diff --git a/.devcontainer/devcontainer.json b/.devcontainer/devcontainer.json
index f339f62f7..ff58994db 100644
--- a/.devcontainer/devcontainer.json
+++ b/.devcontainer/devcontainer.json
@@ -4,10 +4,10 @@
"dockerfile": "Dockerfile",
"args": {
// Update the VARIANT arg to pick a version of Go: 1, 1.15, 1.14
- "VARIANT": "1.24",
+ "VARIANT": "1.25",
// Options
"INSTALL_NODE": "true",
- "NODE_VERSION": "v20"
+ "NODE_VERSION": "v24"
}
},
"workspaceMount": "",
diff --git a/.github/copilot-instructions.md b/.github/copilot-instructions.md
deleted file mode 100644
index 73ad6e727..000000000
--- a/.github/copilot-instructions.md
+++ /dev/null
@@ -1,53 +0,0 @@
-# Navidrome Code Guidelines
-
-This is a music streaming server written in Go with a React frontend. The application manages music libraries, provides streaming capabilities, and offers various features like artist information, artwork handling, and external service integrations.
-
-## Code Standards
-
-### Backend (Go)
-- Follow standard Go conventions and idioms
-- Use context propagation for cancellation signals
-- Write unit tests for new functionality using Ginkgo/Gomega
-- Use mutex appropriately for concurrent operations
-- Implement interfaces for dependencies to facilitate testing
-
-### Frontend (React)
-- Use functional components with hooks
-- Follow React best practices for state management
-- Implement PropTypes for component properties
-- Prefer using React-Admin and Material-UI components
-- Icons should be imported from `react-icons` only
-- Follow existing patterns for API interaction
-
-## Repository Structure
-- `core/`: Server-side business logic (artwork handling, playback, etc.)
-- `ui/`: React frontend components
-- `model/`: Data models and repository interfaces
-- `server/`: API endpoints and server implementation
-- `utils/`: Shared utility functions
-- `persistence/`: Database access layer
-- `scanner/`: Music library scanning functionality
-
-## Key Guidelines
-1. Maintain cache management patterns for performance
-2. Follow the existing concurrency patterns (mutex, atomic)
-3. Use the testing framework appropriately (Ginkgo/Gomega for Go)
-4. Keep UI components focused and reusable
-5. Document configuration options in code
-6. Consider performance implications when working with music libraries
-7. Follow existing error handling patterns
-8. Ensure compatibility with external services (LastFM, Spotify)
-
-## Development Workflow
-- Test changes thoroughly, especially around concurrent operations
-- Validate both backend and frontend interactions
-- Consider how changes will affect user experience and performance
-- Test with different music library sizes and configurations
-- Before committing, ALWAYS run `make format lint test`, and make sure there are no issues
-
-## Important commands
-- `make build`: Build the application
-- `make test`: Run Go tests
-- To run tests for a specific package, use `make test PKG=./pkgname/...`
-- `make lintall`: Run linters
-- `make format`: Format code
\ No newline at end of file
diff --git a/.github/pull_request_template.md b/.github/pull_request_template.md
new file mode 100644
index 000000000..10431e909
--- /dev/null
+++ b/.github/pull_request_template.md
@@ -0,0 +1,38 @@
+### Description
+
+
+### Related Issues
+
+
+### Type of Change
+- [ ] Bug fix
+- [ ] New feature
+- [ ] Documentation update
+- [ ] Refactor
+- [ ] Other (please describe):
+
+### Checklist
+Please review and check all that apply:
+
+- [ ] My code follows the project’s coding style
+- [ ] I have tested the changes locally
+- [ ] I have added or updated documentation as needed
+- [ ] I have added tests that prove my fix/feature works (or explain why not)
+- [ ] All existing and new tests pass
+
+### How to Test
+
+
+### Screenshots / Demos (if applicable)
+
+
+### Additional Notes
+
+
+
\ No newline at end of file
diff --git a/.github/workflows/pipeline.yml b/.github/workflows/pipeline.yml
index 4ac1b2c6b..0767346fa 100644
--- a/.github/workflows/pipeline.yml
+++ b/.github/workflows/pipeline.yml
@@ -14,7 +14,7 @@ concurrency:
cancel-in-progress: true
env:
- CROSS_TAGLIB_VERSION: "2.0.2-1"
+ CROSS_TAGLIB_VERSION: "2.1.1-1"
IS_RELEASE: ${{ startsWith(github.ref, 'refs/tags/') && 'true' || 'false' }}
jobs:
@@ -25,7 +25,7 @@ jobs:
git_tag: ${{ steps.git-version.outputs.GIT_TAG }}
git_sha: ${{ steps.git-version.outputs.GIT_SHA }}
steps:
- - uses: actions/checkout@v4
+ - uses: actions/checkout@v5
with:
fetch-depth: 0
fetch-tags: true
@@ -63,7 +63,7 @@ jobs:
name: Lint Go code
runs-on: ubuntu-latest
steps:
- - uses: actions/checkout@v4
+ - uses: actions/checkout@v5
- name: Download TagLib
uses: ./.github/actions/download-taglib
@@ -78,7 +78,7 @@ jobs:
args: --timeout 2m
- name: Run go goimports
- run: go run golang.org/x/tools/cmd/goimports@latest -w `find . -name '*.go' | grep -v '_gen.go$'`
+ run: go run golang.org/x/tools/cmd/goimports@latest -w `find . -name '*.go' | grep -v '_gen.go$' | grep -v '.pb.go$'`
- run: go mod tidy
- name: Verify no changes from goimports and go mod tidy
run: |
@@ -93,7 +93,7 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Check out code into the Go module directory
- uses: actions/checkout@v4
+ uses: actions/checkout@v5
- name: Download TagLib
uses: ./.github/actions/download-taglib
@@ -106,7 +106,7 @@ jobs:
- name: Test
run: |
pkg-config --define-prefix --cflags --libs taglib # for debugging
- go test -shuffle=on -tags netgo -race -cover ./... -v
+ go test -shuffle=on -tags netgo -race ./... -v
js:
name: Test JS code
@@ -114,10 +114,10 @@ jobs:
env:
NODE_OPTIONS: "--max_old_space_size=4096"
steps:
- - uses: actions/checkout@v4
- - uses: actions/setup-node@v4
+ - uses: actions/checkout@v5
+ - uses: actions/setup-node@v6
with:
- node-version: 20
+ node-version: 24
cache: "npm"
cache-dependency-path: "**/package-lock.json"
@@ -145,7 +145,7 @@ jobs:
name: Lint i18n files
runs-on: ubuntu-latest
steps:
- - uses: actions/checkout@v4
+ - uses: actions/checkout@v5
- run: |
set -e
for file in resources/i18n/*.json; do
@@ -157,6 +157,8 @@ jobs:
exit 1
fi
done
+ - run: ./.github/workflows/validate-translations.sh -v
+
check-push-enabled:
name: Check Docker configuration
@@ -189,7 +191,7 @@ jobs:
PLATFORM=$(echo ${{ matrix.platform }} | tr '/' '_')
echo "PLATFORM=$PLATFORM" >> $GITHUB_ENV
- - uses: actions/checkout@v4
+ - uses: actions/checkout@v5
- name: Prepare Docker Buildx
uses: ./.github/actions/prepare-docker
@@ -215,7 +217,7 @@ jobs:
CROSS_TAGLIB_VERSION=${{ env.CROSS_TAGLIB_VERSION }}
- name: Upload Binaries
- uses: actions/upload-artifact@v4
+ uses: actions/upload-artifact@v5
with:
name: navidrome-${{ env.PLATFORM }}
path: ./output
@@ -246,7 +248,7 @@ jobs:
touch "/tmp/digests/${digest#sha256:}"
- name: Upload digest
- uses: actions/upload-artifact@v4
+ uses: actions/upload-artifact@v5
if: env.IS_LINUX == 'true' && env.IS_DOCKER_PUSH_CONFIGURED == 'true' && env.IS_ARMV5 == 'false'
with:
name: digests-${{ env.PLATFORM }}
@@ -262,10 +264,10 @@ jobs:
env:
REGISTRY_IMAGE: ghcr.io/${{ github.repository }}
steps:
- - uses: actions/checkout@v4
+ - uses: actions/checkout@v5
- name: Download digests
- uses: actions/download-artifact@v4
+ uses: actions/download-artifact@v6
with:
path: /tmp/digests
pattern: digests-*
@@ -316,9 +318,9 @@ jobs:
runs-on: ubuntu-24.04
steps:
- - uses: actions/checkout@v4
+ - uses: actions/checkout@v5
- - uses: actions/download-artifact@v4
+ - uses: actions/download-artifact@v6
with:
path: ./binaries
pattern: navidrome-windows*
@@ -337,7 +339,7 @@ jobs:
du -h binaries/msi/*.msi
- name: Upload MSI files
- uses: actions/upload-artifact@v4
+ uses: actions/upload-artifact@v5
with:
name: navidrome-windows-installers
path: binaries/msi/*.msi
@@ -350,12 +352,12 @@ jobs:
outputs:
package_list: ${{ steps.set-package-list.outputs.package_list }}
steps:
- - uses: actions/checkout@v4
+ - uses: actions/checkout@v5
with:
fetch-depth: 0
fetch-tags: true
- - uses: actions/download-artifact@v4
+ - uses: actions/download-artifact@v6
with:
path: ./binaries
pattern: navidrome-*
@@ -381,7 +383,7 @@ jobs:
rm ./dist/*.tar.gz ./dist/*.zip
- name: Upload all-packages artifact
- uses: actions/upload-artifact@v4
+ uses: actions/upload-artifact@v5
with:
name: packages
path: dist/navidrome_0*
@@ -404,13 +406,13 @@ jobs:
item: ${{ fromJson(needs.release.outputs.package_list) }}
steps:
- name: Download all-packages artifact
- uses: actions/download-artifact@v4
+ uses: actions/download-artifact@v6
with:
name: packages
path: ./dist
- name: Upload all-packages artifact
- uses: actions/upload-artifact@v4
+ uses: actions/upload-artifact@v5
with:
name: navidrome_linux_${{ matrix.item }}
path: dist/navidrome_0*_linux_${{ matrix.item }}
diff --git a/.github/workflows/update-translations.yml b/.github/workflows/update-translations.yml
index 70a9de3d8..69ca1cc94 100644
--- a/.github/workflows/update-translations.yml
+++ b/.github/workflows/update-translations.yml
@@ -8,7 +8,7 @@ jobs:
runs-on: ubuntu-latest
if: ${{ github.repository_owner == 'navidrome' }}
steps:
- - uses: actions/checkout@v4
+ - uses: actions/checkout@v5
- name: Get updated translations
id: poeditor
env:
diff --git a/.github/workflows/validate-translations.sh b/.github/workflows/validate-translations.sh
new file mode 100755
index 000000000..a6b346e78
--- /dev/null
+++ b/.github/workflows/validate-translations.sh
@@ -0,0 +1,236 @@
+#!/bin/bash
+
+# validate-translations.sh
+#
+# This script validates the structure of JSON translation files by comparing them
+# against the reference English translation file (ui/src/i18n/en.json).
+#
+# The script performs the following validations:
+# 1. JSON syntax validation using jq
+# 2. Structural validation - ensures all keys from English file are present
+# 3. Reports missing keys (translation incomplete)
+# 4. Reports extra keys (keys not in English reference, possibly deprecated)
+# 5. Emits GitHub Actions annotations for CI/CD integration
+#
+# Usage:
+# ./validate-translations.sh
+#
+# Environment Variables:
+# EN_FILE - Path to reference English file (default: ui/src/i18n/en.json)
+# TRANSLATION_DIR - Directory containing translation files (default: resources/i18n)
+#
+# Exit codes:
+# 0 - All translations are valid
+# 1 - One or more translations have structural issues
+#
+# GitHub Actions Integration:
+# The script outputs GitHub Actions annotations using ::error and ::warning
+# format that will be displayed in PR checks and workflow summaries.
+
+# Script to validate JSON translation files structure against en.json
+set -e
+
+# Path to the reference English translation file
+EN_FILE="${EN_FILE:-ui/src/i18n/en.json}"
+TRANSLATION_DIR="${TRANSLATION_DIR:-resources/i18n}"
+VERBOSE=false
+
+# Parse command line arguments
+while [[ $# -gt 0 ]]; do
+ case "$1" in
+ -v|--verbose)
+ VERBOSE=true
+ shift
+ ;;
+ -h|--help)
+ echo "Usage: $0 [options]"
+ echo ""
+ echo "Validates JSON translation files structure against English reference file."
+ echo ""
+ echo "Options:"
+ echo " -h, --help Show this help message"
+ echo " -v, --verbose Show detailed output (default: only show errors)"
+ echo ""
+ echo "Environment Variables:"
+ echo " EN_FILE Path to reference English file (default: ui/src/i18n/en.json)"
+ echo " TRANSLATION_DIR Directory with translation files (default: resources/i18n)"
+ echo ""
+ echo "Examples:"
+ echo " $0 # Validate all translation files (quiet mode)"
+ echo " $0 -v # Validate with detailed output"
+ echo " EN_FILE=custom/en.json $0 # Use custom reference file"
+ echo " TRANSLATION_DIR=custom/i18n $0 # Use custom translations directory"
+ exit 0
+ ;;
+ *)
+ echo "Unknown option: $1" >&2
+ echo "Use --help for usage information" >&2
+ exit 1
+ ;;
+ esac
+done
+
+# Color codes for output
+RED='\033[0;31m'
+GREEN='\033[0;32m'
+YELLOW='\033[1;33m'
+NC='\033[0m' # No Color
+
+if [[ "$VERBOSE" == "true" ]]; then
+ echo "Validating translation files structure against ${EN_FILE}..."
+fi
+
+# Check if English reference file exists
+if [[ ! -f "$EN_FILE" ]]; then
+ echo "::error::Reference file $EN_FILE not found"
+ exit 1
+fi
+
+# Function to extract all JSON keys from a file, creating a flat list of dot-separated paths
+extract_keys() {
+ local file="$1"
+ jq -r 'paths(scalars) as $p | $p | join(".")' "$file" 2>/dev/null | sort
+}
+
+# Function to extract all non-empty string keys (to identify structural issues)
+extract_structure_keys() {
+ local file="$1"
+ # Get only keys where values are not empty strings
+ jq -r 'paths(scalars) as $p | select(getpath($p) != "") | $p | join(".")' "$file" 2>/dev/null | sort
+}
+
+# Function to validate a single translation file
+validate_translation() {
+ local translation_file="$1"
+ local filename=$(basename "$translation_file")
+ local has_errors=false
+ local verbose=${2:-false}
+
+ if [[ "$verbose" == "true" ]]; then
+ echo "Validating $filename..."
+ fi
+
+ # First validate JSON syntax
+ if ! jq empty "$translation_file" 2>/dev/null; then
+ echo "::error file=$translation_file::Invalid JSON syntax"
+ echo -e "${RED}✗ $filename has invalid JSON syntax${NC}"
+ return 1
+ fi
+
+ # Extract all keys from both files (for statistics)
+ local en_keys_file=$(mktemp)
+ local translation_keys_file=$(mktemp)
+
+ extract_keys "$EN_FILE" > "$en_keys_file"
+ extract_keys "$translation_file" > "$translation_keys_file"
+
+ # Extract only non-empty structure keys (to validate structural issues)
+ local en_structure_file=$(mktemp)
+ local translation_structure_file=$(mktemp)
+
+ extract_structure_keys "$EN_FILE" > "$en_structure_file"
+ extract_structure_keys "$translation_file" > "$translation_structure_file"
+
+ # Find structural issues: keys in translation not in English (misplaced)
+ local extra_keys=$(comm -13 "$en_keys_file" "$translation_keys_file")
+
+ # Find missing keys (for statistics only)
+ local missing_keys=$(comm -23 "$en_keys_file" "$translation_keys_file")
+
+ # Count keys for statistics
+ local total_en_keys=$(wc -l < "$en_keys_file")
+ local total_translation_keys=$(wc -l < "$translation_keys_file")
+ local missing_count=0
+ local extra_count=0
+
+ if [[ -n "$missing_keys" ]]; then
+ missing_count=$(echo "$missing_keys" | grep -c '^' || echo 0)
+ fi
+
+ if [[ -n "$extra_keys" ]]; then
+ extra_count=$(echo "$extra_keys" | grep -c '^' || echo 0)
+ has_errors=true
+ fi
+
+ # Report extra/misplaced keys (these are structural issues)
+ if [[ -n "$extra_keys" ]]; then
+ if [[ "$verbose" == "true" ]]; then
+ echo -e "${YELLOW}Misplaced keys in $filename ($extra_count):${NC}"
+ fi
+
+ while IFS= read -r key; do
+ # Try to find the line number
+ line=$(grep -n "\"$(echo "$key" | sed 's/.*\.//')" "$translation_file" | head -1 | cut -d: -f1)
+ line=${line:-1} # Default to line 1 if not found
+
+ echo "::error file=$translation_file,line=$line::Misplaced key: $key"
+
+ if [[ "$verbose" == "true" ]]; then
+ echo " + $key (line ~$line)"
+ fi
+ done <<< "$extra_keys"
+ fi
+
+ # Clean up temp files
+ rm -f "$en_keys_file" "$translation_keys_file" "$en_structure_file" "$translation_structure_file"
+
+ # Print statistics
+ if [[ "$verbose" == "true" ]]; then
+ echo " Keys: $total_translation_keys/$total_en_keys (Missing: $missing_count, Extra/Misplaced: $extra_count)"
+
+ if [[ "$has_errors" == "true" ]]; then
+ echo -e "${RED}✗ $filename has structural issues${NC}"
+ else
+ echo -e "${GREEN}✓ $filename structure is valid${NC}"
+ fi
+ elif [[ "$has_errors" == "true" ]]; then
+ echo -e "${RED}✗ $filename has structural issues (Extra/Misplaced: $extra_count)${NC}"
+ fi
+
+ return $([[ "$has_errors" == "true" ]] && echo 1 || echo 0)
+}
+
+# Main validation loop
+validation_failed=false
+total_files=0
+failed_files=0
+valid_files=0
+
+for translation_file in "$TRANSLATION_DIR"/*.json; do
+ if [[ -f "$translation_file" ]]; then
+ total_files=$((total_files + 1))
+ if ! validate_translation "$translation_file" "$VERBOSE"; then
+ validation_failed=true
+ failed_files=$((failed_files + 1))
+ else
+ valid_files=$((valid_files + 1))
+ fi
+
+ if [[ "$VERBOSE" == "true" ]]; then
+ echo "" # Add spacing between files
+ fi
+ fi
+done
+
+# Summary
+if [[ "$VERBOSE" == "true" ]]; then
+ echo "========================================="
+ echo "Translation Validation Summary:"
+ echo " Total files: $total_files"
+ echo " Valid files: $valid_files"
+ echo " Files with structural issues: $failed_files"
+ echo "========================================="
+fi
+
+if [[ "$validation_failed" == "true" ]]; then
+ if [[ "$VERBOSE" == "true" ]]; then
+ echo -e "${RED}Translation validation failed - $failed_files file(s) have structural issues${NC}"
+ else
+ echo -e "${RED}Translation validation failed - $failed_files/$total_files file(s) have structural issues${NC}"
+ fi
+ exit 1
+elif [[ "$VERBOSE" == "true" ]]; then
+ echo -e "${GREEN}All translation files are structurally valid${NC}"
+fi
+
+exit 0
\ No newline at end of file
diff --git a/.gitignore b/.gitignore
index 4e32e14fd..74d7ee46f 100644
--- a/.gitignore
+++ b/.gitignore
@@ -5,6 +5,7 @@
/navidrome
/iTunes*.xml
/tmp
+/bin
data/*
vendor/*/
wiki
@@ -23,7 +24,11 @@ music
docker-compose.yml
!contrib/docker-compose.yml
binaries
-navidrome-master
+navidrome-*
AGENTS.md
+.github/prompts
+.github/instructions
+.github/git-commit-instructions.md
*.exe
-bin/
\ No newline at end of file
+*.test
+*.wasm
\ No newline at end of file
diff --git a/.nvmrc b/.nvmrc
index 9a2a0e219..54c65116f 100644
--- a/.nvmrc
+++ b/.nvmrc
@@ -1 +1 @@
-v20
+v24
diff --git a/Dockerfile b/Dockerfile
index 4b4c3d18c..fb1cf997b 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -1,8 +1,8 @@
FROM --platform=$BUILDPLATFORM ghcr.io/crazy-max/osxcross:14.5-debian AS osxcross
########################################################################################################################
-### Build xx (orignal image: tonistiigi/xx)
-FROM --platform=$BUILDPLATFORM public.ecr.aws/docker/library/alpine:3.21 AS xx-build
+### Build xx (original image: tonistiigi/xx)
+FROM --platform=$BUILDPLATFORM public.ecr.aws/docker/library/alpine:3.19 AS xx-build
# v1.5.0
ENV XX_VERSION=b4e4c451c778822e6742bfc9d9a91d7c7d885c8a
@@ -26,12 +26,14 @@ COPY --from=xx-build /out/ /usr/bin/
########################################################################################################################
### Get TagLib
-FROM --platform=$BUILDPLATFORM public.ecr.aws/docker/library/alpine:3.21 AS taglib-build
+FROM --platform=$BUILDPLATFORM public.ecr.aws/docker/library/alpine:3.19 AS taglib-build
ARG TARGETPLATFORM
-ARG CROSS_TAGLIB_VERSION=2.0.2-1
+ARG CROSS_TAGLIB_VERSION=2.1.1-1
ENV CROSS_TAGLIB_RELEASES_URL=https://github.com/navidrome/cross-taglib/releases/download/v${CROSS_TAGLIB_VERSION}/
+# wget in busybox can't follow redirects
RUN < /dev/null || (echo "Installing golangci-lint..." && curl -sSfL https://raw.githubusercontent.com/golangci/golangci-lint/HEAD/install.sh | sh -s v2.1.6)
+ @INSTALL=false; \
+ if PATH=$$PATH:./bin which golangci-lint > /dev/null 2>&1; then \
+ CURRENT_VERSION=$$(PATH=$$PATH:./bin golangci-lint version 2>/dev/null | grep -oE '[0-9]+\.[0-9]+\.[0-9]+' | head -n1); \
+ REQUIRED_VERSION=$$(echo "$(GOLANGCI_LINT_VERSION)" | sed 's/^v//'); \
+ if [ "$$CURRENT_VERSION" != "$$REQUIRED_VERSION" ]; then \
+ echo "Found golangci-lint $$CURRENT_VERSION, but $$REQUIRED_VERSION is required. Reinstalling..."; \
+ rm -f ./bin/golangci-lint; \
+ INSTALL=true; \
+ fi; \
+ else \
+ INSTALL=true; \
+ fi; \
+ if [ "$$INSTALL" = "true" ]; then \
+ echo "Installing golangci-lint $(GOLANGCI_LINT_VERSION)..."; \
+ curl -sSfL https://raw.githubusercontent.com/golangci/golangci-lint/HEAD/install.sh | sh -s $(GOLANGCI_LINT_VERSION); \
+ fi
.PHONY: install-golangci-lint
lint: install-golangci-lint ##@Development Lint Go code
@@ -64,7 +95,7 @@ lintall: lint ##@Development Lint Go and JS code
format: ##@Development Format code
@(cd ./ui && npm run prettier)
- @go tool goimports -w `find . -name '*.go' | grep -v _gen.go$$`
+ @go tool goimports -w `find . -name '*.go' | grep -v _gen.go$$ | grep -v .pb.go$$`
@go mod tidy
.PHONY: format
@@ -153,6 +184,20 @@ docker-msi: ##@Cross_Compilation Build MSI installer for Windows
@du -h binaries/msi/*.msi
.PHONY: docker-msi
+run-docker: ##@Development Run a Navidrome Docker image. Usage: make run-docker tag=
+ @if [ -z "$(tag)" ]; then echo "Usage: make run-docker tag="; exit 1; fi
+ @TAG_DIR="tmp/$$(echo '$(tag)' | tr '/:' '_')"; mkdir -p "$$TAG_DIR"; \
+ VOLUMES="-v $(PWD)/$$TAG_DIR:/data"; \
+ if [ -f navidrome.toml ]; then \
+ VOLUMES="$$VOLUMES -v $(PWD)/navidrome.toml:/data/navidrome.toml:ro"; \
+ MUSIC_FOLDER=$$(grep '^MusicFolder' navidrome.toml | head -n1 | sed 's/.*= *"//' | sed 's/".*//'); \
+ if [ -n "$$MUSIC_FOLDER" ] && [ -d "$$MUSIC_FOLDER" ]; then \
+ VOLUMES="$$VOLUMES -v $$MUSIC_FOLDER:/music:ro"; \
+ fi; \
+ fi; \
+ echo "Running: docker run --rm -p 4533:4533 $$VOLUMES $(tag)"; docker run --rm -p 4533:4533 $$VOLUMES $(tag)
+.PHONY: run-docker
+
package: docker-build ##@Cross_Compilation Create binaries and packages for ALL supported platforms
@if [ -z `which goreleaser` ]; then echo "Please install goreleaser first: https://goreleaser.com/install/"; exit 1; fi
goreleaser release -f release/goreleaser.yml --clean --skip=publish --snapshot
@@ -221,6 +266,24 @@ deprecated:
@echo "WARNING: This target is deprecated and will be removed in future releases. Use 'make build' instead."
.PHONY: deprecated
+# Generate Go code from plugins/api/api.proto
+plugin-gen: check_go_env ##@Development Generate Go code from plugins protobuf files
+ go generate ./plugins/...
+.PHONY: plugin-gen
+
+plugin-examples: check_go_env ##@Development Build all example plugins
+ $(MAKE) -C plugins/examples clean all
+.PHONY: plugin-examples
+
+plugin-clean: check_go_env ##@Development Clean all plugins
+ $(MAKE) -C plugins/examples clean
+ $(MAKE) -C plugins/testdata clean
+.PHONY: plugin-clean
+
+plugin-tests: check_go_env ##@Development Build all test plugins
+ $(MAKE) -C plugins/testdata clean all
+.PHONY: plugin-tests
+
.DEFAULT_GOAL := help
HELP_FUN = \
diff --git a/adapters/taglib/end_to_end_test.go b/adapters/taglib/end_to_end_test.go
index 08fc1a506..e4d94bb24 100644
--- a/adapters/taglib/end_to_end_test.go
+++ b/adapters/taglib/end_to_end_test.go
@@ -8,6 +8,7 @@ import (
"github.com/djherbis/times"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/model/metadata"
+ "github.com/navidrome/navidrome/utils/gg"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
)
@@ -78,22 +79,116 @@ var _ = Describe("Extractor", func() {
var e *extractor
+ parseTestFile := func(path string) *model.MediaFile {
+ mds, err := e.Parse(path)
+ Expect(err).ToNot(HaveOccurred())
+
+ info, ok := mds[path]
+ Expect(ok).To(BeTrue())
+
+ fileInfo, err := os.Stat(path)
+ Expect(err).ToNot(HaveOccurred())
+ info.FileInfo = testFileInfo{FileInfo: fileInfo}
+
+ metadata := metadata.New(path, info)
+ mf := metadata.ToMediaFile(1, "folderID")
+ return &mf
+ }
+
BeforeEach(func() {
e = &extractor{}
})
+ Describe("ReplayGain", func() {
+ DescribeTable("test replaygain end-to-end", func(file string, trackGain, trackPeak, albumGain, albumPeak *float64) {
+ mf := parseTestFile("tests/fixtures/" + file)
+
+ Expect(mf.RGTrackGain).To(Equal(trackGain))
+ Expect(mf.RGTrackPeak).To(Equal(trackPeak))
+ Expect(mf.RGAlbumGain).To(Equal(albumGain))
+ Expect(mf.RGAlbumPeak).To(Equal(albumPeak))
+ },
+ Entry("mp3 with no replaygain", "no_replaygain.mp3", nil, nil, nil, nil),
+ Entry("mp3 with no zero replaygain", "zero_replaygain.mp3", gg.P(0.0), gg.P(1.0), gg.P(0.0), gg.P(1.0)),
+ )
+ })
+
+ Describe("lyrics", func() {
+ makeLyrics := func(code, secondLine string) model.Lyrics {
+ return model.Lyrics{
+ DisplayArtist: "",
+ DisplayTitle: "",
+ Lang: code,
+ Line: []model.Line{
+ {Start: gg.P(int64(0)), Value: "This is"},
+ {Start: gg.P(int64(2500)), Value: secondLine},
+ },
+ Offset: nil,
+ Synced: true,
+ }
+ }
+
+ It("should fetch both synced and unsynced lyrics in mixed flac", func() {
+ mf := parseTestFile("tests/fixtures/mixed-lyrics.flac")
+
+ lyrics, err := mf.StructuredLyrics()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(lyrics).To(HaveLen(2))
+
+ Expect(lyrics[0].Synced).To(BeTrue())
+ Expect(lyrics[1].Synced).To(BeFalse())
+ })
+
+ It("should handle mp3 with uslt and sylt", func() {
+ mf := parseTestFile("tests/fixtures/test.mp3")
+
+ lyrics, err := mf.StructuredLyrics()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(lyrics).To(HaveLen(4))
+
+ engSylt := makeLyrics("eng", "English SYLT")
+ engUslt := makeLyrics("eng", "English")
+ unsSylt := makeLyrics("xxx", "unspecified SYLT")
+ unsUslt := makeLyrics("xxx", "unspecified")
+
+ // Why is the order inconsistent between runs? Nobody knows
+ Expect(lyrics).To(Or(
+ Equal(model.LyricList{engSylt, engUslt, unsSylt, unsUslt}),
+ Equal(model.LyricList{unsSylt, unsUslt, engSylt, engUslt}),
+ ))
+ })
+
+ DescribeTable("format-specific lyrics", func(file string, isId3 bool) {
+ mf := parseTestFile("tests/fixtures/" + file)
+
+ lyrics, err := mf.StructuredLyrics()
+ Expect(err).To(Not(HaveOccurred()))
+ Expect(lyrics).To(HaveLen(2))
+
+ unspec := makeLyrics("xxx", "unspecified")
+ eng := makeLyrics("xxx", "English")
+
+ if isId3 {
+ eng.Lang = "eng"
+ }
+
+ Expect(lyrics).To(Or(
+ Equal(model.LyricList{unspec, eng}),
+ Equal(model.LyricList{eng, unspec})))
+ },
+ Entry("flac", "test.flac", false),
+ Entry("m4a", "test.m4a", false),
+ Entry("ogg", "test.ogg", false),
+ Entry("wma", "test.wma", false),
+ Entry("wv", "test.wv", false),
+ Entry("wav", "test.wav", true),
+ Entry("aiff", "test.aiff", true),
+ )
+ })
+
Describe("Participants", func() {
DescribeTable("test tags consistent across formats", func(format string) {
- path := "tests/fixtures/test." + format
- mds, err := e.Parse(path)
- Expect(err).ToNot(HaveOccurred())
-
- info := mds[path]
- fileInfo, _ := os.Stat(path)
- info.FileInfo = testFileInfo{FileInfo: fileInfo}
-
- metadata := metadata.New(path, info)
- mf := metadata.ToMediaFile(1, "folderID")
+ mf := parseTestFile("tests/fixtures/test." + format)
for _, data := range roles {
role := data.Role
@@ -144,11 +239,40 @@ var _ = Describe("Extractor", func() {
Entry("FLAC format", "flac"),
Entry("M4a format", "m4a"),
Entry("OGG format", "ogg"),
- Entry("WMA format", "wv"),
+ Entry("WV format", "wv"),
Entry("MP3 format", "mp3"),
Entry("WAV format", "wav"),
Entry("AIFF format", "aiff"),
)
+
+ It("should parse wma", func() {
+ mf := parseTestFile("tests/fixtures/test.wma")
+
+ for _, data := range roles {
+ role := data.Role
+ artists := data.ParticipantList
+ actual := mf.Participants[role]
+
+ // WMA has no Arranger role
+ if role == model.RoleArranger {
+ Expect(actual).To(HaveLen(0))
+ continue
+ }
+
+ Expect(actual).To(HaveLen(len(artists)), role.String())
+
+ // For some bizarre reason, the order is inverted. We also don't get
+ // sort names or MBIDs
+ for i := range artists {
+ idx := len(artists) - 1 - i
+
+ actualArtist := actual[i]
+ expectedArtist := artists[idx]
+
+ Expect(actualArtist.Name).To(Equal(expectedArtist.Name))
+ }
+ }
+ })
})
})
diff --git a/adapters/taglib/taglib.go b/adapters/taglib/taglib.go
index c89dabf62..d32adf4ed 100644
--- a/adapters/taglib/taglib.go
+++ b/adapters/taglib/taglib.go
@@ -7,6 +7,7 @@ import (
"strings"
"time"
+ "github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/core/storage/local"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model/metadata"
@@ -42,23 +43,21 @@ func (e extractor) extractMetadata(filePath string) (*metadata.Info, error) {
// Parse audio properties
ap := metadata.AudioProperties{}
- if length, ok := tags["_lengthinmilliseconds"]; ok && len(length) > 0 {
- millis, _ := strconv.Atoi(length[0])
- if millis > 0 {
- ap.Duration = (time.Millisecond * time.Duration(millis)).Round(time.Millisecond * 10)
- }
- delete(tags, "_lengthinmilliseconds")
- }
- parseProp := func(prop string, target *int) {
- if value, ok := tags[prop]; ok && len(value) > 0 {
- *target, _ = strconv.Atoi(value[0])
- delete(tags, prop)
- }
- }
- parseProp("_bitrate", &ap.BitRate)
- parseProp("_channels", &ap.Channels)
- parseProp("_samplerate", &ap.SampleRate)
- parseProp("_bitspersample", &ap.BitDepth)
+ ap.BitRate = parseProp(tags, "__bitrate")
+ ap.Channels = parseProp(tags, "__channels")
+ ap.SampleRate = parseProp(tags, "__samplerate")
+ ap.BitDepth = parseProp(tags, "__bitspersample")
+ length := parseProp(tags, "__lengthinmilliseconds")
+ ap.Duration = (time.Millisecond * time.Duration(length)).Round(time.Millisecond * 10)
+
+ // Extract basic tags
+ parseBasicTag(tags, "__title", "title")
+ parseBasicTag(tags, "__artist", "artist")
+ parseBasicTag(tags, "__album", "album")
+ parseBasicTag(tags, "__comment", "comment")
+ parseBasicTag(tags, "__genre", "genre")
+ parseBasicTag(tags, "__year", "year")
+ parseBasicTag(tags, "__track", "tracknumber")
// Parse track/disc totals
parseTuple := func(prop string) {
@@ -106,6 +105,31 @@ var tiplMapping = map[string]string{
"DJ-mix": "djmixer",
}
+// parseProp parses a property from the tags map and sets it to the target integer.
+// It also deletes the property from the tags map after parsing.
+func parseProp(tags map[string][]string, prop string) int {
+ if value, ok := tags[prop]; ok && len(value) > 0 {
+ v, _ := strconv.Atoi(value[0])
+ delete(tags, prop)
+ return v
+ }
+ return 0
+}
+
+// parseBasicTag checks if a basic tag (like __title, __artist, etc.) exists in the tags map.
+// If it does, it moves the value to a more appropriate tag name (like title, artist, etc.),
+// and deletes the basic tag from the map. If the target tag already exists, it ignores the basic tag.
+func parseBasicTag(tags map[string][]string, basicName string, tagName string) {
+ basicValue := tags[basicName]
+ if len(basicValue) == 0 {
+ return
+ }
+ delete(tags, basicName)
+ if len(tags[tagName]) == 0 {
+ tags[tagName] = basicValue
+ }
+}
+
// parseTIPL parses the ID3v2.4 TIPL frame string, which is received from TagLib in the format:
//
// "arranger Andrew Powell engineer Chris Blair engineer Pat Stapley producer Eric Woolfson".
@@ -148,4 +172,7 @@ func init() {
// ignores fs, as taglib extractor only works with local files
return &extractor{baseDir}
})
+ conf.AddHook(func() {
+ log.Debug("TagLib version", "version", Version())
+ })
}
diff --git a/adapters/taglib/taglib_test.go b/adapters/taglib/taglib_test.go
index 37b012763..f24c0e839 100644
--- a/adapters/taglib/taglib_test.go
+++ b/adapters/taglib/taglib_test.go
@@ -179,7 +179,7 @@ var _ = Describe("Extractor", func() {
Entry("correctly parses wma/asf tags", "test.wma", "1.02s", 1, 44100, 16, "3.27 dB", "0.132914", "3.27 dB", "0.132914", false, true),
// ffmpeg -f lavfi -i "sine=frequency=800:duration=1" test.wv
- Entry("correctly parses wv (wavpak) tags", "test.wv", "1s", 1, 44100, 16, "3.43 dB", "0.125061", "3.43 dB", "0.125061", false, false),
+ Entry("correctly parses wv (wavpak) tags", "test.wv", "1s", 1, 44100, 16, "3.43 dB", "0.125061", "3.43 dB", "0.125061", false, true),
// ffmpeg -f lavfi -i "sine=frequency=1000:duration=1" test.wav
Entry("correctly parses wav tags", "test.wav", "1s", 1, 44100, 16, "3.06 dB", "0.125056", "3.06 dB", "0.125056", true, true),
diff --git a/adapters/taglib/taglib_wrapper.cpp b/adapters/taglib/taglib_wrapper.cpp
index 17c95bfc0..2985e8f18 100644
--- a/adapters/taglib/taglib_wrapper.cpp
+++ b/adapters/taglib/taglib_wrapper.cpp
@@ -1,6 +1,5 @@
#include
#include
-#include
#define TAGLIB_STATIC
#include
@@ -46,31 +45,63 @@ int taglib_read(const FILENAME_CHAR_T *filename, unsigned long id) {
// Add audio properties to the tags
const TagLib::AudioProperties *props(f.audioProperties());
- goPutInt(id, (char *)"_lengthinmilliseconds", props->lengthInMilliseconds());
- goPutInt(id, (char *)"_bitrate", props->bitrate());
- goPutInt(id, (char *)"_channels", props->channels());
- goPutInt(id, (char *)"_samplerate", props->sampleRate());
+ goPutInt(id, (char *)"__lengthinmilliseconds", props->lengthInMilliseconds());
+ goPutInt(id, (char *)"__bitrate", props->bitrate());
+ goPutInt(id, (char *)"__channels", props->channels());
+ goPutInt(id, (char *)"__samplerate", props->sampleRate());
+ // Extract bits per sample for supported formats
+ int bitsPerSample = 0;
if (const auto* apeProperties{ dynamic_cast(props) })
- goPutInt(id, (char *)"_bitspersample", apeProperties->bitsPerSample());
- if (const auto* asfProperties{ dynamic_cast(props) })
- goPutInt(id, (char *)"_bitspersample", asfProperties->bitsPerSample());
+ bitsPerSample = apeProperties->bitsPerSample();
+ else if (const auto* asfProperties{ dynamic_cast(props) })
+ bitsPerSample = asfProperties->bitsPerSample();
else if (const auto* flacProperties{ dynamic_cast(props) })
- goPutInt(id, (char *)"_bitspersample", flacProperties->bitsPerSample());
+ bitsPerSample = flacProperties->bitsPerSample();
else if (const auto* mp4Properties{ dynamic_cast(props) })
- goPutInt(id, (char *)"_bitspersample", mp4Properties->bitsPerSample());
+ bitsPerSample = mp4Properties->bitsPerSample();
else if (const auto* wavePackProperties{ dynamic_cast(props) })
- goPutInt(id, (char *)"_bitspersample", wavePackProperties->bitsPerSample());
+ bitsPerSample = wavePackProperties->bitsPerSample();
else if (const auto* aiffProperties{ dynamic_cast(props) })
- goPutInt(id, (char *)"_bitspersample", aiffProperties->bitsPerSample());
+ bitsPerSample = aiffProperties->bitsPerSample();
else if (const auto* wavProperties{ dynamic_cast(props) })
- goPutInt(id, (char *)"_bitspersample", wavProperties->bitsPerSample());
+ bitsPerSample = wavProperties->bitsPerSample();
else if (const auto* dsfProperties{ dynamic_cast(props) })
- goPutInt(id, (char *)"_bitspersample", dsfProperties->bitsPerSample());
+ bitsPerSample = dsfProperties->bitsPerSample();
+
+ if (bitsPerSample > 0) {
+ goPutInt(id, (char *)"__bitspersample", bitsPerSample);
+ }
// Send all properties to the Go map
TagLib::PropertyMap tags = f.file()->properties();
+ // Make sure at least the basic properties are extracted
+ TagLib::Tag *basic = f.file()->tag();
+ if (!basic->isEmpty()) {
+ if (!basic->title().isEmpty()) {
+ tags.insert("__title", basic->title());
+ }
+ if (!basic->artist().isEmpty()) {
+ tags.insert("__artist", basic->artist());
+ }
+ if (!basic->album().isEmpty()) {
+ tags.insert("__album", basic->album());
+ }
+ if (!basic->comment().isEmpty()) {
+ tags.insert("__comment", basic->comment());
+ }
+ if (!basic->genre().isEmpty()) {
+ tags.insert("__genre", basic->genre());
+ }
+ if (basic->year() > 0) {
+ tags.insert("__year", TagLib::String::number(basic->year()));
+ }
+ if (basic->track() > 0) {
+ tags.insert("__track", TagLib::String::number(basic->track()));
+ }
+ }
+
TagLib::ID3v2::Tag *id3Tags = NULL;
// Get some extended/non-standard ID3-only tags (ex: iTunes extended frames)
@@ -113,7 +144,7 @@ int taglib_read(const FILENAME_CHAR_T *filename, unsigned long id) {
strncpy(language, bv.data(), 3);
}
- char *val = (char *)frame->text().toCString(true);
+ char *val = const_cast(frame->text().toCString(true));
goPutLyrics(id, language, val);
}
@@ -132,7 +163,7 @@ int taglib_read(const FILENAME_CHAR_T *filename, unsigned long id) {
if (format == TagLib::ID3v2::SynchronizedLyricsFrame::AbsoluteMilliseconds) {
for (const auto &line: frame->synchedText()) {
- char *text = (char *)line.text.toCString(true);
+ char *text = const_cast(line.text.toCString(true));
goPutLyricLine(id, language, text, line.time);
}
} else if (format == TagLib::ID3v2::SynchronizedLyricsFrame::AbsoluteMpegFrames) {
@@ -141,7 +172,7 @@ int taglib_read(const FILENAME_CHAR_T *filename, unsigned long id) {
if (sampleRate != 0) {
for (const auto &line: frame->synchedText()) {
const int timeInMs = (line.time * 1000) / sampleRate;
- char *text = (char *)line.text.toCString(true);
+ char *text = const_cast(line.text.toCString(true));
goPutLyricLine(id, language, text, timeInMs);
}
}
@@ -160,9 +191,9 @@ int taglib_read(const FILENAME_CHAR_T *filename, unsigned long id) {
if (m4afile != NULL) {
const auto itemListMap = m4afile->tag()->itemMap();
for (const auto item: itemListMap) {
- char *key = (char *)item.first.toCString(true);
+ char *key = const_cast(item.first.toCString(true));
for (const auto value: item.second.toStringList()) {
- char *val = (char *)value.toCString(true);
+ char *val = const_cast(value.toCString(true));
goPutM4AStr(id, key, val);
}
}
@@ -174,17 +205,24 @@ int taglib_read(const FILENAME_CHAR_T *filename, unsigned long id) {
const TagLib::ASF::Tag *asfTags{asfFile->tag()};
const auto itemListMap = asfTags->attributeListMap();
for (const auto item : itemListMap) {
- tags.insert(item.first, item.second.front().toString());
+ char *key = const_cast(item.first.toCString(true));
+
+ for (auto j = item.second.begin();
+ j != item.second.end(); ++j) {
+
+ char *val = const_cast(j->toString().toCString(true));
+ goPutStr(id, key, val);
+ }
}
}
// Send all collected tags to the Go map
for (TagLib::PropertyMap::ConstIterator i = tags.begin(); i != tags.end();
++i) {
- char *key = (char *)i->first.toCString(true);
+ char *key = const_cast(i->first.toCString(true));
for (TagLib::StringList::ConstIterator j = i->second.begin();
j != i->second.end(); ++j) {
- char *val = (char *)(*j).toCString(true);
+ char *val = const_cast((*j).toCString(true));
goPutStr(id, key, val);
}
}
@@ -242,7 +280,19 @@ char has_cover(const TagLib::FileRef f) {
// ----- WMA
else if (TagLib::ASF::File * asfFile{dynamic_cast(f.file())}) {
const TagLib::ASF::Tag *tag{ asfFile->tag() };
- hasCover = tag && asfFile->tag()->attributeListMap().contains("WM/Picture");
+ hasCover = tag && tag->attributeListMap().contains("WM/Picture");
+ }
+ // ----- DSF
+ else if (TagLib::DSF::File * dsffile{ dynamic_cast(f.file())}) {
+ const TagLib::ID3v2::Tag *tag { dsffile->tag() };
+ hasCover = tag && !tag->frameListMap()["APIC"].isEmpty();
+ }
+ // ----- WAVPAK (APE tag)
+ else if (TagLib::WavPack::File * wvFile{dynamic_cast(f.file())}) {
+ if (wvFile->hasAPETag()) {
+ // This is the particular string that Picard uses
+ hasCover = !wvFile->APETag()->itemListMap()["COVER ART (FRONT)"].isEmpty();
+ }
}
return hasCover;
diff --git a/cmd/cmd_suite_test.go b/cmd/cmd_suite_test.go
new file mode 100644
index 000000000..f2ddf6a9c
--- /dev/null
+++ b/cmd/cmd_suite_test.go
@@ -0,0 +1,17 @@
+package cmd
+
+import (
+ "testing"
+
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+func TestCmd(t *testing.T) {
+ tests.Init(t, false)
+ log.SetLevel(log.LevelFatal)
+ RegisterFailHandler(Fail)
+ RunSpecs(t, "Cmd Suite")
+}
diff --git a/cmd/plugin.go b/cmd/plugin.go
new file mode 100644
index 000000000..0f3b66078
--- /dev/null
+++ b/cmd/plugin.go
@@ -0,0 +1,716 @@
+package cmd
+
+import (
+ "cmp"
+ "crypto/sha256"
+ "encoding/hex"
+ "fmt"
+ "io"
+ "os"
+ "path/filepath"
+ "strings"
+ "text/tabwriter"
+ "time"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/plugins"
+ "github.com/navidrome/navidrome/plugins/schema"
+ "github.com/navidrome/navidrome/utils"
+ "github.com/navidrome/navidrome/utils/slice"
+ "github.com/spf13/cobra"
+)
+
+const (
+ pluginPackageExtension = ".ndp"
+ pluginDirPermissions = 0700
+ pluginFilePermissions = 0600
+)
+
+func init() {
+ pluginCmd := &cobra.Command{
+ Use: "plugin",
+ Short: "Manage Navidrome plugins",
+ Long: "Commands for managing Navidrome plugins",
+ }
+
+ listCmd := &cobra.Command{
+ Use: "list",
+ Short: "List installed plugins",
+ Long: "List all installed plugins with their metadata",
+ Run: pluginList,
+ }
+
+ infoCmd := &cobra.Command{
+ Use: "info [pluginPackage|pluginName]",
+ Short: "Show details of a plugin",
+ Long: "Show detailed information about a plugin package (.ndp file) or an installed plugin",
+ Args: cobra.ExactArgs(1),
+ Run: pluginInfo,
+ }
+
+ installCmd := &cobra.Command{
+ Use: "install [pluginPackage]",
+ Short: "Install a plugin from a .ndp file",
+ Long: "Install a Navidrome Plugin Package (.ndp) file",
+ Args: cobra.ExactArgs(1),
+ Run: pluginInstall,
+ }
+
+ removeCmd := &cobra.Command{
+ Use: "remove [pluginName]",
+ Short: "Remove an installed plugin",
+ Long: "Remove a plugin by name",
+ Args: cobra.ExactArgs(1),
+ Run: pluginRemove,
+ }
+
+ updateCmd := &cobra.Command{
+ Use: "update [pluginPackage]",
+ Short: "Update an existing plugin",
+ Long: "Update an installed plugin with a new version from a .ndp file",
+ Args: cobra.ExactArgs(1),
+ Run: pluginUpdate,
+ }
+
+ refreshCmd := &cobra.Command{
+ Use: "refresh [pluginName]",
+ Short: "Reload a plugin without restarting Navidrome",
+ Long: "Reload and recompile a plugin without needing to restart Navidrome",
+ Args: cobra.ExactArgs(1),
+ Run: pluginRefresh,
+ }
+
+ devCmd := &cobra.Command{
+ Use: "dev [folder_path]",
+ Short: "Create symlink to development folder",
+ Long: "Create a symlink from a plugin development folder to the plugins directory for easier development",
+ Args: cobra.ExactArgs(1),
+ Run: pluginDev,
+ }
+
+ pluginCmd.AddCommand(listCmd, infoCmd, installCmd, removeCmd, updateCmd, refreshCmd, devCmd)
+ rootCmd.AddCommand(pluginCmd)
+}
+
+// Validation helpers
+
+func validatePluginPackageFile(path string) error {
+ if !utils.FileExists(path) {
+ return fmt.Errorf("plugin package not found: %s", path)
+ }
+ if filepath.Ext(path) != pluginPackageExtension {
+ return fmt.Errorf("not a valid plugin package: %s (expected %s extension)", path, pluginPackageExtension)
+ }
+ return nil
+}
+
+func validatePluginDirectory(pluginsDir, pluginName string) (string, error) {
+ pluginDir := filepath.Join(pluginsDir, pluginName)
+ if !utils.FileExists(pluginDir) {
+ return "", fmt.Errorf("plugin not found: %s (path: %s)", pluginName, pluginDir)
+ }
+ return pluginDir, nil
+}
+
+func resolvePluginPath(pluginDir string) (resolvedPath string, isSymlink bool, err error) {
+ // Check if it's a directory or a symlink
+ lstat, err := os.Lstat(pluginDir)
+ if err != nil {
+ return "", false, fmt.Errorf("failed to stat plugin: %w", err)
+ }
+
+ isSymlink = lstat.Mode()&os.ModeSymlink != 0
+
+ if isSymlink {
+ // Resolve the symlink target
+ targetDir, err := os.Readlink(pluginDir)
+ if err != nil {
+ return "", true, fmt.Errorf("failed to resolve symlink: %w", err)
+ }
+
+ // If target is a relative path, make it absolute
+ if !filepath.IsAbs(targetDir) {
+ targetDir = filepath.Join(filepath.Dir(pluginDir), targetDir)
+ }
+
+ // Verify the target exists and is a directory
+ targetInfo, err := os.Stat(targetDir)
+ if err != nil {
+ return "", true, fmt.Errorf("failed to access symlink target %s: %w", targetDir, err)
+ }
+
+ if !targetInfo.IsDir() {
+ return "", true, fmt.Errorf("symlink target is not a directory: %s", targetDir)
+ }
+
+ return targetDir, true, nil
+ } else if !lstat.IsDir() {
+ return "", false, fmt.Errorf("not a valid plugin directory: %s", pluginDir)
+ }
+
+ return pluginDir, false, nil
+}
+
+// Package handling helpers
+
+func loadAndValidatePackage(ndpPath string) (*plugins.PluginPackage, error) {
+ if err := validatePluginPackageFile(ndpPath); err != nil {
+ return nil, err
+ }
+
+ pkg, err := plugins.LoadPackage(ndpPath)
+ if err != nil {
+ return nil, fmt.Errorf("failed to load plugin package: %w", err)
+ }
+
+ return pkg, nil
+}
+
+func extractAndSetupPlugin(ndpPath, targetDir string) error {
+ if err := plugins.ExtractPackage(ndpPath, targetDir); err != nil {
+ return fmt.Errorf("failed to extract plugin package: %w", err)
+ }
+
+ ensurePluginDirPermissions(targetDir)
+ return nil
+}
+
+// Display helpers
+
+func displayPluginTableRow(w *tabwriter.Writer, discovery plugins.PluginDiscoveryEntry) {
+ if discovery.Error != nil {
+ // Handle global errors (like directory read failure)
+ if discovery.ID == "" {
+ log.Error("Failed to read plugins directory", "folder", conf.Server.Plugins.Folder, discovery.Error)
+ return
+ }
+ // Handle individual plugin errors - show them in the table
+ fmt.Fprintf(w, "%s\tERROR\tERROR\tERROR\tERROR\t%v\n", discovery.ID, discovery.Error)
+ return
+ }
+
+ // Mark symlinks with an indicator
+ nameDisplay := discovery.Manifest.Name
+ if discovery.IsSymlink {
+ nameDisplay = nameDisplay + " (dev)"
+ }
+
+ // Convert capabilities to strings
+ capabilities := slice.Map(discovery.Manifest.Capabilities, func(cap schema.PluginManifestCapabilitiesElem) string {
+ return string(cap)
+ })
+
+ fmt.Fprintf(w, "%s\t%s\t%s\t%s\t%s\t%s\n",
+ discovery.ID,
+ nameDisplay,
+ cmp.Or(discovery.Manifest.Author, "-"),
+ cmp.Or(discovery.Manifest.Version, "-"),
+ strings.Join(capabilities, ", "),
+ cmp.Or(discovery.Manifest.Description, "-"))
+}
+
+func displayTypedPermissions(permissions schema.PluginManifestPermissions, indent string) {
+ if permissions.Http != nil {
+ fmt.Printf("%shttp:\n", indent)
+ fmt.Printf("%s Reason: %s\n", indent, permissions.Http.Reason)
+ fmt.Printf("%s Allow Local Network: %t\n", indent, permissions.Http.AllowLocalNetwork)
+ fmt.Printf("%s Allowed URLs:\n", indent)
+ for urlPattern, methodEnums := range permissions.Http.AllowedUrls {
+ methods := make([]string, len(methodEnums))
+ for i, methodEnum := range methodEnums {
+ methods[i] = string(methodEnum)
+ }
+ fmt.Printf("%s %s: [%s]\n", indent, urlPattern, strings.Join(methods, ", "))
+ }
+ fmt.Println()
+ }
+
+ if permissions.Config != nil {
+ fmt.Printf("%sconfig:\n", indent)
+ fmt.Printf("%s Reason: %s\n", indent, permissions.Config.Reason)
+ fmt.Println()
+ }
+
+ if permissions.Scheduler != nil {
+ fmt.Printf("%sscheduler:\n", indent)
+ fmt.Printf("%s Reason: %s\n", indent, permissions.Scheduler.Reason)
+ fmt.Println()
+ }
+
+ if permissions.Websocket != nil {
+ fmt.Printf("%swebsocket:\n", indent)
+ fmt.Printf("%s Reason: %s\n", indent, permissions.Websocket.Reason)
+ fmt.Printf("%s Allow Local Network: %t\n", indent, permissions.Websocket.AllowLocalNetwork)
+ fmt.Printf("%s Allowed URLs: [%s]\n", indent, strings.Join(permissions.Websocket.AllowedUrls, ", "))
+ fmt.Println()
+ }
+
+ if permissions.Cache != nil {
+ fmt.Printf("%scache:\n", indent)
+ fmt.Printf("%s Reason: %s\n", indent, permissions.Cache.Reason)
+ fmt.Println()
+ }
+
+ if permissions.Artwork != nil {
+ fmt.Printf("%sartwork:\n", indent)
+ fmt.Printf("%s Reason: %s\n", indent, permissions.Artwork.Reason)
+ fmt.Println()
+ }
+
+ if permissions.Subsonicapi != nil {
+ allowedUsers := "All Users"
+ if len(permissions.Subsonicapi.AllowedUsernames) > 0 {
+ allowedUsers = strings.Join(permissions.Subsonicapi.AllowedUsernames, ", ")
+ }
+ fmt.Printf("%ssubsonicapi:\n", indent)
+ fmt.Printf("%s Reason: %s\n", indent, permissions.Subsonicapi.Reason)
+ fmt.Printf("%s Allow Admins: %t\n", indent, permissions.Subsonicapi.AllowAdmins)
+ fmt.Printf("%s Allowed Usernames: [%s]\n", indent, allowedUsers)
+ fmt.Println()
+ }
+}
+
+func displayPluginDetails(manifest *schema.PluginManifest, fileInfo *pluginFileInfo, permInfo *pluginPermissionInfo) {
+ fmt.Println("\nPlugin Information:")
+ fmt.Printf(" Name: %s\n", manifest.Name)
+ fmt.Printf(" Author: %s\n", manifest.Author)
+ fmt.Printf(" Version: %s\n", manifest.Version)
+ fmt.Printf(" Description: %s\n", manifest.Description)
+
+ fmt.Print(" Capabilities: ")
+ capabilities := make([]string, len(manifest.Capabilities))
+ for i, cap := range manifest.Capabilities {
+ capabilities[i] = string(cap)
+ }
+ fmt.Print(strings.Join(capabilities, ", "))
+ fmt.Println()
+
+ // Display manifest permissions using the typed permissions
+ fmt.Println(" Required Permissions:")
+ displayTypedPermissions(manifest.Permissions, " ")
+
+ // Print file information if available
+ if fileInfo != nil {
+ fmt.Println("Package Information:")
+ fmt.Printf(" File: %s\n", fileInfo.path)
+ fmt.Printf(" Size: %d bytes (%.2f KB)\n", fileInfo.size, float64(fileInfo.size)/1024)
+ fmt.Printf(" SHA-256: %s\n", fileInfo.hash)
+ fmt.Printf(" Modified: %s\n", fileInfo.modTime.Format(time.RFC3339))
+ }
+
+ // Print file permissions information if available
+ if permInfo != nil {
+ fmt.Println("File Permissions:")
+ fmt.Printf(" Plugin Directory: %s (%s)\n", permInfo.dirPath, permInfo.dirMode)
+ if permInfo.isSymlink {
+ fmt.Printf(" Symlink Target: %s (%s)\n", permInfo.targetPath, permInfo.targetMode)
+ }
+ fmt.Printf(" Manifest File: %s\n", permInfo.manifestMode)
+ if permInfo.wasmMode != "" {
+ fmt.Printf(" WASM File: %s\n", permInfo.wasmMode)
+ }
+ }
+}
+
+type pluginFileInfo struct {
+ path string
+ size int64
+ hash string
+ modTime time.Time
+}
+
+type pluginPermissionInfo struct {
+ dirPath string
+ dirMode string
+ isSymlink bool
+ targetPath string
+ targetMode string
+ manifestMode string
+ wasmMode string
+}
+
+func getFileInfo(path string) *pluginFileInfo {
+ fileInfo, err := os.Stat(path)
+ if err != nil {
+ log.Error("Failed to get file information", err)
+ return nil
+ }
+
+ return &pluginFileInfo{
+ path: path,
+ size: fileInfo.Size(),
+ hash: calculateSHA256(path),
+ modTime: fileInfo.ModTime(),
+ }
+}
+
+func getPermissionInfo(pluginDir string) *pluginPermissionInfo {
+ // Get plugin directory permissions
+ dirInfo, err := os.Lstat(pluginDir)
+ if err != nil {
+ log.Error("Failed to get plugin directory permissions", err)
+ return nil
+ }
+
+ permInfo := &pluginPermissionInfo{
+ dirPath: pluginDir,
+ dirMode: dirInfo.Mode().String(),
+ }
+
+ // Check if it's a symlink
+ if dirInfo.Mode()&os.ModeSymlink != 0 {
+ permInfo.isSymlink = true
+
+ // Get target path and permissions
+ targetPath, err := os.Readlink(pluginDir)
+ if err == nil {
+ if !filepath.IsAbs(targetPath) {
+ targetPath = filepath.Join(filepath.Dir(pluginDir), targetPath)
+ }
+ permInfo.targetPath = targetPath
+
+ if targetInfo, err := os.Stat(targetPath); err == nil {
+ permInfo.targetMode = targetInfo.Mode().String()
+ }
+ }
+ }
+
+ // Get manifest file permissions
+ manifestPath := filepath.Join(pluginDir, "manifest.json")
+ if manifestInfo, err := os.Stat(manifestPath); err == nil {
+ permInfo.manifestMode = manifestInfo.Mode().String()
+ }
+
+ // Get WASM file permissions (look for .wasm files)
+ entries, err := os.ReadDir(pluginDir)
+ if err == nil {
+ for _, entry := range entries {
+ if filepath.Ext(entry.Name()) == ".wasm" {
+ wasmPath := filepath.Join(pluginDir, entry.Name())
+ if wasmInfo, err := os.Stat(wasmPath); err == nil {
+ permInfo.wasmMode = wasmInfo.Mode().String()
+ break // Just show the first WASM file found
+ }
+ }
+ }
+ }
+
+ return permInfo
+}
+
+// Command implementations
+
+func pluginList(cmd *cobra.Command, args []string) {
+ discoveries := plugins.DiscoverPlugins(conf.Server.Plugins.Folder)
+
+ w := tabwriter.NewWriter(os.Stdout, 0, 0, 2, ' ', 0)
+ fmt.Fprintln(w, "ID\tNAME\tAUTHOR\tVERSION\tCAPABILITIES\tDESCRIPTION")
+
+ for _, discovery := range discoveries {
+ displayPluginTableRow(w, discovery)
+ }
+ w.Flush()
+}
+
+func pluginInfo(cmd *cobra.Command, args []string) {
+ path := args[0]
+ pluginsDir := conf.Server.Plugins.Folder
+
+ var manifest *schema.PluginManifest
+ var fileInfo *pluginFileInfo
+ var permInfo *pluginPermissionInfo
+
+ if filepath.Ext(path) == pluginPackageExtension {
+ // It's a package file
+ pkg, err := loadAndValidatePackage(path)
+ if err != nil {
+ log.Fatal("Failed to load plugin package", err)
+ }
+ manifest = pkg.Manifest
+ fileInfo = getFileInfo(path)
+ // No permission info for package files
+ } else {
+ // It's a plugin name
+ pluginDir, err := validatePluginDirectory(pluginsDir, path)
+ if err != nil {
+ log.Fatal("Plugin validation failed", err)
+ }
+
+ manifest, err = plugins.LoadManifest(pluginDir)
+ if err != nil {
+ log.Fatal("Failed to load plugin manifest", err)
+ }
+
+ // Get permission info for installed plugins
+ permInfo = getPermissionInfo(pluginDir)
+ }
+
+ displayPluginDetails(manifest, fileInfo, permInfo)
+}
+
+func pluginInstall(cmd *cobra.Command, args []string) {
+ ndpPath := args[0]
+ pluginsDir := conf.Server.Plugins.Folder
+
+ pkg, err := loadAndValidatePackage(ndpPath)
+ if err != nil {
+ log.Fatal("Package validation failed", err)
+ }
+
+ // Create target directory based on plugin name
+ targetDir := filepath.Join(pluginsDir, pkg.Manifest.Name)
+
+ // Check if plugin already exists
+ if utils.FileExists(targetDir) {
+ log.Fatal("Plugin already installed", "name", pkg.Manifest.Name, "path", targetDir,
+ "use", "navidrome plugin update")
+ }
+
+ if err := extractAndSetupPlugin(ndpPath, targetDir); err != nil {
+ log.Fatal("Plugin installation failed", err)
+ }
+
+ fmt.Printf("Plugin '%s' v%s installed successfully\n", pkg.Manifest.Name, pkg.Manifest.Version)
+}
+
+func pluginRemove(cmd *cobra.Command, args []string) {
+ pluginName := args[0]
+ pluginsDir := conf.Server.Plugins.Folder
+
+ pluginDir, err := validatePluginDirectory(pluginsDir, pluginName)
+ if err != nil {
+ log.Fatal("Plugin validation failed", err)
+ }
+
+ _, isSymlink, err := resolvePluginPath(pluginDir)
+ if err != nil {
+ log.Fatal("Failed to resolve plugin path", err)
+ }
+
+ if isSymlink {
+ // For symlinked plugins (dev mode), just remove the symlink
+ if err := os.Remove(pluginDir); err != nil {
+ log.Fatal("Failed to remove plugin symlink", "name", pluginName, err)
+ }
+ fmt.Printf("Development plugin symlink '%s' removed successfully (target directory preserved)\n", pluginName)
+ } else {
+ // For regular plugins, remove the entire directory
+ if err := os.RemoveAll(pluginDir); err != nil {
+ log.Fatal("Failed to remove plugin directory", "name", pluginName, err)
+ }
+ fmt.Printf("Plugin '%s' removed successfully\n", pluginName)
+ }
+}
+
+func pluginUpdate(cmd *cobra.Command, args []string) {
+ ndpPath := args[0]
+ pluginsDir := conf.Server.Plugins.Folder
+
+ pkg, err := loadAndValidatePackage(ndpPath)
+ if err != nil {
+ log.Fatal("Package validation failed", err)
+ }
+
+ // Check if plugin exists
+ targetDir := filepath.Join(pluginsDir, pkg.Manifest.Name)
+ if !utils.FileExists(targetDir) {
+ log.Fatal("Plugin not found", "name", pkg.Manifest.Name, "path", targetDir,
+ "use", "navidrome plugin install")
+ }
+
+ // Create a backup of the existing plugin
+ backupDir := targetDir + ".bak." + time.Now().Format("20060102150405")
+ if err := os.Rename(targetDir, backupDir); err != nil {
+ log.Fatal("Failed to backup existing plugin", err)
+ }
+
+ // Extract the new package
+ if err := extractAndSetupPlugin(ndpPath, targetDir); err != nil {
+ // Restore backup if extraction failed
+ os.RemoveAll(targetDir)
+ _ = os.Rename(backupDir, targetDir) // Ignore error as we're already in a fatal path
+ log.Fatal("Plugin update failed", err)
+ }
+
+ // Remove the backup
+ os.RemoveAll(backupDir)
+
+ fmt.Printf("Plugin '%s' updated to v%s successfully\n", pkg.Manifest.Name, pkg.Manifest.Version)
+}
+
+func pluginRefresh(cmd *cobra.Command, args []string) {
+ pluginName := args[0]
+ pluginsDir := conf.Server.Plugins.Folder
+
+ pluginDir, err := validatePluginDirectory(pluginsDir, pluginName)
+ if err != nil {
+ log.Fatal("Plugin validation failed", err)
+ }
+
+ resolvedPath, isSymlink, err := resolvePluginPath(pluginDir)
+ if err != nil {
+ log.Fatal("Failed to resolve plugin path", err)
+ }
+
+ if isSymlink {
+ log.Debug("Processing symlinked plugin", "name", pluginName, "link", pluginDir, "target", resolvedPath)
+ }
+
+ fmt.Printf("Refreshing plugin '%s'...\n", pluginName)
+
+ // Get the plugin manager and refresh
+ mgr := GetPluginManager(cmd.Context())
+ log.Debug("Scanning plugins directory", "path", pluginsDir)
+ mgr.ScanPlugins()
+
+ log.Info("Waiting for plugin compilation to complete", "name", pluginName)
+
+ // Wait for compilation to complete
+ if err := mgr.EnsureCompiled(pluginName); err != nil {
+ log.Fatal("Failed to compile refreshed plugin", "name", pluginName, err)
+ }
+
+ log.Info("Plugin compilation completed successfully", "name", pluginName)
+ fmt.Printf("Plugin '%s' refreshed successfully\n", pluginName)
+}
+
+func pluginDev(cmd *cobra.Command, args []string) {
+ sourcePath, err := filepath.Abs(args[0])
+ if err != nil {
+ log.Fatal("Invalid path", "path", args[0], err)
+ }
+ pluginsDir := conf.Server.Plugins.Folder
+
+ // Validate source directory and manifest
+ if err := validateDevSource(sourcePath); err != nil {
+ log.Fatal("Source validation failed", err)
+ }
+
+ // Load manifest to get plugin name
+ manifest, err := plugins.LoadManifest(sourcePath)
+ if err != nil {
+ log.Fatal("Failed to load plugin manifest", "path", filepath.Join(sourcePath, "manifest.json"), err)
+ }
+
+ pluginName := cmp.Or(manifest.Name, filepath.Base(sourcePath))
+ targetPath := filepath.Join(pluginsDir, pluginName)
+
+ // Handle existing target
+ if err := handleExistingTarget(targetPath, sourcePath); err != nil {
+ log.Fatal("Failed to handle existing target", err)
+ }
+
+ // Create target directory if needed
+ if err := os.MkdirAll(filepath.Dir(targetPath), 0755); err != nil {
+ log.Fatal("Failed to create plugins directory", "path", filepath.Dir(targetPath), err)
+ }
+
+ // Create the symlink
+ if err := os.Symlink(sourcePath, targetPath); err != nil {
+ log.Fatal("Failed to create symlink", "source", sourcePath, "target", targetPath, err)
+ }
+
+ fmt.Printf("Development symlink created: '%s' -> '%s'\n", targetPath, sourcePath)
+ fmt.Println("Plugin can be refreshed with: navidrome plugin refresh", pluginName)
+}
+
+// Utility functions
+
+func validateDevSource(sourcePath string) error {
+ sourceInfo, err := os.Stat(sourcePath)
+ if err != nil {
+ return fmt.Errorf("source folder not found: %s (%w)", sourcePath, err)
+ }
+ if !sourceInfo.IsDir() {
+ return fmt.Errorf("source path is not a directory: %s", sourcePath)
+ }
+
+ manifestPath := filepath.Join(sourcePath, "manifest.json")
+ if !utils.FileExists(manifestPath) {
+ return fmt.Errorf("source folder missing manifest.json: %s", sourcePath)
+ }
+
+ return nil
+}
+
+func handleExistingTarget(targetPath, sourcePath string) error {
+ if !utils.FileExists(targetPath) {
+ return nil // Nothing to handle
+ }
+
+ // Check if it's already a symlink to our source
+ existingLink, err := os.Readlink(targetPath)
+ if err == nil && existingLink == sourcePath {
+ fmt.Printf("Symlink already exists and points to the correct source\n")
+ return fmt.Errorf("symlink already exists") // This will cause early return in caller
+ }
+
+ // Handle case where target exists but is not a symlink to our source
+ fmt.Printf("Target path '%s' already exists.\n", targetPath)
+ fmt.Print("Do you want to replace it? (y/N): ")
+ var response string
+ _, err = fmt.Scanln(&response)
+ if err != nil || strings.ToLower(response) != "y" {
+ if err != nil {
+ log.Debug("Error reading input, assuming 'no'", err)
+ }
+ return fmt.Errorf("operation canceled")
+ }
+
+ // Remove existing target
+ if err := os.RemoveAll(targetPath); err != nil {
+ return fmt.Errorf("failed to remove existing target %s: %w", targetPath, err)
+ }
+
+ return nil
+}
+
+func ensurePluginDirPermissions(dir string) {
+ if err := os.Chmod(dir, pluginDirPermissions); err != nil {
+ log.Error("Failed to set plugin directory permissions", "dir", dir, err)
+ }
+
+ // Apply permissions to all files in the directory
+ entries, err := os.ReadDir(dir)
+ if err != nil {
+ log.Error("Failed to read plugin directory", "dir", dir, err)
+ return
+ }
+
+ for _, entry := range entries {
+ path := filepath.Join(dir, entry.Name())
+ info, err := os.Stat(path)
+ if err != nil {
+ log.Error("Failed to stat file", "path", path, err)
+ continue
+ }
+
+ mode := os.FileMode(pluginFilePermissions) // Files
+ if info.IsDir() {
+ mode = os.FileMode(pluginDirPermissions) // Directories
+ ensurePluginDirPermissions(path) // Recursive
+ }
+
+ if err := os.Chmod(path, mode); err != nil {
+ log.Error("Failed to set file permissions", "path", path, err)
+ }
+ }
+}
+
+func calculateSHA256(filePath string) string {
+ file, err := os.Open(filePath)
+ if err != nil {
+ log.Error("Failed to open file for hashing", err)
+ return "N/A"
+ }
+ defer file.Close()
+
+ hasher := sha256.New()
+ if _, err := io.Copy(hasher, file); err != nil {
+ log.Error("Failed to calculate hash", err)
+ return "N/A"
+ }
+
+ return hex.EncodeToString(hasher.Sum(nil))
+}
diff --git a/cmd/plugin_test.go b/cmd/plugin_test.go
new file mode 100644
index 000000000..3a4aefa88
--- /dev/null
+++ b/cmd/plugin_test.go
@@ -0,0 +1,193 @@
+package cmd
+
+import (
+ "io"
+ "os"
+ "path/filepath"
+ "strings"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+ "github.com/spf13/cobra"
+)
+
+var _ = Describe("Plugin CLI Commands", func() {
+ var tempDir string
+ var cmd *cobra.Command
+ var stdOut *os.File
+ var origStdout *os.File
+ var outReader *os.File
+
+ // Helper to create a test plugin with the given name and details
+ createTestPlugin := func(name, author, version string, capabilities []string) string {
+ pluginDir := filepath.Join(tempDir, name)
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+
+ // Create a properly formatted capabilities JSON array
+ capabilitiesJSON := `"` + strings.Join(capabilities, `", "`) + `"`
+
+ manifest := `{
+ "name": "` + name + `",
+ "author": "` + author + `",
+ "version": "` + version + `",
+ "description": "Plugin for testing",
+ "website": "https://test.navidrome.org/` + name + `",
+ "capabilities": [` + capabilitiesJSON + `],
+ "permissions": {}
+ }`
+
+ Expect(os.WriteFile(filepath.Join(pluginDir, "manifest.json"), []byte(manifest), 0600)).To(Succeed())
+
+ // Create a dummy WASM file
+ wasmContent := []byte("dummy wasm content for testing")
+ Expect(os.WriteFile(filepath.Join(pluginDir, "plugin.wasm"), wasmContent, 0600)).To(Succeed())
+
+ return pluginDir
+ }
+
+ // Helper to execute a command and return captured output
+ captureOutput := func(reader io.Reader) string {
+ stdOut.Close()
+ outputBytes, err := io.ReadAll(reader)
+ Expect(err).NotTo(HaveOccurred())
+ return string(outputBytes)
+ }
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ tempDir = GinkgoT().TempDir()
+
+ // Setup config
+ conf.Server.Plugins.Enabled = true
+ conf.Server.Plugins.Folder = tempDir
+
+ // Create a command for testing
+ cmd = &cobra.Command{Use: "test"}
+
+ // Setup stdout capture
+ origStdout = os.Stdout
+ var err error
+ outReader, stdOut, err = os.Pipe()
+ Expect(err).NotTo(HaveOccurred())
+ os.Stdout = stdOut
+
+ DeferCleanup(func() {
+ os.Stdout = origStdout
+ })
+ })
+
+ AfterEach(func() {
+ os.Stdout = origStdout
+ if stdOut != nil {
+ stdOut.Close()
+ }
+ if outReader != nil {
+ outReader.Close()
+ }
+ })
+
+ Describe("Plugin list command", func() {
+ It("should list installed plugins", func() {
+ // Create test plugins
+ createTestPlugin("plugin1", "Test Author", "1.0.0", []string{"MetadataAgent"})
+ createTestPlugin("plugin2", "Another Author", "2.1.0", []string{"Scrobbler"})
+
+ // Execute command
+ pluginList(cmd, []string{})
+
+ // Verify output
+ output := captureOutput(outReader)
+
+ Expect(output).To(ContainSubstring("plugin1"))
+ Expect(output).To(ContainSubstring("Test Author"))
+ Expect(output).To(ContainSubstring("1.0.0"))
+ Expect(output).To(ContainSubstring("MetadataAgent"))
+
+ Expect(output).To(ContainSubstring("plugin2"))
+ Expect(output).To(ContainSubstring("Another Author"))
+ Expect(output).To(ContainSubstring("2.1.0"))
+ Expect(output).To(ContainSubstring("Scrobbler"))
+ })
+ })
+
+ Describe("Plugin info command", func() {
+ It("should display information about an installed plugin", func() {
+ // Create test plugin with multiple capabilities
+ createTestPlugin("test-plugin", "Test Author", "1.0.0",
+ []string{"MetadataAgent", "Scrobbler"})
+
+ // Execute command
+ pluginInfo(cmd, []string{"test-plugin"})
+
+ // Verify output
+ output := captureOutput(outReader)
+
+ Expect(output).To(ContainSubstring("Name: test-plugin"))
+ Expect(output).To(ContainSubstring("Author: Test Author"))
+ Expect(output).To(ContainSubstring("Version: 1.0.0"))
+ Expect(output).To(ContainSubstring("Description: Plugin for testing"))
+ Expect(output).To(ContainSubstring("Capabilities: MetadataAgent, Scrobbler"))
+ })
+ })
+
+ Describe("Plugin remove command", func() {
+ It("should remove a regular plugin directory", func() {
+ // Create test plugin
+ pluginDir := createTestPlugin("regular-plugin", "Test Author", "1.0.0",
+ []string{"MetadataAgent"})
+
+ // Execute command
+ pluginRemove(cmd, []string{"regular-plugin"})
+
+ // Verify output
+ output := captureOutput(outReader)
+ Expect(output).To(ContainSubstring("Plugin 'regular-plugin' removed successfully"))
+
+ // Verify directory is actually removed
+ _, err := os.Stat(pluginDir)
+ Expect(os.IsNotExist(err)).To(BeTrue())
+ })
+
+ It("should remove only the symlink for a development plugin", func() {
+ // Create a real source directory
+ sourceDir := filepath.Join(GinkgoT().TempDir(), "dev-plugin-source")
+ Expect(os.MkdirAll(sourceDir, 0755)).To(Succeed())
+
+ manifest := `{
+ "name": "dev-plugin",
+ "author": "Dev Author",
+ "version": "0.1.0",
+ "description": "Development plugin for testing",
+ "website": "https://test.navidrome.org/dev-plugin",
+ "capabilities": ["Scrobbler"],
+ "permissions": {}
+ }`
+ Expect(os.WriteFile(filepath.Join(sourceDir, "manifest.json"), []byte(manifest), 0600)).To(Succeed())
+
+ // Create a dummy WASM file
+ wasmContent := []byte("dummy wasm content for testing")
+ Expect(os.WriteFile(filepath.Join(sourceDir, "plugin.wasm"), wasmContent, 0600)).To(Succeed())
+
+ // Create a symlink in the plugins directory
+ symlinkPath := filepath.Join(tempDir, "dev-plugin")
+ Expect(os.Symlink(sourceDir, symlinkPath)).To(Succeed())
+
+ // Execute command
+ pluginRemove(cmd, []string{"dev-plugin"})
+
+ // Verify output
+ output := captureOutput(outReader)
+ Expect(output).To(ContainSubstring("Development plugin symlink 'dev-plugin' removed successfully"))
+ Expect(output).To(ContainSubstring("target directory preserved"))
+
+ // Verify the symlink is removed but source directory exists
+ _, err := os.Lstat(symlinkPath)
+ Expect(os.IsNotExist(err)).To(BeTrue())
+
+ _, err = os.Stat(sourceDir)
+ Expect(err).NotTo(HaveOccurred())
+ })
+ })
+})
diff --git a/cmd/root.go b/cmd/root.go
index e1e92228f..9618b16e6 100644
--- a/cmd/root.go
+++ b/cmd/root.go
@@ -82,8 +82,9 @@ func runNavidrome(ctx context.Context) {
g.Go(schedulePeriodicBackup(ctx))
g.Go(startInsightsCollector(ctx))
g.Go(scheduleDBOptimizer(ctx))
+ g.Go(startPluginManager(ctx))
+ g.Go(runInitialScan(ctx))
if conf.Server.Scanner.Enabled {
- g.Go(runInitialScan(ctx))
g.Go(startScanWatcher(ctx))
g.Go(schedulePeriodicScan(ctx))
} else {
@@ -109,7 +110,7 @@ func mainContext(ctx context.Context) (context.Context, context.CancelFunc) {
func startServer(ctx context.Context) func() error {
return func() error {
a := CreateServer()
- a.MountRouter("Native API", consts.URLPathNativeAPI, CreateNativeAPIRouter())
+ a.MountRouter("Native API", consts.URLPathNativeAPI, CreateNativeAPIRouter(ctx))
a.MountRouter("Subsonic API", consts.URLPathSubsonicAPI, CreateSubsonicAPIRouter(ctx))
a.MountRouter("Public Endpoints", consts.URLPathPublic, CreatePublicRouter())
if conf.Server.LastFM.Enabled {
@@ -147,7 +148,7 @@ func schedulePeriodicScan(ctx context.Context) func() error {
schedulerInstance := scheduler.GetInstance()
log.Info("Scheduling periodic scan", "schedule", schedule)
- err := schedulerInstance.Add(schedule, func() {
+ _, err := schedulerInstance.Add(schedule, func() {
_, err := s.ScanAll(ctx, false)
if err != nil {
log.Error(ctx, "Error executing periodic scan", err)
@@ -172,6 +173,7 @@ func pidHashChanged(ds model.DataStore) (bool, error) {
return !strings.EqualFold(pidAlbum, conf.Server.PID.Album) || !strings.EqualFold(pidTrack, conf.Server.PID.Track), nil
}
+// runInitialScan runs an initial scan of the music library if needed.
func runInitialScan(ctx context.Context) func() error {
return func() error {
ds := CreateDataStore()
@@ -190,7 +192,7 @@ func runInitialScan(ctx context.Context) func() error {
scanNeeded := conf.Server.Scanner.ScanOnStartup || inProgress || fullScanRequired == "1" || pidHasChanged
time.Sleep(2 * time.Second) // Wait 2 seconds before the initial scan
if scanNeeded {
- scanner := CreateScanner(ctx)
+ s := CreateScanner(ctx)
switch {
case fullScanRequired == "1":
log.Warn(ctx, "Full scan required after migration")
@@ -204,7 +206,7 @@ func runInitialScan(ctx context.Context) func() error {
log.Info("Executing initial scan")
}
- _, err = scanner.ScanAll(ctx, fullScanRequired == "1")
+ _, err = s.ScanAll(ctx, fullScanRequired == "1")
if err != nil {
log.Error(ctx, "Scan failed", err)
} else {
@@ -243,7 +245,7 @@ func schedulePeriodicBackup(ctx context.Context) func() error {
schedulerInstance := scheduler.GetInstance()
log.Info("Scheduling periodic backup", "schedule", schedule)
- err := schedulerInstance.Add(schedule, func() {
+ _, err := schedulerInstance.Add(schedule, func() {
start := time.Now()
path, err := db.Backup(ctx)
elapsed := time.Since(start)
@@ -271,7 +273,7 @@ func scheduleDBOptimizer(ctx context.Context) func() error {
return func() error {
log.Info(ctx, "Scheduling DB optimizer", "schedule", consts.OptimizeDBSchedule)
schedulerInstance := scheduler.GetInstance()
- err := schedulerInstance.Add(consts.OptimizeDBSchedule, func() {
+ _, err := schedulerInstance.Add(consts.OptimizeDBSchedule, func() {
if scanner.IsScanning() {
log.Debug(ctx, "Skipping DB optimization because a scan is in progress")
return
@@ -325,6 +327,22 @@ func startPlaybackServer(ctx context.Context) func() error {
}
}
+// startPluginManager starts the plugin manager, if configured.
+func startPluginManager(ctx context.Context) func() error {
+ return func() error {
+ if !conf.Server.Plugins.Enabled {
+ log.Debug("Plugins are DISABLED")
+ return nil
+ }
+ log.Info(ctx, "Starting plugin manager")
+ // Get the manager instance and scan for plugins
+ manager := GetPluginManager(ctx)
+ manager.ScanPlugins()
+
+ return nil
+ }
+}
+
// TODO: Implement some struct tags to map flags to viper
func init() {
cobra.OnInitialize(func() {
diff --git a/cmd/scan.go b/cmd/scan.go
index 26eb7d7a2..41d281070 100644
--- a/cmd/scan.go
+++ b/cmd/scan.go
@@ -4,12 +4,12 @@ import (
"context"
"encoding/gob"
"os"
+ "strings"
"github.com/navidrome/navidrome/core"
- "github.com/navidrome/navidrome/core/artwork"
- "github.com/navidrome/navidrome/core/metrics"
"github.com/navidrome/navidrome/db"
"github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/persistence"
"github.com/navidrome/navidrome/scanner"
"github.com/navidrome/navidrome/utils/pl"
@@ -19,11 +19,13 @@ import (
var (
fullScan bool
subprocess bool
+ targets string
)
func init() {
scanCmd.Flags().BoolVarP(&fullScan, "full", "f", false, "check all subfolders, ignoring timestamps")
scanCmd.Flags().BoolVarP(&subprocess, "subprocess", "", false, "run as subprocess (internal use)")
+ scanCmd.Flags().StringVarP(&targets, "targets", "t", "", "comma-separated list of libraryID:folderPath pairs (e.g., \"1:Music/Rock,1:Music/Jazz,2:Classical\")")
rootCmd.AddCommand(scanCmd)
}
@@ -70,7 +72,18 @@ func runScanner(ctx context.Context) {
ds := persistence.New(sqlDB)
pls := core.NewPlaylists(ds)
- progress, err := scanner.CallScan(ctx, ds, artwork.NoopCacheWarmer(), pls, metrics.NewNoopInstance(), fullScan)
+ // Parse targets if provided
+ var scanTargets []model.ScanTarget
+ if targets != "" {
+ var err error
+ scanTargets, err = model.ParseTargets(strings.Split(targets, ","))
+ if err != nil {
+ log.Fatal(ctx, "Failed to parse targets", err)
+ }
+ log.Info(ctx, "Scanning specific folders", "numTargets", len(scanTargets))
+ }
+
+ progress, err := scanner.CallScan(ctx, ds, pls, fullScan, scanTargets)
if err != nil {
log.Fatal(ctx, "Failed to scan", err)
}
diff --git a/cmd/wire_gen.go b/cmd/wire_gen.go
index d57aadc71..d7b6a3ad2 100644
--- a/cmd/wire_gen.go
+++ b/cmd/wire_gen.go
@@ -22,6 +22,7 @@ import (
"github.com/navidrome/navidrome/db"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/persistence"
+ "github.com/navidrome/navidrome/plugins"
"github.com/navidrome/navidrome/scanner"
"github.com/navidrome/navidrome/server"
"github.com/navidrome/navidrome/server/events"
@@ -46,18 +47,33 @@ func CreateServer() *server.Server {
sqlDB := db.Db()
dataStore := persistence.New(sqlDB)
broker := events.GetBroker()
- insights := metrics.GetInstance(dataStore)
+ metricsMetrics := metrics.GetPrometheusInstance(dataStore)
+ manager := plugins.GetManager(dataStore, metricsMetrics)
+ insights := metrics.GetInstance(dataStore, manager)
serverServer := server.New(dataStore, broker, insights)
return serverServer
}
-func CreateNativeAPIRouter() *nativeapi.Router {
+func CreateNativeAPIRouter(ctx context.Context) *nativeapi.Router {
sqlDB := db.Db()
dataStore := persistence.New(sqlDB)
share := core.NewShare(dataStore)
playlists := core.NewPlaylists(dataStore)
- insights := metrics.GetInstance(dataStore)
- router := nativeapi.New(dataStore, share, playlists, insights)
+ metricsMetrics := metrics.GetPrometheusInstance(dataStore)
+ manager := plugins.GetManager(dataStore, metricsMetrics)
+ insights := metrics.GetInstance(dataStore, manager)
+ fileCache := artwork.GetImageCache()
+ fFmpeg := ffmpeg.New()
+ agentsAgents := agents.GetAgents(dataStore, manager)
+ provider := external.NewProvider(dataStore, agentsAgents)
+ artworkArtwork := artwork.NewArtwork(dataStore, fileCache, fFmpeg, provider)
+ cacheWarmer := artwork.NewCacheWarmer(artworkArtwork, fileCache)
+ broker := events.GetBroker()
+ modelScanner := scanner.New(ctx, dataStore, cacheWarmer, broker, playlists, metricsMetrics)
+ watcher := scanner.GetWatcher(dataStore, modelScanner)
+ library := core.NewLibrary(dataStore, modelScanner, watcher, broker)
+ maintenance := core.NewMaintenance(dataStore)
+ router := nativeapi.New(dataStore, share, playlists, insights, library, maintenance)
return router
}
@@ -66,7 +82,9 @@ func CreateSubsonicAPIRouter(ctx context.Context) *subsonic.Router {
dataStore := persistence.New(sqlDB)
fileCache := artwork.GetImageCache()
fFmpeg := ffmpeg.New()
- agentsAgents := agents.GetAgents(dataStore)
+ metricsMetrics := metrics.GetPrometheusInstance(dataStore)
+ manager := plugins.GetManager(dataStore, metricsMetrics)
+ agentsAgents := agents.GetAgents(dataStore, manager)
provider := external.NewProvider(dataStore, agentsAgents)
artworkArtwork := artwork.NewArtwork(dataStore, fileCache, fFmpeg, provider)
transcodingCache := core.GetTranscodingCache()
@@ -77,11 +95,10 @@ func CreateSubsonicAPIRouter(ctx context.Context) *subsonic.Router {
cacheWarmer := artwork.NewCacheWarmer(artworkArtwork, fileCache)
broker := events.GetBroker()
playlists := core.NewPlaylists(dataStore)
- metricsMetrics := metrics.NewPrometheusInstance(dataStore)
- scannerScanner := scanner.New(ctx, dataStore, cacheWarmer, broker, playlists, metricsMetrics)
- playTracker := scrobbler.GetPlayTracker(dataStore, broker)
+ modelScanner := scanner.New(ctx, dataStore, cacheWarmer, broker, playlists, metricsMetrics)
+ playTracker := scrobbler.GetPlayTracker(dataStore, broker, manager)
playbackServer := playback.GetInstance(dataStore)
- router := subsonic.New(dataStore, artworkArtwork, mediaStreamer, archiver, players, provider, scannerScanner, broker, playlists, playTracker, share, playbackServer)
+ router := subsonic.New(dataStore, artworkArtwork, mediaStreamer, archiver, players, provider, modelScanner, broker, playlists, playTracker, share, playbackServer, metricsMetrics)
return router
}
@@ -90,7 +107,9 @@ func CreatePublicRouter() *public.Router {
dataStore := persistence.New(sqlDB)
fileCache := artwork.GetImageCache()
fFmpeg := ffmpeg.New()
- agentsAgents := agents.GetAgents(dataStore)
+ metricsMetrics := metrics.GetPrometheusInstance(dataStore)
+ manager := plugins.GetManager(dataStore, metricsMetrics)
+ agentsAgents := agents.GetAgents(dataStore, manager)
provider := external.NewProvider(dataStore, agentsAgents)
artworkArtwork := artwork.NewArtwork(dataStore, fileCache, fFmpeg, provider)
transcodingCache := core.GetTranscodingCache()
@@ -118,31 +137,34 @@ func CreateListenBrainzRouter() *listenbrainz.Router {
func CreateInsights() metrics.Insights {
sqlDB := db.Db()
dataStore := persistence.New(sqlDB)
- insights := metrics.GetInstance(dataStore)
+ metricsMetrics := metrics.GetPrometheusInstance(dataStore)
+ manager := plugins.GetManager(dataStore, metricsMetrics)
+ insights := metrics.GetInstance(dataStore, manager)
return insights
}
func CreatePrometheus() metrics.Metrics {
sqlDB := db.Db()
dataStore := persistence.New(sqlDB)
- metricsMetrics := metrics.NewPrometheusInstance(dataStore)
+ metricsMetrics := metrics.GetPrometheusInstance(dataStore)
return metricsMetrics
}
-func CreateScanner(ctx context.Context) scanner.Scanner {
+func CreateScanner(ctx context.Context) model.Scanner {
sqlDB := db.Db()
dataStore := persistence.New(sqlDB)
fileCache := artwork.GetImageCache()
fFmpeg := ffmpeg.New()
- agentsAgents := agents.GetAgents(dataStore)
+ metricsMetrics := metrics.GetPrometheusInstance(dataStore)
+ manager := plugins.GetManager(dataStore, metricsMetrics)
+ agentsAgents := agents.GetAgents(dataStore, manager)
provider := external.NewProvider(dataStore, agentsAgents)
artworkArtwork := artwork.NewArtwork(dataStore, fileCache, fFmpeg, provider)
cacheWarmer := artwork.NewCacheWarmer(artworkArtwork, fileCache)
broker := events.GetBroker()
playlists := core.NewPlaylists(dataStore)
- metricsMetrics := metrics.NewPrometheusInstance(dataStore)
- scannerScanner := scanner.New(ctx, dataStore, cacheWarmer, broker, playlists, metricsMetrics)
- return scannerScanner
+ modelScanner := scanner.New(ctx, dataStore, cacheWarmer, broker, playlists, metricsMetrics)
+ return modelScanner
}
func CreateScanWatcher(ctx context.Context) scanner.Watcher {
@@ -150,15 +172,16 @@ func CreateScanWatcher(ctx context.Context) scanner.Watcher {
dataStore := persistence.New(sqlDB)
fileCache := artwork.GetImageCache()
fFmpeg := ffmpeg.New()
- agentsAgents := agents.GetAgents(dataStore)
+ metricsMetrics := metrics.GetPrometheusInstance(dataStore)
+ manager := plugins.GetManager(dataStore, metricsMetrics)
+ agentsAgents := agents.GetAgents(dataStore, manager)
provider := external.NewProvider(dataStore, agentsAgents)
artworkArtwork := artwork.NewArtwork(dataStore, fileCache, fFmpeg, provider)
cacheWarmer := artwork.NewCacheWarmer(artworkArtwork, fileCache)
broker := events.GetBroker()
playlists := core.NewPlaylists(dataStore)
- metricsMetrics := metrics.NewPrometheusInstance(dataStore)
- scannerScanner := scanner.New(ctx, dataStore, cacheWarmer, broker, playlists, metricsMetrics)
- watcher := scanner.NewWatcher(dataStore, scannerScanner)
+ modelScanner := scanner.New(ctx, dataStore, cacheWarmer, broker, playlists, metricsMetrics)
+ watcher := scanner.GetWatcher(dataStore, modelScanner)
return watcher
}
@@ -169,6 +192,20 @@ func GetPlaybackServer() playback.PlaybackServer {
return playbackServer
}
+func getPluginManager() plugins.Manager {
+ sqlDB := db.Db()
+ dataStore := persistence.New(sqlDB)
+ metricsMetrics := metrics.GetPrometheusInstance(dataStore)
+ manager := plugins.GetManager(dataStore, metricsMetrics)
+ return manager
+}
+
// wire_injectors.go:
-var allProviders = wire.NewSet(core.Set, artwork.Set, server.New, subsonic.New, nativeapi.New, public.New, persistence.New, lastfm.NewRouter, listenbrainz.NewRouter, events.GetBroker, scanner.New, scanner.NewWatcher, metrics.NewPrometheusInstance, db.Db)
+var allProviders = wire.NewSet(core.Set, artwork.Set, server.New, subsonic.New, nativeapi.New, public.New, persistence.New, lastfm.NewRouter, listenbrainz.NewRouter, events.GetBroker, scanner.New, scanner.GetWatcher, plugins.GetManager, metrics.GetPrometheusInstance, db.Db, wire.Bind(new(agents.PluginLoader), new(plugins.Manager)), wire.Bind(new(scrobbler.PluginLoader), new(plugins.Manager)), wire.Bind(new(metrics.PluginLoader), new(plugins.Manager)), wire.Bind(new(core.Watcher), new(scanner.Watcher)))
+
+func GetPluginManager(ctx context.Context) plugins.Manager {
+ manager := getPluginManager()
+ manager.SetSubsonicRouter(CreateSubsonicAPIRouter(ctx))
+ return manager
+}
diff --git a/cmd/wire_injectors.go b/cmd/wire_injectors.go
index c431945dc..595d406b9 100644
--- a/cmd/wire_injectors.go
+++ b/cmd/wire_injectors.go
@@ -7,14 +7,17 @@ import (
"github.com/google/wire"
"github.com/navidrome/navidrome/core"
+ "github.com/navidrome/navidrome/core/agents"
"github.com/navidrome/navidrome/core/agents/lastfm"
"github.com/navidrome/navidrome/core/agents/listenbrainz"
"github.com/navidrome/navidrome/core/artwork"
"github.com/navidrome/navidrome/core/metrics"
"github.com/navidrome/navidrome/core/playback"
+ "github.com/navidrome/navidrome/core/scrobbler"
"github.com/navidrome/navidrome/db"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/persistence"
+ "github.com/navidrome/navidrome/plugins"
"github.com/navidrome/navidrome/scanner"
"github.com/navidrome/navidrome/server"
"github.com/navidrome/navidrome/server/events"
@@ -35,9 +38,14 @@ var allProviders = wire.NewSet(
listenbrainz.NewRouter,
events.GetBroker,
scanner.New,
- scanner.NewWatcher,
- metrics.NewPrometheusInstance,
+ scanner.GetWatcher,
+ plugins.GetManager,
+ metrics.GetPrometheusInstance,
db.Db,
+ wire.Bind(new(agents.PluginLoader), new(plugins.Manager)),
+ wire.Bind(new(scrobbler.PluginLoader), new(plugins.Manager)),
+ wire.Bind(new(metrics.PluginLoader), new(plugins.Manager)),
+ wire.Bind(new(core.Watcher), new(scanner.Watcher)),
)
func CreateDataStore() model.DataStore {
@@ -52,7 +60,7 @@ func CreateServer() *server.Server {
))
}
-func CreateNativeAPIRouter() *nativeapi.Router {
+func CreateNativeAPIRouter(ctx context.Context) *nativeapi.Router {
panic(wire.Build(
allProviders,
))
@@ -94,7 +102,7 @@ func CreatePrometheus() metrics.Metrics {
))
}
-func CreateScanner(ctx context.Context) scanner.Scanner {
+func CreateScanner(ctx context.Context) model.Scanner {
panic(wire.Build(
allProviders,
))
@@ -111,3 +119,15 @@ func GetPlaybackServer() playback.PlaybackServer {
allProviders,
))
}
+
+func getPluginManager() plugins.Manager {
+ panic(wire.Build(
+ allProviders,
+ ))
+}
+
+func GetPluginManager(ctx context.Context) plugins.Manager {
+ manager := getPluginManager()
+ manager.SetSubsonicRouter(CreateSubsonicAPIRouter(ctx))
+ return manager
+}
diff --git a/conf/configuration.go b/conf/configuration.go
index 8561f343f..0ad81492a 100644
--- a/conf/configuration.go
+++ b/conf/configuration.go
@@ -14,7 +14,7 @@ import (
"github.com/kr/pretty"
"github.com/navidrome/navidrome/consts"
"github.com/navidrome/navidrome/log"
- "github.com/navidrome/navidrome/utils/chain"
+ "github.com/navidrome/navidrome/utils/run"
"github.com/robfig/cron/v3"
"github.com/spf13/viper"
)
@@ -66,6 +66,7 @@ type configOptions struct {
CoverArtPriority string
CoverJpegQuality int
ArtistArtPriority string
+ LyricsPriority string
EnableGravatar bool
EnableFavourites bool
EnableStarRating bool
@@ -79,6 +80,7 @@ type configOptions struct {
DefaultUIVolume int
EnableReplayGain bool
EnableCoverAnimation bool
+ EnableNowPlaying bool
GATrackingID string
EnableLogRedacting bool
AuthRequestLimit int
@@ -86,25 +88,26 @@ type configOptions struct {
PasswordEncryptionKey string
ReverseProxyUserHeader string
ReverseProxyWhitelist string
- HTTPSecurityHeaders secureOptions
- Prometheus prometheusOptions
- Scanner scannerOptions
- Jukebox jukeboxOptions
- Backup backupOptions
- PID pidOptions
- Inspect inspectOptions
- Subsonic subsonicOptions
- LyricsPriority string
-
- Agents string
- LastFM lastfmOptions
- Spotify spotifyOptions
- ListenBrainz listenBrainzOptions
- Tags map[string]TagConf
+ Plugins pluginsOptions
+ PluginConfig map[string]map[string]string
+ HTTPSecurityHeaders secureOptions `json:",omitzero"`
+ Prometheus prometheusOptions `json:",omitzero"`
+ Scanner scannerOptions `json:",omitzero"`
+ Jukebox jukeboxOptions `json:",omitzero"`
+ Backup backupOptions `json:",omitzero"`
+ PID pidOptions `json:",omitzero"`
+ Inspect inspectOptions `json:",omitzero"`
+ Subsonic subsonicOptions `json:",omitzero"`
+ LastFM lastfmOptions `json:",omitzero"`
+ Spotify spotifyOptions `json:",omitzero"`
+ Deezer deezerOptions `json:",omitzero"`
+ ListenBrainz listenBrainzOptions `json:",omitzero"`
+ Tags map[string]TagConf `json:",omitempty"`
+ Agents string
// DevFlags. These are used to enable/disable debugging and incomplete features
+ DevLogLevels map[string]string `json:",omitempty"`
DevLogSourceLine bool
- DevLogLevels map[string]string
DevEnableProfiler bool
DevAutoCreateAdminPassword string
DevAutoLoginUsername string
@@ -112,6 +115,8 @@ type configOptions struct {
DevActivityPanelUpdateRate time.Duration
DevSidebarPlaylists bool
DevShowArtistPage bool
+ DevUIShowConfig bool
+ DevNewEventStream bool
DevOffsetOptimize int
DevArtworkMaxRequests int
DevArtworkThrottleBacklogLimit int
@@ -120,8 +125,12 @@ type configOptions struct {
DevAlbumInfoTimeToLive time.Duration
DevExternalScanner bool
DevScannerThreads uint
+ DevSelectiveWatcher bool
DevInsightsInitialDelay time.Duration
DevEnablePlayerInsights bool
+ DevEnablePluginsInsights bool
+ DevPluginCompilationTimeout time.Duration
+ DevExternalArtistFetchMultiplier float64
}
type scannerOptions struct {
@@ -145,19 +154,20 @@ type subsonicOptions struct {
}
type TagConf struct {
- Ignore bool `yaml:"ignore"`
- Aliases []string `yaml:"aliases"`
- Type string `yaml:"type"`
- MaxLength int `yaml:"maxLength"`
- Split []string `yaml:"split"`
- Album bool `yaml:"album"`
+ Ignore bool `yaml:"ignore" json:",omitempty"`
+ Aliases []string `yaml:"aliases" json:",omitempty"`
+ Type string `yaml:"type" json:",omitempty"`
+ MaxLength int `yaml:"maxLength" json:",omitempty"`
+ Split []string `yaml:"split" json:",omitempty"`
+ Album bool `yaml:"album" json:",omitempty"`
}
type lastfmOptions struct {
- Enabled bool
- ApiKey string
- Secret string
- Language string
+ Enabled bool
+ ApiKey string
+ Secret string
+ Language string
+ ScrobbleFirstArtistOnly bool
}
type spotifyOptions struct {
@@ -165,6 +175,11 @@ type spotifyOptions struct {
Secret string
}
+type deezerOptions struct {
+ Enabled bool
+ Language string
+}
+
type listenBrainzOptions struct {
Enabled bool
BaseURL string
@@ -207,6 +222,12 @@ type inspectOptions struct {
BacklogTimeout int
}
+type pluginsOptions struct {
+ Enabled bool
+ Folder string
+ CacheSize string
+}
+
var (
Server = &configOptions{}
hooks []func()
@@ -246,6 +267,17 @@ func Load(noConfigDump bool) {
os.Exit(1)
}
+ if Server.Plugins.Enabled {
+ if Server.Plugins.Folder == "" {
+ Server.Plugins.Folder = filepath.Join(Server.DataFolder, "plugins")
+ }
+ err = os.MkdirAll(Server.Plugins.Folder, 0700)
+ if err != nil {
+ _, _ = fmt.Fprintln(os.Stderr, "FATAL: Error creating plugins path:", err)
+ os.Exit(1)
+ }
+ }
+
Server.ConfigFile = viper.GetViper().ConfigFileUsed()
if Server.DbPath == "" {
Server.DbPath = filepath.Join(Server.DataFolder, consts.DefaultDbPath)
@@ -274,7 +306,7 @@ func Load(noConfigDump bool) {
log.SetLogSourceLine(Server.DevLogSourceLine)
log.SetRedacting(Server.EnableLogRedacting)
- err = chain.RunSequentially(
+ err = run.Sequentially(
validateScanSchedule,
validateBackupSchedule,
validatePlaylistsPath,
@@ -366,6 +398,7 @@ func disableExternalServices() {
Server.EnableInsightsCollector = false
Server.LastFM.Enabled = false
Server.Spotify.ID = ""
+ Server.Deezer.Enabled = false
Server.ListenBrainz.Enabled = false
Server.Agents = ""
if Server.UILoginBackgroundURL == consts.DefaultUILoginBackgroundURL {
@@ -477,10 +510,11 @@ func setViperDefaults() {
viper.SetDefault("ignoredarticles", "The El La Los Las Le Les Os As O A")
viper.SetDefault("indexgroups", "A B C D E F G H I J K L M N O P Q R S T U V W X-Z(XYZ) [Unknown]([)")
viper.SetDefault("ffmpegpath", "")
- viper.SetDefault("mpvcmdtemplate", "mpv --audio-device=%d --no-audio-display --pause %f --input-ipc-server=%s")
+ viper.SetDefault("mpvcmdtemplate", "mpv --audio-device=%d --no-audio-display %f --input-ipc-server=%s")
viper.SetDefault("coverartpriority", "cover.*, folder.*, front.*, embedded, external")
viper.SetDefault("coverjpegquality", 75)
viper.SetDefault("artistartpriority", "artist.*, album/artist.*, external")
+ viper.SetDefault("lyricspriority", ".lrc,.txt,embedded")
viper.SetDefault("enablegravatar", false)
viper.SetDefault("enablefavourites", true)
viper.SetDefault("enablestarrating", true)
@@ -490,6 +524,7 @@ func setViperDefaults() {
viper.SetDefault("defaultuivolume", consts.DefaultUIVolume)
viper.SetDefault("enablereplaygain", true)
viper.SetDefault("enablecoveranimation", true)
+ viper.SetDefault("enablenowplaying", true)
viper.SetDefault("enablesharing", false)
viper.SetDefault("shareurl", "")
viper.SetDefault("defaultshareexpiration", 8760*time.Hour)
@@ -518,18 +553,21 @@ func setViperDefaults() {
viper.SetDefault("scanner.genreseparators", "")
viper.SetDefault("scanner.groupalbumreleases", false)
viper.SetDefault("scanner.followsymlinks", true)
- viper.SetDefault("scanner.purgemissing", "never")
+ viper.SetDefault("scanner.purgemissing", consts.PurgeMissingNever)
viper.SetDefault("subsonic.appendsubtitle", true)
viper.SetDefault("subsonic.artistparticipations", false)
viper.SetDefault("subsonic.defaultreportrealpath", false)
viper.SetDefault("subsonic.legacyclients", "DSub")
- viper.SetDefault("agents", "lastfm,spotify")
+ viper.SetDefault("agents", "lastfm,spotify,deezer")
viper.SetDefault("lastfm.enabled", true)
viper.SetDefault("lastfm.language", "en")
viper.SetDefault("lastfm.apikey", "")
viper.SetDefault("lastfm.secret", "")
+ viper.SetDefault("lastfm.scrobblefirstartistonly", false)
viper.SetDefault("spotify.id", "")
viper.SetDefault("spotify.secret", "")
+ viper.SetDefault("deezer.enabled", true)
+ viper.SetDefault("deezer.language", "en")
viper.SetDefault("listenbrainz.enabled", true)
viper.SetDefault("listenbrainz.baseurl", "https://api.listenbrainz.org/1/")
viper.SetDefault("httpsecurityheaders.customframeoptionsvalue", "DENY")
@@ -542,7 +580,11 @@ func setViperDefaults() {
viper.SetDefault("inspect.maxrequests", 1)
viper.SetDefault("inspect.backloglimit", consts.RequestThrottleBacklogLimit)
viper.SetDefault("inspect.backlogtimeout", consts.RequestThrottleBacklogTimeout)
- viper.SetDefault("lyricspriority", ".lrc,.txt,embedded")
+ viper.SetDefault("plugins.folder", "")
+ viper.SetDefault("plugins.enabled", false)
+ viper.SetDefault("plugins.cachesize", "100MB")
+
+ // DevFlags. These are used to enable/disable debugging and incomplete features
viper.SetDefault("devlogsourceline", false)
viper.SetDefault("devenableprofiler", false)
viper.SetDefault("devautocreateadminpassword", "")
@@ -551,6 +593,8 @@ func setViperDefaults() {
viper.SetDefault("devactivitypanelupdaterate", 300*time.Millisecond)
viper.SetDefault("devsidebarplaylists", true)
viper.SetDefault("devshowartistpage", true)
+ viper.SetDefault("devuishowconfig", true)
+ viper.SetDefault("devneweventstream", true)
viper.SetDefault("devoffsetoptimize", 50000)
viper.SetDefault("devartworkmaxrequests", max(2, runtime.NumCPU()/3))
viper.SetDefault("devartworkthrottlebackloglimit", consts.RequestThrottleBacklogLimit)
@@ -559,8 +603,12 @@ func setViperDefaults() {
viper.SetDefault("devalbuminfotimetolive", consts.AlbumInfoTimeToLive)
viper.SetDefault("devexternalscanner", true)
viper.SetDefault("devscannerthreads", 5)
+ viper.SetDefault("devselectivewatcher", true)
viper.SetDefault("devinsightsinitialdelay", consts.InsightsInitialDelay)
viper.SetDefault("devenableplayerinsights", true)
+ viper.SetDefault("devenablepluginsinsights", true)
+ viper.SetDefault("devplugincompilationtimeout", time.Minute)
+ viper.SetDefault("devexternalartistfetchmultiplier", 1.5)
}
func init() {
diff --git a/core/agents/agents.go b/core/agents/agents.go
index 50a1e04ad..cb10d2c4c 100644
--- a/core/agents/agents.go
+++ b/core/agents/agents.go
@@ -2,6 +2,7 @@ package agents
import (
"context"
+ "slices"
"strings"
"time"
@@ -13,43 +14,107 @@ import (
"github.com/navidrome/navidrome/utils/singleton"
)
-type Agents struct {
- ds model.DataStore
- agents []Interface
+// PluginLoader defines an interface for loading plugins
+type PluginLoader interface {
+ // PluginNames returns the names of all plugins that implement a particular service
+ PluginNames(capability string) []string
+ // LoadMediaAgent loads and returns a media agent plugin
+ LoadMediaAgent(name string) (Interface, bool)
}
-func GetAgents(ds model.DataStore) *Agents {
+type Agents struct {
+ ds model.DataStore
+ pluginLoader PluginLoader
+}
+
+// GetAgents returns the singleton instance of Agents
+func GetAgents(ds model.DataStore, pluginLoader PluginLoader) *Agents {
return singleton.GetInstance(func() *Agents {
- return createAgents(ds)
+ return createAgents(ds, pluginLoader)
})
}
-func createAgents(ds model.DataStore) *Agents {
- var order []string
- if conf.Server.Agents != "" {
- order = strings.Split(conf.Server.Agents, ",")
+// createAgents creates a new Agents instance. Used in tests
+func createAgents(ds model.DataStore, pluginLoader PluginLoader) *Agents {
+ return &Agents{
+ ds: ds,
+ pluginLoader: pluginLoader,
}
- order = append(order, LocalAgentName)
- var res []Interface
- var enabled []string
- for _, name := range order {
- init, ok := Map[name]
- if !ok {
- log.Error("Invalid agent. Check `Agents` configuration", "name", name, "conf", conf.Server.Agents)
- continue
- }
+}
- agent := init(ds)
- if agent == nil {
- log.Debug("Agent not available. Missing configuration?", "name", name)
- continue
- }
- enabled = append(enabled, name)
- res = append(res, init(ds))
+// enabledAgent represents an enabled agent with its type information
+type enabledAgent struct {
+ name string
+ isPlugin bool
+}
+
+// getEnabledAgentNames returns the current list of enabled agents, including:
+// 1. Built-in agents and plugins from config (in the specified order)
+// 2. Always include LocalAgentName
+// 3. If config is empty, include ONLY LocalAgentName
+// Each enabledAgent contains the name and whether it's a plugin (true) or built-in (false)
+func (a *Agents) getEnabledAgentNames() []enabledAgent {
+ // If no agents configured, ONLY use the local agent
+ if conf.Server.Agents == "" {
+ return []enabledAgent{{name: LocalAgentName, isPlugin: false}}
}
- log.Debug("List of agents enabled", "names", enabled)
- return &Agents{ds: ds, agents: res}
+ // Get all available plugin names
+ var availablePlugins []string
+ if a.pluginLoader != nil {
+ availablePlugins = a.pluginLoader.PluginNames("MetadataAgent")
+ }
+
+ configuredAgents := strings.Split(conf.Server.Agents, ",")
+
+ // Always add LocalAgentName if not already included
+ hasLocalAgent := slices.Contains(configuredAgents, LocalAgentName)
+ if !hasLocalAgent {
+ configuredAgents = append(configuredAgents, LocalAgentName)
+ }
+
+ // Filter to only include valid agents (built-in or plugins)
+ var validAgents []enabledAgent
+ for _, name := range configuredAgents {
+ // Check if it's a built-in agent
+ isBuiltIn := Map[name] != nil
+
+ // Check if it's a plugin
+ isPlugin := slices.Contains(availablePlugins, name)
+
+ if isBuiltIn {
+ validAgents = append(validAgents, enabledAgent{name: name, isPlugin: false})
+ } else if isPlugin {
+ validAgents = append(validAgents, enabledAgent{name: name, isPlugin: true})
+ } else {
+ log.Debug("Unknown agent ignored", "name", name)
+ }
+ }
+ return validAgents
+}
+
+func (a *Agents) getAgent(ea enabledAgent) Interface {
+ if ea.isPlugin {
+ // Try to load WASM plugin agent (if plugin loader is available)
+ if a.pluginLoader != nil {
+ agent, ok := a.pluginLoader.LoadMediaAgent(ea.name)
+ if ok && agent != nil {
+ return agent
+ }
+ }
+ } else {
+ // Try to get built-in agent
+ constructor, ok := Map[ea.name]
+ if ok {
+ agent := constructor(a.ds)
+ if agent != nil {
+ return agent
+ }
+ log.Debug("Built-in agent not available. Missing configuration?", "name", ea.name)
+ }
+ }
+
+ return nil
}
func (a *Agents) AgentName() string {
@@ -64,15 +129,19 @@ func (a *Agents) GetArtistMBID(ctx context.Context, id string, name string) (str
return "", nil
}
start := time.Now()
- for _, ag := range a.agents {
+ for _, enabledAgent := range a.getEnabledAgentNames() {
+ ag := a.getAgent(enabledAgent)
+ if ag == nil {
+ continue
+ }
if utils.IsCtxDone(ctx) {
break
}
- agent, ok := ag.(ArtistMBIDRetriever)
+ retriever, ok := ag.(ArtistMBIDRetriever)
if !ok {
continue
}
- mbid, err := agent.GetArtistMBID(ctx, id, name)
+ mbid, err := retriever.GetArtistMBID(ctx, id, name)
if mbid != "" && err == nil {
log.Debug(ctx, "Got MBID", "agent", ag.AgentName(), "artist", name, "mbid", mbid, "elapsed", time.Since(start))
return mbid, nil
@@ -89,15 +158,19 @@ func (a *Agents) GetArtistURL(ctx context.Context, id, name, mbid string) (strin
return "", nil
}
start := time.Now()
- for _, ag := range a.agents {
+ for _, enabledAgent := range a.getEnabledAgentNames() {
+ ag := a.getAgent(enabledAgent)
+ if ag == nil {
+ continue
+ }
if utils.IsCtxDone(ctx) {
break
}
- agent, ok := ag.(ArtistURLRetriever)
+ retriever, ok := ag.(ArtistURLRetriever)
if !ok {
continue
}
- url, err := agent.GetArtistURL(ctx, id, name, mbid)
+ url, err := retriever.GetArtistURL(ctx, id, name, mbid)
if url != "" && err == nil {
log.Debug(ctx, "Got External Url", "agent", ag.AgentName(), "artist", name, "url", url, "elapsed", time.Since(start))
return url, nil
@@ -114,15 +187,19 @@ func (a *Agents) GetArtistBiography(ctx context.Context, id, name, mbid string)
return "", nil
}
start := time.Now()
- for _, ag := range a.agents {
+ for _, enabledAgent := range a.getEnabledAgentNames() {
+ ag := a.getAgent(enabledAgent)
+ if ag == nil {
+ continue
+ }
if utils.IsCtxDone(ctx) {
break
}
- agent, ok := ag.(ArtistBiographyRetriever)
+ retriever, ok := ag.(ArtistBiographyRetriever)
if !ok {
continue
}
- bio, err := agent.GetArtistBiography(ctx, id, name, mbid)
+ bio, err := retriever.GetArtistBiography(ctx, id, name, mbid)
if err == nil {
log.Debug(ctx, "Got Biography", "agent", ag.AgentName(), "artist", name, "len", len(bio), "elapsed", time.Since(start))
return bio, nil
@@ -131,6 +208,8 @@ func (a *Agents) GetArtistBiography(ctx context.Context, id, name, mbid string)
return "", ErrNotFound
}
+// GetSimilarArtists returns similar artists by id, name, and/or mbid. Because some artists returned from an enabled
+// agent may not exist in the database, return at most limit * conf.Server.DevExternalArtistFetchMultiplier items.
func (a *Agents) GetSimilarArtists(ctx context.Context, id, name, mbid string, limit int) ([]Artist, error) {
switch id {
case consts.UnknownArtistID:
@@ -138,16 +217,23 @@ func (a *Agents) GetSimilarArtists(ctx context.Context, id, name, mbid string, l
case consts.VariousArtistsID:
return nil, nil
}
+
+ overLimit := int(float64(limit) * conf.Server.DevExternalArtistFetchMultiplier)
+
start := time.Now()
- for _, ag := range a.agents {
+ for _, enabledAgent := range a.getEnabledAgentNames() {
+ ag := a.getAgent(enabledAgent)
+ if ag == nil {
+ continue
+ }
if utils.IsCtxDone(ctx) {
break
}
- agent, ok := ag.(ArtistSimilarRetriever)
+ retriever, ok := ag.(ArtistSimilarRetriever)
if !ok {
continue
}
- similar, err := agent.GetSimilarArtists(ctx, id, name, mbid, limit)
+ similar, err := retriever.GetSimilarArtists(ctx, id, name, mbid, overLimit)
if len(similar) > 0 && err == nil {
if log.IsGreaterOrEqualTo(log.LevelTrace) {
log.Debug(ctx, "Got Similar Artists", "agent", ag.AgentName(), "artist", name, "similar", similar, "elapsed", time.Since(start))
@@ -168,15 +254,19 @@ func (a *Agents) GetArtistImages(ctx context.Context, id, name, mbid string) ([]
return nil, nil
}
start := time.Now()
- for _, ag := range a.agents {
+ for _, enabledAgent := range a.getEnabledAgentNames() {
+ ag := a.getAgent(enabledAgent)
+ if ag == nil {
+ continue
+ }
if utils.IsCtxDone(ctx) {
break
}
- agent, ok := ag.(ArtistImageRetriever)
+ retriever, ok := ag.(ArtistImageRetriever)
if !ok {
continue
}
- images, err := agent.GetArtistImages(ctx, id, name, mbid)
+ images, err := retriever.GetArtistImages(ctx, id, name, mbid)
if len(images) > 0 && err == nil {
log.Debug(ctx, "Got Images", "agent", ag.AgentName(), "artist", name, "images", images, "elapsed", time.Since(start))
return images, nil
@@ -185,6 +275,8 @@ func (a *Agents) GetArtistImages(ctx context.Context, id, name, mbid string) ([]
return nil, ErrNotFound
}
+// GetArtistTopSongs returns top songs by id, name, and/or mbid. Because some songs returned from an enabled
+// agent may not exist in the database, return at most limit * conf.Server.DevExternalArtistFetchMultiplier items.
func (a *Agents) GetArtistTopSongs(ctx context.Context, id, artistName, mbid string, count int) ([]Song, error) {
switch id {
case consts.UnknownArtistID:
@@ -192,16 +284,23 @@ func (a *Agents) GetArtistTopSongs(ctx context.Context, id, artistName, mbid str
case consts.VariousArtistsID:
return nil, nil
}
+
+ overLimit := int(float64(count) * conf.Server.DevExternalArtistFetchMultiplier)
+
start := time.Now()
- for _, ag := range a.agents {
+ for _, enabledAgent := range a.getEnabledAgentNames() {
+ ag := a.getAgent(enabledAgent)
+ if ag == nil {
+ continue
+ }
if utils.IsCtxDone(ctx) {
break
}
- agent, ok := ag.(ArtistTopSongsRetriever)
+ retriever, ok := ag.(ArtistTopSongsRetriever)
if !ok {
continue
}
- songs, err := agent.GetArtistTopSongs(ctx, id, artistName, mbid, count)
+ songs, err := retriever.GetArtistTopSongs(ctx, id, artistName, mbid, overLimit)
if len(songs) > 0 && err == nil {
log.Debug(ctx, "Got Top Songs", "agent", ag.AgentName(), "artist", artistName, "songs", songs, "elapsed", time.Since(start))
return songs, nil
@@ -215,15 +314,19 @@ func (a *Agents) GetAlbumInfo(ctx context.Context, name, artist, mbid string) (*
return nil, ErrNotFound
}
start := time.Now()
- for _, ag := range a.agents {
+ for _, enabledAgent := range a.getEnabledAgentNames() {
+ ag := a.getAgent(enabledAgent)
+ if ag == nil {
+ continue
+ }
if utils.IsCtxDone(ctx) {
break
}
- agent, ok := ag.(AlbumInfoRetriever)
+ retriever, ok := ag.(AlbumInfoRetriever)
if !ok {
continue
}
- album, err := agent.GetAlbumInfo(ctx, name, artist, mbid)
+ album, err := retriever.GetAlbumInfo(ctx, name, artist, mbid)
if err == nil {
log.Debug(ctx, "Got Album Info", "agent", ag.AgentName(), "album", name, "artist", artist,
"mbid", mbid, "elapsed", time.Since(start))
@@ -233,6 +336,33 @@ func (a *Agents) GetAlbumInfo(ctx context.Context, name, artist, mbid string) (*
return nil, ErrNotFound
}
+func (a *Agents) GetAlbumImages(ctx context.Context, name, artist, mbid string) ([]ExternalImage, error) {
+ if name == consts.UnknownAlbum {
+ return nil, ErrNotFound
+ }
+ start := time.Now()
+ for _, enabledAgent := range a.getEnabledAgentNames() {
+ ag := a.getAgent(enabledAgent)
+ if ag == nil {
+ continue
+ }
+ if utils.IsCtxDone(ctx) {
+ break
+ }
+ retriever, ok := ag.(AlbumImageRetriever)
+ if !ok {
+ continue
+ }
+ images, err := retriever.GetAlbumImages(ctx, name, artist, mbid)
+ if len(images) > 0 && err == nil {
+ log.Debug(ctx, "Got Album Images", "agent", ag.AgentName(), "album", name, "artist", artist,
+ "mbid", mbid, "elapsed", time.Since(start))
+ return images, nil
+ }
+ }
+ return nil, ErrNotFound
+}
+
var _ Interface = (*Agents)(nil)
var _ ArtistMBIDRetriever = (*Agents)(nil)
var _ ArtistURLRetriever = (*Agents)(nil)
@@ -241,3 +371,4 @@ var _ ArtistSimilarRetriever = (*Agents)(nil)
var _ ArtistImageRetriever = (*Agents)(nil)
var _ ArtistTopSongsRetriever = (*Agents)(nil)
var _ AlbumInfoRetriever = (*Agents)(nil)
+var _ AlbumImageRetriever = (*Agents)(nil)
diff --git a/core/agents/agents_plugin_test.go b/core/agents/agents_plugin_test.go
new file mode 100644
index 000000000..b2791c00e
--- /dev/null
+++ b/core/agents/agents_plugin_test.go
@@ -0,0 +1,281 @@
+package agents
+
+import (
+ "context"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/utils/slice"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+// MockPluginLoader implements PluginLoader for testing
+type MockPluginLoader struct {
+ pluginNames []string
+ loadedAgents map[string]*MockAgent
+ pluginCallCount map[string]int
+}
+
+func NewMockPluginLoader() *MockPluginLoader {
+ return &MockPluginLoader{
+ pluginNames: []string{},
+ loadedAgents: make(map[string]*MockAgent),
+ pluginCallCount: make(map[string]int),
+ }
+}
+
+func (m *MockPluginLoader) PluginNames(serviceName string) []string {
+ return m.pluginNames
+}
+
+func (m *MockPluginLoader) LoadMediaAgent(name string) (Interface, bool) {
+ m.pluginCallCount[name]++
+ agent, exists := m.loadedAgents[name]
+ return agent, exists
+}
+
+// MockAgent is a mock agent implementation for testing
+type MockAgent struct {
+ name string
+ mbid string
+}
+
+func (m *MockAgent) AgentName() string {
+ return m.name
+}
+
+func (m *MockAgent) GetArtistMBID(ctx context.Context, id string, name string) (string, error) {
+ return m.mbid, nil
+}
+
+var _ Interface = (*MockAgent)(nil)
+var _ ArtistMBIDRetriever = (*MockAgent)(nil)
+
+var _ PluginLoader = (*MockPluginLoader)(nil)
+
+var _ = Describe("Agents with Plugin Loading", func() {
+ var mockLoader *MockPluginLoader
+ var agents *Agents
+
+ BeforeEach(func() {
+ mockLoader = NewMockPluginLoader()
+
+ // Create the agents instance with our mock loader
+ agents = createAgents(nil, mockLoader)
+ })
+
+ Context("Dynamic agent discovery", func() {
+ It("should include ONLY local agent when no config is specified", func() {
+ // Ensure no specific agents are configured
+ conf.Server.Agents = ""
+
+ // Add some plugin agents that should be ignored
+ mockLoader.pluginNames = append(mockLoader.pluginNames, "plugin_agent", "another_plugin")
+
+ // Should only include the local agent
+ enabledAgents := agents.getEnabledAgentNames()
+ Expect(enabledAgents).To(HaveLen(1))
+ Expect(enabledAgents[0].name).To(Equal(LocalAgentName))
+ Expect(enabledAgents[0].isPlugin).To(BeFalse()) // LocalAgent is built-in, not plugin
+ })
+
+ It("should NOT include plugin agents when no config is specified", func() {
+ // Ensure no specific agents are configured
+ conf.Server.Agents = ""
+
+ // Add a plugin agent
+ mockLoader.pluginNames = append(mockLoader.pluginNames, "plugin_agent")
+
+ // Should only include the local agent
+ enabledAgents := agents.getEnabledAgentNames()
+ Expect(enabledAgents).To(HaveLen(1))
+ Expect(enabledAgents[0].name).To(Equal(LocalAgentName))
+ Expect(enabledAgents[0].isPlugin).To(BeFalse()) // LocalAgent is built-in, not plugin
+ })
+
+ It("should include plugin agents in the enabled agents list ONLY when explicitly configured", func() {
+ // Add a plugin agent
+ mockLoader.pluginNames = append(mockLoader.pluginNames, "plugin_agent")
+
+ // With no config, should not include plugin
+ conf.Server.Agents = ""
+ enabledAgents := agents.getEnabledAgentNames()
+ Expect(enabledAgents).To(HaveLen(1))
+ Expect(enabledAgents[0].name).To(Equal(LocalAgentName))
+
+ // When explicitly configured, should include plugin
+ conf.Server.Agents = "plugin_agent"
+ enabledAgents = agents.getEnabledAgentNames()
+ var agentNames []string
+ var pluginAgentFound bool
+ for _, agent := range enabledAgents {
+ agentNames = append(agentNames, agent.name)
+ if agent.name == "plugin_agent" {
+ pluginAgentFound = true
+ Expect(agent.isPlugin).To(BeTrue()) // plugin_agent is a plugin
+ }
+ }
+ Expect(agentNames).To(ContainElements(LocalAgentName, "plugin_agent"))
+ Expect(pluginAgentFound).To(BeTrue())
+ })
+
+ It("should only include configured plugin agents when config is specified", func() {
+ // Add two plugin agents
+ mockLoader.pluginNames = append(mockLoader.pluginNames, "plugin_one", "plugin_two")
+
+ // Configure only one of them
+ conf.Server.Agents = "plugin_one"
+
+ // Verify only the configured one is included
+ enabledAgents := agents.getEnabledAgentNames()
+ var agentNames []string
+ var pluginOneFound bool
+ for _, agent := range enabledAgents {
+ agentNames = append(agentNames, agent.name)
+ if agent.name == "plugin_one" {
+ pluginOneFound = true
+ Expect(agent.isPlugin).To(BeTrue()) // plugin_one is a plugin
+ }
+ }
+ Expect(agentNames).To(ContainElements(LocalAgentName, "plugin_one"))
+ Expect(agentNames).NotTo(ContainElement("plugin_two"))
+ Expect(pluginOneFound).To(BeTrue())
+ })
+
+ It("should load plugin agents on demand", func() {
+ ctx := context.Background()
+
+ // Configure to use our plugin
+ conf.Server.Agents = "plugin_agent"
+
+ // Add a plugin agent
+ mockLoader.pluginNames = append(mockLoader.pluginNames, "plugin_agent")
+ mockLoader.loadedAgents["plugin_agent"] = &MockAgent{
+ name: "plugin_agent",
+ mbid: "plugin-mbid",
+ }
+
+ // Try to get data from it
+ mbid, err := agents.GetArtistMBID(ctx, "123", "Artist")
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(mbid).To(Equal("plugin-mbid"))
+ Expect(mockLoader.pluginCallCount["plugin_agent"]).To(Equal(1))
+ })
+
+ It("should try both built-in and plugin agents", func() {
+ // Create a mock built-in agent
+ Register("built_in", func(ds model.DataStore) Interface {
+ return &MockAgent{
+ name: "built_in",
+ mbid: "built-in-mbid",
+ }
+ })
+ defer func() {
+ delete(Map, "built_in")
+ }()
+
+ // Configure to use both built-in and plugin
+ conf.Server.Agents = "built_in,plugin_agent"
+
+ // Add a plugin agent
+ mockLoader.pluginNames = append(mockLoader.pluginNames, "plugin_agent")
+ mockLoader.loadedAgents["plugin_agent"] = &MockAgent{
+ name: "plugin_agent",
+ mbid: "plugin-mbid",
+ }
+
+ // Verify that both are in the enabled list
+ enabledAgents := agents.getEnabledAgentNames()
+ var agentNames []string
+ var builtInFound, pluginFound bool
+ for _, agent := range enabledAgents {
+ agentNames = append(agentNames, agent.name)
+ if agent.name == "built_in" {
+ builtInFound = true
+ Expect(agent.isPlugin).To(BeFalse()) // built-in agent
+ }
+ if agent.name == "plugin_agent" {
+ pluginFound = true
+ Expect(agent.isPlugin).To(BeTrue()) // plugin agent
+ }
+ }
+ Expect(agentNames).To(ContainElements("built_in", "plugin_agent", LocalAgentName))
+ Expect(builtInFound).To(BeTrue())
+ Expect(pluginFound).To(BeTrue())
+ })
+
+ It("should respect the order specified in configuration", func() {
+ // Create mock built-in agents
+ Register("agent_a", func(ds model.DataStore) Interface {
+ return &MockAgent{name: "agent_a"}
+ })
+ Register("agent_b", func(ds model.DataStore) Interface {
+ return &MockAgent{name: "agent_b"}
+ })
+ defer func() {
+ delete(Map, "agent_a")
+ delete(Map, "agent_b")
+ }()
+
+ // Add plugin agents
+ mockLoader.pluginNames = append(mockLoader.pluginNames, "plugin_x", "plugin_y")
+
+ // Configure specific order - plugin first, then built-ins
+ conf.Server.Agents = "plugin_y,agent_b,plugin_x,agent_a"
+
+ // Get the agent names
+ enabledAgents := agents.getEnabledAgentNames()
+
+ // Extract just the names to verify the order
+ agentNames := slice.Map(enabledAgents, func(a enabledAgent) string { return a.name })
+
+ // Verify the order matches configuration, with LocalAgentName at the end
+ Expect(agentNames).To(HaveExactElements("plugin_y", "agent_b", "plugin_x", "agent_a", LocalAgentName))
+ })
+
+ It("should NOT call LoadMediaAgent for built-in agents", func() {
+ ctx := context.Background()
+
+ // Create a mock built-in agent
+ Register("builtin_agent", func(ds model.DataStore) Interface {
+ return &MockAgent{
+ name: "builtin_agent",
+ mbid: "builtin-mbid",
+ }
+ })
+ defer func() {
+ delete(Map, "builtin_agent")
+ }()
+
+ // Configure to use only built-in agents
+ conf.Server.Agents = "builtin_agent"
+
+ // Call GetArtistMBID which should only use the built-in agent
+ mbid, err := agents.GetArtistMBID(ctx, "123", "Artist")
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(mbid).To(Equal("builtin-mbid"))
+
+ // Verify LoadMediaAgent was NEVER called (no plugin loading for built-in agents)
+ Expect(mockLoader.pluginCallCount).To(BeEmpty())
+ })
+
+ It("should NOT call LoadMediaAgent for invalid agent names", func() {
+ ctx := context.Background()
+
+ // Configure with an invalid agent name (not built-in, not a plugin)
+ conf.Server.Agents = "invalid_agent"
+
+ // This should only result in using the local agent (as the invalid one is ignored)
+ _, err := agents.GetArtistMBID(ctx, "123", "Artist")
+
+ // Should get ErrNotFound since only local agent is available and it returns not found for this operation
+ Expect(err).To(MatchError(ErrNotFound))
+
+ // Verify LoadMediaAgent was NEVER called for the invalid agent
+ Expect(mockLoader.pluginCallCount).To(BeEmpty())
+ })
+ })
+})
diff --git a/core/agents/agents_test.go b/core/agents/agents_test.go
index ea12fb746..0b7eec282 100644
--- a/core/agents/agents_test.go
+++ b/core/agents/agents_test.go
@@ -4,10 +4,10 @@ import (
"context"
"errors"
+ "github.com/navidrome/navidrome/conf/configtest"
"github.com/navidrome/navidrome/consts"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/tests"
- "github.com/navidrome/navidrome/utils/slice"
"github.com/navidrome/navidrome/conf"
. "github.com/onsi/ginkgo/v2"
@@ -20,6 +20,7 @@ var _ = Describe("Agents", func() {
var ds model.DataStore
var mfRepo *tests.MockMediaFileRepo
BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
ctx, cancel = context.WithCancel(context.Background())
mfRepo = tests.CreateMockMediaFileRepo()
ds = &tests.MockDataStore{MockedMediaFile: mfRepo}
@@ -29,7 +30,7 @@ var _ = Describe("Agents", func() {
var ag *Agents
BeforeEach(func() {
conf.Server.Agents = ""
- ag = createAgents(ds)
+ ag = createAgents(ds, nil)
})
It("calls the placeholder GetArtistImages", func() {
@@ -49,12 +50,18 @@ var _ = Describe("Agents", func() {
Register("disabled", func(model.DataStore) Interface { return nil })
Register("empty", func(model.DataStore) Interface { return &emptyAgent{} })
conf.Server.Agents = "empty,fake,disabled"
- ag = createAgents(ds)
+ ag = createAgents(ds, nil)
Expect(ag.AgentName()).To(Equal("agents"))
})
It("does not register disabled agents", func() {
- ags := slice.Map(ag.agents, func(a Interface) string { return a.AgentName() })
+ var ags []string
+ for _, enabledAgent := range ag.getEnabledAgentNames() {
+ agent := ag.getAgent(enabledAgent)
+ if agent != nil {
+ ags = append(ags, agent.AgentName())
+ }
+ }
// local agent is always appended to the end of the agents list
Expect(ags).To(HaveExactElements("empty", "fake", "local"))
Expect(ags).ToNot(ContainElement("disabled"))
@@ -173,6 +180,42 @@ var _ = Describe("Agents", func() {
Expect(err).To(MatchError(ErrNotFound))
Expect(mock.Args).To(BeEmpty())
})
+
+ Context("with multiple image agents", func() {
+ var first *testImageAgent
+ var second *testImageAgent
+
+ BeforeEach(func() {
+ first = &testImageAgent{Name: "imgFail", Err: errors.New("fail")}
+ second = &testImageAgent{Name: "imgOk", Images: []ExternalImage{{URL: "ok", Size: 1}}}
+ Register("imgFail", func(model.DataStore) Interface { return first })
+ Register("imgOk", func(model.DataStore) Interface { return second })
+ })
+
+ It("falls back to the next agent on error", func() {
+ conf.Server.Agents = "imgFail,imgOk"
+ ag = createAgents(ds, nil)
+
+ images, err := ag.GetArtistImages(ctx, "id", "artist", "mbid")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(images).To(Equal([]ExternalImage{{URL: "ok", Size: 1}}))
+ Expect(first.Args).To(HaveExactElements("id", "artist", "mbid"))
+ Expect(second.Args).To(HaveExactElements("id", "artist", "mbid"))
+ })
+
+ It("falls back if the first agent returns no images", func() {
+ first.Err = nil
+ first.Images = []ExternalImage{}
+ conf.Server.Agents = "imgFail,imgOk"
+ ag = createAgents(ds, nil)
+
+ images, err := ag.GetArtistImages(ctx, "id", "artist", "mbid")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(images).To(Equal([]ExternalImage{{URL: "ok", Size: 1}}))
+ Expect(first.Args).To(HaveExactElements("id", "artist", "mbid"))
+ Expect(second.Args).To(HaveExactElements("id", "artist", "mbid"))
+ })
+ })
})
Describe("GetSimilarArtists", func() {
@@ -199,6 +242,7 @@ var _ = Describe("Agents", func() {
Describe("GetArtistTopSongs", func() {
It("returns on first match", func() {
+ conf.Server.DevExternalArtistFetchMultiplier = 1
Expect(ag.GetArtistTopSongs(ctx, "123", "test", "mb123", 2)).To(Equal([]Song{{
Name: "A Song",
MBID: "mbid444",
@@ -206,6 +250,7 @@ var _ = Describe("Agents", func() {
Expect(mock.Args).To(HaveExactElements("123", "test", "mb123", 2))
})
It("skips the agent if it returns an error", func() {
+ conf.Server.DevExternalArtistFetchMultiplier = 1
mock.Err = errors.New("error")
_, err := ag.GetArtistTopSongs(ctx, "123", "test", "mb123", 2)
Expect(err).To(MatchError(ErrNotFound))
@@ -217,6 +262,14 @@ var _ = Describe("Agents", func() {
Expect(err).To(MatchError(ErrNotFound))
Expect(mock.Args).To(BeEmpty())
})
+ It("fetches with multiplier", func() {
+ conf.Server.DevExternalArtistFetchMultiplier = 2
+ Expect(ag.GetArtistTopSongs(ctx, "123", "test", "mb123", 2)).To(Equal([]Song{{
+ Name: "A Song",
+ MBID: "mbid444",
+ }}))
+ Expect(mock.Args).To(HaveExactElements("123", "test", "mb123", 4))
+ })
})
Describe("GetAlbumInfo", func() {
@@ -226,18 +279,6 @@ var _ = Describe("Agents", func() {
MBID: "mbid444",
Description: "A Description",
URL: "External URL",
- Images: []ExternalImage{
- {
- Size: 174,
- URL: "https://lastfm.freetls.fastly.net/i/u/174s/00000000000000000000000000000000.png",
- }, {
- Size: 64,
- URL: "https://lastfm.freetls.fastly.net/i/u/64s/00000000000000000000000000000000.png",
- }, {
- Size: 34,
- URL: "https://lastfm.freetls.fastly.net/i/u/34s/00000000000000000000000000000000.png",
- },
- },
}))
Expect(mock.Args).To(HaveExactElements("album", "artist", "mbid"))
})
@@ -333,18 +374,6 @@ func (a *mockAgent) GetAlbumInfo(ctx context.Context, name, artist, mbid string)
MBID: "mbid444",
Description: "A Description",
URL: "External URL",
- Images: []ExternalImage{
- {
- Size: 174,
- URL: "https://lastfm.freetls.fastly.net/i/u/174s/00000000000000000000000000000000.png",
- }, {
- Size: 64,
- URL: "https://lastfm.freetls.fastly.net/i/u/64s/00000000000000000000000000000000.png",
- }, {
- Size: 34,
- URL: "https://lastfm.freetls.fastly.net/i/u/34s/00000000000000000000000000000000.png",
- },
- },
}, nil
}
@@ -355,3 +384,17 @@ type emptyAgent struct {
func (e *emptyAgent) AgentName() string {
return "empty"
}
+
+type testImageAgent struct {
+ Name string
+ Images []ExternalImage
+ Err error
+ Args []interface{}
+}
+
+func (t *testImageAgent) AgentName() string { return t.Name }
+
+func (t *testImageAgent) GetArtistImages(_ context.Context, id, name, mbid string) ([]ExternalImage, error) {
+ t.Args = []interface{}{id, name, mbid}
+ return t.Images, t.Err
+}
diff --git a/core/agents/deezer/client.go b/core/agents/deezer/client.go
new file mode 100644
index 000000000..32d93bad6
--- /dev/null
+++ b/core/agents/deezer/client.go
@@ -0,0 +1,218 @@
+package deezer
+
+import (
+ bytes "bytes"
+ "context"
+ "encoding/json"
+ "errors"
+ "fmt"
+ "io"
+ "net/http"
+ "net/url"
+ "strconv"
+ "strings"
+
+ "github.com/microcosm-cc/bluemonday"
+ "github.com/navidrome/navidrome/log"
+)
+
+const apiBaseURL = "https://api.deezer.com"
+const authBaseURL = "https://auth.deezer.com"
+
+var (
+ ErrNotFound = errors.New("deezer: not found")
+)
+
+type httpDoer interface {
+ Do(req *http.Request) (*http.Response, error)
+}
+
+type client struct {
+ httpDoer httpDoer
+ language string
+ jwt jwtToken
+}
+
+func newClient(hc httpDoer, language string) *client {
+ return &client{
+ httpDoer: hc,
+ language: language,
+ }
+}
+
+func (c *client) searchArtists(ctx context.Context, name string, limit int) ([]Artist, error) {
+ params := url.Values{}
+ params.Add("q", name)
+ params.Add("limit", strconv.Itoa(limit))
+ req, err := http.NewRequestWithContext(ctx, "GET", apiBaseURL+"/search/artist", nil)
+ if err != nil {
+ return nil, err
+ }
+ req.URL.RawQuery = params.Encode()
+
+ var results SearchArtistResults
+ err = c.makeRequest(req, &results)
+ if err != nil {
+ return nil, err
+ }
+
+ if len(results.Data) == 0 {
+ return nil, ErrNotFound
+ }
+ return results.Data, nil
+}
+
+func (c *client) makeRequest(req *http.Request, response any) error {
+ log.Trace(req.Context(), fmt.Sprintf("Sending Deezer %s request", req.Method), "url", req.URL)
+ resp, err := c.httpDoer.Do(req)
+ if err != nil {
+ return err
+ }
+
+ defer resp.Body.Close()
+ data, err := io.ReadAll(resp.Body)
+ if err != nil {
+ return err
+ }
+
+ if resp.StatusCode != 200 {
+ return c.parseError(data)
+ }
+
+ return json.Unmarshal(data, response)
+}
+
+func (c *client) parseError(data []byte) error {
+ var deezerError Error
+ err := json.Unmarshal(data, &deezerError)
+ if err != nil {
+ return err
+ }
+ return fmt.Errorf("deezer error(%d): %s", deezerError.Error.Code, deezerError.Error.Message)
+}
+
+func (c *client) getRelatedArtists(ctx context.Context, artistID int) ([]Artist, error) {
+ req, err := http.NewRequestWithContext(ctx, "GET", fmt.Sprintf("%s/artist/%d/related", apiBaseURL, artistID), nil)
+ if err != nil {
+ return nil, err
+ }
+
+ var results RelatedArtists
+ err = c.makeRequest(req, &results)
+ if err != nil {
+ return nil, err
+ }
+
+ return results.Data, nil
+}
+
+func (c *client) getTopTracks(ctx context.Context, artistID int, limit int) ([]Track, error) {
+ params := url.Values{}
+ params.Add("limit", strconv.Itoa(limit))
+ req, err := http.NewRequestWithContext(ctx, "GET", fmt.Sprintf("%s/artist/%d/top", apiBaseURL, artistID), nil)
+ if err != nil {
+ return nil, err
+ }
+ req.URL.RawQuery = params.Encode()
+
+ var results TopTracks
+ err = c.makeRequest(req, &results)
+ if err != nil {
+ return nil, err
+ }
+
+ return results.Data, nil
+}
+
+const pipeAPIURL = "https://pipe.deezer.com/api"
+
+var strictPolicy = bluemonday.StrictPolicy()
+
+func (c *client) getArtistBio(ctx context.Context, artistID int) (string, error) {
+ jwt, err := c.getJWT(ctx)
+ if err != nil {
+ return "", fmt.Errorf("deezer: failed to get JWT: %w", err)
+ }
+
+ query := map[string]any{
+ "operationName": "ArtistBio",
+ "variables": map[string]any{
+ "artistId": strconv.Itoa(artistID),
+ },
+ "query": `query ArtistBio($artistId: String!) {
+ artist(artistId: $artistId) {
+ bio {
+ full
+ }
+ }
+ }`,
+ }
+
+ body, err := json.Marshal(query)
+ if err != nil {
+ return "", err
+ }
+
+ req, err := http.NewRequestWithContext(ctx, "POST", pipeAPIURL, bytes.NewReader(body))
+ if err != nil {
+ return "", err
+ }
+
+ req.Header.Set("Content-Type", "application/json")
+ req.Header.Set("Accept-Language", c.language)
+ req.Header.Set("Authorization", "Bearer "+jwt)
+
+ log.Trace(ctx, "Fetching Deezer artist biography via GraphQL", "artistId", artistID, "language", c.language)
+ resp, err := c.httpDoer.Do(req)
+ if err != nil {
+ return "", err
+ }
+ defer resp.Body.Close()
+
+ if resp.StatusCode != 200 {
+ return "", fmt.Errorf("deezer: failed to fetch biography: %s", resp.Status)
+ }
+
+ data, err := io.ReadAll(resp.Body)
+ if err != nil {
+ return "", err
+ }
+
+ type graphQLResponse struct {
+ Data struct {
+ Artist struct {
+ Bio struct {
+ Full string `json:"full"`
+ } `json:"bio"`
+ } `json:"artist"`
+ } `json:"data"`
+ Errors []struct {
+ Message string `json:"message"`
+ }
+ }
+
+ var result graphQLResponse
+ if err := json.Unmarshal(data, &result); err != nil {
+ return "", fmt.Errorf("deezer: failed to parse GraphQL response: %w", err)
+ }
+
+ if len(result.Errors) > 0 {
+ var errs []error
+ for m := range result.Errors {
+ errs = append(errs, errors.New(result.Errors[m].Message))
+ }
+ err := errors.Join(errs...)
+ return "", fmt.Errorf("deezer: GraphQL error: %w", err)
+ }
+
+ if result.Data.Artist.Bio.Full == "" {
+ return "", errors.New("deezer: biography not found")
+ }
+
+ return cleanBio(result.Data.Artist.Bio.Full), nil
+}
+
+func cleanBio(bio string) string {
+ bio = strings.ReplaceAll(bio, "
", "\n")
+ return strictPolicy.Sanitize(bio)
+}
diff --git a/core/agents/deezer/client_auth.go b/core/agents/deezer/client_auth.go
new file mode 100644
index 000000000..c88c2bcb6
--- /dev/null
+++ b/core/agents/deezer/client_auth.go
@@ -0,0 +1,101 @@
+package deezer
+
+import (
+ "context"
+ "encoding/json"
+ "errors"
+ "fmt"
+ "io"
+ "net/http"
+ "sync"
+ "time"
+
+ "github.com/lestrrat-go/jwx/v2/jwt"
+ "github.com/navidrome/navidrome/log"
+)
+
+type jwtToken struct {
+ token string
+ expiresAt time.Time
+ mu sync.RWMutex
+}
+
+func (j *jwtToken) get() (string, bool) {
+ j.mu.RLock()
+ defer j.mu.RUnlock()
+ if time.Now().Before(j.expiresAt) {
+ return j.token, true
+ }
+ return "", false
+}
+
+func (j *jwtToken) set(token string, expiresIn time.Duration) {
+ j.mu.Lock()
+ defer j.mu.Unlock()
+ j.token = token
+ j.expiresAt = time.Now().Add(expiresIn)
+}
+
+func (c *client) getJWT(ctx context.Context) (string, error) {
+ // Check if we have a valid cached token
+ if token, valid := c.jwt.get(); valid {
+ return token, nil
+ }
+
+ // Fetch a new anonymous token
+ req, err := http.NewRequestWithContext(ctx, "GET", authBaseURL+"/login/anonymous?jo=p&rto=c", nil)
+ if err != nil {
+ return "", err
+ }
+ req.Header.Set("Accept", "application/json")
+
+ resp, err := c.httpDoer.Do(req)
+ if err != nil {
+ return "", err
+ }
+ defer resp.Body.Close()
+
+ if resp.StatusCode != 200 {
+ return "", fmt.Errorf("deezer: failed to get JWT token: %s", resp.Status)
+ }
+
+ data, err := io.ReadAll(resp.Body)
+ if err != nil {
+ return "", err
+ }
+
+ type authResponse struct {
+ JWT string `json:"jwt"`
+ }
+
+ var result authResponse
+ if err := json.Unmarshal(data, &result); err != nil {
+ return "", fmt.Errorf("deezer: failed to parse auth response: %w", err)
+ }
+
+ if result.JWT == "" {
+ return "", errors.New("deezer: no JWT token in response")
+ }
+
+ // Parse JWT to get actual expiration time
+ token, err := jwt.ParseString(result.JWT, jwt.WithVerify(false), jwt.WithValidate(false))
+ if err != nil {
+ return "", fmt.Errorf("deezer: failed to parse JWT token: %w", err)
+ }
+
+ // Calculate TTL with a 1-minute buffer for clock skew and network delays
+ expiresAt := token.Expiration()
+ if expiresAt.IsZero() {
+ return "", errors.New("deezer: JWT token has no expiration time")
+ }
+
+ ttl := time.Until(expiresAt) - 1*time.Minute
+ if ttl <= 0 {
+ return "", errors.New("deezer: JWT token already expired or expires too soon")
+ }
+
+ c.jwt.set(result.JWT, ttl)
+ log.Trace(ctx, "Fetched new Deezer JWT token", "expiresAt", expiresAt, "ttl", ttl)
+
+ return result.JWT, nil
+}
diff --git a/core/agents/deezer/client_auth_test.go b/core/agents/deezer/client_auth_test.go
new file mode 100644
index 000000000..b0c2d195d
--- /dev/null
+++ b/core/agents/deezer/client_auth_test.go
@@ -0,0 +1,293 @@
+package deezer
+
+import (
+ "bytes"
+ "context"
+ "fmt"
+ "io"
+ "net/http"
+ "sync"
+ "time"
+
+ "github.com/lestrrat-go/jwx/v2/jwt"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("JWT Authentication", func() {
+ var httpClient *fakeHttpClient
+ var client *client
+ var ctx context.Context
+
+ BeforeEach(func() {
+ httpClient = &fakeHttpClient{}
+ client = newClient(httpClient, "en")
+ ctx = context.Background()
+ })
+
+ Describe("getJWT", func() {
+ Context("with a valid JWT response", func() {
+ It("successfully fetches and caches a JWT token", func() {
+ testJWT := createTestJWT(5 * time.Minute)
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(fmt.Sprintf(`{"jwt":"%s"}`, testJWT))),
+ })
+
+ token, err := client.getJWT(ctx)
+ Expect(err).To(BeNil())
+ Expect(token).To(Equal(testJWT))
+ })
+
+ It("returns the cached token on subsequent calls", func() {
+ testJWT := createTestJWT(5 * time.Minute)
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(fmt.Sprintf(`{"jwt":"%s"}`, testJWT))),
+ })
+
+ // First call should fetch from API
+ token1, err := client.getJWT(ctx)
+ Expect(err).To(BeNil())
+ Expect(token1).To(Equal(testJWT))
+ Expect(httpClient.lastRequest.URL.Path).To(Equal("/login/anonymous"))
+
+ // Second call should return cached token without hitting API
+ httpClient.lastRequest = nil // Clear last request to verify no new request is made
+ token2, err := client.getJWT(ctx)
+ Expect(err).To(BeNil())
+ Expect(token2).To(Equal(testJWT))
+ Expect(httpClient.lastRequest).To(BeNil()) // No new request made
+ })
+
+ It("parses the JWT expiration time correctly", func() {
+ expectedExpiration := time.Now().Add(5 * time.Minute)
+ testToken, err := jwt.NewBuilder().
+ Expiration(expectedExpiration).
+ Build()
+ Expect(err).To(BeNil())
+ testJWT, err := jwt.Sign(testToken, jwt.WithInsecureNoSignature())
+ Expect(err).To(BeNil())
+
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(fmt.Sprintf(`{"jwt":"%s"}`, string(testJWT)))),
+ })
+
+ token, err := client.getJWT(ctx)
+ Expect(err).To(BeNil())
+ Expect(token).ToNot(BeEmpty())
+
+ // Verify the token is cached until close to expiration
+ // The cache should expire 1 minute before the JWT expires
+ expectedCacheExpiry := expectedExpiration.Add(-1 * time.Minute)
+ Expect(client.jwt.expiresAt).To(BeTemporally("~", expectedCacheExpiry, 2*time.Second))
+ })
+ })
+
+ Context("with JWT tokens that expire soon", func() {
+ It("rejects tokens that expire in less than 1 minute", func() {
+ // Create a token that expires in 30 seconds (less than 1-minute buffer)
+ testJWT := createTestJWT(30 * time.Second)
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(fmt.Sprintf(`{"jwt":"%s"}`, testJWT))),
+ })
+
+ _, err := client.getJWT(ctx)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("JWT token already expired or expires too soon"))
+ })
+
+ It("rejects already expired tokens", func() {
+ // Create a token that expired 1 minute ago
+ testJWT := createTestJWT(-1 * time.Minute)
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(fmt.Sprintf(`{"jwt":"%s"}`, testJWT))),
+ })
+
+ _, err := client.getJWT(ctx)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("JWT token already expired or expires too soon"))
+ })
+
+ It("accepts tokens that expire in more than 1 minute", func() {
+ // Create a token that expires in 2 minutes (just over the 1-minute buffer)
+ testJWT := createTestJWT(2 * time.Minute)
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(fmt.Sprintf(`{"jwt":"%s"}`, testJWT))),
+ })
+
+ token, err := client.getJWT(ctx)
+ Expect(err).To(BeNil())
+ Expect(token).ToNot(BeEmpty())
+ })
+ })
+
+ Context("with invalid responses", func() {
+ It("handles HTTP error responses", func() {
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 500,
+ Body: io.NopCloser(bytes.NewBufferString(`{"error":"Internal server error"}`)),
+ })
+
+ _, err := client.getJWT(ctx)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("failed to get JWT token"))
+ })
+
+ It("handles malformed JSON responses", func() {
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(`{invalid json}`)),
+ })
+
+ _, err := client.getJWT(ctx)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("failed to parse auth response"))
+ })
+
+ It("handles responses with empty JWT field", func() {
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(`{"jwt":""}`)),
+ })
+
+ _, err := client.getJWT(ctx)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(Equal("deezer: no JWT token in response"))
+ })
+
+ It("handles invalid JWT tokens", func() {
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(`{"jwt":"not-a-valid-jwt"}`)),
+ })
+
+ _, err := client.getJWT(ctx)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("failed to parse JWT token"))
+ })
+
+ It("rejects JWT tokens without expiration", func() {
+ // Create a JWT without expiration claim
+ testToken, err := jwt.NewBuilder().
+ Claim("custom", "value").
+ Build()
+ Expect(err).To(BeNil())
+
+ // Verify token has no expiration
+ Expect(testToken.Expiration().IsZero()).To(BeTrue())
+
+ testJWT, err := jwt.Sign(testToken, jwt.WithInsecureNoSignature())
+ Expect(err).To(BeNil())
+
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(fmt.Sprintf(`{"jwt":"%s"}`, string(testJWT)))),
+ })
+
+ _, err = client.getJWT(ctx)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(Equal("deezer: JWT token has no expiration time"))
+ })
+ })
+
+ Context("token caching behavior", func() {
+ It("fetches a new token when the cached token expires", func() {
+ // First token expires in 5 minutes
+ firstJWT := createTestJWT(5 * time.Minute)
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(fmt.Sprintf(`{"jwt":"%s"}`, firstJWT))),
+ })
+
+ token1, err := client.getJWT(ctx)
+ Expect(err).To(BeNil())
+ Expect(token1).To(Equal(firstJWT))
+
+ // Manually expire the cached token
+ client.jwt.expiresAt = time.Now().Add(-1 * time.Second)
+
+ // Second token with different expiration (10 minutes)
+ secondJWT := createTestJWT(10 * time.Minute)
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(fmt.Sprintf(`{"jwt":"%s"}`, secondJWT))),
+ })
+
+ token2, err := client.getJWT(ctx)
+ Expect(err).To(BeNil())
+ Expect(token2).To(Equal(secondJWT))
+ Expect(token2).ToNot(Equal(token1))
+ })
+ })
+ })
+
+ Describe("jwtToken cache", func() {
+ var cache *jwtToken
+
+ BeforeEach(func() {
+ cache = &jwtToken{}
+ })
+
+ It("returns false for expired tokens", func() {
+ cache.set("test-token", -1*time.Second) // Already expired
+ token, valid := cache.get()
+ Expect(valid).To(BeFalse())
+ Expect(token).To(BeEmpty())
+ })
+
+ It("returns true for valid tokens", func() {
+ cache.set("test-token", 4*time.Minute)
+ token, valid := cache.get()
+ Expect(valid).To(BeTrue())
+ Expect(token).To(Equal("test-token"))
+ })
+
+ It("is thread-safe for concurrent access", func() {
+ wg := sync.WaitGroup{}
+
+ // Writer goroutine
+ wg.Go(func() {
+ for i := 0; i < 100; i++ {
+ cache.set(fmt.Sprintf("token-%d", i), 1*time.Hour)
+ time.Sleep(1 * time.Millisecond)
+ }
+ })
+
+ // Reader goroutine
+ wg.Go(func() {
+ for i := 0; i < 100; i++ {
+ cache.get()
+ time.Sleep(1 * time.Millisecond)
+ }
+ })
+
+ // Wait for both goroutines to complete
+ wg.Wait()
+
+ // Verify final state is valid
+ token, valid := cache.get()
+ Expect(valid).To(BeTrue())
+ Expect(token).To(HavePrefix("token-"))
+ })
+ })
+})
+
+// createTestJWT creates a valid JWT token for testing purposes
+func createTestJWT(expiresIn time.Duration) string {
+ token, err := jwt.NewBuilder().
+ Expiration(time.Now().Add(expiresIn)).
+ Build()
+ if err != nil {
+ panic(fmt.Sprintf("failed to create test JWT: %v", err))
+ }
+ signed, err := jwt.Sign(token, jwt.WithInsecureNoSignature())
+ if err != nil {
+ panic(fmt.Sprintf("failed to sign test JWT: %v", err))
+ }
+ return string(signed)
+}
diff --git a/core/agents/deezer/client_test.go b/core/agents/deezer/client_test.go
new file mode 100644
index 000000000..7e4f7a49f
--- /dev/null
+++ b/core/agents/deezer/client_test.go
@@ -0,0 +1,195 @@
+package deezer
+
+import (
+ "bytes"
+ "fmt"
+ "io"
+ "net/http"
+ "os"
+ "time"
+
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("client", func() {
+ var httpClient *fakeHttpClient
+ var client *client
+
+ BeforeEach(func() {
+ httpClient = &fakeHttpClient{}
+ client = newClient(httpClient, "en")
+ })
+
+ Describe("ArtistImages", func() {
+ It("returns artist images from a successful request", func() {
+ f, err := os.Open("tests/fixtures/deezer.search.artist.json")
+ Expect(err).To(BeNil())
+ httpClient.mock("https://api.deezer.com/search/artist", http.Response{Body: f, StatusCode: 200})
+
+ artists, err := client.searchArtists(GinkgoT().Context(), "Michael Jackson", 20)
+ Expect(err).To(BeNil())
+ Expect(artists).To(HaveLen(17))
+ Expect(artists[0].Name).To(Equal("Michael Jackson"))
+ Expect(artists[0].PictureXl).To(Equal("https://cdn-images.dzcdn.net/images/artist/97fae13b2b30e4aec2e8c9e0c7839d92/1000x1000-000000-80-0-0.jpg"))
+ })
+
+ It("fails if artist was not found", func() {
+ httpClient.mock("https://api.deezer.com/search/artist", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(`{"data":[],"total":0}`)),
+ })
+
+ _, err := client.searchArtists(GinkgoT().Context(), "Michael Jackson", 20)
+ Expect(err).To(MatchError(ErrNotFound))
+ })
+ })
+
+ Describe("ArtistBio", func() {
+ BeforeEach(func() {
+ // Mock the JWT token endpoint with a valid JWT that expires in 5 minutes
+ testJWT := createTestJWT(5 * time.Minute)
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(fmt.Sprintf(`{"jwt":"%s","refresh_token":""}`, testJWT))),
+ })
+ })
+
+ It("returns artist bio from a successful request", func() {
+ f, err := os.Open("tests/fixtures/deezer.artist.bio.json")
+ Expect(err).To(BeNil())
+ httpClient.mock("https://pipe.deezer.com/api", http.Response{Body: f, StatusCode: 200})
+
+ bio, err := client.getArtistBio(GinkgoT().Context(), 27)
+ Expect(err).To(BeNil())
+ Expect(bio).To(ContainSubstring("Schoolmates Thomas and Guy-Manuel"))
+ Expect(bio).ToNot(ContainSubstring(""))
+ Expect(bio).ToNot(ContainSubstring("
"))
+ })
+
+ It("uses the configured language", func() {
+ client = newClient(httpClient, "fr")
+ // Mock JWT token for the new client instance with a valid JWT
+ testJWT := createTestJWT(5 * time.Minute)
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(fmt.Sprintf(`{"jwt":"%s","refresh_token":""}`, testJWT))),
+ })
+ f, err := os.Open("tests/fixtures/deezer.artist.bio.json")
+ Expect(err).To(BeNil())
+ httpClient.mock("https://pipe.deezer.com/api", http.Response{Body: f, StatusCode: 200})
+
+ _, err = client.getArtistBio(GinkgoT().Context(), 27)
+ Expect(err).To(BeNil())
+ Expect(httpClient.lastRequest.Header.Get("Accept-Language")).To(Equal("fr"))
+ })
+
+ It("includes the JWT token in the request", func() {
+ f, err := os.Open("tests/fixtures/deezer.artist.bio.json")
+ Expect(err).To(BeNil())
+ httpClient.mock("https://pipe.deezer.com/api", http.Response{Body: f, StatusCode: 200})
+
+ _, err = client.getArtistBio(GinkgoT().Context(), 27)
+ Expect(err).To(BeNil())
+ // Verify that the Authorization header has the Bearer token format
+ authHeader := httpClient.lastRequest.Header.Get("Authorization")
+ Expect(authHeader).To(HavePrefix("Bearer "))
+ Expect(len(authHeader)).To(BeNumerically(">", 20)) // JWT tokens are longer than 20 chars
+ })
+
+ It("handles GraphQL errors", func() {
+ errorResponse := `{
+ "data": {
+ "artist": {
+ "bio": {
+ "full": ""
+ }
+ }
+ },
+ "errors": [
+ {
+ "message": "Artist not found"
+ },
+ {
+ "message": "Invalid artist ID"
+ }
+ ]
+ }`
+ httpClient.mock("https://pipe.deezer.com/api", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(errorResponse)),
+ })
+
+ _, err := client.getArtistBio(GinkgoT().Context(), 999)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("GraphQL error"))
+ Expect(err.Error()).To(ContainSubstring("Artist not found"))
+ Expect(err.Error()).To(ContainSubstring("Invalid artist ID"))
+ })
+
+ It("handles empty biography", func() {
+ emptyBioResponse := `{
+ "data": {
+ "artist": {
+ "bio": {
+ "full": ""
+ }
+ }
+ }
+ }`
+ httpClient.mock("https://pipe.deezer.com/api", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(emptyBioResponse)),
+ })
+
+ _, err := client.getArtistBio(GinkgoT().Context(), 27)
+ Expect(err).To(MatchError("deezer: biography not found"))
+ })
+
+ It("handles JWT token fetch failure", func() {
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 500,
+ Body: io.NopCloser(bytes.NewBufferString(`{"error":"Internal server error"}`)),
+ })
+
+ _, err := client.getArtistBio(GinkgoT().Context(), 27)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("failed to get JWT"))
+ })
+
+ It("handles JWT token that expires too soon", func() {
+ // Create a JWT that expires in 30 seconds (less than the 1-minute buffer)
+ expiredJWT := createTestJWT(30 * time.Second)
+ httpClient.mock("https://auth.deezer.com/login/anonymous", http.Response{
+ StatusCode: 200,
+ Body: io.NopCloser(bytes.NewBufferString(fmt.Sprintf(`{"jwt":"%s","refresh_token":""}`, expiredJWT))),
+ })
+
+ _, err := client.getArtistBio(GinkgoT().Context(), 27)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("JWT token already expired or expires too soon"))
+ })
+ })
+})
+
+type fakeHttpClient struct {
+ responses map[string]*http.Response
+ lastRequest *http.Request
+}
+
+func (c *fakeHttpClient) mock(url string, response http.Response) {
+ if c.responses == nil {
+ c.responses = make(map[string]*http.Response)
+ }
+ c.responses[url] = &response
+}
+
+func (c *fakeHttpClient) Do(req *http.Request) (*http.Response, error) {
+ c.lastRequest = req
+ u := req.URL
+ u.RawQuery = ""
+ if resp, ok := c.responses[u.String()]; ok {
+ return resp, nil
+ }
+ panic("URL not mocked: " + u.String())
+}
diff --git a/core/agents/deezer/deezer.go b/core/agents/deezer/deezer.go
new file mode 100644
index 000000000..8f3e505ec
--- /dev/null
+++ b/core/agents/deezer/deezer.go
@@ -0,0 +1,148 @@
+package deezer
+
+import (
+ "context"
+ "errors"
+ "net/http"
+ "strings"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/consts"
+ "github.com/navidrome/navidrome/core/agents"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/utils/cache"
+ "github.com/navidrome/navidrome/utils/slice"
+)
+
+const deezerAgentName = "deezer"
+const deezerApiPictureXlSize = 1000
+const deezerApiPictureBigSize = 500
+const deezerApiPictureMediumSize = 250
+const deezerApiPictureSmallSize = 56
+const deezerArtistSearchLimit = 50
+
+type deezerAgent struct {
+ dataStore model.DataStore
+ client *client
+}
+
+func deezerConstructor(dataStore model.DataStore) agents.Interface {
+ agent := &deezerAgent{dataStore: dataStore}
+ httpClient := &http.Client{
+ Timeout: consts.DefaultHttpClientTimeOut,
+ }
+ cachedHttpClient := cache.NewHTTPClient(httpClient, consts.DefaultHttpClientTimeOut)
+ agent.client = newClient(cachedHttpClient, conf.Server.Deezer.Language)
+ return agent
+}
+
+func (s *deezerAgent) AgentName() string {
+ return deezerAgentName
+}
+
+func (s *deezerAgent) GetArtistImages(ctx context.Context, _, name, _ string) ([]agents.ExternalImage, error) {
+ artist, err := s.searchArtist(ctx, name)
+ if err != nil {
+ if errors.Is(err, agents.ErrNotFound) {
+ log.Warn(ctx, "Artist not found in deezer", "artist", name)
+ } else {
+ log.Error(ctx, "Error calling deezer", "artist", name, err)
+ }
+ return nil, err
+ }
+
+ var res []agents.ExternalImage
+ possibleImages := []struct {
+ URL string
+ Size int
+ }{
+ {artist.PictureXl, deezerApiPictureXlSize},
+ {artist.PictureBig, deezerApiPictureBigSize},
+ {artist.PictureMedium, deezerApiPictureMediumSize},
+ {artist.PictureSmall, deezerApiPictureSmallSize},
+ }
+ for _, imgData := range possibleImages {
+ if imgData.URL != "" {
+ res = append(res, agents.ExternalImage{
+ URL: imgData.URL,
+ Size: imgData.Size,
+ })
+ }
+ }
+ return res, nil
+}
+
+func (s *deezerAgent) searchArtist(ctx context.Context, name string) (*Artist, error) {
+ artists, err := s.client.searchArtists(ctx, name, deezerArtistSearchLimit)
+ if errors.Is(err, ErrNotFound) || len(artists) == 0 {
+ return nil, agents.ErrNotFound
+ }
+ if err != nil {
+ return nil, err
+ }
+
+ // If the first one has the same name, that's the one
+ if !strings.EqualFold(artists[0].Name, name) {
+ return nil, agents.ErrNotFound
+ }
+ return &artists[0], err
+}
+
+func (s *deezerAgent) GetSimilarArtists(ctx context.Context, _, name, _ string, limit int) ([]agents.Artist, error) {
+ artist, err := s.searchArtist(ctx, name)
+ if err != nil {
+ return nil, err
+ }
+
+ related, err := s.client.getRelatedArtists(ctx, artist.ID)
+ if err != nil {
+ return nil, err
+ }
+
+ res := slice.Map(related, func(r Artist) agents.Artist {
+ return agents.Artist{
+ Name: r.Name,
+ }
+ })
+ if len(res) > limit {
+ res = res[:limit]
+ }
+ return res, nil
+}
+
+func (s *deezerAgent) GetArtistTopSongs(ctx context.Context, _, artistName, _ string, count int) ([]agents.Song, error) {
+ artist, err := s.searchArtist(ctx, artistName)
+ if err != nil {
+ return nil, err
+ }
+
+ tracks, err := s.client.getTopTracks(ctx, artist.ID, count)
+ if err != nil {
+ return nil, err
+ }
+
+ res := slice.Map(tracks, func(r Track) agents.Song {
+ return agents.Song{
+ Name: r.Title,
+ }
+ })
+ return res, nil
+}
+
+func (s *deezerAgent) GetArtistBiography(ctx context.Context, _, name, _ string) (string, error) {
+ artist, err := s.searchArtist(ctx, name)
+ if err != nil {
+ return "", err
+ }
+
+ return s.client.getArtistBio(ctx, artist.ID)
+}
+
+func init() {
+ conf.AddHook(func() {
+ if conf.Server.Deezer.Enabled {
+ agents.Register(deezerAgentName, deezerConstructor)
+ }
+ })
+}
diff --git a/core/agents/deezer/deezer_suite_test.go b/core/agents/deezer/deezer_suite_test.go
new file mode 100644
index 000000000..a42282da7
--- /dev/null
+++ b/core/agents/deezer/deezer_suite_test.go
@@ -0,0 +1,17 @@
+package deezer
+
+import (
+ "testing"
+
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+func TestDeezer(t *testing.T) {
+ tests.Init(t, false)
+ log.SetLevel(log.LevelFatal)
+ RegisterFailHandler(Fail)
+ RunSpecs(t, "Deezer Test Suite")
+}
diff --git a/core/agents/deezer/responses.go b/core/agents/deezer/responses.go
new file mode 100644
index 000000000..266c44c62
--- /dev/null
+++ b/core/agents/deezer/responses.go
@@ -0,0 +1,66 @@
+package deezer
+
+type SearchArtistResults struct {
+ Data []Artist `json:"data"`
+ Total int `json:"total"`
+ Next string `json:"next"`
+}
+
+type Artist struct {
+ ID int `json:"id"`
+ Name string `json:"name"`
+ Link string `json:"link"`
+ Picture string `json:"picture"`
+ PictureSmall string `json:"picture_small"`
+ PictureMedium string `json:"picture_medium"`
+ PictureBig string `json:"picture_big"`
+ PictureXl string `json:"picture_xl"`
+ NbAlbum int `json:"nb_album"`
+ NbFan int `json:"nb_fan"`
+ Radio bool `json:"radio"`
+ Tracklist string `json:"tracklist"`
+ Type string `json:"type"`
+}
+
+type Error struct {
+ Error struct {
+ Type string `json:"type"`
+ Message string `json:"message"`
+ Code int `json:"code"`
+ } `json:"error"`
+}
+
+type RelatedArtists struct {
+ Data []Artist `json:"data"`
+ Total int `json:"total"`
+}
+
+type TopTracks struct {
+ Data []Track `json:"data"`
+ Total int `json:"total"`
+ Next string `json:"next"`
+}
+
+type Track struct {
+ ID int `json:"id"`
+ Title string `json:"title"`
+ Link string `json:"link"`
+ Duration int `json:"duration"`
+ Rank int `json:"rank"`
+ Preview string `json:"preview"`
+ Artist Artist `json:"artist"`
+ Album Album `json:"album"`
+ Contributors []Artist `json:"contributors"`
+}
+
+type Album struct {
+ ID int `json:"id"`
+ Title string `json:"title"`
+ Cover string `json:"cover"`
+ CoverSmall string `json:"cover_small"`
+ CoverMedium string `json:"cover_medium"`
+ CoverBig string `json:"cover_big"`
+ CoverXl string `json:"cover_xl"`
+ Tracklist string `json:"tracklist"`
+ Type string `json:"type"`
+}
diff --git a/core/agents/deezer/responses_test.go b/core/agents/deezer/responses_test.go
new file mode 100644
index 000000000..a9de5c5fb
--- /dev/null
+++ b/core/agents/deezer/responses_test.go
@@ -0,0 +1,69 @@
+package deezer
+
+import (
+ "encoding/json"
+ "os"
+
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("Responses", func() {
+ Describe("Search type=artist", func() {
+ It("parses the artist search result correctly ", func() {
+ var resp SearchArtistResults
+ body, err := os.ReadFile("tests/fixtures/deezer.search.artist.json")
+ Expect(err).To(BeNil())
+ err = json.Unmarshal(body, &resp)
+ Expect(err).To(BeNil())
+
+ Expect(resp.Data).To(HaveLen(17))
+ michael := resp.Data[0]
+ Expect(michael.Name).To(Equal("Michael Jackson"))
+ Expect(michael.PictureXl).To(Equal("https://cdn-images.dzcdn.net/images/artist/97fae13b2b30e4aec2e8c9e0c7839d92/1000x1000-000000-80-0-0.jpg"))
+ })
+ })
+
+ Describe("Error", func() {
+ It("parses the error response correctly", func() {
+ var errorResp Error
+ body := []byte(`{"error":{"type":"MissingParameterException","message":"Missing parameters: q","code":501}}`)
+ err := json.Unmarshal(body, &errorResp)
+ Expect(err).To(BeNil())
+
+ Expect(errorResp.Error.Code).To(Equal(501))
+ Expect(errorResp.Error.Message).To(Equal("Missing parameters: q"))
+ })
+ })
+
+ Describe("Related Artists", func() {
+ It("parses the related artists response correctly", func() {
+ var resp RelatedArtists
+ body, err := os.ReadFile("tests/fixtures/deezer.artist.related.json")
+ Expect(err).To(BeNil())
+ err = json.Unmarshal(body, &resp)
+ Expect(err).To(BeNil())
+
+ Expect(resp.Data).To(HaveLen(20))
+ justice := resp.Data[0]
+ Expect(justice.Name).To(Equal("Justice"))
+ Expect(justice.ID).To(Equal(6404))
+ })
+ })
+
+ Describe("Top Tracks", func() {
+ It("parses the top tracks response correctly", func() {
+ var resp TopTracks
+ body, err := os.ReadFile("tests/fixtures/deezer.artist.top.json")
+ Expect(err).To(BeNil())
+ err = json.Unmarshal(body, &resp)
+ Expect(err).To(BeNil())
+
+ Expect(resp.Data).To(HaveLen(5))
+ track := resp.Data[0]
+ Expect(track.Title).To(Equal("Instant Crush (feat. Julian Casablancas)"))
+ Expect(track.ID).To(Equal(67238732))
+ Expect(track.Album.Title).To(Equal("Random Access Memories"))
+ })
+ })
+})
diff --git a/core/agents/interfaces.go b/core/agents/interfaces.go
index 00f75627d..e60c61909 100644
--- a/core/agents/interfaces.go
+++ b/core/agents/interfaces.go
@@ -13,12 +13,12 @@ type Interface interface {
AgentName() string
}
+// AlbumInfo contains album metadata (no images)
type AlbumInfo struct {
Name string
MBID string
Description string
URL string
- Images []ExternalImage
}
type Artist struct {
@@ -40,11 +40,16 @@ var (
ErrNotFound = errors.New("not found")
)
-// TODO Break up this interface in more specific methods, like artists
+// AlbumInfoRetriever provides album info (no images)
type AlbumInfoRetriever interface {
GetAlbumInfo(ctx context.Context, name, artist, mbid string) (*AlbumInfo, error)
}
+// AlbumImageRetriever provides album images
+type AlbumImageRetriever interface {
+ GetAlbumImages(ctx context.Context, name, artist, mbid string) ([]ExternalImage, error)
+}
+
type ArtistMBIDRetriever interface {
GetArtistMBID(ctx context.Context, id string, name string) (string, error)
}
diff --git a/core/agents/lastfm/agent.go b/core/agents/lastfm/agent.go
index 3f5f44d20..fafa6afec 100644
--- a/core/agents/lastfm/agent.go
+++ b/core/agents/lastfm/agent.go
@@ -38,6 +38,7 @@ type lastfmAgent struct {
secret string
lang string
client *client
+ httpClient httpDoer
getInfoMutex sync.Mutex
}
@@ -56,6 +57,7 @@ func lastFMConstructor(ds model.DataStore) *lastfmAgent {
Timeout: consts.DefaultHttpClientTimeOut,
}
chc := cache.NewHTTPClient(hc, consts.DefaultHttpClientTimeOut)
+ l.httpClient = chc
l.client = newClient(l.apiKey, l.secret, l.lang, chc)
return l
}
@@ -72,16 +74,23 @@ func (l *lastfmAgent) GetAlbumInfo(ctx context.Context, name, artist, mbid strin
return nil, err
}
- response := agents.AlbumInfo{
+ return &agents.AlbumInfo{
Name: a.Name,
MBID: a.MBID,
Description: a.Description.Summary,
URL: a.URL,
- Images: make([]agents.ExternalImage, 0),
+ }, nil
+}
+
+func (l *lastfmAgent) GetAlbumImages(ctx context.Context, name, artist, mbid string) ([]agents.ExternalImage, error) {
+ a, err := l.callAlbumGetInfo(ctx, name, artist, mbid)
+ if err != nil {
+ return nil, err
}
// Last.fm can return duplicate sizes.
seenSizes := map[int]bool{}
+ images := make([]agents.ExternalImage, 0)
// This assumes that Last.fm returns images with size small, medium, and large.
// This is true as of December 29, 2022
@@ -92,23 +101,20 @@ func (l *lastfmAgent) GetAlbumInfo(ctx context.Context, name, artist, mbid strin
log.Trace(ctx, "LastFM/albuminfo image URL does not match expected regex or is empty", "url", img.URL, "size", img.Size)
continue
}
-
numericSize, err := strconv.Atoi(size[0][2:])
if err != nil {
log.Error(ctx, "LastFM/albuminfo image URL does not match expected regex", "url", img.URL, "size", img.Size, err)
return nil, err
- } else {
- if _, exists := seenSizes[numericSize]; !exists {
- response.Images = append(response.Images, agents.ExternalImage{
- Size: numericSize,
- URL: img.URL,
- })
- seenSizes[numericSize] = true
- }
+ }
+ if _, exists := seenSizes[numericSize]; !exists {
+ images = append(images, agents.ExternalImage{
+ Size: numericSize,
+ URL: img.URL,
+ })
+ seenSizes[numericSize] = true
}
}
-
- return &response, nil
+ return images, nil
}
func (l *lastfmAgent) GetArtistMBID(ctx context.Context, id string, name string) (string, error) {
@@ -186,13 +192,13 @@ func (l *lastfmAgent) GetArtistTopSongs(ctx context.Context, id, artistName, mbi
return res, nil
}
-var artistOpenGraphQuery = cascadia.MustCompile(`html > head > meta[property="og:image"]`)
+var (
+ artistOpenGraphQuery = cascadia.MustCompile(`html > head > meta[property="og:image"]`)
+ artistIgnoredImage = "2a96cbd8b46e442fc41c2b86b821562f" // Last.fm artist placeholder image name
+)
func (l *lastfmAgent) GetArtistImages(ctx context.Context, _, name, mbid string) ([]agents.ExternalImage, error) {
log.Debug(ctx, "Getting artist images from Last.fm", "name", name)
- hc := http.Client{
- Timeout: consts.DefaultHttpClientTimeOut,
- }
a, err := l.callArtistGetInfo(ctx, name)
if err != nil {
return nil, fmt.Errorf("get artist info: %w", err)
@@ -201,7 +207,7 @@ func (l *lastfmAgent) GetArtistImages(ctx context.Context, _, name, mbid string)
if err != nil {
return nil, fmt.Errorf("create artist image request: %w", err)
}
- resp, err := hc.Do(req)
+ resp, err := l.httpClient.Do(req)
if err != nil {
return nil, fmt.Errorf("get artist url: %w", err)
}
@@ -218,11 +224,16 @@ func (l *lastfmAgent) GetArtistImages(ctx context.Context, _, name, mbid string)
return res, nil
}
for _, attr := range n.Attr {
- if attr.Key == "content" {
- res = []agents.ExternalImage{
- {URL: attr.Val},
- }
- break
+ if attr.Key != "content" {
+ continue
+ }
+ if strings.Contains(attr.Val, artistIgnoredImage) {
+ log.Debug(ctx, "Artist image is ignored default image", "name", name, "url", attr.Val)
+ return res, nil
+ }
+
+ res = []agents.ExternalImage{
+ {URL: attr.Val},
}
}
return res, nil
@@ -279,14 +290,21 @@ func (l *lastfmAgent) callArtistGetTopTracks(ctx context.Context, artistName str
return t.Track, nil
}
-func (l *lastfmAgent) NowPlaying(ctx context.Context, userId string, track *model.MediaFile) error {
+func (l *lastfmAgent) getArtistForScrobble(track *model.MediaFile) string {
+ if conf.Server.LastFM.ScrobbleFirstArtistOnly && len(track.Participants[model.RoleArtist]) > 0 {
+ return track.Participants[model.RoleArtist][0].Name
+ }
+ return track.Artist
+}
+
+func (l *lastfmAgent) NowPlaying(ctx context.Context, userId string, track *model.MediaFile, position int) error {
sk, err := l.sessionKeys.Get(ctx, userId)
if err != nil || sk == "" {
return scrobbler.ErrNotAuthorized
}
err = l.client.updateNowPlaying(ctx, sk, ScrobbleInfo{
- artist: track.Artist,
+ artist: l.getArtistForScrobble(track),
track: track.Title,
album: track.Album,
trackNumber: track.TrackNumber,
@@ -312,7 +330,7 @@ func (l *lastfmAgent) Scrobble(ctx context.Context, userId string, s scrobbler.S
return nil
}
err = l.client.scrobble(ctx, sk, ScrobbleInfo{
- artist: s.Artist,
+ artist: l.getArtistForScrobble(&s.MediaFile),
track: s.Title,
album: s.Album,
trackNumber: s.TrackNumber,
diff --git a/core/agents/lastfm/agent_test.go b/core/agents/lastfm/agent_test.go
index de4fac6d6..18e7facf2 100644
--- a/core/agents/lastfm/agent_test.go
+++ b/core/agents/lastfm/agent_test.go
@@ -196,6 +196,12 @@ var _ = Describe("lastfmAgent", func() {
TrackNumber: 1,
Duration: 180,
MbzRecordingID: "mbz-123",
+ Participants: map[model.Role]model.ParticipantList{
+ model.RoleArtist: []model.Participant{
+ {Artist: model.Artist{ID: "ar-1", Name: "First Artist"}},
+ {Artist: model.Artist{ID: "ar-2", Name: "Second Artist"}},
+ },
+ },
}
})
@@ -203,7 +209,7 @@ var _ = Describe("lastfmAgent", func() {
It("calls Last.fm with correct params", func() {
httpClient.Res = http.Response{Body: io.NopCloser(bytes.NewBufferString("{}")), StatusCode: 200}
- err := agent.NowPlaying(ctx, "user-1", track)
+ err := agent.NowPlaying(ctx, "user-1", track, 0)
Expect(err).ToNot(HaveOccurred())
Expect(httpClient.SavedRequest.Method).To(Equal(http.MethodPost))
@@ -220,7 +226,7 @@ var _ = Describe("lastfmAgent", func() {
})
It("returns ErrNotAuthorized if user is not linked", func() {
- err := agent.NowPlaying(ctx, "user-2", track)
+ err := agent.NowPlaying(ctx, "user-2", track, 0)
Expect(err).To(MatchError(scrobbler.ErrNotAuthorized))
})
})
@@ -247,6 +253,23 @@ var _ = Describe("lastfmAgent", func() {
Expect(sentParams.Get("timestamp")).To(Equal(strconv.FormatInt(ts.Unix(), 10)))
})
+ When("ScrobbleFirstArtistOnly is true", func() {
+ BeforeEach(func() {
+ conf.Server.LastFM.ScrobbleFirstArtistOnly = true
+ })
+
+ It("uses only the first artist", func() {
+ ts := time.Now()
+ httpClient.Res = http.Response{Body: io.NopCloser(bytes.NewBufferString("{}")), StatusCode: 200}
+
+ err := agent.Scrobble(ctx, "user-1", scrobbler.Scrobble{MediaFile: *track, TimeStamp: ts})
+
+ Expect(err).ToNot(HaveOccurred())
+ sentParams := httpClient.SavedRequest.URL.Query()
+ Expect(sentParams.Get("artist")).To(Equal("First Artist"))
+ })
+ })
+
It("skips songs with less than 31 seconds", func() {
track.Duration = 29
httpClient.Res = http.Response{Body: io.NopCloser(bytes.NewBufferString("{}")), StatusCode: 200}
@@ -322,24 +345,6 @@ var _ = Describe("lastfmAgent", func() {
MBID: "03c91c40-49a6-44a7-90e7-a700edf97a62",
Description: "Believe is the twenty-third studio album by American singer-actress Cher, released on November 10, 1998 by Warner Bros. Records. The RIAA certified it Quadruple Platinum on December 23, 1999, recognizing four million shipments in the United States; Worldwide, the album has sold more than 20 million copies, making it the biggest-selling album of her career. In 1999 the album received three Grammy Awards nominations including \"Record of the Year\", \"Best Pop Album\" and winning \"Best Dance Recording\" for the single \"Believe\". It was released by Warner Bros. Records at the end of 1998. The album was executive produced by Rob Read more on Last.fm .",
URL: "https://www.last.fm/music/Cher/Believe",
- Images: []agents.ExternalImage{
- {
- URL: "https://lastfm.freetls.fastly.net/i/u/34s/3b54885952161aaea4ce2965b2db1638.png",
- Size: 34,
- },
- {
- URL: "https://lastfm.freetls.fastly.net/i/u/64s/3b54885952161aaea4ce2965b2db1638.png",
- Size: 64,
- },
- {
- URL: "https://lastfm.freetls.fastly.net/i/u/174s/3b54885952161aaea4ce2965b2db1638.png",
- Size: 174,
- },
- {
- URL: "https://lastfm.freetls.fastly.net/i/u/300x300/3b54885952161aaea4ce2965b2db1638.png",
- Size: 300,
- },
- },
}))
Expect(httpClient.RequestCount).To(Equal(1))
Expect(httpClient.SavedRequest.URL.Query().Get("mbid")).To(Equal("03c91c40-49a6-44a7-90e7-a700edf97a62"))
@@ -349,9 +354,8 @@ var _ = Describe("lastfmAgent", func() {
f, _ := os.Open("tests/fixtures/lastfm.album.getinfo.empty_urls.json")
httpClient.Res = http.Response{Body: f, StatusCode: 200}
Expect(agent.GetAlbumInfo(ctx, "The Definitive Less Damage And More Joy", "The Jesus and Mary Chain", "")).To(Equal(&agents.AlbumInfo{
- Name: "The Definitive Less Damage And More Joy",
- URL: "https://www.last.fm/music/The+Jesus+and+Mary+Chain/The+Definitive+Less+Damage+And+More+Joy",
- Images: []agents.ExternalImage{},
+ Name: "The Definitive Less Damage And More Joy",
+ URL: "https://www.last.fm/music/The+Jesus+and+Mary+Chain/The+Definitive+Less+Damage+And+More+Joy",
}))
Expect(httpClient.RequestCount).To(Equal(1))
Expect(httpClient.SavedRequest.URL.Query().Get("album")).To(Equal("The Definitive Less Damage And More Joy"))
@@ -389,4 +393,73 @@ var _ = Describe("lastfmAgent", func() {
})
})
})
+
+ Describe("GetArtistImages", func() {
+ var agent *lastfmAgent
+ var apiClient *tests.FakeHttpClient
+ var httpClient *tests.FakeHttpClient
+
+ BeforeEach(func() {
+ apiClient = &tests.FakeHttpClient{}
+ httpClient = &tests.FakeHttpClient{}
+ client := newClient("API_KEY", "SECRET", "pt", apiClient)
+ agent = lastFMConstructor(ds)
+ agent.client = client
+ agent.httpClient = httpClient
+ })
+
+ It("returns the artist image from the page", func() {
+ fApi, _ := os.Open("tests/fixtures/lastfm.artist.getinfo.json")
+ apiClient.Res = http.Response{Body: fApi, StatusCode: 200}
+
+ fScraper, _ := os.Open("tests/fixtures/lastfm.artist.page.html")
+ httpClient.Res = http.Response{Body: fScraper, StatusCode: 200}
+
+ images, err := agent.GetArtistImages(ctx, "123", "U2", "")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(images).To(HaveLen(1))
+ Expect(images[0].URL).To(Equal("https://lastfm.freetls.fastly.net/i/u/ar0/818148bf682d429dc21b59a73ef6f68e.png"))
+ })
+
+ It("returns empty list if image is the ignored default image", func() {
+ fApi, _ := os.Open("tests/fixtures/lastfm.artist.getinfo.json")
+ apiClient.Res = http.Response{Body: fApi, StatusCode: 200}
+
+ fScraper, _ := os.Open("tests/fixtures/lastfm.artist.page.ignored.html")
+ httpClient.Res = http.Response{Body: fScraper, StatusCode: 200}
+
+ images, err := agent.GetArtistImages(ctx, "123", "U2", "")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(images).To(BeEmpty())
+ })
+
+ It("returns empty list if page has no meta tags", func() {
+ fApi, _ := os.Open("tests/fixtures/lastfm.artist.getinfo.json")
+ apiClient.Res = http.Response{Body: fApi, StatusCode: 200}
+
+ fScraper, _ := os.Open("tests/fixtures/lastfm.artist.page.no_meta.html")
+ httpClient.Res = http.Response{Body: fScraper, StatusCode: 200}
+
+ images, err := agent.GetArtistImages(ctx, "123", "U2", "")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(images).To(BeEmpty())
+ })
+
+ It("returns error if API call fails", func() {
+ apiClient.Err = errors.New("api error")
+ _, err := agent.GetArtistImages(ctx, "123", "U2", "")
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("get artist info"))
+ })
+
+ It("returns error if scraper call fails", func() {
+ fApi, _ := os.Open("tests/fixtures/lastfm.artist.getinfo.json")
+ apiClient.Res = http.Response{Body: fApi, StatusCode: 200}
+
+ httpClient.Err = errors.New("scraper error")
+ _, err := agent.GetArtistImages(ctx, "123", "U2", "")
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("get artist url"))
+ })
+ })
})
diff --git a/core/agents/listenbrainz/agent.go b/core/agents/listenbrainz/agent.go
index 200e9f63c..769b0f5a6 100644
--- a/core/agents/listenbrainz/agent.go
+++ b/core/agents/listenbrainz/agent.go
@@ -73,7 +73,7 @@ func (l *listenBrainzAgent) formatListen(track *model.MediaFile) listenInfo {
return li
}
-func (l *listenBrainzAgent) NowPlaying(ctx context.Context, userId string, track *model.MediaFile) error {
+func (l *listenBrainzAgent) NowPlaying(ctx context.Context, userId string, track *model.MediaFile, position int) error {
sk, err := l.sessionKeys.Get(ctx, userId)
if err != nil || sk == "" {
return errors.Join(err, scrobbler.ErrNotAuthorized)
diff --git a/core/agents/listenbrainz/agent_test.go b/core/agents/listenbrainz/agent_test.go
index 86a95d5bf..e99b442de 100644
--- a/core/agents/listenbrainz/agent_test.go
+++ b/core/agents/listenbrainz/agent_test.go
@@ -79,12 +79,12 @@ var _ = Describe("listenBrainzAgent", func() {
It("updates NowPlaying successfully", func() {
httpClient.Res = http.Response{Body: io.NopCloser(bytes.NewBufferString(`{"status": "ok"}`)), StatusCode: 200}
- err := agent.NowPlaying(ctx, "user-1", track)
+ err := agent.NowPlaying(ctx, "user-1", track, 0)
Expect(err).ToNot(HaveOccurred())
})
It("returns ErrNotAuthorized if user is not linked", func() {
- err := agent.NowPlaying(ctx, "user-2", track)
+ err := agent.NowPlaying(ctx, "user-2", track, 0)
Expect(err).To(MatchError(scrobbler.ErrNotAuthorized))
})
})
diff --git a/core/archiver.go b/core/archiver.go
index a15d0d713..63459816e 100644
--- a/core/archiver.go
+++ b/core/archiver.go
@@ -98,7 +98,7 @@ func (a *archiver) ZipShare(ctx context.Context, id string, out io.Writer) error
return model.ErrNotAuthorized
}
log.Debug(ctx, "Zipping share", "name", s.ID, "format", s.Format, "bitrate", s.MaxBitRate, "numTracks", len(s.Tracks))
- return a.zipMediaFiles(ctx, id, s.Format, s.MaxBitRate, out, s.Tracks)
+ return a.zipMediaFiles(ctx, id, s.ID, s.Format, s.MaxBitRate, out, s.Tracks, false)
}
func (a *archiver) ZipPlaylist(ctx context.Context, id string, format string, bitrate int, out io.Writer) error {
@@ -109,15 +109,40 @@ func (a *archiver) ZipPlaylist(ctx context.Context, id string, format string, bi
}
mfs := pls.MediaFiles()
log.Debug(ctx, "Zipping playlist", "name", pls.Name, "format", format, "bitrate", bitrate, "numTracks", len(mfs))
- return a.zipMediaFiles(ctx, id, format, bitrate, out, mfs)
+ return a.zipMediaFiles(ctx, id, pls.Name, format, bitrate, out, mfs, true)
}
-func (a *archiver) zipMediaFiles(ctx context.Context, id string, format string, bitrate int, out io.Writer, mfs model.MediaFiles) error {
+func (a *archiver) zipMediaFiles(ctx context.Context, id, name string, format string, bitrate int, out io.Writer, mfs model.MediaFiles, addM3U bool) error {
z := createZipWriter(out, format, bitrate)
+
+ zippedMfs := make(model.MediaFiles, len(mfs))
for idx, mf := range mfs {
file := a.playlistFilename(mf, format, idx)
_ = a.addFileToZip(ctx, z, mf, format, bitrate, file)
+ mf.Path = file
+ zippedMfs[idx] = mf
}
+
+ // Add M3U file if requested
+ if addM3U && len(zippedMfs) > 0 {
+ plsName := sanitizeName(name)
+ w, err := z.CreateHeader(&zip.FileHeader{
+ Name: plsName + ".m3u",
+ Modified: mfs[0].UpdatedAt,
+ Method: zip.Store,
+ })
+ if err != nil {
+ log.Error(ctx, "Error creating playlist zip entry", err)
+ return err
+ }
+
+ _, err = w.Write([]byte(zippedMfs.ToM3U8(plsName, false)))
+ if err != nil {
+ log.Error(ctx, "Error writing m3u in zip", err)
+ return err
+ }
+ }
+
err := z.Close()
if err != nil {
log.Error(ctx, "Error closing zip file", "id", id, err)
diff --git a/core/archiver_test.go b/core/archiver_test.go
index f1db5520f..37c4ef9ab 100644
--- a/core/archiver_test.go
+++ b/core/archiver_test.go
@@ -145,9 +145,21 @@ var _ = Describe("Archiver", func() {
zr, err := zip.NewReader(bytes.NewReader(out.Bytes()), int64(out.Len()))
Expect(err).To(BeNil())
- Expect(len(zr.File)).To(Equal(2))
+ Expect(len(zr.File)).To(Equal(3))
Expect(zr.File[0].Name).To(Equal("01 - AC_DC - track1.mp3"))
Expect(zr.File[1].Name).To(Equal("02 - Artist 2 - track2.mp3"))
+ Expect(zr.File[2].Name).To(Equal("Test Playlist.m3u"))
+
+ // Verify M3U content
+ m3uFile, err := zr.File[2].Open()
+ Expect(err).To(BeNil())
+ defer m3uFile.Close()
+
+ m3uContent, err := io.ReadAll(m3uFile)
+ Expect(err).To(BeNil())
+
+ expectedM3U := "#EXTM3U\n#PLAYLIST:Test Playlist\n#EXTINF:0,AC/DC - track1\n01 - AC_DC - track1.mp3\n#EXTINF:0,Artist 2 - track2\n02 - Artist 2 - track2.mp3\n"
+ Expect(string(m3uContent)).To(Equal(expectedM3U))
})
})
})
diff --git a/core/artwork/cache_warmer.go b/core/artwork/cache_warmer.go
index 2e60ca00b..909d299d8 100644
--- a/core/artwork/cache_warmer.go
+++ b/core/artwork/cache_warmer.go
@@ -96,8 +96,11 @@ func (a *cacheWarmer) run(ctx context.Context) {
// If cache not available, keep waiting
if !a.cache.Available(ctx) {
- if len(a.buffer) > 0 {
- log.Trace(ctx, "Cache not available, buffering precache request", "bufferLen", len(a.buffer))
+ a.mutex.Lock()
+ bufferLen := len(a.buffer)
+ a.mutex.Unlock()
+ if bufferLen > 0 {
+ log.Trace(ctx, "Cache not available, buffering precache request", "bufferLen", bufferLen)
}
continue
}
diff --git a/core/artwork/cache_warmer_test.go b/core/artwork/cache_warmer_test.go
index d35fb6e82..7ae3a16e0 100644
--- a/core/artwork/cache_warmer_test.go
+++ b/core/artwork/cache_warmer_test.go
@@ -80,6 +80,7 @@ var _ = Describe("CacheWarmer", func() {
})
It("adds multiple items to buffer", func() {
+ fc.SetReady(false) // Make cache unavailable so items stay in buffer
cw := NewCacheWarmer(aw, fc).(*cacheWarmer)
cw.PreCache(model.MustParseArtworkID("al-1"))
cw.PreCache(model.MustParseArtworkID("al-2"))
@@ -89,6 +90,7 @@ var _ = Describe("CacheWarmer", func() {
})
It("deduplicates items in buffer", func() {
+ fc.SetReady(false) // Make cache unavailable so items stay in buffer
cw := NewCacheWarmer(aw, fc).(*cacheWarmer)
cw.PreCache(model.MustParseArtworkID("al-1"))
cw.PreCache(model.MustParseArtworkID("al-1"))
@@ -214,3 +216,7 @@ func (f *mockFileCache) SetDisabled(v bool) {
f.disabled.Store(v)
f.ready.Store(true)
}
+
+func (f *mockFileCache) SetReady(v bool) {
+ f.ready.Store(v)
+}
diff --git a/core/artwork/reader_album.go b/core/artwork/reader_album.go
index 55d8b4352..cb4db97fe 100644
--- a/core/artwork/reader_album.go
+++ b/core/artwork/reader_album.go
@@ -1,6 +1,7 @@
package artwork
import (
+ "cmp"
"context"
"crypto/md5"
"fmt"
@@ -11,6 +12,7 @@ import (
"time"
"github.com/Masterminds/squirrel"
+ "github.com/maruel/natural"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/core"
"github.com/navidrome/navidrome/core/external"
@@ -116,8 +118,30 @@ func loadAlbumFoldersPaths(ctx context.Context, ds model.DataStore, albums ...mo
}
// Sort image files to ensure consistent selection of cover art
- // This prioritizes files from lower-numbered disc folders by sorting the paths
- slices.Sort(imgFiles)
+ // This prioritizes files without numeric suffixes (e.g., cover.jpg over cover.1.jpg)
+ // by comparing base filenames without extensions
+ slices.SortFunc(imgFiles, compareImageFiles)
return paths, imgFiles, &updatedAt, nil
}
+
+// compareImageFiles compares two image file paths for sorting.
+// It extracts the base filename (without extension) and compares case-insensitively.
+// This ensures that "cover.jpg" sorts before "cover.1.jpg" since "cover" < "cover.1".
+// Note: This function is called O(n log n) times during sorting, but in practice albums
+// typically have only 1-20 image files, making the repeated string operations negligible.
+func compareImageFiles(a, b string) int {
+ // Case-insensitive comparison
+ a = strings.ToLower(a)
+ b = strings.ToLower(b)
+
+ // Extract base filenames without extensions
+ baseA := strings.TrimSuffix(filepath.Base(a), filepath.Ext(a))
+ baseB := strings.TrimSuffix(filepath.Base(b), filepath.Ext(b))
+
+ // Compare base names first, then full paths if equal
+ return cmp.Or(
+ natural.Compare(baseA, baseB),
+ natural.Compare(a, b),
+ )
+}
diff --git a/core/artwork/reader_album_test.go b/core/artwork/reader_album_test.go
index 2665632b9..fd5f8a2be 100644
--- a/core/artwork/reader_album_test.go
+++ b/core/artwork/reader_album_test.go
@@ -27,26 +27,7 @@ var _ = Describe("Album Artwork Reader", func() {
expectedAt = now.Add(5 * time.Minute)
// Set up the test folders with image files
- repo = &fakeFolderRepo{
- result: []model.Folder{
- {
- Path: "Artist/Album/Disc1",
- ImagesUpdatedAt: expectedAt,
- ImageFiles: []string{"cover.jpg", "back.jpg"},
- },
- {
- Path: "Artist/Album/Disc2",
- ImagesUpdatedAt: now,
- ImageFiles: []string{"cover.jpg"},
- },
- {
- Path: "Artist/Album/Disc10",
- ImagesUpdatedAt: now,
- ImageFiles: []string{"cover.jpg"},
- },
- },
- err: nil,
- }
+ repo = &fakeFolderRepo{}
ds = &fakeDataStore{
folderRepo: repo,
}
@@ -58,19 +39,82 @@ var _ = Describe("Album Artwork Reader", func() {
})
It("returns sorted image files", func() {
+ repo.result = []model.Folder{
+ {
+ Path: "Artist/Album/Disc1",
+ ImagesUpdatedAt: expectedAt,
+ ImageFiles: []string{"cover.jpg", "back.jpg", "cover.1.jpg"},
+ },
+ {
+ Path: "Artist/Album/Disc2",
+ ImagesUpdatedAt: now,
+ ImageFiles: []string{"cover.jpg"},
+ },
+ {
+ Path: "Artist/Album/Disc10",
+ ImagesUpdatedAt: now,
+ ImageFiles: []string{"cover.jpg"},
+ },
+ }
+
_, imgFiles, imagesUpdatedAt, err := loadAlbumFoldersPaths(ctx, ds, album)
Expect(err).ToNot(HaveOccurred())
Expect(*imagesUpdatedAt).To(Equal(expectedAt))
- // Check that image files are sorted alphabetically
- Expect(imgFiles).To(HaveLen(4))
+ // Check that image files are sorted by base name (without extension)
+ Expect(imgFiles).To(HaveLen(5))
- // The files should be sorted by full path
+ // Files should be sorted by base filename without extension, then by full path
+ // "back" < "cover", so back.jpg comes first
+ // Then all cover.jpg files, sorted by path
Expect(imgFiles[0]).To(Equal(filepath.FromSlash("Artist/Album/Disc1/back.jpg")))
Expect(imgFiles[1]).To(Equal(filepath.FromSlash("Artist/Album/Disc1/cover.jpg")))
- Expect(imgFiles[2]).To(Equal(filepath.FromSlash("Artist/Album/Disc10/cover.jpg")))
- Expect(imgFiles[3]).To(Equal(filepath.FromSlash("Artist/Album/Disc2/cover.jpg")))
+ Expect(imgFiles[2]).To(Equal(filepath.FromSlash("Artist/Album/Disc2/cover.jpg")))
+ Expect(imgFiles[3]).To(Equal(filepath.FromSlash("Artist/Album/Disc10/cover.jpg")))
+ Expect(imgFiles[4]).To(Equal(filepath.FromSlash("Artist/Album/Disc1/cover.1.jpg")))
+ })
+
+ It("prioritizes files without numeric suffixes", func() {
+ // Test case for issue #4683: cover.jpg should come before cover.1.jpg
+ repo.result = []model.Folder{
+ {
+ Path: "Artist/Album",
+ ImagesUpdatedAt: now,
+ ImageFiles: []string{"cover.1.jpg", "cover.jpg", "cover.2.jpg"},
+ },
+ }
+
+ _, imgFiles, _, err := loadAlbumFoldersPaths(ctx, ds, album)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(imgFiles).To(HaveLen(3))
+
+ // cover.jpg should come first because "cover" < "cover.1" < "cover.2"
+ Expect(imgFiles[0]).To(Equal(filepath.FromSlash("Artist/Album/cover.jpg")))
+ Expect(imgFiles[1]).To(Equal(filepath.FromSlash("Artist/Album/cover.1.jpg")))
+ Expect(imgFiles[2]).To(Equal(filepath.FromSlash("Artist/Album/cover.2.jpg")))
+ })
+
+ It("handles case-insensitive sorting", func() {
+ // Test that Cover.jpg and cover.jpg are treated as equivalent
+ repo.result = []model.Folder{
+ {
+ Path: "Artist/Album",
+ ImagesUpdatedAt: now,
+ ImageFiles: []string{"Folder.jpg", "cover.jpg", "BACK.jpg"},
+ },
+ }
+
+ _, imgFiles, _, err := loadAlbumFoldersPaths(ctx, ds, album)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(imgFiles).To(HaveLen(3))
+
+ // Files should be sorted case-insensitively: BACK, cover, Folder
+ Expect(imgFiles[0]).To(Equal(filepath.FromSlash("Artist/Album/BACK.jpg")))
+ Expect(imgFiles[1]).To(Equal(filepath.FromSlash("Artist/Album/cover.jpg")))
+ Expect(imgFiles[2]).To(Equal(filepath.FromSlash("Artist/Album/Folder.jpg")))
})
})
})
diff --git a/core/artwork/reader_artist.go b/core/artwork/reader_artist.go
index 487346b4d..da8141a2d 100644
--- a/core/artwork/reader_artist.go
+++ b/core/artwork/reader_artist.go
@@ -8,6 +8,7 @@ import (
"io/fs"
"os"
"path/filepath"
+ "slices"
"strings"
"time"
@@ -20,6 +21,12 @@ import (
"github.com/navidrome/navidrome/utils/str"
)
+const (
+ // maxArtistFolderTraversalDepth defines how many directory levels to search
+ // when looking for artist images (artist folder + parent directories)
+ maxArtistFolderTraversalDepth = 3
+)
+
type artistReader struct {
cacheKey
a *artwork
@@ -108,36 +115,63 @@ func (a *artistReader) fromArtistArtPriority(ctx context.Context, priority strin
func fromArtistFolder(ctx context.Context, artistFolder string, pattern string) sourceFunc {
return func() (io.ReadCloser, string, error) {
- fsys := os.DirFS(artistFolder)
- matches, err := fs.Glob(fsys, pattern)
- if err != nil {
- log.Warn(ctx, "Error matching artist image pattern", "pattern", pattern, "folder", artistFolder)
- return nil, "", err
- }
- if len(matches) == 0 {
- return nil, "", fmt.Errorf(`no matches for '%s' in '%s'`, pattern, artistFolder)
- }
- for _, m := range matches {
- filePath := filepath.Join(artistFolder, m)
- if !model.IsImageFile(m) {
- continue
+ current := artistFolder
+ for i := 0; i < maxArtistFolderTraversalDepth; i++ {
+ if reader, path, err := findImageInFolder(ctx, current, pattern); err == nil {
+ return reader, path, nil
}
- f, err := os.Open(filePath)
- if err != nil {
- log.Warn(ctx, "Could not open cover art file", "file", filePath, err)
- return nil, "", err
+
+ parent := filepath.Dir(current)
+ if parent == current {
+ break
}
- return f, filePath, nil
+ current = parent
}
- return nil, "", nil
+ return nil, "", fmt.Errorf(`no matches for '%s' in '%s' or its parent directories`, pattern, artistFolder)
}
}
+func findImageInFolder(ctx context.Context, folder, pattern string) (io.ReadCloser, string, error) {
+ log.Trace(ctx, "looking for artist image", "pattern", pattern, "folder", folder)
+ fsys := os.DirFS(folder)
+ matches, err := fs.Glob(fsys, pattern)
+ if err != nil {
+ log.Warn(ctx, "Error matching artist image pattern", "pattern", pattern, "folder", folder, err)
+ return nil, "", err
+ }
+
+ // Filter to valid image files
+ var imagePaths []string
+ for _, m := range matches {
+ if !model.IsImageFile(m) {
+ continue
+ }
+ imagePaths = append(imagePaths, m)
+ }
+
+ // Sort image files by prioritizing base filenames without numeric
+ // suffixes (e.g., artist.jpg before artist.1.jpg)
+ slices.SortFunc(imagePaths, compareImageFiles)
+
+ // Try to open files in sorted order
+ for _, p := range imagePaths {
+ filePath := filepath.Join(folder, p)
+ f, err := os.Open(filePath)
+ if err != nil {
+ log.Warn(ctx, "Could not open cover art file", "file", filePath, err)
+ continue
+ }
+ return f, filePath, nil
+ }
+
+ return nil, "", fmt.Errorf(`no matches for '%s' in '%s'`, pattern, folder)
+}
+
func loadArtistFolder(ctx context.Context, ds model.DataStore, albums model.Albums, paths []string) (string, time.Time, error) {
if len(albums) == 0 {
return "", time.Time{}, nil
}
- libID := albums[0].LibraryID // Just need one of the albums, as they should all be in the same Library
+ libID := albums[0].LibraryID // Just need one of the albums, as they should all be in the same Library - for now! TODO: Support multiple libraries
folderPath := str.LongestCommonPrefix(paths)
if !strings.HasSuffix(folderPath, string(filepath.Separator)) {
diff --git a/core/artwork/reader_artist_test.go b/core/artwork/reader_artist_test.go
index 294a5db0b..e6a0168f8 100644
--- a/core/artwork/reader_artist_test.go
+++ b/core/artwork/reader_artist_test.go
@@ -3,6 +3,8 @@ package artwork
import (
"context"
"errors"
+ "io"
+ "os"
"path/filepath"
"time"
@@ -108,6 +110,309 @@ var _ = Describe("artistArtworkReader", func() {
})
})
})
+
+ var _ = Describe("fromArtistFolder", func() {
+ var (
+ ctx context.Context
+ tempDir string
+ testFunc sourceFunc
+ )
+
+ BeforeEach(func() {
+ ctx = context.Background()
+ tempDir = GinkgoT().TempDir()
+ })
+
+ When("artist folder contains matching image", func() {
+ BeforeEach(func() {
+ // Create test structure: /temp/artist/artist.jpg
+ artistDir := filepath.Join(tempDir, "artist")
+ Expect(os.MkdirAll(artistDir, 0755)).To(Succeed())
+
+ artistImagePath := filepath.Join(artistDir, "artist.jpg")
+ Expect(os.WriteFile(artistImagePath, []byte("fake image data"), 0600)).To(Succeed())
+
+ testFunc = fromArtistFolder(ctx, artistDir, "artist.*")
+ })
+
+ It("finds and returns the image", func() {
+ reader, path, err := testFunc()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(reader).ToNot(BeNil())
+ Expect(path).To(ContainSubstring("artist.jpg"))
+
+ // Verify we can read the content
+ data, err := io.ReadAll(reader)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(string(data)).To(Equal("fake image data"))
+ reader.Close()
+ })
+ })
+
+ When("artist folder is empty but parent contains image", func() {
+ BeforeEach(func() {
+ // Create test structure: /temp/parent/artist.jpg and /temp/parent/artist/album/
+ parentDir := filepath.Join(tempDir, "parent")
+ artistDir := filepath.Join(parentDir, "artist")
+ albumDir := filepath.Join(artistDir, "album")
+ Expect(os.MkdirAll(albumDir, 0755)).To(Succeed())
+
+ // Put artist image in parent directory
+ artistImagePath := filepath.Join(parentDir, "artist.jpg")
+ Expect(os.WriteFile(artistImagePath, []byte("parent image"), 0600)).To(Succeed())
+
+ testFunc = fromArtistFolder(ctx, artistDir, "artist.*")
+ })
+
+ It("finds image in parent directory", func() {
+ reader, path, err := testFunc()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(reader).ToNot(BeNil())
+ Expect(path).To(ContainSubstring("parent" + string(filepath.Separator) + "artist.jpg"))
+
+ data, err := io.ReadAll(reader)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(string(data)).To(Equal("parent image"))
+ reader.Close()
+ })
+ })
+
+ When("image is two levels up", func() {
+ BeforeEach(func() {
+ // Create test structure: /temp/grandparent/artist.jpg and /temp/grandparent/parent/artist/
+ grandparentDir := filepath.Join(tempDir, "grandparent")
+ parentDir := filepath.Join(grandparentDir, "parent")
+ artistDir := filepath.Join(parentDir, "artist")
+ Expect(os.MkdirAll(artistDir, 0755)).To(Succeed())
+
+ // Put artist image in grandparent directory
+ artistImagePath := filepath.Join(grandparentDir, "artist.jpg")
+ Expect(os.WriteFile(artistImagePath, []byte("grandparent image"), 0600)).To(Succeed())
+
+ testFunc = fromArtistFolder(ctx, artistDir, "artist.*")
+ })
+
+ It("finds image in grandparent directory", func() {
+ reader, path, err := testFunc()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(reader).ToNot(BeNil())
+ Expect(path).To(ContainSubstring("grandparent" + string(filepath.Separator) + "artist.jpg"))
+
+ data, err := io.ReadAll(reader)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(string(data)).To(Equal("grandparent image"))
+ reader.Close()
+ })
+ })
+
+ When("images exist at multiple levels", func() {
+ BeforeEach(func() {
+ // Create test structure with images at multiple levels
+ grandparentDir := filepath.Join(tempDir, "grandparent")
+ parentDir := filepath.Join(grandparentDir, "parent")
+ artistDir := filepath.Join(parentDir, "artist")
+ Expect(os.MkdirAll(artistDir, 0755)).To(Succeed())
+
+ // Put artist images at all levels
+ Expect(os.WriteFile(filepath.Join(artistDir, "artist.jpg"), []byte("artist level"), 0600)).To(Succeed())
+ Expect(os.WriteFile(filepath.Join(parentDir, "artist.jpg"), []byte("parent level"), 0600)).To(Succeed())
+ Expect(os.WriteFile(filepath.Join(grandparentDir, "artist.jpg"), []byte("grandparent level"), 0600)).To(Succeed())
+
+ testFunc = fromArtistFolder(ctx, artistDir, "artist.*")
+ })
+
+ It("prioritizes the closest (artist folder) image", func() {
+ reader, path, err := testFunc()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(reader).ToNot(BeNil())
+ Expect(path).To(ContainSubstring("artist" + string(filepath.Separator) + "artist.jpg"))
+
+ data, err := io.ReadAll(reader)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(string(data)).To(Equal("artist level"))
+ reader.Close()
+ })
+ })
+
+ When("pattern matches multiple files", func() {
+ BeforeEach(func() {
+ artistDir := filepath.Join(tempDir, "artist")
+ Expect(os.MkdirAll(artistDir, 0755)).To(Succeed())
+
+ // Create multiple matching files
+ Expect(os.WriteFile(filepath.Join(artistDir, "artist.abc"), []byte("text file"), 0600)).To(Succeed())
+ Expect(os.WriteFile(filepath.Join(artistDir, "artist.png"), []byte("png image"), 0600)).To(Succeed())
+ Expect(os.WriteFile(filepath.Join(artistDir, "artist.jpg"), []byte("jpg image"), 0600)).To(Succeed())
+
+ testFunc = fromArtistFolder(ctx, artistDir, "artist.*")
+ })
+
+ It("returns the first valid image file in sorted order", func() {
+ reader, path, err := testFunc()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(reader).ToNot(BeNil())
+
+ // Should return an image file,
+ // Files are sorted: jpg comes before png alphabetically.
+ // .abc comes first, but it's not an image.
+ Expect(path).To(ContainSubstring("artist.jpg"))
+ reader.Close()
+ })
+ })
+
+ When("prioritizing files without numeric suffixes", func() {
+ BeforeEach(func() {
+ // Test case for issue #4683: artist.jpg should come before artist.1.jpg
+ artistDir := filepath.Join(tempDir, "artist")
+ Expect(os.MkdirAll(artistDir, 0755)).To(Succeed())
+
+ // Create multiple matches with and without numeric suffixes
+ Expect(os.WriteFile(filepath.Join(artistDir, "artist.1.jpg"), []byte("artist 1"), 0600)).To(Succeed())
+ Expect(os.WriteFile(filepath.Join(artistDir, "artist.jpg"), []byte("artist main"), 0600)).To(Succeed())
+ Expect(os.WriteFile(filepath.Join(artistDir, "artist.2.jpg"), []byte("artist 2"), 0600)).To(Succeed())
+
+ testFunc = fromArtistFolder(ctx, artistDir, "artist.*")
+ })
+
+ It("returns artist.jpg before artist.1.jpg and artist.2.jpg", func() {
+ reader, path, err := testFunc()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(reader).ToNot(BeNil())
+ Expect(path).To(ContainSubstring("artist.jpg"))
+
+ // Verify it's the main file, not a numbered variant
+ data, err := io.ReadAll(reader)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(string(data)).To(Equal("artist main"))
+ reader.Close()
+ })
+ })
+
+ When("handling case-insensitive sorting", func() {
+ BeforeEach(func() {
+ // Test case to ensure case-insensitive natural sorting
+ artistDir := filepath.Join(tempDir, "artist")
+ Expect(os.MkdirAll(artistDir, 0755)).To(Succeed())
+
+ // Create files with mixed case names
+ Expect(os.WriteFile(filepath.Join(artistDir, "Folder.jpg"), []byte("folder"), 0600)).To(Succeed())
+ Expect(os.WriteFile(filepath.Join(artistDir, "artist.jpg"), []byte("artist"), 0600)).To(Succeed())
+ Expect(os.WriteFile(filepath.Join(artistDir, "BACK.jpg"), []byte("back"), 0600)).To(Succeed())
+
+ testFunc = fromArtistFolder(ctx, artistDir, "*.*")
+ })
+
+ It("sorts case-insensitively", func() {
+ reader, path, err := testFunc()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(reader).ToNot(BeNil())
+
+ // Should return artist.jpg first (case-insensitive: "artist" < "back" < "folder")
+ Expect(path).To(ContainSubstring("artist.jpg"))
+
+ data, err := io.ReadAll(reader)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(string(data)).To(Equal("artist"))
+ reader.Close()
+ })
+ })
+
+ When("no matching files exist anywhere", func() {
+ BeforeEach(func() {
+ artistDir := filepath.Join(tempDir, "artist")
+ Expect(os.MkdirAll(artistDir, 0755)).To(Succeed())
+
+ // Create non-matching files
+ Expect(os.WriteFile(filepath.Join(artistDir, "cover.jpg"), []byte("cover image"), 0600)).To(Succeed())
+
+ testFunc = fromArtistFolder(ctx, artistDir, "artist.*")
+ })
+
+ It("returns an error", func() {
+ reader, path, err := testFunc()
+ Expect(err).To(HaveOccurred())
+ Expect(reader).To(BeNil())
+ Expect(path).To(BeEmpty())
+ Expect(err.Error()).To(ContainSubstring("no matches for 'artist.*'"))
+ Expect(err.Error()).To(ContainSubstring("parent directories"))
+ })
+ })
+
+ When("directory traversal reaches filesystem root", func() {
+ BeforeEach(func() {
+ // Start from a shallow directory to test root boundary
+ artistDir := filepath.Join(tempDir, "artist")
+ Expect(os.MkdirAll(artistDir, 0755)).To(Succeed())
+
+ testFunc = fromArtistFolder(ctx, artistDir, "artist.*")
+ })
+
+ It("handles root boundary gracefully", func() {
+ reader, path, err := testFunc()
+ Expect(err).To(HaveOccurred())
+ Expect(reader).To(BeNil())
+ Expect(path).To(BeEmpty())
+ // Should not panic or cause infinite loop
+ })
+ })
+
+ When("file exists but cannot be opened", func() {
+ BeforeEach(func() {
+ artistDir := filepath.Join(tempDir, "artist")
+ Expect(os.MkdirAll(artistDir, 0755)).To(Succeed())
+
+ // Create a file that cannot be opened (permission denied)
+ restrictedFile := filepath.Join(artistDir, "artist.jpg")
+ Expect(os.WriteFile(restrictedFile, []byte("restricted"), 0600)).To(Succeed())
+
+ testFunc = fromArtistFolder(ctx, artistDir, "artist.*")
+ })
+
+ It("logs warning and continues searching", func() {
+ // This test depends on the ability to restrict file permissions
+ // For now, we'll just ensure it doesn't panic and returns appropriate error
+ reader, _, err := testFunc()
+ // The file should be readable in test environment, so this will succeed
+ // In a real scenario with permission issues, it would continue searching
+ if err == nil {
+ Expect(reader).ToNot(BeNil())
+ reader.Close()
+ }
+ })
+ })
+
+ When("single album artist scenario (original issue)", func() {
+ BeforeEach(func() {
+ // Simulate the exact folder structure from the issue:
+ // /music/artist/album1/ (single album)
+ // /music/artist/artist.jpg (artist image that should be found)
+ artistDir := filepath.Join(tempDir, "music", "artist")
+ albumDir := filepath.Join(artistDir, "album1")
+ Expect(os.MkdirAll(albumDir, 0755)).To(Succeed())
+
+ // Create artist.jpg in the artist folder (this was not being found before)
+ artistImagePath := filepath.Join(artistDir, "artist.jpg")
+ Expect(os.WriteFile(artistImagePath, []byte("single album artist image"), 0600)).To(Succeed())
+
+ // The fromArtistFolder is called with the artist folder path
+ testFunc = fromArtistFolder(ctx, artistDir, "artist.*")
+ })
+
+ It("finds artist.jpg in artist folder for single album artist", func() {
+ reader, path, err := testFunc()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(reader).ToNot(BeNil())
+ Expect(path).To(ContainSubstring("artist.jpg"))
+ Expect(path).To(ContainSubstring("artist"))
+
+ // Verify the content
+ data, err := io.ReadAll(reader)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(string(data)).To(Equal("single album artist image"))
+ reader.Close()
+ })
+ })
+ })
})
type fakeFolderRepo struct {
diff --git a/core/artwork/sources.go b/core/artwork/sources.go
index 121e6c38b..4250a373b 100644
--- a/core/artwork/sources.go
+++ b/core/artwork/sources.go
@@ -188,7 +188,7 @@ func fromURL(ctx context.Context, imageUrl *url.URL) (io.ReadCloser, string, err
}
if resp.StatusCode != http.StatusOK {
resp.Body.Close()
- return nil, "", fmt.Errorf("error retrieveing artwork from %s: %s", imageUrl, resp.Status)
+ return nil, "", fmt.Errorf("error retrieving artwork from %s: %s", imageUrl, resp.Status)
}
return resp.Body, imageUrl.String(), nil
}
diff --git a/core/external/extdata_helper_test.go b/core/external/extdata_helper_test.go
index 367437815..29975e5c5 100644
--- a/core/external/extdata_helper_test.go
+++ b/core/external/extdata_helper_test.go
@@ -190,10 +190,13 @@ type mockAgents struct {
topSongsAgent agents.ArtistTopSongsRetriever
similarAgent agents.ArtistSimilarRetriever
imageAgent agents.ArtistImageRetriever
- albumInfoAgent agents.AlbumInfoRetriever
- bioAgent agents.ArtistBiographyRetriever
- mbidAgent agents.ArtistMBIDRetriever
- urlAgent agents.ArtistURLRetriever
+ albumInfoAgent interface {
+ agents.AlbumInfoRetriever
+ agents.AlbumImageRetriever
+ }
+ bioAgent agents.ArtistBiographyRetriever
+ mbidAgent agents.ArtistMBIDRetriever
+ urlAgent agents.ArtistURLRetriever
agents.Interface
}
@@ -268,3 +271,14 @@ func (m *mockAgents) GetArtistImages(ctx context.Context, id, name, mbid string)
}
return nil, args.Error(1)
}
+
+func (m *mockAgents) GetAlbumImages(ctx context.Context, name, artist, mbid string) ([]agents.ExternalImage, error) {
+ if m.albumInfoAgent != nil {
+ return m.albumInfoAgent.GetAlbumImages(ctx, name, artist, mbid)
+ }
+ args := m.Called(ctx, name, artist, mbid)
+ if args.Get(0) != nil {
+ return args.Get(0).([]agents.ExternalImage), args.Error(1)
+ }
+ return nil, args.Error(1)
+}
diff --git a/core/external/provider.go b/core/external/provider.go
index f27ded11b..8e9a458c1 100644
--- a/core/external/provider.go
+++ b/core/external/provider.go
@@ -3,6 +3,7 @@ package external
import (
"context"
"errors"
+ "fmt"
"net/url"
"sort"
"strings"
@@ -11,6 +12,7 @@ import (
"github.com/Masterminds/squirrel"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/core/agents"
+ _ "github.com/navidrome/navidrome/core/agents/deezer"
_ "github.com/navidrome/navidrome/core/agents/lastfm"
_ "github.com/navidrome/navidrome/core/agents/listenbrainz"
_ "github.com/navidrome/navidrome/core/agents/spotify"
@@ -34,7 +36,7 @@ const (
type Provider interface {
UpdateAlbumInfo(ctx context.Context, id string) (*model.Album, error)
UpdateArtistInfo(ctx context.Context, id string, count int, includeNotPresent bool) (*model.Artist, error)
- SimilarSongs(ctx context.Context, id string, count int) (model.MediaFiles, error)
+ ArtistRadio(ctx context.Context, id string, count int) (model.MediaFiles, error)
TopSongs(ctx context.Context, artist string, count int) (model.MediaFiles, error)
ArtistImage(ctx context.Context, id string) (*url.URL, error)
AlbumImage(ctx context.Context, id string) (*url.URL, error)
@@ -59,6 +61,7 @@ type auxArtist struct {
type Agents interface {
agents.AlbumInfoRetriever
+ agents.AlbumImageRetriever
agents.ArtistBiographyRetriever
agents.ArtistMBIDRetriever
agents.ArtistImageRetriever
@@ -139,19 +142,20 @@ func (e *provider) populateAlbumInfo(ctx context.Context, album auxAlbum) (auxAl
album.Description = info.Description
}
- if len(info.Images) > 0 {
- sort.Slice(info.Images, func(i, j int) bool {
- return info.Images[i].Size > info.Images[j].Size
+ images, err := e.ag.GetAlbumImages(ctx, album.Name, album.AlbumArtist, album.MbzAlbumID)
+ if err == nil && len(images) > 0 {
+ sort.Slice(images, func(i, j int) bool {
+ return images[i].Size > images[j].Size
})
- album.LargeImageUrl = info.Images[0].URL
+ album.LargeImageUrl = images[0].URL
- if len(info.Images) >= 2 {
- album.MediumImageUrl = info.Images[1].URL
+ if len(images) >= 2 {
+ album.MediumImageUrl = images[1].URL
}
- if len(info.Images) >= 3 {
- album.SmallImageUrl = info.Images[2].URL
+ if len(images) >= 3 {
+ album.SmallImageUrl = images[2].URL
}
}
@@ -257,7 +261,7 @@ func (e *provider) populateArtistInfo(ctx context.Context, artist auxArtist) (au
return artist, nil
}
-func (e *provider) SimilarSongs(ctx context.Context, id string, count int) (model.MediaFiles, error) {
+func (e *provider) ArtistRadio(ctx context.Context, id string, count int) (model.MediaFiles, error) {
artist, err := e.getArtist(ctx, id)
if err != nil {
return nil, err
@@ -265,14 +269,14 @@ func (e *provider) SimilarSongs(ctx context.Context, id string, count int) (mode
e.callGetSimilar(ctx, e.ag, &artist, 15, false)
if utils.IsCtxDone(ctx) {
- log.Warn(ctx, "SimilarSongs call canceled", ctx.Err())
+ log.Warn(ctx, "ArtistRadio call canceled", ctx.Err())
return nil, ctx.Err()
}
weightedSongs := random.NewWeightedChooser[model.MediaFile]()
addArtist := func(a model.Artist, weightedSongs *random.WeightedChooser[model.MediaFile], count, artistWeight int) error {
if utils.IsCtxDone(ctx) {
- log.Warn(ctx, "SimilarSongs call canceled", ctx.Err())
+ log.Warn(ctx, "ArtistRadio call canceled", ctx.Err())
return ctx.Err()
}
@@ -340,29 +344,28 @@ func (e *provider) AlbumImage(ctx context.Context, id string) (*url.URL, error)
return nil, err
}
- info, err := e.ag.GetAlbumInfo(ctx, album.Name, album.AlbumArtist, album.MbzAlbumID)
+ images, err := e.ag.GetAlbumImages(ctx, album.Name, album.AlbumArtist, album.MbzAlbumID)
if err != nil {
switch {
case errors.Is(err, agents.ErrNotFound):
log.Trace(ctx, "Album not found in agent", "albumID", id, "name", album.Name, "artist", album.AlbumArtist)
return nil, model.ErrNotFound
case errors.Is(err, context.Canceled):
- log.Debug(ctx, "GetAlbumInfo call canceled", err)
+ log.Debug(ctx, "GetAlbumImages call canceled", err)
default:
- log.Warn(ctx, "Error getting album info from agent", "albumID", id, "name", album.Name, "artist", album.AlbumArtist, err)
+ log.Warn(ctx, "Error getting album images from agent", "albumID", id, "name", album.Name, "artist", album.AlbumArtist, err)
}
-
return nil, err
}
- if info == nil {
- log.Warn(ctx, "Agent returned nil info without error", "albumID", id, "name", album.Name, "artist", album.AlbumArtist)
+ if len(images) == 0 {
+ log.Warn(ctx, "Agent returned no images without error", "albumID", id, "name", album.Name, "artist", album.AlbumArtist)
return nil, model.ErrNotFound
}
// Return the biggest image
var img agents.ExternalImage
- for _, i := range info.Images {
+ for _, i := range images {
if img.Size <= i.Size {
img = i
}
@@ -400,20 +403,21 @@ func (e *provider) TopSongs(ctx context.Context, artistName string, count int) (
func (e *provider) getMatchingTopSongs(ctx context.Context, agent agents.ArtistTopSongsRetriever, artist *auxArtist, count int) (model.MediaFiles, error) {
songs, err := agent.GetArtistTopSongs(ctx, artist.ID, artist.Name, artist.MbzArtistID, count)
if err != nil {
- return nil, err
+ return nil, fmt.Errorf("failed to get top songs for artist %s: %w", artist.Name, err)
}
- var mfs model.MediaFiles
- for _, t := range songs {
- mf, err := e.findMatchingTrack(ctx, t.MBID, artist.ID, t.Name)
- if err != nil {
- continue
- }
- mfs = append(mfs, *mf)
- if len(mfs) == count {
- break
- }
+ mbidMatches, err := e.loadTracksByMBID(ctx, songs)
+ if err != nil {
+ return nil, fmt.Errorf("failed to load tracks by MBID: %w", err)
}
+ titleMatches, err := e.loadTracksByTitle(ctx, songs, artist, mbidMatches)
+ if err != nil {
+ return nil, fmt.Errorf("failed to load tracks by title: %w", err)
+ }
+
+ log.Trace(ctx, "Top Songs loaded", "name", artist.Name, "numSongs", len(songs), "numMBIDMatches", len(mbidMatches), "numTitleMatches", len(titleMatches))
+ mfs := e.selectTopSongs(songs, mbidMatches, titleMatches, count)
+
if len(mfs) == 0 {
log.Debug(ctx, "No matching top songs found", "name", artist.Name)
} else {
@@ -423,35 +427,94 @@ func (e *provider) getMatchingTopSongs(ctx context.Context, agent agents.ArtistT
return mfs, nil
}
-func (e *provider) findMatchingTrack(ctx context.Context, mbid string, artistID, title string) (*model.MediaFile, error) {
- if mbid != "" {
- mfs, err := e.ds.MediaFile(ctx).GetAll(model.QueryOptions{
- Filters: squirrel.And{
- squirrel.Eq{"mbz_recording_id": mbid},
- squirrel.Eq{"missing": false},
- },
- })
- if err == nil && len(mfs) > 0 {
- return &mfs[0], nil
+func (e *provider) loadTracksByMBID(ctx context.Context, songs []agents.Song) (map[string]model.MediaFile, error) {
+ var mbids []string
+ for _, s := range songs {
+ if s.MBID != "" {
+ mbids = append(mbids, s.MBID)
}
- return e.findMatchingTrack(ctx, "", artistID, title)
}
- mfs, err := e.ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ matches := map[string]model.MediaFile{}
+ if len(mbids) == 0 {
+ return matches, nil
+ }
+ res, err := e.ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.And{
+ squirrel.Eq{"mbz_recording_id": mbids},
+ squirrel.Eq{"missing": false},
+ },
+ })
+ if err != nil {
+ return matches, err
+ }
+ for _, mf := range res {
+ if id := mf.MbzRecordingID; id != "" {
+ if _, ok := matches[id]; !ok {
+ matches[id] = mf
+ }
+ }
+ }
+ return matches, nil
+}
+
+func (e *provider) loadTracksByTitle(ctx context.Context, songs []agents.Song, artist *auxArtist, mbidMatches map[string]model.MediaFile) (map[string]model.MediaFile, error) {
+ titleMap := map[string]string{}
+ for _, s := range songs {
+ if s.MBID != "" && mbidMatches[s.MBID].ID != "" {
+ continue
+ }
+ sanitized := str.SanitizeFieldForSorting(s.Name)
+ titleMap[sanitized] = s.Name
+ }
+ matches := map[string]model.MediaFile{}
+ if len(titleMap) == 0 {
+ return matches, nil
+ }
+ titleFilters := squirrel.Or{}
+ for sanitized := range titleMap {
+ titleFilters = append(titleFilters, squirrel.Like{"order_title": sanitized})
+ }
+
+ res, err := e.ds.MediaFile(ctx).GetAll(model.QueryOptions{
Filters: squirrel.And{
squirrel.Or{
- squirrel.Eq{"artist_id": artistID},
- squirrel.Eq{"album_artist_id": artistID},
+ squirrel.Eq{"artist_id": artist.ID},
+ squirrel.Eq{"album_artist_id": artist.ID},
},
- squirrel.Like{"order_title": str.SanitizeFieldForSorting(title)},
+ titleFilters,
squirrel.Eq{"missing": false},
},
Sort: "starred desc, rating desc, year asc, compilation asc ",
- Max: 1,
})
- if err != nil || len(mfs) == 0 {
- return nil, model.ErrNotFound
+ if err != nil {
+ return matches, err
}
- return &mfs[0], nil
+ for _, mf := range res {
+ sanitized := str.SanitizeFieldForSorting(mf.Title)
+ if _, ok := matches[sanitized]; !ok {
+ matches[sanitized] = mf
+ }
+ }
+ return matches, nil
+}
+
+func (e *provider) selectTopSongs(songs []agents.Song, byMBID, byTitle map[string]model.MediaFile, count int) model.MediaFiles {
+ var mfs model.MediaFiles
+ for _, t := range songs {
+ if len(mfs) == count {
+ break
+ }
+ if t.MBID != "" {
+ if mf, ok := byMBID[t.MBID]; ok {
+ mfs = append(mfs, mf)
+ continue
+ }
+ }
+ if mf, ok := byTitle[str.SanitizeFieldForSorting(t.Name)]; ok {
+ mfs = append(mfs, mf)
+ }
+ }
+ return mfs
}
func (e *provider) callGetURL(ctx context.Context, agent agents.ArtistURLRetriever, artist *auxArtist) {
@@ -497,7 +560,7 @@ func (e *provider) callGetSimilar(ctx context.Context, agent agents.ArtistSimila
return
}
start := time.Now()
- sa, err := e.mapSimilarArtists(ctx, similar, includeNotPresent)
+ sa, err := e.mapSimilarArtists(ctx, similar, limit, includeNotPresent)
log.Debug(ctx, "Mapped Similar Artists", "artist", artist.Name, "numSimilar", len(sa), "elapsed", time.Since(start))
if err != nil {
return
@@ -505,7 +568,7 @@ func (e *provider) callGetSimilar(ctx context.Context, agent agents.ArtistSimila
artist.SimilarArtists = sa
}
-func (e *provider) mapSimilarArtists(ctx context.Context, similar []agents.Artist, includeNotPresent bool) (model.Artists, error) {
+func (e *provider) mapSimilarArtists(ctx context.Context, similar []agents.Artist, limit int, includeNotPresent bool) (model.Artists, error) {
var result model.Artists
var notPresent []string
@@ -528,21 +591,33 @@ func (e *provider) mapSimilarArtists(ctx context.Context, similar []agents.Artis
artistMap[artist.Name] = artist
}
+ count := 0
+
// Process the similar artists
for _, s := range similar {
if artist, found := artistMap[s.Name]; found {
result = append(result, artist)
+ count++
+
+ if count >= limit {
+ break
+ }
} else {
notPresent = append(notPresent, s.Name)
}
}
// Then fill up with non-present artists
- if includeNotPresent {
+ if includeNotPresent && count < limit {
for _, s := range notPresent {
// Let the ID empty to indicate that the artist is not present in the DB
sa := model.Artist{Name: s}
result = append(result, sa)
+
+ count++
+ if count >= limit {
+ break
+ }
}
}
diff --git a/core/external/provider_albumimage_test.go b/core/external/provider_albumimage_test.go
index e248813c1..9b682462d 100644
--- a/core/external/provider_albumimage_test.go
+++ b/core/external/provider_albumimage_test.go
@@ -23,7 +23,6 @@ var _ = Describe("Provider - AlbumImage", func() {
var mockAlbumRepo *mockAlbumRepo
var mockMediaFileRepo *mockMediaFileRepo
var mockAlbumAgent *mockAlbumInfoAgent
- var agentsCombined *mockAgents
var ctx context.Context
BeforeEach(func() {
@@ -43,10 +42,7 @@ var _ = Describe("Provider - AlbumImage", func() {
mockAlbumAgent = newMockAlbumInfoAgent()
- agentsCombined = &mockAgents{
- albumInfoAgent: mockAlbumAgent,
- }
-
+ agentsCombined := &mockAgents{albumInfoAgent: mockAlbumAgent}
provider = NewProvider(ds, agentsCombined)
// Default mocks
@@ -66,13 +62,11 @@ var _ = Describe("Provider - AlbumImage", func() {
mockArtistRepo.On("Get", "album-1").Return(nil, model.ErrNotFound).Once() // Expect GetEntityByID sequence
mockAlbumRepo.On("Get", "album-1").Return(&model.Album{ID: "album-1", Name: "Album One", AlbumArtistID: "artist-1"}, nil).Once()
// Explicitly mock agent call for this test
- mockAlbumAgent.On("GetAlbumInfo", ctx, "Album One", "", "").
- Return(&agents.AlbumInfo{
- Images: []agents.ExternalImage{
- {URL: "http://example.com/large.jpg", Size: 1000},
- {URL: "http://example.com/medium.jpg", Size: 500},
- {URL: "http://example.com/small.jpg", Size: 200},
- },
+ mockAlbumAgent.On("GetAlbumImages", ctx, "Album One", "", "").
+ Return([]agents.ExternalImage{
+ {URL: "http://example.com/large.jpg", Size: 1000},
+ {URL: "http://example.com/medium.jpg", Size: 500},
+ {URL: "http://example.com/small.jpg", Size: 200},
}, nil).Once()
expectedURL, _ := url.Parse("http://example.com/large.jpg")
@@ -82,8 +76,8 @@ var _ = Describe("Provider - AlbumImage", func() {
Expect(imgURL).To(Equal(expectedURL))
mockArtistRepo.AssertCalled(GinkgoT(), "Get", "album-1") // From GetEntityByID
mockAlbumRepo.AssertCalled(GinkgoT(), "Get", "album-1")
- mockArtistRepo.AssertNotCalled(GinkgoT(), "Get", "artist-1") // Artist lookup no longer happens in getAlbum
- mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumInfo", ctx, "Album One", "", "") // Expect empty artist name
+ mockArtistRepo.AssertNotCalled(GinkgoT(), "Get", "artist-1") // Artist lookup no longer happens in getAlbum
+ mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumImages", ctx, "Album One", "", "") // Expect empty artist name
})
It("returns ErrNotFound if the album is not found in the DB", func() {
@@ -99,7 +93,7 @@ var _ = Describe("Provider - AlbumImage", func() {
mockArtistRepo.AssertCalled(GinkgoT(), "Get", "not-found")
mockAlbumRepo.AssertCalled(GinkgoT(), "Get", "not-found")
mockMediaFileRepo.AssertCalled(GinkgoT(), "Get", "not-found")
- mockAlbumAgent.AssertNotCalled(GinkgoT(), "GetAlbumInfo", mock.Anything, mock.Anything, mock.Anything, mock.Anything)
+ mockAlbumAgent.AssertNotCalled(GinkgoT(), "GetAlbumImages", mock.Anything, mock.Anything, mock.Anything)
})
It("returns the agent error if the agent fails", func() {
@@ -109,7 +103,7 @@ var _ = Describe("Provider - AlbumImage", func() {
agentErr := errors.New("agent failure")
// Explicitly mock agent call for this test
- mockAlbumAgent.On("GetAlbumInfo", ctx, "Album One", "", "").Return(nil, agentErr).Once() // Expect empty artist
+ mockAlbumAgent.On("GetAlbumImages", ctx, "Album One", "", "").Return(nil, agentErr).Once() // Expect empty artist
imgURL, err := provider.AlbumImage(ctx, "album-1")
@@ -118,7 +112,7 @@ var _ = Describe("Provider - AlbumImage", func() {
mockArtistRepo.AssertCalled(GinkgoT(), "Get", "album-1")
mockAlbumRepo.AssertCalled(GinkgoT(), "Get", "album-1")
mockArtistRepo.AssertNotCalled(GinkgoT(), "Get", "artist-1")
- mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumInfo", ctx, "Album One", "", "") // Expect empty artist
+ mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumImages", ctx, "Album One", "", "") // Expect empty artist
})
It("returns ErrNotFound if the agent returns ErrNotFound", func() {
@@ -127,7 +121,7 @@ var _ = Describe("Provider - AlbumImage", func() {
mockAlbumRepo.On("Get", "album-1").Return(&model.Album{ID: "album-1", Name: "Album One", AlbumArtistID: "artist-1"}, nil).Once()
// Explicitly mock agent call for this test
- mockAlbumAgent.On("GetAlbumInfo", ctx, "Album One", "", "").Return(nil, agents.ErrNotFound).Once() // Expect empty artist
+ mockAlbumAgent.On("GetAlbumImages", ctx, "Album One", "", "").Return(nil, agents.ErrNotFound).Once() // Expect empty artist
imgURL, err := provider.AlbumImage(ctx, "album-1")
@@ -135,7 +129,7 @@ var _ = Describe("Provider - AlbumImage", func() {
Expect(imgURL).To(BeNil())
mockArtistRepo.AssertCalled(GinkgoT(), "Get", "album-1")
mockAlbumRepo.AssertCalled(GinkgoT(), "Get", "album-1")
- mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumInfo", ctx, "Album One", "", "") // Expect empty artist
+ mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumImages", ctx, "Album One", "", "") // Expect empty artist
})
It("returns ErrNotFound if the agent returns no images", func() {
@@ -144,8 +138,8 @@ var _ = Describe("Provider - AlbumImage", func() {
mockAlbumRepo.On("Get", "album-1").Return(&model.Album{ID: "album-1", Name: "Album One", AlbumArtistID: "artist-1"}, nil).Once()
// Explicitly mock agent call for this test
- mockAlbumAgent.On("GetAlbumInfo", ctx, "Album One", "", "").
- Return(&agents.AlbumInfo{Images: []agents.ExternalImage{}}, nil).Once() // Expect empty artist
+ mockAlbumAgent.On("GetAlbumImages", ctx, "Album One", "", "").
+ Return([]agents.ExternalImage{}, nil).Once() // Expect empty artist
imgURL, err := provider.AlbumImage(ctx, "album-1")
@@ -153,7 +147,7 @@ var _ = Describe("Provider - AlbumImage", func() {
Expect(imgURL).To(BeNil())
mockArtistRepo.AssertCalled(GinkgoT(), "Get", "album-1")
mockAlbumRepo.AssertCalled(GinkgoT(), "Get", "album-1")
- mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumInfo", ctx, "Album One", "", "") // Expect empty artist
+ mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumImages", ctx, "Album One", "", "") // Expect empty artist
})
It("returns context error if context is canceled", func() {
@@ -163,7 +157,7 @@ var _ = Describe("Provider - AlbumImage", func() {
mockArtistRepo.On("Get", "album-1").Return(nil, model.ErrNotFound).Once()
mockAlbumRepo.On("Get", "album-1").Return(&model.Album{ID: "album-1", Name: "Album One", AlbumArtistID: "artist-1"}, nil).Once()
// Expect the agent call even if context is cancelled, returning the context error
- mockAlbumAgent.On("GetAlbumInfo", cctx, "Album One", "", "").Return(nil, context.Canceled).Once()
+ mockAlbumAgent.On("GetAlbumImages", cctx, "Album One", "", "").Return(nil, context.Canceled).Once()
// Cancel the context *before* calling the function under test
cancelCtx()
@@ -174,7 +168,7 @@ var _ = Describe("Provider - AlbumImage", func() {
mockArtistRepo.AssertCalled(GinkgoT(), "Get", "album-1")
mockAlbumRepo.AssertCalled(GinkgoT(), "Get", "album-1")
// Agent should now be called, verify this expectation
- mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumInfo", cctx, "Album One", "", "")
+ mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumImages", cctx, "Album One", "", "")
})
It("derives album ID from MediaFile ID", func() {
@@ -186,13 +180,11 @@ var _ = Describe("Provider - AlbumImage", func() {
mockAlbumRepo.On("Get", "album-1").Return(&model.Album{ID: "album-1", Name: "Album One", AlbumArtistID: "artist-1"}, nil).Once()
// Explicitly mock agent call for this test
- mockAlbumAgent.On("GetAlbumInfo", ctx, "Album One", "", "").
- Return(&agents.AlbumInfo{
- Images: []agents.ExternalImage{
- {URL: "http://example.com/large.jpg", Size: 1000},
- {URL: "http://example.com/medium.jpg", Size: 500},
- {URL: "http://example.com/small.jpg", Size: 200},
- },
+ mockAlbumAgent.On("GetAlbumImages", ctx, "Album One", "", "").
+ Return([]agents.ExternalImage{
+ {URL: "http://example.com/large.jpg", Size: 1000},
+ {URL: "http://example.com/medium.jpg", Size: 500},
+ {URL: "http://example.com/small.jpg", Size: 200},
}, nil).Once()
expectedURL, _ := url.Parse("http://example.com/large.jpg")
@@ -206,7 +198,7 @@ var _ = Describe("Provider - AlbumImage", func() {
mockArtistRepo.AssertCalled(GinkgoT(), "Get", "album-1")
mockAlbumRepo.AssertCalled(GinkgoT(), "Get", "album-1")
mockArtistRepo.AssertNotCalled(GinkgoT(), "Get", "artist-1")
- mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumInfo", ctx, "Album One", "", "")
+ mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumImages", ctx, "Album One", "", "")
})
It("handles different image orders from agent", func() {
@@ -214,13 +206,11 @@ var _ = Describe("Provider - AlbumImage", func() {
mockArtistRepo.On("Get", "album-1").Return(nil, model.ErrNotFound).Once() // Expect GetEntityByID sequence
mockAlbumRepo.On("Get", "album-1").Return(&model.Album{ID: "album-1", Name: "Album One", AlbumArtistID: "artist-1"}, nil).Once()
// Explicitly mock agent call for this test
- mockAlbumAgent.On("GetAlbumInfo", ctx, "Album One", "", "").
- Return(&agents.AlbumInfo{
- Images: []agents.ExternalImage{
- {URL: "http://example.com/small.jpg", Size: 200},
- {URL: "http://example.com/large.jpg", Size: 1000},
- {URL: "http://example.com/medium.jpg", Size: 500},
- },
+ mockAlbumAgent.On("GetAlbumImages", ctx, "Album One", "", "").
+ Return([]agents.ExternalImage{
+ {URL: "http://example.com/small.jpg", Size: 200},
+ {URL: "http://example.com/large.jpg", Size: 1000},
+ {URL: "http://example.com/medium.jpg", Size: 500},
}, nil).Once()
expectedURL, _ := url.Parse("http://example.com/large.jpg")
@@ -228,7 +218,7 @@ var _ = Describe("Provider - AlbumImage", func() {
Expect(err).ToNot(HaveOccurred())
Expect(imgURL).To(Equal(expectedURL)) // Should still pick the largest
- mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumInfo", ctx, "Album One", "", "")
+ mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumImages", ctx, "Album One", "", "")
})
It("handles agent returning only one image", func() {
@@ -236,11 +226,9 @@ var _ = Describe("Provider - AlbumImage", func() {
mockArtistRepo.On("Get", "album-1").Return(nil, model.ErrNotFound).Once() // Expect GetEntityByID sequence
mockAlbumRepo.On("Get", "album-1").Return(&model.Album{ID: "album-1", Name: "Album One", AlbumArtistID: "artist-1"}, nil).Once()
// Explicitly mock agent call for this test
- mockAlbumAgent.On("GetAlbumInfo", ctx, "Album One", "", "").
- Return(&agents.AlbumInfo{
- Images: []agents.ExternalImage{
- {URL: "http://example.com/single.jpg", Size: 700},
- },
+ mockAlbumAgent.On("GetAlbumImages", ctx, "Album One", "", "").
+ Return([]agents.ExternalImage{
+ {URL: "http://example.com/single.jpg", Size: 700},
}, nil).Once()
expectedURL, _ := url.Parse("http://example.com/single.jpg")
@@ -248,7 +236,7 @@ var _ = Describe("Provider - AlbumImage", func() {
Expect(err).ToNot(HaveOccurred())
Expect(imgURL).To(Equal(expectedURL))
- mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumInfo", ctx, "Album One", "", "")
+ mockAlbumAgent.AssertCalled(GinkgoT(), "GetAlbumImages", ctx, "Album One", "", "")
})
It("returns ErrNotFound if deriving album ID fails", func() {
@@ -270,14 +258,15 @@ var _ = Describe("Provider - AlbumImage", func() {
mockArtistRepo.AssertCalled(GinkgoT(), "Get", "not-found")
mockAlbumRepo.AssertCalled(GinkgoT(), "Get", "not-found")
mockMediaFileRepo.AssertCalled(GinkgoT(), "Get", "not-found")
- mockAlbumAgent.AssertNotCalled(GinkgoT(), "GetAlbumInfo", mock.Anything, mock.Anything, mock.Anything, mock.Anything)
+ mockAlbumAgent.AssertNotCalled(GinkgoT(), "GetAlbumImages", mock.Anything, mock.Anything, mock.Anything)
})
})
// mockAlbumInfoAgent implementation
type mockAlbumInfoAgent struct {
mock.Mock
- agents.AlbumInfoRetriever // Embed interface
+ agents.AlbumInfoRetriever
+ agents.AlbumImageRetriever
}
func newMockAlbumInfoAgent() *mockAlbumInfoAgent {
@@ -299,5 +288,14 @@ func (m *mockAlbumInfoAgent) GetAlbumInfo(ctx context.Context, name, artist, mbi
return args.Get(0).(*agents.AlbumInfo), args.Error(1)
}
-// Ensure mockAgent implements the interface
+func (m *mockAlbumInfoAgent) GetAlbumImages(ctx context.Context, name, artist, mbid string) ([]agents.ExternalImage, error) {
+ args := m.Called(ctx, name, artist, mbid)
+ if args.Get(0) == nil {
+ return nil, args.Error(1)
+ }
+ return args.Get(0).([]agents.ExternalImage), args.Error(1)
+}
+
+// Ensure mockAgent implements the interfaces
var _ agents.AlbumInfoRetriever = (*mockAlbumInfoAgent)(nil)
+var _ agents.AlbumImageRetriever = (*mockAlbumInfoAgent)(nil)
diff --git a/core/external/provider_similarsongs_test.go b/core/external/provider_artistradio_test.go
similarity index 85%
rename from core/external/provider_similarsongs_test.go
rename to core/external/provider_artistradio_test.go
index fd622746a..21ea07706 100644
--- a/core/external/provider_similarsongs_test.go
+++ b/core/external/provider_artistradio_test.go
@@ -13,7 +13,7 @@ import (
"github.com/stretchr/testify/mock"
)
-var _ = Describe("Provider - SimilarSongs", func() {
+var _ = Describe("Provider - ArtistRadio", func() {
var ds model.DataStore
var provider Provider
var mockAgent *mockSimilarArtistAgent
@@ -50,9 +50,9 @@ var _ = Describe("Provider - SimilarSongs", func() {
It("returns similar songs from main artist and similar artists", func() {
artist1 := model.Artist{ID: "artist-1", Name: "Artist One"}
similarArtist := model.Artist{ID: "artist-3", Name: "Similar Artist"}
- song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1"}
- song2 := model.MediaFile{ID: "song-2", Title: "Song Two", ArtistID: "artist-1"}
- song3 := model.MediaFile{ID: "song-3", Title: "Song Three", ArtistID: "artist-3"}
+ song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1", MbzRecordingID: "mbid-1"}
+ song2 := model.MediaFile{ID: "song-2", Title: "Song Two", ArtistID: "artist-1", MbzRecordingID: "mbid-2"}
+ song3 := model.MediaFile{ID: "song-3", Title: "Song Three", ArtistID: "artist-3", MbzRecordingID: "mbid-3"}
artistRepo.On("Get", "artist-1").Return(&artist1, nil).Maybe()
artistRepo.On("Get", "artist-3").Return(&similarArtist, nil).Maybe()
@@ -82,11 +82,10 @@ var _ = Describe("Provider - SimilarSongs", func() {
{Name: "Song Three", MBID: "mbid-3"},
}, nil).Once()
- mediaFileRepo.FindByMBID("mbid-1", song1)
- mediaFileRepo.FindByMBID("mbid-2", song2)
- mediaFileRepo.FindByMBID("mbid-3", song3)
+ mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song1, song2}, nil).Once()
+ mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song3}, nil).Once()
- songs, err := provider.SimilarSongs(ctx, "artist-1", 3)
+ songs, err := provider.ArtistRadio(ctx, "artist-1", 3)
Expect(err).ToNot(HaveOccurred())
Expect(songs).To(HaveLen(3))
@@ -103,7 +102,7 @@ var _ = Describe("Provider - SimilarSongs", func() {
return opt.Max == 1 && opt.Filters != nil
})).Return(model.Artists{}, nil).Maybe()
- songs, err := provider.SimilarSongs(ctx, "artist-unknown-artist", 5)
+ songs, err := provider.ArtistRadio(ctx, "artist-unknown-artist", 5)
Expect(err).To(Equal(model.ErrNotFound))
Expect(songs).To(BeNil())
@@ -111,7 +110,7 @@ var _ = Describe("Provider - SimilarSongs", func() {
It("returns songs from main artist when GetSimilarArtists returns error", func() {
artist1 := model.Artist{ID: "artist-1", Name: "Artist One"}
- song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1"}
+ song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1", MbzRecordingID: "mbid-1"}
artistRepo.On("Get", "artist-1").Return(&artist1, nil).Maybe()
artistRepo.On("GetAll", mock.MatchedBy(func(opt model.QueryOptions) bool {
@@ -130,9 +129,9 @@ var _ = Describe("Provider - SimilarSongs", func() {
{Name: "Song One", MBID: "mbid-1"},
}, nil).Once()
- mediaFileRepo.FindByMBID("mbid-1", song1)
+ mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song1}, nil).Once()
- songs, err := provider.SimilarSongs(ctx, "artist-1", 5)
+ songs, err := provider.ArtistRadio(ctx, "artist-1", 5)
Expect(err).ToNot(HaveOccurred())
Expect(songs).To(HaveLen(1))
@@ -157,7 +156,7 @@ var _ = Describe("Provider - SimilarSongs", func() {
mockAgent.On("GetArtistTopSongs", mock.Anything, "artist-1", "Artist One", "", mock.Anything).
Return(nil, errors.New("error getting top songs")).Once()
- songs, err := provider.SimilarSongs(ctx, "artist-1", 5)
+ songs, err := provider.ArtistRadio(ctx, "artist-1", 5)
Expect(err).ToNot(HaveOccurred())
Expect(songs).To(BeEmpty())
@@ -165,8 +164,8 @@ var _ = Describe("Provider - SimilarSongs", func() {
It("respects count parameter", func() {
artist1 := model.Artist{ID: "artist-1", Name: "Artist One"}
- song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1"}
- song2 := model.MediaFile{ID: "song-2", Title: "Song Two", ArtistID: "artist-1"}
+ song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1", MbzRecordingID: "mbid-1"}
+ song2 := model.MediaFile{ID: "song-2", Title: "Song Two", ArtistID: "artist-1", MbzRecordingID: "mbid-2"}
artistRepo.On("Get", "artist-1").Return(&artist1, nil).Maybe()
artistRepo.On("GetAll", mock.MatchedBy(func(opt model.QueryOptions) bool {
@@ -186,10 +185,9 @@ var _ = Describe("Provider - SimilarSongs", func() {
{Name: "Song Two", MBID: "mbid-2"},
}, nil).Once()
- mediaFileRepo.FindByMBID("mbid-1", song1)
- mediaFileRepo.FindByMBID("mbid-2", song2)
+ mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song1, song2}, nil).Once()
- songs, err := provider.SimilarSongs(ctx, "artist-1", 1)
+ songs, err := provider.ArtistRadio(ctx, "artist-1", 1)
Expect(err).ToNot(HaveOccurred())
Expect(songs).To(HaveLen(1))
diff --git a/core/external/provider_topsongs_test.go b/core/external/provider_topsongs_test.go
index 4ce7911de..5a5a25714 100644
--- a/core/external/provider_topsongs_test.go
+++ b/core/external/provider_topsongs_test.go
@@ -42,10 +42,6 @@ var _ = Describe("Provider - TopSongs", func() {
p = NewProvider(ds, ag)
})
- BeforeEach(func() {
- // Setup expectations in individual tests
- })
-
It("returns top songs for a known artist", func() {
// Mock finding the artist
artist1 := model.Artist{ID: "artist-1", Name: "Artist One", MbzArtistID: "mbid-artist-1"}
@@ -58,11 +54,10 @@ var _ = Describe("Provider - TopSongs", func() {
}
ag.On("GetArtistTopSongs", ctx, "artist-1", "Artist One", "mbid-artist-1", 2).Return(agentSongs, nil).Once()
- // Mock finding matching tracks
+ // Mock finding matching tracks (both returned in a single query)
song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1", MbzRecordingID: "mbid-song-1"}
song2 := model.MediaFile{ID: "song-2", Title: "Song Two", ArtistID: "artist-1", MbzRecordingID: "mbid-song-2"}
- mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song1}, nil).Once()
- mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song2}, nil).Once()
+ mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song1, song2}, nil).Once()
songs, err := p.TopSongs(ctx, "Artist One", 2)
@@ -155,11 +150,10 @@ var _ = Describe("Provider - TopSongs", func() {
}
ag.On("GetArtistTopSongs", ctx, "artist-1", "Artist One", "mbid-artist-1", 2).Return(agentSongs, nil).Once()
- // Mock finding matching tracks (only find song 1)
+ // Mock finding matching tracks (only find song 1 on bulk query)
song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1", MbzRecordingID: "mbid-song-1"}
- mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song1}, nil).Once()
- mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{}, nil).Once() // For mbid-song-2 (fails)
- mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{}, nil).Once() // For title fallback (fails)
+ mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song1}, nil).Once() // bulk MBID query
+ mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{}, nil).Once() // title fallback for song2
songs, err := p.TopSongs(ctx, "Artist One", 2)
@@ -190,4 +184,91 @@ var _ = Describe("Provider - TopSongs", func() {
artistRepo.AssertExpectations(GinkgoT())
ag.AssertExpectations(GinkgoT())
})
+
+ It("falls back to title matching when MbzRecordingID is missing", func() {
+ // Mock finding the artist
+ artist1 := model.Artist{ID: "artist-1", Name: "Artist One", MbzArtistID: "mbid-artist-1"}
+ artistRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.Artists{artist1}, nil).Once()
+
+ // Mock agent response with songs that have NO MBID (empty string)
+ agentSongs := []agents.Song{
+ {Name: "Song One", MBID: ""}, // No MBID, should fall back to title matching
+ {Name: "Song Two", MBID: ""}, // No MBID, should fall back to title matching
+ }
+ ag.On("GetArtistTopSongs", ctx, "artist-1", "Artist One", "mbid-artist-1", 2).Return(agentSongs, nil).Once()
+
+ // Since there are no MBIDs, loadTracksByMBID should not make any database call
+ // loadTracksByTitle should make a database call for title matching
+ song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1", MbzRecordingID: "", OrderTitle: "song one"}
+ song2 := model.MediaFile{ID: "song-2", Title: "Song Two", ArtistID: "artist-1", MbzRecordingID: "", OrderTitle: "song two"}
+ mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song1, song2}, nil).Once()
+
+ songs, err := p.TopSongs(ctx, "Artist One", 2)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(songs).To(HaveLen(2))
+ Expect(songs[0].ID).To(Equal("song-1"))
+ Expect(songs[1].ID).To(Equal("song-2"))
+ artistRepo.AssertExpectations(GinkgoT())
+ ag.AssertExpectations(GinkgoT())
+ mediaFileRepo.AssertExpectations(GinkgoT())
+ })
+
+ It("combines MBID and title matching when some songs have missing MbzRecordingID", func() {
+ // Mock finding the artist
+ artist1 := model.Artist{ID: "artist-1", Name: "Artist One", MbzArtistID: "mbid-artist-1"}
+ artistRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.Artists{artist1}, nil).Once()
+
+ // Mock agent response with mixed MBID availability
+ agentSongs := []agents.Song{
+ {Name: "Song One", MBID: "mbid-song-1"}, // Has MBID, should match by MBID
+ {Name: "Song Two", MBID: ""}, // No MBID, should fall back to title matching
+ }
+ ag.On("GetArtistTopSongs", ctx, "artist-1", "Artist One", "mbid-artist-1", 2).Return(agentSongs, nil).Once()
+
+ // Mock the MBID query (finds song1 by MBID)
+ song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1", MbzRecordingID: "mbid-song-1", OrderTitle: "song one"}
+ mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song1}, nil).Once()
+
+ // Mock the title fallback query (finds song2 by title)
+ song2 := model.MediaFile{ID: "song-2", Title: "Song Two", ArtistID: "artist-1", MbzRecordingID: "", OrderTitle: "song two"}
+ mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song2}, nil).Once()
+
+ songs, err := p.TopSongs(ctx, "Artist One", 2)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(songs).To(HaveLen(2))
+ Expect(songs[0].ID).To(Equal("song-1")) // Found by MBID
+ Expect(songs[1].ID).To(Equal("song-2")) // Found by title
+ artistRepo.AssertExpectations(GinkgoT())
+ ag.AssertExpectations(GinkgoT())
+ mediaFileRepo.AssertExpectations(GinkgoT())
+ })
+
+ It("only returns requested count when provider returns additional items", func() {
+ // Mock finding the artist
+ artist1 := model.Artist{ID: "artist-1", Name: "Artist One", MbzArtistID: "mbid-artist-1"}
+ artistRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.Artists{artist1}, nil).Once()
+
+ // Mock agent response
+ agentSongs := []agents.Song{
+ {Name: "Song One", MBID: "mbid-song-1"},
+ {Name: "Song Two", MBID: "mbid-song-2"},
+ }
+ ag.On("GetArtistTopSongs", ctx, "artist-1", "Artist One", "mbid-artist-1", 1).Return(agentSongs, nil).Once()
+
+ // Mock finding matching tracks (both returned in a single query)
+ song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1", MbzRecordingID: "mbid-song-1"}
+ song2 := model.MediaFile{ID: "song-2", Title: "Song Two", ArtistID: "artist-1", MbzRecordingID: "mbid-song-2"}
+ mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song1, song2}, nil).Once()
+
+ songs, err := p.TopSongs(ctx, "Artist One", 1)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(songs).To(HaveLen(1))
+ Expect(songs[0].ID).To(Equal("song-1"))
+ artistRepo.AssertExpectations(GinkgoT())
+ ag.AssertExpectations(GinkgoT())
+ mediaFileRepo.AssertExpectations(GinkgoT())
+ })
})
diff --git a/core/external/provider_updatealbuminfo_test.go b/core/external/provider_updatealbuminfo_test.go
index 0622849f0..5f5d41a87 100644
--- a/core/external/provider_updatealbuminfo_test.go
+++ b/core/external/provider_updatealbuminfo_test.go
@@ -59,13 +59,13 @@ var _ = Describe("Provider - UpdateAlbumInfo", func() {
expectedInfo := &agents.AlbumInfo{
URL: "http://example.com/album",
Description: "Album Description",
- Images: []agents.ExternalImage{
- {URL: "http://example.com/large.jpg", Size: 300},
- {URL: "http://example.com/medium.jpg", Size: 200},
- {URL: "http://example.com/small.jpg", Size: 100},
- },
}
ag.On("GetAlbumInfo", ctx, "Test Album", "Test Artist", "mbid-album").Return(expectedInfo, nil)
+ ag.On("GetAlbumImages", ctx, "Test Album", "Test Artist", "mbid-album").Return([]agents.ExternalImage{
+ {URL: "http://example.com/large.jpg", Size: 300},
+ {URL: "http://example.com/medium.jpg", Size: 200},
+ {URL: "http://example.com/small.jpg", Size: 100},
+ }, nil)
updatedAlbum, err := p.UpdateAlbumInfo(ctx, "al-existing")
@@ -74,9 +74,6 @@ var _ = Describe("Provider - UpdateAlbumInfo", func() {
Expect(updatedAlbum.ID).To(Equal("al-existing"))
Expect(updatedAlbum.ExternalUrl).To(Equal("http://example.com/album"))
Expect(updatedAlbum.Description).To(Equal("Album Description"))
- Expect(updatedAlbum.LargeImageUrl).To(Equal("http://example.com/large.jpg"))
- Expect(updatedAlbum.MediumImageUrl).To(Equal("http://example.com/medium.jpg"))
- Expect(updatedAlbum.SmallImageUrl).To(Equal("http://example.com/small.jpg"))
Expect(updatedAlbum.ExternalInfoUpdatedAt).NotTo(BeNil())
Expect(*updatedAlbum.ExternalInfoUpdatedAt).To(BeTemporally("~", time.Now(), time.Second))
diff --git a/core/library.go b/core/library.go
new file mode 100644
index 000000000..f4f55ec5a
--- /dev/null
+++ b/core/library.go
@@ -0,0 +1,407 @@
+package core
+
+import (
+ "context"
+ "errors"
+ "fmt"
+ "io/fs"
+ "os"
+ "path/filepath"
+ "strconv"
+ "strings"
+ "time"
+
+ "github.com/Masterminds/squirrel"
+ "github.com/deluan/rest"
+ "github.com/navidrome/navidrome/core/storage"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/server/events"
+ "github.com/navidrome/navidrome/utils/slice"
+)
+
+// Watcher interface for managing file system watchers
+type Watcher interface {
+ Watch(ctx context.Context, lib *model.Library) error
+ StopWatching(ctx context.Context, libraryID int) error
+}
+
+// Library provides business logic for library management and user-library associations
+type Library interface {
+ GetUserLibraries(ctx context.Context, userID string) (model.Libraries, error)
+ SetUserLibraries(ctx context.Context, userID string, libraryIDs []int) error
+ ValidateLibraryAccess(ctx context.Context, userID string, libraryID int) error
+
+ NewRepository(ctx context.Context) rest.Repository
+}
+
+type libraryService struct {
+ ds model.DataStore
+ scanner model.Scanner
+ watcher Watcher
+ broker events.Broker
+}
+
+// NewLibrary creates a new Library service
+func NewLibrary(ds model.DataStore, scanner model.Scanner, watcher Watcher, broker events.Broker) Library {
+ return &libraryService{
+ ds: ds,
+ scanner: scanner,
+ watcher: watcher,
+ broker: broker,
+ }
+}
+
+// User-library association operations
+
+func (s *libraryService) GetUserLibraries(ctx context.Context, userID string) (model.Libraries, error) {
+ // Verify user exists
+ if _, err := s.ds.User(ctx).Get(userID); err != nil {
+ return nil, err
+ }
+
+ return s.ds.User(ctx).GetUserLibraries(userID)
+}
+
+func (s *libraryService) SetUserLibraries(ctx context.Context, userID string, libraryIDs []int) error {
+ // Verify user exists
+ user, err := s.ds.User(ctx).Get(userID)
+ if err != nil {
+ return err
+ }
+
+ // Admin users get all libraries automatically - don't allow manual assignment
+ if user.IsAdmin {
+ return fmt.Errorf("%w: cannot manually assign libraries to admin users", model.ErrValidation)
+ }
+
+ // Regular users must have at least one library
+ if len(libraryIDs) == 0 {
+ return fmt.Errorf("%w: at least one library must be assigned to non-admin users", model.ErrValidation)
+ }
+
+ // Validate all library IDs exist
+ if len(libraryIDs) > 0 {
+ if err := s.validateLibraryIDs(ctx, libraryIDs); err != nil {
+ return err
+ }
+ }
+
+ // Set user libraries
+ err = s.ds.User(ctx).SetUserLibraries(userID, libraryIDs)
+ if err != nil {
+ return fmt.Errorf("error setting user libraries: %w", err)
+ }
+
+ // Send refresh event to all clients
+ event := &events.RefreshResource{}
+ libIDs := slice.Map(libraryIDs, func(id int) string { return strconv.Itoa(id) })
+ event = event.With("user", userID).With("library", libIDs...)
+ s.broker.SendBroadcastMessage(ctx, event)
+ return nil
+}
+
+func (s *libraryService) ValidateLibraryAccess(ctx context.Context, userID string, libraryID int) error {
+ user, ok := request.UserFrom(ctx)
+ if !ok {
+ return fmt.Errorf("user not found in context")
+ }
+
+ // Admin users have access to all libraries
+ if user.IsAdmin {
+ return nil
+ }
+
+ // Check if user has explicit access to this library
+ libraries, err := s.ds.User(ctx).GetUserLibraries(userID)
+ if err != nil {
+ log.Error(ctx, "Error checking library access", "userID", userID, "libraryID", libraryID, err)
+ return fmt.Errorf("error checking library access: %w", err)
+ }
+
+ for _, lib := range libraries {
+ if lib.ID == libraryID {
+ return nil
+ }
+ }
+
+ return fmt.Errorf("%w: user does not have access to library %d", model.ErrNotAuthorized, libraryID)
+}
+
+// REST repository wrapper
+
+func (s *libraryService) NewRepository(ctx context.Context) rest.Repository {
+ repo := s.ds.Library(ctx)
+ wrapper := &libraryRepositoryWrapper{
+ ctx: ctx,
+ LibraryRepository: repo,
+ Repository: repo.(rest.Repository),
+ ds: s.ds,
+ scanner: s.scanner,
+ watcher: s.watcher,
+ broker: s.broker,
+ }
+ return wrapper
+}
+
+type libraryRepositoryWrapper struct {
+ rest.Repository
+ model.LibraryRepository
+ ctx context.Context
+ ds model.DataStore
+ scanner model.Scanner
+ watcher Watcher
+ broker events.Broker
+}
+
+func (r *libraryRepositoryWrapper) Save(entity interface{}) (string, error) {
+ lib := entity.(*model.Library)
+ if err := r.validateLibrary(lib); err != nil {
+ return "", err
+ }
+
+ err := r.LibraryRepository.Put(lib)
+ if err != nil {
+ return "", r.mapError(err)
+ }
+
+ // Start watcher and trigger scan after successful library creation
+ if r.watcher != nil {
+ if err := r.watcher.Watch(r.ctx, lib); err != nil {
+ log.Warn(r.ctx, "Failed to start watcher for new library", "libraryID", lib.ID, "name", lib.Name, "path", lib.Path, err)
+ }
+ }
+
+ if r.scanner != nil {
+ go r.triggerScan(lib, "new")
+ }
+
+ // Send library refresh event to all clients
+ if r.broker != nil {
+ event := &events.RefreshResource{}
+ r.broker.SendBroadcastMessage(r.ctx, event.With("library", strconv.Itoa(lib.ID)))
+ log.Debug(r.ctx, "Library created - sent refresh event", "libraryID", lib.ID, "name", lib.Name)
+ }
+
+ return strconv.Itoa(lib.ID), nil
+}
+
+func (r *libraryRepositoryWrapper) Update(id string, entity interface{}, _ ...string) error {
+ lib := entity.(*model.Library)
+ libID, err := strconv.Atoi(id)
+ if err != nil {
+ return fmt.Errorf("invalid library ID: %s", id)
+ }
+
+ lib.ID = libID
+ if err := r.validateLibrary(lib); err != nil {
+ return err
+ }
+
+ // Get the original library to check if path changed
+ originalLib, err := r.Get(libID)
+ if err != nil {
+ return r.mapError(err)
+ }
+
+ pathChanged := originalLib.Path != lib.Path
+
+ err = r.LibraryRepository.Put(lib)
+ if err != nil {
+ return r.mapError(err)
+ }
+
+ // Restart watcher and trigger scan if path was updated
+ if pathChanged {
+ if r.watcher != nil {
+ if err := r.watcher.Watch(r.ctx, lib); err != nil {
+ log.Warn(r.ctx, "Failed to restart watcher for updated library", "libraryID", lib.ID, "name", lib.Name, "path", lib.Path, err)
+ }
+ }
+
+ if r.scanner != nil {
+ go r.triggerScan(lib, "updated")
+ }
+ }
+
+ // Send library refresh event to all clients
+ if r.broker != nil {
+ event := &events.RefreshResource{}
+ r.broker.SendBroadcastMessage(r.ctx, event.With("library", id))
+ log.Debug(r.ctx, "Library updated - sent refresh event", "libraryID", libID, "name", lib.Name)
+ }
+
+ return nil
+}
+
+func (r *libraryRepositoryWrapper) Delete(id string) error {
+ libID, err := strconv.Atoi(id)
+ if err != nil {
+ return &rest.ValidationError{Errors: map[string]string{
+ "id": "invalid library ID format",
+ }}
+ }
+
+ // Get library info before deletion for logging
+ lib, err := r.Get(libID)
+ if err != nil {
+ return r.mapError(err)
+ }
+
+ err = r.LibraryRepository.Delete(libID)
+ if err != nil {
+ return r.mapError(err)
+ }
+
+ // Stop watcher and trigger scan after successful library deletion to clean up orphaned data
+ if r.watcher != nil {
+ if err := r.watcher.StopWatching(r.ctx, libID); err != nil {
+ log.Warn(r.ctx, "Failed to stop watcher for deleted library", "libraryID", libID, "name", lib.Name, "path", lib.Path, err)
+ }
+ }
+
+ if r.scanner != nil {
+ go r.triggerScan(lib, "deleted")
+ }
+
+ // Send library refresh event to all clients
+ if r.broker != nil {
+ event := &events.RefreshResource{}
+ r.broker.SendBroadcastMessage(r.ctx, event.With("library", id))
+ log.Debug(r.ctx, "Library deleted - sent refresh event", "libraryID", libID, "name", lib.Name)
+ }
+
+ return nil
+}
+
+// Helper methods
+
+func (r *libraryRepositoryWrapper) mapError(err error) error {
+ if err == nil {
+ return nil
+ }
+
+ errStr := err.Error()
+
+ // Handle database constraint violations.
+ // TODO: Being tied to react-admin translations is not ideal, but this will probably go away with the new UI/API
+ if strings.Contains(errStr, "UNIQUE constraint failed") {
+ if strings.Contains(errStr, "library.name") {
+ return &rest.ValidationError{Errors: map[string]string{"name": "ra.validation.unique"}}
+ }
+ if strings.Contains(errStr, "library.path") {
+ return &rest.ValidationError{Errors: map[string]string{"path": "ra.validation.unique"}}
+ }
+ }
+
+ switch {
+ case errors.Is(err, model.ErrNotFound):
+ return rest.ErrNotFound
+ case errors.Is(err, model.ErrNotAuthorized):
+ return rest.ErrPermissionDenied
+ default:
+ return err
+ }
+}
+
+func (r *libraryRepositoryWrapper) validateLibrary(library *model.Library) error {
+ validationErrors := make(map[string]string)
+
+ if library.Name == "" {
+ validationErrors["name"] = "ra.validation.required"
+ }
+
+ if library.Path == "" {
+ validationErrors["path"] = "ra.validation.required"
+ } else {
+ // Validate path format and accessibility
+ if err := r.validateLibraryPath(library); err != nil {
+ validationErrors["path"] = err.Error()
+ }
+ }
+
+ if len(validationErrors) > 0 {
+ return &rest.ValidationError{Errors: validationErrors}
+ }
+
+ return nil
+}
+
+func (r *libraryRepositoryWrapper) validateLibraryPath(library *model.Library) error {
+ // Validate path format
+ if !filepath.IsAbs(library.Path) {
+ return fmt.Errorf("library path must be absolute")
+ }
+
+ // Clean the path to normalize it
+ cleanPath := filepath.Clean(library.Path)
+ library.Path = cleanPath
+
+ // Check if path exists and is accessible using storage abstraction
+ fileStore, err := storage.For(library.Path)
+ if err != nil {
+ return fmt.Errorf("invalid storage scheme: %w", err)
+ }
+
+ fsys, err := fileStore.FS()
+ if err != nil {
+ log.Warn(r.ctx, "Error validating library.path", "path", library.Path, err)
+ return fmt.Errorf("resources.library.validation.pathInvalid")
+ }
+
+ // Check if root directory exists
+ info, err := fs.Stat(fsys, ".")
+ if err != nil {
+ // Parse the error message to check for "not a directory"
+ log.Warn(r.ctx, "Error stating library.path", "path", library.Path, err)
+ errStr := err.Error()
+ if strings.Contains(errStr, "not a directory") ||
+ strings.Contains(errStr, "The directory name is invalid.") {
+ return fmt.Errorf("resources.library.validation.pathNotDirectory")
+ } else if os.IsNotExist(err) {
+ return fmt.Errorf("resources.library.validation.pathNotFound")
+ } else if os.IsPermission(err) {
+ return fmt.Errorf("resources.library.validation.pathNotAccessible")
+ } else {
+ return fmt.Errorf("resources.library.validation.pathInvalid")
+ }
+ }
+
+ if !info.IsDir() {
+ return fmt.Errorf("resources.library.validation.pathNotDirectory")
+ }
+
+ return nil
+}
+
+func (s *libraryService) validateLibraryIDs(ctx context.Context, libraryIDs []int) error {
+ if len(libraryIDs) == 0 {
+ return nil
+ }
+
+ // Use CountAll to efficiently validate library IDs exist
+ count, err := s.ds.Library(ctx).CountAll(model.QueryOptions{
+ Filters: squirrel.Eq{"id": libraryIDs},
+ })
+ if err != nil {
+ return fmt.Errorf("error validating library IDs: %w", err)
+ }
+
+ if int(count) != len(libraryIDs) {
+ return fmt.Errorf("%w: one or more library IDs are invalid", model.ErrValidation)
+ }
+
+ return nil
+}
+
+func (r *libraryRepositoryWrapper) triggerScan(lib *model.Library, action string) {
+ log.Info(r.ctx, fmt.Sprintf("Triggering scan for %s library", action), "libraryID", lib.ID, "name", lib.Name, "path", lib.Path)
+ start := time.Now()
+ warnings, err := r.scanner.ScanAll(r.ctx, false) // Quick scan for new library
+ if err != nil {
+ log.Error(r.ctx, fmt.Sprintf("Error scanning %s library", action), "libraryID", lib.ID, "name", lib.Name, err)
+ } else {
+ log.Info(r.ctx, fmt.Sprintf("Scan completed for %s library", action), "libraryID", lib.ID, "name", lib.Name, "warnings", len(warnings), "elapsed", time.Since(start))
+ }
+}
diff --git a/core/library_test.go b/core/library_test.go
new file mode 100644
index 000000000..bf73a62b7
--- /dev/null
+++ b/core/library_test.go
@@ -0,0 +1,958 @@
+package core_test
+
+import (
+ "context"
+ "errors"
+ "net/http"
+ "os"
+ "path/filepath"
+ "sync"
+
+ "github.com/deluan/rest"
+ _ "github.com/navidrome/navidrome/adapters/taglib" // Register taglib extractor
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/core"
+ _ "github.com/navidrome/navidrome/core/storage/local" // Register local storage
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/server/events"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+// These tests require the local storage adapter and the taglib extractor to be registered.
+var _ = Describe("Library Service", func() {
+ var service core.Library
+ var ds *tests.MockDataStore
+ var libraryRepo *tests.MockLibraryRepo
+ var userRepo *tests.MockedUserRepo
+ var ctx context.Context
+ var tempDir string
+ var scanner *tests.MockScanner
+ var watcherManager *mockWatcherManager
+ var broker *mockEventBroker
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+
+ ds = &tests.MockDataStore{}
+ libraryRepo = &tests.MockLibraryRepo{}
+ userRepo = tests.CreateMockUserRepo()
+ ds.MockedLibrary = libraryRepo
+ ds.MockedUser = userRepo
+
+ // Create a mock scanner that tracks calls
+ scanner = tests.NewMockScanner()
+ // Create a mock watcher manager
+ watcherManager = &mockWatcherManager{
+ libraryStates: make(map[int]model.Library),
+ }
+ // Create a mock event broker
+ broker = &mockEventBroker{}
+ service = core.NewLibrary(ds, scanner, watcherManager, broker)
+ ctx = context.Background()
+
+ // Create a temporary directory for testing valid paths
+ var err error
+ tempDir, err = os.MkdirTemp("", "navidrome-library-test-")
+ Expect(err).NotTo(HaveOccurred())
+ DeferCleanup(func() {
+ os.RemoveAll(tempDir)
+ })
+ })
+
+ Describe("Library CRUD Operations", func() {
+ var repo rest.Persistable
+
+ BeforeEach(func() {
+ r := service.NewRepository(ctx)
+ repo = r.(rest.Persistable)
+ })
+
+ Describe("Create", func() {
+ It("creates a new library successfully", func() {
+ library := &model.Library{ID: 1, Name: "New Library", Path: tempDir}
+
+ _, err := repo.Save(library)
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(libraryRepo.Data[1].Name).To(Equal("New Library"))
+ Expect(libraryRepo.Data[1].Path).To(Equal(tempDir))
+ })
+
+ It("fails when library name is empty", func() {
+ library := &model.Library{Path: tempDir}
+
+ _, err := repo.Save(library)
+
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("ra.validation.required"))
+ })
+
+ It("fails when library path is empty", func() {
+ library := &model.Library{Name: "Test"}
+
+ _, err := repo.Save(library)
+
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("ra.validation.required"))
+ })
+
+ It("fails when library path is not absolute", func() {
+ library := &model.Library{Name: "Test", Path: "relative/path"}
+
+ _, err := repo.Save(library)
+
+ Expect(err).To(HaveOccurred())
+ var validationErr *rest.ValidationError
+ Expect(errors.As(err, &validationErr)).To(BeTrue())
+ Expect(validationErr.Errors["path"]).To(Equal("library path must be absolute"))
+ })
+
+ Context("Database constraint violations", func() {
+ BeforeEach(func() {
+ // Set up an existing library that will cause constraint violations
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Existing Library", Path: tempDir},
+ })
+ })
+
+ AfterEach(func() {
+ // Reset custom PutFn after each test
+ libraryRepo.PutFn = nil
+ })
+
+ It("handles name uniqueness constraint violation from database", func() {
+ // Create the directory that will be used for the test
+ otherTempDir, err := os.MkdirTemp("", "navidrome-other-")
+ Expect(err).NotTo(HaveOccurred())
+ DeferCleanup(func() { os.RemoveAll(otherTempDir) })
+
+ // Try to create another library with the same name
+ library := &model.Library{ID: 2, Name: "Existing Library", Path: otherTempDir}
+
+ // Mock the repository to return a UNIQUE constraint error
+ libraryRepo.PutFn = func(library *model.Library) error {
+ return errors.New("UNIQUE constraint failed: library.name")
+ }
+
+ _, err = repo.Save(library)
+
+ Expect(err).To(HaveOccurred())
+ var validationErr *rest.ValidationError
+ Expect(errors.As(err, &validationErr)).To(BeTrue())
+ Expect(validationErr.Errors["name"]).To(Equal("ra.validation.unique"))
+ })
+
+ It("handles path uniqueness constraint violation from database", func() {
+ // Try to create another library with the same path
+ library := &model.Library{ID: 2, Name: "Different Library", Path: tempDir}
+
+ // Mock the repository to return a UNIQUE constraint error
+ libraryRepo.PutFn = func(library *model.Library) error {
+ return errors.New("UNIQUE constraint failed: library.path")
+ }
+
+ _, err := repo.Save(library)
+
+ Expect(err).To(HaveOccurred())
+ var validationErr *rest.ValidationError
+ Expect(errors.As(err, &validationErr)).To(BeTrue())
+ Expect(validationErr.Errors["path"]).To(Equal("ra.validation.unique"))
+ })
+ })
+ })
+
+ Describe("Update", func() {
+ BeforeEach(func() {
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Original Library", Path: tempDir},
+ })
+ })
+
+ It("updates an existing library successfully", func() {
+ newTempDir, err := os.MkdirTemp("", "navidrome-library-update-")
+ Expect(err).NotTo(HaveOccurred())
+ DeferCleanup(func() { os.RemoveAll(newTempDir) })
+
+ library := &model.Library{ID: 1, Name: "Updated Library", Path: newTempDir}
+
+ err = repo.Update("1", library)
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(libraryRepo.Data[1].Name).To(Equal("Updated Library"))
+ Expect(libraryRepo.Data[1].Path).To(Equal(newTempDir))
+ })
+
+ It("fails when library doesn't exist", func() {
+ // Create a unique temporary directory to avoid path conflicts
+ uniqueTempDir, err := os.MkdirTemp("", "navidrome-nonexistent-")
+ Expect(err).NotTo(HaveOccurred())
+ DeferCleanup(func() { os.RemoveAll(uniqueTempDir) })
+
+ library := &model.Library{ID: 999, Name: "Non-existent", Path: uniqueTempDir}
+
+ err = repo.Update("999", library)
+
+ Expect(err).To(HaveOccurred())
+ Expect(err).To(Equal(model.ErrNotFound))
+ })
+
+ It("fails when library name is empty", func() {
+ library := &model.Library{ID: 1, Path: tempDir}
+
+ err := repo.Update("1", library)
+
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("ra.validation.required"))
+ })
+
+ It("cleans and normalizes the path on update", func() {
+ unnormalizedPath := tempDir + "//../" + filepath.Base(tempDir)
+ library := &model.Library{ID: 1, Name: "Updated Library", Path: unnormalizedPath}
+
+ err := repo.Update("1", library)
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(libraryRepo.Data[1].Path).To(Equal(filepath.Clean(unnormalizedPath)))
+ })
+
+ It("allows updating library with same name (no change)", func() {
+ // Set up a library
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Test Library", Path: tempDir},
+ })
+
+ // Update the library keeping the same name (should be allowed)
+ library := &model.Library{ID: 1, Name: "Test Library", Path: tempDir}
+
+ err := repo.Update("1", library)
+
+ Expect(err).NotTo(HaveOccurred())
+ })
+
+ It("allows updating library with same path (no change)", func() {
+ // Set up a library
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Test Library", Path: tempDir},
+ })
+
+ // Update the library keeping the same path (should be allowed)
+ library := &model.Library{ID: 1, Name: "Test Library", Path: tempDir}
+
+ err := repo.Update("1", library)
+
+ Expect(err).NotTo(HaveOccurred())
+ })
+
+ Context("Database constraint violations during update", func() {
+ BeforeEach(func() {
+ // Reset any custom PutFn from previous tests
+ libraryRepo.PutFn = nil
+ })
+
+ It("handles name uniqueness constraint violation during update", func() {
+ // Create additional temp directory for the test
+ otherTempDir, err := os.MkdirTemp("", "navidrome-other-")
+ Expect(err).NotTo(HaveOccurred())
+ DeferCleanup(func() { os.RemoveAll(otherTempDir) })
+
+ // Set up two libraries
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Library One", Path: tempDir},
+ {ID: 2, Name: "Library Two", Path: otherTempDir},
+ })
+
+ // Mock database constraint violation
+ libraryRepo.PutFn = func(library *model.Library) error {
+ return errors.New("UNIQUE constraint failed: library.name")
+ }
+
+ // Try to update library 2 to have the same name as library 1
+ library := &model.Library{ID: 2, Name: "Library One", Path: otherTempDir}
+
+ err = repo.Update("2", library)
+
+ Expect(err).To(HaveOccurred())
+ var validationErr *rest.ValidationError
+ Expect(errors.As(err, &validationErr)).To(BeTrue())
+ Expect(validationErr.Errors["name"]).To(Equal("ra.validation.unique"))
+ })
+
+ It("handles path uniqueness constraint violation during update", func() {
+ // Create additional temp directory for the test
+ otherTempDir, err := os.MkdirTemp("", "navidrome-other-")
+ Expect(err).NotTo(HaveOccurred())
+ DeferCleanup(func() { os.RemoveAll(otherTempDir) })
+
+ // Set up two libraries
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Library One", Path: tempDir},
+ {ID: 2, Name: "Library Two", Path: otherTempDir},
+ })
+
+ // Mock database constraint violation
+ libraryRepo.PutFn = func(library *model.Library) error {
+ return errors.New("UNIQUE constraint failed: library.path")
+ }
+
+ // Try to update library 2 to have the same path as library 1
+ library := &model.Library{ID: 2, Name: "Library Two", Path: tempDir}
+
+ err = repo.Update("2", library)
+
+ Expect(err).To(HaveOccurred())
+ var validationErr *rest.ValidationError
+ Expect(errors.As(err, &validationErr)).To(BeTrue())
+ Expect(validationErr.Errors["path"]).To(Equal("ra.validation.unique"))
+ })
+ })
+ })
+
+ Describe("Path Validation", func() {
+ Context("Create operation", func() {
+ It("fails when path is not absolute", func() {
+ library := &model.Library{Name: "Test", Path: "relative/path"}
+
+ _, err := repo.Save(library)
+
+ Expect(err).To(HaveOccurred())
+ var validationErr *rest.ValidationError
+ Expect(errors.As(err, &validationErr)).To(BeTrue())
+ Expect(validationErr.Errors["path"]).To(Equal("library path must be absolute"))
+ })
+
+ It("fails when path does not exist", func() {
+ nonExistentPath := filepath.Join(tempDir, "nonexistent")
+ library := &model.Library{Name: "Test", Path: nonExistentPath}
+
+ _, err := repo.Save(library)
+
+ Expect(err).To(HaveOccurred())
+ var validationErr *rest.ValidationError
+ Expect(errors.As(err, &validationErr)).To(BeTrue())
+ Expect(validationErr.Errors["path"]).To(Equal("resources.library.validation.pathInvalid"))
+ })
+
+ It("fails when path is a file instead of directory", func() {
+ testFile := filepath.Join(tempDir, "testfile.txt")
+ err := os.WriteFile(testFile, []byte("test"), 0600)
+ Expect(err).NotTo(HaveOccurred())
+
+ library := &model.Library{Name: "Test", Path: testFile}
+
+ _, err = repo.Save(library)
+
+ Expect(err).To(HaveOccurred())
+ var validationErr *rest.ValidationError
+ Expect(errors.As(err, &validationErr)).To(BeTrue())
+ Expect(validationErr.Errors["path"]).To(Equal("resources.library.validation.pathNotDirectory"))
+ })
+
+ It("fails when path is not accessible due to permissions", func() {
+ Skip("Permission tests are environment-dependent and may fail in CI")
+ // This test is skipped because creating a directory with no read permissions
+ // is complex and may not work consistently across different environments
+ })
+
+ It("handles multiple validation errors", func() {
+ library := &model.Library{Name: "", Path: "relative/path"}
+
+ _, err := repo.Save(library)
+
+ Expect(err).To(HaveOccurred())
+ var validationErr *rest.ValidationError
+ Expect(errors.As(err, &validationErr)).To(BeTrue())
+ Expect(validationErr.Errors).To(HaveKey("name"))
+ Expect(validationErr.Errors).To(HaveKey("path"))
+ Expect(validationErr.Errors["name"]).To(Equal("ra.validation.required"))
+ Expect(validationErr.Errors["path"]).To(Equal("library path must be absolute"))
+ })
+ })
+
+ Context("Update operation", func() {
+ BeforeEach(func() {
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Test Library", Path: tempDir},
+ })
+ })
+
+ It("fails when updated path is not absolute", func() {
+ library := &model.Library{ID: 1, Name: "Test", Path: "relative/path"}
+
+ err := repo.Update("1", library)
+
+ Expect(err).To(HaveOccurred())
+ var validationErr *rest.ValidationError
+ Expect(errors.As(err, &validationErr)).To(BeTrue())
+ Expect(validationErr.Errors["path"]).To(Equal("library path must be absolute"))
+ })
+
+ It("allows updating library with same name (no change)", func() {
+ // Set up a library
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Test Library", Path: tempDir},
+ })
+
+ // Update the library keeping the same name (should be allowed)
+ library := &model.Library{ID: 1, Name: "Test Library", Path: tempDir}
+
+ err := repo.Update("1", library)
+
+ Expect(err).NotTo(HaveOccurred())
+ })
+
+ It("fails when updated path does not exist", func() {
+ nonExistentPath := filepath.Join(tempDir, "nonexistent")
+ library := &model.Library{ID: 1, Name: "Test", Path: nonExistentPath}
+
+ err := repo.Update("1", library)
+
+ Expect(err).To(HaveOccurred())
+ var validationErr *rest.ValidationError
+ Expect(errors.As(err, &validationErr)).To(BeTrue())
+ Expect(validationErr.Errors["path"]).To(Equal("resources.library.validation.pathInvalid"))
+ })
+
+ It("fails when updated path is a file instead of directory", func() {
+ testFile := filepath.Join(tempDir, "updatefile.txt")
+ err := os.WriteFile(testFile, []byte("test"), 0600)
+ Expect(err).NotTo(HaveOccurred())
+
+ library := &model.Library{ID: 1, Name: "Test", Path: testFile}
+
+ err = repo.Update("1", library)
+
+ Expect(err).To(HaveOccurred())
+ var validationErr *rest.ValidationError
+ Expect(errors.As(err, &validationErr)).To(BeTrue())
+ Expect(validationErr.Errors["path"]).To(Equal("resources.library.validation.pathNotDirectory"))
+ })
+
+ It("handles multiple validation errors on update", func() {
+ // Try to update with empty name and invalid path
+ library := &model.Library{ID: 1, Name: "", Path: "relative/path"}
+
+ err := repo.Update("1", library)
+
+ Expect(err).To(HaveOccurred())
+ var validationErr *rest.ValidationError
+ Expect(errors.As(err, &validationErr)).To(BeTrue())
+ Expect(validationErr.Errors).To(HaveKey("name"))
+ Expect(validationErr.Errors).To(HaveKey("path"))
+ Expect(validationErr.Errors["name"]).To(Equal("ra.validation.required"))
+ Expect(validationErr.Errors["path"]).To(Equal("library path must be absolute"))
+ })
+ })
+ })
+
+ Describe("Delete", func() {
+ BeforeEach(func() {
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Library to Delete", Path: tempDir},
+ })
+ })
+
+ It("deletes an existing library successfully", func() {
+ err := repo.Delete("1")
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(libraryRepo.Data).To(HaveLen(0))
+ })
+
+ It("fails when library doesn't exist", func() {
+ err := repo.Delete("999")
+
+ Expect(err).To(HaveOccurred())
+ Expect(err).To(Equal(model.ErrNotFound))
+ })
+ })
+ })
+
+ Describe("User-Library Association Operations", func() {
+ var regularUser, adminUser *model.User
+
+ BeforeEach(func() {
+ regularUser = &model.User{ID: "user1", UserName: "regular", IsAdmin: false}
+ adminUser = &model.User{ID: "admin1", UserName: "admin", IsAdmin: true}
+
+ userRepo.Data = map[string]*model.User{
+ "regular": regularUser,
+ "admin": adminUser,
+ }
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Library 1", Path: "/music1"},
+ {ID: 2, Name: "Library 2", Path: "/music2"},
+ {ID: 3, Name: "Library 3", Path: "/music3"},
+ })
+ })
+
+ Describe("GetUserLibraries", func() {
+ It("returns user's libraries", func() {
+ userRepo.UserLibraries = map[string][]int{
+ "user1": {1},
+ }
+
+ result, err := service.GetUserLibraries(ctx, "user1")
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(result).To(HaveLen(1))
+ Expect(result[0].ID).To(Equal(1))
+ })
+
+ It("fails when user doesn't exist", func() {
+ _, err := service.GetUserLibraries(ctx, "nonexistent")
+
+ Expect(err).To(HaveOccurred())
+ Expect(err).To(Equal(model.ErrNotFound))
+ })
+ })
+
+ Describe("SetUserLibraries", func() {
+ It("sets libraries for regular user successfully", func() {
+ err := service.SetUserLibraries(ctx, "user1", []int{1, 2})
+
+ Expect(err).NotTo(HaveOccurred())
+ libraries := userRepo.UserLibraries["user1"]
+ Expect(libraries).To(HaveLen(2))
+ })
+
+ It("fails when user doesn't exist", func() {
+ err := service.SetUserLibraries(ctx, "nonexistent", []int{1})
+
+ Expect(err).To(HaveOccurred())
+ Expect(err).To(Equal(model.ErrNotFound))
+ })
+
+ It("fails when trying to set libraries for admin user", func() {
+ err := service.SetUserLibraries(ctx, "admin1", []int{1})
+
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("cannot manually assign libraries to admin users"))
+ })
+
+ It("fails when no libraries provided for regular user", func() {
+ err := service.SetUserLibraries(ctx, "user1", []int{})
+
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("at least one library must be assigned to non-admin users"))
+ })
+
+ It("fails when library doesn't exist", func() {
+ err := service.SetUserLibraries(ctx, "user1", []int{999})
+
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("one or more library IDs are invalid"))
+ })
+
+ It("fails when some libraries don't exist", func() {
+ err := service.SetUserLibraries(ctx, "user1", []int{1, 999, 2})
+
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("one or more library IDs are invalid"))
+ })
+ })
+
+ Describe("ValidateLibraryAccess", func() {
+ Context("admin user", func() {
+ BeforeEach(func() {
+ ctx = request.WithUser(ctx, *adminUser)
+ })
+
+ It("allows access to any library", func() {
+ err := service.ValidateLibraryAccess(ctx, "admin1", 1)
+
+ Expect(err).NotTo(HaveOccurred())
+ })
+ })
+
+ Context("regular user", func() {
+ BeforeEach(func() {
+ ctx = request.WithUser(ctx, *regularUser)
+ userRepo.UserLibraries = map[string][]int{
+ "user1": {1},
+ }
+ })
+
+ It("allows access to user's libraries", func() {
+ err := service.ValidateLibraryAccess(ctx, "user1", 1)
+
+ Expect(err).NotTo(HaveOccurred())
+ })
+
+ It("denies access to libraries user doesn't have", func() {
+ err := service.ValidateLibraryAccess(ctx, "user1", 2)
+
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("user does not have access to library 2"))
+ })
+ })
+
+ Context("no user in context", func() {
+ It("fails with user not found error", func() {
+ err := service.ValidateLibraryAccess(ctx, "user1", 1)
+
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("user not found in context"))
+ })
+ })
+ })
+ })
+
+ Describe("Scan Triggering", func() {
+ var repo rest.Persistable
+
+ BeforeEach(func() {
+ r := service.NewRepository(ctx)
+ repo = r.(rest.Persistable)
+ })
+
+ It("triggers scan when creating a new library", func() {
+ library := &model.Library{ID: 1, Name: "New Library", Path: tempDir}
+
+ _, err := repo.Save(library)
+ Expect(err).NotTo(HaveOccurred())
+
+ // Wait briefly for the goroutine to complete
+ Eventually(func() int {
+ return scanner.GetScanAllCallCount()
+ }, "1s", "10ms").Should(Equal(1))
+
+ // Verify scan was called with correct parameters
+ calls := scanner.GetScanAllCalls()
+ Expect(calls[0].FullScan).To(BeFalse()) // Should be quick scan
+ })
+
+ It("triggers scan when updating library path", func() {
+ // First create a library
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Original Library", Path: tempDir},
+ })
+
+ // Create a new temporary directory for the update
+ newTempDir, err := os.MkdirTemp("", "navidrome-library-update-")
+ Expect(err).NotTo(HaveOccurred())
+ DeferCleanup(func() { os.RemoveAll(newTempDir) })
+
+ // Update the library with a new path
+ library := &model.Library{ID: 1, Name: "Updated Library", Path: newTempDir}
+ err = repo.Update("1", library)
+ Expect(err).NotTo(HaveOccurred())
+
+ // Wait briefly for the goroutine to complete
+ Eventually(func() int {
+ return scanner.GetScanAllCallCount()
+ }, "1s", "10ms").Should(Equal(1))
+
+ // Verify scan was called with correct parameters
+ calls := scanner.GetScanAllCalls()
+ Expect(calls[0].FullScan).To(BeFalse()) // Should be quick scan
+ })
+
+ It("does not trigger scan when updating library without path change", func() {
+ // First create a library
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Original Library", Path: tempDir},
+ })
+
+ // Update the library name only (same path)
+ library := &model.Library{ID: 1, Name: "Updated Name", Path: tempDir}
+ err := repo.Update("1", library)
+ Expect(err).NotTo(HaveOccurred())
+
+ // Wait a bit to ensure no scan was triggered
+ Consistently(func() int {
+ return scanner.GetScanAllCallCount()
+ }, "100ms", "10ms").Should(Equal(0))
+ })
+
+ It("does not trigger scan when library creation fails", func() {
+ // Try to create library with invalid data (empty name)
+ library := &model.Library{Path: tempDir}
+
+ _, err := repo.Save(library)
+ Expect(err).To(HaveOccurred())
+
+ // Ensure no scan was triggered since creation failed
+ Consistently(func() int {
+ return scanner.GetScanAllCallCount()
+ }, "100ms", "10ms").Should(Equal(0))
+ })
+
+ It("does not trigger scan when library update fails", func() {
+ // First create a library
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Original Library", Path: tempDir},
+ })
+
+ // Try to update with invalid data (empty name)
+ library := &model.Library{ID: 1, Name: "", Path: tempDir}
+ err := repo.Update("1", library)
+ Expect(err).To(HaveOccurred())
+
+ // Ensure no scan was triggered since update failed
+ Consistently(func() int {
+ return scanner.GetScanAllCallCount()
+ }, "100ms", "10ms").Should(Equal(0))
+ })
+
+ It("triggers scan when deleting a library", func() {
+ // First create a library
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Library to Delete", Path: tempDir},
+ })
+
+ // Delete the library
+ err := repo.Delete("1")
+ Expect(err).NotTo(HaveOccurred())
+
+ // Wait briefly for the goroutine to complete
+ Eventually(func() int {
+ return scanner.GetScanAllCallCount()
+ }, "1s", "10ms").Should(Equal(1))
+
+ // Verify scan was called with correct parameters
+ calls := scanner.GetScanAllCalls()
+ Expect(calls[0].FullScan).To(BeFalse()) // Should be quick scan
+ })
+
+ It("does not trigger scan when library deletion fails", func() {
+ // Try to delete a non-existent library
+ err := repo.Delete("999")
+ Expect(err).To(HaveOccurred())
+
+ // Ensure no scan was triggered since deletion failed
+ Consistently(func() int {
+ return scanner.GetScanAllCallCount()
+ }, "100ms", "10ms").Should(Equal(0))
+ })
+
+ Context("Watcher Integration", func() {
+ It("starts watcher when creating a new library", func() {
+ library := &model.Library{ID: 1, Name: "New Library", Path: tempDir}
+
+ _, err := repo.Save(library)
+ Expect(err).NotTo(HaveOccurred())
+
+ // Verify watcher was started
+ Eventually(func() int {
+ return watcherManager.lenStarted()
+ }, "1s", "10ms").Should(Equal(1))
+
+ Expect(watcherManager.StartedWatchers[0].ID).To(Equal(1))
+ Expect(watcherManager.StartedWatchers[0].Name).To(Equal("New Library"))
+ Expect(watcherManager.StartedWatchers[0].Path).To(Equal(tempDir))
+ })
+
+ It("restarts watcher when library path is updated", func() {
+ // First create a library
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Original Library", Path: tempDir},
+ })
+
+ // Simulate that this library already has a watcher
+ watcherManager.simulateExistingLibrary(model.Library{ID: 1, Name: "Original Library", Path: tempDir})
+
+ // Create a new temp directory for the update
+ newTempDir, err := os.MkdirTemp("", "navidrome-library-update-")
+ Expect(err).NotTo(HaveOccurred())
+ DeferCleanup(func() { os.RemoveAll(newTempDir) })
+
+ // Update library with new path
+ library := &model.Library{ID: 1, Name: "Updated Library", Path: newTempDir}
+ err = repo.Update("1", library)
+ Expect(err).NotTo(HaveOccurred())
+
+ // Verify watcher was restarted
+ Eventually(func() int {
+ return watcherManager.lenRestarted()
+ }, "1s", "10ms").Should(Equal(1))
+
+ Expect(watcherManager.RestartedWatchers[0].ID).To(Equal(1))
+ Expect(watcherManager.RestartedWatchers[0].Path).To(Equal(newTempDir))
+ })
+
+ It("does not restart watcher when only library name is updated", func() {
+ // First create a library
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Original Library", Path: tempDir},
+ })
+
+ // Update library with same path but different name
+ library := &model.Library{ID: 1, Name: "Updated Name", Path: tempDir}
+ err := repo.Update("1", library)
+ Expect(err).NotTo(HaveOccurred())
+
+ // Verify watcher was NOT restarted (since path didn't change)
+ Consistently(func() int {
+ return watcherManager.lenRestarted()
+ }, "100ms", "10ms").Should(Equal(0))
+ })
+
+ It("stops watcher when library is deleted", func() {
+ // Set up a library
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Test Library", Path: tempDir},
+ })
+
+ err := repo.Delete("1")
+ Expect(err).NotTo(HaveOccurred())
+
+ // Verify watcher was stopped
+ Eventually(func() int {
+ return watcherManager.lenStopped()
+ }, "1s", "10ms").Should(Equal(1))
+
+ Expect(watcherManager.StoppedWatchers[0]).To(Equal(1))
+ })
+
+ It("does not stop watcher when library deletion fails", func() {
+ // Set up a library
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Test Library", Path: tempDir},
+ })
+
+ // Mock deletion to fail by trying to delete non-existent library
+ err := repo.Delete("999")
+ Expect(err).To(HaveOccurred())
+
+ // Verify watcher was NOT stopped since deletion failed
+ Consistently(func() int {
+ return watcherManager.lenStopped()
+ }, "100ms", "10ms").Should(Equal(0))
+ })
+ })
+ })
+
+ Describe("Event Broadcasting", func() {
+ var repo rest.Persistable
+
+ BeforeEach(func() {
+ r := service.NewRepository(ctx)
+ repo = r.(rest.Persistable)
+ // Clear any events from broker
+ broker.Events = []events.Event{}
+ })
+
+ It("sends refresh event when creating a library", func() {
+ library := &model.Library{ID: 1, Name: "New Library", Path: tempDir}
+
+ _, err := repo.Save(library)
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(broker.Events).To(HaveLen(1))
+ })
+
+ It("sends refresh event when updating a library", func() {
+ // First create a library
+ libraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Original Library", Path: tempDir},
+ })
+
+ library := &model.Library{ID: 1, Name: "Updated Library", Path: tempDir}
+ err := repo.Update("1", library)
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(broker.Events).To(HaveLen(1))
+ })
+
+ It("sends refresh event when deleting a library", func() {
+ // First create a library
+ libraryRepo.SetData(model.Libraries{
+ {ID: 2, Name: "Library to Delete", Path: tempDir},
+ })
+
+ err := repo.Delete("2")
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(broker.Events).To(HaveLen(1))
+ })
+ })
+})
+
+// mockWatcherManager provides a simple mock implementation of core.Watcher for testing
+type mockWatcherManager struct {
+ StartedWatchers []model.Library
+ StoppedWatchers []int
+ RestartedWatchers []model.Library
+ libraryStates map[int]model.Library // Track which libraries we know about
+ mu sync.RWMutex
+}
+
+func (m *mockWatcherManager) Watch(ctx context.Context, lib *model.Library) error {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+
+ // Check if we already know about this library ID
+ if _, exists := m.libraryStates[lib.ID]; exists {
+ // This is a restart - the library already existed
+ // Update our tracking and record the restart
+ for i, startedLib := range m.StartedWatchers {
+ if startedLib.ID == lib.ID {
+ m.StartedWatchers[i] = *lib
+ break
+ }
+ }
+ m.RestartedWatchers = append(m.RestartedWatchers, *lib)
+ m.libraryStates[lib.ID] = *lib
+ return nil
+ }
+
+ // This is a new library - first time we're seeing it
+ m.StartedWatchers = append(m.StartedWatchers, *lib)
+ m.libraryStates[lib.ID] = *lib
+ return nil
+}
+
+func (m *mockWatcherManager) StopWatching(ctx context.Context, libraryID int) error {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+ m.StoppedWatchers = append(m.StoppedWatchers, libraryID)
+ return nil
+}
+
+func (m *mockWatcherManager) lenStarted() int {
+ m.mu.RLock()
+ defer m.mu.RUnlock()
+ return len(m.StartedWatchers)
+}
+
+func (m *mockWatcherManager) lenStopped() int {
+ m.mu.RLock()
+ defer m.mu.RUnlock()
+ return len(m.StoppedWatchers)
+}
+
+func (m *mockWatcherManager) lenRestarted() int {
+ m.mu.RLock()
+ defer m.mu.RUnlock()
+ return len(m.RestartedWatchers)
+}
+
+// simulateExistingLibrary simulates the scenario where a library already exists
+// and has a watcher running (used by tests to set up the initial state)
+func (m *mockWatcherManager) simulateExistingLibrary(lib model.Library) {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+ m.libraryStates[lib.ID] = lib
+}
+
+// mockEventBroker provides a mock implementation of events.Broker for testing
+type mockEventBroker struct {
+ http.Handler
+ Events []events.Event
+ mu sync.RWMutex
+}
+
+func (m *mockEventBroker) SendMessage(ctx context.Context, event events.Event) {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+ m.Events = append(m.Events, event)
+}
+
+func (m *mockEventBroker) SendBroadcastMessage(ctx context.Context, event events.Event) {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+ m.Events = append(m.Events, event)
+}
diff --git a/core/lyrics/sources.go b/core/lyrics/sources.go
index 6d4a4cc6f..857dc2eef 100644
--- a/core/lyrics/sources.go
+++ b/core/lyrics/sources.go
@@ -8,6 +8,7 @@ import (
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/utils/ioutils"
)
func fromEmbedded(ctx context.Context, mf *model.MediaFile) (model.LyricList, error) {
@@ -27,8 +28,7 @@ func fromExternalFile(ctx context.Context, mf *model.MediaFile, suffix string) (
externalLyric := basePath[0:len(basePath)-len(ext)] + suffix
- contents, err := os.ReadFile(externalLyric)
-
+ contents, err := ioutils.UTF8ReadFile(externalLyric)
if errors.Is(err, os.ErrNotExist) {
log.Trace(ctx, "no lyrics found at path", "path", externalLyric)
return nil, nil
diff --git a/core/lyrics/sources_test.go b/core/lyrics/sources_test.go
index e92564c00..b3d502101 100644
--- a/core/lyrics/sources_test.go
+++ b/core/lyrics/sources_test.go
@@ -108,5 +108,39 @@ var _ = Describe("sources", func() {
},
}))
})
+
+ It("should handle LRC files with UTF-8 BOM marker (issue #4631)", func() {
+ // The function looks for , so we need to pass
+ // a MediaFile with .mp3 path and look for .lrc suffix
+ mf := model.MediaFile{Path: "tests/fixtures/bom-test.mp3"}
+ lyrics, err := fromExternalFile(ctx, &mf, ".lrc")
+
+ Expect(err).To(BeNil())
+ Expect(lyrics).ToNot(BeNil())
+ Expect(lyrics).To(HaveLen(1))
+
+ // The critical assertion: even with BOM, synced should be true
+ Expect(lyrics[0].Synced).To(BeTrue(), "Lyrics with BOM marker should be recognized as synced")
+ Expect(lyrics[0].Line).To(HaveLen(1))
+ Expect(lyrics[0].Line[0].Start).To(Equal(gg.P(int64(0))))
+ Expect(lyrics[0].Line[0].Value).To(ContainSubstring("作曲"))
+ })
+
+ It("should handle UTF-16 LE encoded LRC files", func() {
+ mf := model.MediaFile{Path: "tests/fixtures/bom-utf16-test.mp3"}
+ lyrics, err := fromExternalFile(ctx, &mf, ".lrc")
+
+ Expect(err).To(BeNil())
+ Expect(lyrics).ToNot(BeNil())
+ Expect(lyrics).To(HaveLen(1))
+
+ // UTF-16 should be properly converted to UTF-8
+ Expect(lyrics[0].Synced).To(BeTrue(), "UTF-16 encoded lyrics should be recognized as synced")
+ Expect(lyrics[0].Line).To(HaveLen(2))
+ Expect(lyrics[0].Line[0].Start).To(Equal(gg.P(int64(18800))))
+ Expect(lyrics[0].Line[0].Value).To(Equal("We're no strangers to love"))
+ Expect(lyrics[0].Line[1].Start).To(Equal(gg.P(int64(22801))))
+ Expect(lyrics[0].Line[1].Value).To(Equal("You know the rules and so do I"))
+ })
})
})
diff --git a/core/maintenance.go b/core/maintenance.go
new file mode 100644
index 000000000..750fd3a9e
--- /dev/null
+++ b/core/maintenance.go
@@ -0,0 +1,226 @@
+package core
+
+import (
+ "context"
+ "fmt"
+ "slices"
+ "sync"
+ "time"
+
+ "github.com/Masterminds/squirrel"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/utils/slice"
+)
+
+type Maintenance interface {
+ // DeleteMissingFiles deletes specific missing files by their IDs
+ DeleteMissingFiles(ctx context.Context, ids []string) error
+ // DeleteAllMissingFiles deletes all files marked as missing
+ DeleteAllMissingFiles(ctx context.Context) error
+}
+
+type maintenanceService struct {
+ ds model.DataStore
+ wg sync.WaitGroup
+}
+
+func NewMaintenance(ds model.DataStore) Maintenance {
+ return &maintenanceService{
+ ds: ds,
+ }
+}
+
+func (s *maintenanceService) DeleteMissingFiles(ctx context.Context, ids []string) error {
+ return s.deleteMissing(ctx, ids)
+}
+
+func (s *maintenanceService) DeleteAllMissingFiles(ctx context.Context) error {
+ return s.deleteMissing(ctx, nil)
+}
+
+// deleteMissing handles the deletion of missing files and triggers necessary cleanup operations
+func (s *maintenanceService) deleteMissing(ctx context.Context, ids []string) error {
+ // Track affected album IDs before deletion for refresh
+ affectedAlbumIDs, err := s.getAffectedAlbumIDs(ctx, ids)
+ if err != nil {
+ log.Warn(ctx, "Error tracking affected albums for refresh", err)
+ // Don't fail the operation, just log the warning
+ }
+
+ // Delete missing files within a transaction
+ err = s.ds.WithTx(func(tx model.DataStore) error {
+ if len(ids) == 0 {
+ _, err := tx.MediaFile(ctx).DeleteAllMissing()
+ return err
+ }
+ return tx.MediaFile(ctx).DeleteMissing(ids)
+ })
+ if err != nil {
+ log.Error(ctx, "Error deleting missing tracks from DB", "ids", ids, err)
+ return err
+ }
+
+ // Run garbage collection to clean up orphaned records
+ if err := s.ds.GC(ctx); err != nil {
+ log.Error(ctx, "Error running GC after deleting missing tracks", err)
+ return err
+ }
+
+ // Refresh statistics in background
+ s.refreshStatsAsync(ctx, affectedAlbumIDs)
+
+ return nil
+}
+
+// refreshAlbums recalculates album attributes (size, duration, song count, etc.) from media files.
+// It uses batch queries to minimize database round-trips for efficiency.
+func (s *maintenanceService) refreshAlbums(ctx context.Context, albumIDs []string) error {
+ if len(albumIDs) == 0 {
+ return nil
+ }
+
+ log.Debug(ctx, "Refreshing albums", "count", len(albumIDs))
+
+ // Process in chunks to avoid query size limits
+ const chunkSize = 100
+ for chunk := range slice.CollectChunks(slices.Values(albumIDs), chunkSize) {
+ if err := s.refreshAlbumChunk(ctx, chunk); err != nil {
+ return fmt.Errorf("refreshing album chunk: %w", err)
+ }
+ }
+
+ log.Debug(ctx, "Successfully refreshed albums", "count", len(albumIDs))
+ return nil
+}
+
+// refreshAlbumChunk processes a single chunk of album IDs
+func (s *maintenanceService) refreshAlbumChunk(ctx context.Context, albumIDs []string) error {
+ albumRepo := s.ds.Album(ctx)
+ mfRepo := s.ds.MediaFile(ctx)
+
+ // Batch load existing albums
+ albums, err := albumRepo.GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"album.id": albumIDs},
+ })
+ if err != nil {
+ return fmt.Errorf("loading albums: %w", err)
+ }
+
+ // Create a map for quick lookup
+ albumMap := make(map[string]*model.Album, len(albums))
+ for i := range albums {
+ albumMap[albums[i].ID] = &albums[i]
+ }
+
+ // Batch load all media files for these albums
+ mediaFiles, err := mfRepo.GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"album_id": albumIDs},
+ Sort: "album_id, path",
+ })
+ if err != nil {
+ return fmt.Errorf("loading media files: %w", err)
+ }
+
+ // Group media files by album ID
+ filesByAlbum := make(map[string]model.MediaFiles)
+ for i := range mediaFiles {
+ albumID := mediaFiles[i].AlbumID
+ filesByAlbum[albumID] = append(filesByAlbum[albumID], mediaFiles[i])
+ }
+
+ // Recalculate each album from its media files
+ for albumID, oldAlbum := range albumMap {
+ mfs, hasTracks := filesByAlbum[albumID]
+ if !hasTracks {
+ // Album has no tracks anymore, skip (will be cleaned up by GC)
+ log.Debug(ctx, "Skipping album with no tracks", "albumID", albumID)
+ continue
+ }
+
+ // Recalculate album from media files
+ newAlbum := mfs.ToAlbum()
+
+ // Only update if something changed (avoid unnecessary writes)
+ if !oldAlbum.Equals(newAlbum) {
+ // Preserve original timestamps
+ newAlbum.UpdatedAt = time.Now()
+ newAlbum.CreatedAt = oldAlbum.CreatedAt
+
+ if err := albumRepo.Put(&newAlbum); err != nil {
+ log.Error(ctx, "Error updating album during refresh", "albumID", albumID, err)
+ // Continue with other albums instead of failing entirely
+ continue
+ }
+ log.Trace(ctx, "Refreshed album", "albumID", albumID, "name", newAlbum.Name)
+ }
+ }
+
+ return nil
+}
+
+// getAffectedAlbumIDs returns distinct album IDs from missing media files
+func (s *maintenanceService) getAffectedAlbumIDs(ctx context.Context, ids []string) ([]string, error) {
+ var filters squirrel.Sqlizer = squirrel.Eq{"missing": true}
+ if len(ids) > 0 {
+ filters = squirrel.And{
+ squirrel.Eq{"missing": true},
+ squirrel.Eq{"media_file.id": ids},
+ }
+ }
+
+ mfs, err := s.ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: filters,
+ })
+ if err != nil {
+ return nil, err
+ }
+
+ // Extract unique album IDs
+ albumIDMap := make(map[string]struct{}, len(mfs))
+ for _, mf := range mfs {
+ if mf.AlbumID != "" {
+ albumIDMap[mf.AlbumID] = struct{}{}
+ }
+ }
+
+ albumIDs := make([]string, 0, len(albumIDMap))
+ for id := range albumIDMap {
+ albumIDs = append(albumIDs, id)
+ }
+
+ return albumIDs, nil
+}
+
+// refreshStatsAsync refreshes artist and album statistics in background goroutines
+func (s *maintenanceService) refreshStatsAsync(ctx context.Context, affectedAlbumIDs []string) {
+ // Refresh artist stats in background
+ s.wg.Add(1)
+ go func() {
+ defer s.wg.Done()
+ bgCtx := request.AddValues(context.Background(), ctx)
+ if _, err := s.ds.Artist(bgCtx).RefreshStats(true); err != nil {
+ log.Error(bgCtx, "Error refreshing artist stats after deleting missing files", err)
+ } else {
+ log.Debug(bgCtx, "Successfully refreshed artist stats after deleting missing files")
+ }
+
+ // Refresh album stats in background if we have affected albums
+ if len(affectedAlbumIDs) > 0 {
+ if err := s.refreshAlbums(bgCtx, affectedAlbumIDs); err != nil {
+ log.Error(bgCtx, "Error refreshing album stats after deleting missing files", err)
+ } else {
+ log.Debug(bgCtx, "Successfully refreshed album stats after deleting missing files", "count", len(affectedAlbumIDs))
+ }
+ }
+ }()
+}
+
+// Wait waits for all background goroutines to complete.
+// WARNING: This method is ONLY for testing. Never call this in production code.
+// Calling Wait() in production will block until ALL background operations complete
+// and may cause race conditions with new operations starting.
+func (s *maintenanceService) wait() {
+ s.wg.Wait()
+}
diff --git a/core/maintenance_test.go b/core/maintenance_test.go
new file mode 100644
index 000000000..09b442438
--- /dev/null
+++ b/core/maintenance_test.go
@@ -0,0 +1,364 @@
+package core
+
+import (
+ "context"
+ "errors"
+ "sync"
+
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+ "github.com/sirupsen/logrus"
+)
+
+var _ = Describe("Maintenance", func() {
+ var ds *tests.MockDataStore
+ var mfRepo *extendedMediaFileRepo
+ var service Maintenance
+ var ctx context.Context
+
+ BeforeEach(func() {
+ ctx = context.Background()
+ ctx = request.WithUser(ctx, model.User{ID: "user1", IsAdmin: true})
+
+ ds = createTestDataStore()
+ mfRepo = ds.MockedMediaFile.(*extendedMediaFileRepo)
+ service = NewMaintenance(ds)
+ })
+
+ Describe("DeleteMissingFiles", func() {
+ Context("with specific IDs", func() {
+ It("deletes specific missing files and runs GC", func() {
+ // Setup: mock missing files with album IDs
+ mfRepo.SetData(model.MediaFiles{
+ {ID: "mf1", AlbumID: "album1", Missing: true},
+ {ID: "mf2", AlbumID: "album2", Missing: true},
+ })
+
+ err := service.DeleteMissingFiles(ctx, []string{"mf1", "mf2"})
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(mfRepo.deleteMissingCalled).To(BeTrue())
+ Expect(mfRepo.deletedIDs).To(Equal([]string{"mf1", "mf2"}))
+ Expect(ds.GCCalled).To(BeTrue(), "GC should be called after deletion")
+ })
+
+ It("triggers artist stats refresh and album refresh after deletion", func() {
+ artistRepo := ds.MockedArtist.(*extendedArtistRepo)
+ // Setup: mock missing files with albums
+ albumRepo := ds.MockedAlbum.(*extendedAlbumRepo)
+ albumRepo.SetData(model.Albums{
+ {ID: "album1", Name: "Test Album", SongCount: 5},
+ })
+ mfRepo.SetData(model.MediaFiles{
+ {ID: "mf1", AlbumID: "album1", Missing: true},
+ {ID: "mf2", AlbumID: "album1", Missing: false, Size: 1000, Duration: 180},
+ {ID: "mf3", AlbumID: "album1", Missing: false, Size: 2000, Duration: 200},
+ })
+
+ err := service.DeleteMissingFiles(ctx, []string{"mf1"})
+
+ Expect(err).ToNot(HaveOccurred())
+
+ // Wait for background goroutines to complete
+ service.(*maintenanceService).wait()
+
+ // RefreshStats should be called
+ Expect(artistRepo.IsRefreshStatsCalled()).To(BeTrue(), "Artist stats should be refreshed")
+
+ // Album should be updated with new calculated values
+ Expect(albumRepo.GetPutCallCount()).To(BeNumerically(">", 0), "Album.Put() should be called to refresh album data")
+ })
+
+ It("returns error if deletion fails", func() {
+ mfRepo.deleteMissingError = errors.New("delete failed")
+
+ err := service.DeleteMissingFiles(ctx, []string{"mf1"})
+
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("delete failed"))
+ })
+
+ It("continues even if album tracking fails", func() {
+ mfRepo.SetError(true)
+
+ err := service.DeleteMissingFiles(ctx, []string{"mf1"})
+
+ // Should not fail, just log warning
+ Expect(err).ToNot(HaveOccurred())
+ Expect(mfRepo.deleteMissingCalled).To(BeTrue())
+ })
+
+ It("returns error if GC fails", func() {
+ mfRepo.SetData(model.MediaFiles{
+ {ID: "mf1", AlbumID: "album1", Missing: true},
+ })
+
+ // Set GC to return error
+ ds.GCError = errors.New("gc failed")
+
+ err := service.DeleteMissingFiles(ctx, []string{"mf1"})
+
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("gc failed"))
+ })
+ })
+
+ Context("album ID extraction", func() {
+ It("extracts unique album IDs from missing files", func() {
+ mfRepo.SetData(model.MediaFiles{
+ {ID: "mf1", AlbumID: "album1", Missing: true},
+ {ID: "mf2", AlbumID: "album1", Missing: true},
+ {ID: "mf3", AlbumID: "album2", Missing: true},
+ })
+
+ err := service.DeleteMissingFiles(ctx, []string{"mf1", "mf2", "mf3"})
+
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ It("skips files without album IDs", func() {
+ mfRepo.SetData(model.MediaFiles{
+ {ID: "mf1", AlbumID: "", Missing: true},
+ {ID: "mf2", AlbumID: "album1", Missing: true},
+ })
+
+ err := service.DeleteMissingFiles(ctx, []string{"mf1", "mf2"})
+
+ Expect(err).ToNot(HaveOccurred())
+ })
+ })
+ })
+
+ Describe("DeleteAllMissingFiles", func() {
+ It("deletes all missing files and runs GC", func() {
+ mfRepo.SetData(model.MediaFiles{
+ {ID: "mf1", AlbumID: "album1", Missing: true},
+ {ID: "mf2", AlbumID: "album2", Missing: true},
+ {ID: "mf3", AlbumID: "album3", Missing: true},
+ })
+
+ err := service.DeleteAllMissingFiles(ctx)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ds.GCCalled).To(BeTrue(), "GC should be called after deletion")
+ })
+
+ It("returns error if deletion fails", func() {
+ mfRepo.SetError(true)
+
+ err := service.DeleteAllMissingFiles(ctx)
+
+ Expect(err).To(HaveOccurred())
+ })
+
+ It("handles empty result gracefully", func() {
+ mfRepo.SetData(model.MediaFiles{})
+
+ err := service.DeleteAllMissingFiles(ctx)
+
+ Expect(err).ToNot(HaveOccurred())
+ })
+ })
+
+ Describe("Album refresh logic", func() {
+ var albumRepo *extendedAlbumRepo
+
+ BeforeEach(func() {
+ albumRepo = ds.MockedAlbum.(*extendedAlbumRepo)
+ })
+
+ Context("when album has no tracks after deletion", func() {
+ It("skips the album without updating it", func() {
+ // Setup album with no remaining tracks
+ albumRepo.SetData(model.Albums{
+ {ID: "album1", Name: "Empty Album", SongCount: 1},
+ })
+ mfRepo.SetData(model.MediaFiles{
+ {ID: "mf1", AlbumID: "album1", Missing: true},
+ })
+
+ err := service.DeleteMissingFiles(ctx, []string{"mf1"})
+
+ Expect(err).ToNot(HaveOccurred())
+
+ // Wait for background goroutines to complete
+ service.(*maintenanceService).wait()
+
+ // Album should NOT be updated because it has no tracks left
+ Expect(albumRepo.GetPutCallCount()).To(Equal(0), "Album with no tracks should not be updated")
+ })
+ })
+
+ Context("when Put fails for one album", func() {
+ It("continues processing other albums", func() {
+ albumRepo.SetData(model.Albums{
+ {ID: "album1", Name: "Album 1"},
+ {ID: "album2", Name: "Album 2"},
+ })
+ mfRepo.SetData(model.MediaFiles{
+ {ID: "mf1", AlbumID: "album1", Missing: true},
+ {ID: "mf2", AlbumID: "album1", Missing: false, Size: 1000, Duration: 180},
+ {ID: "mf3", AlbumID: "album2", Missing: true},
+ {ID: "mf4", AlbumID: "album2", Missing: false, Size: 2000, Duration: 200},
+ })
+
+ // Make Put fail on first call but succeed on subsequent calls
+ albumRepo.putError = errors.New("put failed")
+ albumRepo.failOnce = true
+
+ err := service.DeleteMissingFiles(ctx, []string{"mf1", "mf3"})
+
+ // Should not fail even if one album's Put fails
+ Expect(err).ToNot(HaveOccurred())
+
+ // Wait for background goroutines to complete
+ service.(*maintenanceService).wait()
+
+ // Put should have been called multiple times
+ Expect(albumRepo.GetPutCallCount()).To(BeNumerically(">", 0), "Put should be attempted")
+ })
+ })
+
+ Context("when media file loading fails", func() {
+ It("logs warning but continues when tracking affected albums fails", func() {
+ // Set up log capturing
+ hook, cleanup := tests.LogHook()
+ defer cleanup()
+
+ albumRepo.SetData(model.Albums{
+ {ID: "album1", Name: "Album 1"},
+ })
+ mfRepo.SetData(model.MediaFiles{
+ {ID: "mf1", AlbumID: "album1", Missing: true},
+ })
+ // Make GetAll fail when loading media files
+ mfRepo.SetError(true)
+
+ err := service.DeleteMissingFiles(ctx, []string{"mf1"})
+
+ // Deletion should succeed despite the tracking error
+ Expect(err).ToNot(HaveOccurred())
+ Expect(mfRepo.deleteMissingCalled).To(BeTrue())
+
+ // Verify the warning was logged
+ Expect(hook.LastEntry()).ToNot(BeNil())
+ Expect(hook.LastEntry().Level).To(Equal(logrus.WarnLevel))
+ Expect(hook.LastEntry().Message).To(Equal("Error tracking affected albums for refresh"))
+ })
+ })
+ })
+})
+
+// Test helper to create a mock DataStore with controllable behavior
+func createTestDataStore() *tests.MockDataStore {
+ ds := &tests.MockDataStore{}
+
+ // Create extended album repo with Put tracking
+ albumRepo := &extendedAlbumRepo{
+ MockAlbumRepo: tests.CreateMockAlbumRepo(),
+ }
+ ds.MockedAlbum = albumRepo
+
+ // Create extended artist repo with RefreshStats tracking
+ artistRepo := &extendedArtistRepo{
+ MockArtistRepo: tests.CreateMockArtistRepo(),
+ }
+ ds.MockedArtist = artistRepo
+
+ // Create extended media file repo with DeleteMissing support
+ mfRepo := &extendedMediaFileRepo{
+ MockMediaFileRepo: tests.CreateMockMediaFileRepo(),
+ }
+ ds.MockedMediaFile = mfRepo
+
+ return ds
+}
+
+// Extension of MockMediaFileRepo to add DeleteMissing method
+type extendedMediaFileRepo struct {
+ *tests.MockMediaFileRepo
+ deleteMissingCalled bool
+ deletedIDs []string
+ deleteMissingError error
+}
+
+func (m *extendedMediaFileRepo) DeleteMissing(ids []string) error {
+ m.deleteMissingCalled = true
+ m.deletedIDs = ids
+ if m.deleteMissingError != nil {
+ return m.deleteMissingError
+ }
+ // Actually delete from the mock data
+ for _, id := range ids {
+ delete(m.Data, id)
+ }
+ return nil
+}
+
+// Extension of MockAlbumRepo to track Put calls
+type extendedAlbumRepo struct {
+ *tests.MockAlbumRepo
+ mu sync.RWMutex
+ putCallCount int
+ lastPutData *model.Album
+ putError error
+ failOnce bool
+}
+
+func (m *extendedAlbumRepo) Put(album *model.Album) error {
+ m.mu.Lock()
+ m.putCallCount++
+ m.lastPutData = album
+
+ // Handle failOnce behavior
+ var err error
+ if m.putError != nil {
+ if m.failOnce {
+ err = m.putError
+ m.putError = nil // Clear error after first failure
+ m.mu.Unlock()
+ return err
+ }
+ err = m.putError
+ m.mu.Unlock()
+ return err
+ }
+ m.mu.Unlock()
+
+ return m.MockAlbumRepo.Put(album)
+}
+
+func (m *extendedAlbumRepo) GetPutCallCount() int {
+ m.mu.RLock()
+ defer m.mu.RUnlock()
+ return m.putCallCount
+}
+
+// Extension of MockArtistRepo to track RefreshStats calls
+type extendedArtistRepo struct {
+ *tests.MockArtistRepo
+ mu sync.RWMutex
+ refreshStatsCalled bool
+ refreshStatsError error
+}
+
+func (m *extendedArtistRepo) RefreshStats(allArtists bool) (int64, error) {
+ m.mu.Lock()
+ m.refreshStatsCalled = true
+ err := m.refreshStatsError
+ m.mu.Unlock()
+
+ if err != nil {
+ return 0, err
+ }
+ return m.MockArtistRepo.RefreshStats(allArtists)
+}
+
+func (m *extendedArtistRepo) IsRefreshStatsCalled() bool {
+ m.mu.RLock()
+ defer m.mu.RUnlock()
+ return m.refreshStatsCalled
+}
diff --git a/core/metrics/insights.go b/core/metrics/insights.go
index 6076be0a5..010c24c28 100644
--- a/core/metrics/insights.go
+++ b/core/metrics/insights.go
@@ -6,6 +6,7 @@ import (
"encoding/json"
"math"
"net/http"
+ "os"
"path/filepath"
"runtime"
"runtime/debug"
@@ -21,6 +22,7 @@ import (
"github.com/navidrome/navidrome/core/metrics/insights"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/plugins/schema"
"github.com/navidrome/navidrome/utils/singleton"
)
@@ -34,12 +36,18 @@ var (
)
type insightsCollector struct {
- ds model.DataStore
- lastRun atomic.Int64
- lastStatus atomic.Bool
+ ds model.DataStore
+ pluginLoader PluginLoader
+ lastRun atomic.Int64
+ lastStatus atomic.Bool
}
-func GetInstance(ds model.DataStore) Insights {
+// PluginLoader defines an interface for loading plugins
+type PluginLoader interface {
+ PluginList() map[string]schema.PluginManifest
+}
+
+func GetInstance(ds model.DataStore, pluginLoader PluginLoader) Insights {
return singleton.GetInstance(func() *insightsCollector {
id, err := ds.Property(context.TODO()).Get(consts.InsightsIDKey)
if err != nil {
@@ -51,7 +59,7 @@ func GetInstance(ds model.DataStore) Insights {
}
}
insightsID = id
- return &insightsCollector{ds: ds}
+ return &insightsCollector{ds: ds, pluginLoader: pluginLoader}
})
}
@@ -153,6 +161,13 @@ var staticData = sync.OnceValue(func() insights.Data {
data.Build.Settings, data.Build.GoVersion = buildInfo()
data.OS.Containerized = consts.InContainer
+ // Install info
+ packageFilename := filepath.Join(conf.Server.DataFolder, ".package")
+ packageFileData, err := os.ReadFile(packageFilename)
+ if err == nil {
+ data.OS.Package = string(packageFileData)
+ }
+
// OS info
data.OS.Type = runtime.GOOS
data.OS.Arch = runtime.GOARCH
@@ -176,13 +191,15 @@ var staticData = sync.OnceValue(func() insights.Data {
data.Config.DefaultBackgroundURLSet = conf.Server.UILoginBackgroundURL == consts.DefaultUILoginBackgroundURL
data.Config.EnableArtworkPrecache = conf.Server.EnableArtworkPrecache
data.Config.EnableCoverAnimation = conf.Server.EnableCoverAnimation
+ data.Config.EnableNowPlaying = conf.Server.EnableNowPlaying
data.Config.EnableDownloads = conf.Server.EnableDownloads
data.Config.EnableSharing = conf.Server.EnableSharing
data.Config.EnableStarRating = conf.Server.EnableStarRating
- data.Config.EnableLastFM = conf.Server.LastFM.Enabled
+ data.Config.EnableLastFM = conf.Server.LastFM.Enabled && conf.Server.LastFM.ApiKey != "" && conf.Server.LastFM.Secret != ""
+ data.Config.EnableSpotify = conf.Server.Spotify.ID != "" && conf.Server.Spotify.Secret != ""
data.Config.EnableListenBrainz = conf.Server.ListenBrainz.Enabled
+ data.Config.EnableDeezer = conf.Server.Deezer.Enabled
data.Config.EnableMediaFileCoverArt = conf.Server.EnableMediaFileCoverArt
- data.Config.EnableSpotify = conf.Server.Spotify.ID != ""
data.Config.EnableJukebox = conf.Server.Jukebox.Enabled
data.Config.EnablePrometheus = conf.Server.Prometheus.Enabled
data.Config.TranscodingCacheSize = conf.Server.TranscodingCacheSize
@@ -198,6 +215,9 @@ var staticData = sync.OnceValue(func() insights.Data {
data.Config.ScanSchedule = conf.Server.Scanner.Schedule
data.Config.ScanWatcherWait = uint64(math.Trunc(conf.Server.Scanner.WatcherWait.Seconds()))
data.Config.ScanOnStartup = conf.Server.Scanner.ScanOnStartup
+ data.Config.ReverseProxyConfigured = conf.Server.ReverseProxyWhitelist != ""
+ data.Config.HasCustomPID = conf.Server.PID.Track != "" || conf.Server.PID.Album != ""
+ data.Config.HasCustomTags = len(conf.Server.Tags) > 0
return data
})
@@ -232,12 +252,29 @@ func (c *insightsCollector) collect(ctx context.Context) []byte {
if err != nil {
log.Trace(ctx, "Error reading radios count", err)
}
+ data.Library.Libraries, err = c.ds.Library(ctx).CountAll()
+ if err != nil {
+ log.Trace(ctx, "Error reading libraries count", err)
+ }
data.Library.ActiveUsers, err = c.ds.User(ctx).CountAll(model.QueryOptions{
Filters: squirrel.Gt{"last_access_at": time.Now().Add(-7 * 24 * time.Hour)},
})
if err != nil {
log.Trace(ctx, "Error reading active users count", err)
}
+
+ // Check for smart playlists
+ data.Config.HasSmartPlaylists, err = c.hasSmartPlaylists(ctx)
+ if err != nil {
+ log.Trace(ctx, "Error checking for smart playlists", err)
+ }
+
+ // Collect plugins if permitted and enabled
+ if conf.Server.DevEnablePluginsInsights && conf.Server.Plugins.Enabled {
+ data.Plugins = c.collectPlugins(ctx)
+ }
+
+ // Collect active players if permitted
if conf.Server.DevEnablePlayerInsights {
data.Library.ActivePlayers, err = c.ds.Player(ctx).CountByClient(model.QueryOptions{
Filters: squirrel.Gt{"last_seen": time.Now().Add(-7 * 24 * time.Hour)},
@@ -263,3 +300,23 @@ func (c *insightsCollector) collect(ctx context.Context) []byte {
}
return resp
}
+
+// hasSmartPlaylists checks if there are any smart playlists (playlists with rules)
+func (c *insightsCollector) hasSmartPlaylists(ctx context.Context) (bool, error) {
+ count, err := c.ds.Playlist(ctx).CountAll(model.QueryOptions{
+ Filters: squirrel.And{squirrel.NotEq{"rules": ""}, squirrel.NotEq{"rules": nil}},
+ })
+ return count > 0, err
+}
+
+// collectPlugins collects information about installed plugins
+func (c *insightsCollector) collectPlugins(_ context.Context) map[string]insights.PluginInfo {
+ plugins := make(map[string]insights.PluginInfo)
+ for id, manifest := range c.pluginLoader.PluginList() {
+ plugins[id] = insights.PluginInfo{
+ Name: manifest.Name,
+ Version: manifest.Version,
+ }
+ }
+ return plugins
+}
diff --git a/core/metrics/insights/data.go b/core/metrics/insights/data.go
index 9df547b4a..c46eb8743 100644
--- a/core/metrics/insights/data.go
+++ b/core/metrics/insights/data.go
@@ -16,6 +16,7 @@ type Data struct {
Containerized bool `json:"containerized"`
Arch string `json:"arch"`
NumCPU int `json:"numCPU"`
+ Package string `json:"package,omitempty"`
} `json:"os"`
Mem struct {
Alloc uint64 `json:"alloc"`
@@ -36,6 +37,7 @@ type Data struct {
Playlists int64 `json:"playlists"`
Shares int64 `json:"shares"`
Radios int64 `json:"radios"`
+ Libraries int64 `json:"libraries"`
ActiveUsers int64 `json:"activeUsers"`
ActivePlayers map[string]int64 `json:"activePlayers,omitempty"`
} `json:"library"`
@@ -55,11 +57,13 @@ type Data struct {
EnableStarRating bool `json:"enableStarRating,omitempty"`
EnableLastFM bool `json:"enableLastFM,omitempty"`
EnableListenBrainz bool `json:"enableListenBrainz,omitempty"`
+ EnableDeezer bool `json:"enableDeezer,omitempty"`
EnableMediaFileCoverArt bool `json:"enableMediaFileCoverArt,omitempty"`
EnableSpotify bool `json:"enableSpotify,omitempty"`
EnableJukebox bool `json:"enableJukebox,omitempty"`
EnablePrometheus bool `json:"enablePrometheus,omitempty"`
EnableCoverAnimation bool `json:"enableCoverAnimation,omitempty"`
+ EnableNowPlaying bool `json:"enableNowPlaying,omitempty"`
SessionTimeout uint64 `json:"sessionTimeout,omitempty"`
SearchFullString bool `json:"searchFullString,omitempty"`
RecentlyAddedByModTime bool `json:"recentlyAddedByModTime,omitempty"`
@@ -68,7 +72,17 @@ type Data struct {
BackupCount int `json:"backupCount,omitempty"`
DevActivityPanel bool `json:"devActivityPanel,omitempty"`
DefaultBackgroundURLSet bool `json:"defaultBackgroundURL,omitempty"`
+ HasSmartPlaylists bool `json:"hasSmartPlaylists,omitempty"`
+ ReverseProxyConfigured bool `json:"reverseProxyConfigured,omitempty"`
+ HasCustomPID bool `json:"hasCustomPID,omitempty"`
+ HasCustomTags bool `json:"hasCustomTags,omitempty"`
} `json:"config"`
+ Plugins map[string]PluginInfo `json:"plugins,omitempty"`
+}
+
+type PluginInfo struct {
+ Name string `json:"name"`
+ Version string `json:"version"`
}
type FSInfo struct {
diff --git a/core/metrics/prometheus.go b/core/metrics/prometheus.go
index 5dabf29ce..412483156 100644
--- a/core/metrics/prometheus.go
+++ b/core/metrics/prometheus.go
@@ -2,7 +2,6 @@ package metrics
import (
"context"
- "fmt"
"net/http"
"strconv"
"sync"
@@ -13,6 +12,7 @@ import (
"github.com/navidrome/navidrome/consts"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/utils/singleton"
"github.com/prometheus/client_golang/prometheus"
"github.com/prometheus/client_golang/prometheus/promhttp"
)
@@ -20,6 +20,8 @@ import (
type Metrics interface {
WriteInitialMetrics(ctx context.Context)
WriteAfterScanMetrics(ctx context.Context, success bool)
+ RecordRequest(ctx context.Context, endpoint, method, client string, status int32, elapsed int64)
+ RecordPluginRequest(ctx context.Context, plugin, method string, ok bool, elapsed int64)
GetHandler() http.Handler
}
@@ -27,11 +29,14 @@ type metrics struct {
ds model.DataStore
}
-func NewPrometheusInstance(ds model.DataStore) Metrics {
- if conf.Server.Prometheus.Enabled {
- return &metrics{ds: ds}
+func GetPrometheusInstance(ds model.DataStore) Metrics {
+ if !conf.Server.Prometheus.Enabled {
+ return noopMetrics{}
}
- return noopMetrics{}
+
+ return singleton.GetInstance(func() *metrics {
+ return &metrics{ds: ds}
+ })
}
func NewNoopInstance() Metrics {
@@ -51,6 +56,38 @@ func (m *metrics) WriteAfterScanMetrics(ctx context.Context, success bool) {
getPrometheusMetrics().mediaScansCounter.With(scanLabels).Inc()
}
+func (m *metrics) RecordRequest(_ context.Context, endpoint, method, client string, status int32, elapsed int64) {
+ httpLabel := prometheus.Labels{
+ "endpoint": endpoint,
+ "method": method,
+ "client": client,
+ "status": strconv.FormatInt(int64(status), 10),
+ }
+ getPrometheusMetrics().httpRequestCounter.With(httpLabel).Inc()
+
+ httpLatencyLabel := prometheus.Labels{
+ "endpoint": endpoint,
+ "method": method,
+ "client": client,
+ }
+ getPrometheusMetrics().httpRequestDuration.With(httpLatencyLabel).Observe(float64(elapsed))
+}
+
+func (m *metrics) RecordPluginRequest(_ context.Context, plugin, method string, ok bool, elapsed int64) {
+ pluginLabel := prometheus.Labels{
+ "plugin": plugin,
+ "method": method,
+ "ok": strconv.FormatBool(ok),
+ }
+ getPrometheusMetrics().pluginRequestCounter.With(pluginLabel).Inc()
+
+ pluginLatencyLabel := prometheus.Labels{
+ "plugin": plugin,
+ "method": method,
+ }
+ getPrometheusMetrics().pluginRequestDuration.With(pluginLatencyLabel).Observe(float64(elapsed))
+}
+
func (m *metrics) GetHandler() http.Handler {
r := chi.NewRouter()
@@ -59,20 +96,31 @@ func (m *metrics) GetHandler() http.Handler {
consts.PrometheusAuthUser: conf.Server.Prometheus.Password,
}))
}
- r.Handle("/", promhttp.Handler())
+ // Enable created at timestamp to handle zero counter on create.
+ // This requires --enable-feature=created-timestamp-zero-ingestion to be passed in Prometheus
+ r.Handle("/", promhttp.HandlerFor(prometheus.DefaultGatherer, promhttp.HandlerOpts{
+ EnableOpenMetrics: true,
+ EnableOpenMetricsTextCreatedSamples: true,
+ }))
return r
}
type prometheusMetrics struct {
- dbTotal *prometheus.GaugeVec
- versionInfo *prometheus.GaugeVec
- lastMediaScan *prometheus.GaugeVec
- mediaScansCounter *prometheus.CounterVec
+ dbTotal *prometheus.GaugeVec
+ versionInfo *prometheus.GaugeVec
+ lastMediaScan *prometheus.GaugeVec
+ mediaScansCounter *prometheus.CounterVec
+ httpRequestCounter *prometheus.CounterVec
+ httpRequestDuration *prometheus.SummaryVec
+ pluginRequestCounter *prometheus.CounterVec
+ pluginRequestDuration *prometheus.SummaryVec
}
// Prometheus' metrics requires initialization. But not more than once
var getPrometheusMetrics = sync.OnceValue(func() *prometheusMetrics {
+ quartilesToEstimate := map[float64]float64{0.5: 0.05, 0.75: 0.025, 0.9: 0.01, 0.99: 0.001}
+
instance := &prometheusMetrics{
dbTotal: prometheus.NewGaugeVec(
prometheus.GaugeOpts{
@@ -102,23 +150,49 @@ var getPrometheusMetrics = sync.OnceValue(func() *prometheusMetrics {
},
[]string{"success"},
),
+ httpRequestCounter: prometheus.NewCounterVec(
+ prometheus.CounterOpts{
+ Name: "http_request_count",
+ Help: "Request types by status",
+ },
+ []string{"endpoint", "method", "client", "status"},
+ ),
+ httpRequestDuration: prometheus.NewSummaryVec(
+ prometheus.SummaryOpts{
+ Name: "http_request_latency",
+ Help: "Latency (in ms) of HTTP requests",
+ Objectives: quartilesToEstimate,
+ },
+ []string{"endpoint", "method", "client"},
+ ),
+ pluginRequestCounter: prometheus.NewCounterVec(
+ prometheus.CounterOpts{
+ Name: "plugin_request_count",
+ Help: "Plugin requests by method/status",
+ },
+ []string{"plugin", "method", "ok"},
+ ),
+ pluginRequestDuration: prometheus.NewSummaryVec(
+ prometheus.SummaryOpts{
+ Name: "plugin_request_latency",
+ Help: "Latency (in ms) of plugin requests",
+ Objectives: quartilesToEstimate,
+ },
+ []string{"plugin", "method"},
+ ),
}
- err := prometheus.DefaultRegisterer.Register(instance.dbTotal)
- if err != nil {
- log.Fatal("Unable to create Prometheus metric instance", fmt.Errorf("unable to register db_model_totals metrics: %w", err))
- }
- err = prometheus.DefaultRegisterer.Register(instance.versionInfo)
- if err != nil {
- log.Fatal("Unable to create Prometheus metric instance", fmt.Errorf("unable to register navidrome_info metrics: %w", err))
- }
- err = prometheus.DefaultRegisterer.Register(instance.lastMediaScan)
- if err != nil {
- log.Fatal("Unable to create Prometheus metric instance", fmt.Errorf("unable to register media_scan_last metrics: %w", err))
- }
- err = prometheus.DefaultRegisterer.Register(instance.mediaScansCounter)
- if err != nil {
- log.Fatal("Unable to create Prometheus metric instance", fmt.Errorf("unable to register media_scans metrics: %w", err))
- }
+
+ prometheus.DefaultRegisterer.MustRegister(
+ instance.dbTotal,
+ instance.versionInfo,
+ instance.lastMediaScan,
+ instance.mediaScansCounter,
+ instance.httpRequestCounter,
+ instance.httpRequestDuration,
+ instance.pluginRequestCounter,
+ instance.pluginRequestDuration,
+ )
+
return instance
})
@@ -159,4 +233,8 @@ func (n noopMetrics) WriteInitialMetrics(context.Context) {}
func (n noopMetrics) WriteAfterScanMetrics(context.Context, bool) {}
+func (n noopMetrics) RecordRequest(context.Context, string, string, string, int32, int64) {}
+
+func (n noopMetrics) RecordPluginRequest(context.Context, string, string, bool, int64) {}
+
func (n noopMetrics) GetHandler() http.Handler { return nil }
diff --git a/core/mock_library_service.go b/core/mock_library_service.go
new file mode 100644
index 000000000..56f2abd4c
--- /dev/null
+++ b/core/mock_library_service.go
@@ -0,0 +1,46 @@
+package core
+
+import (
+ "context"
+
+ "github.com/deluan/rest"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/tests"
+)
+
+// MockLibraryWrapper provides a simple wrapper around MockLibraryRepo
+// that implements the core.Library interface for testing
+type MockLibraryWrapper struct {
+ *tests.MockLibraryRepo
+}
+
+// MockLibraryRestAdapter adapts MockLibraryRepo to rest.Repository interface
+type MockLibraryRestAdapter struct {
+ *tests.MockLibraryRepo
+}
+
+// NewMockLibraryService creates a new mock library service for testing
+func NewMockLibraryService() Library {
+ repo := &tests.MockLibraryRepo{
+ Data: make(map[int]model.Library),
+ }
+ // Set up default test data
+ repo.SetData(model.Libraries{
+ {ID: 1, Name: "Test Library 1", Path: "/music/library1"},
+ {ID: 2, Name: "Test Library 2", Path: "/music/library2"},
+ })
+ return &MockLibraryWrapper{MockLibraryRepo: repo}
+}
+
+func (m *MockLibraryWrapper) NewRepository(ctx context.Context) rest.Repository {
+ return &MockLibraryRestAdapter{MockLibraryRepo: m.MockLibraryRepo}
+}
+
+// rest.Repository interface implementation
+
+func (a *MockLibraryRestAdapter) Delete(id string) error {
+ return a.DeleteByStringID(id)
+}
+
+var _ Library = (*MockLibraryWrapper)(nil)
+var _ rest.Repository = (*MockLibraryRestAdapter)(nil)
diff --git a/core/playback/mpv/mpv.go b/core/playback/mpv/mpv.go
index 495d27512..f356a1410 100644
--- a/core/playback/mpv/mpv.go
+++ b/core/playback/mpv/mpv.go
@@ -10,11 +10,15 @@ import (
"strings"
"sync"
+ "github.com/kballard/go-shellquote"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/log"
)
func start(ctx context.Context, args []string) (Executor, error) {
+ if len(args) == 0 {
+ return Executor{}, fmt.Errorf("no command arguments provided")
+ }
log.Debug("Executing mpv command", "cmd", args)
j := Executor{args: args}
j.PipeReader, j.out = io.Pipe()
@@ -71,28 +75,32 @@ func (j *Executor) wait() {
// Path will always be an absolute path
func createMPVCommand(deviceName string, filename string, socketName string) []string {
- split := strings.Split(fixCmd(conf.Server.MPVCmdTemplate), " ")
- for i, s := range split {
- s = strings.ReplaceAll(s, "%d", deviceName)
- s = strings.ReplaceAll(s, "%f", filename)
- s = strings.ReplaceAll(s, "%s", socketName)
- split[i] = s
+ // Parse the template structure using shell parsing to handle quoted arguments
+ templateArgs, err := shellquote.Split(conf.Server.MPVCmdTemplate)
+ if err != nil {
+ log.Error("Failed to parse MPV command template", "template", conf.Server.MPVCmdTemplate, err)
+ return nil
}
- return split
-}
-func fixCmd(cmd string) string {
- split := strings.Split(cmd, " ")
- var result []string
- cmdPath, _ := mpvCommand()
- for _, s := range split {
- if s == "mpv" || s == "mpv.exe" {
- result = append(result, cmdPath)
- } else {
- result = append(result, s)
+ // Replace placeholders in each parsed argument to preserve spaces in substituted values
+ for i, arg := range templateArgs {
+ arg = strings.ReplaceAll(arg, "%d", deviceName)
+ arg = strings.ReplaceAll(arg, "%f", filename)
+ arg = strings.ReplaceAll(arg, "%s", socketName)
+ templateArgs[i] = arg
+ }
+
+ // Replace mpv executable references with the configured path
+ if len(templateArgs) > 0 {
+ cmdPath, err := mpvCommand()
+ if err == nil {
+ if templateArgs[0] == "mpv" || templateArgs[0] == "mpv.exe" {
+ templateArgs[0] = cmdPath
+ }
}
}
- return strings.Join(result, " ")
+
+ return templateArgs
}
// This is a 1:1 copy of the stuff in ffmpeg.go, need to be unified.
diff --git a/core/playback/mpv/mpv_suite_test.go b/core/playback/mpv/mpv_suite_test.go
new file mode 100644
index 000000000..f8f827620
--- /dev/null
+++ b/core/playback/mpv/mpv_suite_test.go
@@ -0,0 +1,17 @@
+package mpv
+
+import (
+ "testing"
+
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+func TestMPV(t *testing.T) {
+ tests.Init(t, false)
+ log.SetLevel(log.LevelFatal)
+ RegisterFailHandler(Fail)
+ RunSpecs(t, "MPV Suite")
+}
diff --git a/core/playback/mpv/mpv_test.go b/core/playback/mpv/mpv_test.go
new file mode 100644
index 000000000..20c02501b
--- /dev/null
+++ b/core/playback/mpv/mpv_test.go
@@ -0,0 +1,390 @@
+package mpv
+
+import (
+ "context"
+ "fmt"
+ "io"
+ "os"
+ "path/filepath"
+ "runtime"
+ "strings"
+ "sync"
+ "time"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/model"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("MPV", func() {
+ var (
+ testScript string
+ tempDir string
+ )
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+
+ // Reset MPV cache
+ mpvOnce = sync.Once{}
+ mpvPath = ""
+ mpvErr = nil
+
+ // Create temporary directory for test files
+ var err error
+ tempDir, err = os.MkdirTemp("", "mpv_test_*")
+ Expect(err).ToNot(HaveOccurred())
+ DeferCleanup(func() { os.RemoveAll(tempDir) })
+
+ // Create mock MPV script that outputs arguments to stdout
+ testScript = createMockMPVScript(tempDir)
+
+ // Configure test MPV path
+ conf.Server.MPVPath = testScript
+ })
+
+ Describe("createMPVCommand", func() {
+ Context("with default template", func() {
+ BeforeEach(func() {
+ conf.Server.MPVCmdTemplate = "mpv --audio-device=%d --no-audio-display --pause %f --input-ipc-server=%s"
+ })
+
+ It("creates correct command with simple paths", func() {
+ args := createMPVCommand("auto", "/music/test.mp3", "/tmp/socket")
+ Expect(args).To(Equal([]string{
+ testScript,
+ "--audio-device=auto",
+ "--no-audio-display",
+ "--pause",
+ "/music/test.mp3",
+ "--input-ipc-server=/tmp/socket",
+ }))
+ })
+
+ It("handles paths with spaces", func() {
+ args := createMPVCommand("auto", "/music/My Album/01 - Song.mp3", "/tmp/socket")
+ Expect(args).To(Equal([]string{
+ testScript,
+ "--audio-device=auto",
+ "--no-audio-display",
+ "--pause",
+ "/music/My Album/01 - Song.mp3",
+ "--input-ipc-server=/tmp/socket",
+ }))
+ })
+
+ It("handles complex device names", func() {
+ deviceName := "coreaudio/AppleUSBAudioEngine:Cambridge Audio :Cambridge Audio USB Audio 1.0:0000:1"
+ args := createMPVCommand(deviceName, "/music/test.mp3", "/tmp/socket")
+ Expect(args).To(Equal([]string{
+ testScript,
+ "--audio-device=" + deviceName,
+ "--no-audio-display",
+ "--pause",
+ "/music/test.mp3",
+ "--input-ipc-server=/tmp/socket",
+ }))
+ })
+ })
+
+ Context("with snapcast template (issue #3619)", func() {
+ BeforeEach(func() {
+ // This is the template that fails with naive space splitting
+ conf.Server.MPVCmdTemplate = "mpv --no-audio-display --pause %f --input-ipc-server=%s --audio-channels=stereo --audio-samplerate=48000 --audio-format=s16 --ao=pcm --ao-pcm-file=/audio/snapcast_fifo"
+ })
+
+ It("creates correct command for snapcast integration", func() {
+ args := createMPVCommand("auto", "/music/test.mp3", "/tmp/socket")
+ Expect(args).To(Equal([]string{
+ testScript,
+ "--no-audio-display",
+ "--pause",
+ "/music/test.mp3",
+ "--input-ipc-server=/tmp/socket",
+ "--audio-channels=stereo",
+ "--audio-samplerate=48000",
+ "--audio-format=s16",
+ "--ao=pcm",
+ "--ao-pcm-file=/audio/snapcast_fifo",
+ }))
+ })
+ })
+
+ Context("with wrapper script template", func() {
+ BeforeEach(func() {
+ // Test case that would break with naive splitting due to quoted arguments
+ conf.Server.MPVCmdTemplate = `/tmp/mpv.sh --no-audio-display --pause %f --input-ipc-server=%s --audio-channels=stereo`
+ })
+
+ It("handles wrapper script paths", func() {
+ args := createMPVCommand("auto", "/music/test.mp3", "/tmp/socket")
+ Expect(args).To(Equal([]string{
+ "/tmp/mpv.sh",
+ "--no-audio-display",
+ "--pause",
+ "/music/test.mp3",
+ "--input-ipc-server=/tmp/socket",
+ "--audio-channels=stereo",
+ }))
+ })
+ })
+
+ Context("with extra spaces in template", func() {
+ BeforeEach(func() {
+ conf.Server.MPVCmdTemplate = "mpv --audio-device=%d --no-audio-display --pause %f --input-ipc-server=%s"
+ })
+
+ It("handles extra spaces correctly", func() {
+ args := createMPVCommand("auto", "/music/test.mp3", "/tmp/socket")
+ Expect(args).To(Equal([]string{
+ testScript,
+ "--audio-device=auto",
+ "--no-audio-display",
+ "--pause",
+ "/music/test.mp3",
+ "--input-ipc-server=/tmp/socket",
+ }))
+ })
+ })
+ Context("with paths containing spaces in template arguments", func() {
+ BeforeEach(func() {
+ // Template with spaces in the path arguments themselves
+ conf.Server.MPVCmdTemplate = `mpv --no-audio-display --pause %f --ao-pcm-file="/audio/my folder/snapcast_fifo" --input-ipc-server=%s`
+ })
+
+ It("handles spaces in quoted template argument paths", func() {
+ args := createMPVCommand("auto", "/music/test.mp3", "/tmp/socket")
+ // This test reveals the limitation of strings.Fields() - it will split on all spaces
+ // Expected behavior would be to keep the path as one argument
+ Expect(args).To(Equal([]string{
+ testScript,
+ "--no-audio-display",
+ "--pause",
+ "/music/test.mp3",
+ "--ao-pcm-file=/audio/my folder/snapcast_fifo", // This should be one argument
+ "--input-ipc-server=/tmp/socket",
+ }))
+ })
+ })
+
+ Context("with malformed template", func() {
+ BeforeEach(func() {
+ // Template with unmatched quotes that will cause shell parsing to fail
+ conf.Server.MPVCmdTemplate = `mpv --no-audio-display --pause %f --input-ipc-server=%s --ao-pcm-file="/unclosed/quote`
+ })
+
+ It("returns nil when shell parsing fails", func() {
+ args := createMPVCommand("auto", "/music/test.mp3", "/tmp/socket")
+ Expect(args).To(BeNil())
+ })
+ })
+
+ Context("with empty template", func() {
+ BeforeEach(func() {
+ conf.Server.MPVCmdTemplate = ""
+ })
+
+ It("returns empty slice for empty template", func() {
+ args := createMPVCommand("auto", "/music/test.mp3", "/tmp/socket")
+ Expect(args).To(Equal([]string{}))
+ })
+ })
+ })
+
+ Describe("start", func() {
+ BeforeEach(func() {
+ conf.Server.MPVCmdTemplate = "mpv --audio-device=%d --no-audio-display --pause %f --input-ipc-server=%s"
+ })
+
+ It("executes MPV command and captures arguments correctly", func() {
+ ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
+ defer cancel()
+
+ deviceName := "auto"
+ filename := "/music/test.mp3"
+ socketName := "/tmp/test_socket"
+
+ args := createMPVCommand(deviceName, filename, socketName)
+ executor, err := start(ctx, args)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Read all the output from stdout (this will block until the process finishes or is canceled)
+ output, err := io.ReadAll(executor)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Parse the captured arguments
+ lines := strings.Split(strings.TrimSpace(string(output)), "\n")
+ Expect(lines).To(HaveLen(6))
+ Expect(lines[0]).To(Equal(testScript))
+ Expect(lines[1]).To(Equal("--audio-device=auto"))
+ Expect(lines[2]).To(Equal("--no-audio-display"))
+ Expect(lines[3]).To(Equal("--pause"))
+ Expect(lines[4]).To(Equal("/music/test.mp3"))
+ Expect(lines[5]).To(Equal("--input-ipc-server=/tmp/test_socket"))
+ })
+
+ It("handles file paths with spaces", func() {
+ ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
+ defer cancel()
+
+ deviceName := "auto"
+ filename := "/music/My Album/01 - My Song.mp3"
+ socketName := "/tmp/test socket"
+
+ args := createMPVCommand(deviceName, filename, socketName)
+ executor, err := start(ctx, args)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Read all the output from stdout (this will block until the process finishes or is canceled)
+ output, err := io.ReadAll(executor)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Parse the captured arguments
+ lines := strings.Split(strings.TrimSpace(string(output)), "\n")
+ Expect(lines).To(ContainElement("/music/My Album/01 - My Song.mp3"))
+ Expect(lines).To(ContainElement("--input-ipc-server=/tmp/test socket"))
+ })
+
+ Context("with complex snapcast configuration", func() {
+ BeforeEach(func() {
+ conf.Server.MPVCmdTemplate = "mpv --no-audio-display --pause %f --input-ipc-server=%s --audio-channels=stereo --audio-samplerate=48000 --audio-format=s16 --ao=pcm --ao-pcm-file=/audio/snapcast_fifo"
+ })
+
+ It("passes all snapcast arguments correctly", func() {
+ ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
+ defer cancel()
+
+ deviceName := "auto"
+ filename := "/music/album/track.flac"
+ socketName := "/tmp/mpv-ctrl-test.socket"
+
+ args := createMPVCommand(deviceName, filename, socketName)
+ executor, err := start(ctx, args)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Read all the output from stdout (this will block until the process finishes or is canceled)
+ output, err := io.ReadAll(executor)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Parse the captured arguments
+ lines := strings.Split(strings.TrimSpace(string(output)), "\n")
+
+ // Verify all expected arguments are present
+ Expect(lines).To(ContainElement("--no-audio-display"))
+ Expect(lines).To(ContainElement("--pause"))
+ Expect(lines).To(ContainElement("/music/album/track.flac"))
+ Expect(lines).To(ContainElement("--input-ipc-server=/tmp/mpv-ctrl-test.socket"))
+ Expect(lines).To(ContainElement("--audio-channels=stereo"))
+ Expect(lines).To(ContainElement("--audio-samplerate=48000"))
+ Expect(lines).To(ContainElement("--audio-format=s16"))
+ Expect(lines).To(ContainElement("--ao=pcm"))
+ Expect(lines).To(ContainElement("--ao-pcm-file=/audio/snapcast_fifo"))
+ })
+ })
+
+ Context("with nil args", func() {
+ It("returns error when args is nil", func() {
+ ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
+ defer cancel()
+
+ _, err := start(ctx, nil)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(Equal("no command arguments provided"))
+ })
+
+ It("returns error when args is empty", func() {
+ ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
+ defer cancel()
+
+ _, err := start(ctx, []string{})
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(Equal("no command arguments provided"))
+ })
+ })
+ })
+
+ Describe("mpvCommand", func() {
+ BeforeEach(func() {
+ // Reset the mpv command cache
+ mpvOnce = sync.Once{}
+ mpvPath = ""
+ mpvErr = nil
+ })
+
+ It("finds the configured MPV path", func() {
+ conf.Server.MPVPath = testScript
+ path, err := mpvCommand()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(path).To(Equal(testScript))
+ })
+ })
+
+ Describe("NewTrack integration", func() {
+ var testMediaFile model.MediaFile
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ conf.Server.MPVPath = testScript
+
+ // Create a test media file
+ testMediaFile = model.MediaFile{
+ ID: "test-id",
+ Path: "/music/test.mp3",
+ }
+ })
+
+ Context("with malformed template", func() {
+ BeforeEach(func() {
+ // Template with unmatched quotes that will cause shell parsing to fail
+ conf.Server.MPVCmdTemplate = `mpv --no-audio-display --pause %f --input-ipc-server=%s --ao-pcm-file="/unclosed/quote`
+ })
+
+ It("returns error when createMPVCommand fails", func() {
+ ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
+ defer cancel()
+
+ playbackDone := make(chan bool, 1)
+ _, err := NewTrack(ctx, playbackDone, "auto", testMediaFile)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(Equal("no mpv command arguments provided"))
+ })
+ })
+ })
+})
+
+// createMockMPVScript creates a mock script that outputs arguments to stdout
+func createMockMPVScript(tempDir string) string {
+ var scriptContent string
+ var scriptExt string
+
+ if runtime.GOOS == "windows" {
+ scriptExt = ".bat"
+ scriptContent = `@echo off
+echo %0
+:loop
+if "%~1"=="" goto end
+echo %~1
+shift
+goto loop
+:end
+`
+ } else {
+ scriptExt = ".sh"
+ scriptContent = `#!/bin/sh
+echo "$0"
+for arg in "$@"; do
+ echo "$arg"
+done
+`
+ }
+
+ scriptPath := filepath.Join(tempDir, "mock_mpv"+scriptExt)
+ err := os.WriteFile(scriptPath, []byte(scriptContent), 0755) // nolint:gosec
+ if err != nil {
+ panic(fmt.Sprintf("Failed to create mock script: %v", err))
+ }
+
+ return scriptPath
+}
diff --git a/core/playback/mpv/track.go b/core/playback/mpv/track.go
index b894ff3ad..14170efd4 100644
--- a/core/playback/mpv/track.go
+++ b/core/playback/mpv/track.go
@@ -34,7 +34,10 @@ func NewTrack(ctx context.Context, playbackDoneChannel chan bool, deviceName str
tmpSocketName := socketName("mpv-ctrl-", ".socket")
- args := createMPVCommand(deviceName, mf.Path, tmpSocketName)
+ args := createMPVCommand(deviceName, mf.AbsolutePath(), tmpSocketName)
+ if len(args) == 0 {
+ return nil, fmt.Errorf("no mpv command arguments provided")
+ }
exe, err := start(ctx, args)
if err != nil {
log.Error("Error starting mpv process", err)
diff --git a/core/playlists.go b/core/playlists.go
index 4cdab0d38..f98179f88 100644
--- a/core/playlists.go
+++ b/core/playlists.go
@@ -20,7 +20,9 @@ import (
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/model/criteria"
"github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/utils/ioutils"
"github.com/navidrome/navidrome/utils/slice"
+ "golang.org/x/text/unicode/norm"
)
type Playlists interface {
@@ -96,12 +98,13 @@ func (s *playlists) parsePlaylist(ctx context.Context, playlistFile string, fold
}
defer file.Close()
+ reader := ioutils.UTF8Reader(file)
extension := strings.ToLower(filepath.Ext(playlistFile))
switch extension {
case ".nsp":
- err = s.parseNSP(ctx, pls, file)
+ err = s.parseNSP(ctx, pls, reader)
default:
- err = s.parseM3U(ctx, pls, folder, file)
+ err = s.parseM3U(ctx, pls, folder, reader)
}
return pls, err
}
@@ -203,10 +206,10 @@ func (s *playlists) parseM3U(ctx context.Context, pls *model.Playlist, folder *m
}
existing := make(map[string]int, len(found))
for idx := range found {
- existing[strings.ToLower(found[idx].Path)] = idx
+ existing[normalizePathForComparison(found[idx].Path)] = idx
}
for _, path := range paths {
- idx, ok := existing[strings.ToLower(path)]
+ idx, ok := existing[normalizePathForComparison(path)]
if ok {
mfs = append(mfs, found[idx])
} else {
@@ -223,6 +226,13 @@ func (s *playlists) parseM3U(ctx context.Context, pls *model.Playlist, folder *m
return nil
}
+// normalizePathForComparison normalizes a file path to NFC form and converts to lowercase
+// for consistent comparison. This fixes Unicode normalization issues on macOS where
+// Apple Music creates playlists with NFC-encoded paths but the filesystem uses NFD.
+func normalizePathForComparison(path string) string {
+ return strings.ToLower(norm.NFC.String(path))
+}
+
// TODO This won't work for multiple libraries
func (s *playlists) normalizePaths(ctx context.Context, pls *model.Playlist, folder *model.Folder, lines []string) ([]string, error) {
libRegex, err := s.compileLibraryPaths(ctx)
@@ -326,7 +336,7 @@ func (s *playlists) Update(ctx context.Context, playlistID string,
if needsTrackRefresh {
pls, err = repo.GetWithTracks(playlistID, true, false)
pls.RemoveTracks(idxToRemove)
- pls.AddTracks(idsToAdd)
+ pls.AddMediaFilesByID(idsToAdd)
} else {
if len(idsToAdd) > 0 {
_, err = tracks.Add(idsToAdd)
diff --git a/core/playlists_test.go b/core/playlists_test.go
index 3a3c9aafc..fb42f9c9f 100644
--- a/core/playlists_test.go
+++ b/core/playlists_test.go
@@ -15,6 +15,7 @@ import (
"github.com/navidrome/navidrome/tests"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
+ "golang.org/x/text/unicode/norm"
)
var _ = Describe("Playlists", func() {
@@ -73,6 +74,24 @@ var _ = Describe("Playlists", func() {
Expect(err).ToNot(HaveOccurred())
Expect(pls.Tracks).To(HaveLen(2))
})
+
+ It("parses playlists with UTF-8 BOM marker", func() {
+ pls, err := ps.ImportFile(ctx, folder, "bom-test.m3u")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(pls.OwnerID).To(Equal("123"))
+ Expect(pls.Name).To(Equal("Test Playlist"))
+ Expect(pls.Tracks).To(HaveLen(1))
+ Expect(pls.Tracks[0].Path).To(Equal("tests/fixtures/playlists/test.mp3"))
+ })
+
+ It("parses UTF-16 LE encoded playlists with BOM and converts to UTF-8", func() {
+ pls, err := ps.ImportFile(ctx, folder, "bom-test-utf16.m3u")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(pls.OwnerID).To(Equal("123"))
+ Expect(pls.Name).To(Equal("UTF-16 Test Playlist"))
+ Expect(pls.Tracks).To(HaveLen(1))
+ Expect(pls.Tracks[0].Path).To(Equal("tests/fixtures/playlists/test.mp3"))
+ })
})
Describe("NSP", func() {
@@ -186,6 +205,54 @@ var _ = Describe("Playlists", func() {
Expect(pls.Tracks).To(HaveLen(1))
Expect(pls.Tracks[0].Path).To(Equal("abc/tEsT1.Mp3"))
})
+
+ It("handles Unicode normalization when comparing paths", func() {
+ // Test case for Apple Music playlists that use NFC encoding vs macOS filesystem NFD
+ // The character "è" can be represented as NFC (single codepoint) or NFD (e + combining accent)
+
+ const pathWithAccents = "artist/Michèle Desrosiers/album/Noël.m4a"
+
+ // Simulate a database entry with NFD encoding (as stored by macOS filesystem)
+ nfdPath := norm.NFD.String(pathWithAccents)
+ repo.data = []string{nfdPath}
+
+ // Simulate an Apple Music M3U playlist entry with NFC encoding
+ nfcPath := norm.NFC.String("/music/" + pathWithAccents)
+ m3u := strings.Join([]string{
+ nfcPath,
+ }, "\n")
+ f := strings.NewReader(m3u)
+
+ pls, err := ps.ImportM3U(ctx, f)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(pls.Tracks).To(HaveLen(1), "Should find the track despite Unicode normalization differences")
+ Expect(pls.Tracks[0].Path).To(Equal(nfdPath))
+ })
+ })
+
+ Describe("normalizePathForComparison", func() {
+ It("normalizes Unicode characters to NFC form and converts to lowercase", func() {
+ // Test with NFD (decomposed) input - as would come from macOS filesystem
+ nfdPath := norm.NFD.String("Michèle") // Explicitly convert to NFD form
+ normalized := normalizePathForComparison(nfdPath)
+ Expect(normalized).To(Equal("michèle"))
+
+ // Test with NFC (composed) input - as would come from Apple Music M3U
+ nfcPath := "Michèle" // This might be in NFC form
+ normalizedNfc := normalizePathForComparison(nfcPath)
+
+ // Ensure the two paths are not equal in their original forms
+ Expect(nfdPath).ToNot(Equal(nfcPath))
+
+ // Both should normalize to the same result
+ Expect(normalized).To(Equal(normalizedNfc))
+ })
+
+ It("handles paths with mixed case and Unicode characters", func() {
+ path := "Artist/Noël Coward/Album/Song.mp3"
+ normalized := normalizePathForComparison(path)
+ Expect(normalized).To(Equal("artist/noël coward/album/song.mp3"))
+ })
})
Describe("InPlaylistsPath", func() {
diff --git a/core/scrobbler/buffered_scrobbler.go b/core/scrobbler/buffered_scrobbler.go
index 047e43eef..4f64a3c2b 100644
--- a/core/scrobbler/buffered_scrobbler.go
+++ b/core/scrobbler/buffered_scrobbler.go
@@ -10,9 +10,16 @@ import (
)
func newBufferedScrobbler(ds model.DataStore, s Scrobbler, service string) *bufferedScrobbler {
- b := &bufferedScrobbler{ds: ds, wrapped: s, service: service}
- b.wakeSignal = make(chan struct{}, 1)
- go b.run(context.TODO())
+ ctx, cancel := context.WithCancel(context.Background())
+ b := &bufferedScrobbler{
+ ds: ds,
+ wrapped: s,
+ service: service,
+ wakeSignal: make(chan struct{}, 1),
+ ctx: ctx,
+ cancel: cancel,
+ }
+ go b.run(ctx)
return b
}
@@ -21,14 +28,22 @@ type bufferedScrobbler struct {
wrapped Scrobbler
service string
wakeSignal chan struct{}
+ ctx context.Context
+ cancel context.CancelFunc
+}
+
+func (b *bufferedScrobbler) Stop() {
+ if b.cancel != nil {
+ b.cancel()
+ }
}
func (b *bufferedScrobbler) IsAuthorized(ctx context.Context, userId string) bool {
return b.wrapped.IsAuthorized(ctx, userId)
}
-func (b *bufferedScrobbler) NowPlaying(ctx context.Context, userId string, track *model.MediaFile) error {
- return b.wrapped.NowPlaying(ctx, userId, track)
+func (b *bufferedScrobbler) NowPlaying(ctx context.Context, userId string, track *model.MediaFile, position int) error {
+ return b.wrapped.NowPlaying(ctx, userId, track, position)
}
func (b *bufferedScrobbler) Scrobble(ctx context.Context, userId string, s Scrobble) error {
diff --git a/core/scrobbler/buffered_scrobbler_test.go b/core/scrobbler/buffered_scrobbler_test.go
new file mode 100644
index 000000000..c1440046d
--- /dev/null
+++ b/core/scrobbler/buffered_scrobbler_test.go
@@ -0,0 +1,88 @@
+package scrobbler
+
+import (
+ "context"
+ "time"
+
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("BufferedScrobbler", func() {
+ var ds model.DataStore
+ var scr *fakeScrobbler
+ var bs *bufferedScrobbler
+ var ctx context.Context
+ var buffer *tests.MockedScrobbleBufferRepo
+
+ BeforeEach(func() {
+ ctx = context.Background()
+ buffer = tests.CreateMockedScrobbleBufferRepo()
+ ds = &tests.MockDataStore{
+ MockedScrobbleBuffer: buffer,
+ }
+ scr = &fakeScrobbler{Authorized: true}
+ bs = newBufferedScrobbler(ds, scr, "test")
+ })
+
+ It("forwards IsAuthorized calls", func() {
+ scr.Authorized = true
+ Expect(bs.IsAuthorized(ctx, "user1")).To(BeTrue())
+
+ scr.Authorized = false
+ Expect(bs.IsAuthorized(ctx, "user1")).To(BeFalse())
+ })
+
+ It("forwards NowPlaying calls", func() {
+ track := &model.MediaFile{ID: "123", Title: "Test Track"}
+ Expect(bs.NowPlaying(ctx, "user1", track, 0)).To(Succeed())
+ Expect(scr.NowPlayingCalled).To(BeTrue())
+ Expect(scr.UserID).To(Equal("user1"))
+ Expect(scr.Track).To(Equal(track))
+ })
+
+ It("enqueues scrobbles to buffer", func() {
+ track := model.MediaFile{ID: "123", Title: "Test Track"}
+ now := time.Now()
+ scrobble := Scrobble{MediaFile: track, TimeStamp: now}
+ Expect(buffer.Length()).To(Equal(int64(0)))
+ Expect(scr.ScrobbleCalled.Load()).To(BeFalse())
+
+ Expect(bs.Scrobble(ctx, "user1", scrobble)).To(Succeed())
+ Expect(buffer.Length()).To(Equal(int64(1)))
+
+ // Wait for the scrobble to be sent
+ Eventually(scr.ScrobbleCalled.Load).Should(BeTrue())
+
+ lastScrobble := scr.LastScrobble.Load()
+ Expect(lastScrobble.MediaFile.ID).To(Equal("123"))
+ Expect(lastScrobble.TimeStamp).To(BeTemporally("==", now))
+ })
+
+ It("stops the background goroutine when Stop is called", func() {
+ // Replace the real run method with one that signals when it exits
+ done := make(chan struct{})
+
+ // Start our instrumented run function that will signal when it exits
+ go func() {
+ defer close(done)
+ bs.run(bs.ctx)
+ }()
+
+ // Wait a bit to ensure the goroutine is running
+ time.Sleep(10 * time.Millisecond)
+
+ // Call the real Stop method
+ bs.Stop()
+
+ // Wait for the goroutine to exit or timeout
+ select {
+ case <-done:
+ // Success, goroutine exited
+ case <-time.After(100 * time.Millisecond):
+ Fail("Goroutine did not exit in time after Stop was called")
+ }
+ })
+})
diff --git a/core/scrobbler/interfaces.go b/core/scrobbler/interfaces.go
index 90141f112..f8567e91b 100644
--- a/core/scrobbler/interfaces.go
+++ b/core/scrobbler/interfaces.go
@@ -21,7 +21,7 @@ var (
type Scrobbler interface {
IsAuthorized(ctx context.Context, userId string) bool
- NowPlaying(ctx context.Context, userId string, track *model.MediaFile) error
+ NowPlaying(ctx context.Context, userId string, track *model.MediaFile, position int) error
Scrobble(ctx context.Context, userId string, s Scrobble) error
}
diff --git a/core/scrobbler/play_tracker.go b/core/scrobbler/play_tracker.go
index 53f397647..3b71a2100 100644
--- a/core/scrobbler/play_tracker.go
+++ b/core/scrobbler/play_tracker.go
@@ -2,9 +2,12 @@ package scrobbler
import (
"context"
+ "maps"
"sort"
+ "sync"
"time"
+ "github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/consts"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
@@ -17,6 +20,7 @@ import (
type NowPlayingInfo struct {
MediaFile model.MediaFile
Start time.Time
+ Position int
Username string
PlayerId string
PlayerName string
@@ -28,30 +32,52 @@ type Submission struct {
}
type PlayTracker interface {
- NowPlaying(ctx context.Context, playerId string, playerName string, trackId string) error
+ NowPlaying(ctx context.Context, playerId string, playerName string, trackId string, position int) error
GetNowPlaying(ctx context.Context) ([]NowPlayingInfo, error)
Submit(ctx context.Context, submissions []Submission) error
}
-type playTracker struct {
- ds model.DataStore
- broker events.Broker
- playMap cache.SimpleCache[string, NowPlayingInfo]
- scrobblers map[string]Scrobbler
+// PluginLoader is a minimal interface for plugin manager usage in PlayTracker
+// (avoids import cycles)
+type PluginLoader interface {
+ PluginNames(capability string) []string
+ LoadScrobbler(name string) (Scrobbler, bool)
}
-func GetPlayTracker(ds model.DataStore, broker events.Broker) PlayTracker {
+type playTracker struct {
+ ds model.DataStore
+ broker events.Broker
+ playMap cache.SimpleCache[string, NowPlayingInfo]
+ builtinScrobblers map[string]Scrobbler
+ pluginScrobblers map[string]Scrobbler
+ pluginLoader PluginLoader
+ mu sync.RWMutex
+}
+
+func GetPlayTracker(ds model.DataStore, broker events.Broker, pluginManager PluginLoader) PlayTracker {
return singleton.GetInstance(func() *playTracker {
- return newPlayTracker(ds, broker)
+ return newPlayTracker(ds, broker, pluginManager)
})
}
// This constructor only exists for testing. For normal usage, the PlayTracker has to be a singleton, returned by
// the GetPlayTracker function above
-func newPlayTracker(ds model.DataStore, broker events.Broker) *playTracker {
+func newPlayTracker(ds model.DataStore, broker events.Broker, pluginManager PluginLoader) *playTracker {
m := cache.NewSimpleCache[string, NowPlayingInfo]()
- p := &playTracker{ds: ds, playMap: m, broker: broker}
- p.scrobblers = make(map[string]Scrobbler)
+ p := &playTracker{
+ ds: ds,
+ playMap: m,
+ broker: broker,
+ builtinScrobblers: make(map[string]Scrobbler),
+ pluginScrobblers: make(map[string]Scrobbler),
+ pluginLoader: pluginManager,
+ }
+ if conf.Server.EnableNowPlaying {
+ m.OnExpiration(func(_ string, _ NowPlayingInfo) {
+ broker.SendBroadcastMessage(context.Background(), &events.NowPlayingCount{Count: m.Len()})
+ })
+ }
+
var enabled []string
for name, constructor := range constructors {
s := constructor(ds)
@@ -61,13 +87,87 @@ func newPlayTracker(ds model.DataStore, broker events.Broker) *playTracker {
}
enabled = append(enabled, name)
s = newBufferedScrobbler(ds, s, name)
- p.scrobblers[name] = s
+ p.builtinScrobblers[name] = s
}
- log.Debug("List of scrobblers enabled", "names", enabled)
+ log.Debug("List of builtin scrobblers enabled", "names", enabled)
return p
}
-func (p *playTracker) NowPlaying(ctx context.Context, playerId string, playerName string, trackId string) error {
+// pluginNamesMatchScrobblers returns true if the set of pluginNames matches the keys in pluginScrobblers
+func pluginNamesMatchScrobblers(pluginNames []string, scrobblers map[string]Scrobbler) bool {
+ if len(pluginNames) != len(scrobblers) {
+ return false
+ }
+ for _, name := range pluginNames {
+ if _, ok := scrobblers[name]; !ok {
+ return false
+ }
+ }
+ return true
+}
+
+// refreshPluginScrobblers updates the pluginScrobblers map to match the current set of plugin scrobblers
+func (p *playTracker) refreshPluginScrobblers() {
+ p.mu.Lock()
+ defer p.mu.Unlock()
+ if p.pluginLoader == nil {
+ return
+ }
+
+ // Get the list of available plugin names
+ pluginNames := p.pluginLoader.PluginNames("Scrobbler")
+
+ // Early return if plugin names match existing scrobblers (no change)
+ if pluginNamesMatchScrobblers(pluginNames, p.pluginScrobblers) {
+ return
+ }
+
+ // Build a set of current plugins for faster lookups
+ current := make(map[string]struct{}, len(pluginNames))
+
+ // Process additions - add new plugins
+ for _, name := range pluginNames {
+ current[name] = struct{}{}
+ // Only create a new scrobbler if it doesn't exist
+ if _, exists := p.pluginScrobblers[name]; !exists {
+ s, ok := p.pluginLoader.LoadScrobbler(name)
+ if ok && s != nil {
+ p.pluginScrobblers[name] = newBufferedScrobbler(p.ds, s, name)
+ }
+ }
+ }
+
+ type stoppableScrobbler interface {
+ Scrobbler
+ Stop()
+ }
+
+ // Process removals - remove plugins that no longer exist
+ for name, scrobbler := range p.pluginScrobblers {
+ if _, exists := current[name]; !exists {
+ // If the scrobbler implements stoppableScrobbler, call Stop() before removing it
+ if stoppable, ok := scrobbler.(stoppableScrobbler); ok {
+ log.Debug("Stopping scrobbler", "name", name)
+ stoppable.Stop()
+ }
+ delete(p.pluginScrobblers, name)
+ }
+ }
+}
+
+// getActiveScrobblers refreshes plugin scrobblers, acquires a read lock,
+// combines builtin and plugin scrobblers into a new map, releases the lock,
+// and returns the combined map.
+func (p *playTracker) getActiveScrobblers() map[string]Scrobbler {
+ p.refreshPluginScrobblers()
+ p.mu.RLock()
+ defer p.mu.RUnlock()
+ combined := maps.Clone(p.builtinScrobblers)
+ maps.Copy(combined, p.pluginScrobblers)
+ return combined
+}
+
+func (p *playTracker) NowPlaying(ctx context.Context, playerId string, playerName string, trackId string, position int) error {
mf, err := p.ds.MediaFile(ctx).GetWithParticipants(trackId)
if err != nil {
log.Error(ctx, "Error retrieving mediaFile", "id", trackId, err)
@@ -78,31 +178,43 @@ func (p *playTracker) NowPlaying(ctx context.Context, playerId string, playerNam
info := NowPlayingInfo{
MediaFile: *mf,
Start: time.Now(),
+ Position: position,
Username: user.UserName,
PlayerId: playerId,
PlayerName: playerName,
}
- ttl := time.Duration(int(mf.Duration)+5) * time.Second
+ // Calculate TTL based on remaining track duration. If position exceeds track duration,
+ // remaining is set to 0 to avoid negative TTL.
+ remaining := int(mf.Duration) - position
+ if remaining < 0 {
+ remaining = 0
+ }
+ // Add 5 seconds buffer to ensure the NowPlaying info is available slightly longer than the track duration.
+ ttl := time.Duration(remaining+5) * time.Second
_ = p.playMap.AddWithTTL(playerId, info, ttl)
+ if conf.Server.EnableNowPlaying {
+ p.broker.SendBroadcastMessage(ctx, &events.NowPlayingCount{Count: p.playMap.Len()})
+ }
player, _ := request.PlayerFrom(ctx)
if player.ScrobbleEnabled {
- p.dispatchNowPlaying(ctx, user.ID, mf)
+ p.dispatchNowPlaying(ctx, user.ID, mf, position)
}
return nil
}
-func (p *playTracker) dispatchNowPlaying(ctx context.Context, userId string, t *model.MediaFile) {
+func (p *playTracker) dispatchNowPlaying(ctx context.Context, userId string, t *model.MediaFile, position int) {
if t.Artist == consts.UnknownArtist {
log.Debug(ctx, "Ignoring external NowPlaying update for track with unknown artist", "track", t.Title, "artist", t.Artist)
return
}
- for name, s := range p.scrobblers {
+ allScrobblers := p.getActiveScrobblers()
+ for name, s := range allScrobblers {
if !s.IsAuthorized(ctx, userId) {
continue
}
- log.Debug(ctx, "Sending NowPlaying update", "scrobbler", name, "track", t.Title, "artist", t.Artist)
- err := s.NowPlaying(ctx, userId, t)
+ log.Debug(ctx, "Sending NowPlaying update", "scrobbler", name, "track", t.Title, "artist", t.Artist, "position", position)
+ err := s.NowPlaying(ctx, userId, t, position)
if err != nil {
log.Error(ctx, "Error sending NowPlayingInfo", "scrobbler", name, "track", t.Title, "artist", t.Artist, err)
continue
@@ -174,9 +286,11 @@ func (p *playTracker) dispatchScrobble(ctx context.Context, t *model.MediaFile,
log.Debug(ctx, "Ignoring external Scrobble for track with unknown artist", "track", t.Title, "artist", t.Artist)
return
}
+
+ allScrobblers := p.getActiveScrobblers()
u, _ := request.UserFrom(ctx)
scrobble := Scrobble{MediaFile: *t, TimeStamp: playTime}
- for name, s := range p.scrobblers {
+ for name, s := range allScrobblers {
if !s.IsAuthorized(ctx, u.ID) {
continue
}
diff --git a/core/scrobbler/play_tracker_test.go b/core/scrobbler/play_tracker_test.go
index 0ff025f15..7b4785bb5 100644
--- a/core/scrobbler/play_tracker_test.go
+++ b/core/scrobbler/play_tracker_test.go
@@ -3,8 +3,13 @@ package scrobbler
import (
"context"
"errors"
+ "net/http"
+ "sync"
+ "sync/atomic"
"time"
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
"github.com/navidrome/navidrome/consts"
"github.com/navidrome/navidrome/model"
@@ -15,10 +20,28 @@ import (
. "github.com/onsi/gomega"
)
+// mockPluginLoader is a test implementation of PluginLoader for plugin scrobbler tests
+// Moved to top-level scope to avoid linter issues
+
+type mockPluginLoader struct {
+ names []string
+ scrobblers map[string]Scrobbler
+}
+
+func (m *mockPluginLoader) PluginNames(service string) []string {
+ return m.names
+}
+
+func (m *mockPluginLoader) LoadScrobbler(name string) (Scrobbler, bool) {
+ s, ok := m.scrobblers[name]
+ return s, ok
+}
+
var _ = Describe("PlayTracker", func() {
var ctx context.Context
var ds model.DataStore
var tracker PlayTracker
+ var eventBroker *fakeEventBroker
var track model.MediaFile
var album model.Album
var artist1 model.Artist
@@ -26,6 +49,7 @@ var _ = Describe("PlayTracker", func() {
var fake fakeScrobbler
BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
ctx = context.Background()
ctx = request.WithUser(ctx, model.User{ID: "u-1"})
ctx = request.WithPlayer(ctx, model.Player{ScrobbleEnabled: true})
@@ -37,8 +61,9 @@ var _ = Describe("PlayTracker", func() {
Register("disabled", func(model.DataStore) Scrobbler {
return nil
})
- tracker = newPlayTracker(ds, events.GetBroker())
- tracker.(*playTracker).scrobblers["fake"] = &fake // Bypass buffering for tests
+ eventBroker = &fakeEventBroker{}
+ tracker = newPlayTracker(ds, eventBroker, nil)
+ tracker.(*playTracker).builtinScrobblers["fake"] = &fake // Bypass buffering for tests
track = model.MediaFile{
ID: "123",
@@ -62,13 +87,13 @@ var _ = Describe("PlayTracker", func() {
})
It("does not register disabled scrobblers", func() {
- Expect(tracker.(*playTracker).scrobblers).To(HaveKey("fake"))
- Expect(tracker.(*playTracker).scrobblers).ToNot(HaveKey("disabled"))
+ Expect(tracker.(*playTracker).builtinScrobblers).To(HaveKey("fake"))
+ Expect(tracker.(*playTracker).builtinScrobblers).ToNot(HaveKey("disabled"))
})
Describe("NowPlaying", func() {
It("sends track to agent", func() {
- err := tracker.NowPlaying(ctx, "player-1", "player-one", "123")
+ err := tracker.NowPlaying(ctx, "player-1", "player-one", "123", 0)
Expect(err).ToNot(HaveOccurred())
Expect(fake.NowPlayingCalled).To(BeTrue())
Expect(fake.UserID).To(Equal("u-1"))
@@ -78,7 +103,7 @@ var _ = Describe("PlayTracker", func() {
It("does not send track to agent if user has not authorized", func() {
fake.Authorized = false
- err := tracker.NowPlaying(ctx, "player-1", "player-one", "123")
+ err := tracker.NowPlaying(ctx, "player-1", "player-one", "123", 0)
Expect(err).ToNot(HaveOccurred())
Expect(fake.NowPlayingCalled).To(BeFalse())
@@ -86,7 +111,7 @@ var _ = Describe("PlayTracker", func() {
It("does not send track to agent if player is not enabled to send scrobbles", func() {
ctx = request.WithPlayer(ctx, model.Player{ScrobbleEnabled: false})
- err := tracker.NowPlaying(ctx, "player-1", "player-one", "123")
+ err := tracker.NowPlaying(ctx, "player-1", "player-one", "123", 0)
Expect(err).ToNot(HaveOccurred())
Expect(fake.NowPlayingCalled).To(BeFalse())
@@ -94,11 +119,40 @@ var _ = Describe("PlayTracker", func() {
It("does not send track to agent if artist is unknown", func() {
track.Artist = consts.UnknownArtist
- err := tracker.NowPlaying(ctx, "player-1", "player-one", "123")
+ err := tracker.NowPlaying(ctx, "player-1", "player-one", "123", 0)
Expect(err).ToNot(HaveOccurred())
Expect(fake.NowPlayingCalled).To(BeFalse())
})
+
+ It("stores position when greater than zero", func() {
+ pos := 42
+ err := tracker.NowPlaying(ctx, "player-1", "player-one", "123", pos)
+ Expect(err).ToNot(HaveOccurred())
+
+ playing, err := tracker.GetNowPlaying(ctx)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(playing).To(HaveLen(1))
+ Expect(playing[0].Position).To(Equal(pos))
+ Expect(fake.Position).To(Equal(pos))
+ })
+
+ It("sends event with count", func() {
+ err := tracker.NowPlaying(ctx, "player-1", "player-one", "123", 0)
+ Expect(err).ToNot(HaveOccurred())
+ eventList := eventBroker.getEvents()
+ Expect(eventList).ToNot(BeEmpty())
+ evt, ok := eventList[0].(*events.NowPlayingCount)
+ Expect(ok).To(BeTrue())
+ Expect(evt.Count).To(Equal(1))
+ })
+
+ It("does not send event when disabled", func() {
+ conf.Server.EnableNowPlaying = false
+ err := tracker.NowPlaying(ctx, "player-1", "player-one", "123", 0)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(eventBroker.getEvents()).To(BeEmpty())
+ })
})
Describe("GetNowPlaying", func() {
@@ -107,9 +161,9 @@ var _ = Describe("PlayTracker", func() {
track2.ID = "456"
_ = ds.MediaFile(ctx).Put(&track2)
ctx = request.WithUser(context.Background(), model.User{UserName: "user-1"})
- _ = tracker.NowPlaying(ctx, "player-1", "player-one", "123")
+ _ = tracker.NowPlaying(ctx, "player-1", "player-one", "123", 0)
ctx = request.WithUser(context.Background(), model.User{UserName: "user-2"})
- _ = tracker.NowPlaying(ctx, "player-2", "player-two", "456")
+ _ = tracker.NowPlaying(ctx, "player-2", "player-two", "456", 0)
playing, err := tracker.GetNowPlaying(ctx)
@@ -127,6 +181,26 @@ var _ = Describe("PlayTracker", func() {
})
})
+ Describe("Expiration events", func() {
+ It("sends event when entry expires", func() {
+ info := NowPlayingInfo{MediaFile: track, Start: time.Now(), Username: "user"}
+ _ = tracker.(*playTracker).playMap.AddWithTTL("player-1", info, 10*time.Millisecond)
+ Eventually(func() int { return len(eventBroker.getEvents()) }).Should(BeNumerically(">", 0))
+ eventList := eventBroker.getEvents()
+ evt, ok := eventList[len(eventList)-1].(*events.NowPlayingCount)
+ Expect(ok).To(BeTrue())
+ Expect(evt.Count).To(Equal(0))
+ })
+
+ It("does not send event when disabled", func() {
+ conf.Server.EnableNowPlaying = false
+ tracker = newPlayTracker(ds, eventBroker, nil)
+ info := NowPlayingInfo{MediaFile: track, Start: time.Now(), Username: "user"}
+ _ = tracker.(*playTracker).playMap.AddWithTTL("player-2", info, 10*time.Millisecond)
+ Consistently(func() int { return len(eventBroker.getEvents()) }).Should(Equal(0))
+ })
+ })
+
Describe("Submit", func() {
It("sends track to agent", func() {
ctx = request.WithUser(ctx, model.User{ID: "u-1", UserName: "user-1"})
@@ -135,10 +209,12 @@ var _ = Describe("PlayTracker", func() {
err := tracker.Submit(ctx, []Submission{{TrackID: "123", Timestamp: ts}})
Expect(err).ToNot(HaveOccurred())
- Expect(fake.ScrobbleCalled).To(BeTrue())
+ Expect(fake.ScrobbleCalled.Load()).To(BeTrue())
Expect(fake.UserID).To(Equal("u-1"))
- Expect(fake.LastScrobble.ID).To(Equal("123"))
- Expect(fake.LastScrobble.Participants).To(Equal(track.Participants))
+ lastScrobble := fake.LastScrobble.Load()
+ Expect(lastScrobble.TimeStamp).To(BeTemporally("~", ts, 1*time.Second))
+ Expect(lastScrobble.ID).To(Equal("123"))
+ Expect(lastScrobble.Participants).To(Equal(track.Participants))
})
It("increments play counts in the DB", func() {
@@ -162,7 +238,7 @@ var _ = Describe("PlayTracker", func() {
err := tracker.Submit(ctx, []Submission{{TrackID: "123", Timestamp: time.Now()}})
Expect(err).ToNot(HaveOccurred())
- Expect(fake.ScrobbleCalled).To(BeFalse())
+ Expect(fake.ScrobbleCalled.Load()).To(BeFalse())
})
It("does not send track to agent if player is not enabled to send scrobbles", func() {
@@ -171,7 +247,7 @@ var _ = Describe("PlayTracker", func() {
err := tracker.Submit(ctx, []Submission{{TrackID: "123", Timestamp: time.Now()}})
Expect(err).ToNot(HaveOccurred())
- Expect(fake.ScrobbleCalled).To(BeFalse())
+ Expect(fake.ScrobbleCalled.Load()).To(BeFalse())
})
It("does not send track to agent if artist is unknown", func() {
@@ -180,7 +256,7 @@ var _ = Describe("PlayTracker", func() {
err := tracker.Submit(ctx, []Submission{{TrackID: "123", Timestamp: time.Now()}})
Expect(err).ToNot(HaveOccurred())
- Expect(fake.ScrobbleCalled).To(BeFalse())
+ Expect(fake.ScrobbleCalled.Load()).To(BeFalse())
})
It("increments play counts even if it cannot scrobble", func() {
@@ -189,7 +265,7 @@ var _ = Describe("PlayTracker", func() {
err := tracker.Submit(ctx, []Submission{{TrackID: "123", Timestamp: time.Now()}})
Expect(err).ToNot(HaveOccurred())
- Expect(fake.ScrobbleCalled).To(BeFalse())
+ Expect(fake.ScrobbleCalled.Load()).To(BeFalse())
Expect(track.PlayCount).To(Equal(int64(1)))
Expect(album.PlayCount).To(Equal(int64(1)))
@@ -200,15 +276,111 @@ var _ = Describe("PlayTracker", func() {
})
})
+ Describe("Plugin scrobbler logic", func() {
+ var pluginLoader *mockPluginLoader
+ var pluginFake fakeScrobbler
+
+ BeforeEach(func() {
+ pluginFake = fakeScrobbler{Authorized: true}
+ pluginLoader = &mockPluginLoader{
+ names: []string{"plugin1"},
+ scrobblers: map[string]Scrobbler{"plugin1": &pluginFake},
+ }
+ tracker = newPlayTracker(ds, events.GetBroker(), pluginLoader)
+
+ // Bypass buffering for both built-in and plugin scrobblers
+ tracker.(*playTracker).builtinScrobblers["fake"] = &fake
+ tracker.(*playTracker).pluginScrobblers["plugin1"] = &pluginFake
+ })
+
+ It("registers and uses plugin scrobbler for NowPlaying", func() {
+ err := tracker.NowPlaying(ctx, "player-1", "player-one", "123", 0)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(pluginFake.NowPlayingCalled).To(BeTrue())
+ })
+
+ It("removes plugin scrobbler if not present anymore", func() {
+ // First call: plugin present
+ _ = tracker.NowPlaying(ctx, "player-1", "player-one", "123", 0)
+ Expect(pluginFake.NowPlayingCalled).To(BeTrue())
+ pluginFake.NowPlayingCalled = false
+ // Remove plugin
+ pluginLoader.names = []string{}
+ _ = tracker.NowPlaying(ctx, "player-1", "player-one", "123", 0)
+ Expect(pluginFake.NowPlayingCalled).To(BeFalse())
+ })
+
+ It("calls both builtin and plugin scrobblers for NowPlaying", func() {
+ fake.NowPlayingCalled = false
+ pluginFake.NowPlayingCalled = false
+ err := tracker.NowPlaying(ctx, "player-1", "player-one", "123", 0)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(fake.NowPlayingCalled).To(BeTrue())
+ Expect(pluginFake.NowPlayingCalled).To(BeTrue())
+ })
+
+ It("calls plugin scrobbler for Submit", func() {
+ ts := time.Now()
+ err := tracker.Submit(ctx, []Submission{{TrackID: "123", Timestamp: ts}})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(pluginFake.ScrobbleCalled.Load()).To(BeTrue())
+ })
+ })
+
+ Describe("Plugin Scrobbler Management", func() {
+ var pluginScr *fakeScrobbler
+ var mockPlugin *mockPluginLoader
+ var pTracker *playTracker
+ var mockedBS *mockBufferedScrobbler
+
+ BeforeEach(func() {
+ ctx = context.Background()
+ ctx = request.WithUser(ctx, model.User{ID: "u-1"})
+ ctx = request.WithPlayer(ctx, model.Player{ScrobbleEnabled: true})
+ ds = &tests.MockDataStore{}
+
+ // Setup plugin scrobbler
+ pluginScr = &fakeScrobbler{Authorized: true}
+ mockPlugin = &mockPluginLoader{
+ names: []string{"plugin1"},
+ scrobblers: map[string]Scrobbler{"plugin1": pluginScr},
+ }
+
+ // Create a tracker with the mock plugin loader
+ pTracker = newPlayTracker(ds, events.GetBroker(), mockPlugin)
+
+ // Create a mock buffered scrobbler and explicitly cast it to Scrobbler
+ mockedBS = &mockBufferedScrobbler{
+ wrapped: pluginScr,
+ }
+ // Make sure the instance is added with its concrete type preserved
+ pTracker.pluginScrobblers["plugin1"] = mockedBS
+ })
+
+ It("calls Stop on scrobblers when removing them", func() {
+ // Change the plugin names to simulate a plugin being removed
+ mockPlugin.names = []string{}
+
+ // Call refreshPluginScrobblers which should detect the removed plugin
+ pTracker.refreshPluginScrobblers()
+
+ // Verify the Stop method was called
+ Expect(mockedBS.stopCalled).To(BeTrue())
+
+ // Verify the scrobbler was removed from the map
+ Expect(pTracker.pluginScrobblers).NotTo(HaveKey("plugin1"))
+ })
+ })
})
type fakeScrobbler struct {
Authorized bool
NowPlayingCalled bool
- ScrobbleCalled bool
+ ScrobbleCalled atomic.Bool
UserID string
Track *model.MediaFile
- LastScrobble Scrobble
+ Position int
+ LastScrobble atomic.Pointer[Scrobble]
Error error
}
@@ -216,23 +388,24 @@ func (f *fakeScrobbler) IsAuthorized(ctx context.Context, userId string) bool {
return f.Error == nil && f.Authorized
}
-func (f *fakeScrobbler) NowPlaying(ctx context.Context, userId string, track *model.MediaFile) error {
+func (f *fakeScrobbler) NowPlaying(ctx context.Context, userId string, track *model.MediaFile, position int) error {
f.NowPlayingCalled = true
if f.Error != nil {
return f.Error
}
f.UserID = userId
f.Track = track
+ f.Position = position
return nil
}
func (f *fakeScrobbler) Scrobble(ctx context.Context, userId string, s Scrobble) error {
- f.ScrobbleCalled = true
+ f.UserID = userId
+ f.LastScrobble.Store(&s)
+ f.ScrobbleCalled.Store(true)
if f.Error != nil {
return f.Error
}
- f.UserID = userId
- f.LastScrobble = s
return nil
}
@@ -243,3 +416,51 @@ func _p(id, name string, sortName ...string) model.Participant {
}
return p
}
+
+type fakeEventBroker struct {
+ http.Handler
+ events []events.Event
+ mu sync.Mutex
+}
+
+func (f *fakeEventBroker) SendMessage(_ context.Context, event events.Event) {
+ f.mu.Lock()
+ defer f.mu.Unlock()
+ f.events = append(f.events, event)
+}
+
+func (f *fakeEventBroker) SendBroadcastMessage(_ context.Context, event events.Event) {
+ f.mu.Lock()
+ defer f.mu.Unlock()
+ f.events = append(f.events, event)
+}
+
+func (f *fakeEventBroker) getEvents() []events.Event {
+ f.mu.Lock()
+ defer f.mu.Unlock()
+ return f.events
+}
+
+var _ events.Broker = (*fakeEventBroker)(nil)
+
+// mockBufferedScrobbler used to test that Stop is called
+type mockBufferedScrobbler struct {
+ wrapped Scrobbler
+ stopCalled bool
+}
+
+func (m *mockBufferedScrobbler) Stop() {
+ m.stopCalled = true
+}
+
+func (m *mockBufferedScrobbler) IsAuthorized(ctx context.Context, userId string) bool {
+ return m.wrapped.IsAuthorized(ctx, userId)
+}
+
+func (m *mockBufferedScrobbler) NowPlaying(ctx context.Context, userId string, track *model.MediaFile, position int) error {
+ return m.wrapped.NowPlaying(ctx, userId, track, position)
+}
+
+func (m *mockBufferedScrobbler) Scrobble(ctx context.Context, userId string, s Scrobble) error {
+ return m.wrapped.Scrobble(ctx, userId, s)
+}
diff --git a/core/share.go b/core/share.go
index add88322d..eb5e6679b 100644
--- a/core/share.go
+++ b/core/share.go
@@ -13,6 +13,7 @@ import (
"github.com/navidrome/navidrome/model"
. "github.com/navidrome/navidrome/utils/gg"
"github.com/navidrome/navidrome/utils/slice"
+ "github.com/navidrome/navidrome/utils/str"
)
type Share interface {
@@ -119,9 +120,8 @@ func (r *shareRepositoryWrapper) Save(entity interface{}) (string, error) {
log.Error(r.ctx, "Invalid Resource ID", "id", firstId)
return "", model.ErrNotFound
}
- if len(s.Contents) > 30 {
- s.Contents = s.Contents[:26] + "..."
- }
+
+ s.Contents = str.TruncateRunes(s.Contents, 30, "...")
id, err = r.Persistable.Save(s)
return id, err
@@ -149,7 +149,7 @@ func (r *shareRepositoryWrapper) contentsLabelFromArtist(shareID string, ids str
func (r *shareRepositoryWrapper) contentsLabelFromAlbums(shareID string, ids string) string {
idList := strings.Split(ids, ",")
- all, err := r.ds.Album(r.ctx).GetAll(model.QueryOptions{Filters: squirrel.Eq{"id": idList}})
+ all, err := r.ds.Album(r.ctx).GetAll(model.QueryOptions{Filters: squirrel.Eq{"album.id": idList}})
if err != nil {
log.Error(r.ctx, "Error retrieving album names for share", "share", shareID, err)
return ""
diff --git a/core/share_test.go b/core/share_test.go
index 21069bb59..475d40ec9 100644
--- a/core/share_test.go
+++ b/core/share_test.go
@@ -38,6 +38,38 @@ var _ = Describe("Share", func() {
Expect(id).ToNot(BeEmpty())
Expect(entity.ID).To(Equal(id))
})
+
+ It("does not truncate ASCII labels shorter than 30 characters", func() {
+ _ = ds.MediaFile(ctx).Put(&model.MediaFile{ID: "456", Title: "Example Media File"})
+ entity := &model.Share{Description: "test", ResourceIDs: "456"}
+ _, err := repo.Save(entity)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(entity.Contents).To(Equal("Example Media File"))
+ })
+
+ It("truncates ASCII labels longer than 30 characters", func() {
+ _ = ds.MediaFile(ctx).Put(&model.MediaFile{ID: "789", Title: "Example Media File But The Title Is Really Long For Testing Purposes"})
+ entity := &model.Share{Description: "test", ResourceIDs: "789"}
+ _, err := repo.Save(entity)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(entity.Contents).To(Equal("Example Media File But The ..."))
+ })
+
+ It("does not truncate CJK labels shorter than 30 runes", func() {
+ _ = ds.MediaFile(ctx).Put(&model.MediaFile{ID: "456", Title: "青春コンプレックス"})
+ entity := &model.Share{Description: "test", ResourceIDs: "456"}
+ _, err := repo.Save(entity)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(entity.Contents).To(Equal("青春コンプレックス"))
+ })
+
+ It("truncates CJK labels longer than 30 runes", func() {
+ _ = ds.MediaFile(ctx).Put(&model.MediaFile{ID: "789", Title: "私の中の幻想的世界観及びその顕現を想起させたある現実での出来事に関する一考察"})
+ entity := &model.Share{Description: "test", ResourceIDs: "789"}
+ _, err := repo.Save(entity)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(entity.Contents).To(Equal("私の中の幻想的世界観及びその顕現を想起させたある現実で..."))
+ })
})
Describe("Update", func() {
diff --git a/core/storage/local/local_suite_test.go b/core/storage/local/local_suite_test.go
index 98dfcbd4b..5934cde5d 100644
--- a/core/storage/local/local_suite_test.go
+++ b/core/storage/local/local_suite_test.go
@@ -3,11 +3,15 @@ package local
import (
"testing"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/tests"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
)
func TestLocal(t *testing.T) {
+ tests.Init(t, false)
+ log.SetLevel(log.LevelFatal)
RegisterFailHandler(Fail)
- RunSpecs(t, "Local Storage Test Suite")
+ RunSpecs(t, "Local Storage Suite")
}
diff --git a/core/storage/local/local_test.go b/core/storage/local/local_test.go
new file mode 100644
index 000000000..3ed01bbc4
--- /dev/null
+++ b/core/storage/local/local_test.go
@@ -0,0 +1,428 @@
+package local
+
+import (
+ "io/fs"
+ "net/url"
+ "os"
+ "path/filepath"
+ "runtime"
+ "time"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/core/storage"
+ "github.com/navidrome/navidrome/model/metadata"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("LocalStorage", func() {
+ var tempDir string
+ var testExtractor *mockTestExtractor
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+
+ // Create a temporary directory for testing
+ var err error
+ tempDir, err = os.MkdirTemp("", "navidrome-local-storage-test-")
+ Expect(err).ToNot(HaveOccurred())
+
+ DeferCleanup(func() {
+ os.RemoveAll(tempDir)
+ })
+
+ // Create and register a test extractor
+ testExtractor = &mockTestExtractor{
+ results: make(map[string]metadata.Info),
+ }
+ RegisterExtractor("test", func(fs.FS, string) Extractor {
+ return testExtractor
+ })
+ conf.Server.Scanner.Extractor = "test"
+ })
+
+ Describe("newLocalStorage", func() {
+ Context("with valid path", func() {
+ It("should create a localStorage instance with correct path", func() {
+ u, err := url.Parse("file://" + tempDir)
+ Expect(err).ToNot(HaveOccurred())
+
+ storage := newLocalStorage(*u)
+ localStorage := storage.(*localStorage)
+
+ Expect(localStorage.u.Scheme).To(Equal("file"))
+ // Check that the path is set correctly (could be resolved to real path on macOS)
+ Expect(localStorage.u.Path).To(ContainSubstring("navidrome-local-storage-test"))
+ Expect(localStorage.resolvedPath).To(ContainSubstring("navidrome-local-storage-test"))
+ Expect(localStorage.extractor).ToNot(BeNil())
+ })
+
+ It("should handle URL-decoded paths correctly", func() {
+ // Create a directory with spaces to test URL decoding
+ spacedDir := filepath.Join(tempDir, "test folder")
+ err := os.MkdirAll(spacedDir, 0755)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Use proper URL construction instead of manual escaping
+ u := &url.URL{
+ Scheme: "file",
+ Path: spacedDir,
+ }
+
+ storage := newLocalStorage(*u)
+ localStorage, ok := storage.(*localStorage)
+ Expect(ok).To(BeTrue())
+
+ Expect(localStorage.u.Path).To(Equal(spacedDir))
+ })
+
+ It("should resolve symlinks when possible", func() {
+ // Create a real directory and a symlink to it
+ realDir := filepath.Join(tempDir, "real")
+ linkDir := filepath.Join(tempDir, "link")
+
+ err := os.MkdirAll(realDir, 0755)
+ Expect(err).ToNot(HaveOccurred())
+
+ err = os.Symlink(realDir, linkDir)
+ Expect(err).ToNot(HaveOccurred())
+
+ u, err := url.Parse("file://" + linkDir)
+ Expect(err).ToNot(HaveOccurred())
+
+ storage := newLocalStorage(*u)
+ localStorage, ok := storage.(*localStorage)
+ Expect(ok).To(BeTrue())
+
+ Expect(localStorage.u.Path).To(Equal(linkDir))
+ // Check that the resolved path contains the real directory name
+ Expect(localStorage.resolvedPath).To(ContainSubstring("real"))
+ })
+
+ It("should use u.Path as resolvedPath when symlink resolution fails", func() {
+ // Use a non-existent path to trigger symlink resolution failure
+ nonExistentPath := filepath.Join(tempDir, "non-existent")
+
+ u, err := url.Parse("file://" + nonExistentPath)
+ Expect(err).ToNot(HaveOccurred())
+
+ storage := newLocalStorage(*u)
+ localStorage, ok := storage.(*localStorage)
+ Expect(ok).To(BeTrue())
+
+ Expect(localStorage.u.Path).To(Equal(nonExistentPath))
+ Expect(localStorage.resolvedPath).To(Equal(nonExistentPath))
+ })
+ })
+
+ Context("with Windows path", func() {
+ BeforeEach(func() {
+ if runtime.GOOS != "windows" {
+ Skip("Windows-specific test")
+ }
+ })
+
+ It("should handle Windows drive letters correctly", func() {
+ u, err := url.Parse("file://C:/music")
+ Expect(err).ToNot(HaveOccurred())
+
+ storage := newLocalStorage(*u)
+ localStorage, ok := storage.(*localStorage)
+ Expect(ok).To(BeTrue())
+
+ Expect(localStorage.u.Path).To(Equal("C:/music"))
+ })
+ })
+
+ Context("with invalid extractor", func() {
+ It("should handle extractor validation correctly", func() {
+ // Note: The actual implementation uses log.Fatal which exits the process,
+ // so we test the normal path where extractors exist
+
+ u, err := url.Parse("file://" + tempDir)
+ Expect(err).ToNot(HaveOccurred())
+
+ storage := newLocalStorage(*u)
+ Expect(storage).ToNot(BeNil())
+ })
+ })
+ })
+
+ Describe("localStorage.FS", func() {
+ Context("with existing directory", func() {
+ It("should return a localFS instance", func() {
+ u, err := url.Parse("file://" + tempDir)
+ Expect(err).ToNot(HaveOccurred())
+
+ storage := newLocalStorage(*u)
+ musicFS, err := storage.FS()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(musicFS).ToNot(BeNil())
+
+ _, ok := musicFS.(*localFS)
+ Expect(ok).To(BeTrue())
+ })
+ })
+
+ Context("with non-existent directory", func() {
+ It("should return an error", func() {
+ nonExistentPath := filepath.Join(tempDir, "non-existent")
+ u, err := url.Parse("file://" + nonExistentPath)
+ Expect(err).ToNot(HaveOccurred())
+
+ storage := newLocalStorage(*u)
+ _, err = storage.FS()
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring(nonExistentPath))
+ })
+ })
+ })
+
+ Describe("localFS.ReadTags", func() {
+ var testFile string
+
+ BeforeEach(func() {
+ // Create a test file
+ testFile = filepath.Join(tempDir, "test.mp3")
+ err := os.WriteFile(testFile, []byte("test data"), 0600)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Reset extractor state
+ testExtractor.results = make(map[string]metadata.Info)
+ testExtractor.err = nil
+ })
+
+ Context("when extractor returns complete metadata", func() {
+ It("should return the metadata as-is", func() {
+ expectedInfo := metadata.Info{
+ Tags: map[string][]string{
+ "title": {"Test Song"},
+ "artist": {"Test Artist"},
+ },
+ AudioProperties: metadata.AudioProperties{
+ Duration: 180,
+ BitRate: 320,
+ },
+ FileInfo: &testFileInfo{name: "test.mp3"},
+ }
+
+ testExtractor.results["test.mp3"] = expectedInfo
+
+ u, err := url.Parse("file://" + tempDir)
+ Expect(err).ToNot(HaveOccurred())
+ storage := newLocalStorage(*u)
+ musicFS, err := storage.FS()
+ Expect(err).ToNot(HaveOccurred())
+
+ results, err := musicFS.ReadTags("test.mp3")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveKey("test.mp3"))
+ Expect(results["test.mp3"]).To(Equal(expectedInfo))
+ })
+ })
+
+ Context("when extractor returns metadata without FileInfo", func() {
+ It("should populate FileInfo from filesystem", func() {
+ incompleteInfo := metadata.Info{
+ Tags: map[string][]string{
+ "title": {"Test Song"},
+ },
+ FileInfo: nil, // Missing FileInfo
+ }
+
+ testExtractor.results["test.mp3"] = incompleteInfo
+
+ u, err := url.Parse("file://" + tempDir)
+ Expect(err).ToNot(HaveOccurred())
+ storage := newLocalStorage(*u)
+ musicFS, err := storage.FS()
+ Expect(err).ToNot(HaveOccurred())
+
+ results, err := musicFS.ReadTags("test.mp3")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveKey("test.mp3"))
+
+ result := results["test.mp3"]
+ Expect(result.FileInfo).ToNot(BeNil())
+ Expect(result.FileInfo.Name()).To(Equal("test.mp3"))
+
+ // Should be wrapped in localFileInfo
+ _, ok := result.FileInfo.(localFileInfo)
+ Expect(ok).To(BeTrue())
+ })
+ })
+
+ Context("when filesystem stat fails", func() {
+ It("should return an error", func() {
+ incompleteInfo := metadata.Info{
+ Tags: map[string][]string{"title": {"Test Song"}},
+ FileInfo: nil,
+ }
+
+ testExtractor.results["non-existent.mp3"] = incompleteInfo
+
+ u, err := url.Parse("file://" + tempDir)
+ Expect(err).ToNot(HaveOccurred())
+ storage := newLocalStorage(*u)
+ musicFS, err := storage.FS()
+ Expect(err).ToNot(HaveOccurred())
+
+ _, err = musicFS.ReadTags("non-existent.mp3")
+ Expect(err).To(HaveOccurred())
+ })
+ })
+
+ Context("when extractor fails", func() {
+ It("should return the extractor error", func() {
+ testExtractor.err = &extractorError{message: "extractor failed"}
+
+ u, err := url.Parse("file://" + tempDir)
+ Expect(err).ToNot(HaveOccurred())
+ storage := newLocalStorage(*u)
+ musicFS, err := storage.FS()
+ Expect(err).ToNot(HaveOccurred())
+
+ _, err = musicFS.ReadTags("test.mp3")
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("extractor failed"))
+ })
+ })
+
+ Context("with multiple files", func() {
+ It("should process all files correctly", func() {
+ // Create another test file
+ testFile2 := filepath.Join(tempDir, "test2.mp3")
+ err := os.WriteFile(testFile2, []byte("test data 2"), 0600)
+ Expect(err).ToNot(HaveOccurred())
+
+ info1 := metadata.Info{
+ Tags: map[string][]string{"title": {"Song 1"}},
+ FileInfo: &testFileInfo{name: "test.mp3"},
+ }
+ info2 := metadata.Info{
+ Tags: map[string][]string{"title": {"Song 2"}},
+ FileInfo: nil, // This one needs FileInfo populated
+ }
+
+ testExtractor.results["test.mp3"] = info1
+ testExtractor.results["test2.mp3"] = info2
+
+ u, err := url.Parse("file://" + tempDir)
+ Expect(err).ToNot(HaveOccurred())
+ storage := newLocalStorage(*u)
+ musicFS, err := storage.FS()
+ Expect(err).ToNot(HaveOccurred())
+
+ results, err := musicFS.ReadTags("test.mp3", "test2.mp3")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(2))
+
+ Expect(results["test.mp3"].FileInfo).To(Equal(&testFileInfo{name: "test.mp3"}))
+ Expect(results["test2.mp3"].FileInfo).ToNot(BeNil())
+ Expect(results["test2.mp3"].FileInfo.Name()).To(Equal("test2.mp3"))
+ })
+ })
+ })
+
+ Describe("localFileInfo", func() {
+ var testFile string
+ var fileInfo fs.FileInfo
+
+ BeforeEach(func() {
+ testFile = filepath.Join(tempDir, "test.mp3")
+ err := os.WriteFile(testFile, []byte("test data"), 0600)
+ Expect(err).ToNot(HaveOccurred())
+
+ fileInfo, err = os.Stat(testFile)
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ Describe("BirthTime", func() {
+ It("should return birth time when available", func() {
+ lfi := localFileInfo{FileInfo: fileInfo}
+ birthTime := lfi.BirthTime()
+
+ // Birth time should be a valid time (not zero value)
+ Expect(birthTime).ToNot(BeZero())
+ // Should be around the current time (within last few minutes)
+ Expect(birthTime).To(BeTemporally("~", time.Now(), 5*time.Minute))
+ })
+ })
+
+ It("should delegate all other FileInfo methods", func() {
+ lfi := localFileInfo{FileInfo: fileInfo}
+
+ Expect(lfi.Name()).To(Equal(fileInfo.Name()))
+ Expect(lfi.Size()).To(Equal(fileInfo.Size()))
+ Expect(lfi.Mode()).To(Equal(fileInfo.Mode()))
+ Expect(lfi.ModTime()).To(Equal(fileInfo.ModTime()))
+ Expect(lfi.IsDir()).To(Equal(fileInfo.IsDir()))
+ Expect(lfi.Sys()).To(Equal(fileInfo.Sys()))
+ })
+ })
+
+ Describe("Storage registration", func() {
+ It("should register localStorage for file scheme", func() {
+ // This tests the init() function indirectly
+ storage, err := storage.For("file://" + tempDir)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(storage).To(BeAssignableToTypeOf(&localStorage{}))
+ })
+ })
+})
+
+// Test extractor for testing
+type mockTestExtractor struct {
+ results map[string]metadata.Info
+ err error
+}
+
+func (m *mockTestExtractor) Parse(files ...string) (map[string]metadata.Info, error) {
+ if m.err != nil {
+ return nil, m.err
+ }
+
+ result := make(map[string]metadata.Info)
+ for _, file := range files {
+ if info, exists := m.results[file]; exists {
+ result[file] = info
+ }
+ }
+ return result, nil
+}
+
+func (m *mockTestExtractor) Version() string {
+ return "test-1.0"
+}
+
+type extractorError struct {
+ message string
+}
+
+func (e *extractorError) Error() string {
+ return e.message
+}
+
+// Test FileInfo that implements metadata.FileInfo
+type testFileInfo struct {
+ name string
+ size int64
+ mode fs.FileMode
+ modTime time.Time
+ isDir bool
+ birthTime time.Time
+}
+
+func (t *testFileInfo) Name() string { return t.name }
+func (t *testFileInfo) Size() int64 { return t.size }
+func (t *testFileInfo) Mode() fs.FileMode { return t.mode }
+func (t *testFileInfo) ModTime() time.Time { return t.modTime }
+func (t *testFileInfo) IsDir() bool { return t.isDir }
+func (t *testFileInfo) Sys() any { return nil }
+func (t *testFileInfo) BirthTime() time.Time {
+ if t.birthTime.IsZero() {
+ return time.Now()
+ }
+ return t.birthTime
+}
diff --git a/core/storage/storage.go b/core/storage/storage.go
index 84bcae0d6..b9fceb1fd 100644
--- a/core/storage/storage.go
+++ b/core/storage/storage.go
@@ -6,6 +6,8 @@ import (
"path/filepath"
"strings"
"sync"
+
+ "github.com/navidrome/navidrome/utils/slice"
)
const LocalSchemaID = "file"
@@ -36,7 +38,14 @@ func For(uri string) (Storage, error) {
if len(parts) < 2 {
uri, _ = filepath.Abs(uri)
uri = filepath.ToSlash(uri)
- uri = LocalSchemaID + "://" + uri
+
+ // Properly escape each path component using URL standards
+ pathParts := strings.Split(uri, "/")
+ escapedParts := slice.Map(pathParts, func(s string) string {
+ return url.PathEscape(s)
+ })
+
+ uri = LocalSchemaID + "://" + strings.Join(escapedParts, "/")
}
u, err := url.Parse(uri)
diff --git a/core/storage/storage_test.go b/core/storage/storage_test.go
index c74c7c6ed..60496e611 100644
--- a/core/storage/storage_test.go
+++ b/core/storage/storage_test.go
@@ -65,6 +65,21 @@ var _ = Describe("Storage", func() {
_, err := For("webdav:///tmp")
Expect(err).To(HaveOccurred())
})
+ DescribeTable("should handle paths with special characters correctly",
+ func(inputPath string) {
+ s, err := For(inputPath)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(s).To(BeAssignableToTypeOf(&fakeLocalStorage{}))
+ Expect(s.(*fakeLocalStorage).u.Scheme).To(Equal("file"))
+ // The path should be exactly the same as the input - after URL parsing it gets decoded back
+ Expect(s.(*fakeLocalStorage).u.Path).To(Equal(inputPath))
+ },
+ Entry("hash symbols", "/tmp/test#folder/file.mp3"),
+ Entry("spaces", "/tmp/test folder/file with spaces.mp3"),
+ Entry("question marks", "/tmp/test?query/file.mp3"),
+ Entry("ampersands", "/tmp/test&/file.mp3"),
+ Entry("multiple special chars", "/tmp/Song #1 & More?.mp3"),
+ )
})
})
diff --git a/core/wire_providers.go b/core/wire_providers.go
index 482cfbefe..16335645c 100644
--- a/core/wire_providers.go
+++ b/core/wire_providers.go
@@ -17,6 +17,8 @@ var Set = wire.NewSet(
NewPlayers,
NewShare,
NewPlaylists,
+ NewLibrary,
+ NewMaintenance,
agents.GetAgents,
external.NewProvider,
wire.Bind(new(external.Agents), new(*agents.Agents)),
diff --git a/db/migrations/20241026183640_support_new_scanner.go b/db/migrations/20241026183640_support_new_scanner.go
index 251b27f63..fcbef7e4e 100644
--- a/db/migrations/20241026183640_support_new_scanner.go
+++ b/db/migrations/20241026183640_support_new_scanner.go
@@ -13,7 +13,7 @@ import (
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
- "github.com/navidrome/navidrome/utils/chain"
+ "github.com/navidrome/navidrome/utils/run"
"github.com/pressly/goose/v3"
)
@@ -25,7 +25,7 @@ func upSupportNewScanner(ctx context.Context, tx *sql.Tx) error {
execute := createExecuteFunc(ctx, tx)
addColumn := createAddColumnFunc(ctx, tx)
- return chain.RunSequentially(
+ return run.Sequentially(
upSupportNewScanner_CreateTableFolder(ctx, execute),
upSupportNewScanner_PopulateTableFolder(ctx, tx),
upSupportNewScanner_UpdateTableMediaFile(ctx, execute, addColumn),
@@ -213,7 +213,7 @@ update media_file set path = replace(substr(path, %d), '\', '/');`, libPathLen+2
func upSupportNewScanner_UpdateTableMediaFile(_ context.Context, execute execStmtFunc, addColumn addColumnFunc) execFunc {
return func() error {
- return chain.RunSequentially(
+ return run.Sequentially(
execute(`
alter table media_file
add column folder_id varchar default '' not null;
@@ -288,7 +288,7 @@ create index if not exists album_mbz_release_group_id
func upSupportNewScanner_UpdateTableArtist(_ context.Context, execute execStmtFunc, addColumn addColumnFunc) execFunc {
return func() error {
- return chain.RunSequentially(
+ return run.Sequentially(
execute(`
alter table artist
drop column album_count;
diff --git a/db/migrations/20250611010101_playqueue_current_to_index.go b/db/migrations/20250611010101_playqueue_current_to_index.go
new file mode 100644
index 000000000..d9250eba2
--- /dev/null
+++ b/db/migrations/20250611010101_playqueue_current_to_index.go
@@ -0,0 +1,80 @@
+package migrations
+
+import (
+ "context"
+ "database/sql"
+ "strings"
+
+ "github.com/pressly/goose/v3"
+)
+
+func init() {
+ goose.AddMigrationContext(upPlayQueueCurrentToIndex, downPlayQueueCurrentToIndex)
+}
+
+func upPlayQueueCurrentToIndex(ctx context.Context, tx *sql.Tx) error {
+ _, err := tx.ExecContext(ctx, `
+create table playqueue_dg_tmp(
+ id varchar(255) not null,
+ user_id varchar(255) not null
+ references user(id)
+ on update cascade on delete cascade,
+ current integer not null default 0,
+ position real,
+ changed_by varchar(255),
+ items varchar(255),
+ created_at datetime,
+ updated_at datetime
+);`)
+ if err != nil {
+ return err
+ }
+
+ rows, err := tx.QueryContext(ctx, `select id, user_id, current, position, changed_by, items, created_at, updated_at from playqueue`)
+ if err != nil {
+ return err
+ }
+ defer rows.Close()
+
+ stmt, err := tx.PrepareContext(ctx, `insert into playqueue_dg_tmp(id, user_id, current, position, changed_by, items, created_at, updated_at) values(?,?,?,?,?,?,?,?)`)
+ if err != nil {
+ return err
+ }
+ defer stmt.Close()
+
+ for rows.Next() {
+ var id, userID, currentID, changedBy, items string
+ var position sql.NullFloat64
+ var createdAt, updatedAt sql.NullString
+ if err = rows.Scan(&id, &userID, ¤tID, &position, &changedBy, &items, &createdAt, &updatedAt); err != nil {
+ return err
+ }
+ index := 0
+ if currentID != "" && items != "" {
+ parts := strings.Split(items, ",")
+ for i, p := range parts {
+ if p == currentID {
+ index = i
+ break
+ }
+ }
+ }
+ _, err = stmt.Exec(id, userID, index, position, changedBy, items, createdAt, updatedAt)
+ if err != nil {
+ return err
+ }
+ }
+ if err = rows.Err(); err != nil {
+ return err
+ }
+
+ if _, err = tx.ExecContext(ctx, `drop table playqueue;`); err != nil {
+ return err
+ }
+ _, err = tx.ExecContext(ctx, `alter table playqueue_dg_tmp rename to playqueue;`)
+ return err
+}
+
+func downPlayQueueCurrentToIndex(ctx context.Context, tx *sql.Tx) error {
+ return nil
+}
diff --git a/db/migrations/20250701010101_add_folder_hash.go b/db/migrations/20250701010101_add_folder_hash.go
new file mode 100644
index 000000000..e82a0749f
--- /dev/null
+++ b/db/migrations/20250701010101_add_folder_hash.go
@@ -0,0 +1,21 @@
+package migrations
+
+import (
+ "context"
+ "database/sql"
+
+ "github.com/pressly/goose/v3"
+)
+
+func init() {
+ goose.AddMigrationContext(upAddFolderHash, downAddFolderHash)
+}
+
+func upAddFolderHash(ctx context.Context, tx *sql.Tx) error {
+ _, err := tx.ExecContext(ctx, `alter table folder add column hash varchar default '' not null;`)
+ return err
+}
+
+func downAddFolderHash(ctx context.Context, tx *sql.Tx) error {
+ return nil
+}
diff --git a/db/migrations/20250701010102_add_annotation_user_foreign_key.sql b/db/migrations/20250701010102_add_annotation_user_foreign_key.sql
new file mode 100644
index 000000000..114de2a88
--- /dev/null
+++ b/db/migrations/20250701010102_add_annotation_user_foreign_key.sql
@@ -0,0 +1,46 @@
+-- +goose Up
+-- +goose StatementBegin
+CREATE TABLE IF NOT EXISTS annotation_tmp
+(
+ user_id varchar(255) not null
+ REFERENCES user(id)
+ ON DELETE CASCADE
+ ON UPDATE CASCADE,
+ item_id varchar(255) default '' not null,
+ item_type varchar(255) default '' not null,
+ play_count integer default 0,
+ play_date datetime,
+ rating integer default 0,
+ starred bool default FALSE not null,
+ starred_at datetime,
+ unique (user_id, item_id, item_type)
+);
+
+
+INSERT INTO annotation_tmp(
+ user_id, item_id, item_type, play_count, play_date, rating, starred, starred_at
+)
+SELECT user_id, item_id, item_type, play_count, play_date, rating, starred, starred_at
+FROM annotation
+WHERE user_id IN (
+ SELECT id FROM user
+);
+
+DROP TABLE annotation;
+ALTER TABLE annotation_tmp RENAME TO annotation;
+
+CREATE INDEX annotation_play_count
+ on annotation (play_count);
+CREATE INDEX annotation_play_date
+ on annotation (play_date);
+CREATE INDEX annotation_rating
+ on annotation (rating);
+CREATE INDEX annotation_starred
+ on annotation (starred);
+CREATE INDEX annotation_starred_at
+ on annotation (starred_at);
+
+-- +goose StatementEnd
+
+-- +goose Down
+
diff --git a/db/migrations/20250701010103_add_library_stats.go b/db/migrations/20250701010103_add_library_stats.go
new file mode 100644
index 000000000..8025229cc
--- /dev/null
+++ b/db/migrations/20250701010103_add_library_stats.go
@@ -0,0 +1,48 @@
+package migrations
+
+import (
+ "context"
+ "database/sql"
+
+ "github.com/pressly/goose/v3"
+)
+
+func init() {
+ goose.AddMigrationContext(upAddLibraryStats, downAddLibraryStats)
+}
+
+func upAddLibraryStats(ctx context.Context, tx *sql.Tx) error {
+ _, err := tx.ExecContext(ctx, `
+alter table library add column total_songs integer default 0 not null;
+alter table library add column total_albums integer default 0 not null;
+alter table library add column total_artists integer default 0 not null;
+alter table library add column total_folders integer default 0 not null;
+ alter table library add column total_files integer default 0 not null;
+ alter table library add column total_missing_files integer default 0 not null;
+ alter table library add column total_size integer default 0 not null;
+update library set
+ total_songs = (
+ select count(*) from media_file where library_id = library.id and missing = 0
+ ),
+ total_albums = (select count(*) from album where library_id = library.id and missing = 0),
+ total_artists = (
+ select count(*) from library_artist la
+ join artist a on la.artist_id = a.id
+ where la.library_id = library.id and a.missing = 0
+ ),
+ total_folders = (select count(*) from folder where library_id = library.id and missing = 0 and num_audio_files > 0),
+ total_files = (
+ select ifnull(sum(num_audio_files + num_playlists + json_array_length(image_files)),0)
+ from folder where library_id = library.id and missing = 0
+ ),
+ total_missing_files = (
+ select count(*) from media_file where library_id = library.id and missing = 1
+ ),
+ total_size = (select ifnull(sum(size),0) from album where library_id = library.id and missing = 0);
+`)
+ return err
+}
+
+func downAddLibraryStats(ctx context.Context, tx *sql.Tx) error {
+ return nil
+}
diff --git a/db/migrations/20250701010104_make_replaygain_fields_nullable.go b/db/migrations/20250701010104_make_replaygain_fields_nullable.go
new file mode 100644
index 000000000..163608d32
--- /dev/null
+++ b/db/migrations/20250701010104_make_replaygain_fields_nullable.go
@@ -0,0 +1,49 @@
+package migrations
+
+import (
+ "context"
+ "database/sql"
+
+ "github.com/pressly/goose/v3"
+)
+
+func init() {
+ goose.AddMigrationContext(upMakeReplaygainFieldsNullable, downMakeReplaygainFieldsNullable)
+}
+
+func upMakeReplaygainFieldsNullable(ctx context.Context, tx *sql.Tx) error {
+ _, err := tx.ExecContext(ctx, `
+ALTER TABLE media_file ADD COLUMN rg_album_gain_new real;
+ALTER TABLE media_file ADD COLUMN rg_album_peak_new real;
+ALTER TABLE media_file ADD COLUMN rg_track_gain_new real;
+ALTER TABLE media_file ADD COLUMN rg_track_peak_new real;
+
+UPDATE media_file SET
+ rg_album_gain_new = rg_album_gain,
+ rg_album_peak_new = rg_album_peak,
+ rg_track_gain_new = rg_track_gain,
+ rg_track_peak_new = rg_track_peak;
+
+ALTER TABLE media_file DROP COLUMN rg_album_gain;
+ALTER TABLE media_file DROP COLUMN rg_album_peak;
+ALTER TABLE media_file DROP COLUMN rg_track_gain;
+ALTER TABLE media_file DROP COLUMN rg_track_peak;
+
+ALTER TABLE media_file RENAME COLUMN rg_album_gain_new TO rg_album_gain;
+ALTER TABLE media_file RENAME COLUMN rg_album_peak_new TO rg_album_peak;
+ALTER TABLE media_file RENAME COLUMN rg_track_gain_new TO rg_track_gain;
+ALTER TABLE media_file RENAME COLUMN rg_track_peak_new TO rg_track_peak;
+ `)
+
+ if err != nil {
+ return err
+ }
+
+ notice(tx, "Fetching replaygain fields properly will require a full scan")
+ return nil
+}
+
+func downMakeReplaygainFieldsNullable(ctx context.Context, tx *sql.Tx) error {
+ // This code is executed when the migration is rolled back.
+ return nil
+}
diff --git a/db/migrations/20250701010105_remove_dangling_items.sql b/db/migrations/20250701010105_remove_dangling_items.sql
new file mode 100644
index 000000000..aede49b6e
--- /dev/null
+++ b/db/migrations/20250701010105_remove_dangling_items.sql
@@ -0,0 +1,7 @@
+-- +goose Up
+-- +goose StatementBegin
+update media_file set missing = 1 where folder_id = '';
+update album set missing = 1 where folder_ids = '[]';
+-- +goose StatementEnd
+
+-- +goose Down
diff --git a/db/migrations/20250701010106_add_participant_stats_to_all_artists.sql b/db/migrations/20250701010106_add_participant_stats_to_all_artists.sql
new file mode 100644
index 000000000..1cd67dc32
--- /dev/null
+++ b/db/migrations/20250701010106_add_participant_stats_to_all_artists.sql
@@ -0,0 +1,65 @@
+-- +goose Up
+-- +goose StatementBegin
+WITH artist_role_counters AS (
+ SELECT jt.atom AS artist_id,
+ substr(
+ replace(jt.path, '$.', ''),
+ 1,
+ CASE WHEN instr(replace(jt.path, '$.', ''), '[') > 0
+ THEN instr(replace(jt.path, '$.', ''), '[') - 1
+ ELSE length(replace(jt.path, '$.', ''))
+ END
+ ) AS role,
+ count(DISTINCT mf.album_id) AS album_count,
+ count(mf.id) AS count,
+ sum(mf.size) AS size
+ FROM media_file mf
+ JOIN json_tree(mf.participants) jt ON jt.key = 'id' AND jt.atom IS NOT NULL
+ GROUP BY jt.atom, role
+),
+artist_total_counters AS (
+ SELECT mfa.artist_id,
+ 'total' AS role,
+ count(DISTINCT mf.album_id) AS album_count,
+ count(DISTINCT mf.id) AS count,
+ sum(mf.size) AS size
+ FROM media_file_artists mfa
+ JOIN media_file mf ON mfa.media_file_id = mf.id
+ GROUP BY mfa.artist_id
+),
+artist_participant_counter AS (
+ SELECT mfa.artist_id,
+ 'maincredit' AS role,
+ count(DISTINCT mf.album_id) AS album_count,
+ count(DISTINCT mf.id) AS count,
+ sum(mf.size) AS size
+ FROM media_file_artists mfa
+ JOIN media_file mf ON mfa.media_file_id = mf.id
+ AND mfa.role IN ('albumartist', 'artist')
+ GROUP BY mfa.artist_id
+),
+combined_counters AS (
+ SELECT artist_id, role, album_count, count, size FROM artist_role_counters
+ UNION
+ SELECT artist_id, role, album_count, count, size FROM artist_total_counters
+ UNION
+ SELECT artist_id, role, album_count, count, size FROM artist_participant_counter
+),
+artist_counters AS (
+ SELECT artist_id AS id,
+ json_group_object(
+ replace(role, '"', ''),
+ json_object('a', album_count, 'm', count, 's', size)
+ ) AS counters
+ FROM combined_counters
+ GROUP BY artist_id
+)
+UPDATE artist
+SET stats = coalesce((SELECT counters FROM artist_counters ac WHERE ac.id = artist.id), '{}'),
+ updated_at = datetime(current_timestamp, 'localtime')
+WHERE artist.id <> '';
+-- +goose StatementEnd
+
+-- +goose Down
+-- +goose StatementBegin
+-- +goose StatementEnd
diff --git a/db/migrations/20250701010107_add_mbid_indexes.sql b/db/migrations/20250701010107_add_mbid_indexes.sql
new file mode 100644
index 000000000..f8a5a444b
--- /dev/null
+++ b/db/migrations/20250701010107_add_mbid_indexes.sql
@@ -0,0 +1,27 @@
+-- +goose Up
+-- +goose StatementBegin
+
+-- Add indexes for MBID fields to improve lookup performance
+-- Artists table
+create index if not exists artist_mbz_artist_id
+ on artist (mbz_artist_id);
+
+-- Albums table
+create index if not exists album_mbz_album_id
+ on album (mbz_album_id);
+
+-- Media files table
+create index if not exists media_file_mbz_release_track_id
+ on media_file (mbz_release_track_id);
+
+-- +goose StatementEnd
+
+-- +goose Down
+-- +goose StatementBegin
+
+-- Remove MBID indexes
+drop index if exists artist_mbz_artist_id;
+drop index if exists album_mbz_album_id;
+drop index if exists media_file_mbz_release_track_id;
+
+-- +goose StatementEnd
diff --git a/db/migrations/20250701010108_add_multi_library_support.go b/db/migrations/20250701010108_add_multi_library_support.go
new file mode 100644
index 000000000..654784d09
--- /dev/null
+++ b/db/migrations/20250701010108_add_multi_library_support.go
@@ -0,0 +1,119 @@
+package migrations
+
+import (
+ "context"
+ "database/sql"
+
+ "github.com/pressly/goose/v3"
+)
+
+func init() {
+ goose.AddMigrationContext(upAddMultiLibrarySupport, downAddMultiLibrarySupport)
+}
+
+func upAddMultiLibrarySupport(ctx context.Context, tx *sql.Tx) error {
+ _, err := tx.ExecContext(ctx, `
+ -- Create user_library association table
+ CREATE TABLE user_library (
+ user_id VARCHAR(255) NOT NULL,
+ library_id INTEGER NOT NULL,
+ PRIMARY KEY (user_id, library_id),
+ FOREIGN KEY (user_id) REFERENCES user(id) ON DELETE CASCADE,
+ FOREIGN KEY (library_id) REFERENCES library(id) ON DELETE CASCADE
+ );
+ -- Create indexes for performance
+ CREATE INDEX idx_user_library_user_id ON user_library(user_id);
+ CREATE INDEX idx_user_library_library_id ON user_library(library_id);
+
+ -- Populate with existing users having access to library ID 1 (existing setup)
+ -- Admin users get access to all libraries, regular users get access to library 1
+ INSERT INTO user_library (user_id, library_id)
+ SELECT u.id, 1
+ FROM user u;
+
+ -- Add total_duration column to library table
+ ALTER TABLE library ADD COLUMN total_duration real DEFAULT 0;
+ UPDATE library SET total_duration = (
+ SELECT IFNULL(SUM(duration),0) from album where album.library_id = library.id and missing = 0
+ );
+
+ -- Add default_new_users column to library table
+ ALTER TABLE library ADD COLUMN default_new_users boolean DEFAULT false;
+ -- Set library ID 1 (default library) as default for new users
+ UPDATE library SET default_new_users = true WHERE id = 1;
+
+ -- Add stats column to library_artist junction table for per-library artist statistics
+ ALTER TABLE library_artist ADD COLUMN stats text DEFAULT '{}';
+
+ -- Migrate existing global artist stats to per-library format in library_artist table
+ -- For each library_artist association, copy the artist's global stats
+ UPDATE library_artist
+ SET stats = (
+ SELECT COALESCE(artist.stats, '{}')
+ FROM artist
+ WHERE artist.id = library_artist.artist_id
+ );
+
+ -- Remove stats column from artist table to eliminate duplication
+ -- Stats are now stored per-library in library_artist table
+ ALTER TABLE artist DROP COLUMN stats;
+
+ -- Create library_tag table for per-library tag statistics
+ CREATE TABLE library_tag (
+ tag_id VARCHAR NOT NULL,
+ library_id INTEGER NOT NULL,
+ album_count INTEGER DEFAULT 0 NOT NULL,
+ media_file_count INTEGER DEFAULT 0 NOT NULL,
+ PRIMARY KEY (tag_id, library_id),
+ FOREIGN KEY (tag_id) REFERENCES tag(id) ON DELETE CASCADE,
+ FOREIGN KEY (library_id) REFERENCES library(id) ON DELETE CASCADE
+ );
+
+ -- Create indexes for optimal query performance
+ CREATE INDEX idx_library_tag_tag_id ON library_tag(tag_id);
+ CREATE INDEX idx_library_tag_library_id ON library_tag(library_id);
+
+ -- Migrate existing tag stats to per-library format in library_tag table
+ -- For existing installations, copy current global stats to library ID 1 (default library)
+ INSERT INTO library_tag (tag_id, library_id, album_count, media_file_count)
+ SELECT t.id, 1, t.album_count, t.media_file_count
+ FROM tag t
+ WHERE EXISTS (SELECT 1 FROM library WHERE id = 1);
+
+ -- Remove global stats from tag table as they are now per-library
+ ALTER TABLE tag DROP COLUMN album_count;
+ ALTER TABLE tag DROP COLUMN media_file_count;
+ `)
+
+ return err
+}
+
+func downAddMultiLibrarySupport(ctx context.Context, tx *sql.Tx) error {
+ _, err := tx.ExecContext(ctx, `
+ -- Restore stats column to artist table before removing from library_artist
+ ALTER TABLE artist ADD COLUMN stats text DEFAULT '{}';
+
+ -- Restore global stats by aggregating from library_artist (simplified approach)
+ -- In a real rollback scenario, this might need more sophisticated logic
+ UPDATE artist
+ SET stats = (
+ SELECT COALESCE(la.stats, '{}')
+ FROM library_artist la
+ WHERE la.artist_id = artist.id
+ LIMIT 1
+ );
+
+ ALTER TABLE library_artist DROP COLUMN IF EXISTS stats;
+ DROP INDEX IF EXISTS idx_user_library_library_id;
+ DROP INDEX IF EXISTS idx_user_library_user_id;
+ DROP TABLE IF EXISTS user_library;
+ ALTER TABLE library DROP COLUMN IF EXISTS total_duration;
+ ALTER TABLE library DROP COLUMN IF EXISTS default_new_users;
+
+ -- Drop library_tag table and its indexes
+ DROP INDEX IF EXISTS idx_library_tag_library_id;
+ DROP INDEX IF EXISTS idx_library_tag_tag_id;
+ DROP TABLE IF EXISTS library_tag;
+ `)
+ return err
+}
diff --git a/db/migrations/20250823142158_make_playqueue_position_int.sql b/db/migrations/20250823142158_make_playqueue_position_int.sql
new file mode 100644
index 000000000..de20f0c79
--- /dev/null
+++ b/db/migrations/20250823142158_make_playqueue_position_int.sql
@@ -0,0 +1,9 @@
+-- +goose Up
+-- +goose StatementBegin
+ALTER TABLE playqueue ADD COLUMN position_int integer;
+UPDATE playqueue SET position_int = CAST(position as INTEGER) ;
+ALTER TABLE playqueue DROP COLUMN position;
+ALTER TABLE playqueue RENAME COLUMN position_int TO position;
+-- +goose StatementEnd
+
+-- +goose Down
diff --git a/git/pre-commit b/git/pre-commit
index 04f87994b..39ec8797f 100755
--- a/git/pre-commit
+++ b/git/pre-commit
@@ -12,7 +12,7 @@
gofmtcmd="go tool goimports"
-gofiles=$(git diff --cached --name-only --diff-filter=ACM | grep '.go$' | grep -v '_gen.go$')
+gofiles=$(git diff --cached --name-only --diff-filter=ACM | grep '.go$' | grep -v '_gen.go$' | grep -v '.pb.go$')
[ -z "$gofiles" ] && exit 0
unformatted=$($gofmtcmd -l $gofiles)
diff --git a/go.mod b/go.mod
index 5c09c0b17..f680bda51 100644
--- a/go.mod
+++ b/go.mod
@@ -1,15 +1,15 @@
module github.com/navidrome/navidrome
-go 1.24.2
+go 1.25
-// Fork to fix https://github.com/navidrome/navidrome/pull/3254
+// Fork to fix https://github.com/navidrome/navidrome/issues/3254
replace github.com/dhowden/tag v0.0.0-20240417053706-3d75831295e8 => github.com/deluan/tag v0.0.0-20241002021117-dfe5e6ea396d
require (
github.com/Masterminds/squirrel v1.5.4
github.com/RaveNoX/go-jsoncommentstrip v1.0.0
github.com/andybalholm/cascadia v1.3.3
- github.com/bmatcuk/doublestar/v4 v4.8.1
+ github.com/bmatcuk/doublestar/v4 v4.9.1
github.com/bradleyjkemp/cupaloy/v2 v2.8.0
github.com/deluan/rest v0.0.0-20211102003136-6260bc399cbf
github.com/deluan/sanitize v0.0.0-20241120162836-fdfd8fdfaa55
@@ -22,51 +22,60 @@ require (
github.com/djherbis/times v1.6.0
github.com/dustin/go-humanize v1.0.1
github.com/fatih/structs v1.1.0
- github.com/go-chi/chi/v5 v5.2.1
- github.com/go-chi/cors v1.2.1
+ github.com/go-chi/chi/v5 v5.2.3
+ github.com/go-chi/cors v1.2.2
github.com/go-chi/httprate v0.15.0
github.com/go-chi/jwtauth/v5 v5.3.3
github.com/go-viper/encoding/ini v0.1.1
- github.com/gohugoio/hashstructure v0.5.0
+ github.com/gohugoio/hashstructure v0.6.0
github.com/google/go-pipeline v0.0.0-20230411140531-6cbedfc1d3fc
github.com/google/uuid v1.6.0
- github.com/google/wire v0.6.0
+ github.com/google/wire v0.7.0
+ github.com/gorilla/websocket v1.5.3
github.com/hashicorp/go-multierror v1.1.1
- github.com/jellydator/ttlcache/v3 v3.3.0
- github.com/kardianos/service v1.2.2
+ github.com/jellydator/ttlcache/v3 v3.4.0
+ github.com/kardianos/service v1.2.4
+ github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51
+ github.com/knqyf263/go-plugin v0.9.0
github.com/kr/pretty v0.3.1
github.com/lestrrat-go/jwx/v2 v2.1.6
+ github.com/maruel/natural v1.2.1
github.com/matoous/go-nanoid/v2 v2.1.0
- github.com/mattn/go-sqlite3 v1.14.28
+ github.com/mattn/go-sqlite3 v1.14.32
github.com/microcosm-cc/bluemonday v1.0.27
github.com/mileusna/useragent v1.3.5
- github.com/onsi/ginkgo/v2 v2.23.4
- github.com/onsi/gomega v1.37.0
+ github.com/onsi/ginkgo/v2 v2.27.2
+ github.com/onsi/gomega v1.38.2
github.com/pelletier/go-toml/v2 v2.2.4
github.com/pocketbase/dbx v1.11.0
- github.com/pressly/goose/v3 v3.24.3
- github.com/prometheus/client_golang v1.22.0
+ github.com/pressly/goose/v3 v3.26.0
+ github.com/prometheus/client_golang v1.23.2
github.com/rjeczalik/notify v0.9.3
github.com/robfig/cron/v3 v3.0.1
github.com/sabhiram/go-gitignore v0.0.0-20210923224102-525f6e181f06
github.com/sirupsen/logrus v1.9.3
- github.com/spf13/cobra v1.9.1
- github.com/spf13/viper v1.20.1
- github.com/stretchr/testify v1.10.0
+ github.com/spf13/cobra v1.10.1
+ github.com/spf13/viper v1.21.0
+ github.com/stretchr/testify v1.11.1
+ github.com/tetratelabs/wazero v1.10.1
github.com/unrolled/secure v1.17.0
- github.com/xrash/smetrics v0.0.0-20240521201337-686a1a2994c1
+ github.com/xrash/smetrics v0.0.0-20250705151800-55b8f293f342
go.uber.org/goleak v1.3.0
- golang.org/x/exp v0.0.0-20250506013437-ce4c2cf36ca6
- golang.org/x/image v0.27.0
- golang.org/x/net v0.40.0
- golang.org/x/sync v0.14.0
- golang.org/x/sys v0.33.0
- golang.org/x/text v0.25.0
- golang.org/x/time v0.11.0
+ golang.org/x/exp v0.0.0-20251113190631-e25ba8c21ef6
+ golang.org/x/image v0.33.0
+ golang.org/x/net v0.47.0
+ golang.org/x/sync v0.18.0
+ golang.org/x/sys v0.38.0
+ golang.org/x/text v0.31.0
+ golang.org/x/time v0.14.0
+ google.golang.org/protobuf v1.36.10
gopkg.in/yaml.v3 v3.0.1
)
require (
+ dario.cat/mergo v1.0.2 // indirect
+ github.com/Masterminds/semver/v3 v3.4.0 // indirect
+ github.com/atombender/go-jsonschema v0.20.0 // indirect
github.com/aymerick/douceur v0.2.0 // indirect
github.com/beorn7/perks v1.0.1 // indirect
github.com/cespare/reflex v0.3.1 // indirect
@@ -75,55 +84,60 @@ require (
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect
github.com/decred/dcrd/dcrec/secp256k1/v4 v4.4.0 // indirect
github.com/fsnotify/fsnotify v1.9.0 // indirect
- github.com/go-logr/logr v1.4.2 // indirect
+ github.com/go-logr/logr v1.4.3 // indirect
github.com/go-task/slim-sprig/v3 v3.0.0 // indirect
- github.com/go-viper/mapstructure/v2 v2.2.1 // indirect
+ github.com/go-viper/mapstructure/v2 v2.4.0 // indirect
github.com/goccy/go-json v0.10.5 // indirect
+ github.com/goccy/go-yaml v1.18.0 // indirect
github.com/google/go-cmp v0.7.0 // indirect
- github.com/google/pprof v0.0.0-20250501235452-c0086092b71a // indirect
+ github.com/google/pprof v0.0.0-20251114195745-4902fdda35c8 // indirect
github.com/google/subcommands v1.2.0 // indirect
github.com/gorilla/css v1.0.1 // indirect
github.com/hashicorp/errwrap v1.1.0 // indirect
github.com/inconshreveable/mousetrap v1.1.0 // indirect
- github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51 // indirect
- github.com/klauspost/cpuid/v2 v2.2.10 // indirect
+ github.com/klauspost/cpuid/v2 v2.3.0 // indirect
github.com/kr/text v0.2.0 // indirect
github.com/lann/builder v0.0.0-20180802200727-47ae307949d0 // indirect
github.com/lann/ps v0.0.0-20150810152359-62de8c46ede0 // indirect
- github.com/lestrrat-go/blackmagic v1.0.3 // indirect
+ github.com/lestrrat-go/blackmagic v1.0.4 // indirect
github.com/lestrrat-go/httpcc v1.0.1 // indirect
github.com/lestrrat-go/httprc v1.0.6 // indirect
github.com/lestrrat-go/iter v1.0.2 // indirect
github.com/lestrrat-go/option v1.0.1 // indirect
github.com/mfridman/interpolate v0.0.2 // indirect
+ github.com/mitchellh/go-wordwrap v1.0.1 // indirect
github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 // indirect
github.com/ogier/pflag v0.0.1 // indirect
+ github.com/pkg/errors v0.9.1 // indirect
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect
- github.com/prometheus/client_model v0.6.1 // indirect
- github.com/prometheus/common v0.62.0 // indirect
+ github.com/prometheus/client_model v0.6.2 // indirect
+ github.com/prometheus/common v0.66.1 // indirect
github.com/prometheus/procfs v0.16.1 // indirect
github.com/rogpeppe/go-internal v1.14.1 // indirect
- github.com/sagikazarmark/locafero v0.9.0 // indirect
- github.com/segmentio/asm v1.2.0 // indirect
+ github.com/sagikazarmark/locafero v0.12.0 // indirect
+ github.com/sanity-io/litter v1.5.8 // indirect
+ github.com/segmentio/asm v1.2.1 // indirect
github.com/sethvargo/go-retry v0.3.0 // indirect
- github.com/sourcegraph/conc v0.3.0 // indirect
- github.com/spf13/afero v1.14.0 // indirect
- github.com/spf13/cast v1.8.0 // indirect
- github.com/spf13/pflag v1.0.6 // indirect
+ github.com/sosodev/duration v1.3.1 // indirect
+ github.com/spf13/afero v1.15.0 // indirect
+ github.com/spf13/cast v1.10.0 // indirect
+ github.com/spf13/pflag v1.0.10 // indirect
github.com/stretchr/objx v0.5.2 // indirect
github.com/subosito/gotenv v1.6.0 // indirect
github.com/zeebo/xxh3 v1.0.2 // indirect
- go.uber.org/automaxprocs v1.6.0 // indirect
go.uber.org/multierr v1.11.0 // indirect
- golang.org/x/crypto v0.38.0 // indirect
- golang.org/x/mod v0.24.0 // indirect
- golang.org/x/tools v0.33.0 // indirect
- google.golang.org/protobuf v1.36.6 // indirect
+ go.yaml.in/yaml/v2 v2.4.2 // indirect
+ go.yaml.in/yaml/v3 v3.0.4 // indirect
+ golang.org/x/crypto v0.45.0 // indirect
+ golang.org/x/mod v0.30.0 // indirect
+ golang.org/x/telemetry v0.0.0-20251111182119-bc8e575c7b54 // indirect
+ golang.org/x/tools v0.39.0 // indirect
gopkg.in/ini.v1 v1.67.0 // indirect
gopkg.in/natefinch/npipe.v2 v2.0.0-20160621034901-c1b8fa8bdcce // indirect
)
tool (
+ github.com/atombender/go-jsonschema
github.com/cespare/reflex
github.com/google/wire/cmd/wire
github.com/onsi/ginkgo/v2/ginkgo
diff --git a/go.sum b/go.sum
index d8a1a8c45..77c0cbb40 100644
--- a/go.sum
+++ b/go.sum
@@ -1,17 +1,23 @@
+dario.cat/mergo v1.0.2 h1:85+piFYR1tMbRrLcDwR18y4UKJ3aH1Tbzi24VRW1TK8=
+dario.cat/mergo v1.0.2/go.mod h1:E/hbnu0NxMFBjpMIE34DRGLWqDy0g5FuKDhCb31ngxA=
filippo.io/edwards25519 v1.1.0 h1:FNf4tywRC1HmFuKW5xopWpigGjJKiJSV0Cqo0cJWDaA=
filippo.io/edwards25519 v1.1.0/go.mod h1:BxyFTGdWcka3PhytdK4V28tE5sGfRvvvRV7EaN4VDT4=
+github.com/Masterminds/semver/v3 v3.4.0 h1:Zog+i5UMtVoCU8oKka5P7i9q9HgrJeGzI9SA1Xbatp0=
+github.com/Masterminds/semver/v3 v3.4.0/go.mod h1:4V+yj/TJE1HU9XfppCwVMZq3I84lprf4nC11bSS5beM=
github.com/Masterminds/squirrel v1.5.4 h1:uUcX/aBc8O7Fg9kaISIUsHXdKuqehiXAMQTYX8afzqM=
github.com/Masterminds/squirrel v1.5.4/go.mod h1:NNaOrjSoIDfDA40n7sr2tPNZRfjzjA400rg+riTZj10=
github.com/RaveNoX/go-jsoncommentstrip v1.0.0 h1:t527LHHE3HmiHrq74QMpNPZpGCIJzTx+apLkMKt4HC0=
github.com/RaveNoX/go-jsoncommentstrip v1.0.0/go.mod h1:78ihd09MekBnJnxpICcwzCMzGrKSKYe4AqU6PDYYpjk=
github.com/andybalholm/cascadia v1.3.3 h1:AG2YHrzJIm4BZ19iwJ/DAua6Btl3IwJX+VI4kktS1LM=
github.com/andybalholm/cascadia v1.3.3/go.mod h1:xNd9bqTn98Ln4DwST8/nG+H0yuB8Hmgu1YHNnWw0GeA=
+github.com/atombender/go-jsonschema v0.20.0 h1:AHg0LeI0HcjQ686ALwUNqVJjNRcSXpIR6U+wC2J0aFY=
+github.com/atombender/go-jsonschema v0.20.0/go.mod h1:ZmbuR11v2+cMM0PdP6ySxtyZEGFBmhgF4xa4J6Hdls8=
github.com/aymerick/douceur v0.2.0 h1:Mv+mAeH1Q+n9Fr+oyamOlAkUNPWPlA8PPGR0QAaYuPk=
github.com/aymerick/douceur v0.2.0/go.mod h1:wlT5vV2O3h55X9m7iVYN0TBM0NH/MmbLnd30/FjWUq4=
github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM=
github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw=
-github.com/bmatcuk/doublestar/v4 v4.8.1 h1:54Bopc5c2cAvhLRAzqOGCYHYyhcDHsFF4wWIR5wKP38=
-github.com/bmatcuk/doublestar/v4 v4.8.1/go.mod h1:xBQ8jztBU6kakFMg+8WGxn0c6z1fTSPVIjEY1Wr7jzc=
+github.com/bmatcuk/doublestar/v4 v4.9.1 h1:X8jg9rRZmJd4yRy7ZeNDRnM+T3ZfHv15JiBJ/avrEXE=
+github.com/bmatcuk/doublestar/v4 v4.9.1/go.mod h1:xBQ8jztBU6kakFMg+8WGxn0c6z1fTSPVIjEY1Wr7jzc=
github.com/bradleyjkemp/cupaloy/v2 v2.8.0 h1:any4BmKE+jGIaMpnU8YgH/I2LPiLBufr6oMMlVBbn9M=
github.com/bradleyjkemp/cupaloy/v2 v2.8.0/go.mod h1:bm7JXdkRd4BHJk9HpwqAI8BoAY1lps46Enkdqw6aRX0=
github.com/cespare/reflex v0.3.1 h1:N4Y/UmRrjwOkNT0oQQnYsdr6YBxvHqtSfPB4mqOyAKk=
@@ -22,6 +28,7 @@ github.com/cpuguy83/go-md2man/v2 v2.0.6/go.mod h1:oOW0eioCTA6cOiMLiUPZOpcVxMig6N
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
github.com/creack/pty v1.1.11 h1:07n33Z8lZxZ2qwegKbObQohDhXDQxiMMz1NOUGYlesw=
github.com/creack/pty v1.1.11/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
+github.com/davecgh/go-spew v0.0.0-20161028175848-04cdfd42973b/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM=
@@ -55,48 +62,57 @@ github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7z
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
github.com/fsnotify/fsnotify v1.9.0 h1:2Ml+OJNzbYCTzsxtv8vKSFD9PbJjmhYF14k/jKC7S9k=
github.com/fsnotify/fsnotify v1.9.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0=
-github.com/go-chi/chi/v5 v5.2.1 h1:KOIHODQj58PmL80G2Eak4WdvUzjSJSm0vG72crDCqb8=
-github.com/go-chi/chi/v5 v5.2.1/go.mod h1:L2yAIGWB3H+phAw1NxKwWM+7eUH/lU8pOMm5hHcoops=
-github.com/go-chi/cors v1.2.1 h1:xEC8UT3Rlp2QuWNEr4Fs/c2EAGVKBwy/1vHx3bppil4=
-github.com/go-chi/cors v1.2.1/go.mod h1:sSbTewc+6wYHBBCW7ytsFSn836hqM7JxpglAy2Vzc58=
+github.com/gkampitakis/ciinfo v0.3.2 h1:JcuOPk8ZU7nZQjdUhctuhQofk7BGHuIy0c9Ez8BNhXs=
+github.com/gkampitakis/ciinfo v0.3.2/go.mod h1:1NIwaOcFChN4fa/B0hEBdAb6npDlFL8Bwx4dfRLRqAo=
+github.com/gkampitakis/go-diff v1.3.2 h1:Qyn0J9XJSDTgnsgHRdz9Zp24RaJeKMUHg2+PDZZdC4M=
+github.com/gkampitakis/go-diff v1.3.2/go.mod h1:LLgOrpqleQe26cte8s36HTWcTmMEur6OPYerdAAS9tk=
+github.com/gkampitakis/go-snaps v0.5.15 h1:amyJrvM1D33cPHwVrjo9jQxX8g/7E2wYdZ+01KS3zGE=
+github.com/gkampitakis/go-snaps v0.5.15/go.mod h1:HNpx/9GoKisdhw9AFOBT1N7DBs9DiHo/hGheFGBZ+mc=
+github.com/go-chi/chi/v5 v5.2.3 h1:WQIt9uxdsAbgIYgid+BpYc+liqQZGMHRaUwp0JUcvdE=
+github.com/go-chi/chi/v5 v5.2.3/go.mod h1:L2yAIGWB3H+phAw1NxKwWM+7eUH/lU8pOMm5hHcoops=
+github.com/go-chi/cors v1.2.2 h1:Jmey33TE+b+rB7fT8MUy1u0I4L+NARQlK6LhzKPSyQE=
+github.com/go-chi/cors v1.2.2/go.mod h1:sSbTewc+6wYHBBCW7ytsFSn836hqM7JxpglAy2Vzc58=
github.com/go-chi/httprate v0.15.0 h1:j54xcWV9KGmPf/X4H32/aTH+wBlrvxL7P+SdnRqxh5g=
github.com/go-chi/httprate v0.15.0/go.mod h1:rzGHhVrsBn3IMLYDOZQsSU4fJNWcjui4fWKJcCId1R4=
github.com/go-chi/jwtauth/v5 v5.3.3 h1:50Uzmacu35/ZP9ER2Ht6SazwPsnLQ9LRJy6zTZJpHEo=
github.com/go-chi/jwtauth/v5 v5.3.3/go.mod h1:O4QvPRuZLZghl9WvfVaON+ARfGzpD2PBX/QY5vUz7aQ=
-github.com/go-logr/logr v1.4.2 h1:6pFjapn8bFcIbiKo3XT4j/BhANplGihG6tvd+8rYgrY=
-github.com/go-logr/logr v1.4.2/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
+github.com/go-logr/logr v1.4.3 h1:CjnDlHq8ikf6E492q6eKboGOC0T8CDaOvkHCIg8idEI=
+github.com/go-logr/logr v1.4.3/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
github.com/go-sql-driver/mysql v1.4.1/go.mod h1:zAC/RDZ24gD3HViQzih4MyKcchzm+sOG5ZlKdlhCg5w=
-github.com/go-sql-driver/mysql v1.9.2 h1:4cNKDYQ1I84SXslGddlsrMhc8k4LeDVj6Ad6WRjiHuU=
-github.com/go-sql-driver/mysql v1.9.2/go.mod h1:qn46aNg1333BRMNU69Lq93t8du/dwxI64Gl8i5p1WMU=
+github.com/go-sql-driver/mysql v1.9.3 h1:U/N249h2WzJ3Ukj8SowVFjdtZKfu9vlLZxjPXV1aweo=
+github.com/go-sql-driver/mysql v1.9.3/go.mod h1:qn46aNg1333BRMNU69Lq93t8du/dwxI64Gl8i5p1WMU=
github.com/go-task/slim-sprig/v3 v3.0.0 h1:sUs3vkvUymDpBKi3qH1YSqBQk9+9D/8M2mN1vB6EwHI=
github.com/go-task/slim-sprig/v3 v3.0.0/go.mod h1:W848ghGpv3Qj3dhTPRyJypKRiqCdHZiAzKg9hl15HA8=
github.com/go-viper/encoding/ini v0.1.1 h1:MVWY7B2XNw7lnOqHutGRc97bF3rP7omOdgjdMPAJgbs=
github.com/go-viper/encoding/ini v0.1.1/go.mod h1:Pfi4M2V1eAGJVZ5q6FrkHPhtHED2YgLlXhvgMVrB+YQ=
-github.com/go-viper/mapstructure/v2 v2.2.1 h1:ZAaOCxANMuZx5RCeg0mBdEZk7DZasvvZIxtHqx8aGss=
-github.com/go-viper/mapstructure/v2 v2.2.1/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM=
+github.com/go-viper/mapstructure/v2 v2.4.0 h1:EBsztssimR/CONLSZZ04E8qAkxNYq4Qp9LvH92wZUgs=
+github.com/go-viper/mapstructure/v2 v2.4.0/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM=
github.com/goccy/go-json v0.10.5 h1:Fq85nIqj+gXn/S5ahsiTlK3TmC85qgirsdTP/+DeaC4=
github.com/goccy/go-json v0.10.5/go.mod h1:oq7eo15ShAhp70Anwd5lgX2pLfOS3QCiwU/PULtXL6M=
-github.com/gohugoio/hashstructure v0.5.0 h1:G2fjSBU36RdwEJBWJ+919ERvOVqAg9tfcYp47K9swqg=
-github.com/gohugoio/hashstructure v0.5.0/go.mod h1:Ser0TniXuu/eauYmrwM4o64EBvySxNzITEOLlm4igec=
+github.com/goccy/go-yaml v1.18.0 h1:8W7wMFS12Pcas7KU+VVkaiCng+kG8QiFeFwzFb+rwuw=
+github.com/goccy/go-yaml v1.18.0/go.mod h1:XBurs7gK8ATbW4ZPGKgcbrY1Br56PdM69F7LkFRi1kA=
+github.com/gohugoio/hashstructure v0.6.0 h1:7wMB/2CfXoThFYhdWRGv3u3rUM761Cq29CxUW+NltUg=
+github.com/gohugoio/hashstructure v0.6.0/go.mod h1:lapVLk9XidheHG1IQ4ZSbyYrXcaILU1ZEP/+vno5rBQ=
github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
-github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
github.com/google/go-cmp v0.7.0 h1:wk8382ETsv4JYUZwIsn6YpYiWiBsYLSJiTsyBybVuN8=
github.com/google/go-cmp v0.7.0/go.mod h1:pXiqmnSA92OHEEa9HXL2W4E7lf9JzCmGVUdgjX3N/iU=
github.com/google/go-pipeline v0.0.0-20230411140531-6cbedfc1d3fc h1:hd+uUVsB1vdxohPneMrhGH2YfQuH5hRIK9u4/XCeUtw=
github.com/google/go-pipeline v0.0.0-20230411140531-6cbedfc1d3fc/go.mod h1:SL66SJVysrh7YbDCP9tH30b8a9o/N2HeiQNUm85EKhc=
-github.com/google/pprof v0.0.0-20250501235452-c0086092b71a h1:rDA3FfmxwXR+BVKKdz55WwMJ1pD2hJQNW31d+l3mPk4=
-github.com/google/pprof v0.0.0-20250501235452-c0086092b71a/go.mod h1:5hDyRhoBCxViHszMt12TnOpEI4VVi+U8Gm9iphldiMA=
+github.com/google/pprof v0.0.0-20251114195745-4902fdda35c8 h1:3DsUAV+VNEQa2CUVLxCY3f87278uWfIDhJnbdvDjvmE=
+github.com/google/pprof v0.0.0-20251114195745-4902fdda35c8/go.mod h1:I6V7YzU0XDpsHqbsyrghnFZLO1gwK6NPTNvmetQIk9U=
github.com/google/subcommands v1.2.0 h1:vWQspBTo2nEqTUFita5/KeEWlUL8kQObDFbub/EN9oE=
github.com/google/subcommands v1.2.0/go.mod h1:ZjhPrFU+Olkh9WazFPsl27BQ4UPiG37m3yTrtFlrHVk=
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
-github.com/google/wire v0.6.0 h1:HBkoIh4BdSxoyo9PveV8giw7ZsaBOvzWKfcg/6MrVwI=
-github.com/google/wire v0.6.0/go.mod h1:F4QhpQ9EDIdJ1Mbop/NZBRB+5yrR6qg3BnctaoUk6NA=
+github.com/google/wire v0.7.0 h1:JxUKI6+CVBgCO2WToKy/nQk0sS+amI9z9EjVmdaocj4=
+github.com/google/wire v0.7.0/go.mod h1:n6YbUQD9cPKTnHXEBN2DXlOp/mVADhVErcMFb0v3J18=
github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1 h1:EGx4pi6eqNxGaHF6qqu48+N2wcFQ5qg5FXgOdqsJ5d8=
github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY=
github.com/gorilla/css v1.0.1 h1:ntNaBIghp6JmvWnxbZKANoLyuXTPZ4cAMlo6RyhlbO8=
github.com/gorilla/css v1.0.1/go.mod h1:BvnYkspnSzMmwRK+b8/xgNPLiIuNZr6vbZBTPQ2A3b0=
+github.com/gorilla/websocket v1.5.3 h1:saDtZ6Pbx/0u+bgYQ3q96pZgCzfhKXGPqt7kZ72aNNg=
+github.com/gorilla/websocket v1.5.3/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
github.com/hashicorp/errwrap v1.1.0 h1:OxrOeh75EUXMY8TBjag2fzXGZ40LB6IKw45YeGUDY2I=
github.com/hashicorp/errwrap v1.1.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
@@ -104,18 +120,22 @@ github.com/hashicorp/go-multierror v1.1.1 h1:H5DkEtf6CXdFp0N0Em5UCwQpXMWke8IA0+l
github.com/hashicorp/go-multierror v1.1.1/go.mod h1:iw975J/qwKPdAO1clOe2L8331t/9/fmwbPZ6JB6eMoM=
github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8=
github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw=
-github.com/jellydator/ttlcache/v3 v3.3.0 h1:BdoC9cE81qXfrxeb9eoJi9dWrdhSuwXMAnHTbnBm4Wc=
-github.com/jellydator/ttlcache/v3 v3.3.0/go.mod h1:bj2/e0l4jRnQdrnSTaGTsh4GSXvMjQcy41i7th0GVGw=
+github.com/jellydator/ttlcache/v3 v3.4.0 h1:YS4P125qQS0tNhtL6aeYkheEaB/m8HCqdMMP4mnWdTY=
+github.com/jellydator/ttlcache/v3 v3.4.0/go.mod h1:Hw9EgjymziQD3yGsQdf1FqFdpp7YjFMd4Srg5EJlgD4=
+github.com/joshdk/go-junit v1.0.0 h1:S86cUKIdwBHWwA6xCmFlf3RTLfVXYQfvanM5Uh+K6GE=
+github.com/joshdk/go-junit v1.0.0/go.mod h1:TiiV0PqkaNfFXjEiyjWM3XXrhVyCa1K4Zfga6W52ung=
github.com/jtolds/gls v4.20.0+incompatible h1:xdiiI2gbIgH/gLH7ADydsJ1uDOEzR8yvV7C0MuV77Wo=
github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU=
-github.com/kardianos/service v1.2.2 h1:ZvePhAHfvo0A7Mftk/tEzqEZ7Q4lgnR8sGz4xu1YX60=
-github.com/kardianos/service v1.2.2/go.mod h1:CIMRFEJVL+0DS1a3Nx06NaMn4Dz63Ng6O7dl0qH0zVM=
+github.com/kardianos/service v1.2.4 h1:XNlGtZOYNx2u91urOdg/Kfmc+gfmuIo1Dd3rEi2OgBk=
+github.com/kardianos/service v1.2.4/go.mod h1:E4V9ufUuY82F7Ztlu1eN9VXWIQxg8NoLQlmFe0MtrXc=
github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51 h1:Z9n2FFNUXsshfwJMBgNA0RU6/i7WVaAegv3PtuIHPMs=
github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51/go.mod h1:CzGEWj7cYgsdH8dAjBGEr58BoE7ScuLd+fwFZ44+/x8=
github.com/klauspost/compress v1.18.0 h1:c/Cqfb0r+Yi+JtIEq73FWXVkRonBlf0CRNYc8Zttxdo=
github.com/klauspost/compress v1.18.0/go.mod h1:2Pp+KzxcywXVXMr50+X0Q/Lsb43OQHYWRCY2AiWywWQ=
-github.com/klauspost/cpuid/v2 v2.2.10 h1:tBs3QSyvjDyFTq3uoc/9xFpCuOsJQFNPiAhYdw2skhE=
-github.com/klauspost/cpuid/v2 v2.2.10/go.mod h1:hqwkgyIinND0mEev00jJYCxPNVRVXFQeu1XKlok6oO0=
+github.com/klauspost/cpuid/v2 v2.3.0 h1:S4CRMLnYUhGeDFDqkGriYKdfoFlDnMtqTiI/sFzhA9Y=
+github.com/klauspost/cpuid/v2 v2.3.0/go.mod h1:hqwkgyIinND0mEev00jJYCxPNVRVXFQeu1XKlok6oO0=
+github.com/knqyf263/go-plugin v0.9.0 h1:CQs2+lOPIlkZVtcb835ZYDEoyyWJWLbSTWeCs0EwTwI=
+github.com/knqyf263/go-plugin v0.9.0/go.mod h1:2z5lCO1/pez6qGo8CvCxSlBFSEat4MEp1DrnA+f7w8Q=
github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE=
@@ -130,8 +150,8 @@ github.com/lann/builder v0.0.0-20180802200727-47ae307949d0 h1:SOEGU9fKiNWd/HOJuq
github.com/lann/builder v0.0.0-20180802200727-47ae307949d0/go.mod h1:dXGbAdH5GtBTC4WfIxhKZfyBF/HBFgRZSWwZ9g/He9o=
github.com/lann/ps v0.0.0-20150810152359-62de8c46ede0 h1:P6pPBnrTSX3DEVR4fDembhRWSsG5rVo6hYhAB/ADZrk=
github.com/lann/ps v0.0.0-20150810152359-62de8c46ede0/go.mod h1:vmVJ0l/dxyfGW6FmdpVm2joNMFikkuWg0EoCKLGUMNw=
-github.com/lestrrat-go/blackmagic v1.0.3 h1:94HXkVLxkZO9vJI/w2u1T0DAoprShFd13xtnSINtDWs=
-github.com/lestrrat-go/blackmagic v1.0.3/go.mod h1:6AWFyKNNj0zEXQYfTMPfZrAXUWUfTIZ5ECEUEJaijtw=
+github.com/lestrrat-go/blackmagic v1.0.4 h1:IwQibdnf8l2KoO+qC3uT4OaTWsW7tuRQXy9TRN9QanA=
+github.com/lestrrat-go/blackmagic v1.0.4/go.mod h1:6AWFyKNNj0zEXQYfTMPfZrAXUWUfTIZ5ECEUEJaijtw=
github.com/lestrrat-go/httpcc v1.0.1 h1:ydWCStUeJLkpYyjLDHihupbn2tYmZ7m22BGkcvZZrIE=
github.com/lestrrat-go/httpcc v1.0.1/go.mod h1:qiltp3Mt56+55GPVCbTdM9MlqhvzyuL6W/NMDA8vA5E=
github.com/lestrrat-go/httprc v1.0.6 h1:qgmgIRhpvBqexMJjA/PmwSvhNk679oqD1RbovdCGW8k=
@@ -142,46 +162,53 @@ github.com/lestrrat-go/jwx/v2 v2.1.6 h1:hxM1gfDILk/l5ylers6BX/Eq1m/pnxe9NBwW6lVf
github.com/lestrrat-go/jwx/v2 v2.1.6/go.mod h1:Y722kU5r/8mV7fYDifjug0r8FK8mZdw0K0GpJw/l8pU=
github.com/lestrrat-go/option v1.0.1 h1:oAzP2fvZGQKWkvHa1/SAcFolBEca1oN+mQ7eooNBEYU=
github.com/lestrrat-go/option v1.0.1/go.mod h1:5ZHFbivi4xwXxhxY9XHDe2FHo6/Z7WWmtT7T5nBBp3I=
+github.com/maruel/natural v1.2.1 h1:G/y4pwtTA07lbQsMefvsmEO0VN0NfqpxprxXDM4R/4o=
+github.com/maruel/natural v1.2.1/go.mod h1:v+Rfd79xlw1AgVBjbO0BEQmptqb5HvL/k9GRHB7ZKEg=
github.com/matoous/go-nanoid/v2 v2.1.0 h1:P64+dmq21hhWdtvZfEAofnvJULaRR1Yib0+PnU669bE=
github.com/matoous/go-nanoid/v2 v2.1.0/go.mod h1:KlbGNQ+FhrUNIHUxZdL63t7tl4LaPkZNpUULS8H4uVM=
github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY=
github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
-github.com/mattn/go-sqlite3 v1.14.28 h1:ThEiQrnbtumT+QMknw63Befp/ce/nUPgBPMlRFEum7A=
-github.com/mattn/go-sqlite3 v1.14.28/go.mod h1:Uh1q+B4BYcTPb+yiD3kU8Ct7aC0hY9fxUwlHK0RXw+Y=
+github.com/mattn/go-sqlite3 v1.14.32 h1:JD12Ag3oLy1zQA+BNn74xRgaBbdhbNIDYvQUEuuErjs=
+github.com/mattn/go-sqlite3 v1.14.32/go.mod h1:Uh1q+B4BYcTPb+yiD3kU8Ct7aC0hY9fxUwlHK0RXw+Y=
github.com/mfridman/interpolate v0.0.2 h1:pnuTK7MQIxxFz1Gr+rjSIx9u7qVjf5VOoM/u6BbAxPY=
github.com/mfridman/interpolate v0.0.2/go.mod h1:p+7uk6oE07mpE/Ik1b8EckO0O4ZXiGAfshKBWLUM9Xg=
+github.com/mfridman/tparse v0.18.0 h1:wh6dzOKaIwkUGyKgOntDW4liXSo37qg5AXbIhkMV3vE=
+github.com/mfridman/tparse v0.18.0/go.mod h1:gEvqZTuCgEhPbYk/2lS3Kcxg1GmTxxU7kTC8DvP0i/A=
github.com/microcosm-cc/bluemonday v1.0.27 h1:MpEUotklkwCSLeH+Qdx1VJgNqLlpY2KXwXFM08ygZfk=
github.com/microcosm-cc/bluemonday v1.0.27/go.mod h1:jFi9vgW+H7c3V0lb6nR74Ib/DIB5OBs92Dimizgw2cA=
github.com/mileusna/useragent v1.3.5 h1:SJM5NzBmh/hO+4LGeATKpaEX9+b4vcGg2qXGLiNGDws=
github.com/mileusna/useragent v1.3.5/go.mod h1:3d8TOmwL/5I8pJjyVDteHtgDGcefrFUX4ccGOMKNYYc=
+github.com/mitchellh/go-wordwrap v1.0.1 h1:TLuKupo69TCn6TQSyGxwI1EblZZEsQ0vMlAFQflz0v0=
+github.com/mitchellh/go-wordwrap v1.0.1/go.mod h1:R62XHJLzvMFRBbcrT7m7WgmE1eOyTSsCt+hzestvNj0=
github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 h1:C3w9PqII01/Oq1c1nUAm88MOHcQC9l5mIlSMApZMrHA=
github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ=
github.com/ncruces/go-strftime v0.1.9 h1:bY0MQC28UADQmHmaF5dgpLmImcShSi2kHU9XLdhx/f4=
github.com/ncruces/go-strftime v0.1.9/go.mod h1:Fwc5htZGVVkseilnfgOVb9mKy6w1naJmn9CehxcKcls=
github.com/ogier/pflag v0.0.1 h1:RW6JSWSu/RkSatfcLtogGfFgpim5p7ARQ10ECk5O750=
github.com/ogier/pflag v0.0.1/go.mod h1:zkFki7tvTa0tafRvTBIZTvzYyAu6kQhPZFnshFFPE+g=
-github.com/onsi/ginkgo/v2 v2.23.4 h1:ktYTpKJAVZnDT4VjxSbiBenUjmlL/5QkBEocaWXiQus=
-github.com/onsi/ginkgo/v2 v2.23.4/go.mod h1:Bt66ApGPBFzHyR+JO10Zbt0Gsp4uWxu5mIOTusL46e8=
-github.com/onsi/gomega v1.37.0 h1:CdEG8g0S133B4OswTDC/5XPSzE1OeP29QOioj2PID2Y=
-github.com/onsi/gomega v1.37.0/go.mod h1:8D9+Txp43QWKhM24yyOBEdpkzN8FvJyAwecBgsU4KU0=
+github.com/onsi/ginkgo/v2 v2.27.2 h1:LzwLj0b89qtIy6SSASkzlNvX6WktqurSHwkk2ipF/Ns=
+github.com/onsi/ginkgo/v2 v2.27.2/go.mod h1:ArE1D/XhNXBXCBkKOLkbsb2c81dQHCRcF5zwn/ykDRo=
+github.com/onsi/gomega v1.38.2 h1:eZCjf2xjZAqe+LeWvKb5weQ+NcPwX84kqJ0cZNxok2A=
+github.com/onsi/gomega v1.38.2/go.mod h1:W2MJcYxRGV63b418Ai34Ud0hEdTVXq9NW9+Sx6uXf3k=
github.com/pelletier/go-toml/v2 v2.2.4 h1:mye9XuhQ6gvn5h28+VilKrrPoQVanw5PMw/TB0t5Ec4=
github.com/pelletier/go-toml/v2 v2.2.4/go.mod h1:2gIqNv+qfxSVS7cM2xJQKtLSTLUE9V8t9Stt+h56mCY=
github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA=
+github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
+github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
+github.com/pmezard/go-difflib v0.0.0-20151028094244-d8ed2627bdf0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRIccs7FGNTlIRMkT8wgtp5eCXdBlqhYGL6U=
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
github.com/pocketbase/dbx v1.11.0 h1:LpZezioMfT3K4tLrqA55wWFw1EtH1pM4tzSVa7kgszU=
github.com/pocketbase/dbx v1.11.0/go.mod h1:xXRCIAKTHMgUCyCKZm55pUOdvFziJjQfXaWKhu2vhMs=
-github.com/prashantv/gostub v1.1.0 h1:BTyx3RfQjRHnUWaGF9oQos79AlQ5k8WNktv7VGvVH4g=
-github.com/prashantv/gostub v1.1.0/go.mod h1:A5zLQHz7ieHGG7is6LLXLz7I8+3LZzsrV0P1IAHhP5U=
-github.com/pressly/goose/v3 v3.24.3 h1:DSWWNwwggVUsYZ0X2VitiAa9sKuqtBfe+Jr9zFGwWlM=
-github.com/pressly/goose/v3 v3.24.3/go.mod h1:v9zYL4xdViLHCUUJh/mhjnm6JrK7Eul8AS93IxiZM4E=
-github.com/prometheus/client_golang v1.22.0 h1:rb93p9lokFEsctTys46VnV1kLCDpVZ0a/Y92Vm0Zc6Q=
-github.com/prometheus/client_golang v1.22.0/go.mod h1:R7ljNsLXhuQXYZYtw6GAE9AZg8Y7vEW5scdCXrWRXC0=
-github.com/prometheus/client_model v0.6.1 h1:ZKSh/rekM+n3CeS952MLRAdFwIKqeY8b62p8ais2e9E=
-github.com/prometheus/client_model v0.6.1/go.mod h1:OrxVMOVHjw3lKMa8+x6HeMGkHMQyHDk9E3jmP2AmGiY=
-github.com/prometheus/common v0.62.0 h1:xasJaQlnWAeyHdUBeGjXmutelfJHWMRr+Fg4QszZ2Io=
-github.com/prometheus/common v0.62.0/go.mod h1:vyBcEuLSvWos9B1+CyL7JZ2up+uFzXhkqml0W5zIY1I=
+github.com/pressly/goose/v3 v3.26.0 h1:KJakav68jdH0WDvoAcj8+n61WqOIaPGgH0bJWS6jpmM=
+github.com/pressly/goose/v3 v3.26.0/go.mod h1:4hC1KrritdCxtuFsqgs1R4AU5bWtTAf+cnWvfhf2DNY=
+github.com/prometheus/client_golang v1.23.2 h1:Je96obch5RDVy3FDMndoUsjAhG5Edi49h0RJWRi/o0o=
+github.com/prometheus/client_golang v1.23.2/go.mod h1:Tb1a6LWHB3/SPIzCoaDXI4I8UHKeFTEQ1YCr+0Gyqmg=
+github.com/prometheus/client_model v0.6.2 h1:oBsgwpGs7iVziMvrGhE53c/GrLUsZdHnqNwqPLxwZyk=
+github.com/prometheus/client_model v0.6.2/go.mod h1:y3m2F6Gdpfy6Ut/GBsUqTWZqCUvMVzSfMLjcu6wAwpE=
+github.com/prometheus/common v0.66.1 h1:h5E0h5/Y8niHc5DlaLlWLArTQI7tMrsfQjHV+d9ZoGs=
+github.com/prometheus/common v0.66.1/go.mod h1:gcaUsgf3KfRSwHY4dIMXLPV0K/Wg1oZ8+SbZk/HH/dA=
github.com/prometheus/procfs v0.16.1 h1:hZ15bTNuirocR6u0JZ6BAHHmwS1p8B4P6MRqxtzMyRg=
github.com/prometheus/procfs v0.16.1/go.mod h1:teAbpZRB1iIAJYREa1LsoWUXykVXA1KlTmWl8x/U+Is=
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE=
@@ -196,10 +223,12 @@ github.com/rogpeppe/go-internal v1.14.1/go.mod h1:MaRKkUm5W0goXpeCfT7UZI6fk/L7L7
github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
github.com/sabhiram/go-gitignore v0.0.0-20210923224102-525f6e181f06 h1:OkMGxebDjyw0ULyrTYWeN0UNCCkmCWfjPnIA2W6oviI=
github.com/sabhiram/go-gitignore v0.0.0-20210923224102-525f6e181f06/go.mod h1:+ePHsJ1keEjQtpvf9HHw0f4ZeJ0TLRsxhunSI2hYJSs=
-github.com/sagikazarmark/locafero v0.9.0 h1:GbgQGNtTrEmddYDSAH9QLRyfAHY12md+8YFTqyMTC9k=
-github.com/sagikazarmark/locafero v0.9.0/go.mod h1:UBUyz37V+EdMS3hDF3QWIiVr/2dPrx49OMO0Bn0hJqk=
-github.com/segmentio/asm v1.2.0 h1:9BQrFxC+YOHJlTlHGkTrFWf59nbL3XnCoFLTwDCI7ys=
-github.com/segmentio/asm v1.2.0/go.mod h1:BqMnlJP91P8d+4ibuonYZw9mfnzI9HfxselHZr5aAcs=
+github.com/sagikazarmark/locafero v0.12.0 h1:/NQhBAkUb4+fH1jivKHWusDYFjMOOKU88eegjfxfHb4=
+github.com/sagikazarmark/locafero v0.12.0/go.mod h1:sZh36u/YSZ918v0Io+U9ogLYQJ9tLLBmM4eneO6WwsI=
+github.com/sanity-io/litter v1.5.8 h1:uM/2lKrWdGbRXDrIq08Lh9XtVYoeGtcQxk9rtQ7+rYg=
+github.com/sanity-io/litter v1.5.8/go.mod h1:9gzJgR2i4ZpjZHsKvUXIRQVk7P+yM3e+jAF7bU2UI5U=
+github.com/segmentio/asm v1.2.1 h1:DTNbBqs57ioxAD4PrArqftgypG4/qNpXoJx8TVXxPR0=
+github.com/segmentio/asm v1.2.1/go.mod h1:BqMnlJP91P8d+4ibuonYZw9mfnzI9HfxselHZr5aAcs=
github.com/sethvargo/go-retry v0.3.0 h1:EEt31A35QhrcRZtrYFDTBg91cqZVnFL2navjDrah2SE=
github.com/sethvargo/go-retry v0.3.0/go.mod h1:mNX17F0C/HguQMyMyJxcnU471gOZGxCLyYaFyAZraas=
github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE=
@@ -209,68 +238,80 @@ github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d h1:zE9ykE
github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc=
github.com/smartystreets/goconvey v1.6.4 h1:fv0U8FUIMPNf1L9lnHLvLhgicrIVChEkdzIKYqbNC9s=
github.com/smartystreets/goconvey v1.6.4/go.mod h1:syvi0/a8iFYH4r/RixwvyeAJjdLS9QV7WQ/tjFTllLA=
-github.com/sourcegraph/conc v0.3.0 h1:OQTbbt6P72L20UqAkXXuLOj79LfEanQ+YQFNpLA9ySo=
-github.com/sourcegraph/conc v0.3.0/go.mod h1:Sdozi7LEKbFPqYX2/J+iBAM6HpqSLTASQIKqDmF7Mt0=
-github.com/spf13/afero v1.14.0 h1:9tH6MapGnn/j0eb0yIXiLjERO8RB6xIVZRDCX7PtqWA=
-github.com/spf13/afero v1.14.0/go.mod h1:acJQ8t0ohCGuMN3O+Pv0V0hgMxNYDlvdk+VTfyZmbYo=
-github.com/spf13/cast v1.8.0 h1:gEN9K4b8Xws4EX0+a0reLmhq8moKn7ntRlQYgjPeCDk=
-github.com/spf13/cast v1.8.0/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo=
-github.com/spf13/cobra v1.9.1 h1:CXSaggrXdbHK9CF+8ywj8Amf7PBRmPCOJugH954Nnlo=
-github.com/spf13/cobra v1.9.1/go.mod h1:nDyEzZ8ogv936Cinf6g1RU9MRY64Ir93oCnqb9wxYW0=
-github.com/spf13/pflag v1.0.6 h1:jFzHGLGAlb3ruxLB8MhbI6A8+AQX/2eW4qeyNZXNp2o=
-github.com/spf13/pflag v1.0.6/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
-github.com/spf13/viper v1.20.1 h1:ZMi+z/lvLyPSCoNtFCpqjy0S4kPbirhpTMwl8BkW9X4=
-github.com/spf13/viper v1.20.1/go.mod h1:P9Mdzt1zoHIG8m2eZQinpiBjo6kCmZSKBClNNqjJvu4=
+github.com/sosodev/duration v1.3.1 h1:qtHBDMQ6lvMQsL15g4aopM4HEfOaYuhWBw3NPTtlqq4=
+github.com/sosodev/duration v1.3.1/go.mod h1:RQIBBX0+fMLc/D9+Jb/fwvVmo0eZvDDEERAikUR6SDg=
+github.com/spf13/afero v1.15.0 h1:b/YBCLWAJdFWJTN9cLhiXXcD7mzKn9Dm86dNnfyQw1I=
+github.com/spf13/afero v1.15.0/go.mod h1:NC2ByUVxtQs4b3sIUphxK0NioZnmxgyCrfzeuq8lxMg=
+github.com/spf13/cast v1.10.0 h1:h2x0u2shc1QuLHfxi+cTJvs30+ZAHOGRic8uyGTDWxY=
+github.com/spf13/cast v1.10.0/go.mod h1:jNfB8QC9IA6ZuY2ZjDp0KtFO2LZZlg4S/7bzP6qqeHo=
+github.com/spf13/cobra v1.10.1 h1:lJeBwCfmrnXthfAupyUTzJ/J4Nc1RsHC/mSRU2dll/s=
+github.com/spf13/cobra v1.10.1/go.mod h1:7SmJGaTHFVBY0jW4NXGluQoLvhqFQM+6XSKD+P4XaB0=
+github.com/spf13/pflag v1.0.9/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
+github.com/spf13/pflag v1.0.10 h1:4EBh2KAYBwaONj6b2Ye1GiHfwjqyROoF4RwYO+vPwFk=
+github.com/spf13/pflag v1.0.10/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
+github.com/spf13/viper v1.21.0 h1:x5S+0EU27Lbphp4UKm1C+1oQO+rKx36vfCoaVebLFSU=
+github.com/spf13/viper v1.21.0/go.mod h1:P0lhsswPGWD/1lZJ9ny3fYnVqxiegrlNrEmgLjbTCAY=
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/objx v0.5.2 h1:xuMeJ0Sdp5ZMRXx/aWO6RZxdr3beISkG5/G/aIRr3pY=
github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/8L+MA=
+github.com/stretchr/testify v0.0.0-20161117074351-18a02ba4a312/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
-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.11.1 h1:7s2iGBzp5EwR7/aIZr8ao5+dra3wiQyKjjFuvgVKu7U=
+github.com/stretchr/testify v1.11.1/go.mod h1:wZwfW3scLgRK+23gO65QZefKpKQRnfz6sD981Nm4B6U=
github.com/subosito/gotenv v1.6.0 h1:9NlTDc1FTs4qu0DDq7AEtTPNw6SVm7uBMsUCUjABIf8=
github.com/subosito/gotenv v1.6.0/go.mod h1:Dk4QP5c2W3ibzajGcXpNraDfq2IrhjMIvMSWPKKo0FU=
+github.com/tetratelabs/wazero v1.10.1 h1:2DugeJf6VVk58KTPszlNfeeN8AhhpwcZqkJj2wwFuH8=
+github.com/tetratelabs/wazero v1.10.1/go.mod h1:DRm5twOQ5Gr1AoEdSi0CLjDQF1J9ZAuyqFIjl1KKfQU=
+github.com/tidwall/gjson v1.18.0 h1:FIDeeyB800efLX89e5a8Y0BNH+LOngJyGrIWxG2FKQY=
+github.com/tidwall/gjson v1.18.0/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk=
+github.com/tidwall/match v1.1.1 h1:+Ho715JplO36QYgwN9PGYNhgZvoUSc9X2c80KVTi+GA=
+github.com/tidwall/match v1.1.1/go.mod h1:eRSPERbgtNPcGhD8UCthc6PmLEQXEWd3PRB5JTxsfmM=
+github.com/tidwall/pretty v1.2.1 h1:qjsOFOWWQl+N3RsoF5/ssm1pHmJJwhjlSbZ51I6wMl4=
+github.com/tidwall/pretty v1.2.1/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU=
+github.com/tidwall/sjson v1.2.5 h1:kLy8mja+1c9jlljvWTlSazM7cKDRfJuR/bOJhcY5NcY=
+github.com/tidwall/sjson v1.2.5/go.mod h1:Fvgq9kS/6ociJEDnK0Fk1cpYF4FIW6ZF7LAe+6jwd28=
github.com/unrolled/secure v1.17.0 h1:Io7ifFgo99Bnh0J7+Q+qcMzWM6kaDPCA5FroFZEdbWU=
github.com/unrolled/secure v1.17.0/go.mod h1:BmF5hyM6tXczk3MpQkFf1hpKSRqCyhqcbiQtiAF7+40=
-github.com/xrash/smetrics v0.0.0-20240521201337-686a1a2994c1 h1:gEOO8jv9F4OT7lGCjxCBTO/36wtF6j2nSip77qHd4x4=
-github.com/xrash/smetrics v0.0.0-20240521201337-686a1a2994c1/go.mod h1:Ohn+xnUBiLI6FVj/9LpzZWtj1/D6lUovWYBkxHVV3aM=
+github.com/xrash/smetrics v0.0.0-20250705151800-55b8f293f342 h1:FnBeRrxr7OU4VvAzt5X7s6266i6cSVkkFPS0TuXWbIg=
+github.com/xrash/smetrics v0.0.0-20250705151800-55b8f293f342/go.mod h1:Ohn+xnUBiLI6FVj/9LpzZWtj1/D6lUovWYBkxHVV3aM=
github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
github.com/zeebo/assert v1.3.0 h1:g7C04CbJuIDKNPFHmsk4hwZDO5O+kntRxzaUoNXj+IQ=
github.com/zeebo/assert v1.3.0/go.mod h1:Pq9JiuJQpG8JLJdtkwrJESF0Foym2/D9XMU5ciN/wJ0=
github.com/zeebo/xxh3 v1.0.2 h1:xZmwmqxHZA8AI603jOQ0tMqmBr9lPeFwGg6d+xy9DC0=
github.com/zeebo/xxh3 v1.0.2/go.mod h1:5NWz9Sef7zIDm2JHfFlcQvNekmcEl9ekUZQQKCYaDcA=
-go.uber.org/automaxprocs v1.6.0 h1:O3y2/QNTOdbF+e/dpXNNW7Rx2hZ4sTIPyybbxyNqTUs=
-go.uber.org/automaxprocs v1.6.0/go.mod h1:ifeIMSnPZuznNm6jmdzmU3/bfk01Fe2fotchwEFJ8r8=
go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto=
go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE=
go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
+go.yaml.in/yaml/v2 v2.4.2 h1:DzmwEr2rDGHl7lsFgAHxmNz/1NlQ7xLIrlN2h5d1eGI=
+go.yaml.in/yaml/v2 v2.4.2/go.mod h1:081UH+NErpNdqlCXm3TtEran0rJZGxAYx9hb/ELlsPU=
+go.yaml.in/yaml/v3 v3.0.4 h1:tfq32ie2Jv2UxXFdLJdh3jXuOzWiL1fo0bu/FbuKpbc=
+go.yaml.in/yaml/v3 v3.0.4/go.mod h1:DhzuOOF2ATzADvBadXxruRBLzYTpT36CKvDb3+aBEFg=
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
golang.org/x/crypto v0.13.0/go.mod h1:y6Z2r+Rw4iayiXXAIxJIDAJ1zMW4yaTpebo8fPOliYc=
-golang.org/x/crypto v0.18.0/go.mod h1:R0j02AL6hcrfOiy9T4ZYp/rcWeMxM3L6QYxlOuEG1mg=
golang.org/x/crypto v0.19.0/go.mod h1:Iy9bg/ha4yyC70EfRS8jz+B6ybOBKMaSxLj6P6oBDfU=
golang.org/x/crypto v0.23.0/go.mod h1:CKFgDieR+mRhux2Lsu27y0fO304Db0wZe70UKqHu0v8=
golang.org/x/crypto v0.31.0/go.mod h1:kDsLvtWBEx7MV9tJOj9bnXsPbxwJQ6csT/x4KIN4Ssk=
-golang.org/x/crypto v0.38.0 h1:jt+WWG8IZlBnVbomuhg2Mdq0+BBQaHbtqHEFEigjUV8=
-golang.org/x/crypto v0.38.0/go.mod h1:MvrbAqul58NNYPKnOra203SB9vpuZW0e+RRZV+Ggqjw=
-golang.org/x/exp v0.0.0-20250506013437-ce4c2cf36ca6 h1:y5zboxd6LQAqYIhHnB48p0ByQ/GnQx2BE33L8BOHQkI=
-golang.org/x/exp v0.0.0-20250506013437-ce4c2cf36ca6/go.mod h1:U6Lno4MTRCDY+Ba7aCcauB9T60gsv5s4ralQzP72ZoQ=
+golang.org/x/crypto v0.45.0 h1:jMBrvKuj23MTlT0bQEOBcAE0mjg8mK9RXFhRH6nyF3Q=
+golang.org/x/crypto v0.45.0/go.mod h1:XTGrrkGJve7CYK7J8PEww4aY7gM3qMCElcJQ8n8JdX4=
+golang.org/x/exp v0.0.0-20251113190631-e25ba8c21ef6 h1:zfMcR1Cs4KNuomFFgGefv5N0czO2XZpUbxGUy8i8ug0=
+golang.org/x/exp v0.0.0-20251113190631-e25ba8c21ef6/go.mod h1:46edojNIoXTNOhySWIWdix628clX9ODXwPsQuG6hsK0=
golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0=
-golang.org/x/image v0.27.0 h1:C8gA4oWU/tKkdCfYT6T2u4faJu3MeNS5O8UPWlPF61w=
-golang.org/x/image v0.27.0/go.mod h1:xbdrClrAUway1MUTEZDq9mz/UpRwYAkFFNUslZtcB+g=
+golang.org/x/image v0.33.0 h1:LXRZRnv1+zGd5XBUVRFmYEphyyKJjQjCRiOuAP3sZfQ=
+golang.org/x/image v0.33.0/go.mod h1:DD3OsTYT9chzuzTQt+zMcOlBHgfoKQb1gry8p76Y1sc=
golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
golang.org/x/mod v0.12.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
-golang.org/x/mod v0.14.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
golang.org/x/mod v0.15.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
golang.org/x/mod v0.17.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
-golang.org/x/mod v0.24.0 h1:ZfthKaKaT4NrhGVZHO1/WDTwGES4De8KtWO0SIbNJMU=
-golang.org/x/mod v0.24.0/go.mod h1:IXM97Txy2VM4PJ3gI61r1YEk/gAj6zAHN3AdZt6S9Ww=
+golang.org/x/mod v0.30.0 h1:fDEXFVZ/fmCKProc/yAXXUijritrDzahmwwefnjoPFk=
+golang.org/x/mod v0.30.0/go.mod h1:lAsf5O2EvJeSFMiBxXDki7sCgAxEUcZHXoXMKT4GJKc=
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks=
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
@@ -279,12 +320,11 @@ golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug
golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg=
golang.org/x/net v0.15.0/go.mod h1:idbUs1IY1+zTqbi8yxTbhexhEEk5ur9LInksu6HrEpk=
-golang.org/x/net v0.20.0/go.mod h1:z8BVo6PvndSri0LbOE3hAn0apkU+1YvI6E70E9jsnvY=
golang.org/x/net v0.21.0/go.mod h1:bIjVDfnllIU7BJ2DNgfnXvpSvtn8VRwhlsaeUTyUS44=
golang.org/x/net v0.25.0/go.mod h1:JkAGAh7GEvH74S6FOH42FLoXpXbE/aqXSrIQjXgsiwM=
golang.org/x/net v0.33.0/go.mod h1:HXLR5J+9DxmrqMwG9qjGCxZ+zKXxBru04zlTvWlWuN4=
-golang.org/x/net v0.40.0 h1:79Xs7wF06Gbdcg4kdCCIQArK11Z1hr5POQ6+fIYHNuY=
-golang.org/x/net v0.40.0/go.mod h1:y0hY0exeL2Pku80/zKK7tpntoX23cqL3Oa6njdgRtds=
+golang.org/x/net v0.47.0 h1:Mx+4dIFzqraBXUugkia1OOvlD6LemFo1ALMHjrXDOhY=
+golang.org/x/net v0.47.0/go.mod h1:/jNxtkgq5yWUGYkaZGqo27cfGZ1c5Nen03aYrrKpVRU=
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
@@ -292,13 +332,12 @@ golang.org/x/sync v0.3.0/go.mod h1:FU7BRWz2tNW+3quACPkgCx/L+uEAv1htQ0V83Z9Rj+Y=
golang.org/x/sync v0.6.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
golang.org/x/sync v0.7.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
golang.org/x/sync v0.10.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
-golang.org/x/sync v0.14.0 h1:woo0S4Yywslg6hp4eUFjTVOyKt0RookbpAHG4c1HmhQ=
-golang.org/x/sync v0.14.0/go.mod h1:1dzgHSNfp02xaA81J2MS99Qcpr2w7fw1gpm99rleRqA=
+golang.org/x/sync v0.18.0 h1:kr88TuHDroi+UVf+0hZnirlk8o8T+4MrK6mr60WkH/I=
+golang.org/x/sync v0.18.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI=
golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20180926160741-c2ed4eda69e7/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20201015000850-e3ed0017c211/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
@@ -308,19 +347,19 @@ golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBc
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
-golang.org/x/sys v0.16.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
golang.org/x/sys v0.17.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
golang.org/x/sys v0.20.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
golang.org/x/sys v0.28.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
-golang.org/x/sys v0.33.0 h1:q3i8TbbEz+JRD9ywIRlyRAQbM0qF7hu24q3teo2hbuw=
-golang.org/x/sys v0.33.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k=
+golang.org/x/sys v0.38.0 h1:3yZWxaJjBmCWXqhN1qh02AkOnCQ1poK6oF+a7xWL6Gc=
+golang.org/x/sys v0.38.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
golang.org/x/telemetry v0.0.0-20240228155512-f48c80bd79b2/go.mod h1:TeRTkGYfJXctD9OcfyVLyj2J3IxLnKwHJR8f4D8a3YE=
+golang.org/x/telemetry v0.0.0-20251111182119-bc8e575c7b54 h1:E2/AqCUMZGgd73TQkxUMcMla25GB9i/5HOdLr+uH7Vo=
+golang.org/x/telemetry v0.0.0-20251111182119-bc8e575c7b54/go.mod h1:hKdjCMrbv9skySur+Nek8Hd0uJ0GuxJIoIX2payrIdQ=
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo=
golang.org/x/term v0.12.0/go.mod h1:owVbMEjm3cBLCHdkQu9b1opXd4ETQWc3BhuQGKgXgvU=
-golang.org/x/term v0.16.0/go.mod h1:yn7UURbUtPyrVJPGPq404EukNFxcm/foM+bV/bfcDsY=
golang.org/x/term v0.17.0/go.mod h1:lLRBjIVuehSbZlaOtGMbcMncT+aqLLLmKrsjNrUguwk=
golang.org/x/term v0.20.0/go.mod h1:8UkIAJTvZgivsXaD6/pH6U9ecQzZ45awqEOzuCvwpFY=
golang.org/x/term v0.27.0/go.mod h1:iMsnZpn0cago0GOrHO2+Y7u7JPn5AylBrcoWkElMTSM=
@@ -334,24 +373,23 @@ golang.org/x/text v0.13.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE=
golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU=
golang.org/x/text v0.15.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU=
golang.org/x/text v0.21.0/go.mod h1:4IBbMaMmOPCJ8SecivzSH54+73PCFmPWxNTLm+vZkEQ=
-golang.org/x/text v0.25.0 h1:qVyWApTSYLk/drJRO5mDlNYskwQznZmkpV2c8q9zls4=
-golang.org/x/text v0.25.0/go.mod h1:WEdwpYrmk1qmdHvhkSTNPm3app7v4rsT8F2UD6+VHIA=
-golang.org/x/time v0.11.0 h1:/bpjEDfN9tkoN/ryeYHnv5hcMlc8ncjMcM4XBk5NWV0=
-golang.org/x/time v0.11.0/go.mod h1:CDIdPxbZBQxdj6cxyCIdrNogrJKMJ7pr37NYpMcMDSg=
+golang.org/x/text v0.31.0 h1:aC8ghyu4JhP8VojJ2lEHBnochRno1sgL6nEi9WGFGMM=
+golang.org/x/text v0.31.0/go.mod h1:tKRAlv61yKIjGGHX/4tP1LTbc13YSec1pxVEWXzfoeM=
+golang.org/x/time v0.14.0 h1:MRx4UaLrDotUKUdCIqzPC48t1Y9hANFKIRpNx+Te8PI=
+golang.org/x/time v0.14.0/go.mod h1:eL/Oa2bBBK0TkX57Fyni+NgnyQQN4LitPmob2Hjnqw4=
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
golang.org/x/tools v0.0.0-20190328211700-ab21143f2384/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc=
golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU=
golang.org/x/tools v0.13.0/go.mod h1:HvlwmtVNQAhOuCjW7xxvovg8wbNq7LwfXh/k7wXUl58=
-golang.org/x/tools v0.17.0/go.mod h1:xsh6VxdV005rRVaS6SSAf9oiAqljS7UZUacMZ8Bnsps=
golang.org/x/tools v0.21.1-0.20240508182429-e35e4ccd0d2d/go.mod h1:aiJjzUbINMkxbQROHiO6hDPo2LHcIPhhQsa9DLh0yGk=
-golang.org/x/tools v0.33.0 h1:4qz2S3zmRxbGIhDIAgjxvFutSvH5EfnsYrRBj0UI0bc=
-golang.org/x/tools v0.33.0/go.mod h1:CIJMaWEY88juyUfo7UbgPqbC8rU2OqfAV1h2Qp0oMYI=
+golang.org/x/tools v0.39.0 h1:ik4ho21kwuQln40uelmciQPp9SipgNDdrafrYA4TmQQ=
+golang.org/x/tools v0.39.0/go.mod h1:JnefbkDPyD8UU2kI5fuf8ZX4/yUeh9W877ZeBONxUqQ=
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
-google.golang.org/protobuf v1.36.6 h1:z1NpPI8ku2WgiWnf+t9wTPsn6eP1L7ksHUlkfLvd9xY=
-google.golang.org/protobuf v1.36.6/go.mod h1:jduwjTPXsFjZGTmRluh+L6NjiWu7pchiJ2/5YcXBHnY=
+google.golang.org/protobuf v1.36.10 h1:AYd7cD/uASjIL6Q9LiTjz8JLcrh/88q5UObnmY3aOOE=
+google.golang.org/protobuf v1.36.10/go.mod h1:HTf+CrKn2C3g5S8VImy6tdcUvCska2kB7j23XfzDpco=
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q=
@@ -363,11 +401,11 @@ gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
-modernc.org/libc v1.65.0 h1:e183gLDnAp9VJh6gWKdTy0CThL9Pt7MfcR/0bgb7Y1Y=
-modernc.org/libc v1.65.0/go.mod h1:7m9VzGq7APssBTydds2zBcxGREwvIGpuUBaKTXdm2Qs=
+modernc.org/libc v1.66.3 h1:cfCbjTUcdsKyyZZfEUKfoHcP3S0Wkvz3jgSzByEWVCQ=
+modernc.org/libc v1.66.3/go.mod h1:XD9zO8kt59cANKvHPXpx7yS2ELPheAey0vjIuZOhOU8=
modernc.org/mathutil v1.7.1 h1:GCZVGXdaN8gTqB1Mf/usp1Y/hSqgI2vAGGP4jZMCxOU=
modernc.org/mathutil v1.7.1/go.mod h1:4p5IwJITfppl0G4sUEDtCr4DthTaT47/N3aT6MhfgJg=
-modernc.org/memory v1.10.0 h1:fzumd51yQ1DxcOxSO+S6X7+QTuVU+n8/Aj7swYjFfC4=
-modernc.org/memory v1.10.0/go.mod h1:/JP4VbVC+K5sU2wZi9bHoq2MAkCnrt2r98UGeSK7Mjw=
-modernc.org/sqlite v1.37.0 h1:s1TMe7T3Q3ovQiK2Ouz4Jwh7dw4ZDqbebSDTlSJdfjI=
-modernc.org/sqlite v1.37.0/go.mod h1:5YiWv+YviqGMuGw4V+PNplcyaJ5v+vQd7TQOgkACoJM=
+modernc.org/memory v1.11.0 h1:o4QC8aMQzmcwCK3t3Ux/ZHmwFPzE6hf2Y5LbkRs+hbI=
+modernc.org/memory v1.11.0/go.mod h1:/JP4VbVC+K5sU2wZi9bHoq2MAkCnrt2r98UGeSK7Mjw=
+modernc.org/sqlite v1.38.2 h1:Aclu7+tgjgcQVShZqim41Bbw9Cho0y/7WzYptXqkEek=
+modernc.org/sqlite v1.38.2/go.mod h1:cPTJYSlgg3Sfg046yBShXENNtPrWrDX8bsbAQBzgQ5E=
diff --git a/log/log.go b/log/log.go
index 08a487fcd..801fd7214 100644
--- a/log/log.go
+++ b/log/log.go
@@ -11,6 +11,7 @@ import (
"runtime"
"sort"
"strings"
+ "sync"
"time"
"github.com/sirupsen/logrus"
@@ -70,6 +71,7 @@ type levelPath struct {
var (
currentLevel Level
+ loggerMu sync.RWMutex
defaultLogger = logrus.New()
logSourceLine = false
rootPath string
@@ -78,8 +80,10 @@ var (
// SetLevel sets the global log level used by the simple logger.
func SetLevel(l Level) {
+ loggerMu.Lock()
currentLevel = l
defaultLogger.Level = logrus.TraceLevel
+ loggerMu.Unlock()
logrus.SetLevel(logrus.Level(l))
}
@@ -110,6 +114,8 @@ func levelFromString(l string) Level {
// SetLogLevels sets the log levels for specific paths in the codebase.
func SetLogLevels(levels map[string]string) {
+ loggerMu.Lock()
+ defer loggerMu.Unlock()
logLevels = nil
for k, v := range levels {
logLevels = append(logLevels, levelPath{path: k, level: levelFromString(v)})
@@ -125,6 +131,8 @@ func SetLogSourceLine(enabled bool) {
func SetRedacting(enabled bool) {
if enabled {
+ loggerMu.Lock()
+ defer loggerMu.Unlock()
defaultLogger.AddHook(redacted)
}
}
@@ -133,6 +141,8 @@ func SetOutput(w io.Writer) {
if runtime.GOOS == "windows" {
w = CRLFWriter(w)
}
+ loggerMu.Lock()
+ defer loggerMu.Unlock()
defaultLogger.SetOutput(w)
}
@@ -158,10 +168,14 @@ func NewContext(ctx context.Context, keyValuePairs ...interface{}) context.Conte
}
func SetDefaultLogger(l *logrus.Logger) {
+ loggerMu.Lock()
+ defer loggerMu.Unlock()
defaultLogger = l
}
func CurrentLevel() Level {
+ loggerMu.RLock()
+ defer loggerMu.RUnlock()
return currentLevel
}
@@ -203,11 +217,22 @@ func log(level Level, args ...interface{}) {
logger.Log(logrus.Level(level), msg)
}
+func Writer() io.Writer {
+ loggerMu.RLock()
+ defer loggerMu.RUnlock()
+ return defaultLogger.Writer()
+}
+
func shouldLog(requiredLevel Level, skip int) bool {
- if currentLevel >= requiredLevel {
+ loggerMu.RLock()
+ level := currentLevel
+ levels := logLevels
+ loggerMu.RUnlock()
+
+ if level >= requiredLevel {
return true
}
- if len(logLevels) == 0 {
+ if len(levels) == 0 {
return false
}
@@ -217,7 +242,7 @@ func shouldLog(requiredLevel Level, skip int) bool {
}
file = strings.TrimPrefix(file, rootPath)
- for _, lp := range logLevels {
+ for _, lp := range levels {
if strings.HasPrefix(file, lp.path) {
return lp.level >= requiredLevel
}
@@ -310,6 +335,8 @@ func extractLogger(ctx interface{}) (*logrus.Entry, error) {
func createNewLogger() *logrus.Entry {
//logrus.SetFormatter(&logrus.TextFormatter{ForceColors: true, DisableTimestamp: false, FullTimestamp: true})
//l.Formatter = &logrus.TextFormatter{ForceColors: true, DisableTimestamp: false, FullTimestamp: true}
+ loggerMu.RLock()
+ defer loggerMu.RUnlock()
logger := logrus.NewEntry(defaultLogger)
return logger
}
diff --git a/log/redactrus.go b/log/redactrus.go
index d743e3f2d..6e17243e7 100755
--- a/log/redactrus.go
+++ b/log/redactrus.go
@@ -42,8 +42,9 @@ func (h *Hook) Fire(e *logrus.Entry) error {
e.Data[k] = "[REDACTED]"
continue
}
-
- // Redact based on value matching in Data fields
+ if v == nil {
+ continue
+ }
switch reflect.TypeOf(v).Kind() {
case reflect.String:
e.Data[k] = re.ReplaceAllString(v.(string), "$1[REDACTED]$2")
diff --git a/model/album.go b/model/album.go
index c9dc022cb..a8dcfe682 100644
--- a/model/album.go
+++ b/model/album.go
@@ -14,6 +14,8 @@ type Album struct {
ID string `structs:"id" json:"id"`
LibraryID int `structs:"library_id" json:"libraryId"`
+ LibraryPath string `structs:"-" json:"libraryPath" hash:"ignore"`
+ LibraryName string `structs:"-" json:"libraryName" hash:"ignore"`
Name string `structs:"name" json:"name"`
EmbedArtPath string `structs:"embed_art_path" json:"-"`
AlbumArtistID string `structs:"album_artist_id" json:"albumArtistId"` // Deprecated, use Participants
diff --git a/model/artist.go b/model/artist.go
index 68836ff28..309ee800f 100644
--- a/model/artist.go
+++ b/model/artist.go
@@ -78,11 +78,11 @@ type ArtistRepository interface {
UpdateExternalInfo(a *Artist) error
Get(id string) (*Artist, error)
GetAll(options ...QueryOptions) (Artists, error)
- GetIndex(includeMissing bool, roles ...Role) (ArtistIndexes, error)
+ GetIndex(includeMissing bool, libraryIds []int, roles ...Role) (ArtistIndexes, error)
// The following methods are used exclusively by the scanner:
RefreshPlayCounts() (int64, error)
- RefreshStats() (int64, error)
+ RefreshStats(allArtists bool) (int64, error)
AnnotatedRepository
SearchableRepository[Artists]
diff --git a/model/criteria/criteria.go b/model/criteria/criteria.go
index 493e53173..54ac59697 100644
--- a/model/criteria/criteria.go
+++ b/model/criteria/criteria.go
@@ -25,17 +25,48 @@ func (c Criteria) OrderBy() string {
if c.Sort == "" {
c.Sort = "title"
}
- sortField := strings.ToLower(c.Sort)
- f := fieldMap[sortField]
- var mapped string
- if f == nil {
- log.Error("Invalid field in 'sort' field. Using 'title'", "sort", c.Sort)
- mapped = fieldMap["title"].field
- } else {
+
+ order := strings.ToLower(strings.TrimSpace(c.Order))
+ if order != "" && order != "asc" && order != "desc" {
+ log.Error("Invalid value in 'order' field. Valid values: 'asc', 'desc'", "order", c.Order)
+ order = ""
+ }
+
+ parts := strings.Split(c.Sort, ",")
+ fields := make([]string, 0, len(parts))
+
+ for _, p := range parts {
+ p = strings.TrimSpace(p)
+ if p == "" {
+ continue
+ }
+
+ dir := "asc"
+ if strings.HasPrefix(p, "+") || strings.HasPrefix(p, "-") {
+ if strings.HasPrefix(p, "-") {
+ dir = "desc"
+ }
+ p = strings.TrimSpace(p[1:])
+ }
+
+ sortField := strings.ToLower(p)
+ f := fieldMap[sortField]
+ if f == nil {
+ log.Error("Invalid field in 'sort' field", "sort", sortField)
+ continue
+ }
+
+ var mapped string
+
if f.order != "" {
mapped = f.order
} else if f.isTag {
- mapped = "COALESCE(json_extract(media_file.tags, '$." + sortField + "[0].value'), '')"
+ // Use the actual field name (handles aliases like albumtype -> releasetype)
+ tagName := sortField
+ if f.field != "" {
+ tagName = f.field
+ }
+ mapped = "COALESCE(json_extract(media_file.tags, '$." + tagName + "[0].value'), '')"
} else if f.isRole {
mapped = "COALESCE(json_extract(media_file.participants, '$." + sortField + "[0].name'), '')"
} else {
@@ -44,15 +75,20 @@ func (c Criteria) OrderBy() string {
if f.numeric {
mapped = fmt.Sprintf("CAST(%s AS REAL)", mapped)
}
- }
- if c.Order != "" {
- if strings.EqualFold(c.Order, "asc") || strings.EqualFold(c.Order, "desc") {
- mapped = mapped + " " + c.Order
- } else {
- log.Error("Invalid value in 'order' field. Valid values: 'asc', 'desc'", "order", c.Order)
+ // If the global 'order' field is set to 'desc', reverse the default or field-specific sort direction.
+ // This ensures that the global order applies consistently across all fields.
+ if order == "desc" {
+ if dir == "asc" {
+ dir = "desc"
+ } else {
+ dir = "asc"
+ }
}
+
+ fields = append(fields, mapped+" "+dir)
}
- return mapped
+
+ return strings.Join(fields, ", ")
}
func (c Criteria) ToSql() (sql string, args []any, err error) {
diff --git a/model/criteria/criteria_test.go b/model/criteria/criteria_test.go
index 7afb6ec0d..032ead5c8 100644
--- a/model/criteria/criteria_test.go
+++ b/model/criteria/criteria_test.go
@@ -118,11 +118,43 @@ var _ = Describe("Criteria", func() {
)
})
+ It("sorts by albumtype alias (resolves to releasetype)", func() {
+ AddTagNames([]string{"releasetype"})
+ goObj.Sort = "albumtype"
+ gomega.Expect(goObj.OrderBy()).To(
+ gomega.Equal(
+ "COALESCE(json_extract(media_file.tags, '$.releasetype[0].value'), '') asc",
+ ),
+ )
+ })
+
It("sorts by random", func() {
newObj := goObj
newObj.Sort = "random"
gomega.Expect(newObj.OrderBy()).To(gomega.Equal("random() asc"))
})
+
+ It("sorts by multiple fields", func() {
+ goObj.Sort = "title,-rating"
+ gomega.Expect(goObj.OrderBy()).To(gomega.Equal(
+ "media_file.title asc, COALESCE(annotation.rating, 0) desc",
+ ))
+ })
+
+ It("reverts order when order is desc", func() {
+ goObj.Sort = "-date,artist"
+ goObj.Order = "desc"
+ gomega.Expect(goObj.OrderBy()).To(gomega.Equal(
+ "media_file.date asc, COALESCE(json_extract(media_file.participants, '$.artist[0].name'), '') desc",
+ ))
+ })
+
+ It("ignores invalid sort fields", func() {
+ goObj.Sort = "bogus,title"
+ gomega.Expect(goObj.OrderBy()).To(gomega.Equal(
+ "media_file.title asc",
+ ))
+ })
})
})
diff --git a/model/criteria/fields.go b/model/criteria/fields.go
index b7178e540..70719cd6f 100644
--- a/model/criteria/fields.go
+++ b/model/criteria/fields.go
@@ -10,43 +10,52 @@ import (
)
var fieldMap = map[string]*mappedField{
- "title": {field: "media_file.title"},
- "album": {field: "media_file.album"},
- "hascoverart": {field: "media_file.has_cover_art"},
- "tracknumber": {field: "media_file.track_number"},
- "discnumber": {field: "media_file.disc_number"},
- "year": {field: "media_file.year"},
- "date": {field: "media_file.date", alias: "recordingdate"},
- "originalyear": {field: "media_file.original_year"},
- "originaldate": {field: "media_file.original_date"},
- "releaseyear": {field: "media_file.release_year"},
- "releasedate": {field: "media_file.release_date"},
- "size": {field: "media_file.size"},
- "compilation": {field: "media_file.compilation"},
- "dateadded": {field: "media_file.created_at"},
- "datemodified": {field: "media_file.updated_at"},
- "discsubtitle": {field: "media_file.disc_subtitle"},
- "comment": {field: "media_file.comment"},
- "lyrics": {field: "media_file.lyrics"},
- "sorttitle": {field: "media_file.sort_title"},
- "sortalbum": {field: "media_file.sort_album_name"},
- "sortartist": {field: "media_file.sort_artist_name"},
- "sortalbumartist": {field: "media_file.sort_album_artist_name"},
- "albumtype": {field: "media_file.mbz_album_type", alias: "releasetype"},
- "albumcomment": {field: "media_file.mbz_album_comment"},
- "catalognumber": {field: "media_file.catalog_num"},
- "filepath": {field: "media_file.path"},
- "filetype": {field: "media_file.suffix"},
- "duration": {field: "media_file.duration"},
- "bitrate": {field: "media_file.bit_rate"},
- "bitdepth": {field: "media_file.bit_depth"},
- "bpm": {field: "media_file.bpm"},
- "channels": {field: "media_file.channels"},
- "loved": {field: "COALESCE(annotation.starred, false)"},
- "dateloved": {field: "annotation.starred_at"},
- "lastplayed": {field: "annotation.play_date"},
- "playcount": {field: "COALESCE(annotation.play_count, 0)"},
- "rating": {field: "COALESCE(annotation.rating, 0)"},
+ "title": {field: "media_file.title"},
+ "album": {field: "media_file.album"},
+ "hascoverart": {field: "media_file.has_cover_art"},
+ "tracknumber": {field: "media_file.track_number"},
+ "discnumber": {field: "media_file.disc_number"},
+ "year": {field: "media_file.year"},
+ "date": {field: "media_file.date", alias: "recordingdate"},
+ "originalyear": {field: "media_file.original_year"},
+ "originaldate": {field: "media_file.original_date"},
+ "releaseyear": {field: "media_file.release_year"},
+ "releasedate": {field: "media_file.release_date"},
+ "size": {field: "media_file.size"},
+ "compilation": {field: "media_file.compilation"},
+ "dateadded": {field: "media_file.created_at"},
+ "datemodified": {field: "media_file.updated_at"},
+ "discsubtitle": {field: "media_file.disc_subtitle"},
+ "comment": {field: "media_file.comment"},
+ "lyrics": {field: "media_file.lyrics"},
+ "sorttitle": {field: "media_file.sort_title"},
+ "sortalbum": {field: "media_file.sort_album_name"},
+ "sortartist": {field: "media_file.sort_artist_name"},
+ "sortalbumartist": {field: "media_file.sort_album_artist_name"},
+ "albumcomment": {field: "media_file.mbz_album_comment"},
+ "catalognumber": {field: "media_file.catalog_num"},
+ "filepath": {field: "media_file.path"},
+ "filetype": {field: "media_file.suffix"},
+ "duration": {field: "media_file.duration"},
+ "bitrate": {field: "media_file.bit_rate"},
+ "bitdepth": {field: "media_file.bit_depth"},
+ "bpm": {field: "media_file.bpm"},
+ "channels": {field: "media_file.channels"},
+ "loved": {field: "COALESCE(annotation.starred, false)"},
+ "dateloved": {field: "annotation.starred_at"},
+ "lastplayed": {field: "annotation.play_date"},
+ "playcount": {field: "COALESCE(annotation.play_count, 0)"},
+ "rating": {field: "COALESCE(annotation.rating, 0)"},
+ "mbz_album_id": {field: "media_file.mbz_album_id"},
+ "mbz_album_artist_id": {field: "media_file.mbz_album_artist_id"},
+ "mbz_artist_id": {field: "media_file.mbz_artist_id"},
+ "mbz_recording_id": {field: "media_file.mbz_recording_id"},
+ "mbz_release_track_id": {field: "media_file.mbz_release_track_id"},
+ "mbz_release_group_id": {field: "media_file.mbz_release_group_id"},
+ "library_id": {field: "media_file.library_id", numeric: true},
+
+ // Backward compatibility: albumtype is an alias for releasetype tag
+ "albumtype": {field: "releasetype", isTag: true},
// special fields
"random": {field: "", order: "random()"}, // pseudo-field for random sorting
@@ -147,13 +156,19 @@ type tagCond struct {
func (e tagCond) ToSql() (string, []any, error) {
cond, args, err := e.cond.ToSql()
- // Check if this tag is marked as numeric in the fieldMap
- if fm, ok := fieldMap[e.tag]; ok && fm.numeric {
- cond = strings.ReplaceAll(cond, "value", "CAST(value AS REAL)")
+ // Resolve the actual tag name (handles aliases like albumtype -> releasetype)
+ tagName := e.tag
+ if fm, ok := fieldMap[e.tag]; ok {
+ if fm.field != "" {
+ tagName = fm.field
+ }
+ if fm.numeric {
+ cond = strings.ReplaceAll(cond, "value", "CAST(value AS REAL)")
+ }
}
cond = fmt.Sprintf("exists (select 1 from json_tree(tags, '$.%s') where key='value' and %s)",
- e.tag, cond)
+ tagName, cond)
if e.not {
cond = "not " + cond
}
diff --git a/model/criteria/operators_test.go b/model/criteria/operators_test.go
index 95f9fc5f4..4c1db1303 100644
--- a/model/criteria/operators_test.go
+++ b/model/criteria/operators_test.go
@@ -29,7 +29,11 @@ var _ = Describe("Operators", func() {
},
Entry("is [string]", Is{"title": "Low Rider"}, "media_file.title = ?", "Low Rider"),
Entry("is [bool]", Is{"loved": true}, "COALESCE(annotation.starred, false) = ?", true),
+ Entry("is [numeric]", Is{"library_id": 1}, "media_file.library_id = ?", 1),
+ Entry("is [numeric list]", Is{"library_id": []int{1, 2}}, "media_file.library_id IN (?,?)", 1, 2),
Entry("isNot", IsNot{"title": "Low Rider"}, "media_file.title <> ?", "Low Rider"),
+ Entry("isNot [numeric]", IsNot{"library_id": 1}, "media_file.library_id <> ?", 1),
+ Entry("isNot [numeric list]", IsNot{"library_id": []int{1, 2}}, "media_file.library_id NOT IN (?,?)", 1, 2),
Entry("gt", Gt{"playCount": 10}, "COALESCE(annotation.play_count, 0) > ?", 10),
Entry("lt", Lt{"playCount": 10}, "COALESCE(annotation.play_count, 0) < ?", 10),
Entry("contains", Contains{"title": "Low Rider"}, "media_file.title LIKE ?", "%Low Rider%"),
@@ -101,6 +105,40 @@ var _ = Describe("Operators", func() {
gomega.Expect(sql).To(gomega.BeEmpty())
gomega.Expect(args).To(gomega.BeEmpty())
})
+ It("supports releasetype as multi-valued tag", func() {
+ AddTagNames([]string{"releasetype"})
+ op := Contains{"releasetype": "soundtrack"}
+ sql, args, err := op.ToSql()
+ gomega.Expect(err).ToNot(gomega.HaveOccurred())
+ gomega.Expect(sql).To(gomega.Equal("exists (select 1 from json_tree(tags, '$.releasetype') where key='value' and value LIKE ?)"))
+ gomega.Expect(args).To(gomega.HaveExactElements("%soundtrack%"))
+ })
+ It("supports albumtype as alias for releasetype", func() {
+ AddTagNames([]string{"releasetype"})
+ op := Contains{"albumtype": "live"}
+ sql, args, err := op.ToSql()
+ gomega.Expect(err).ToNot(gomega.HaveOccurred())
+ gomega.Expect(sql).To(gomega.Equal("exists (select 1 from json_tree(tags, '$.releasetype') where key='value' and value LIKE ?)"))
+ gomega.Expect(args).To(gomega.HaveExactElements("%live%"))
+ })
+ It("supports albumtype alias with Is operator", func() {
+ AddTagNames([]string{"releasetype"})
+ op := Is{"albumtype": "album"}
+ sql, args, err := op.ToSql()
+ gomega.Expect(err).ToNot(gomega.HaveOccurred())
+ // Should query $.releasetype, not $.albumtype
+ gomega.Expect(sql).To(gomega.Equal("exists (select 1 from json_tree(tags, '$.releasetype') where key='value' and value = ?)"))
+ gomega.Expect(args).To(gomega.HaveExactElements("album"))
+ })
+ It("supports albumtype alias with IsNot operator", func() {
+ AddTagNames([]string{"releasetype"})
+ op := IsNot{"albumtype": "compilation"}
+ sql, args, err := op.ToSql()
+ gomega.Expect(err).ToNot(gomega.HaveOccurred())
+ // Should query $.releasetype, not $.albumtype
+ gomega.Expect(sql).To(gomega.Equal("not exists (select 1 from json_tree(tags, '$.releasetype') where key='value' and value = ?)"))
+ gomega.Expect(args).To(gomega.HaveExactElements("compilation"))
+ })
})
Describe("Custom Roles", func() {
diff --git a/model/datastore.go b/model/datastore.go
index 4290e2134..536a37274 100644
--- a/model/datastore.go
+++ b/model/datastore.go
@@ -43,5 +43,5 @@ type DataStore interface {
WithTx(block func(tx DataStore) error, scope ...string) error
WithTxImmediate(block func(tx DataStore) error, scope ...string) error
- GC(ctx context.Context) error
+ GC(ctx context.Context, libraryIDs ...int) error
}
diff --git a/model/errors.go b/model/errors.go
index ff4be5723..41029d316 100644
--- a/model/errors.go
+++ b/model/errors.go
@@ -8,4 +8,5 @@ var (
ErrNotAuthorized = errors.New("not authorized")
ErrExpired = errors.New("access expired")
ErrNotAvailable = errors.New("functionality not available")
+ ErrValidation = errors.New("validation error")
)
diff --git a/model/folder.go b/model/folder.go
index 3d14e7c53..7a769735e 100644
--- a/model/folder.go
+++ b/model/folder.go
@@ -17,7 +17,7 @@ import (
type Folder struct {
ID string `structs:"id"`
LibraryID int `structs:"library_id"`
- LibraryPath string `structs:"-" json:"-" hash:"-"`
+ LibraryPath string `structs:"-" json:"-" hash:"ignore"`
Path string `structs:"path"`
Name string `structs:"name"`
ParentID string `structs:"parent_id"`
@@ -25,6 +25,7 @@ type Folder struct {
NumPlaylists int `structs:"num_playlists"`
ImageFiles []string `structs:"image_files"`
ImagesUpdatedAt time.Time `structs:"images_updated_at"`
+ Hash string `structs:"hash"`
Missing bool `structs:"missing"`
UpdateAt time.Time `structs:"updated_at"`
CreatedAt time.Time `structs:"created_at"`
@@ -74,12 +75,17 @@ func NewFolder(lib Library, folderPath string) *Folder {
type FolderCursor iter.Seq2[Folder, error]
+type FolderUpdateInfo struct {
+ UpdatedAt time.Time
+ Hash string
+}
+
type FolderRepository interface {
Get(id string) (*Folder, error)
GetByPath(lib Library, path string) (*Folder, error)
GetAll(...QueryOptions) ([]Folder, error)
CountAll(...QueryOptions) (int64, error)
- GetLastUpdates(lib Library) (map[string]time.Time, error)
+ GetFolderUpdateInfo(lib Library, targetPaths ...string) (map[string]FolderUpdateInfo, error)
Put(*Folder) error
MarkMissing(missing bool, ids ...string) error
GetTouchedWithPlaylists() (FolderCursor, error)
diff --git a/model/library.go b/model/library.go
index a29f1c1d6..bcb2864c8 100644
--- a/model/library.go
+++ b/model/library.go
@@ -2,34 +2,60 @@ package model
import (
"time"
+
+ "github.com/navidrome/navidrome/utils/slice"
)
type Library struct {
- ID int
- Name string
- Path string
- RemotePath string
- LastScanAt time.Time
- LastScanStartedAt time.Time
- FullScanInProgress bool
- UpdatedAt time.Time
- CreatedAt time.Time
+ ID int `json:"id" db:"id"`
+ Name string `json:"name" db:"name"`
+ Path string `json:"path" db:"path"`
+ RemotePath string `json:"remotePath" db:"remote_path"`
+ LastScanAt time.Time `json:"lastScanAt" db:"last_scan_at"`
+ LastScanStartedAt time.Time `json:"lastScanStartedAt" db:"last_scan_started_at"`
+ FullScanInProgress bool `json:"fullScanInProgress" db:"full_scan_in_progress"`
+ UpdatedAt time.Time `json:"updatedAt" db:"updated_at"`
+ CreatedAt time.Time `json:"createdAt" db:"created_at"`
+ TotalSongs int `json:"totalSongs" db:"total_songs"`
+ TotalAlbums int `json:"totalAlbums" db:"total_albums"`
+ TotalArtists int `json:"totalArtists" db:"total_artists"`
+ TotalFolders int `json:"totalFolders" db:"total_folders"`
+ TotalFiles int `json:"totalFiles" db:"total_files"`
+ TotalMissingFiles int `json:"totalMissingFiles" db:"total_missing_files"`
+ TotalSize int64 `json:"totalSize" db:"total_size"`
+ TotalDuration float64 `json:"totalDuration" db:"total_duration"`
+ DefaultNewUsers bool `json:"defaultNewUsers" db:"default_new_users"`
}
+const (
+ DefaultLibraryID = 1
+ DefaultLibraryName = "Music Library"
+)
+
type Libraries []Library
+func (l Libraries) IDs() []int {
+ return slice.Map(l, func(lib Library) int { return lib.ID })
+}
+
type LibraryRepository interface {
Get(id int) (*Library, error)
// GetPath returns the path of the library with the given ID.
// Its implementation must be optimized to avoid unnecessary queries.
GetPath(id int) (string, error)
GetAll(...QueryOptions) (Libraries, error)
+ CountAll(...QueryOptions) (int64, error)
Put(*Library) error
+ Delete(id int) error
StoreMusicFolder() error
AddArtist(id int, artistID string) error
+ // User-library association methods
+ GetUsersWithLibraryAccess(libraryID int) (Users, error)
+
// TODO These methods should be moved to a core service
ScanBegin(id int, fullScan bool) error
ScanEnd(id int) error
ScanInProgress() (bool, error)
+ RefreshStats(id int) error
}
diff --git a/model/mediafile.go b/model/mediafile.go
index cdb001c85..0ef26d746 100644
--- a/model/mediafile.go
+++ b/model/mediafile.go
@@ -9,6 +9,7 @@ import (
"mime"
"path/filepath"
"slices"
+ "strings"
"time"
"github.com/gohugoio/hashstructure"
@@ -25,7 +26,8 @@ type MediaFile struct {
ID string `structs:"id" json:"id" hash:"ignore"`
PID string `structs:"pid" json:"-" hash:"ignore"`
LibraryID int `structs:"library_id" json:"libraryId" hash:"ignore"`
- LibraryPath string `structs:"-" json:"libraryPath" hash:"-"`
+ LibraryPath string `structs:"-" json:"libraryPath" hash:"ignore"`
+ LibraryName string `structs:"-" json:"libraryName" hash:"ignore"`
FolderID string `structs:"folder_id" json:"folderId" hash:"ignore"`
Path string `structs:"path" json:"path" hash:"ignore"`
Title string `structs:"title" json:"title"`
@@ -35,53 +37,53 @@ type MediaFile struct {
Artist string `structs:"artist" json:"artist"`
AlbumArtistID string `structs:"album_artist_id" json:"albumArtistId"` // Deprecated: Use Participants instead
// AlbumArtist is the display name used for the album artist.
- AlbumArtist string `structs:"album_artist" json:"albumArtist"`
- AlbumID string `structs:"album_id" json:"albumId"`
- HasCoverArt bool `structs:"has_cover_art" json:"hasCoverArt"`
- TrackNumber int `structs:"track_number" json:"trackNumber"`
- DiscNumber int `structs:"disc_number" json:"discNumber"`
- DiscSubtitle string `structs:"disc_subtitle" json:"discSubtitle,omitempty"`
- Year int `structs:"year" json:"year"`
- Date string `structs:"date" json:"date,omitempty"`
- OriginalYear int `structs:"original_year" json:"originalYear"`
- OriginalDate string `structs:"original_date" json:"originalDate,omitempty"`
- ReleaseYear int `structs:"release_year" json:"releaseYear"`
- ReleaseDate string `structs:"release_date" json:"releaseDate,omitempty"`
- Size int64 `structs:"size" json:"size"`
- Suffix string `structs:"suffix" json:"suffix"`
- Duration float32 `structs:"duration" json:"duration"`
- BitRate int `structs:"bit_rate" json:"bitRate"`
- SampleRate int `structs:"sample_rate" json:"sampleRate"`
- BitDepth int `structs:"bit_depth" json:"bitDepth"`
- Channels int `structs:"channels" json:"channels"`
- Genre string `structs:"genre" json:"genre"`
- Genres Genres `structs:"-" json:"genres,omitempty"`
- SortTitle string `structs:"sort_title" json:"sortTitle,omitempty"`
- SortAlbumName string `structs:"sort_album_name" json:"sortAlbumName,omitempty"`
- SortArtistName string `structs:"sort_artist_name" json:"sortArtistName,omitempty"` // Deprecated: Use Participants instead
- SortAlbumArtistName string `structs:"sort_album_artist_name" json:"sortAlbumArtistName,omitempty"` // Deprecated: Use Participants instead
- OrderTitle string `structs:"order_title" json:"orderTitle,omitempty"`
- OrderAlbumName string `structs:"order_album_name" json:"orderAlbumName"`
- OrderArtistName string `structs:"order_artist_name" json:"orderArtistName"` // Deprecated: Use Participants instead
- OrderAlbumArtistName string `structs:"order_album_artist_name" json:"orderAlbumArtistName"` // Deprecated: Use Participants instead
- Compilation bool `structs:"compilation" json:"compilation"`
- Comment string `structs:"comment" json:"comment,omitempty"`
- Lyrics string `structs:"lyrics" json:"lyrics"`
- BPM int `structs:"bpm" json:"bpm,omitempty"`
- ExplicitStatus string `structs:"explicit_status" json:"explicitStatus"`
- CatalogNum string `structs:"catalog_num" json:"catalogNum,omitempty"`
- MbzRecordingID string `structs:"mbz_recording_id" json:"mbzRecordingID,omitempty"`
- MbzReleaseTrackID string `structs:"mbz_release_track_id" json:"mbzReleaseTrackId,omitempty"`
- MbzAlbumID string `structs:"mbz_album_id" json:"mbzAlbumId,omitempty"`
- MbzReleaseGroupID string `structs:"mbz_release_group_id" json:"mbzReleaseGroupId,omitempty"`
- MbzArtistID string `structs:"mbz_artist_id" json:"mbzArtistId,omitempty"` // Deprecated: Use Participants instead
- MbzAlbumArtistID string `structs:"mbz_album_artist_id" json:"mbzAlbumArtistId,omitempty"` // Deprecated: Use Participants instead
- MbzAlbumType string `structs:"mbz_album_type" json:"mbzAlbumType,omitempty"`
- MbzAlbumComment string `structs:"mbz_album_comment" json:"mbzAlbumComment,omitempty"`
- RGAlbumGain float64 `structs:"rg_album_gain" json:"rgAlbumGain"`
- RGAlbumPeak float64 `structs:"rg_album_peak" json:"rgAlbumPeak"`
- RGTrackGain float64 `structs:"rg_track_gain" json:"rgTrackGain"`
- RGTrackPeak float64 `structs:"rg_track_peak" json:"rgTrackPeak"`
+ AlbumArtist string `structs:"album_artist" json:"albumArtist"`
+ AlbumID string `structs:"album_id" json:"albumId"`
+ HasCoverArt bool `structs:"has_cover_art" json:"hasCoverArt"`
+ TrackNumber int `structs:"track_number" json:"trackNumber"`
+ DiscNumber int `structs:"disc_number" json:"discNumber"`
+ DiscSubtitle string `structs:"disc_subtitle" json:"discSubtitle,omitempty"`
+ Year int `structs:"year" json:"year"`
+ Date string `structs:"date" json:"date,omitempty"`
+ OriginalYear int `structs:"original_year" json:"originalYear"`
+ OriginalDate string `structs:"original_date" json:"originalDate,omitempty"`
+ ReleaseYear int `structs:"release_year" json:"releaseYear"`
+ ReleaseDate string `structs:"release_date" json:"releaseDate,omitempty"`
+ Size int64 `structs:"size" json:"size"`
+ Suffix string `structs:"suffix" json:"suffix"`
+ Duration float32 `structs:"duration" json:"duration"`
+ BitRate int `structs:"bit_rate" json:"bitRate"`
+ SampleRate int `structs:"sample_rate" json:"sampleRate"`
+ BitDepth int `structs:"bit_depth" json:"bitDepth"`
+ Channels int `structs:"channels" json:"channels"`
+ Genre string `structs:"genre" json:"genre"`
+ Genres Genres `structs:"-" json:"genres,omitempty"`
+ SortTitle string `structs:"sort_title" json:"sortTitle,omitempty"`
+ SortAlbumName string `structs:"sort_album_name" json:"sortAlbumName,omitempty"`
+ SortArtistName string `structs:"sort_artist_name" json:"sortArtistName,omitempty"` // Deprecated: Use Participants instead
+ SortAlbumArtistName string `structs:"sort_album_artist_name" json:"sortAlbumArtistName,omitempty"` // Deprecated: Use Participants instead
+ OrderTitle string `structs:"order_title" json:"orderTitle,omitempty"`
+ OrderAlbumName string `structs:"order_album_name" json:"orderAlbumName"`
+ OrderArtistName string `structs:"order_artist_name" json:"orderArtistName"` // Deprecated: Use Participants instead
+ OrderAlbumArtistName string `structs:"order_album_artist_name" json:"orderAlbumArtistName"` // Deprecated: Use Participants instead
+ Compilation bool `structs:"compilation" json:"compilation"`
+ Comment string `structs:"comment" json:"comment,omitempty"`
+ Lyrics string `structs:"lyrics" json:"lyrics"`
+ BPM int `structs:"bpm" json:"bpm,omitempty"`
+ ExplicitStatus string `structs:"explicit_status" json:"explicitStatus"`
+ CatalogNum string `structs:"catalog_num" json:"catalogNum,omitempty"`
+ MbzRecordingID string `structs:"mbz_recording_id" json:"mbzRecordingID,omitempty"`
+ MbzReleaseTrackID string `structs:"mbz_release_track_id" json:"mbzReleaseTrackId,omitempty"`
+ MbzAlbumID string `structs:"mbz_album_id" json:"mbzAlbumId,omitempty"`
+ MbzReleaseGroupID string `structs:"mbz_release_group_id" json:"mbzReleaseGroupId,omitempty"`
+ MbzArtistID string `structs:"mbz_artist_id" json:"mbzArtistId,omitempty"` // Deprecated: Use Participants instead
+ MbzAlbumArtistID string `structs:"mbz_album_artist_id" json:"mbzAlbumArtistId,omitempty"` // Deprecated: Use Participants instead
+ MbzAlbumType string `structs:"mbz_album_type" json:"mbzAlbumType,omitempty"`
+ MbzAlbumComment string `structs:"mbz_album_comment" json:"mbzAlbumComment,omitempty"`
+ RGAlbumGain *float64 `structs:"rg_album_gain" json:"rgAlbumGain"`
+ RGAlbumPeak *float64 `structs:"rg_album_peak" json:"rgAlbumPeak"`
+ RGTrackGain *float64 `structs:"rg_track_gain" json:"rgTrackGain"`
+ RGTrackPeak *float64 `structs:"rg_track_peak" json:"rgTrackPeak"`
Tags Tags `structs:"tags" json:"tags,omitempty" hash:"ignore"` // All imported tags from the original file
Participants Participants `structs:"participants" json:"participants" hash:"ignore"` // All artists that participated in this track
@@ -330,6 +332,23 @@ func firstArtPath(currentPath string, currentDisc int, m MediaFile) (string, int
return currentPath, currentDisc
}
+// ToM3U8 exports the playlist to the Extended M3U8 format, as specified in
+// https://docs.fileformat.com/audio/m3u/#extended-m3u
+func (mfs MediaFiles) ToM3U8(title string, absolutePaths bool) string {
+ buf := strings.Builder{}
+ buf.WriteString("#EXTM3U\n")
+ buf.WriteString(fmt.Sprintf("#PLAYLIST:%s\n", title))
+ for _, t := range mfs {
+ buf.WriteString(fmt.Sprintf("#EXTINF:%.f,%s - %s\n", t.Duration, t.Artist, t.Title))
+ if absolutePaths {
+ buf.WriteString(t.AbsolutePath() + "\n")
+ } else {
+ buf.WriteString(t.Path + "\n")
+ }
+ }
+ return buf.String()
+}
+
type MediaFileCursor iter.Seq2[MediaFile, error]
type MediaFileRepository interface {
@@ -349,6 +368,8 @@ type MediaFileRepository interface {
MarkMissing(bool, ...*MediaFile) error
MarkMissingByFolder(missing bool, folderIDs ...string) error
GetMissingAndMatching(libId int) (MediaFileCursor, error)
+ FindRecentFilesByMBZTrackID(missing MediaFile, since time.Time) (MediaFiles, error)
+ FindRecentFilesByProperties(missing MediaFile, since time.Time) (MediaFiles, error)
AnnotatedRepository
BookmarkableRepository
diff --git a/model/mediafile_test.go b/model/mediafile_test.go
index 7f583cf75..635a61d30 100644
--- a/model/mediafile_test.go
+++ b/model/mediafile_test.go
@@ -402,6 +402,72 @@ var _ = Describe("MediaFiles", func() {
})
})
})
+
+ Describe("ToM3U8", func() {
+ It("returns header only for empty MediaFiles", func() {
+ mfs = MediaFiles{}
+ result := mfs.ToM3U8("My Playlist", false)
+ Expect(result).To(Equal("#EXTM3U\n#PLAYLIST:My Playlist\n"))
+ })
+
+ DescribeTable("duration formatting",
+ func(duration float32, expected string) {
+ mfs = MediaFiles{{Title: "Song", Artist: "Artist", Duration: duration, Path: "song.mp3"}}
+ result := mfs.ToM3U8("Test", false)
+ Expect(result).To(ContainSubstring(expected))
+ },
+ Entry("zero duration", float32(0.0), "#EXTINF:0,"),
+ Entry("whole number", float32(120.0), "#EXTINF:120,"),
+ Entry("rounds 0.5 down", float32(180.5), "#EXTINF:180,"),
+ Entry("rounds 0.6 up", float32(240.6), "#EXTINF:241,"),
+ )
+
+ Context("multiple tracks", func() {
+ BeforeEach(func() {
+ mfs = MediaFiles{
+ {Title: "Song One", Artist: "Artist A", Duration: 120, Path: "a/song1.mp3", LibraryPath: "/music"},
+ {Title: "Song Two", Artist: "Artist B", Duration: 241, Path: "b/song2.mp3", LibraryPath: "/music"},
+ {Title: "Song with \"quotes\" & ampersands", Artist: "Artist with Ümläuts", Duration: 90, Path: "special/file.mp3", LibraryPath: "/música"},
+ }
+ })
+
+ DescribeTable("generates correct output",
+ func(absolutePaths bool, expectedContent string) {
+ result := mfs.ToM3U8("Multi Track", absolutePaths)
+ Expect(result).To(Equal(expectedContent))
+ },
+ Entry("relative paths",
+ false,
+ "#EXTM3U\n#PLAYLIST:Multi Track\n#EXTINF:120,Artist A - Song One\na/song1.mp3\n#EXTINF:241,Artist B - Song Two\nb/song2.mp3\n#EXTINF:90,Artist with Ümläuts - Song with \"quotes\" & ampersands\nspecial/file.mp3\n",
+ ),
+ Entry("absolute paths",
+ true,
+ "#EXTM3U\n#PLAYLIST:Multi Track\n#EXTINF:120,Artist A - Song One\n/music/a/song1.mp3\n#EXTINF:241,Artist B - Song Two\n/music/b/song2.mp3\n#EXTINF:90,Artist with Ümläuts - Song with \"quotes\" & ampersands\n/música/special/file.mp3\n",
+ ),
+ Entry("special characters",
+ false,
+ "#EXTM3U\n#PLAYLIST:Multi Track\n#EXTINF:120,Artist A - Song One\na/song1.mp3\n#EXTINF:241,Artist B - Song Two\nb/song2.mp3\n#EXTINF:90,Artist with Ümläuts - Song with \"quotes\" & ampersands\nspecial/file.mp3\n",
+ ),
+ )
+ })
+
+ Context("path variations", func() {
+ It("handles different path structures", func() {
+ mfs = MediaFiles{
+ {Title: "Root", Artist: "Artist", Duration: 60, Path: "song.mp3", LibraryPath: "/lib"},
+ {Title: "Nested", Artist: "Artist", Duration: 60, Path: "deep/nested/song.mp3", LibraryPath: "/lib"},
+ }
+
+ relativeResult := mfs.ToM3U8("Test", false)
+ Expect(relativeResult).To(ContainSubstring("song.mp3\n"))
+ Expect(relativeResult).To(ContainSubstring("deep/nested/song.mp3\n"))
+
+ absoluteResult := mfs.ToM3U8("Test", true)
+ Expect(absoluteResult).To(ContainSubstring("/lib/song.mp3\n"))
+ Expect(absoluteResult).To(ContainSubstring("/lib/deep/nested/song.mp3\n"))
+ })
+ })
+ })
})
var _ = Describe("MediaFile", func() {
diff --git a/model/metadata/legacy_ids.go b/model/metadata/legacy_ids.go
index 25025ea19..18a273550 100644
--- a/model/metadata/legacy_ids.go
+++ b/model/metadata/legacy_ids.go
@@ -14,18 +14,25 @@ import (
// These are the legacy ID functions that were used in the original Navidrome ID generation.
// They are kept here for backwards compatibility with existing databases.
-func legacyTrackID(mf model.MediaFile) string {
- return fmt.Sprintf("%x", md5.Sum([]byte(mf.Path)))
+func legacyTrackID(mf model.MediaFile, prependLibId bool) string {
+ id := mf.Path
+ if prependLibId && mf.LibraryID != model.DefaultLibraryID {
+ id = fmt.Sprintf("%d\\%s", mf.LibraryID, id)
+ }
+ return fmt.Sprintf("%x", md5.Sum([]byte(id)))
}
-func legacyAlbumID(md Metadata) string {
- releaseDate := legacyReleaseDate(md)
+func legacyAlbumID(mf model.MediaFile, md Metadata, prependLibId bool) string {
+ _, _, releaseDate := md.mapDates()
albumPath := strings.ToLower(fmt.Sprintf("%s\\%s", legacyMapAlbumArtistName(md), legacyMapAlbumName(md)))
if !conf.Server.Scanner.GroupAlbumReleases {
if len(releaseDate) != 0 {
albumPath = fmt.Sprintf("%s\\%s", albumPath, releaseDate)
}
}
+ if prependLibId && mf.LibraryID != model.DefaultLibraryID {
+ albumPath = fmt.Sprintf("%d\\%s", mf.LibraryID, albumPath)
+ }
return fmt.Sprintf("%x", md5.Sum([]byte(albumPath)))
}
@@ -48,9 +55,3 @@ func legacyMapAlbumName(md Metadata) string {
consts.UnknownAlbum,
)
}
-
-// Keep the TaggedLikePicard logic for backwards compatibility
-func legacyReleaseDate(md Metadata) string {
- _, _, releaseDate := md.mapDates()
- return string(releaseDate)
-}
diff --git a/model/metadata/legacy_ids_test.go b/model/metadata/legacy_ids_test.go
deleted file mode 100644
index b6d096763..000000000
--- a/model/metadata/legacy_ids_test.go
+++ /dev/null
@@ -1,30 +0,0 @@
-package metadata
-
-import (
- . "github.com/onsi/ginkgo/v2"
- . "github.com/onsi/gomega"
-)
-
-var _ = Describe("legacyReleaseDate", func() {
-
- DescribeTable("legacyReleaseDate",
- func(recordingDate, originalDate, releaseDate, expected string) {
- md := New("", Info{
- Tags: map[string][]string{
- "DATE": {recordingDate},
- "ORIGINALDATE": {originalDate},
- "RELEASEDATE": {releaseDate},
- },
- })
-
- result := legacyReleaseDate(md)
- Expect(result).To(Equal(expected))
- },
- Entry("regular mapping", "2020-05-15", "2019-02-10", "2021-01-01", "2021-01-01"),
- Entry("legacy mapping", "2020-05-15", "2019-02-10", "", "2020-05-15"),
- Entry("legacy mapping, originalYear < year", "2018-05-15", "2019-02-10", "2021-01-01", "2021-01-01"),
- Entry("legacy mapping, originalYear empty", "2020-05-15", "", "2021-01-01", "2021-01-01"),
- Entry("legacy mapping, releaseYear", "2020-05-15", "2019-02-10", "2021-01-01", "2021-01-01"),
- Entry("legacy mapping, same dates", "2020-05-15", "2020-05-15", "", "2020-05-15"),
- )
-})
diff --git a/model/metadata/map_mediafile.go b/model/metadata/map_mediafile.go
index b4857df85..c64e8c724 100644
--- a/model/metadata/map_mediafile.go
+++ b/model/metadata/map_mediafile.go
@@ -7,9 +7,9 @@ import (
"math"
"strconv"
+ "github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
- "github.com/navidrome/navidrome/model/id"
"github.com/navidrome/navidrome/utils/str"
)
@@ -53,9 +53,9 @@ func (md Metadata) ToMediaFile(libID int, folderID string) model.MediaFile {
mf.MbzAlbumType = md.String(model.TagReleaseType)
// ReplayGain
- mf.RGAlbumPeak = md.Float(model.TagReplayGainAlbumPeak, 1)
+ mf.RGAlbumPeak = md.NullableFloat(model.TagReplayGainAlbumPeak)
mf.RGAlbumGain = md.mapGain(model.TagReplayGainAlbumGain, model.TagR128AlbumGain)
- mf.RGTrackPeak = md.Float(model.TagReplayGainTrackPeak, 1)
+ mf.RGTrackPeak = md.NullableFloat(model.TagReplayGainTrackPeak)
mf.RGTrackGain = md.mapGain(model.TagReplayGainTrackGain, model.TagR128TrackGain)
// General properties
@@ -77,7 +77,7 @@ func (md Metadata) ToMediaFile(libID int, folderID string) model.MediaFile {
// Persistent IDs
mf.PID = md.trackPID(mf)
- mf.AlbumID = md.albumID(mf)
+ mf.AlbumID = md.albumID(mf, conf.Server.PID.Album)
// BFR These IDs will go away once the UI handle multiple participants.
// BFR For Legacy Subsonic compatibility, we will set them in the API handlers
@@ -104,27 +104,27 @@ func (md Metadata) ToMediaFile(libID int, folderID string) model.MediaFile {
}
func (md Metadata) AlbumID(mf model.MediaFile, pidConf string) string {
- getPID := createGetPID(id.NewHash)
- return getPID(mf, md, pidConf)
+ return md.albumID(mf, pidConf)
}
-func (md Metadata) mapGain(rg, r128 model.TagName) float64 {
+func (md Metadata) mapGain(rg, r128 model.TagName) *float64 {
v := md.Gain(rg)
- if v != 0 {
+ if v != nil {
return v
}
r128value := md.String(r128)
if r128value != "" {
var v, err = strconv.Atoi(r128value)
if err != nil {
- return 0
+ return nil
}
// Convert Q7.8 to float
- var value = float64(v) / 256.0
+ value := float64(v) / 256.0
// Adding 5 dB to normalize with ReplayGain level
- return value + 5
+ value += 5
+ return &value
}
- return 0
+ return nil
}
func (md Metadata) mapLyrics() string {
diff --git a/model/metadata/map_mediafile_test.go b/model/metadata/map_mediafile_test.go
index ddda39bc2..e3adf3fae 100644
--- a/model/metadata/map_mediafile_test.go
+++ b/model/metadata/map_mediafile_test.go
@@ -75,6 +75,23 @@ var _ = Describe("ToMediaFile", func() {
Expect(mf.OriginalYear).To(Equal(1966))
Expect(mf.ReleaseYear).To(Equal(2014))
})
+ DescribeTable("legacyReleaseDate (TaggedLikePicard old behavior)",
+ func(recordingDate, originalDate, releaseDate, expected string) {
+ mf := toMediaFile(model.RawTags{
+ "DATE": {recordingDate},
+ "ORIGINALDATE": {originalDate},
+ "RELEASEDATE": {releaseDate},
+ })
+
+ Expect(mf.ReleaseDate).To(Equal(expected))
+ },
+ Entry("regular mapping", "2020-05-15", "2019-02-10", "2021-01-01", "2021-01-01"),
+ Entry("legacy mapping", "2020-05-15", "2019-02-10", "", "2020-05-15"),
+ Entry("legacy mapping, originalYear < year", "2018-05-15", "2019-02-10", "2021-01-01", "2021-01-01"),
+ Entry("legacy mapping, originalYear empty", "2020-05-15", "", "2021-01-01", "2021-01-01"),
+ Entry("legacy mapping, releaseYear", "2020-05-15", "2019-02-10", "2021-01-01", "2021-01-01"),
+ Entry("legacy mapping, same dates", "2020-05-15", "2020-05-15", "", "2020-05-15"),
+ )
})
Describe("Lyrics", func() {
diff --git a/model/metadata/metadata.go b/model/metadata/metadata.go
index 471c2434c..1372d0034 100644
--- a/model/metadata/metadata.go
+++ b/model/metadata/metadata.go
@@ -103,9 +103,11 @@ func (md Metadata) NumAndTotal(key model.TagName) (int, int) { return md.tuple(k
func (md Metadata) Float(key model.TagName, def ...float64) float64 {
return float(md.first(key), def...)
}
-func (md Metadata) Gain(key model.TagName) float64 {
+func (md Metadata) NullableFloat(key model.TagName) *float64 { return nullableFloat(md.first(key)) }
+
+func (md Metadata) Gain(key model.TagName) *float64 {
v := strings.TrimSpace(strings.Replace(md.first(key), "dB", "", 1))
- return float(v)
+ return nullableFloat(v)
}
func (md Metadata) Pairs(key model.TagName) []Pair {
values := md.tags[key]
@@ -119,14 +121,22 @@ func (md Metadata) first(key model.TagName) string {
}
func float(value string, def ...float64) float64 {
+ v := nullableFloat(value)
+ if v != nil {
+ return *v
+ }
+ if len(def) > 0 {
+ return def[0]
+ }
+ return 0
+}
+
+func nullableFloat(value string) *float64 {
v, err := strconv.ParseFloat(value, 64)
if err != nil || v == math.Inf(-1) || math.IsInf(v, 1) || math.IsNaN(v) {
- if len(def) > 0 {
- return def[0]
- }
- return 0
+ return nil
}
- return v
+ return &v
}
// Used for tracks and discs
@@ -235,10 +245,14 @@ func processPairMapping(name model.TagName, mapping model.TagConf, lowered model
}
}
+ // always parse id3 pairs. For lyrics, Taglib appears to always provide lyrics:xxx
+ // Prefer that over format-specific tags
+ id3Base := parseID3Pairs(name, lowered)
+
if len(aliasValues) > 0 {
- return parseVorbisPairs(aliasValues)
+ id3Base = append(id3Base, parseVorbisPairs(aliasValues)...)
}
- return parseID3Pairs(name, lowered)
+ return id3Base
}
func parseID3Pairs(name model.TagName, lowered model.Tags) []string {
diff --git a/model/metadata/metadata_test.go b/model/metadata/metadata_test.go
index d7473afa7..82afd8657 100644
--- a/model/metadata/metadata_test.go
+++ b/model/metadata/metadata_test.go
@@ -8,6 +8,7 @@ import (
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/model/metadata"
"github.com/navidrome/navidrome/utils"
+ "github.com/navidrome/navidrome/utils/gg"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
)
@@ -257,38 +258,39 @@ var _ = Describe("Metadata", func() {
}
DescribeTable("Gain",
- func(tagValue string, expected float64) {
+ func(tagValue string, expected *float64) {
mf := createMF("replaygain_track_gain", tagValue)
Expect(mf.RGTrackGain).To(Equal(expected))
},
- Entry("0", "0", 0.0),
- Entry("1.2dB", "1.2dB", 1.2),
- Entry("Infinity", "Infinity", 0.0),
- Entry("Invalid value", "INVALID VALUE", 0.0),
- Entry("NaN", "NaN", 0.0),
+ Entry("0", "0", gg.P(0.0)),
+ Entry("1.2dB", "1.2dB", gg.P(1.2)),
+ Entry("Infinity", "Infinity", nil),
+ Entry("Invalid value", "INVALID VALUE", nil),
+ Entry("NaN", "NaN", nil),
)
DescribeTable("Peak",
- func(tagValue string, expected float64) {
+ func(tagValue string, expected *float64) {
mf := createMF("replaygain_track_peak", tagValue)
Expect(mf.RGTrackPeak).To(Equal(expected))
},
- Entry("0", "0", 0.0),
- Entry("0.5", "0.5", 0.5),
- Entry("Invalid dB suffix", "0.7dB", 1.0),
- Entry("Infinity", "Infinity", 1.0),
- Entry("Invalid value", "INVALID VALUE", 1.0),
- Entry("NaN", "NaN", 1.0),
+ Entry("0", "0", gg.P(0.0)),
+ Entry("1.0", "1.0", gg.P(1.0)),
+ Entry("0.5", "0.5", gg.P(0.5)),
+ Entry("Invalid dB suffix", "0.7dB", nil),
+ Entry("Infinity", "Infinity", nil),
+ Entry("Invalid value", "INVALID VALUE", nil),
+ Entry("NaN", "NaN", nil),
)
DescribeTable("getR128GainValue",
- func(tagValue string, expected float64) {
+ func(tagValue string, expected *float64) {
mf := createMF("r128_track_gain", tagValue)
Expect(mf.RGTrackGain).To(Equal(expected))
},
- Entry("0", "0", 5.0),
- Entry("-3776", "-3776", -9.75),
- Entry("Infinity", "Infinity", 0.0),
- Entry("Invalid value", "INVALID VALUE", 0.0),
+ Entry("0", "0", gg.P(5.0)),
+ Entry("-3776", "-3776", gg.P(-9.75)),
+ Entry("Infinity", "Infinity", nil),
+ Entry("Invalid value", "INVALID VALUE", nil),
)
})
diff --git a/model/metadata/persistent_ids.go b/model/metadata/persistent_ids.go
index a71749e81..b45882946 100644
--- a/model/metadata/persistent_ids.go
+++ b/model/metadata/persistent_ids.go
@@ -2,6 +2,7 @@ package metadata
import (
"cmp"
+ "fmt"
"path/filepath"
"strings"
@@ -16,17 +17,20 @@ import (
type hashFunc = func(...string) string
-// getPID returns the persistent ID for a given spec, getting the referenced values from the metadata
+// createGetPID returns a function that calculates the persistent ID for a given spec, getting the referenced values from the metadata
// The spec is a pipe-separated list of fields, where each field is a comma-separated list of attributes
// Attributes can be either tags or some processed values like folder, albumid, albumartistid, etc.
// For each field, it gets all its attributes values and concatenates them, then hashes the result.
// If a field is empty, it is skipped and the function looks for the next field.
-func createGetPID(hash hashFunc) func(mf model.MediaFile, md Metadata, spec string) string {
- var getPID func(mf model.MediaFile, md Metadata, spec string) string
- getAttr := func(mf model.MediaFile, md Metadata, attr string) string {
+type getPIDFunc = func(mf model.MediaFile, md Metadata, spec string, prependLibId bool) string
+
+func createGetPID(hash hashFunc) getPIDFunc {
+ var getPID getPIDFunc
+ getAttr := func(mf model.MediaFile, md Metadata, attr string, prependLibId bool) string {
+ attr = strings.TrimSpace(strings.ToLower(attr))
switch attr {
case "albumid":
- return getPID(mf, md, conf.Server.PID.Album)
+ return getPID(mf, md, conf.Server.PID.Album, prependLibId)
case "folder":
return filepath.Dir(mf.Path)
case "albumartistid":
@@ -38,14 +42,14 @@ func createGetPID(hash hashFunc) func(mf model.MediaFile, md Metadata, spec stri
}
return md.String(model.TagName(attr))
}
- getPID = func(mf model.MediaFile, md Metadata, spec string) string {
+ getPID = func(mf model.MediaFile, md Metadata, spec string, prependLibId bool) string {
pid := ""
fields := strings.Split(spec, "|")
for _, field := range fields {
attributes := strings.Split(field, ",")
hasValue := false
values := slice.Map(attributes, func(attr string) string {
- v := getAttr(mf, md, attr)
+ v := getAttr(mf, md, attr, prependLibId)
if v != "" {
hasValue = true
}
@@ -56,32 +60,35 @@ func createGetPID(hash hashFunc) func(mf model.MediaFile, md Metadata, spec stri
break
}
}
+ if prependLibId {
+ pid = fmt.Sprintf("%d\\%s", mf.LibraryID, pid)
+ }
return hash(pid)
}
- return func(mf model.MediaFile, md Metadata, spec string) string {
+ return func(mf model.MediaFile, md Metadata, spec string, prependLibId bool) string {
switch spec {
case "track_legacy":
- return legacyTrackID(mf)
+ return legacyTrackID(mf, prependLibId)
case "album_legacy":
- return legacyAlbumID(md)
+ return legacyAlbumID(mf, md, prependLibId)
}
- return getPID(mf, md, spec)
+ return getPID(mf, md, spec, prependLibId)
}
}
func (md Metadata) trackPID(mf model.MediaFile) string {
- return createGetPID(id.NewHash)(mf, md, conf.Server.PID.Track)
+ return createGetPID(id.NewHash)(mf, md, conf.Server.PID.Track, true)
}
-func (md Metadata) albumID(mf model.MediaFile) string {
- return createGetPID(id.NewHash)(mf, md, conf.Server.PID.Album)
+func (md Metadata) albumID(mf model.MediaFile, pidConf string) string {
+ return createGetPID(id.NewHash)(mf, md, pidConf, true)
}
// BFR Must be configurable?
func (md Metadata) artistID(name string) string {
mf := model.MediaFile{AlbumArtist: name}
- return createGetPID(id.NewHash)(mf, md, "albumartistid")
+ return createGetPID(id.NewHash)(mf, md, "albumartistid", false)
}
func (md Metadata) mapTrackTitle() string {
diff --git a/model/metadata/persistent_ids_test.go b/model/metadata/persistent_ids_test.go
index 6903abc05..7ae0c91f7 100644
--- a/model/metadata/persistent_ids_test.go
+++ b/model/metadata/persistent_ids_test.go
@@ -15,7 +15,7 @@ var _ = Describe("getPID", func() {
md Metadata
mf model.MediaFile
sum hashFunc
- getPID func(mf model.MediaFile, md Metadata, spec string) string
+ getPID getPIDFunc
)
BeforeEach(func() {
@@ -28,7 +28,7 @@ var _ = Describe("getPID", func() {
When("no attributes were present", func() {
It("should return empty pid", func() {
md.tags = map[model.TagName][]string{}
- pid := getPID(mf, md, spec)
+ pid := getPID(mf, md, spec, false)
Expect(pid).To(Equal("()"))
})
})
@@ -40,7 +40,7 @@ var _ = Describe("getPID", func() {
"discnumber": {"1"},
"tracknumber": {"1"},
}
- Expect(getPID(mf, md, spec)).To(Equal("(mbtrackid)"))
+ Expect(getPID(mf, md, spec, false)).To(Equal("(mbtrackid)"))
})
})
When("only first field is present", func() {
@@ -48,7 +48,7 @@ var _ = Describe("getPID", func() {
md.tags = map[model.TagName][]string{
"musicbrainz_trackid": {"mbtrackid"},
}
- Expect(getPID(mf, md, spec)).To(Equal("(mbtrackid)"))
+ Expect(getPID(mf, md, spec, false)).To(Equal("(mbtrackid)"))
})
})
When("first is empty, but second field is present", func() {
@@ -57,10 +57,11 @@ var _ = Describe("getPID", func() {
"album": {"album name"},
"discnumber": {"1"},
}
- Expect(getPID(mf, md, spec)).To(Equal("(album name\\1\\)"))
+ Expect(getPID(mf, md, spec, false)).To(Equal("(album name\\1\\)"))
})
})
})
+
Context("calculated attributes", func() {
BeforeEach(func() {
DeferCleanup(configtest.SetupConfig())
@@ -72,7 +73,7 @@ var _ = Describe("getPID", func() {
md.tags = map[model.TagName][]string{"title": {"title"}}
md.filePath = "/path/to/file.mp3"
mf.Title = "Title"
- Expect(getPID(mf, md, spec)).To(Equal("(Title)"))
+ Expect(getPID(mf, md, spec, false)).To(Equal("(Title)"))
})
})
When("field is folder", func() {
@@ -80,7 +81,7 @@ var _ = Describe("getPID", func() {
spec := "folder|title"
md.tags = map[model.TagName][]string{"title": {"title"}}
mf.Path = "/path/to/file.mp3"
- Expect(getPID(mf, md, spec)).To(Equal("(/path/to)"))
+ Expect(getPID(mf, md, spec, false)).To(Equal("(/path/to)"))
})
})
When("field is albumid", func() {
@@ -93,7 +94,7 @@ var _ = Describe("getPID", func() {
"releasedate": {"2021-01-01"},
}
mf.AlbumArtist = "Album Artist"
- Expect(getPID(mf, md, spec)).To(Equal("(((album artist)\\album name\\version\\2021-01-01))"))
+ Expect(getPID(mf, md, spec, false)).To(Equal("(((album artist)\\album name\\version\\2021-01-01))"))
})
})
When("field is albumartistid", func() {
@@ -103,14 +104,168 @@ var _ = Describe("getPID", func() {
"albumartist": {"Album Artist"},
}
mf.AlbumArtist = "Album Artist"
- Expect(getPID(mf, md, spec)).To(Equal("((album artist))"))
+ Expect(getPID(mf, md, spec, false)).To(Equal("((album artist))"))
})
})
When("field is album", func() {
It("should return the pid", func() {
spec := "album|title"
md.tags = map[model.TagName][]string{"album": {"Album Name"}}
- Expect(getPID(mf, md, spec)).To(Equal("(album name)"))
+ Expect(getPID(mf, md, spec, false)).To(Equal("(album name)"))
+ })
+ })
+ })
+
+ Context("edge cases", func() {
+ When("the spec has spaces between groups", func() {
+ It("should return the pid", func() {
+ spec := "albumartist| Album"
+ md.tags = map[model.TagName][]string{
+ "album": {"album name"},
+ }
+ Expect(getPID(mf, md, spec, false)).To(Equal("(album name)"))
+ })
+ })
+ When("the spec has spaces", func() {
+ It("should return the pid", func() {
+ spec := "albumartist, album"
+ md.tags = map[model.TagName][]string{
+ "albumartist": {"Album Artist"},
+ "album": {"album name"},
+ }
+ Expect(getPID(mf, md, spec, false)).To(Equal("(Album Artist\\album name)"))
+ })
+ })
+ When("the spec has mixed case fields", func() {
+ It("should return the pid", func() {
+ spec := "albumartist,Album"
+ md.tags = map[model.TagName][]string{
+ "albumartist": {"Album Artist"},
+ "album": {"album name"},
+ }
+ Expect(getPID(mf, md, spec, false)).To(Equal("(Album Artist\\album name)"))
+ })
+ })
+ })
+
+ Context("prependLibId functionality", func() {
+ BeforeEach(func() {
+ mf.LibraryID = 42
+ })
+ When("prependLibId is true", func() {
+ It("should prepend library ID to the hash input", func() {
+ spec := "album"
+ md.tags = map[model.TagName][]string{"album": {"Test Album"}}
+ pid := getPID(mf, md, spec, true)
+ // The hash function should receive "42\test album" as input
+ Expect(pid).To(Equal("(42\\test album)"))
+ })
+ })
+ When("prependLibId is false", func() {
+ It("should not prepend library ID to the hash input", func() {
+ spec := "album"
+ md.tags = map[model.TagName][]string{"album": {"Test Album"}}
+ pid := getPID(mf, md, spec, false)
+ // The hash function should receive "test album" as input
+ Expect(pid).To(Equal("(test album)"))
+ })
+ })
+ When("prependLibId is true with complex spec", func() {
+ It("should prepend library ID to the final hash input", func() {
+ spec := "musicbrainz_trackid|album,tracknumber"
+ md.tags = map[model.TagName][]string{
+ "album": {"Test Album"},
+ "tracknumber": {"1"},
+ }
+ pid := getPID(mf, md, spec, true)
+ // Should use the fallback field and prepend library ID
+ Expect(pid).To(Equal("(42\\test album\\1)"))
+ })
+ })
+ When("prependLibId is true with nested albumid", func() {
+ It("should handle nested albumid calls correctly", func() {
+ DeferCleanup(configtest.SetupConfig())
+ conf.Server.PID.Album = "album"
+ spec := "albumid"
+ md.tags = map[model.TagName][]string{"album": {"Test Album"}}
+ mf.AlbumArtist = "Test Artist"
+ pid := getPID(mf, md, spec, true)
+ // The albumid call should also use prependLibId=true
+ Expect(pid).To(Equal("(42\\(42\\test album))"))
+ })
+ })
+ })
+
+ Context("legacy specs", func() {
+ Context("track_legacy", func() {
+ When("library ID is default (1)", func() {
+ It("should not prepend library ID even when prependLibId is true", func() {
+ mf.Path = "/path/to/track.mp3"
+ mf.LibraryID = 1 // Default library ID
+ // With default library, both should be the same
+ pidTrue := getPID(mf, md, "track_legacy", true)
+ pidFalse := getPID(mf, md, "track_legacy", false)
+ Expect(pidTrue).To(Equal(pidFalse))
+ Expect(pidTrue).NotTo(BeEmpty())
+ })
+ })
+ When("library ID is non-default", func() {
+ It("should prepend library ID when prependLibId is true", func() {
+ mf.Path = "/path/to/track.mp3"
+ mf.LibraryID = 2 // Non-default library ID
+ pidTrue := getPID(mf, md, "track_legacy", true)
+ pidFalse := getPID(mf, md, "track_legacy", false)
+ Expect(pidTrue).NotTo(Equal(pidFalse))
+ Expect(pidTrue).NotTo(BeEmpty())
+ Expect(pidFalse).NotTo(BeEmpty())
+ })
+ })
+ When("library ID is non-default but prependLibId is false", func() {
+ It("should not prepend library ID", func() {
+ mf.Path = "/path/to/track.mp3"
+ mf.LibraryID = 3
+ mf2 := mf
+ mf2.LibraryID = 1 // Default library
+ pidNonDefault := getPID(mf, md, "track_legacy", false)
+ pidDefault := getPID(mf2, md, "track_legacy", false)
+ // Should be the same since prependLibId=false
+ Expect(pidNonDefault).To(Equal(pidDefault))
+ })
+ })
+ })
+ Context("album_legacy", func() {
+ When("library ID is default (1)", func() {
+ It("should not prepend library ID even when prependLibId is true", func() {
+ md.tags = map[model.TagName][]string{"album": {"Test Album"}}
+ mf.LibraryID = 1 // Default library ID
+ pidTrue := getPID(mf, md, "album_legacy", true)
+ pidFalse := getPID(mf, md, "album_legacy", false)
+ Expect(pidTrue).To(Equal(pidFalse))
+ Expect(pidTrue).NotTo(BeEmpty())
+ })
+ })
+ When("library ID is non-default", func() {
+ It("should prepend library ID when prependLibId is true", func() {
+ md.tags = map[model.TagName][]string{"album": {"Test Album"}}
+ mf.LibraryID = 2 // Non-default library ID
+ pidTrue := getPID(mf, md, "album_legacy", true)
+ pidFalse := getPID(mf, md, "album_legacy", false)
+ Expect(pidTrue).NotTo(Equal(pidFalse))
+ Expect(pidTrue).NotTo(BeEmpty())
+ Expect(pidFalse).NotTo(BeEmpty())
+ })
+ })
+ When("library ID is non-default but prependLibId is false", func() {
+ It("should not prepend library ID", func() {
+ md.tags = map[model.TagName][]string{"album": {"Test Album"}}
+ mf.LibraryID = 3
+ mf2 := mf
+ mf2.LibraryID = 1 // Default library
+ pidNonDefault := getPID(mf, md, "album_legacy", false)
+ pidDefault := getPID(mf2, md, "album_legacy", false)
+ // Should be the same since prependLibId=false
+ Expect(pidNonDefault).To(Equal(pidDefault))
+ })
})
})
})
diff --git a/model/participants.go b/model/participants.go
index 5f07bf42c..afbda10de 100644
--- a/model/participants.go
+++ b/model/participants.go
@@ -25,6 +25,8 @@ var (
RoleRemixer = Role{"remixer"}
RoleDJMixer = Role{"djmixer"}
RolePerformer = Role{"performer"}
+ // RoleMainCredit is a credit where the artist is an album artist or artist
+ RoleMainCredit = Role{"maincredit"}
)
var AllRoles = map[string]Role{
@@ -41,6 +43,7 @@ var AllRoles = map[string]Role{
RoleRemixer.role: RoleRemixer,
RoleDJMixer.role: RoleDJMixer,
RolePerformer.role: RolePerformer,
+ RoleMainCredit.role: RoleMainCredit,
}
// Role represents the role of an artist in a track or album.
diff --git a/model/playlist.go b/model/playlist.go
index 521adfcd0..a87019ed5 100644
--- a/model/playlist.go
+++ b/model/playlist.go
@@ -1,10 +1,8 @@
package model
import (
- "fmt"
"slices"
"strconv"
- "strings"
"time"
"github.com/navidrome/navidrome/model/criteria"
@@ -42,6 +40,21 @@ func (pls Playlist) MediaFiles() MediaFiles {
return pls.Tracks.MediaFiles()
}
+func (pls *Playlist) refreshStats() {
+ pls.SongCount = len(pls.Tracks)
+ pls.Duration = 0
+ pls.Size = 0
+ for _, t := range pls.Tracks {
+ pls.Duration += t.MediaFile.Duration
+ pls.Size += t.MediaFile.Size
+ }
+}
+
+func (pls *Playlist) SetTracks(tracks PlaylistTracks) {
+ pls.Tracks = tracks
+ pls.refreshStats()
+}
+
func (pls *Playlist) RemoveTracks(idxToRemove []int) {
var newTracks PlaylistTracks
for i, t := range pls.Tracks {
@@ -51,22 +64,15 @@ func (pls *Playlist) RemoveTracks(idxToRemove []int) {
newTracks = append(newTracks, t)
}
pls.Tracks = newTracks
+ pls.refreshStats()
}
-// ToM3U8 exports the playlist to the Extended M3U8 format, as specified in
-// https://docs.fileformat.com/audio/m3u/#extended-m3u
+// ToM3U8 exports the playlist to the Extended M3U8 format
func (pls *Playlist) ToM3U8() string {
- buf := strings.Builder{}
- buf.WriteString("#EXTM3U\n")
- buf.WriteString(fmt.Sprintf("#PLAYLIST:%s\n", pls.Name))
- for _, t := range pls.Tracks {
- buf.WriteString(fmt.Sprintf("#EXTINF:%.f,%s - %s\n", t.Duration, t.Artist, t.Title))
- buf.WriteString(t.AbsolutePath() + "\n")
- }
- return buf.String()
+ return pls.MediaFiles().ToM3U8(pls.Name, true)
}
-func (pls *Playlist) AddTracks(mediaFileIds []string) {
+func (pls *Playlist) AddMediaFilesByID(mediaFileIds []string) {
pos := len(pls.Tracks)
for _, mfId := range mediaFileIds {
pos++
@@ -78,6 +84,7 @@ func (pls *Playlist) AddTracks(mediaFileIds []string) {
}
pls.Tracks = append(pls.Tracks, t)
}
+ pls.refreshStats()
}
func (pls *Playlist) AddMediaFiles(mfs MediaFiles) {
@@ -92,6 +99,7 @@ func (pls *Playlist) AddMediaFiles(mfs MediaFiles) {
}
pls.Tracks = append(pls.Tracks, t)
}
+ pls.refreshStats()
}
func (pls Playlist) CoverArtID() ArtworkID {
@@ -111,6 +119,7 @@ type PlaylistRepository interface {
FindByPath(path string) (*Playlist, error)
Delete(id string) error
Tracks(playlistId string, refreshSmartPlaylist bool) PlaylistTrackRepository
+ GetPlaylists(mediaFileId string) (Playlists, error)
}
type PlaylistTrack struct {
diff --git a/model/playlists_test.go b/model/playlist_test.go
similarity index 71%
rename from model/playlists_test.go
rename to model/playlist_test.go
index 600e116cc..a54cecd53 100644
--- a/model/playlists_test.go
+++ b/model/playlist_test.go
@@ -13,13 +13,17 @@ var _ = Describe("Playlist", func() {
pls = model.Playlist{Name: "Mellow sunset"}
pls.Tracks = model.PlaylistTracks{
{MediaFile: model.MediaFile{Artist: "Morcheeba feat. Kurt Wagner", Title: "What New York Couples Fight About",
- Duration: 377.84, Path: "/music/library/Morcheeba/Charango/01-06 What New York Couples Fight About.mp3"}},
+ Duration: 377.84,
+ LibraryPath: "/music/library", Path: "Morcheeba/Charango/01-06 What New York Couples Fight About.mp3"}},
{MediaFile: model.MediaFile{Artist: "A Tribe Called Quest", Title: "Description of a Fool (Groove Armada's Acoustic mix)",
- Duration: 374.49, Path: "/music/library/Groove Armada/Back to Mine_ Groove Armada/01-01 Description of a Fool (Groove Armada's Acoustic mix).mp3"}},
+ Duration: 374.49,
+ LibraryPath: "/music/library", Path: "Groove Armada/Back to Mine_ Groove Armada/01-01 Description of a Fool (Groove Armada's Acoustic mix).mp3"}},
{MediaFile: model.MediaFile{Artist: "Lou Reed", Title: "Walk on the Wild Side",
- Duration: 253.1, Path: "/music/library/Lou Reed/Walk on the Wild Side_ The Best of Lou Reed/01-06 Walk on the Wild Side.m4a"}},
+ Duration: 253.1,
+ LibraryPath: "/music/library", Path: "Lou Reed/Walk on the Wild Side_ The Best of Lou Reed/01-06 Walk on the Wild Side.m4a"}},
{MediaFile: model.MediaFile{Artist: "Legião Urbana", Title: "On the Way Home",
- Duration: 163.89, Path: "/music/library/Legião Urbana/Música p_ acampamentos/02-05 On the Way Home.mp3"}},
+ Duration: 163.89,
+ LibraryPath: "/music/library", Path: "Legião Urbana/Música p_ acampamentos/02-05 On the Way Home.mp3"}},
}
})
It("generates the correct M3U format", func() {
diff --git a/model/playqueue.go b/model/playqueue.go
index 52ba173d3..03b562253 100644
--- a/model/playqueue.go
+++ b/model/playqueue.go
@@ -7,7 +7,7 @@ import (
type PlayQueue struct {
ID string `structs:"id" json:"id"`
UserID string `structs:"user_id" json:"userId"`
- Current string `structs:"current" json:"current"`
+ Current int `structs:"current" json:"current"`
Position int64 `structs:"position" json:"position"`
ChangedBy string `structs:"changed_by" json:"changedBy"`
Items MediaFiles `structs:"-" json:"items,omitempty"`
@@ -18,6 +18,11 @@ type PlayQueue struct {
type PlayQueues []PlayQueue
type PlayQueueRepository interface {
- Store(queue *PlayQueue) error
+ Store(queue *PlayQueue, colNames ...string) error
+ // Retrieve returns the playqueue without loading the full MediaFiles
+ // (Items only contain IDs)
Retrieve(userId string) (*PlayQueue, error)
+ // RetrieveWithMediaFiles returns the playqueue with full MediaFiles loaded
+ RetrieveWithMediaFiles(userId string) (*PlayQueue, error)
+ Clear(userId string) error
}
diff --git a/model/request/request.go b/model/request/request.go
index 5f2980340..8d7919298 100644
--- a/model/request/request.go
+++ b/model/request/request.go
@@ -17,6 +17,7 @@ const (
Transcoding = contextKey("transcoding")
ClientUniqueId = contextKey("clientUniqueId")
ReverseProxyIp = contextKey("reverseProxyIp")
+ InternalAuth = contextKey("internalAuth") // Used for internal API calls, e.g., from the plugins
)
var allKeys = []contextKey{
@@ -28,6 +29,7 @@ var allKeys = []contextKey{
Transcoding,
ClientUniqueId,
ReverseProxyIp,
+ InternalAuth,
}
func WithUser(ctx context.Context, u model.User) context.Context {
@@ -62,6 +64,10 @@ func WithReverseProxyIp(ctx context.Context, reverseProxyIp string) context.Cont
return context.WithValue(ctx, ReverseProxyIp, reverseProxyIp)
}
+func WithInternalAuth(ctx context.Context, username string) context.Context {
+ return context.WithValue(ctx, InternalAuth, username)
+}
+
func UserFrom(ctx context.Context) (model.User, bool) {
v, ok := ctx.Value(User).(model.User)
return v, ok
@@ -102,6 +108,15 @@ func ReverseProxyIpFrom(ctx context.Context) (string, bool) {
return v, ok
}
+func InternalAuthFrom(ctx context.Context) (string, bool) {
+ if v := ctx.Value(InternalAuth); v != nil {
+ if username, ok := v.(string); ok {
+ return username, true
+ }
+ }
+ return "", false
+}
+
func AddValues(ctx, requestCtx context.Context) context.Context {
for _, key := range allKeys {
if v := requestCtx.Value(key); v != nil {
diff --git a/model/scanner.go b/model/scanner.go
new file mode 100644
index 000000000..389c77f87
--- /dev/null
+++ b/model/scanner.go
@@ -0,0 +1,81 @@
+package model
+
+import (
+ "context"
+ "fmt"
+ "strconv"
+ "strings"
+ "time"
+)
+
+// ScanTarget represents a specific folder within a library to be scanned.
+// NOTE: This struct is used as a map key, so it should only contain comparable types.
+type ScanTarget struct {
+ LibraryID int
+ FolderPath string // Relative path within the library, or "" for entire library
+}
+
+func (st ScanTarget) String() string {
+ return fmt.Sprintf("%d:%s", st.LibraryID, st.FolderPath)
+}
+
+// ScannerStatus holds information about the current scan status
+type ScannerStatus struct {
+ Scanning bool
+ LastScan time.Time
+ Count uint32
+ FolderCount uint32
+ LastError string
+ ScanType string
+ ElapsedTime time.Duration
+}
+
+type Scanner interface {
+ // ScanAll starts a scan of all libraries. This is a blocking operation.
+ ScanAll(ctx context.Context, fullScan bool) (warnings []string, err error)
+ // ScanFolders scans specific library/folder pairs, recursing into subdirectories.
+ // If targets is nil, it scans all libraries. This is a blocking operation.
+ ScanFolders(ctx context.Context, fullScan bool, targets []ScanTarget) (warnings []string, err error)
+ Status(context.Context) (*ScannerStatus, error)
+}
+
+// ParseTargets parses scan targets strings into ScanTarget structs.
+// Example: []string{"1:Music/Rock", "2:Classical"}
+func ParseTargets(libFolders []string) ([]ScanTarget, error) {
+ targets := make([]ScanTarget, 0, len(libFolders))
+
+ for _, part := range libFolders {
+ part = strings.TrimSpace(part)
+ if part == "" {
+ continue
+ }
+
+ // Split by the first colon
+ colonIdx := strings.Index(part, ":")
+ if colonIdx == -1 {
+ return nil, fmt.Errorf("invalid target format: %q (expected libraryID:folderPath)", part)
+ }
+
+ libIDStr := part[:colonIdx]
+ folderPath := part[colonIdx+1:]
+
+ libID, err := strconv.Atoi(libIDStr)
+ if err != nil {
+ return nil, fmt.Errorf("invalid library ID %q: %w", libIDStr, err)
+ }
+ if libID <= 0 {
+ return nil, fmt.Errorf("invalid library ID %q", libIDStr)
+ }
+
+ targets = append(targets, ScanTarget{
+ LibraryID: libID,
+ FolderPath: folderPath,
+ })
+ }
+
+ if len(targets) == 0 {
+ return nil, fmt.Errorf("no valid targets found")
+ }
+
+ return targets, nil
+}
diff --git a/model/scanner_test.go b/model/scanner_test.go
new file mode 100644
index 000000000..8ca0c53fa
--- /dev/null
+++ b/model/scanner_test.go
@@ -0,0 +1,89 @@
+package model_test
+
+import (
+ "github.com/navidrome/navidrome/model"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("ParseTargets", func() {
+ It("parses multiple entries in slice", func() {
+ targets, err := model.ParseTargets([]string{"1:Music/Rock", "1:Music/Jazz", "2:Classical"})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(targets).To(HaveLen(3))
+ Expect(targets[0].LibraryID).To(Equal(1))
+ Expect(targets[0].FolderPath).To(Equal("Music/Rock"))
+ Expect(targets[1].LibraryID).To(Equal(1))
+ Expect(targets[1].FolderPath).To(Equal("Music/Jazz"))
+ Expect(targets[2].LibraryID).To(Equal(2))
+ Expect(targets[2].FolderPath).To(Equal("Classical"))
+ })
+
+ It("handles empty folder paths", func() {
+ targets, err := model.ParseTargets([]string{"1:", "2:"})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(targets).To(HaveLen(2))
+ Expect(targets[0].FolderPath).To(Equal(""))
+ Expect(targets[1].FolderPath).To(Equal(""))
+ })
+
+ It("trims whitespace from entries", func() {
+ targets, err := model.ParseTargets([]string{" 1:Music/Rock", " 2:Classical "})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(targets).To(HaveLen(2))
+ Expect(targets[0].LibraryID).To(Equal(1))
+ Expect(targets[0].FolderPath).To(Equal("Music/Rock"))
+ Expect(targets[1].LibraryID).To(Equal(2))
+ Expect(targets[1].FolderPath).To(Equal("Classical"))
+ })
+
+ It("skips empty strings", func() {
+ targets, err := model.ParseTargets([]string{"1:Music/Rock", "", "2:Classical"})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(targets).To(HaveLen(2))
+ })
+
+ It("handles paths with colons", func() {
+ targets, err := model.ParseTargets([]string{"1:C:/Music/Rock", "2:/path:with:colons"})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(targets).To(HaveLen(2))
+ Expect(targets[0].FolderPath).To(Equal("C:/Music/Rock"))
+ Expect(targets[1].FolderPath).To(Equal("/path:with:colons"))
+ })
+
+ It("returns error for invalid format without colon", func() {
+ _, err := model.ParseTargets([]string{"1Music/Rock"})
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("invalid target format"))
+ })
+
+ It("returns error for non-numeric library ID", func() {
+ _, err := model.ParseTargets([]string{"abc:Music/Rock"})
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("invalid library ID"))
+ })
+
+ It("returns error for negative library ID", func() {
+ _, err := model.ParseTargets([]string{"-1:Music/Rock"})
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("invalid library ID"))
+ })
+
+ It("returns error for zero library ID", func() {
+ _, err := model.ParseTargets([]string{"0:Music/Rock"})
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("invalid library ID"))
+ })
+
+ It("returns error for empty input", func() {
+ _, err := model.ParseTargets([]string{})
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("no valid targets found"))
+ })
+
+ It("returns error for all empty strings", func() {
+ _, err := model.ParseTargets([]string{"", " ", ""})
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("no valid targets found"))
+ })
+})
diff --git a/model/searchable.go b/model/searchable.go
index d37299997..631a11726 100644
--- a/model/searchable.go
+++ b/model/searchable.go
@@ -1,5 +1,5 @@
package model
type SearchableRepository[T any] interface {
- Search(q string, offset, size int, includeMissing bool) (T, error)
+ Search(q string, offset, size int, options ...QueryOptions) (T, error)
}
diff --git a/model/share.go b/model/share.go
index 0f52f5323..acb5fb428 100644
--- a/model/share.go
+++ b/model/share.go
@@ -2,7 +2,6 @@ package model
import (
"cmp"
- "fmt"
"strings"
"time"
@@ -50,17 +49,9 @@ func (s Share) CoverArtID() ArtworkID {
type Shares []Share
-// ToM3U8 exports the playlist to the Extended M3U8 format, as specified in
-// https://docs.fileformat.com/audio/m3u/#extended-m3u
+// ToM3U8 exports the share to the Extended M3U8 format.
func (s Share) ToM3U8() string {
- buf := strings.Builder{}
- buf.WriteString("#EXTM3U\n")
- buf.WriteString(fmt.Sprintf("#PLAYLIST:%s\n", cmp.Or(s.Description, s.ID)))
- for _, t := range s.Tracks {
- buf.WriteString(fmt.Sprintf("#EXTINF:%.f,%s - %s\n", t.Duration, t.Artist, t.Title))
- buf.WriteString(t.Path + "\n")
- }
- return buf.String()
+ return s.Tracks.ToM3U8(cmp.Or(s.Description, s.ID), false)
}
type ShareRepository interface {
diff --git a/model/tag.go b/model/tag.go
index a1f4e28da..674f688ca 100644
--- a/model/tag.go
+++ b/model/tag.go
@@ -12,11 +12,11 @@ import (
)
type Tag struct {
- ID string `json:"id,omitempty"`
- TagName TagName `json:"tagName,omitempty"`
- TagValue string `json:"tagValue,omitempty"`
- AlbumCount int `json:"albumCount,omitempty"`
- MediaFileCount int `json:"songCount,omitempty"`
+ ID string `json:"id,omitempty"`
+ TagName TagName `json:"tagName,omitempty"`
+ TagValue string `json:"tagValue,omitempty"`
+ AlbumCount int `json:"albumCount,omitempty"`
+ SongCount int `json:"songCount,omitempty"`
}
type TagList []Tag
@@ -153,7 +153,7 @@ func (t Tags) Add(name TagName, v string) {
}
type TagRepository interface {
- Add(...Tag) error
+ Add(libraryID int, tags ...Tag) error
UpdateCounts() error
}
diff --git a/model/tag_mappings.go b/model/tag_mappings.go
index d54f51f43..bfe098f77 100644
--- a/model/tag_mappings.go
+++ b/model/tag_mappings.go
@@ -139,7 +139,9 @@ func compileSplitRegex(tagName TagName, split []string) *regexp.Regexp {
}
// If no valid separators remain, return the original value.
if len(escaped) == 0 {
- log.Warn("No valid separators found in split list", "split", split, "tag", tagName)
+ if len(split) > 0 {
+ log.Warn("No valid separators found in split list", "split", split, "tag", tagName)
+ }
return nil
}
@@ -147,7 +149,7 @@ func compileSplitRegex(tagName TagName, split []string) *regexp.Regexp {
pattern := "(?i)(" + strings.Join(escaped, "|") + ")"
re, err := regexp.Compile(pattern)
if err != nil {
- log.Error("Error compiling regexp", "pattern", pattern, "tag", tagName, "err", err)
+ log.Warn("Error compiling regexp for split list", "pattern", pattern, "tag", tagName, "split", split, err)
return nil
}
return re
diff --git a/model/user.go b/model/user.go
index 7c41ac041..aabedc096 100644
--- a/model/user.go
+++ b/model/user.go
@@ -1,6 +1,8 @@
package model
-import "time"
+import (
+ "time"
+)
type User struct {
ID string `structs:"id" json:"id"`
@@ -13,6 +15,9 @@ type User struct {
CreatedAt time.Time `structs:"created_at" json:"createdAt"`
UpdatedAt time.Time `structs:"updated_at" json:"updatedAt"`
+ // Library associations (many-to-many relationship)
+ Libraries Libraries `structs:"-" json:"libraries,omitempty"`
+
// This is only available on the backend, and it is never sent over the wire
Password string `structs:"-" json:"-"`
// This is used to set or change a password when calling Put. If it is empty, the password is not changed.
@@ -22,6 +27,18 @@ type User struct {
CurrentPassword string `structs:"current_password,omitempty" json:"currentPassword,omitempty"`
}
+func (u User) HasLibraryAccess(libraryID int) bool {
+ if u.IsAdmin {
+ return true // Admin users have access to all libraries
+ }
+ for _, lib := range u.Libraries {
+ if lib.ID == libraryID {
+ return true
+ }
+ }
+ return false
+}
+
type Users []User
type UserRepository interface {
@@ -35,4 +52,8 @@ type UserRepository interface {
FindByUsername(username string) (*User, error)
// FindByUsernameWithPassword is the same as above, but also returns the decrypted password
FindByUsernameWithPassword(username string) (*User, error)
+
+ // Library association methods
+ GetUserLibraries(userID string) (Libraries, error)
+ SetUserLibraries(userID string, libraryIDs []int) error
}
diff --git a/model/user_test.go b/model/user_test.go
new file mode 100644
index 000000000..ab66a29a9
--- /dev/null
+++ b/model/user_test.go
@@ -0,0 +1,83 @@
+package model_test
+
+import (
+ "github.com/navidrome/navidrome/model"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("User", func() {
+ var user model.User
+ var libraries model.Libraries
+
+ BeforeEach(func() {
+ libraries = model.Libraries{
+ {ID: 1, Name: "Rock Library", Path: "/music/rock"},
+ {ID: 2, Name: "Jazz Library", Path: "/music/jazz"},
+ {ID: 3, Name: "Classical Library", Path: "/music/classical"},
+ }
+
+ user = model.User{
+ ID: "user1",
+ UserName: "testuser",
+ Name: "Test User",
+ Email: "test@example.com",
+ IsAdmin: false,
+ Libraries: libraries,
+ }
+ })
+
+ Describe("HasLibraryAccess", func() {
+ Context("when user is admin", func() {
+ BeforeEach(func() {
+ user.IsAdmin = true
+ })
+
+ It("returns true for any library ID", func() {
+ Expect(user.HasLibraryAccess(1)).To(BeTrue())
+ Expect(user.HasLibraryAccess(99)).To(BeTrue())
+ Expect(user.HasLibraryAccess(-1)).To(BeTrue())
+ })
+
+ It("returns true even when user has no libraries assigned", func() {
+ user.Libraries = nil
+ Expect(user.HasLibraryAccess(1)).To(BeTrue())
+ })
+ })
+
+ Context("when user is not admin", func() {
+ BeforeEach(func() {
+ user.IsAdmin = false
+ })
+
+ It("returns true for libraries the user has access to", func() {
+ Expect(user.HasLibraryAccess(1)).To(BeTrue())
+ Expect(user.HasLibraryAccess(2)).To(BeTrue())
+ Expect(user.HasLibraryAccess(3)).To(BeTrue())
+ })
+
+ It("returns false for libraries the user does not have access to", func() {
+ Expect(user.HasLibraryAccess(4)).To(BeFalse())
+ Expect(user.HasLibraryAccess(99)).To(BeFalse())
+ Expect(user.HasLibraryAccess(-1)).To(BeFalse())
+ Expect(user.HasLibraryAccess(0)).To(BeFalse())
+ })
+
+ It("returns false when user has no libraries assigned", func() {
+ user.Libraries = nil
+ Expect(user.HasLibraryAccess(1)).To(BeFalse())
+ })
+
+ It("handles duplicate library IDs correctly", func() {
+ user.Libraries = model.Libraries{
+ {ID: 1, Name: "Library 1", Path: "/music1"},
+ {ID: 1, Name: "Library 1 Duplicate", Path: "/music1-dup"},
+ {ID: 2, Name: "Library 2", Path: "/music2"},
+ }
+ Expect(user.HasLibraryAccess(1)).To(BeTrue())
+ Expect(user.HasLibraryAccess(2)).To(BeTrue())
+ Expect(user.HasLibraryAccess(3)).To(BeFalse())
+ })
+ })
+ })
+})
diff --git a/persistence/album_repository.go b/persistence/album_repository.go
index 3f238ee23..b1ce23e2b 100644
--- a/persistence/album_repository.go
+++ b/persistence/album_repository.go
@@ -12,6 +12,7 @@ import (
. "github.com/Masterminds/squirrel"
"github.com/deluan/rest"
+ "github.com/google/uuid"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
@@ -112,7 +113,7 @@ func NewAlbumRepository(ctx context.Context, db dbx.Builder) model.AlbumReposito
var albumFilters = sync.OnceValue(func() map[string]filterFunc {
filters := map[string]filterFunc{
"id": idFilter("album"),
- "name": fullTextFilter("album"),
+ "name": fullTextFilter("album", "mbz_album_id", "mbz_release_group_id"),
"compilation": booleanFilter,
"artist_id": artistFilter,
"year": yearFilter,
@@ -122,6 +123,7 @@ var albumFilters = sync.OnceValue(func() map[string]filterFunc {
"missing": booleanFilter,
"genre_id": tagIDFilter,
"role_total_id": allRolesFilter,
+ "library_id": libraryIdFilter,
}
// Add all album tags as filters
for tag := range model.AlbumLevelTags() {
@@ -183,9 +185,10 @@ func allRolesFilter(_ string, value interface{}) Sqlizer {
}
func (r *albumRepository) CountAll(options ...model.QueryOptions) (int64, error) {
- sql := r.newSelect()
- sql = r.withAnnotation(sql, "album.id")
- return r.count(sql, options...)
+ query := r.newSelect()
+ query = r.withAnnotation(query, "album.id")
+ query = r.applyLibraryFilter(query)
+ return r.count(query, options...)
}
func (r *albumRepository) Exists(id string) (bool, error) {
@@ -215,8 +218,10 @@ func (r *albumRepository) UpdateExternalInfo(al *model.Album) error {
}
func (r *albumRepository) selectAlbum(options ...model.QueryOptions) SelectBuilder {
- sql := r.newSelect(options...).Columns("album.*")
- return r.withAnnotation(sql, "album.id")
+ sql := r.newSelect(options...).Columns("album.*", "library.path as library_path", "library.name as library_name").
+ LeftJoin("library on album.library_id = library.id")
+ sql = r.withAnnotation(sql, "album.id")
+ return r.applyLibraryFilter(sql)
}
func (r *albumRepository) Get(id string) (*model.Album, error) {
@@ -290,7 +295,6 @@ func (r *albumRepository) TouchByMissingFolder() (int64, error) {
// It does not need to load participants, as they are not used by the scanner.
func (r *albumRepository) GetTouchedAlbums(libID int) (model.AlbumCursor, error) {
query := r.selectAlbum().
- Join("library on library.id = album.library_id").
Where(And{
Eq{"library.id": libID},
ConcatExpr("album.imported_at > library.last_scan_at"),
@@ -333,8 +337,12 @@ on conflict (user_id, item_id, item_type) do update
return r.executeSQL(query)
}
-func (r *albumRepository) purgeEmpty() error {
+func (r *albumRepository) purgeEmpty(libraryIDs ...int) error {
del := Delete(r.tableName).Where("id not in (select distinct(album_id) from media_file)")
+ // If libraryIDs are specified, only purge albums from those libraries
+ if len(libraryIDs) > 0 {
+ del = del.Where(Eq{"library_id": libraryIDs})
+ }
c, err := r.executeSQL(del)
if err != nil {
return fmt.Errorf("purging empty albums: %w", err)
@@ -345,13 +353,20 @@ func (r *albumRepository) purgeEmpty() error {
return nil
}
-func (r *albumRepository) Search(q string, offset int, size int, includeMissing bool) (model.Albums, error) {
+func (r *albumRepository) Search(q string, offset int, size int, options ...model.QueryOptions) (model.Albums, error) {
var res dbAlbums
- err := r.doSearch(r.selectAlbum(), q, offset, size, includeMissing, &res, "name")
- if err != nil {
- return nil, err
+ if uuid.Validate(q) == nil {
+ err := r.searchByMBID(r.selectAlbum(options...), q, []string{"mbz_album_id", "mbz_release_group_id"}, &res)
+ if err != nil {
+ return nil, fmt.Errorf("searching album by MBID %q: %w", q, err)
+ }
+ } else {
+ err := r.doSearch(r.selectAlbum(options...), q, offset, size, &res, "album.rowid", "name")
+ if err != nil {
+ return nil, fmt.Errorf("searching album by query %q: %w", q, err)
+ }
}
- return res.toModels(), err
+ return res.toModels(), nil
}
func (r *albumRepository) Count(options ...rest.QueryOptions) (int64, error) {
diff --git a/persistence/album_repository_test.go b/persistence/album_repository_test.go
index 529458c26..a062b4398 100644
--- a/persistence/album_repository_test.go
+++ b/persistence/album_repository_test.go
@@ -1,13 +1,12 @@
package persistence
import (
- "context"
"fmt"
"time"
+ "github.com/Masterminds/squirrel"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/consts"
- "github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/model/id"
"github.com/navidrome/navidrome/model/request"
@@ -16,16 +15,16 @@ import (
)
var _ = Describe("AlbumRepository", func() {
- var repo model.AlbumRepository
+ var albumRepo *albumRepository
BeforeEach(func() {
- ctx := request.WithUser(log.NewContext(context.TODO()), model.User{ID: "userid", UserName: "johndoe"})
- repo = NewAlbumRepository(ctx, GetDBXBuilder())
+ ctx := request.WithUser(GinkgoT().Context(), model.User{ID: "userid", UserName: "johndoe"})
+ albumRepo = NewAlbumRepository(ctx, GetDBXBuilder()).(*albumRepository)
})
Describe("Get", func() {
var Get = func(id string) (*model.Album, error) {
- album, err := repo.Get(id)
+ album, err := albumRepo.Get(id)
if album != nil {
album.ImportedAt = time.Time{}
}
@@ -42,7 +41,7 @@ var _ = Describe("AlbumRepository", func() {
Describe("GetAll", func() {
var GetAll = func(opts ...model.QueryOptions) (model.Albums, error) {
- albums, err := repo.GetAll(opts...)
+ albums, err := albumRepo.GetAll(opts...)
for i := range albums {
albums[i].ImportedAt = time.Time{}
}
@@ -56,6 +55,7 @@ var _ = Describe("AlbumRepository", func() {
It("returns all records sorted", func() {
Expect(GetAll(model.QueryOptions{Sort: "name"})).To(Equal(model.Albums{
albumAbbeyRoad,
+ albumMultiDisc,
albumRadioactivity,
albumSgtPeppers,
}))
@@ -65,6 +65,7 @@ var _ = Describe("AlbumRepository", func() {
Expect(GetAll(model.QueryOptions{Sort: "name", Order: "desc"})).To(Equal(model.Albums{
albumSgtPeppers,
albumRadioactivity,
+ albumMultiDisc,
albumAbbeyRoad,
}))
})
@@ -83,12 +84,12 @@ var _ = Describe("AlbumRepository", func() {
conf.Server.AlbumPlayCountMode = consts.AlbumPlayCountModeAbsolute
newID := id.NewRandom()
- Expect(repo.Put(&model.Album{LibraryID: 1, ID: newID, Name: "name", SongCount: songCount})).To(Succeed())
+ Expect(albumRepo.Put(&model.Album{LibraryID: 1, ID: newID, Name: "name", SongCount: songCount})).To(Succeed())
for i := 0; i < playCount; i++ {
- Expect(repo.IncPlayCount(newID, time.Now())).To(Succeed())
+ Expect(albumRepo.IncPlayCount(newID, time.Now())).To(Succeed())
}
- album, err := repo.Get(newID)
+ album, err := albumRepo.Get(newID)
Expect(err).ToNot(HaveOccurred())
Expect(album.PlayCount).To(Equal(int64(expected)))
},
@@ -106,12 +107,12 @@ var _ = Describe("AlbumRepository", func() {
conf.Server.AlbumPlayCountMode = consts.AlbumPlayCountModeNormalized
newID := id.NewRandom()
- Expect(repo.Put(&model.Album{LibraryID: 1, ID: newID, Name: "name", SongCount: songCount})).To(Succeed())
+ Expect(albumRepo.Put(&model.Album{LibraryID: 1, ID: newID, Name: "name", SongCount: songCount})).To(Succeed())
for i := 0; i < playCount; i++ {
- Expect(repo.IncPlayCount(newID, time.Now())).To(Succeed())
+ Expect(albumRepo.IncPlayCount(newID, time.Now())).To(Succeed())
}
- album, err := repo.Get(newID)
+ album, err := albumRepo.Get(newID)
Expect(err).ToNot(HaveOccurred())
Expect(album.PlayCount).To(Equal(int64(expected)))
},
@@ -283,6 +284,235 @@ var _ = Describe("AlbumRepository", func() {
Expect(err).To(HaveOccurred())
})
})
+
+ Describe("Participant Foreign Key Handling", func() {
+ // albumArtistRecord represents a record in the album_artists table
+ type albumArtistRecord struct {
+ ArtistID string `db:"artist_id"`
+ Role string `db:"role"`
+ SubRole string `db:"sub_role"`
+ }
+
+ var artistRepo *artistRepository
+
+ BeforeEach(func() {
+ ctx := request.WithUser(GinkgoT().Context(), adminUser)
+ artistRepo = NewArtistRepository(ctx, GetDBXBuilder()).(*artistRepository)
+ })
+
+ // Helper to verify album_artists records
+ verifyAlbumArtists := func(albumID string, expected []albumArtistRecord) {
+ GinkgoHelper()
+ var actual []albumArtistRecord
+ sq := squirrel.Select("artist_id", "role", "sub_role").
+ From("album_artists").
+ Where(squirrel.Eq{"album_id": albumID}).
+ OrderBy("role", "artist_id", "sub_role")
+
+ err := albumRepo.queryAll(sq, &actual)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(actual).To(Equal(expected))
+ }
+
+ It("verifies that participant records are actually inserted into database", func() {
+ // Create a real artist in the database first
+ artist := &model.Artist{
+ ID: "real-artist-1",
+ Name: "Real Artist",
+ OrderArtistName: "real artist",
+ SortArtistName: "Artist, Real",
+ }
+ err := createArtistWithLibrary(artistRepo, artist, 1)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Create an album with participants that reference the real artist
+ album := &model.Album{
+ LibraryID: 1,
+ ID: "test-album-db-insert",
+ Name: "Test Album DB Insert",
+ AlbumArtistID: "real-artist-1",
+ AlbumArtist: "Real Artist",
+ Participants: model.Participants{
+ model.RoleArtist: {
+ {Artist: model.Artist{ID: "real-artist-1", Name: "Real Artist"}},
+ },
+ model.RoleComposer: {
+ {Artist: model.Artist{ID: "real-artist-1", Name: "Real Artist"}, SubRole: "primary"},
+ },
+ },
+ }
+
+ // Insert the album
+ err = albumRepo.Put(album)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify that participant records were actually inserted into album_artists table
+ expected := []albumArtistRecord{
+ {ArtistID: "real-artist-1", Role: "artist", SubRole: ""},
+ {ArtistID: "real-artist-1", Role: "composer", SubRole: "primary"},
+ }
+ verifyAlbumArtists(album.ID, expected)
+
+ // Clean up the test artist and album created for this test
+ _, _ = artistRepo.executeSQL(squirrel.Delete("artist").Where(squirrel.Eq{"id": artist.ID}))
+ _, _ = albumRepo.executeSQL(squirrel.Delete("album").Where(squirrel.Eq{"id": album.ID}))
+ })
+
+ It("filters out invalid artist IDs leaving only valid participants in database", func() {
+ // Create two real artists in the database
+ artist1 := &model.Artist{
+ ID: "real-artist-mix-1",
+ Name: "Real Artist 1",
+ OrderArtistName: "real artist 1",
+ }
+ artist2 := &model.Artist{
+ ID: "real-artist-mix-2",
+ Name: "Real Artist 2",
+ OrderArtistName: "real artist 2",
+ }
+ err := createArtistWithLibrary(artistRepo, artist1, 1)
+ Expect(err).ToNot(HaveOccurred())
+ err = createArtistWithLibrary(artistRepo, artist2, 1)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Create an album with mix of valid and invalid artist IDs
+ album := &model.Album{
+ LibraryID: 1,
+ ID: "test-album-mixed-validity",
+ Name: "Test Album Mixed Validity",
+ AlbumArtistID: "real-artist-mix-1",
+ AlbumArtist: "Real Artist 1",
+ Participants: model.Participants{
+ model.RoleArtist: {
+ {Artist: model.Artist{ID: "real-artist-mix-1", Name: "Real Artist 1"}},
+ {Artist: model.Artist{ID: "non-existent-mix-1", Name: "Non Existent 1"}},
+ {Artist: model.Artist{ID: "real-artist-mix-2", Name: "Real Artist 2"}},
+ },
+ model.RoleComposer: {
+ {Artist: model.Artist{ID: "non-existent-mix-2", Name: "Non Existent 2"}},
+ {Artist: model.Artist{ID: "real-artist-mix-1", Name: "Real Artist 1"}},
+ },
+ },
+ }
+
+ // This should not fail - only valid artists should be inserted
+ err = albumRepo.Put(album)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify that only valid artist IDs were inserted into album_artists table
+ // Non-existent artists should be filtered out by the INNER JOIN
+ expected := []albumArtistRecord{
+ {ArtistID: "real-artist-mix-1", Role: "artist", SubRole: ""},
+ {ArtistID: "real-artist-mix-2", Role: "artist", SubRole: ""},
+ {ArtistID: "real-artist-mix-1", Role: "composer", SubRole: ""},
+ }
+ verifyAlbumArtists(album.ID, expected)
+
+ // Clean up the test artists and album created for this test
+ artistIDs := []string{artist1.ID, artist2.ID}
+ _, _ = artistRepo.executeSQL(squirrel.Delete("artist").Where(squirrel.Eq{"id": artistIDs}))
+ _, _ = albumRepo.executeSQL(squirrel.Delete("album").Where(squirrel.Eq{"id": album.ID}))
+ })
+
+ It("handles complex nested JSON with multiple roles and sub-roles", func() {
+ // Create 4 artists for this test
+ artists := []*model.Artist{
+ {ID: "complex-artist-1", Name: "Lead Vocalist", OrderArtistName: "lead vocalist"},
+ {ID: "complex-artist-2", Name: "Guitarist", OrderArtistName: "guitarist"},
+ {ID: "complex-artist-3", Name: "Producer", OrderArtistName: "producer"},
+ {ID: "complex-artist-4", Name: "Engineer", OrderArtistName: "engineer"},
+ }
+
+ for _, artist := range artists {
+ err := createArtistWithLibrary(artistRepo, artist, 1)
+ Expect(err).ToNot(HaveOccurred())
+ }
+
+ // Create album with complex participant structure
+ album := &model.Album{
+ LibraryID: 1,
+ ID: "test-album-complex-json",
+ Name: "Test Album Complex JSON",
+ AlbumArtistID: "complex-artist-1",
+ AlbumArtist: "Lead Vocalist",
+ Participants: model.Participants{
+ model.RoleArtist: {
+ {Artist: model.Artist{ID: "complex-artist-1", Name: "Lead Vocalist"}},
+ {Artist: model.Artist{ID: "complex-artist-2", Name: "Guitarist"}, SubRole: "lead guitar"},
+ {Artist: model.Artist{ID: "complex-artist-2", Name: "Guitarist"}, SubRole: "rhythm guitar"},
+ },
+ model.RoleProducer: {
+ {Artist: model.Artist{ID: "complex-artist-3", Name: "Producer"}, SubRole: "executive"},
+ },
+ model.RoleEngineer: {
+ {Artist: model.Artist{ID: "complex-artist-4", Name: "Engineer"}, SubRole: "mixing"},
+ {Artist: model.Artist{ID: "complex-artist-4", Name: "Engineer"}, SubRole: "mastering"},
+ },
+ },
+ }
+
+ err := albumRepo.Put(album)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify complex JSON structure was correctly parsed and inserted
+ expected := []albumArtistRecord{
+ {ArtistID: "complex-artist-1", Role: "artist", SubRole: ""},
+ {ArtistID: "complex-artist-2", Role: "artist", SubRole: "lead guitar"},
+ {ArtistID: "complex-artist-2", Role: "artist", SubRole: "rhythm guitar"},
+ {ArtistID: "complex-artist-4", Role: "engineer", SubRole: "mastering"},
+ {ArtistID: "complex-artist-4", Role: "engineer", SubRole: "mixing"},
+ {ArtistID: "complex-artist-3", Role: "producer", SubRole: "executive"},
+ }
+ verifyAlbumArtists(album.ID, expected)
+
+ // Clean up the test artists and album created for this test
+ artistIDs := make([]string, len(artists))
+ for i, artist := range artists {
+ artistIDs[i] = artist.ID
+ }
+ _, _ = artistRepo.executeSQL(squirrel.Delete("artist").Where(squirrel.Eq{"id": artistIDs}))
+ _, _ = albumRepo.executeSQL(squirrel.Delete("album").Where(squirrel.Eq{"id": album.ID}))
+ })
+
+ It("handles albums with non-existent artist IDs without constraint errors", func() {
+ // Regression test for foreign key constraint error when album participants
+ // contain artist IDs that don't exist in the artist table
+
+ // Create an album with participants that reference non-existent artist IDs
+ album := &model.Album{
+ LibraryID: 1,
+ ID: "test-album-fk-constraints",
+ Name: "Test Album with Invalid Artist References",
+ AlbumArtistID: "non-existent-artist-1",
+ AlbumArtist: "Non Existent Album Artist",
+ Participants: model.Participants{
+ model.RoleArtist: {
+ {Artist: model.Artist{ID: "non-existent-artist-1", Name: "Non Existent Artist 1"}},
+ {Artist: model.Artist{ID: "non-existent-artist-2", Name: "Non Existent Artist 2"}},
+ },
+ model.RoleComposer: {
+ {Artist: model.Artist{ID: "non-existent-composer-1", Name: "Non Existent Composer 1"}},
+ {Artist: model.Artist{ID: "non-existent-composer-2", Name: "Non Existent Composer 2"}},
+ },
+ model.RoleAlbumArtist: {
+ {Artist: model.Artist{ID: "non-existent-album-artist-1", Name: "Non Existent Album Artist 1"}},
+ },
+ },
+ }
+
+ // This should not fail with foreign key constraint error
+ // The updateParticipants method should handle non-existent artist IDs gracefully
+ err := albumRepo.Put(album)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify that no participant records were inserted since all artist IDs were invalid
+ // The INNER JOIN with the artist table should filter out all non-existent artists
+ verifyAlbumArtists(album.ID, []albumArtistRecord{})
+
+ // Clean up the test album created for this test
+ _, _ = albumRepo.executeSQL(squirrel.Delete("album").Where(squirrel.Eq{"id": album.ID}))
+ })
+ })
})
func _p(id, name string, sortName ...string) model.Participant {
diff --git a/persistence/artist_repository.go b/persistence/artist_repository.go
index eb87ed006..6d08c27db 100644
--- a/persistence/artist_repository.go
+++ b/persistence/artist_repository.go
@@ -11,6 +11,7 @@ import (
. "github.com/Masterminds/squirrel"
"github.com/deluan/rest"
+ "github.com/google/uuid"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
@@ -26,9 +27,9 @@ type artistRepository struct {
}
type dbArtist struct {
- *model.Artist `structs:",flatten"`
- SimilarArtists string `structs:"-" json:"-"`
- Stats string `structs:"-" json:"-"`
+ *model.Artist `structs:",flatten"`
+ SimilarArtists string `structs:"-" json:"-"`
+ LibraryStatsJSON string `structs:"-" json:"-"`
}
type dbSimilarArtist struct {
@@ -37,27 +38,45 @@ type dbSimilarArtist struct {
}
func (a *dbArtist) PostScan() error {
- var stats map[string]map[string]int64
- if err := json.Unmarshal([]byte(a.Stats), &stats); err != nil {
- return fmt.Errorf("parsing artist stats from db: %w", err)
- }
a.Artist.Stats = make(map[model.Role]model.ArtistStats)
- for key, c := range stats {
- if key == "total" {
- a.Artist.Size = c["s"]
- a.Artist.SongCount = int(c["m"])
- a.Artist.AlbumCount = int(c["a"])
+
+ if a.LibraryStatsJSON != "" {
+ var rawLibStats map[string]map[string]map[string]int64
+ if err := json.Unmarshal([]byte(a.LibraryStatsJSON), &rawLibStats); err != nil {
+ return fmt.Errorf("parsing artist stats from db: %w", err)
}
- role := model.RoleFromString(key)
- if role == model.RoleInvalid {
- continue
- }
- a.Artist.Stats[role] = model.ArtistStats{
- SongCount: int(c["m"]),
- AlbumCount: int(c["a"]),
- Size: c["s"],
+
+ for _, stats := range rawLibStats {
+ // Sum all libraries roles stats
+ for key, stat := range stats {
+ // Aggregate stats into the main Artist.Stats map
+ artistStats := model.ArtistStats{
+ SongCount: int(stat["m"]),
+ AlbumCount: int(stat["a"]),
+ Size: stat["s"],
+ }
+
+ // Store total stats into the main attributes
+ if key == "total" {
+ a.Artist.Size += artistStats.Size
+ a.Artist.SongCount += artistStats.SongCount
+ a.Artist.AlbumCount += artistStats.AlbumCount
+ }
+
+ role := model.RoleFromString(key)
+ if role == model.RoleInvalid {
+ continue
+ }
+
+ current := a.Artist.Stats[role]
+ current.Size += artistStats.Size
+ current.SongCount += artistStats.SongCount
+ current.AlbumCount += artistStats.AlbumCount
+ a.Artist.Stats[role] = current
+ }
}
}
+
a.Artist.SimilarArtists = nil
if a.SimilarArtists == "" {
return nil
@@ -112,11 +131,12 @@ func NewArtistRepository(ctx context.Context, db dbx.Builder) model.ArtistReposi
r.indexGroups = utils.ParseIndexGroups(conf.Server.IndexGroups)
r.tableName = "artist" // To be used by the idFilter below
r.registerModel(&model.Artist{}, map[string]filterFunc{
- "id": idFilter(r.tableName),
- "name": fullTextFilter(r.tableName),
- "starred": booleanFilter,
- "role": roleFilter,
- "missing": booleanFilter,
+ "id": idFilter(r.tableName),
+ "name": fullTextFilter(r.tableName, "mbz_artist_id"),
+ "starred": booleanFilter,
+ "role": roleFilter,
+ "missing": booleanFilter,
+ "library_id": artistLibraryIdFilter,
})
r.setSortMappings(map[string]string{
"name": "order_artist_name",
@@ -124,28 +144,72 @@ func NewArtistRepository(ctx context.Context, db dbx.Builder) model.ArtistReposi
"song_count": "stats->>'total'->>'m'",
"album_count": "stats->>'total'->>'a'",
"size": "stats->>'total'->>'s'",
+
+ // Stats by credits that are currently available
+ "maincredit_song_count": "sum(stats->>'maincredit'->>'m')",
+ "maincredit_album_count": "sum(stats->>'maincredit'->>'a')",
+ "maincredit_size": "sum(stats->>'maincredit'->>'s')",
})
return r
}
func roleFilter(_ string, role any) Sqlizer {
- return NotEq{fmt.Sprintf("stats ->> '$.%v'", role): nil}
+ if role, ok := role.(string); ok {
+ if _, ok := model.AllRoles[role]; ok {
+ return Expr("JSON_EXTRACT(library_artist.stats, '$." + role + ".m') IS NOT NULL")
+ }
+ }
+ return Eq{"1": 2}
+}
+
+// artistLibraryIdFilter filters artists based on library access through the library_artist table
+func artistLibraryIdFilter(_ string, value interface{}) Sqlizer {
+ return Eq{"library_artist.library_id": value}
+}
+
+// applyLibraryFilterToArtistQuery applies library filtering to artist queries through the library_artist junction table
+func (r *artistRepository) applyLibraryFilterToArtistQuery(query SelectBuilder) SelectBuilder {
+ user := loggedUser(r.ctx)
+ // Join with library_artist first to ensure only artists with content in libraries are included
+ // Exclude artists with empty stats (no actual content in the library)
+ query = query.Join("library_artist on library_artist.artist_id = artist.id")
+ //query = query.Join("library_artist on library_artist.artist_id = artist.id AND library_artist.stats != '{}'")
+
+ // Admin users see all artists from all libraries, no additional filtering needed
+ if user.ID != invalidUserId && !user.IsAdmin {
+ // Apply library filtering only for non-admin users by joining with their accessible libraries
+ query = query.Join("user_library on user_library.library_id = library_artist.library_id AND user_library.user_id = ?", user.ID)
+ }
+
+ return query
}
func (r *artistRepository) selectArtist(options ...model.QueryOptions) SelectBuilder {
- query := r.newSelect(options...).Columns("artist.*")
- query = r.withAnnotation(query, "artist.id")
- return query
+ // Stats Format: {"1": {"albumartist": {"m": 10, "a": 5, "s": 1024}, "artist": {...}}, "2": {...}}
+ query := r.newSelect(options...).Columns("artist.*",
+ "JSON_GROUP_OBJECT(library_artist.library_id, JSONB(library_artist.stats)) as library_stats_json")
+
+ query = r.applyLibraryFilterToArtistQuery(query)
+ query = query.GroupBy("artist.id")
+ return r.withAnnotation(query, "artist.id")
}
func (r *artistRepository) CountAll(options ...model.QueryOptions) (int64, error) {
query := r.newSelect()
+ query = r.applyLibraryFilterToArtistQuery(query)
query = r.withAnnotation(query, "artist.id")
return r.count(query, options...)
}
+// Exists checks if an artist with the given ID exists in the database and is accessible by the current user.
func (r *artistRepository) Exists(id string) (bool, error) {
- return r.exists(Eq{"artist.id": id})
+ // Create a query using the same library filtering logic as selectArtist()
+ query := r.newSelect().Columns("count(distinct artist.id) as exist").Where(Eq{"artist.id": id})
+ query = r.applyLibraryFilterToArtistQuery(query)
+
+ var res struct{ Exist int64 }
+ err := r.queryOne(query, &res)
+ return res.Exist > 0, err
}
func (r *artistRepository) Put(a *model.Artist, colsToUpdate ...string) error {
@@ -202,14 +266,21 @@ func (r *artistRepository) getIndexKey(a model.Artist) string {
return "#"
}
-// TODO Cache the index (recalculate when there are changes to the DB)
-func (r *artistRepository) GetIndex(includeMissing bool, roles ...model.Role) (model.ArtistIndexes, error) {
+// GetIndex returns a list of artists grouped by the first letter of their name, or by the index group if configured.
+// It can filter by roles and libraries, and optionally include artists that are missing (i.e., have no albums).
+// TODO Cache the index (recalculate at scan time)
+func (r *artistRepository) GetIndex(includeMissing bool, libraryIds []int, roles ...model.Role) (model.ArtistIndexes, error) {
+ // Validate library IDs. If no library IDs are provided, return an empty index.
+ if len(libraryIds) == 0 {
+ return nil, nil
+ }
+
options := model.QueryOptions{Sort: "name"}
if len(roles) > 0 {
roleFilters := slice.Map(roles, func(r model.Role) Sqlizer {
- return roleFilter("role", r)
+ return roleFilter("role", r.String())
})
- options.Filters = And(roleFilters)
+ options.Filters = Or(roleFilters)
}
if !includeMissing {
if options.Filters == nil {
@@ -218,10 +289,19 @@ func (r *artistRepository) GetIndex(includeMissing bool, roles ...model.Role) (m
options.Filters = And{options.Filters, Eq{"artist.missing": false}}
}
}
+
+ libFilter := artistLibraryIdFilter("library_id", libraryIds)
+ if options.Filters == nil {
+ options.Filters = libFilter
+ } else {
+ options.Filters = And{options.Filters, libFilter}
+ }
+
artists, err := r.GetAll(options)
if err != nil {
return nil, err
}
+
var result model.ArtistIndexes
for k, v := range slice.Group(artists, r.getIndexKey) {
result = append(result, model.ArtistIndex{ID: k, Artists: v})
@@ -287,75 +367,97 @@ on conflict (user_id, item_id, item_type) do update
}
// RefreshStats updates the stats field for artists whose associated media files were updated after the oldest recorded library scan time.
-// It processes artists in batches to handle potentially large updates.
-func (r *artistRepository) RefreshStats() (int64, error) {
- touchedArtistsQuerySQL := `
- SELECT DISTINCT mfa.artist_id
- FROM media_file_artists mfa
- JOIN media_file mf ON mfa.media_file_id = mf.id
- WHERE mf.updated_at > (SELECT last_scan_at FROM library ORDER BY last_scan_at ASC LIMIT 1)
- `
-
+// When allArtists is true, it refreshes stats for all artists. It processes artists in batches to handle potentially large updates.
+// This method now calculates per-library statistics and stores them in the library_artist junction table.
+func (r *artistRepository) RefreshStats(allArtists bool) (int64, error) {
var allTouchedArtistIDs []string
- if err := r.db.NewQuery(touchedArtistsQuerySQL).Column(&allTouchedArtistIDs); err != nil {
- return 0, fmt.Errorf("fetching touched artist IDs: %w", err)
+ if allArtists {
+ // Refresh stats for all artists
+ allArtistsQuerySQL := `SELECT DISTINCT id FROM artist WHERE id <> ''`
+ if err := r.db.NewQuery(allArtistsQuerySQL).Column(&allTouchedArtistIDs); err != nil {
+ return 0, fmt.Errorf("fetching all artist IDs: %w", err)
+ }
+ log.Debug(r.ctx, "RefreshStats: Refreshing all artists.", "count", len(allTouchedArtistIDs))
+ } else {
+ // Only refresh artists with updated timestamps
+ touchedArtistsQuerySQL := `
+ SELECT DISTINCT id
+ FROM artist
+ WHERE updated_at > (SELECT last_scan_at FROM library ORDER BY last_scan_at ASC LIMIT 1)
+ `
+ if err := r.db.NewQuery(touchedArtistsQuerySQL).Column(&allTouchedArtistIDs); err != nil {
+ return 0, fmt.Errorf("fetching touched artist IDs: %w", err)
+ }
+ log.Debug(r.ctx, "RefreshStats: Refreshing touched artists.", "count", len(allTouchedArtistIDs))
}
if len(allTouchedArtistIDs) == 0 {
log.Debug(r.ctx, "RefreshStats: No artists to update.")
return 0, nil
}
- log.Debug(r.ctx, "RefreshStats: Found artists to update.", "count", len(allTouchedArtistIDs))
// Template for the batch update with placeholder markers that we'll replace
+ // This now calculates per-library statistics and stores them in library_artist.stats
batchUpdateStatsSQL := `
WITH artist_role_counters AS (
- SELECT jt.atom AS artist_id,
- substr(
- replace(jt.path, '$.', ''),
- 1,
- CASE WHEN instr(replace(jt.path, '$.', ''), '[') > 0
- THEN instr(replace(jt.path, '$.', ''), '[') - 1
- ELSE length(replace(jt.path, '$.', ''))
- END
- ) AS role,
+ SELECT mfa.artist_id,
+ mf.library_id,
+ mfa.role,
count(DISTINCT mf.album_id) AS album_count,
- count(mf.id) AS count,
+ count(DISTINCT mf.id) AS count,
sum(mf.size) AS size
- FROM media_file mf
- JOIN json_tree(mf.participants) jt ON jt.key = 'id' AND jt.atom IS NOT NULL
- WHERE jt.atom IN (ROLE_IDS_PLACEHOLDER) -- Will replace with actual placeholders
- GROUP BY jt.atom, role
+ FROM media_file_artists mfa
+ JOIN media_file mf ON mfa.media_file_id = mf.id
+ WHERE mfa.artist_id IN (ROLE_IDS_PLACEHOLDER) -- Will replace with actual placeholders
+ GROUP BY mfa.artist_id, mf.library_id, mfa.role
),
artist_total_counters AS (
SELECT mfa.artist_id,
+ mf.library_id,
'total' AS role,
count(DISTINCT mf.album_id) AS album_count,
count(DISTINCT mf.id) AS count,
sum(mf.size) AS size
FROM media_file_artists mfa
JOIN media_file mf ON mfa.media_file_id = mf.id
- WHERE mfa.artist_id IN (TOTAL_IDS_PLACEHOLDER) -- Will replace with actual placeholders
- GROUP BY mfa.artist_id
+ WHERE mfa.artist_id IN (ROLE_IDS_PLACEHOLDER) -- Will replace with actual placeholders
+ GROUP BY mfa.artist_id, mf.library_id
+ ),
+ artist_participant_counter AS (
+ SELECT mfa.artist_id,
+ mf.library_id,
+ 'maincredit' AS role,
+ count(DISTINCT mf.album_id) AS album_count,
+ count(DISTINCT mf.id) AS count,
+ sum(mf.size) AS size
+ FROM media_file_artists mfa
+ JOIN media_file mf ON mfa.media_file_id = mf.id
+ WHERE mfa.artist_id IN (ROLE_IDS_PLACEHOLDER) -- Will replace with actual placeholders
+ AND mfa.role IN ('albumartist', 'artist')
+ GROUP BY mfa.artist_id, mf.library_id
),
combined_counters AS (
- SELECT artist_id, role, album_count, count, size FROM artist_role_counters
- UNION
- SELECT artist_id, role, album_count, count, size FROM artist_total_counters
+ SELECT artist_id, library_id, role, album_count, count, size FROM artist_role_counters
+ UNION ALL
+ SELECT artist_id, library_id, role, album_count, count, size FROM artist_total_counters
+ UNION ALL
+ SELECT artist_id, library_id, role, album_count, count, size FROM artist_participant_counter
),
- artist_counters AS (
- SELECT artist_id AS id,
+ library_artist_counters AS (
+ SELECT artist_id,
+ library_id,
json_group_object(
- replace(role, '"', ''),
+ role,
json_object('a', album_count, 'm', count, 's', size)
) AS counters
FROM combined_counters
- GROUP BY artist_id
+ GROUP BY artist_id, library_id
)
- UPDATE artist
- SET stats = coalesce((SELECT counters FROM artist_counters ac WHERE ac.id = artist.id), '{}'),
- updated_at = datetime(current_timestamp, 'localtime')
- WHERE artist.id IN (UPDATE_IDS_PLACEHOLDER) AND artist.id <> '';` // Will replace with actual placeholders
+ UPDATE library_artist
+ SET stats = coalesce((SELECT counters FROM library_artist_counters lac
+ WHERE lac.artist_id = library_artist.artist_id
+ AND lac.library_id = library_artist.library_id), '{}')
+ WHERE library_artist.artist_id IN (ROLE_IDS_PLACEHOLDER);` // Will replace with actual placeholders
var totalRowsAffected int64 = 0
const batchSize = 1000
@@ -374,21 +476,16 @@ func (r *artistRepository) RefreshStats() (int64, error) {
inClause := strings.Join(placeholders, ",")
// Replace the placeholder markers with actual SQL placeholders
- batchSQL := strings.Replace(batchUpdateStatsSQL, "ROLE_IDS_PLACEHOLDER", inClause, 1)
- batchSQL = strings.Replace(batchSQL, "TOTAL_IDS_PLACEHOLDER", inClause, 1)
- batchSQL = strings.Replace(batchSQL, "UPDATE_IDS_PLACEHOLDER", inClause, 1)
+ batchSQL := strings.Replace(batchUpdateStatsSQL, "ROLE_IDS_PLACEHOLDER", inClause, 4)
- // Create a single parameter array with all IDs (repeated 3 times for each IN clause)
- // We need to repeat each ID 3 times (once for each IN clause)
- var args []interface{}
- for _, id := range artistIDBatch {
- args = append(args, id) // For ROLE_IDS_PLACEHOLDER
- }
- for _, id := range artistIDBatch {
- args = append(args, id) // For TOTAL_IDS_PLACEHOLDER
- }
- for _, id := range artistIDBatch {
- args = append(args, id) // For UPDATE_IDS_PLACEHOLDER
+ // Create a single parameter array with all IDs (repeated 4 times for each IN clause)
+ // We need to repeat each ID 4 times (once for each IN clause)
+ args := make([]any, 4*len(artistIDBatch))
+ for idx, id := range artistIDBatch {
+ for i := range 4 {
+ startIdx := i * len(artistIDBatch)
+ args[startIdx+idx] = id
+ }
}
// Now use Expr with the expanded SQL and all parameters
@@ -401,17 +498,35 @@ func (r *artistRepository) RefreshStats() (int64, error) {
totalRowsAffected += rowsAffected
}
+ // // Remove library_artist entries for artists that no longer have any content in any library
+ cleanupSQL := Delete("library_artist").Where("stats = '{}'")
+ cleanupRows, err := r.executeSQL(cleanupSQL)
+ if err != nil {
+ log.Warn(r.ctx, "Failed to cleanup empty library_artist entries", "error", err)
+ } else if cleanupRows > 0 {
+ log.Debug(r.ctx, "Cleaned up empty library_artist entries", "rowsDeleted", cleanupRows)
+ }
+
log.Debug(r.ctx, "RefreshStats: Successfully updated stats.", "totalArtistsProcessed", len(allTouchedArtistIDs), "totalDBRowsAffected", totalRowsAffected)
return totalRowsAffected, nil
}
-func (r *artistRepository) Search(q string, offset int, size int, includeMissing bool) (model.Artists, error) {
- var dba dbArtists
- err := r.doSearch(r.selectArtist(), q, offset, size, includeMissing, &dba, "json_extract(stats, '$.total.m') desc", "name")
- if err != nil {
- return nil, err
+func (r *artistRepository) Search(q string, offset int, size int, options ...model.QueryOptions) (model.Artists, error) {
+ var res dbArtists
+ if uuid.Validate(q) == nil {
+ err := r.searchByMBID(r.selectArtist(options...), q, []string{"mbz_artist_id"}, &res)
+ if err != nil {
+ return nil, fmt.Errorf("searching artist by MBID %q: %w", q, err)
+ }
+ } else {
+ // Natural order for artists is more performant by ID, due to GROUP BY clause in selectArtist
+ err := r.doSearch(r.selectArtist(options...), q, offset, size, &res, "artist.id",
+ "sum(json_extract(stats, '$.total.m')) desc", "name")
+ if err != nil {
+ return nil, fmt.Errorf("searching artist by query %q: %w", q, err)
+ }
}
- return dba.toModels(), nil
+ return res.toModels(), nil
}
func (r *artistRepository) Count(options ...rest.QueryOptions) (int64, error) {
@@ -429,9 +544,9 @@ func (r *artistRepository) ReadAll(options ...rest.QueryOptions) (interface{}, e
role = v
}
}
- r.sortMappings["song_count"] = "stats->>'" + role + "'->>'m'"
- r.sortMappings["album_count"] = "stats->>'" + role + "'->>'a'"
- r.sortMappings["size"] = "stats->>'" + role + "'->>'s'"
+ r.sortMappings["song_count"] = "sum(stats->>'" + role + "'->>'m')"
+ r.sortMappings["album_count"] = "sum(stats->>'" + role + "'->>'a')"
+ r.sortMappings["size"] = "sum(stats->>'" + role + "'->>'s')"
return r.GetAll(r.parseRestOptions(r.ctx, options...))
}
diff --git a/persistence/artist_repository_test.go b/persistence/artist_repository_test.go
index 0c7018dc8..dfaf499ac 100644
--- a/persistence/artist_repository_test.go
+++ b/persistence/artist_repository_test.go
@@ -7,7 +7,6 @@ import (
"github.com/Masterminds/squirrel"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/conf/configtest"
- "github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/model/request"
"github.com/navidrome/navidrome/utils"
@@ -15,310 +14,759 @@ import (
. "github.com/onsi/gomega"
)
+// Test helper functions to reduce duplication
+func createTestArtistWithMBID(id, name, mbid string) model.Artist {
+ return model.Artist{
+ ID: id,
+ Name: name,
+ MbzArtistID: mbid,
+ }
+}
+
+func createUserWithLibraries(userID string, libraryIDs []int) model.User {
+ user := model.User{
+ ID: userID,
+ UserName: userID,
+ Name: userID,
+ Email: userID + "@test.com",
+ IsAdmin: false,
+ }
+
+ if len(libraryIDs) > 0 {
+ user.Libraries = make(model.Libraries, len(libraryIDs))
+ for i, libID := range libraryIDs {
+ user.Libraries[i] = model.Library{ID: libID, Name: "Test Library", Path: "/test"}
+ }
+ }
+
+ return user
+}
+
var _ = Describe("ArtistRepository", func() {
- var repo model.ArtistRepository
- BeforeEach(func() {
- DeferCleanup(configtest.SetupConfig())
- ctx := log.NewContext(context.TODO())
- ctx = request.WithUser(ctx, model.User{ID: "userid"})
- repo = NewArtistRepository(ctx, GetDBXBuilder())
- })
+ Context("Core Functionality", func() {
+ Describe("GetIndexKey", func() {
+ // Note: OrderArtistName should never be empty, so we don't need to test for that
+ r := artistRepository{indexGroups: utils.ParseIndexGroups(conf.Server.IndexGroups)}
- Describe("Count", func() {
- It("returns the number of artists in the DB", func() {
- Expect(repo.CountAll()).To(Equal(int64(2)))
+ DescribeTable("returns correct index key based on PreferSortTags setting",
+ func(preferSortTags bool, sortArtistName, orderArtistName, expectedKey string) {
+ DeferCleanup(configtest.SetupConfig())
+ conf.Server.PreferSortTags = preferSortTags
+ a := model.Artist{SortArtistName: sortArtistName, OrderArtistName: orderArtistName, Name: "Test"}
+ idx := GetIndexKey(&r, a)
+ Expect(idx).To(Equal(expectedKey))
+ },
+ Entry("PreferSortTags=false, SortArtistName empty -> uses OrderArtistName", false, "", "Bar", "B"),
+ Entry("PreferSortTags=false, SortArtistName not empty -> still uses OrderArtistName", false, "Foo", "Bar", "B"),
+ Entry("PreferSortTags=true, SortArtistName not empty -> uses SortArtistName", true, "Foo", "Bar", "F"),
+ Entry("PreferSortTags=true, SortArtistName empty -> falls back to OrderArtistName", true, "", "Bar", "B"),
+ )
})
- })
- Describe("Exists", func() {
- It("returns true for an artist that is in the DB", func() {
- Expect(repo.Exists("3")).To(BeTrue())
- })
- It("returns false for an artist that is in the DB", func() {
- Expect(repo.Exists("666")).To(BeFalse())
- })
- })
+ Describe("roleFilter", func() {
+ DescribeTable("validates roles and returns appropriate SQL expressions",
+ func(role string, shouldBeValid bool) {
+ result := roleFilter("", role)
+ if shouldBeValid {
+ expectedExpr := squirrel.Expr("JSON_EXTRACT(library_artist.stats, '$." + role + ".m') IS NOT NULL")
+ Expect(result).To(Equal(expectedExpr))
+ } else {
+ expectedInvalid := squirrel.Eq{"1": 2}
+ Expect(result).To(Equal(expectedInvalid))
+ }
+ },
+ // Valid roles from model.AllRoles
+ Entry("artist role", "artist", true),
+ Entry("albumartist role", "albumartist", true),
+ Entry("composer role", "composer", true),
+ Entry("conductor role", "conductor", true),
+ Entry("lyricist role", "lyricist", true),
+ Entry("arranger role", "arranger", true),
+ Entry("producer role", "producer", true),
+ Entry("director role", "director", true),
+ Entry("engineer role", "engineer", true),
+ Entry("mixer role", "mixer", true),
+ Entry("remixer role", "remixer", true),
+ Entry("djmixer role", "djmixer", true),
+ Entry("performer role", "performer", true),
+ Entry("maincredit role", "maincredit", true),
+ // Invalid roles
+ Entry("invalid role - wizard", "wizard", false),
+ Entry("invalid role - songanddanceman", "songanddanceman", false),
+ Entry("empty string", "", false),
+ Entry("SQL injection attempt", "artist') SELECT LIKE(CHAR(65,66,67,68,69,70,71),UPPER(HEX(RANDOMBLOB(500000000/2))))--", false),
+ )
- Describe("Get", func() {
- It("saves and retrieves data", func() {
- artist, err := repo.Get("2")
- Expect(err).ToNot(HaveOccurred())
- Expect(artist.Name).To(Equal(artistKraftwerk.Name))
+ It("handles non-string input types", func() {
+ expectedInvalid := squirrel.Eq{"1": 2}
+ Expect(roleFilter("", 123)).To(Equal(expectedInvalid))
+ Expect(roleFilter("", nil)).To(Equal(expectedInvalid))
+ Expect(roleFilter("", []string{"artist"})).To(Equal(expectedInvalid))
+ })
})
- })
- Describe("GetIndexKey", func() {
- // Note: OrderArtistName should never be empty, so we don't need to test for that
- r := artistRepository{indexGroups: utils.ParseIndexGroups(conf.Server.IndexGroups)}
- When("PreferSortTags is false", func() {
+ Describe("dbArtist mapping", func() {
+ var (
+ artist *model.Artist
+ dba *dbArtist
+ )
+
BeforeEach(func() {
- conf.Server.PreferSortTags = false
+ artist = &model.Artist{ID: "1", Name: "Eddie Van Halen", SortArtistName: "Van Halen, Eddie"}
+ dba = &dbArtist{Artist: artist}
})
- It("returns the OrderArtistName key is SortArtistName is empty", func() {
- conf.Server.PreferSortTags = false
- a := model.Artist{SortArtistName: "", OrderArtistName: "Bar", Name: "Qux"}
- idx := GetIndexKey(&r, a)
- Expect(idx).To(Equal("B"))
+
+ Describe("PostScan", func() {
+ It("parses stats and similar artists correctly", func() {
+ stats := map[string]map[string]map[string]int64{
+ "1": {
+ "total": {"s": 1000, "m": 10, "a": 2},
+ "composer": {"s": 500, "m": 5, "a": 1},
+ },
+ }
+ statsJSON, _ := json.Marshal(stats)
+ dba.LibraryStatsJSON = string(statsJSON)
+ dba.SimilarArtists = `[{"id":"2","Name":"AC/DC"},{"name":"Test;With:Sep,Chars"}]`
+
+ err := dba.PostScan()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(dba.Artist.Size).To(Equal(int64(1000)))
+ Expect(dba.Artist.SongCount).To(Equal(10))
+ Expect(dba.Artist.AlbumCount).To(Equal(2))
+ Expect(dba.Artist.Stats).To(HaveLen(1))
+ Expect(dba.Artist.Stats[model.RoleFromString("composer")].Size).To(Equal(int64(500)))
+ Expect(dba.Artist.Stats[model.RoleFromString("composer")].SongCount).To(Equal(5))
+ Expect(dba.Artist.Stats[model.RoleFromString("composer")].AlbumCount).To(Equal(1))
+ Expect(dba.Artist.SimilarArtists).To(HaveLen(2))
+ Expect(dba.Artist.SimilarArtists[0].ID).To(Equal("2"))
+ Expect(dba.Artist.SimilarArtists[0].Name).To(Equal("AC/DC"))
+ Expect(dba.Artist.SimilarArtists[1].ID).To(BeEmpty())
+ Expect(dba.Artist.SimilarArtists[1].Name).To(Equal("Test;With:Sep,Chars"))
+ })
})
- It("returns the OrderArtistName key even if SortArtistName is not empty", func() {
- a := model.Artist{SortArtistName: "Foo", OrderArtistName: "Bar", Name: "Qux"}
- idx := GetIndexKey(&r, a)
- Expect(idx).To(Equal("B"))
- })
- })
- When("PreferSortTags is true", func() {
- BeforeEach(func() {
- conf.Server.PreferSortTags = true
- })
- It("returns the SortArtistName key if it is not empty", func() {
- a := model.Artist{SortArtistName: "Foo", OrderArtistName: "Bar", Name: "Qux"}
- idx := GetIndexKey(&r, a)
- Expect(idx).To(Equal("F"))
- })
- It("returns the OrderArtistName key if SortArtistName is empty", func() {
- a := model.Artist{SortArtistName: "", OrderArtistName: "Bar", Name: "Qux"}
- idx := GetIndexKey(&r, a)
- Expect(idx).To(Equal("B"))
+
+ Describe("PostMapArgs", func() {
+ It("maps empty similar artists correctly", func() {
+ m := make(map[string]any)
+ err := dba.PostMapArgs(m)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(m).To(HaveKeyWithValue("similar_artists", "[]"))
+ })
+
+ It("maps similar artists and full text correctly", func() {
+ artist.SimilarArtists = []model.Artist{
+ {ID: "2", Name: "AC/DC"},
+ {Name: "Test;With:Sep,Chars"},
+ }
+ m := make(map[string]any)
+ err := dba.PostMapArgs(m)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(m).To(HaveKeyWithValue("similar_artists", `[{"id":"2","name":"AC/DC"},{"name":"Test;With:Sep,Chars"}]`))
+ Expect(m).To(HaveKeyWithValue("full_text", " eddie halen van"))
+ })
+
+ It("does not override empty sort_artist_name and mbz_artist_id", func() {
+ m := map[string]any{
+ "sort_artist_name": "",
+ "mbz_artist_id": "",
+ }
+ err := dba.PostMapArgs(m)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(m).ToNot(HaveKey("sort_artist_name"))
+ Expect(m).ToNot(HaveKey("mbz_artist_id"))
+ })
})
})
})
- Describe("GetIndex", func() {
- When("PreferSortTags is true", func() {
- BeforeEach(func() {
- conf.Server.PreferSortTags = true
- })
- It("returns the index when PreferSortTags is true and SortArtistName is not empty", func() {
- // Set SortArtistName to "Foo" for Beatles
- artistBeatles.SortArtistName = "Foo"
- er := repo.Put(&artistBeatles)
- Expect(er).To(BeNil())
-
- idx, err := repo.GetIndex(false)
- Expect(err).ToNot(HaveOccurred())
- Expect(idx).To(HaveLen(2))
- Expect(idx[0].ID).To(Equal("F"))
- Expect(idx[0].Artists).To(HaveLen(1))
- Expect(idx[0].Artists[0].Name).To(Equal(artistBeatles.Name))
- Expect(idx[1].ID).To(Equal("K"))
- Expect(idx[1].Artists).To(HaveLen(1))
- Expect(idx[1].Artists[0].Name).To(Equal(artistKraftwerk.Name))
-
- // Restore the original value
- artistBeatles.SortArtistName = ""
- er = repo.Put(&artistBeatles)
- Expect(er).To(BeNil())
- })
-
- // BFR Empty SortArtistName is not saved in the DB anymore
- XIt("returns the index when PreferSortTags is true and SortArtistName is empty", func() {
- idx, err := repo.GetIndex(false)
- Expect(err).ToNot(HaveOccurred())
- Expect(idx).To(HaveLen(2))
- Expect(idx[0].ID).To(Equal("B"))
- Expect(idx[0].Artists).To(HaveLen(1))
- Expect(idx[0].Artists[0].Name).To(Equal(artistBeatles.Name))
- Expect(idx[1].ID).To(Equal("K"))
- Expect(idx[1].Artists).To(HaveLen(1))
- Expect(idx[1].Artists[0].Name).To(Equal(artistKraftwerk.Name))
- })
- })
-
- When("PreferSortTags is false", func() {
- BeforeEach(func() {
- conf.Server.PreferSortTags = false
- })
- It("returns the index when SortArtistName is NOT empty", func() {
- // Set SortArtistName to "Foo" for Beatles
- artistBeatles.SortArtistName = "Foo"
- er := repo.Put(&artistBeatles)
- Expect(er).To(BeNil())
-
- idx, err := repo.GetIndex(false)
- Expect(err).ToNot(HaveOccurred())
- Expect(idx).To(HaveLen(2))
- Expect(idx[0].ID).To(Equal("B"))
- Expect(idx[0].Artists).To(HaveLen(1))
- Expect(idx[0].Artists[0].Name).To(Equal(artistBeatles.Name))
- Expect(idx[1].ID).To(Equal("K"))
- Expect(idx[1].Artists).To(HaveLen(1))
- Expect(idx[1].Artists[0].Name).To(Equal(artistKraftwerk.Name))
-
- // Restore the original value
- artistBeatles.SortArtistName = ""
- er = repo.Put(&artistBeatles)
- Expect(er).To(BeNil())
- })
-
- It("returns the index when SortArtistName is empty", func() {
- idx, err := repo.GetIndex(false)
- Expect(err).ToNot(HaveOccurred())
- Expect(idx).To(HaveLen(2))
- Expect(idx[0].ID).To(Equal("B"))
- Expect(idx[0].Artists).To(HaveLen(1))
- Expect(idx[0].Artists[0].Name).To(Equal(artistBeatles.Name))
- Expect(idx[1].ID).To(Equal("K"))
- Expect(idx[1].Artists).To(HaveLen(1))
- Expect(idx[1].Artists[0].Name).To(Equal(artistKraftwerk.Name))
- })
- })
- })
-
- Describe("dbArtist mapping", func() {
- var (
- artist *model.Artist
- dba *dbArtist
- )
+ Context("Admin User Operations", func() {
+ var repo model.ArtistRepository
BeforeEach(func() {
- artist = &model.Artist{ID: "1", Name: "Eddie Van Halen", SortArtistName: "Van Halen, Eddie"}
- dba = &dbArtist{Artist: artist}
+ ctx := GinkgoT().Context()
+ ctx = request.WithUser(ctx, adminUser)
+ repo = NewArtistRepository(ctx, GetDBXBuilder())
})
- Describe("PostScan", func() {
- It("parses stats and similar artists correctly", func() {
- stats := map[string]map[string]int64{
- "total": {"s": 1000, "m": 10, "a": 2},
- "composer": {"s": 500, "m": 5, "a": 1},
- }
- statsJSON, _ := json.Marshal(stats)
- dba.Stats = string(statsJSON)
- dba.SimilarArtists = `[{"id":"2","Name":"AC/DC"},{"name":"Test;With:Sep,Chars"}]`
+ Describe("Basic Operations", func() {
+ Describe("Count", func() {
+ It("returns the number of artists in the DB", func() {
+ Expect(repo.CountAll()).To(Equal(int64(2)))
+ })
+ })
- err := dba.PostScan()
- Expect(err).ToNot(HaveOccurred())
- Expect(dba.Artist.Size).To(Equal(int64(1000)))
- Expect(dba.Artist.SongCount).To(Equal(10))
- Expect(dba.Artist.AlbumCount).To(Equal(2))
- Expect(dba.Artist.Stats).To(HaveLen(1))
- Expect(dba.Artist.Stats[model.RoleFromString("composer")].Size).To(Equal(int64(500)))
- Expect(dba.Artist.Stats[model.RoleFromString("composer")].SongCount).To(Equal(5))
- Expect(dba.Artist.Stats[model.RoleFromString("composer")].AlbumCount).To(Equal(1))
- Expect(dba.Artist.SimilarArtists).To(HaveLen(2))
- Expect(dba.Artist.SimilarArtists[0].ID).To(Equal("2"))
- Expect(dba.Artist.SimilarArtists[0].Name).To(Equal("AC/DC"))
- Expect(dba.Artist.SimilarArtists[1].ID).To(BeEmpty())
- Expect(dba.Artist.SimilarArtists[1].Name).To(Equal("Test;With:Sep,Chars"))
+ Describe("Exists", func() {
+ It("returns true for an artist that is in the DB", func() {
+ Expect(repo.Exists("3")).To(BeTrue())
+ })
+ It("returns false for an artist that is NOT in the DB", func() {
+ Expect(repo.Exists("666")).To(BeFalse())
+ })
+ })
+
+ Describe("Get", func() {
+ It("retrieves existing artist data", func() {
+ artist, err := repo.Get("2")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(artist.Name).To(Equal(artistKraftwerk.Name))
+ })
})
})
- Describe("PostMapArgs", func() {
- It("maps empty similar artists correctly", func() {
- m := make(map[string]any)
- err := dba.PostMapArgs(m)
- Expect(err).ToNot(HaveOccurred())
- Expect(m).To(HaveKeyWithValue("similar_artists", "[]"))
+ Describe("GetIndex", func() {
+ When("PreferSortTags is true", func() {
+ BeforeEach(func() {
+ conf.Server.PreferSortTags = true
+ })
+ It("returns the index when PreferSortTags is true and SortArtistName is not empty", func() {
+ // Set SortArtistName to "Foo" for Beatles
+ artistBeatles.SortArtistName = "Foo"
+ er := repo.Put(&artistBeatles)
+ Expect(er).To(BeNil())
+
+ idx, err := repo.GetIndex(false, []int{1})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(idx).To(HaveLen(2))
+ Expect(idx[0].ID).To(Equal("F"))
+ Expect(idx[0].Artists).To(HaveLen(1))
+ Expect(idx[0].Artists[0].Name).To(Equal(artistBeatles.Name))
+ Expect(idx[1].ID).To(Equal("K"))
+ Expect(idx[1].Artists).To(HaveLen(1))
+ Expect(idx[1].Artists[0].Name).To(Equal(artistKraftwerk.Name))
+
+ // Restore the original value
+ artistBeatles.SortArtistName = ""
+ er = repo.Put(&artistBeatles)
+ Expect(er).To(BeNil())
+ })
+
+ // BFR Empty SortArtistName is not saved in the DB anymore
+ XIt("returns the index when PreferSortTags is true and SortArtistName is empty", func() {
+ idx, err := repo.GetIndex(false, []int{1})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(idx).To(HaveLen(2))
+ Expect(idx[0].ID).To(Equal("B"))
+ Expect(idx[0].Artists).To(HaveLen(1))
+ Expect(idx[0].Artists[0].Name).To(Equal(artistBeatles.Name))
+ Expect(idx[1].ID).To(Equal("K"))
+ Expect(idx[1].Artists).To(HaveLen(1))
+ Expect(idx[1].Artists[0].Name).To(Equal(artistKraftwerk.Name))
+ })
})
- It("maps similar artists and full text correctly", func() {
- artist.SimilarArtists = []model.Artist{
- {ID: "2", Name: "AC/DC"},
- {Name: "Test;With:Sep,Chars"},
- }
- m := make(map[string]any)
- err := dba.PostMapArgs(m)
- Expect(err).ToNot(HaveOccurred())
- Expect(m).To(HaveKeyWithValue("similar_artists", `[{"id":"2","name":"AC/DC"},{"name":"Test;With:Sep,Chars"}]`))
- Expect(m).To(HaveKeyWithValue("full_text", " eddie halen van"))
+ When("PreferSortTags is false", func() {
+ BeforeEach(func() {
+ conf.Server.PreferSortTags = false
+ })
+ It("returns the index when SortArtistName is NOT empty", func() {
+ // Set SortArtistName to "Foo" for Beatles
+ artistBeatles.SortArtistName = "Foo"
+ er := repo.Put(&artistBeatles)
+ Expect(er).To(BeNil())
+
+ idx, err := repo.GetIndex(false, []int{1})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(idx).To(HaveLen(2))
+ Expect(idx[0].ID).To(Equal("B"))
+ Expect(idx[0].Artists).To(HaveLen(1))
+ Expect(idx[0].Artists[0].Name).To(Equal(artistBeatles.Name))
+ Expect(idx[1].ID).To(Equal("K"))
+ Expect(idx[1].Artists).To(HaveLen(1))
+ Expect(idx[1].Artists[0].Name).To(Equal(artistKraftwerk.Name))
+
+ // Restore the original value
+ artistBeatles.SortArtistName = ""
+ er = repo.Put(&artistBeatles)
+ Expect(er).To(BeNil())
+ })
+
+ It("returns the index when SortArtistName is empty", func() {
+ idx, err := repo.GetIndex(false, []int{1})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(idx).To(HaveLen(2))
+ Expect(idx[0].ID).To(Equal("B"))
+ Expect(idx[0].Artists).To(HaveLen(1))
+ Expect(idx[0].Artists[0].Name).To(Equal(artistBeatles.Name))
+ Expect(idx[1].ID).To(Equal("K"))
+ Expect(idx[1].Artists).To(HaveLen(1))
+ Expect(idx[1].Artists[0].Name).To(Equal(artistKraftwerk.Name))
+ })
})
- It("does not override empty sort_artist_name and mbz_artist_id", func() {
- m := map[string]any{
- "sort_artist_name": "",
- "mbz_artist_id": "",
- }
- err := dba.PostMapArgs(m)
- Expect(err).ToNot(HaveOccurred())
- Expect(m).ToNot(HaveKey("sort_artist_name"))
- Expect(m).ToNot(HaveKey("mbz_artist_id"))
+ When("filtering by role", func() {
+ var raw *artistRepository
+
+ BeforeEach(func() {
+ raw = repo.(*artistRepository)
+ // Add stats to library_artist table since stats are now stored per-library
+ composerStats := `{"composer": {"s": 1000, "m": 5, "a": 2}}`
+ producerStats := `{"producer": {"s": 500, "m": 3, "a": 1}}`
+
+ // Set Beatles as composer in library 1
+ _, err := raw.executeSQL(squirrel.Insert("library_artist").
+ Columns("library_id", "artist_id", "stats").
+ Values(1, artistBeatles.ID, composerStats).
+ Suffix("ON CONFLICT(library_id, artist_id) DO UPDATE SET stats = excluded.stats"))
+ Expect(err).ToNot(HaveOccurred())
+
+ // Set Kraftwerk as producer in library 1
+ _, err = raw.executeSQL(squirrel.Insert("library_artist").
+ Columns("library_id", "artist_id", "stats").
+ Values(1, artistKraftwerk.ID, producerStats).
+ Suffix("ON CONFLICT(library_id, artist_id) DO UPDATE SET stats = excluded.stats"))
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ AfterEach(func() {
+ // Clean up stats from library_artist table
+ _, _ = raw.executeSQL(squirrel.Update("library_artist").
+ Set("stats", "{}").
+ Where(squirrel.Eq{"artist_id": artistBeatles.ID, "library_id": 1}))
+ _, _ = raw.executeSQL(squirrel.Update("library_artist").
+ Set("stats", "{}").
+ Where(squirrel.Eq{"artist_id": artistKraftwerk.ID, "library_id": 1}))
+ })
+
+ It("returns only artists with the specified role", func() {
+ idx, err := repo.GetIndex(false, []int{1}, model.RoleComposer)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(idx).To(HaveLen(1))
+ Expect(idx[0].ID).To(Equal("B"))
+ Expect(idx[0].Artists).To(HaveLen(1))
+ Expect(idx[0].Artists[0].Name).To(Equal(artistBeatles.Name))
+ })
+
+ It("returns artists with any of the specified roles", func() {
+ idx, err := repo.GetIndex(false, []int{1}, model.RoleComposer, model.RoleProducer)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(idx).To(HaveLen(2))
+
+ // Find Beatles and Kraftwerk in the results
+ var beatlesFound, kraftwerkFound bool
+ for _, index := range idx {
+ for _, artist := range index.Artists {
+ if artist.Name == artistBeatles.Name {
+ beatlesFound = true
+ }
+ if artist.Name == artistKraftwerk.Name {
+ kraftwerkFound = true
+ }
+ }
+ }
+ Expect(beatlesFound).To(BeTrue())
+ Expect(kraftwerkFound).To(BeTrue())
+ })
+
+ It("returns empty index when no artists have the specified role", func() {
+ idx, err := repo.GetIndex(false, []int{1}, model.RoleDirector)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(idx).To(HaveLen(0))
+ })
+ })
+
+ When("validating library IDs", func() {
+ It("returns nil when no library IDs are provided", func() {
+ idx, err := repo.GetIndex(false, []int{})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(idx).To(BeNil())
+ })
+
+ It("returns artists when library IDs are provided (admin user sees all content)", func() {
+ // Admin users can see all content when valid library IDs are provided
+ idx, err := repo.GetIndex(false, []int{1})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(idx).To(HaveLen(2))
+
+ // With non-existent library ID, admin users see no content because no artists are associated with that library
+ idx, err = repo.GetIndex(false, []int{999})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(idx).To(HaveLen(0)) // Even admin users need valid library associations
+ })
})
})
- Describe("Missing artist visibility", func() {
+ Describe("MBID and Text Search", func() {
+ var lib2 model.Library
+ var lr model.LibraryRepository
+ var restrictedUser model.User
+ var restrictedRepo model.ArtistRepository
+ var headlessRepo model.ArtistRepository
+
+ BeforeEach(func() {
+ // Set up headless repo (no user context)
+ headlessRepo = NewArtistRepository(context.Background(), GetDBXBuilder())
+
+ // Create library for testing access restrictions
+ lib2 = model.Library{ID: 0, Name: "Artist Test Library", Path: "/artist/test/lib"}
+ lr = NewLibraryRepository(request.WithUser(GinkgoT().Context(), adminUser), GetDBXBuilder())
+ err := lr.Put(&lib2)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Create a user with access to only library 1
+ restrictedUser = createUserWithLibraries("search_user", []int{1})
+
+ // Create repository context for the restricted user
+ ctx := request.WithUser(GinkgoT().Context(), restrictedUser)
+ restrictedRepo = NewArtistRepository(ctx, GetDBXBuilder())
+
+ // Ensure both test artists are associated with library 1
+ err = lr.AddArtist(1, artistBeatles.ID)
+ Expect(err).ToNot(HaveOccurred())
+ err = lr.AddArtist(1, artistKraftwerk.ID)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Create the restricted user in the database
+ ur := NewUserRepository(request.WithUser(GinkgoT().Context(), adminUser), GetDBXBuilder())
+ err = ur.Put(&restrictedUser)
+ Expect(err).ToNot(HaveOccurred())
+ err = ur.SetUserLibraries(restrictedUser.ID, []int{1})
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ AfterEach(func() {
+ // Clean up library 2
+ lr := NewLibraryRepository(request.WithUser(GinkgoT().Context(), adminUser), GetDBXBuilder())
+ _ = lr.(*libraryRepository).delete(squirrel.Eq{"id": lib2.ID})
+ })
+
+ DescribeTable("MBID search behavior across different user types",
+ func(testRepo *model.ArtistRepository, shouldFind bool, testDesc string) {
+ // Create test artist with MBID
+ artistWithMBID := createTestArtistWithMBID("test-mbid-artist", "Test MBID Artist", "550e8400-e29b-41d4-a716-446655440010")
+
+ err := createArtistWithLibrary(*testRepo, &artistWithMBID, 1)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Test the search
+ results, err := (*testRepo).Search("550e8400-e29b-41d4-a716-446655440010", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+
+ if shouldFind {
+ Expect(results).To(HaveLen(1), testDesc)
+ Expect(results[0].ID).To(Equal("test-mbid-artist"))
+ } else {
+ Expect(results).To(BeEmpty(), testDesc)
+ }
+
+ // Clean up
+ if raw, ok := (*testRepo).(*artistRepository); ok {
+ _, _ = raw.executeSQL(squirrel.Delete(raw.tableName).Where(squirrel.Eq{"id": artistWithMBID.ID}))
+ }
+ },
+ Entry("Admin user can find artist by MBID", &repo, true, "Admin should find MBID artist"),
+ Entry("Restricted user can find artist by MBID in accessible library", &restrictedRepo, true, "Restricted user should find MBID artist in accessible library"),
+ Entry("Headless process can find artist by MBID", &headlessRepo, true, "Headless process should find MBID artist"),
+ )
+
+ It("prevents restricted user from finding artist by MBID when not in accessible library", func() {
+ // Create an artist in library 2 (not accessible to restricted user)
+ inaccessibleArtist := createTestArtistWithMBID("inaccessible-mbid-artist", "Inaccessible MBID Artist", "a74b1b7f-71a5-4011-9441-d0b5e4122711")
+ err := repo.Put(&inaccessibleArtist)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Add to library 2 (not accessible to restricted user)
+ err = lr.AddArtist(lib2.ID, inaccessibleArtist.ID)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Restricted user should not find this artist
+ results, err := restrictedRepo.Search("a74b1b7f-71a5-4011-9441-d0b5e4122711", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(BeEmpty())
+
+ // But admin should find it
+ results, err = repo.Search("a74b1b7f-71a5-4011-9441-d0b5e4122711", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(1))
+
+ // Clean up
+ if raw, ok := repo.(*artistRepository); ok {
+ _, _ = raw.executeSQL(squirrel.Delete(raw.tableName).Where(squirrel.Eq{"id": inaccessibleArtist.ID}))
+ }
+ })
+
+ Context("Text Search", func() {
+ It("allows admin to find artists by name regardless of library", func() {
+ results, err := repo.Search("Beatles", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(1))
+ Expect(results[0].Name).To(Equal("The Beatles"))
+ })
+
+ It("correctly prevents restricted user from finding artists by name when not in accessible library", func() {
+ // Create an artist in library 2 (not accessible to restricted user)
+ inaccessibleArtist := model.Artist{
+ ID: "inaccessible-text-artist",
+ Name: "Unique Search Name Artist",
+ }
+ err := repo.Put(&inaccessibleArtist)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Add to library 2 (not accessible to restricted user)
+ err = lr.AddArtist(lib2.ID, inaccessibleArtist.ID)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Restricted user should not find this artist
+ results, err := restrictedRepo.Search("Unique Search Name", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(BeEmpty(), "Text search should respect library filtering")
+
+ // Clean up
+ if raw, ok := repo.(*artistRepository); ok {
+ _, _ = raw.executeSQL(squirrel.Delete(raw.tableName).Where(squirrel.Eq{"id": inaccessibleArtist.ID}))
+ }
+ })
+ })
+
+ Context("Headless Processes (No User Context)", func() {
+ It("should see all artists from all libraries when no user is in context", func() {
+ // Add artists to different libraries
+ err := lr.AddArtist(lib2.ID, artistBeatles.ID)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Headless processes should see all artists regardless of library
+ artists, err := headlessRepo.GetAll()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Should see all artists from all libraries
+ found := false
+ for _, artist := range artists {
+ if artist.ID == artistBeatles.ID {
+ found = true
+ break
+ }
+ }
+ Expect(found).To(BeTrue(), "Headless process should see artists from all libraries")
+ })
+
+ It("should allow headless processes to apply explicit library_id filters", func() {
+ // Add artists to different libraries
+ err := lr.AddArtist(lib2.ID, artistBeatles.ID)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Filter by specific library
+ artists, err := headlessRepo.GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+
+ // Should see only artists from the specified library
+ for _, artist := range artists {
+ if artist.ID == artistBeatles.ID {
+ return // Found the expected artist
+ }
+ }
+ Expect(false).To(BeTrue(), "Should find artist from specified library")
+ })
+
+ It("should get individual artists when no user is in context", func() {
+ // Add artist to a library
+ err := lr.AddArtist(lib2.ID, artistBeatles.ID)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Headless process should be able to get the artist
+ artist, err := headlessRepo.Get(artistBeatles.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(artist.ID).To(Equal(artistBeatles.ID))
+ })
+ })
+ })
+
+ Describe("Admin User Library Access", func() {
+ It("sees all artists regardless of library permissions", func() {
+ count, err := repo.CountAll()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(count).To(Equal(int64(2)))
+
+ artists, err := repo.GetAll()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(artists).To(HaveLen(2))
+
+ exists, err := repo.Exists(artistBeatles.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(exists).To(BeTrue())
+ })
+ })
+
+ Describe("Missing Artist Handling", func() {
+ var missingArtist model.Artist
var raw *artistRepository
- var missing model.Artist
- insertMissing := func() {
- missing = model.Artist{ID: "m1", Name: "Missing", OrderArtistName: "missing"}
- Expect(repo.Put(&missing)).To(Succeed())
+ BeforeEach(func() {
raw = repo.(*artistRepository)
- _, err := raw.executeSQL(squirrel.Update(raw.tableName).Set("missing", true).Where(squirrel.Eq{"id": missing.ID}))
+ missingArtist = model.Artist{ID: "missing_test", Name: "Missing Artist", OrderArtistName: "missing artist"}
+
+ // Create and mark as missing
+ err := createArtistWithLibrary(repo, &missingArtist, 1)
Expect(err).ToNot(HaveOccurred())
- }
- removeMissing := func() {
- if raw != nil {
- _, _ = raw.executeSQL(squirrel.Delete(raw.tableName).Where(squirrel.Eq{"id": missing.ID}))
- }
- }
-
- Context("regular user", func() {
- BeforeEach(func() {
- ctx := log.NewContext(context.TODO())
- ctx = request.WithUser(ctx, model.User{ID: "u1"})
- repo = NewArtistRepository(ctx, GetDBXBuilder())
- insertMissing()
- })
-
- AfterEach(func() { removeMissing() })
-
- It("does not return missing artist in GetAll", func() {
- artists, err := repo.GetAll(model.QueryOptions{Filters: squirrel.Eq{"artist.missing": false}})
- Expect(err).ToNot(HaveOccurred())
- Expect(artists).To(HaveLen(2))
- })
-
- It("does not return missing artist in Search", func() {
- res, err := repo.Search("missing", 0, 10, false)
- Expect(err).ToNot(HaveOccurred())
- Expect(res).To(BeEmpty())
- })
-
- It("does not return missing artist in GetIndex", func() {
- idx, err := repo.GetIndex(false)
- Expect(err).ToNot(HaveOccurred())
- // Only 2 artists should be present
- total := 0
- for _, ix := range idx {
- total += len(ix.Artists)
- }
- Expect(total).To(Equal(2))
- })
+ _, err = raw.executeSQL(squirrel.Update(raw.tableName).Set("missing", true).Where(squirrel.Eq{"id": missingArtist.ID}))
+ Expect(err).ToNot(HaveOccurred())
})
- Context("admin user", func() {
- BeforeEach(func() {
- ctx := log.NewContext(context.TODO())
- ctx = request.WithUser(ctx, model.User{ID: "admin", IsAdmin: true})
- repo = NewArtistRepository(ctx, GetDBXBuilder())
- insertMissing()
- })
+ AfterEach(func() {
+ _, _ = raw.executeSQL(squirrel.Delete(raw.tableName).Where(squirrel.Eq{"id": missingArtist.ID}))
+ })
- AfterEach(func() { removeMissing() })
+ It("missing artists are never returned by search", func() {
+ // Should see missing artist in GetAll by default for admin users
+ artists, err := repo.GetAll()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(artists).To(HaveLen(3)) // Including the missing artist
- It("returns missing artist in GetAll", func() {
- artists, err := repo.GetAll()
- Expect(err).ToNot(HaveOccurred())
- Expect(artists).To(HaveLen(3))
- })
+ // Search never returns missing artists (hardcoded behavior)
+ results, err := repo.Search("Missing Artist", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(BeEmpty())
+ })
+ })
+ })
- It("returns missing artist in Search", func() {
- res, err := repo.Search("missing", 0, 10, true)
- Expect(err).ToNot(HaveOccurred())
- Expect(res).To(HaveLen(1))
- })
+ Context("Regular User Operations", func() {
+ var restrictedRepo model.ArtistRepository
+ var unauthorizedUser model.User
- It("returns missing artist in GetIndex when included", func() {
- idx, err := repo.GetIndex(true)
- Expect(err).ToNot(HaveOccurred())
- total := 0
- for _, ix := range idx {
- total += len(ix.Artists)
- }
- Expect(total).To(Equal(3))
- })
+ BeforeEach(func() {
+ // Create a user without access to any libraries
+ unauthorizedUser = model.User{ID: "restricted_user", UserName: "restricted", Name: "Restricted User", Email: "restricted@test.com", IsAdmin: false}
+
+ // Create repository context for the unauthorized user
+ ctx := GinkgoT().Context()
+ ctx = request.WithUser(ctx, unauthorizedUser)
+ restrictedRepo = NewArtistRepository(ctx, GetDBXBuilder())
+ })
+
+ Describe("Library Access Restrictions", func() {
+ It("CountAll returns 0 for users without library access", func() {
+ count, err := restrictedRepo.CountAll()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(count).To(Equal(int64(0)))
+ })
+
+ It("GetAll returns empty list for users without library access", func() {
+ artists, err := restrictedRepo.GetAll()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(artists).To(BeEmpty())
+ })
+
+ It("Exists returns false for existing artists when user has no library access", func() {
+ // These artists exist in the DB but the user has no access to them
+ exists, err := restrictedRepo.Exists(artistBeatles.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(exists).To(BeFalse())
+
+ exists, err = restrictedRepo.Exists(artistKraftwerk.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(exists).To(BeFalse())
+ })
+
+ It("Get returns ErrNotFound for existing artists when user has no library access", func() {
+ _, err := restrictedRepo.Get(artistBeatles.ID)
+ Expect(err).To(Equal(model.ErrNotFound))
+
+ _, err = restrictedRepo.Get(artistKraftwerk.ID)
+ Expect(err).To(Equal(model.ErrNotFound))
+ })
+
+ It("Search returns empty results for users without library access", func() {
+ results, err := restrictedRepo.Search("Beatles", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(BeEmpty())
+
+ results, err = restrictedRepo.Search("Kraftwerk", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(BeEmpty())
+ })
+
+ It("GetIndex returns empty index for users without library access", func() {
+ idx, err := restrictedRepo.GetIndex(false, []int{1})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(idx).To(HaveLen(0))
+ })
+ })
+
+ Context("when user gains library access", func() {
+ BeforeEach(func() {
+ ctx := GinkgoT().Context()
+ // Give the user access to library 1
+ ur := NewUserRepository(request.WithUser(ctx, adminUser), GetDBXBuilder())
+
+ // First create the user if not exists
+ err := ur.Put(&unauthorizedUser)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Then add library access
+ err = ur.SetUserLibraries(unauthorizedUser.ID, []int{1})
+ Expect(err).ToNot(HaveOccurred())
+
+ // Update the user object with the libraries to simulate middleware behavior
+ libraries, err := ur.GetUserLibraries(unauthorizedUser.ID)
+ Expect(err).ToNot(HaveOccurred())
+ unauthorizedUser.Libraries = libraries
+
+ // Recreate repository context with updated user
+ ctx = request.WithUser(ctx, unauthorizedUser)
+ restrictedRepo = NewArtistRepository(ctx, GetDBXBuilder())
+ })
+
+ AfterEach(func() {
+ // Clean up: remove the user's library access
+ ur := NewUserRepository(request.WithUser(GinkgoT().Context(), adminUser), GetDBXBuilder())
+ _ = ur.SetUserLibraries(unauthorizedUser.ID, []int{})
+ })
+
+ It("CountAll returns correct count after gaining access", func() {
+ count, err := restrictedRepo.CountAll()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(count).To(Equal(int64(2))) // Beatles and Kraftwerk
+ })
+
+ It("GetAll returns artists after gaining access", func() {
+ artists, err := restrictedRepo.GetAll()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(artists).To(HaveLen(2))
+
+ var names []string
+ for _, artist := range artists {
+ names = append(names, artist.Name)
+ }
+ Expect(names).To(ContainElements("The Beatles", "Kraftwerk"))
+ })
+
+ It("Exists returns true for accessible artists", func() {
+ exists, err := restrictedRepo.Exists(artistBeatles.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(exists).To(BeTrue())
+
+ exists, err = restrictedRepo.Exists(artistKraftwerk.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(exists).To(BeTrue())
+ })
+
+ It("GetIndex returns artists with proper library filtering", func() {
+ // With valid library access, should see artists
+ idx, err := restrictedRepo.GetIndex(false, []int{1})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(idx).To(HaveLen(2))
+
+ // With non-existent library ID, should see nothing (non-admin user)
+ idx, err = restrictedRepo.GetIndex(false, []int{999})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(idx).To(HaveLen(0))
})
})
})
})
+
+// Helper function to create an artist with proper library association.
+// This ensures test artists always have library_artist associations to avoid orphaned artists in tests.
+func createArtistWithLibrary(repo model.ArtistRepository, artist *model.Artist, libraryID int) error {
+ err := repo.Put(artist)
+ if err != nil {
+ return err
+ }
+
+ // Add the artist to the specified library
+ lr := NewLibraryRepository(request.WithUser(GinkgoT().Context(), adminUser), GetDBXBuilder())
+ return lr.AddArtist(libraryID, artist.ID)
+}
diff --git a/persistence/folder_repository.go b/persistence/folder_repository.go
index a8b7884b7..a586746a0 100644
--- a/persistence/folder_repository.go
+++ b/persistence/folder_repository.go
@@ -4,7 +4,10 @@ import (
"context"
"encoding/json"
"fmt"
+ "os"
+ "path/filepath"
"slices"
+ "strings"
"time"
. "github.com/Masterminds/squirrel"
@@ -61,8 +64,9 @@ func newFolderRepository(ctx context.Context, db dbx.Builder) model.FolderReposi
}
func (r folderRepository) selectFolder(options ...model.QueryOptions) SelectBuilder {
- return r.newSelect(options...).Columns("folder.*", "library.path as library_path").
+ sql := r.newSelect(options...).Columns("folder.*", "library.path as library_path").
Join("library on library.id = folder.library_id")
+ return r.applyLibraryFilter(sql)
}
func (r folderRepository) Get(id string) (*model.Folder, error) {
@@ -85,23 +89,64 @@ func (r folderRepository) GetAll(opt ...model.QueryOptions) ([]model.Folder, err
}
func (r folderRepository) CountAll(opt ...model.QueryOptions) (int64, error) {
- sq := r.newSelect(opt...).Columns("count(*)")
- return r.count(sq)
+ query := r.newSelect(opt...).Columns("count(*)")
+ query = r.applyLibraryFilter(query)
+ return r.count(query)
}
-func (r folderRepository) GetLastUpdates(lib model.Library) (map[string]time.Time, error) {
- sq := r.newSelect().Columns("id", "updated_at").Where(Eq{"library_id": lib.ID, "missing": false})
+func (r folderRepository) GetFolderUpdateInfo(lib model.Library, targetPaths ...string) (map[string]model.FolderUpdateInfo, error) {
+ where := And{
+ Eq{"library_id": lib.ID},
+ Eq{"missing": false},
+ }
+
+ // If specific paths are requested, include those folders and all their descendants
+ if len(targetPaths) > 0 {
+ // Collect folder IDs for exact target folders and path conditions for descendants
+ folderIDs := make([]string, 0, len(targetPaths))
+ pathConditions := make(Or, 0, len(targetPaths)*2)
+
+ for _, targetPath := range targetPaths {
+ if targetPath == "" || targetPath == "." {
+ // Root path - include everything in this library
+ pathConditions = Or{}
+ folderIDs = nil
+ break
+ }
+ // Clean the path to normalize it. Paths stored in the folder table do not have leading/trailing slashes.
+ cleanPath := strings.TrimPrefix(targetPath, string(os.PathSeparator))
+ cleanPath = filepath.Clean(cleanPath)
+
+ // Include the target folder itself by ID
+ folderIDs = append(folderIDs, model.FolderID(lib, cleanPath))
+
+ // Include all descendants: folders whose path field equals or starts with the target path
+ // Note: Folder.Path is the directory path, so children have path = targetPath
+ pathConditions = append(pathConditions, Eq{"path": cleanPath})
+ pathConditions = append(pathConditions, Like{"path": cleanPath + "/%"})
+ }
+
+ // Combine conditions: exact folder IDs OR descendant path patterns
+ if len(folderIDs) > 0 {
+ where = append(where, Or{Eq{"id": folderIDs}, pathConditions})
+ } else if len(pathConditions) > 0 {
+ where = append(where, pathConditions)
+ }
+ }
+
+ sq := r.newSelect().Columns("id", "updated_at", "hash").Where(where)
var res []struct {
ID string
UpdatedAt time.Time
+ Hash string
}
err := r.queryAll(sq, &res)
if err != nil {
return nil, err
}
- m := make(map[string]time.Time, len(res))
+ m := make(map[string]model.FolderUpdateInfo, len(res))
for _, f := range res {
- m[f.ID] = f.UpdatedAt
+ m[f.ID] = model.FolderUpdateInfo{UpdatedAt: f.UpdatedAt, Hash: f.Hash}
}
return m, nil
}
@@ -146,7 +191,7 @@ func (r folderRepository) GetTouchedWithPlaylists() (model.FolderCursor, error)
}, nil
}
-func (r folderRepository) purgeEmpty() error {
+func (r folderRepository) purgeEmpty(libraryIDs ...int) error {
sq := Delete(r.tableName).Where(And{
Eq{"num_audio_files": 0},
Eq{"num_playlists": 0},
@@ -154,6 +199,10 @@ func (r folderRepository) purgeEmpty() error {
ConcatExpr("id not in (select parent_id from folder)"),
ConcatExpr("id not in (select folder_id from media_file)"),
})
+ // If libraryIDs are specified, only purge folders from those libraries
+ if len(libraryIDs) > 0 {
+ sq = sq.Where(Eq{"library_id": libraryIDs})
+ }
c, err := r.executeSQL(sq)
if err != nil {
return fmt.Errorf("purging empty folders: %w", err)
diff --git a/persistence/folder_repository_test.go b/persistence/folder_repository_test.go
new file mode 100644
index 000000000..6c24741c9
--- /dev/null
+++ b/persistence/folder_repository_test.go
@@ -0,0 +1,213 @@
+package persistence
+
+import (
+ "context"
+ "fmt"
+
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+ "github.com/pocketbase/dbx"
+)
+
+var _ = Describe("FolderRepository", func() {
+ var repo model.FolderRepository
+ var ctx context.Context
+ var conn *dbx.DB
+ var testLib, otherLib model.Library
+
+ BeforeEach(func() {
+ ctx = request.WithUser(log.NewContext(context.TODO()), model.User{ID: "userid"})
+ conn = GetDBXBuilder()
+ repo = newFolderRepository(ctx, conn)
+
+ // Use existing library ID 1 from test fixtures
+ libRepo := NewLibraryRepository(ctx, conn)
+ lib, err := libRepo.Get(1)
+ Expect(err).ToNot(HaveOccurred())
+ testLib = *lib
+
+ // Create a second library with its own folder to verify isolation
+ otherLib = model.Library{Name: "Other Library", Path: "/other/path"}
+ Expect(libRepo.Put(&otherLib)).To(Succeed())
+ })
+
+ AfterEach(func() {
+ // Clean up only test folders created by our tests (paths starting with "Test")
+ // This prevents interference with fixture data needed by other tests
+ _, _ = conn.NewQuery("DELETE FROM folder WHERE library_id = 1 AND path LIKE 'Test%'").Execute()
+ _, _ = conn.NewQuery(fmt.Sprintf("DELETE FROM library WHERE id = %d", otherLib.ID)).Execute()
+ })
+
+ Describe("GetFolderUpdateInfo", func() {
+ Context("with no target paths", func() {
+ It("returns all folders in the library", func() {
+ // Create test folders with unique names to avoid conflicts
+ folder1 := model.NewFolder(testLib, "TestGetLastUpdates/Folder1")
+ folder2 := model.NewFolder(testLib, "TestGetLastUpdates/Folder2")
+
+ err := repo.Put(folder1)
+ Expect(err).ToNot(HaveOccurred())
+ err = repo.Put(folder2)
+ Expect(err).ToNot(HaveOccurred())
+
+ otherFolder := model.NewFolder(otherLib, "TestOtherLib/Folder")
+ err = repo.Put(otherFolder)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Query all folders (no target paths) - should only return folders from testLib
+ results, err := repo.GetFolderUpdateInfo(testLib)
+ Expect(err).ToNot(HaveOccurred())
+ // Should include folders from testLib
+ Expect(results).To(HaveKey(folder1.ID))
+ Expect(results).To(HaveKey(folder2.ID))
+ // Should NOT include folders from other library
+ Expect(results).ToNot(HaveKey(otherFolder.ID))
+ })
+ })
+
+ Context("with specific target paths", func() {
+ It("returns folder info for existing folders", func() {
+ // Create test folders with unique names
+ folder1 := model.NewFolder(testLib, "TestSpecific/Rock")
+ folder2 := model.NewFolder(testLib, "TestSpecific/Jazz")
+ folder3 := model.NewFolder(testLib, "TestSpecific/Classical")
+
+ err := repo.Put(folder1)
+ Expect(err).ToNot(HaveOccurred())
+ err = repo.Put(folder2)
+ Expect(err).ToNot(HaveOccurred())
+ err = repo.Put(folder3)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Query specific paths
+ results, err := repo.GetFolderUpdateInfo(testLib, "TestSpecific/Rock", "TestSpecific/Classical")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(2))
+
+ // Verify folder IDs are in results
+ Expect(results).To(HaveKey(folder1.ID))
+ Expect(results).To(HaveKey(folder3.ID))
+ Expect(results).ToNot(HaveKey(folder2.ID))
+
+ // Verify update info is populated
+ Expect(results[folder1.ID].UpdatedAt).ToNot(BeZero())
+ Expect(results[folder1.ID].Hash).To(Equal(folder1.Hash))
+ })
+
+ It("includes all child folders when querying parent", func() {
+ // Create a parent folder with multiple children
+ parent := model.NewFolder(testLib, "TestParent/Music")
+ child1 := model.NewFolder(testLib, "TestParent/Music/Rock/Queen")
+ child2 := model.NewFolder(testLib, "TestParent/Music/Jazz")
+ otherParent := model.NewFolder(testLib, "TestParent2/Music/Jazz")
+
+ Expect(repo.Put(parent)).To(Succeed())
+ Expect(repo.Put(child1)).To(Succeed())
+ Expect(repo.Put(child2)).To(Succeed())
+
+ // Query the parent folder - should return parent and all children
+ results, err := repo.GetFolderUpdateInfo(testLib, "TestParent/Music")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(3))
+ Expect(results).To(HaveKey(parent.ID))
+ Expect(results).To(HaveKey(child1.ID))
+ Expect(results).To(HaveKey(child2.ID))
+ Expect(results).ToNot(HaveKey(otherParent.ID))
+ })
+
+ It("excludes children from other libraries", func() {
+ // Create parent in testLib
+ parent := model.NewFolder(testLib, "TestIsolation/Parent")
+ child := model.NewFolder(testLib, "TestIsolation/Parent/Child")
+
+ Expect(repo.Put(parent)).To(Succeed())
+ Expect(repo.Put(child)).To(Succeed())
+
+ // Create similar path in other library
+ otherParent := model.NewFolder(otherLib, "TestIsolation/Parent")
+ otherChild := model.NewFolder(otherLib, "TestIsolation/Parent/Child")
+
+ Expect(repo.Put(otherParent)).To(Succeed())
+ Expect(repo.Put(otherChild)).To(Succeed())
+
+ // Query should only return folders from testLib
+ results, err := repo.GetFolderUpdateInfo(testLib, "TestIsolation/Parent")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(2))
+ Expect(results).To(HaveKey(parent.ID))
+ Expect(results).To(HaveKey(child.ID))
+ Expect(results).ToNot(HaveKey(otherParent.ID))
+ Expect(results).ToNot(HaveKey(otherChild.ID))
+ })
+
+ It("excludes missing children when querying parent", func() {
+ // Create parent and children, mark one as missing
+ parent := model.NewFolder(testLib, "TestMissingChild/Parent")
+ child1 := model.NewFolder(testLib, "TestMissingChild/Parent/Child1")
+ child2 := model.NewFolder(testLib, "TestMissingChild/Parent/Child2")
+ child2.Missing = true
+
+ Expect(repo.Put(parent)).To(Succeed())
+ Expect(repo.Put(child1)).To(Succeed())
+ Expect(repo.Put(child2)).To(Succeed())
+
+ // Query parent - should only return parent and non-missing child
+ results, err := repo.GetFolderUpdateInfo(testLib, "TestMissingChild/Parent")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(2))
+ Expect(results).To(HaveKey(parent.ID))
+ Expect(results).To(HaveKey(child1.ID))
+ Expect(results).ToNot(HaveKey(child2.ID))
+ })
+
+ It("handles mix of existing and non-existing target paths", func() {
+ // Create folders for one path but not the other
+ existingParent := model.NewFolder(testLib, "TestMixed/Exists")
+ existingChild := model.NewFolder(testLib, "TestMixed/Exists/Child")
+
+ Expect(repo.Put(existingParent)).To(Succeed())
+ Expect(repo.Put(existingChild)).To(Succeed())
+
+ // Query both existing and non-existing paths
+ results, err := repo.GetFolderUpdateInfo(testLib, "TestMixed/Exists", "TestMixed/DoesNotExist")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(2))
+ Expect(results).To(HaveKey(existingParent.ID))
+ Expect(results).To(HaveKey(existingChild.ID))
+ })
+
+ It("handles empty folder path as root", func() {
+ // Test querying for root folder without creating it (fixtures should have one)
+ rootFolderID := model.FolderID(testLib, ".")
+
+ results, err := repo.GetFolderUpdateInfo(testLib, "")
+ Expect(err).ToNot(HaveOccurred())
+ // Should return the root folder if it exists
+ if len(results) > 0 {
+ Expect(results).To(HaveKey(rootFolderID))
+ }
+ })
+
+ It("returns empty map for non-existent folders", func() {
+ results, err := repo.GetFolderUpdateInfo(testLib, "NonExistent/Path")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(BeEmpty())
+ })
+
+ It("skips missing folders", func() {
+ // Create a folder and mark it as missing
+ folder := model.NewFolder(testLib, "TestMissing/Folder")
+ folder.Missing = true
+ err := repo.Put(folder)
+ Expect(err).ToNot(HaveOccurred())
+
+ results, err := repo.GetFolderUpdateInfo(testLib, "TestMissing/Folder")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(BeEmpty())
+ })
+ })
+ })
+})
diff --git a/persistence/genre_repository.go b/persistence/genre_repository.go
index e92e1491a..5857350a6 100644
--- a/persistence/genre_repository.go
+++ b/persistence/genre_repository.go
@@ -10,31 +10,18 @@ import (
)
type genreRepository struct {
- sqlRepository
+ *baseTagRepository
}
func NewGenreRepository(ctx context.Context, db dbx.Builder) model.GenreRepository {
- r := &genreRepository{}
- r.ctx = ctx
- r.db = db
- r.registerModel(&model.Tag{}, map[string]filterFunc{
- "name": containsFilter("tag_value"),
- })
- r.setSortMappings(map[string]string{
- "name": "tag_name",
- })
- return r
+ genreFilter := model.TagGenre
+ return &genreRepository{
+ baseTagRepository: newBaseTagRepository(ctx, db, &genreFilter),
+ }
}
func (r *genreRepository) selectGenre(opt ...model.QueryOptions) SelectBuilder {
- return r.newSelect(opt...).
- Columns(
- "id",
- "tag_value as name",
- "album_count",
- "media_file_count as song_count",
- ).
- Where(Eq{"tag.tag_name": model.TagGenre})
+ return r.newSelect(opt...).Columns("tag.tag_value as name")
}
func (r *genreRepository) GetAll(opt ...model.QueryOptions) (model.Genres, error) {
@@ -44,12 +31,10 @@ func (r *genreRepository) GetAll(opt ...model.QueryOptions) (model.Genres, error
return res, err
}
-func (r *genreRepository) Count(options ...rest.QueryOptions) (int64, error) {
- return r.count(r.selectGenre(), r.parseRestOptions(r.ctx, options...))
-}
+// Override ResourceRepository methods to return Genre objects instead of Tag objects
func (r *genreRepository) Read(id string) (interface{}, error) {
- sel := r.selectGenre().Columns("*").Where(Eq{"id": id})
+ sel := r.selectGenre().Where(Eq{"tag.id": id})
var res model.Genre
err := r.queryOne(sel, &res)
return &res, err
@@ -59,10 +44,6 @@ func (r *genreRepository) ReadAll(options ...rest.QueryOptions) (interface{}, er
return r.GetAll(r.parseRestOptions(r.ctx, options...))
}
-func (r *genreRepository) EntityName() string {
- return r.tableName
-}
-
func (r *genreRepository) NewInstance() interface{} {
return &model.Genre{}
}
diff --git a/persistence/genre_repository_test.go b/persistence/genre_repository_test.go
new file mode 100644
index 000000000..67e84ce51
--- /dev/null
+++ b/persistence/genre_repository_test.go
@@ -0,0 +1,329 @@
+package persistence
+
+import (
+ "context"
+ "slices"
+ "strings"
+
+ "github.com/Masterminds/squirrel"
+ "github.com/deluan/rest"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/id"
+ "github.com/navidrome/navidrome/model/request"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("GenreRepository", func() {
+ var repo model.GenreRepository
+ var restRepo model.ResourceRepository
+ var tagRepo model.TagRepository
+ var ctx context.Context
+
+ BeforeEach(func() {
+ ctx = request.WithUser(GinkgoT().Context(), model.User{ID: "userid", UserName: "johndoe", IsAdmin: true})
+ genreRepo := NewGenreRepository(ctx, GetDBXBuilder())
+ repo = genreRepo
+ restRepo = genreRepo.(model.ResourceRepository)
+ tagRepo = NewTagRepository(ctx, GetDBXBuilder())
+
+ // Clear any existing tags to ensure test isolation
+ db := GetDBXBuilder()
+ _, err := db.NewQuery("DELETE FROM tag").Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Ensure library 1 exists and user has access to it
+ _, err = db.NewQuery("INSERT OR IGNORE INTO library (id, name, path, default_new_users) VALUES (1, 'Test Library', '/test', true)").Execute()
+ Expect(err).ToNot(HaveOccurred())
+ _, err = db.NewQuery("INSERT OR IGNORE INTO user_library (user_id, library_id) VALUES ('userid', 1)").Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Add comprehensive test data that covers all test scenarios
+ newTag := func(name, value string) model.Tag {
+ return model.Tag{ID: id.NewTagID(name, value), TagName: model.TagName(name), TagValue: value}
+ }
+
+ err = tagRepo.Add(1,
+ newTag("genre", "rock"),
+ newTag("genre", "pop"),
+ newTag("genre", "jazz"),
+ newTag("genre", "electronic"),
+ newTag("genre", "classical"),
+ newTag("genre", "ambient"),
+ newTag("genre", "techno"),
+ newTag("genre", "house"),
+ newTag("genre", "trance"),
+ newTag("genre", "Alternative Rock"),
+ newTag("genre", "Blues"),
+ newTag("genre", "Country"),
+ // These should not be counted as genres
+ newTag("mood", "happy"),
+ newTag("mood", "ambient"),
+ )
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ Describe("GetAll", func() {
+ It("should return all genres", func() {
+ genres, err := repo.GetAll()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(genres).To(HaveLen(12))
+
+ // Verify that all returned items are genres (TagName = "genre")
+ genreNames := make([]string, len(genres))
+ for i, genre := range genres {
+ genreNames[i] = genre.Name
+ }
+ Expect(genreNames).To(ContainElement("rock"))
+ Expect(genreNames).To(ContainElement("pop"))
+ Expect(genreNames).To(ContainElement("jazz"))
+ // Should not contain mood tags
+ Expect(genreNames).ToNot(ContainElement("happy"))
+ })
+
+ It("should support query options", func() {
+ // Test with limiting results
+ genres, err := repo.GetAll(model.QueryOptions{Max: 1})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(genres).To(HaveLen(1))
+ })
+
+ It("should handle empty results gracefully", func() {
+ // Clear all genre tags
+ _, err := GetDBXBuilder().NewQuery("DELETE FROM tag WHERE tag_name = 'genre'").Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ genres, err := repo.GetAll()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(genres).To(BeEmpty())
+ })
+ Describe("filtering and sorting", func() {
+ It("should filter by name using like match", func() {
+ // Test filtering by partial name match using the "name" filter which maps to containsFilter("tag_value")
+ options := model.QueryOptions{
+ Filters: squirrel.Like{"tag_value": "%rock%"}, // Direct field access
+ }
+ genres, err := repo.GetAll(options)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(genres).To(HaveLen(2)) // Should match "rock" and "Alternative Rock"
+
+ // Verify all returned genres contain "rock" in their name
+ for _, genre := range genres {
+ Expect(strings.ToLower(genre.Name)).To(ContainSubstring("rock"))
+ }
+ })
+
+ It("should sort by name in ascending order", func() {
+ // Test sorting by name with the fixed mapping
+ options := model.QueryOptions{
+ Filters: squirrel.Like{"tag_value": "%e%"}, // Should match genres containing "e"
+ Sort: "name",
+ }
+ genres, err := repo.GetAll(options)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(genres).To(HaveLen(7))
+
+ Expect(slices.IsSortedFunc(genres, func(a, b model.Genre) int {
+ return strings.Compare(b.Name, a.Name) // Inverted to check descending order
+ }))
+ })
+
+ It("should sort by name in descending order", func() {
+ // Test sorting by name in descending order
+ options := model.QueryOptions{
+ Filters: squirrel.Like{"tag_value": "%e%"}, // Should match genres containing "e"
+ Sort: "name",
+ Order: "desc",
+ }
+ genres, err := repo.GetAll(options)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(genres).To(HaveLen(7))
+
+ Expect(slices.IsSortedFunc(genres, func(a, b model.Genre) int {
+ return strings.Compare(a.Name, b.Name)
+ }))
+ })
+ })
+ })
+
+ Describe("Count", func() {
+ It("should return correct count of genres", func() {
+ count, err := restRepo.Count()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(count).To(Equal(int64(12))) // We have 12 genre tags
+ })
+
+ It("should handle zero count", func() {
+ // Clear all genre tags
+ _, err := GetDBXBuilder().NewQuery("DELETE FROM tag WHERE tag_name = 'genre'").Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ count, err := restRepo.Count()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(count).To(BeZero())
+ })
+
+ It("should only count genre tags", func() {
+ // Add a non-genre tag
+ nonGenreTag := model.Tag{
+ ID: id.NewTagID("mood", "energetic"),
+ TagName: "mood",
+ TagValue: "energetic",
+ }
+ err := tagRepo.Add(1, nonGenreTag)
+ Expect(err).ToNot(HaveOccurred())
+
+ count, err := restRepo.Count()
+ Expect(err).ToNot(HaveOccurred())
+ // Count should not include the mood tag
+ Expect(count).To(Equal(int64(12))) // Should still be 12 genre tags
+ })
+
+ It("should filter by name using like match", func() {
+ // Test filtering by partial name match using the "name" filter which maps to containsFilter("tag_value")
+ options := rest.QueryOptions{
+ Filters: map[string]interface{}{"name": "%rock%"},
+ }
+ count, err := restRepo.Count(options)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(count).To(BeNumerically("==", 2))
+ })
+ })
+
+ Describe("Read", func() {
+ It("should return existing genre", func() {
+ // Use one of the existing genres from our consolidated dataset
+ genreID := id.NewTagID("genre", "rock")
+ result, err := restRepo.Read(genreID)
+ Expect(err).ToNot(HaveOccurred())
+ genre := result.(*model.Genre)
+ Expect(genre.ID).To(Equal(genreID))
+ Expect(genre.Name).To(Equal("rock"))
+ })
+
+ It("should return error for non-existent genre", func() {
+ _, err := restRepo.Read("non-existent-id")
+ Expect(err).To(HaveOccurred())
+ })
+
+ It("should not return non-genre tags", func() {
+ moodID := id.NewTagID("mood", "happy") // This exists as a mood tag, not genre
+ _, err := restRepo.Read(moodID)
+ Expect(err).To(HaveOccurred()) // Should not find it as a genre
+ })
+ })
+
+ Describe("ReadAll", func() {
+ It("should return all genres through ReadAll", func() {
+ result, err := restRepo.ReadAll()
+ Expect(err).ToNot(HaveOccurred())
+ genres := result.(model.Genres)
+ Expect(genres).To(HaveLen(12)) // We have 12 genre tags
+
+ genreNames := make([]string, len(genres))
+ for i, genre := range genres {
+ genreNames[i] = genre.Name
+ }
+ // Check for some of our consolidated dataset genres
+ Expect(genreNames).To(ContainElement("rock"))
+ Expect(genreNames).To(ContainElement("pop"))
+ Expect(genreNames).To(ContainElement("jazz"))
+ })
+
+ It("should support rest query options", func() {
+ result, err := restRepo.ReadAll()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(result).ToNot(BeNil())
+ })
+ })
+
+ Describe("Library Filtering", func() {
+ Context("Headless Processes (No User Context)", func() {
+ var headlessRepo model.GenreRepository
+ var headlessRestRepo model.ResourceRepository
+
+ BeforeEach(func() {
+ // Create a repository with no user context (headless)
+ headlessGenreRepo := NewGenreRepository(context.Background(), GetDBXBuilder())
+ headlessRepo = headlessGenreRepo
+ headlessRestRepo = headlessGenreRepo.(model.ResourceRepository)
+
+ // Add genres to different libraries
+ db := GetDBXBuilder()
+ _, err := db.NewQuery("INSERT OR IGNORE INTO library (id, name, path) VALUES (2, 'Test Library 2', '/test2')").Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Add tags to different libraries
+ newTag := func(name, value string) model.Tag {
+ return model.Tag{ID: id.NewTagID(name, value), TagName: model.TagName(name), TagValue: value}
+ }
+
+ err = tagRepo.Add(2, newTag("genre", "jazz"))
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ It("should see all genres from all libraries when no user is in context", func() {
+ // Headless processes should see all genres regardless of library
+ genres, err := headlessRepo.GetAll()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Should see genres from all libraries
+ var genreNames []string
+ for _, genre := range genres {
+ genreNames = append(genreNames, genre.Name)
+ }
+
+ // Should include both rock (library 1) and jazz (library 2)
+ Expect(genreNames).To(ContainElement("rock"))
+ Expect(genreNames).To(ContainElement("jazz"))
+ })
+
+ It("should count all genres from all libraries when no user is in context", func() {
+ count, err := headlessRestRepo.Count()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Should count all genres from all libraries
+ Expect(count).To(BeNumerically(">=", 2))
+ })
+
+ It("should allow headless processes to apply explicit library_id filters", func() {
+ // Filter by specific library
+ genres, err := headlessRestRepo.ReadAll(rest.QueryOptions{
+ Filters: map[string]interface{}{"library_id": 2},
+ })
+ Expect(err).ToNot(HaveOccurred())
+
+ genreList := genres.(model.Genres)
+ // Should see only genres from library 2
+ Expect(genreList).To(HaveLen(1))
+ Expect(genreList[0].Name).To(Equal("jazz"))
+ })
+
+ It("should get individual genres when no user is in context", func() {
+ // Get all genres first to find an ID
+ genres, err := headlessRepo.GetAll()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(genres).ToNot(BeEmpty())
+
+ // Headless process should be able to get the genre
+ genre, err := headlessRestRepo.Read(genres[0].ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(genre).ToNot(BeNil())
+ })
+ })
+ })
+
+ Describe("EntityName", func() {
+ It("should return correct entity name", func() {
+ name := restRepo.EntityName()
+ Expect(name).To(Equal("tag")) // Genre repository uses tag table
+ })
+ })
+
+ Describe("NewInstance", func() {
+ It("should return new genre instance", func() {
+ instance := restRepo.NewInstance()
+ Expect(instance).To(BeAssignableToTypeOf(&model.Genre{}))
+ })
+ })
+})
diff --git a/persistence/library_repository.go b/persistence/library_repository.go
index 5ec54b964..5621e1719 100644
--- a/persistence/library_repository.go
+++ b/persistence/library_repository.go
@@ -2,13 +2,17 @@ package persistence
import (
"context"
+ "fmt"
+ "strconv"
"sync"
"time"
. "github.com/Masterminds/squirrel"
+ "github.com/deluan/rest"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/utils/run"
"github.com/pocketbase/dbx"
)
@@ -67,41 +71,78 @@ func (r *libraryRepository) GetPath(id int) (string, error) {
}
func (r *libraryRepository) Put(l *model.Library) error {
- cols := map[string]any{
- "name": l.Name,
- "path": l.Path,
- "remote_path": l.RemotePath,
- "updated_at": time.Now(),
- }
- if l.ID != 0 {
- cols["id"] = l.ID
+ if l.ID == model.DefaultLibraryID {
+ currentLib, err := r.Get(1)
+ // if we are creating it, it's ok.
+ if err == nil { // it exists, so we are updating it
+ if currentLib.Path != l.Path {
+ return fmt.Errorf("%w: path for library with ID 1 cannot be changed", model.ErrValidation)
+ }
+ }
}
- sq := Insert(r.tableName).SetMap(cols).
- Suffix(`on conflict(id) do update set name = excluded.name, path = excluded.path,
- remote_path = excluded.remote_path, updated_at = excluded.updated_at`)
- _, err := r.executeSQL(sq)
+ var err error
+ l.UpdatedAt = time.Now()
+ if l.ID == 0 {
+ // Insert with autoassigned ID
+ l.CreatedAt = time.Now()
+ err = r.db.Model(l).Insert()
+ } else {
+ // Try to update first
+ cols := map[string]any{
+ "name": l.Name,
+ "path": l.Path,
+ "remote_path": l.RemotePath,
+ "default_new_users": l.DefaultNewUsers,
+ "updated_at": l.UpdatedAt,
+ }
+ sq := Update(r.tableName).SetMap(cols).Where(Eq{"id": l.ID})
+ rowsAffected, updateErr := r.executeSQL(sq)
+ if updateErr != nil {
+ return updateErr
+ }
+
+ // If no rows were affected, the record doesn't exist, so insert it
+ if rowsAffected == 0 {
+ l.CreatedAt = time.Now()
+ l.UpdatedAt = time.Now()
+ err = r.db.Model(l).Insert()
+ }
+ }
if err != nil {
- libLock.Lock()
- defer libLock.Unlock()
- libCache[l.ID] = l.Path
+ return err
}
- return err
-}
-const hardCodedMusicFolderID = 1
+ // Auto-assign all libraries to all admin users
+ sql := Expr(`
+INSERT INTO user_library (user_id, library_id)
+SELECT u.id, l.id
+FROM user u
+CROSS JOIN library l
+WHERE u.is_admin = true
+ON CONFLICT (user_id, library_id) DO NOTHING;`,
+ )
+ if _, err = r.executeSQL(sql); err != nil {
+ return fmt.Errorf("failed to assign library to admin users: %w", err)
+ }
+
+ libLock.Lock()
+ defer libLock.Unlock()
+ libCache[l.ID] = l.Path
+ return nil
+}
// TODO Remove this method when we have a proper UI to add libraries
// This is a temporary method to store the music folder path from the config in the DB
func (r *libraryRepository) StoreMusicFolder() error {
sq := Update(r.tableName).Set("path", conf.Server.MusicFolder).
Set("updated_at", time.Now()).
- Where(Eq{"id": hardCodedMusicFolderID})
+ Where(Eq{"id": model.DefaultLibraryID})
_, err := r.executeSQL(sq)
if err != nil {
libLock.Lock()
defer libLock.Unlock()
- libCache[hardCodedMusicFolderID] = conf.Server.MusicFolder
+ libCache[model.DefaultLibraryID] = conf.Server.MusicFolder
}
return err
}
@@ -136,7 +177,9 @@ func (r *libraryRepository) ScanEnd(id int) error {
return err
}
// https://www.sqlite.org/pragma.html#pragma_optimize
- _, err = r.executeSQL(Expr("PRAGMA optimize=0x10012;"))
+ // Use mask 0x10000 to check table sizes without running ANALYZE
+ // Running ANALYZE can cause query planner issues with expression-based collation indexes
+ _, err = r.executeSQL(Expr("PRAGMA optimize=0x10000;"))
return err
}
@@ -146,6 +189,84 @@ func (r *libraryRepository) ScanInProgress() (bool, error) {
return count > 0, err
}
+func (r *libraryRepository) RefreshStats(id int) error {
+ var songsRes, albumsRes, artistsRes, foldersRes, filesRes, missingRes struct{ Count int64 }
+ var sizeRes struct{ Sum int64 }
+ var durationRes struct{ Sum float64 }
+
+ err := run.Parallel(
+ func() error {
+ return r.queryOne(Select("count(*) as count").From("media_file").Where(Eq{"library_id": id, "missing": false}), &songsRes)
+ },
+ func() error {
+ return r.queryOne(Select("count(*) as count").From("album").Where(Eq{"library_id": id, "missing": false}), &albumsRes)
+ },
+ func() error {
+ return r.queryOne(Select("count(*) as count").From("library_artist la").
+ Join("artist a on la.artist_id = a.id").
+ Where(Eq{"la.library_id": id, "a.missing": false}), &artistsRes)
+ },
+ func() error {
+ return r.queryOne(Select("count(*) as count").From("folder").
+ Where(And{
+ Eq{"library_id": id, "missing": false},
+ Gt{"num_audio_files": 0},
+ }), &foldersRes)
+ },
+ func() error {
+ return r.queryOne(Select("ifnull(sum(num_audio_files + num_playlists + json_array_length(image_files)),0) as count").
+ From("folder").Where(Eq{"library_id": id, "missing": false}), &filesRes)
+ },
+ func() error {
+ return r.queryOne(Select("count(*) as count").From("media_file").Where(Eq{"library_id": id, "missing": true}), &missingRes)
+ },
+ func() error {
+ return r.queryOne(Select("ifnull(sum(size),0) as sum").From("album").Where(Eq{"library_id": id, "missing": false}), &sizeRes)
+ },
+ func() error {
+ return r.queryOne(Select("ifnull(sum(duration),0) as sum").From("album").Where(Eq{"library_id": id, "missing": false}), &durationRes)
+ },
+ )()
+ if err != nil {
+ return err
+ }
+
+ sq := Update(r.tableName).
+ Set("total_songs", songsRes.Count).
+ Set("total_albums", albumsRes.Count).
+ Set("total_artists", artistsRes.Count).
+ Set("total_folders", foldersRes.Count).
+ Set("total_files", filesRes.Count).
+ Set("total_missing_files", missingRes.Count).
+ Set("total_size", sizeRes.Sum).
+ Set("total_duration", durationRes.Sum).
+ Set("updated_at", time.Now()).
+ Where(Eq{"id": id})
+ _, err = r.executeSQL(sq)
+ return err
+}
+
+func (r *libraryRepository) Delete(id int) error {
+ if !loggedUser(r.ctx).IsAdmin {
+ return model.ErrNotAuthorized
+ }
+ if id == 1 {
+ return fmt.Errorf("%w: library with ID 1 cannot be deleted", model.ErrValidation)
+ }
+
+ err := r.delete(Eq{"id": id})
+ if err != nil {
+ return err
+ }
+
+ // Clear cache entry for this library only if DB operation was successful
+ libLock.Lock()
+ defer libLock.Unlock()
+ delete(libCache, id)
+
+ return nil
+}
+
func (r *libraryRepository) GetAll(ops ...model.QueryOptions) (model.Libraries, error) {
sq := r.newSelect(ops...).Columns("*")
res := model.Libraries{}
@@ -153,4 +274,72 @@ func (r *libraryRepository) GetAll(ops ...model.QueryOptions) (model.Libraries,
return res, err
}
+func (r *libraryRepository) CountAll(ops ...model.QueryOptions) (int64, error) {
+ sq := r.newSelect(ops...)
+ return r.count(sq)
+}
+
+// User-library association methods
+
+func (r *libraryRepository) GetUsersWithLibraryAccess(libraryID int) (model.Users, error) {
+ sel := Select("u.*").
+ From("user u").
+ Join("user_library ul ON u.id = ul.user_id").
+ Where(Eq{"ul.library_id": libraryID}).
+ OrderBy("u.name")
+
+ var res model.Users
+ err := r.queryAll(sel, &res)
+ return res, err
+}
+
+// REST interface methods
+
+func (r *libraryRepository) Count(options ...rest.QueryOptions) (int64, error) {
+ return r.CountAll(r.parseRestOptions(r.ctx, options...))
+}
+
+func (r *libraryRepository) Read(id string) (interface{}, error) {
+ idInt, err := strconv.Atoi(id)
+ if err != nil {
+ log.Trace(r.ctx, "invalid library id: %s", id, err)
+ return nil, rest.ErrNotFound
+ }
+ return r.Get(idInt)
+}
+
+func (r *libraryRepository) ReadAll(options ...rest.QueryOptions) (interface{}, error) {
+ return r.GetAll(r.parseRestOptions(r.ctx, options...))
+}
+
+func (r *libraryRepository) EntityName() string {
+ return "library"
+}
+
+func (r *libraryRepository) NewInstance() interface{} {
+ return &model.Library{}
+}
+
+func (r *libraryRepository) Save(entity interface{}) (string, error) {
+ lib := entity.(*model.Library)
+ lib.ID = 0 // Reset ID to ensure we create a new library
+ err := r.Put(lib)
+ if err != nil {
+ return "", err
+ }
+ return strconv.Itoa(lib.ID), nil
+}
+
+func (r *libraryRepository) Update(id string, entity interface{}, cols ...string) error {
+ lib := entity.(*model.Library)
+ idInt, err := strconv.Atoi(id)
+ if err != nil {
+ return fmt.Errorf("invalid library ID: %s", id)
+ }
+
+ lib.ID = idInt
+ return r.Put(lib)
+}
+
var _ model.LibraryRepository = (*libraryRepository)(nil)
+var _ rest.Repository = (*libraryRepository)(nil)
diff --git a/persistence/library_repository_test.go b/persistence/library_repository_test.go
new file mode 100644
index 000000000..3e3972bdb
--- /dev/null
+++ b/persistence/library_repository_test.go
@@ -0,0 +1,203 @@
+package persistence
+
+import (
+ "context"
+
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+ "github.com/pocketbase/dbx"
+)
+
+var _ = Describe("LibraryRepository", func() {
+ var repo model.LibraryRepository
+ var ctx context.Context
+ var conn *dbx.DB
+
+ BeforeEach(func() {
+ ctx = request.WithUser(log.NewContext(context.TODO()), model.User{ID: "userid"})
+ conn = GetDBXBuilder()
+ repo = NewLibraryRepository(ctx, conn)
+ })
+
+ AfterEach(func() {
+ // Clean up test libraries (keep ID 1 which is the default library)
+ _, _ = conn.NewQuery("DELETE FROM library WHERE id > 1").Execute()
+ })
+
+ Describe("Put", func() {
+ Context("when ID is 0", func() {
+ It("inserts a new library with autoassigned ID", func() {
+ lib := &model.Library{
+ ID: 0,
+ Name: "Test Library",
+ Path: "/music/test",
+ }
+
+ err := repo.Put(lib)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(lib.ID).To(BeNumerically(">", 0))
+ Expect(lib.CreatedAt).ToNot(BeZero())
+ Expect(lib.UpdatedAt).ToNot(BeZero())
+
+ // Verify it was inserted
+ savedLib, err := repo.Get(lib.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(savedLib.Name).To(Equal("Test Library"))
+ Expect(savedLib.Path).To(Equal("/music/test"))
+ })
+ })
+
+ Context("when ID is non-zero and record exists", func() {
+ It("updates the existing record", func() {
+ // First create a library
+ lib := &model.Library{
+ ID: 0,
+ Name: "Original Library",
+ Path: "/music/original",
+ }
+ err := repo.Put(lib)
+ Expect(err).ToNot(HaveOccurred())
+
+ originalID := lib.ID
+ originalCreatedAt := lib.CreatedAt
+
+ // Now update it
+ lib.Name = "Updated Library"
+ lib.Path = "/music/updated"
+ err = repo.Put(lib)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify it was updated, not inserted
+ Expect(lib.ID).To(Equal(originalID))
+ Expect(lib.CreatedAt).To(Equal(originalCreatedAt))
+ Expect(lib.UpdatedAt).To(BeTemporally(">", originalCreatedAt))
+
+ // Verify the changes were saved
+ savedLib, err := repo.Get(lib.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(savedLib.Name).To(Equal("Updated Library"))
+ Expect(savedLib.Path).To(Equal("/music/updated"))
+ })
+ })
+
+ Context("when ID is non-zero but record doesn't exist", func() {
+ It("inserts a new record with the specified ID", func() {
+ lib := &model.Library{
+ ID: 999,
+ Name: "New Library with ID",
+ Path: "/music/new",
+ }
+
+ // Ensure the record doesn't exist
+ _, err := repo.Get(999)
+ Expect(err).To(HaveOccurred())
+
+ // Put should insert it
+ err = repo.Put(lib)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(lib.ID).To(Equal(999))
+ Expect(lib.CreatedAt).ToNot(BeZero())
+ Expect(lib.UpdatedAt).ToNot(BeZero())
+
+ // Verify it was inserted with the correct ID
+ savedLib, err := repo.Get(999)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(savedLib.ID).To(Equal(999))
+ Expect(savedLib.Name).To(Equal("New Library with ID"))
+ Expect(savedLib.Path).To(Equal("/music/new"))
+ })
+ })
+ })
+
+ It("refreshes stats", func() {
+ libBefore, err := repo.Get(1)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(repo.RefreshStats(1)).To(Succeed())
+ libAfter, err := repo.Get(1)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(libAfter.UpdatedAt).To(BeTemporally(">", libBefore.UpdatedAt))
+
+ var songsRes, albumsRes, artistsRes, foldersRes, filesRes, missingRes struct{ Count int64 }
+ var sizeRes struct{ Sum int64 }
+ var durationRes struct{ Sum float64 }
+
+ Expect(conn.NewQuery("select count(*) as count from media_file where library_id = {:id} and missing = 0").Bind(dbx.Params{"id": 1}).One(&songsRes)).To(Succeed())
+ Expect(conn.NewQuery("select count(*) as count from album where library_id = {:id} and missing = 0").Bind(dbx.Params{"id": 1}).One(&albumsRes)).To(Succeed())
+ Expect(conn.NewQuery("select count(*) as count from library_artist la join artist a on la.artist_id = a.id where la.library_id = {:id} and a.missing = 0").Bind(dbx.Params{"id": 1}).One(&artistsRes)).To(Succeed())
+ Expect(conn.NewQuery("select count(*) as count from folder where library_id = {:id} and missing = 0").Bind(dbx.Params{"id": 1}).One(&foldersRes)).To(Succeed())
+ Expect(conn.NewQuery("select ifnull(sum(num_audio_files + num_playlists + json_array_length(image_files)),0) as count from folder where library_id = {:id} and missing = 0").Bind(dbx.Params{"id": 1}).One(&filesRes)).To(Succeed())
+ Expect(conn.NewQuery("select count(*) as count from media_file where library_id = {:id} and missing = 1").Bind(dbx.Params{"id": 1}).One(&missingRes)).To(Succeed())
+ Expect(conn.NewQuery("select ifnull(sum(size),0) as sum from album where library_id = {:id} and missing = 0").Bind(dbx.Params{"id": 1}).One(&sizeRes)).To(Succeed())
+ Expect(conn.NewQuery("select ifnull(sum(duration),0) as sum from album where library_id = {:id} and missing = 0").Bind(dbx.Params{"id": 1}).One(&durationRes)).To(Succeed())
+
+ Expect(libAfter.TotalSongs).To(Equal(int(songsRes.Count)))
+ Expect(libAfter.TotalAlbums).To(Equal(int(albumsRes.Count)))
+ Expect(libAfter.TotalArtists).To(Equal(int(artistsRes.Count)))
+ Expect(libAfter.TotalFolders).To(Equal(int(foldersRes.Count)))
+ Expect(libAfter.TotalFiles).To(Equal(int(filesRes.Count)))
+ Expect(libAfter.TotalMissingFiles).To(Equal(int(missingRes.Count)))
+ Expect(libAfter.TotalSize).To(Equal(sizeRes.Sum))
+ Expect(libAfter.TotalDuration).To(Equal(durationRes.Sum))
+ })
+
+ Describe("ScanBegin and ScanEnd", func() {
+ var lib *model.Library
+
+ BeforeEach(func() {
+ lib = &model.Library{
+ ID: 0,
+ Name: "Test Scan Library",
+ Path: "/music/test-scan",
+ }
+ err := repo.Put(lib)
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ DescribeTable("ScanBegin",
+ func(fullScan bool, expectedFullScanInProgress bool) {
+ err := repo.ScanBegin(lib.ID, fullScan)
+ Expect(err).ToNot(HaveOccurred())
+
+ updatedLib, err := repo.Get(lib.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(updatedLib.LastScanStartedAt).ToNot(BeZero())
+ Expect(updatedLib.FullScanInProgress).To(Equal(expectedFullScanInProgress))
+ },
+ Entry("sets FullScanInProgress to true for full scan", true, true),
+ Entry("sets FullScanInProgress to false for quick scan", false, false),
+ )
+
+ Context("ScanEnd", func() {
+ BeforeEach(func() {
+ err := repo.ScanBegin(lib.ID, true)
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ It("sets LastScanAt and clears FullScanInProgress and LastScanStartedAt", func() {
+ err := repo.ScanEnd(lib.ID)
+ Expect(err).ToNot(HaveOccurred())
+
+ updatedLib, err := repo.Get(lib.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(updatedLib.LastScanAt).ToNot(BeZero())
+ Expect(updatedLib.FullScanInProgress).To(BeFalse())
+ Expect(updatedLib.LastScanStartedAt).To(BeZero())
+ })
+
+ It("sets LastScanAt to be after LastScanStartedAt", func() {
+ libBefore, err := repo.Get(lib.ID)
+ Expect(err).ToNot(HaveOccurred())
+
+ err = repo.ScanEnd(lib.ID)
+ Expect(err).ToNot(HaveOccurred())
+
+ libAfter, err := repo.Get(lib.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(libAfter.LastScanAt).To(BeTemporally(">=", libBefore.LastScanStartedAt))
+ })
+ })
+ })
+})
diff --git a/persistence/mediafile_repository.go b/persistence/mediafile_repository.go
index b0ed637c1..e7883947a 100644
--- a/persistence/mediafile_repository.go
+++ b/persistence/mediafile_repository.go
@@ -9,6 +9,8 @@ import (
. "github.com/Masterminds/squirrel"
"github.com/deluan/rest"
+ "github.com/google/uuid"
+ "github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/utils/slice"
@@ -25,10 +27,10 @@ type dbMediaFile struct {
Tags string `structs:"-" json:"-"`
// These are necessary to map the correct names (rg_*) to the correct fields (RG*)
// without using `db` struct tags in the model.MediaFile struct
- RgAlbumGain float64 `structs:"-" json:"-"`
- RgAlbumPeak float64 `structs:"-" json:"-"`
- RgTrackGain float64 `structs:"-" json:"-"`
- RgTrackPeak float64 `structs:"-" json:"-"`
+ RgAlbumGain *float64 `structs:"-" json:"-"`
+ RgAlbumPeak *float64 `structs:"-" json:"-"`
+ RgTrackGain *float64 `structs:"-" json:"-"`
+ RgTrackPeak *float64 `structs:"-" json:"-"`
}
func (m *dbMediaFile) PostScan() error {
@@ -74,13 +76,14 @@ func NewMediaFileRepository(ctx context.Context, db dbx.Builder) model.MediaFile
r.tableName = "media_file"
r.registerModel(&model.MediaFile{}, mediaFileFilter())
r.setSortMappings(map[string]string{
- "title": "order_title",
- "artist": "order_artist_name, order_album_name, release_date, disc_number, track_number",
- "album_artist": "order_album_artist_name, order_album_name, release_date, disc_number, track_number",
- "album": "order_album_name, album_id, disc_number, track_number, order_artist_name, title",
- "random": "random",
- "created_at": "media_file.created_at",
- "starred_at": "starred, starred_at",
+ "title": "order_title",
+ "artist": "order_artist_name, order_album_name, release_date, disc_number, track_number",
+ "album_artist": "order_album_artist_name, order_album_name, release_date, disc_number, track_number",
+ "album": "order_album_name, album_id, disc_number, track_number, order_artist_name, title",
+ "random": "random",
+ "created_at": "media_file.created_at",
+ "recently_added": mediaFileRecentlyAddedSort(),
+ "starred_at": "starred, starred_at",
})
return r
}
@@ -88,11 +91,12 @@ func NewMediaFileRepository(ctx context.Context, db dbx.Builder) model.MediaFile
var mediaFileFilter = sync.OnceValue(func() map[string]filterFunc {
filters := map[string]filterFunc{
"id": idFilter("media_file"),
- "title": fullTextFilter("media_file"),
+ "title": fullTextFilter("media_file", "mbz_recording_id", "mbz_release_track_id"),
"starred": booleanFilter,
"genre_id": tagIDFilter,
"missing": booleanFilter,
"artists_id": artistFilter,
+ "library_id": libraryIdFilter,
}
// Add all album tags as filters
for tag := range model.TagMappings() {
@@ -103,9 +107,17 @@ var mediaFileFilter = sync.OnceValue(func() map[string]filterFunc {
return filters
})
+func mediaFileRecentlyAddedSort() string {
+ if conf.Server.RecentlyAddedByModTime {
+ return "media_file.updated_at"
+ }
+ return "media_file.created_at"
+}
+
func (r *mediaFileRepository) CountAll(options ...model.QueryOptions) (int64, error) {
query := r.newSelect()
query = r.withAnnotation(query, "media_file.id")
+ query = r.applyLibraryFilter(query)
return r.count(query, options...)
}
@@ -124,10 +136,11 @@ func (r *mediaFileRepository) Put(m *model.MediaFile) error {
}
func (r *mediaFileRepository) selectMediaFile(options ...model.QueryOptions) SelectBuilder {
- sql := r.newSelect(options...).Columns("media_file.*", "library.path as library_path").
+ sql := r.newSelect(options...).Columns("media_file.*", "library.path as library_path", "library.name as library_name").
LeftJoin("library on media_file.library_id = library.id")
sql = r.withAnnotation(sql, "media_file.id")
- return r.withBookmark(sql, "media_file.id")
+ sql = r.withBookmark(sql, "media_file.id")
+ return r.applyLibraryFilter(sql)
}
func (r *mediaFileRepository) Get(id string) (*model.MediaFile, error) {
@@ -263,7 +276,7 @@ func (r *mediaFileRepository) GetMissingAndMatching(libId int) (model.MediaFileC
if err != nil {
return nil, err
}
- sel := r.newSelect().Columns("media_file.*", "library.path as library_path").
+ sel := r.newSelect().Columns("media_file.*", "library.path as library_path", "library.name as library_name").
LeftJoin("library on media_file.library_id = library.id").
Where("pid in ("+subQText+")", subQArgs...).
Where(Or{
@@ -284,13 +297,62 @@ func (r *mediaFileRepository) GetMissingAndMatching(libId int) (model.MediaFileC
}, nil
}
-func (r *mediaFileRepository) Search(q string, offset int, size int, includeMissing bool) (model.MediaFiles, error) {
- results := dbMediaFiles{}
- err := r.doSearch(r.selectMediaFile(), q, offset, size, includeMissing, &results, "title")
+// FindRecentFilesByMBZTrackID finds recently added files by MusicBrainz Track ID in other libraries
+func (r *mediaFileRepository) FindRecentFilesByMBZTrackID(missing model.MediaFile, since time.Time) (model.MediaFiles, error) {
+ sel := r.selectMediaFile().Where(And{
+ NotEq{"media_file.library_id": missing.LibraryID},
+ Eq{"media_file.mbz_release_track_id": missing.MbzReleaseTrackID},
+ NotEq{"media_file.mbz_release_track_id": ""}, // Exclude empty MBZ Track IDs
+ Eq{"media_file.suffix": missing.Suffix},
+ Gt{"media_file.created_at": since},
+ Eq{"media_file.missing": false},
+ }).OrderBy("media_file.created_at DESC")
+
+ var res dbMediaFiles
+ err := r.queryAll(sel, &res)
if err != nil {
return nil, err
}
- return results.toModels(), err
+ return res.toModels(), nil
+}
+
+// FindRecentFilesByProperties finds recently added files by intrinsic properties in other libraries
+func (r *mediaFileRepository) FindRecentFilesByProperties(missing model.MediaFile, since time.Time) (model.MediaFiles, error) {
+ sel := r.selectMediaFile().Where(And{
+ NotEq{"media_file.library_id": missing.LibraryID},
+ Eq{"media_file.title": missing.Title},
+ Eq{"media_file.size": missing.Size},
+ Eq{"media_file.suffix": missing.Suffix},
+ Eq{"media_file.disc_number": missing.DiscNumber},
+ Eq{"media_file.track_number": missing.TrackNumber},
+ Eq{"media_file.album": missing.Album},
+ Eq{"media_file.mbz_release_track_id": ""}, // Exclude files with MBZ Track ID
+ Gt{"media_file.created_at": since},
+ Eq{"media_file.missing": false},
+ }).OrderBy("media_file.created_at DESC")
+
+ var res dbMediaFiles
+ err := r.queryAll(sel, &res)
+ if err != nil {
+ return nil, err
+ }
+ return res.toModels(), nil
+}
+
+func (r *mediaFileRepository) Search(q string, offset int, size int, options ...model.QueryOptions) (model.MediaFiles, error) {
+ var res dbMediaFiles
+ if uuid.Validate(q) == nil {
+ err := r.searchByMBID(r.selectMediaFile(options...), q, []string{"mbz_recording_id", "mbz_release_track_id"}, &res)
+ if err != nil {
+ return nil, fmt.Errorf("searching media_file by MBID %q: %w", q, err)
+ }
+ } else {
+ err := r.doSearch(r.selectMediaFile(options...), q, offset, size, &res, "media_file.rowid", "title")
+ if err != nil {
+ return nil, fmt.Errorf("searching media_file by query %q: %w", q, err)
+ }
+ }
+ return res.toModels(), nil
}
func (r *mediaFileRepository) Count(options ...rest.QueryOptions) (int64, error) {
diff --git a/persistence/mediafile_repository_test.go b/persistence/mediafile_repository_test.go
index c17bc595b..ab926c00d 100644
--- a/persistence/mediafile_repository_test.go
+++ b/persistence/mediafile_repository_test.go
@@ -5,12 +5,15 @@ import (
"time"
"github.com/Masterminds/squirrel"
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/model/id"
"github.com/navidrome/navidrome/model/request"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
+ "github.com/pocketbase/dbx"
)
var _ = Describe("MediaRepository", func() {
@@ -35,7 +38,31 @@ var _ = Describe("MediaRepository", func() {
})
It("counts the number of mediafiles in the DB", func() {
- Expect(mr.CountAll()).To(Equal(int64(4)))
+ Expect(mr.CountAll()).To(Equal(int64(10)))
+ })
+
+ It("returns songs ordered by lyrics with a specific title/artist", func() {
+ // attempt to mimic filters.SongsByArtistTitleWithLyricsFirst, except we want all items
+ results, err := mr.GetAll(model.QueryOptions{
+ Sort: "lyrics, updated_at",
+ Order: "desc",
+ Filters: squirrel.And{
+ squirrel.Eq{"title": "Antenna"},
+ squirrel.Or{
+ Exists("json_tree(participants, '$.albumartist')", squirrel.Eq{"value": "Kraftwerk"}),
+ Exists("json_tree(participants, '$.artist')", squirrel.Eq{"value": "Kraftwerk"}),
+ },
+ },
+ })
+
+ Expect(err).To(BeNil())
+ Expect(results).To(HaveLen(3))
+ Expect(results[0].Lyrics).To(Equal(`[{"lang":"xxx","line":[{"value":"This is a set of lyrics"}],"synced":false}]`))
+ for _, item := range results[1:] {
+ Expect(item.Lyrics).To(Equal("[]"))
+ Expect(item.Title).To(Equal("Antenna"))
+ Expect(item.Participants[model.RoleArtist][0].Name).To(Equal("Kraftwerk"))
+ }
})
It("checks existence of mediafiles in the DB", func() {
@@ -131,4 +158,256 @@ var _ = Describe("MediaRepository", func() {
Expect(mf.PlayCount).To(Equal(int64(1)))
})
})
+
+ Context("Sort options", func() {
+ Context("recently_added sort", func() {
+ var testMediaFiles []model.MediaFile
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+
+ // Create test media files with specific timestamps
+ testMediaFiles = []model.MediaFile{
+ {
+ ID: id.NewRandom(),
+ LibraryID: 1,
+ Title: "Old Song",
+ Path: "/test/old.mp3",
+ },
+ {
+ ID: id.NewRandom(),
+ LibraryID: 1,
+ Title: "Middle Song",
+ Path: "/test/middle.mp3",
+ },
+ {
+ ID: id.NewRandom(),
+ LibraryID: 1,
+ Title: "New Song",
+ Path: "/test/new.mp3",
+ },
+ }
+
+ // Insert test data first
+ for i := range testMediaFiles {
+ Expect(mr.Put(&testMediaFiles[i])).To(Succeed())
+ }
+
+ // Then manually update timestamps using direct SQL to bypass the repository logic
+ db := GetDBXBuilder()
+
+ // Set specific timestamps for testing
+ oldTime := time.Date(2020, 1, 1, 0, 0, 0, 0, time.UTC)
+ middleTime := time.Date(2021, 1, 1, 0, 0, 0, 0, time.UTC)
+ newTime := time.Date(2022, 1, 1, 0, 0, 0, 0, time.UTC)
+
+ // Update "Old Song": created long ago, updated recently
+ _, err := db.Update("media_file",
+ map[string]interface{}{
+ "created_at": oldTime,
+ "updated_at": newTime,
+ },
+ dbx.HashExp{"id": testMediaFiles[0].ID}).Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Update "Middle Song": created and updated at the same middle time
+ _, err = db.Update("media_file",
+ map[string]interface{}{
+ "created_at": middleTime,
+ "updated_at": middleTime,
+ },
+ dbx.HashExp{"id": testMediaFiles[1].ID}).Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Update "New Song": created recently, updated long ago
+ _, err = db.Update("media_file",
+ map[string]interface{}{
+ "created_at": newTime,
+ "updated_at": oldTime,
+ },
+ dbx.HashExp{"id": testMediaFiles[2].ID}).Execute()
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ AfterEach(func() {
+ // Clean up test data
+ for _, mf := range testMediaFiles {
+ _ = mr.Delete(mf.ID)
+ }
+ })
+
+ When("RecentlyAddedByModTime is false", func() {
+ var testRepo model.MediaFileRepository
+
+ BeforeEach(func() {
+ conf.Server.RecentlyAddedByModTime = false
+ // Create repository AFTER setting config
+ ctx := log.NewContext(GinkgoT().Context())
+ ctx = request.WithUser(ctx, model.User{ID: "userid"})
+ testRepo = NewMediaFileRepository(ctx, GetDBXBuilder())
+ })
+
+ It("sorts by created_at", func() {
+ // Get results sorted by recently_added (should use created_at)
+ results, err := testRepo.GetAll(model.QueryOptions{
+ Sort: "recently_added",
+ Order: "desc",
+ Filters: squirrel.Eq{"media_file.id": []string{testMediaFiles[0].ID, testMediaFiles[1].ID, testMediaFiles[2].ID}},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(3))
+
+ // Verify sorting by created_at (newest first in descending order)
+ Expect(results[0].Title).To(Equal("New Song")) // created 2022
+ Expect(results[1].Title).To(Equal("Middle Song")) // created 2021
+ Expect(results[2].Title).To(Equal("Old Song")) // created 2020
+ })
+
+ It("sorts in ascending order when specified", func() {
+ // Get results sorted by recently_added in ascending order
+ results, err := testRepo.GetAll(model.QueryOptions{
+ Sort: "recently_added",
+ Order: "asc",
+ Filters: squirrel.Eq{"media_file.id": []string{testMediaFiles[0].ID, testMediaFiles[1].ID, testMediaFiles[2].ID}},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(3))
+
+ // Verify sorting by created_at (oldest first)
+ Expect(results[0].Title).To(Equal("Old Song")) // created 2020
+ Expect(results[1].Title).To(Equal("Middle Song")) // created 2021
+ Expect(results[2].Title).To(Equal("New Song")) // created 2022
+ })
+ })
+
+ When("RecentlyAddedByModTime is true", func() {
+ var testRepo model.MediaFileRepository
+
+ BeforeEach(func() {
+ conf.Server.RecentlyAddedByModTime = true
+ // Create repository AFTER setting config
+ ctx := log.NewContext(GinkgoT().Context())
+ ctx = request.WithUser(ctx, model.User{ID: "userid"})
+ testRepo = NewMediaFileRepository(ctx, GetDBXBuilder())
+ })
+
+ It("sorts by updated_at", func() {
+ // Get results sorted by recently_added (should use updated_at)
+ results, err := testRepo.GetAll(model.QueryOptions{
+ Sort: "recently_added",
+ Order: "desc",
+ Filters: squirrel.Eq{"media_file.id": []string{testMediaFiles[0].ID, testMediaFiles[1].ID, testMediaFiles[2].ID}},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(3))
+
+ // Verify sorting by updated_at (newest first in descending order)
+ Expect(results[0].Title).To(Equal("Old Song")) // updated 2022
+ Expect(results[1].Title).To(Equal("Middle Song")) // updated 2021
+ Expect(results[2].Title).To(Equal("New Song")) // updated 2020
+ })
+ })
+
+ })
+ })
+
+ Describe("Search", func() {
+ Context("text search", func() {
+ It("finds media files by title", func() {
+ results, err := mr.Search("Antenna", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(3)) // songAntenna, songAntennaWithLyrics, songAntenna2
+ for _, result := range results {
+ Expect(result.Title).To(Equal("Antenna"))
+ }
+ })
+
+ It("finds media files case insensitively", func() {
+ results, err := mr.Search("antenna", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(3))
+ for _, result := range results {
+ Expect(result.Title).To(Equal("Antenna"))
+ }
+ })
+
+ It("returns empty result when no matches found", func() {
+ results, err := mr.Search("nonexistent", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(BeEmpty())
+ })
+ })
+
+ Context("MBID search", func() {
+ var mediaFileWithMBID model.MediaFile
+ var raw *mediaFileRepository
+
+ BeforeEach(func() {
+ raw = mr.(*mediaFileRepository)
+ // Create a test media file with MBID
+ mediaFileWithMBID = model.MediaFile{
+ ID: "test-mbid-mediafile",
+ Title: "Test MBID MediaFile",
+ MbzRecordingID: "550e8400-e29b-41d4-a716-446655440020", // Valid UUID v4
+ MbzReleaseTrackID: "550e8400-e29b-41d4-a716-446655440021", // Valid UUID v4
+ LibraryID: 1,
+ Path: "/test/path/test.mp3",
+ }
+
+ // Insert the test media file into the database
+ err := mr.Put(&mediaFileWithMBID)
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ AfterEach(func() {
+ // Clean up test data using direct SQL
+ _, _ = raw.executeSQL(squirrel.Delete(raw.tableName).Where(squirrel.Eq{"id": mediaFileWithMBID.ID}))
+ })
+
+ It("finds media file by mbz_recording_id", func() {
+ results, err := mr.Search("550e8400-e29b-41d4-a716-446655440020", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(1))
+ Expect(results[0].ID).To(Equal("test-mbid-mediafile"))
+ Expect(results[0].Title).To(Equal("Test MBID MediaFile"))
+ })
+
+ It("finds media file by mbz_release_track_id", func() {
+ results, err := mr.Search("550e8400-e29b-41d4-a716-446655440021", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(HaveLen(1))
+ Expect(results[0].ID).To(Equal("test-mbid-mediafile"))
+ Expect(results[0].Title).To(Equal("Test MBID MediaFile"))
+ })
+
+ It("returns empty result when MBID is not found", func() {
+ results, err := mr.Search("550e8400-e29b-41d4-a716-446655440099", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(BeEmpty())
+ })
+
+ It("missing media files are never returned by search", func() {
+ // Create a missing media file with MBID
+ missingMediaFile := model.MediaFile{
+ ID: "test-missing-mbid-mediafile",
+ Title: "Test Missing MBID MediaFile",
+ MbzRecordingID: "550e8400-e29b-41d4-a716-446655440022",
+ LibraryID: 1,
+ Path: "/test/path/missing.mp3",
+ Missing: true,
+ }
+
+ err := mr.Put(&missingMediaFile)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Search never returns missing media files (hardcoded behavior)
+ results, err := mr.Search("550e8400-e29b-41d4-a716-446655440022", 0, 10)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(results).To(BeEmpty())
+
+ // Clean up
+ _, _ = raw.executeSQL(squirrel.Delete(raw.tableName).Where(squirrel.Eq{"id": missingMediaFile.ID}))
+ })
+ })
+ })
})
diff --git a/persistence/persistence.go b/persistence/persistence.go
index 2536b9c35..1de0bae61 100644
--- a/persistence/persistence.go
+++ b/persistence/persistence.go
@@ -9,7 +9,7 @@ import (
"github.com/navidrome/navidrome/db"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
- "github.com/navidrome/navidrome/utils/chain"
+ "github.com/navidrome/navidrome/utils/run"
"github.com/pocketbase/dbx"
)
@@ -157,7 +157,7 @@ func (s *SQLStore) WithTxImmediate(block func(tx model.DataStore) error, scope .
}, scope...)
}
-func (s *SQLStore) GC(ctx context.Context) error {
+func (s *SQLStore) GC(ctx context.Context, libraryIDs ...int) error {
trace := func(ctx context.Context, msg string, f func() error) func() error {
return func() error {
start := time.Now()
@@ -167,11 +167,17 @@ func (s *SQLStore) GC(ctx context.Context) error {
}
}
- err := chain.RunSequentially(
- trace(ctx, "purge empty albums", func() error { return s.Album(ctx).(*albumRepository).purgeEmpty() }),
+ // If libraryIDs are provided, scope operations to those libraries where possible
+ scoped := len(libraryIDs) > 0
+ if scoped {
+ log.Debug(ctx, "GC: Running selective garbage collection", "libraryIDs", libraryIDs)
+ }
+
+ err := run.Sequentially(
+ trace(ctx, "purge empty albums", func() error { return s.Album(ctx).(*albumRepository).purgeEmpty(libraryIDs...) }),
trace(ctx, "purge empty artists", func() error { return s.Artist(ctx).(*artistRepository).purgeEmpty() }),
trace(ctx, "mark missing artists", func() error { return s.Artist(ctx).(*artistRepository).markMissing() }),
- trace(ctx, "purge empty folders", func() error { return s.Folder(ctx).(*folderRepository).purgeEmpty() }),
+ trace(ctx, "purge empty folders", func() error { return s.Folder(ctx).(*folderRepository).purgeEmpty(libraryIDs...) }),
trace(ctx, "clean album annotations", func() error { return s.Album(ctx).(*albumRepository).cleanAnnotations() }),
trace(ctx, "clean artist annotations", func() error { return s.Artist(ctx).(*artistRepository).cleanAnnotations() }),
trace(ctx, "clean media file annotations", func() error { return s.MediaFile(ctx).(*mediaFileRepository).cleanAnnotations() }),
diff --git a/persistence/persistence_suite_test.go b/persistence/persistence_suite_test.go
index 43e4c292b..f3cb4f3d0 100644
--- a/persistence/persistence_suite_test.go
+++ b/persistence/persistence_suite_test.go
@@ -12,6 +12,7 @@ import (
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/model/request"
"github.com/navidrome/navidrome/tests"
+ "github.com/navidrome/navidrome/utils/gg"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
"github.com/pocketbase/dbx"
@@ -33,16 +34,22 @@ func mf(mf model.MediaFile) model.MediaFile {
mf.Tags = model.Tags{}
mf.LibraryID = 1
mf.LibraryPath = "music" // Default folder
+ mf.LibraryName = "Music Library"
mf.Participants = model.Participants{
model.RoleArtist: model.ParticipantList{
model.Participant{Artist: model.Artist{ID: mf.ArtistID, Name: mf.Artist}},
},
}
+ if mf.Lyrics == "" {
+ mf.Lyrics = "[]"
+ }
return mf
}
func al(al model.Album) model.Album {
al.LibraryID = 1
+ al.LibraryPath = "music"
+ al.LibraryName = "Music Library"
al.Discs = model.Discs{}
al.Tags = model.Tags{}
al.Participants = model.Participants{}
@@ -62,10 +69,12 @@ var (
albumSgtPeppers = al(model.Album{ID: "101", Name: "Sgt Peppers", AlbumArtist: "The Beatles", OrderAlbumName: "sgt peppers", AlbumArtistID: "3", EmbedArtPath: p("/beatles/1/sgt/a day.mp3"), SongCount: 1, MaxYear: 1967})
albumAbbeyRoad = al(model.Album{ID: "102", Name: "Abbey Road", AlbumArtist: "The Beatles", OrderAlbumName: "abbey road", AlbumArtistID: "3", EmbedArtPath: p("/beatles/1/come together.mp3"), SongCount: 1, MaxYear: 1969})
albumRadioactivity = al(model.Album{ID: "103", Name: "Radioactivity", AlbumArtist: "Kraftwerk", OrderAlbumName: "radioactivity", AlbumArtistID: "2", EmbedArtPath: p("/kraft/radio/radio.mp3"), SongCount: 2})
+ albumMultiDisc = al(model.Album{ID: "104", Name: "Multi Disc Album", AlbumArtist: "Test Artist", OrderAlbumName: "multi disc album", AlbumArtistID: "1", EmbedArtPath: p("/test/multi/disc1/track1.mp3"), SongCount: 4})
testAlbums = model.Albums{
albumSgtPeppers,
albumAbbeyRoad,
albumRadioactivity,
+ albumMultiDisc,
}
)
@@ -76,13 +85,33 @@ var (
songAntenna = mf(model.MediaFile{ID: "1004", Title: "Antenna", ArtistID: "2", Artist: "Kraftwerk",
AlbumID: "103",
Path: p("/kraft/radio/antenna.mp3"),
- RGAlbumGain: 1.0, RGAlbumPeak: 2.0, RGTrackGain: 3.0, RGTrackPeak: 4.0,
+ RGAlbumGain: gg.P(1.0), RGAlbumPeak: gg.P(2.0), RGTrackGain: gg.P(3.0), RGTrackPeak: gg.P(4.0),
})
- testSongs = model.MediaFiles{
+ songAntennaWithLyrics = mf(model.MediaFile{
+ ID: "1005",
+ Title: "Antenna",
+ ArtistID: "2",
+ Artist: "Kraftwerk",
+ AlbumID: "103",
+ Lyrics: `[{"lang":"xxx","line":[{"value":"This is a set of lyrics"}],"synced":false}]`,
+ })
+ songAntenna2 = mf(model.MediaFile{ID: "1006", Title: "Antenna", ArtistID: "2", Artist: "Kraftwerk", AlbumID: "103"})
+ // Multi-disc album tracks (intentionally out of order to test sorting)
+ songDisc2Track11 = mf(model.MediaFile{ID: "2001", Title: "Disc 2 Track 11", ArtistID: "1", Artist: "Test Artist", AlbumID: "104", Album: "Multi Disc Album", DiscNumber: 2, TrackNumber: 11, Path: p("/test/multi/disc2/track11.mp3"), OrderAlbumName: "multi disc album", OrderArtistName: "test artist"})
+ songDisc1Track01 = mf(model.MediaFile{ID: "2002", Title: "Disc 1 Track 1", ArtistID: "1", Artist: "Test Artist", AlbumID: "104", Album: "Multi Disc Album", DiscNumber: 1, TrackNumber: 1, Path: p("/test/multi/disc1/track1.mp3"), OrderAlbumName: "multi disc album", OrderArtistName: "test artist"})
+ songDisc2Track01 = mf(model.MediaFile{ID: "2003", Title: "Disc 2 Track 1", ArtistID: "1", Artist: "Test Artist", AlbumID: "104", Album: "Multi Disc Album", DiscNumber: 2, TrackNumber: 1, Path: p("/test/multi/disc2/track1.mp3"), OrderAlbumName: "multi disc album", OrderArtistName: "test artist"})
+ songDisc1Track02 = mf(model.MediaFile{ID: "2004", Title: "Disc 1 Track 2", ArtistID: "1", Artist: "Test Artist", AlbumID: "104", Album: "Multi Disc Album", DiscNumber: 1, TrackNumber: 2, Path: p("/test/multi/disc1/track2.mp3"), OrderAlbumName: "multi disc album", OrderArtistName: "test artist"})
+ testSongs = model.MediaFiles{
songDayInALife,
songComeTogether,
songRadioactivity,
songAntenna,
+ songAntennaWithLyrics,
+ songAntenna2,
+ songDisc2Track11,
+ songDisc1Track01,
+ songDisc2Track01,
+ songDisc1Track02,
}
)
@@ -123,14 +152,13 @@ var _ = BeforeSuite(func() {
}
}
- //gr := NewGenreRepository(ctx, conn)
- //for i := range testGenres {
- // g := testGenres[i]
- // err := gr.Put(&g)
- // if err != nil {
- // panic(err)
- // }
- //}
+ // Associate users with library 1 (default test library)
+ for i := range testUsers {
+ err := ur.SetUserLibraries(testUsers[i].ID, []int{1})
+ if err != nil {
+ panic(err)
+ }
+ }
alr := NewAlbumRepository(ctx, conn).(*albumRepository)
for i := range testAlbums {
@@ -150,6 +178,15 @@ var _ = BeforeSuite(func() {
}
}
+ // Associate artists with library 1 (default test library)
+ lr := NewLibraryRepository(ctx, conn)
+ for i := range testArtists {
+ err := lr.AddArtist(1, testArtists[i].ID)
+ if err != nil {
+ panic(err)
+ }
+ }
+
mr := NewMediaFileRepository(ctx, conn)
for i := range testSongs {
err := mr.Put(&testSongs[i])
@@ -175,9 +212,9 @@ var _ = BeforeSuite(func() {
Public: true,
SongCount: 2,
}
- plsBest.AddTracks([]string{"1001", "1003"})
+ plsBest.AddMediaFilesByID([]string{"1001", "1003"})
plsCool = model.Playlist{Name: "Cool", OwnerID: "userid", OwnerName: "userid"}
- plsCool.AddTracks([]string{"1004"})
+ plsCool.AddMediaFilesByID([]string{"1004"})
testPlaylists = []*model.Playlist{&plsBest, &plsCool}
pr := NewPlaylistRepository(ctx, conn)
@@ -192,7 +229,13 @@ var _ = BeforeSuite(func() {
if err := arr.SetStar(true, artistBeatles.ID); err != nil {
panic(err)
}
- ar, _ := arr.Get(artistBeatles.ID)
+ ar, err := arr.Get(artistBeatles.ID)
+ if err != nil {
+ panic(err)
+ }
+ if ar == nil {
+ panic("artist not found after SetStar")
+ }
artistBeatles.Starred = true
artistBeatles.StarredAt = ar.StarredAt
testArtists[1] = artistBeatles
@@ -204,6 +247,9 @@ var _ = BeforeSuite(func() {
if err != nil {
panic(err)
}
+ if al == nil {
+ panic("album not found after SetStar")
+ }
albumRadioactivity.Starred = true
albumRadioactivity.StarredAt = al.StarredAt
testAlbums[2] = albumRadioactivity
diff --git a/persistence/playlist_repository.go b/persistence/playlist_repository.go
index 743eca470..046284e1f 100644
--- a/persistence/playlist_repository.go
+++ b/persistence/playlist_repository.go
@@ -161,7 +161,7 @@ func (r *playlistRepository) GetWithTracks(id string, refreshSmartPlaylist, incl
log.Error(r.ctx, "Error loading playlist tracks ", "playlist", pls.Name, "id", pls.ID, err)
return nil, err
}
- pls.Tracks = tracks
+ pls.SetTracks(tracks)
return pls, nil
}
@@ -197,6 +197,25 @@ func (r *playlistRepository) GetAll(options ...model.QueryOptions) (model.Playli
return playlists, err
}
+func (r *playlistRepository) GetPlaylists(mediaFileId string) (model.Playlists, error) {
+ sel := r.selectPlaylist(model.QueryOptions{Sort: "name"}).
+ Join("playlist_tracks on playlist.id = playlist_tracks.playlist_id").
+ Where(And{Eq{"playlist_tracks.media_file_id": mediaFileId}, r.userFilter()})
+ var res []dbPlaylist
+ err := r.queryAll(sel, &res)
+ if err != nil {
+ if errors.Is(err, model.ErrNotFound) {
+ return model.Playlists{}, nil
+ }
+ return nil, err
+ }
+ playlists := make(model.Playlists, len(res))
+ for i, p := range res {
+ playlists[i] = p.Playlist
+ }
+ return playlists, nil
+}
+
func (r *playlistRepository) selectPlaylist(options ...model.QueryOptions) SelectBuilder {
return r.newSelect(options...).Join("user on user.id = owner_id").
Columns(r.tableName+".*", "user.user_name as owner_name")
@@ -244,7 +263,7 @@ func (r *playlistRepository) refreshSmartPlaylist(pls *model.Playlist) bool {
From("media_file").LeftJoin("annotation on (" +
"annotation.item_id = media_file.id" +
" AND annotation.item_type = 'media_file'" +
- " AND annotation.user_id = '" + userId(r.ctx) + "')")
+ " AND annotation.user_id = '" + usr.ID + "')")
sq = r.addCriteria(sq, rules)
insSql := Insert("playlist_tracks").Columns("id", "playlist_id", "media_file_id").Select(sq)
_, err = r.executeSQL(insSql)
@@ -360,6 +379,8 @@ func (r *playlistRepository) refreshCounters(pls *model.Playlist) error {
}
func (r *playlistRepository) loadTracks(sel SelectBuilder, id string) (model.PlaylistTracks, error) {
+ sel = r.applyLibraryFilter(sel, "f")
+ userID := loggedUser(r.ctx).ID
tracksQuery := sel.
Columns(
"coalesce(starred, 0) as starred",
@@ -370,11 +391,12 @@ func (r *playlistRepository) loadTracks(sel SelectBuilder, id string) (model.Pla
"f.*",
"playlist_tracks.*",
"library.path as library_path",
+ "library.name as library_name",
).
LeftJoin("annotation on (" +
"annotation.item_id = media_file_id" +
" AND annotation.item_type = 'media_file'" +
- " AND annotation.user_id = '" + userId(r.ctx) + "')").
+ " AND annotation.user_id = '" + userID + "')").
Join("media_file f on f.id = media_file_id").
Join("library on f.library_id = library.id").
Where(Eq{"playlist_id": id})
diff --git a/persistence/playlist_repository_test.go b/persistence/playlist_repository_test.go
index 5a82964c9..7fad93b1e 100644
--- a/persistence/playlist_repository_test.go
+++ b/persistence/playlist_repository_test.go
@@ -79,13 +79,13 @@ var _ = Describe("PlaylistRepository", func() {
It("Put/Exists/Delete", func() {
By("saves the playlist to the DB")
newPls := model.Playlist{Name: "Great!", OwnerID: "userid"}
- newPls.AddTracks([]string{"1004", "1003"})
+ newPls.AddMediaFilesByID([]string{"1004", "1003"})
By("saves the playlist to the DB")
Expect(repo.Put(&newPls)).To(BeNil())
By("adds repeated songs to a playlist and keeps the order")
- newPls.AddTracks([]string{"1004"})
+ newPls.AddMediaFilesByID([]string{"1004"})
Expect(repo.Put(&newPls)).To(BeNil())
saved, _ := repo.GetWithTracks(newPls.ID, true, false)
Expect(saved.Tracks).To(HaveLen(3))
@@ -112,6 +112,21 @@ var _ = Describe("PlaylistRepository", func() {
})
})
+ Describe("GetPlaylists", func() {
+ It("returns playlists for a track", func() {
+ pls, err := repo.GetPlaylists(songRadioactivity.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(pls).To(HaveLen(1))
+ Expect(pls[0].ID).To(Equal(plsBest.ID))
+ })
+
+ It("returns empty when none", func() {
+ pls, err := repo.GetPlaylists("9999")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(pls).To(HaveLen(0))
+ })
+ })
+
Context("Smart Playlists", func() {
var rules *criteria.Criteria
BeforeEach(func() {
@@ -204,4 +219,37 @@ var _ = Describe("PlaylistRepository", func() {
})
})
})
+
+ Describe("Playlist Track Sorting", func() {
+ var testPlaylistID string
+
+ AfterEach(func() {
+ if testPlaylistID != "" {
+ Expect(repo.Delete(testPlaylistID)).To(BeNil())
+ testPlaylistID = ""
+ }
+ })
+
+ It("sorts tracks correctly by album (disc and track number)", func() {
+ By("creating a playlist with multi-disc album tracks in arbitrary order")
+ newPls := model.Playlist{Name: "Multi-Disc Test", OwnerID: "userid"}
+ // Add tracks in intentionally scrambled order
+ newPls.AddMediaFilesByID([]string{"2001", "2002", "2003", "2004"})
+ Expect(repo.Put(&newPls)).To(Succeed())
+ testPlaylistID = newPls.ID
+
+ By("retrieving tracks sorted by album")
+ tracksRepo := repo.Tracks(newPls.ID, false)
+ tracks, err := tracksRepo.GetAll(model.QueryOptions{Sort: "album", Order: "asc"})
+ Expect(err).ToNot(HaveOccurred())
+
+ By("verifying tracks are sorted by disc number then track number")
+ Expect(tracks).To(HaveLen(4))
+ // Expected order: Disc 1 Track 1, Disc 1 Track 2, Disc 2 Track 1, Disc 2 Track 11
+ Expect(tracks[0].MediaFileID).To(Equal("2002")) // Disc 1, Track 1
+ Expect(tracks[1].MediaFileID).To(Equal("2004")) // Disc 1, Track 2
+ Expect(tracks[2].MediaFileID).To(Equal("2003")) // Disc 2, Track 1
+ Expect(tracks[3].MediaFileID).To(Equal("2001")) // Disc 2, Track 11
+ })
+ })
})
diff --git a/persistence/playlist_track_repository.go b/persistence/playlist_track_repository.go
index d33bd5113..b3f9e0c07 100644
--- a/persistence/playlist_track_repository.go
+++ b/persistence/playlist_track_repository.go
@@ -47,14 +47,15 @@ func (r *playlistRepository) Tracks(playlistId string, refreshSmartPlaylist bool
p.db = r.db
p.tableName = "playlist_tracks"
p.registerModel(&model.PlaylistTrack{}, map[string]filterFunc{
- "missing": booleanFilter,
+ "missing": booleanFilter,
+ "library_id": libraryIdFilter,
})
p.setSortMappings(
map[string]string{
"id": "playlist_tracks.id",
"artist": "order_artist_name",
"album_artist": "order_album_artist_name",
- "album": "order_album_name, order_album_artist_name",
+ "album": "order_album_name, album_id, disc_number, track_number, order_artist_name, title",
"title": "order_title",
// To make sure these fields will be whitelisted
"duration": "duration",
@@ -84,11 +85,12 @@ func (r *playlistTrackRepository) Count(options ...rest.QueryOptions) (int64, er
}
func (r *playlistTrackRepository) Read(id string) (interface{}, error) {
+ userID := loggedUser(r.ctx).ID
sel := r.newSelect().
LeftJoin("annotation on ("+
"annotation.item_id = media_file_id"+
" AND annotation.item_type = 'media_file'"+
- " AND annotation.user_id = '"+userId(r.ctx)+"')").
+ " AND annotation.user_id = '"+userID+"')").
Columns(
"coalesce(starred, 0) as starred",
"coalesce(play_count, 0) as play_count",
@@ -99,10 +101,10 @@ func (r *playlistTrackRepository) Read(id string) (interface{}, error) {
"playlist_tracks.*",
).
Join("media_file f on f.id = media_file_id").
- Where(And{Eq{"playlist_id": r.playlistId}, Eq{"id": id}})
+ Where(And{Eq{"playlist_id": r.playlistId}, Eq{"playlist_tracks.id": id}})
var trk dbPlaylistTrack
err := r.queryOne(sel, &trk)
- return trk.PlaylistTrack.MediaFile, err
+ return trk.PlaylistTrack, err
}
func (r *playlistTrackRepository) GetAll(options ...model.QueryOptions) (model.PlaylistTracks, error) {
diff --git a/persistence/playqueue_repository.go b/persistence/playqueue_repository.go
index fe42dd7fc..74c80ee92 100644
--- a/persistence/playqueue_repository.go
+++ b/persistence/playqueue_repository.go
@@ -2,6 +2,7 @@ package persistence
import (
"context"
+ "errors"
"strings"
"time"
@@ -27,7 +28,7 @@ func NewPlayQueueRepository(ctx context.Context, db dbx.Builder) model.PlayQueue
type playQueue struct {
ID string `structs:"id"`
UserID string `structs:"user_id"`
- Current string `structs:"current"`
+ Current int `structs:"current"`
Position int64 `structs:"position"`
ChangedBy string `structs:"changed_by"`
Items string `structs:"items"`
@@ -35,22 +36,39 @@ type playQueue struct {
UpdatedAt time.Time `structs:"updated_at"`
}
-func (r *playQueueRepository) Store(q *model.PlayQueue) error {
+func (r *playQueueRepository) Store(q *model.PlayQueue, colNames ...string) error {
u := loggedUser(r.ctx)
- err := r.clearPlayQueue(q.UserID)
- if err != nil {
- log.Error(r.ctx, "Error deleting previous playqueue", "user", u.UserName, err)
+
+ // Always find existing playqueue for this user
+ existingQueue, err := r.Retrieve(q.UserID)
+ if err != nil && !errors.Is(err, model.ErrNotFound) {
+ log.Error(r.ctx, "Error retrieving existing playqueue", "user", u.UserName, err)
return err
}
- if len(q.Items) == 0 {
- return nil
+
+ // Use existing ID if found, otherwise keep the provided ID (which may be empty for new records)
+ if !errors.Is(err, model.ErrNotFound) && existingQueue.ID != "" {
+ q.ID = existingQueue.ID
}
+
+ // When no specific columns are provided, we replace the whole queue
+ if len(colNames) == 0 {
+ err := r.clearPlayQueue(q.UserID)
+ if err != nil {
+ log.Error(r.ctx, "Error deleting previous playqueue", "user", u.UserName, err)
+ return err
+ }
+ if len(q.Items) == 0 {
+ return nil
+ }
+ }
+
pq := r.fromModel(q)
if pq.ID == "" {
pq.CreatedAt = time.Now()
}
pq.UpdatedAt = time.Now()
- _, err = r.put(pq.ID, pq)
+ _, err = r.put(pq.ID, pq, colNames...)
if err != nil {
log.Error(r.ctx, "Error saving playqueue", "user", u.UserName, err)
return err
@@ -58,12 +76,21 @@ func (r *playQueueRepository) Store(q *model.PlayQueue) error {
return nil
}
+func (r *playQueueRepository) RetrieveWithMediaFiles(userId string) (*model.PlayQueue, error) {
+ sel := r.newSelect().Columns("*").Where(Eq{"user_id": userId})
+ var res playQueue
+ err := r.queryOne(sel, &res)
+ q := r.toModel(&res)
+ q.Items = r.loadTracks(q.Items)
+ return &q, err
+}
+
func (r *playQueueRepository) Retrieve(userId string) (*model.PlayQueue, error) {
sel := r.newSelect().Columns("*").Where(Eq{"user_id": userId})
var res playQueue
err := r.queryOne(sel, &res)
- pls := r.toModel(&res)
- return &pls, err
+ q := r.toModel(&res)
+ return &q, err
}
func (r *playQueueRepository) fromModel(q *model.PlayQueue) playQueue {
@@ -100,7 +127,6 @@ func (r *playQueueRepository) toModel(pq *playQueue) model.PlayQueue {
q.Items = append(q.Items, model.MediaFile{ID: t})
}
}
- q.Items = r.loadTracks(q.Items)
return q
}
@@ -145,4 +171,8 @@ func (r *playQueueRepository) clearPlayQueue(userId string) error {
return r.delete(Eq{"user_id": userId})
}
+func (r *playQueueRepository) Clear(userId string) error {
+ return r.clearPlayQueue(userId)
+}
+
var _ model.PlayQueueRepository = (*playQueueRepository)(nil)
diff --git a/persistence/playqueue_repository_test.go b/persistence/playqueue_repository_test.go
index a370e1162..2bcc88fd0 100644
--- a/persistence/playqueue_repository_test.go
+++ b/persistence/playqueue_repository_test.go
@@ -5,6 +5,7 @@ import (
"time"
"github.com/Masterminds/squirrel"
+ "github.com/navidrome/navidrome/conf/configtest"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/model/id"
@@ -18,38 +19,296 @@ var _ = Describe("PlayQueueRepository", func() {
var ctx context.Context
BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
ctx = log.NewContext(context.TODO())
ctx = request.WithUser(ctx, model.User{ID: "userid", UserName: "userid", IsAdmin: true})
repo = NewPlayQueueRepository(ctx, GetDBXBuilder())
})
- Describe("PlayQueues", func() {
+ Describe("Store", func() {
+ It("stores a complete playqueue", func() {
+ expected := aPlayQueue("userid", 1, 123, songComeTogether, songDayInALife)
+ Expect(repo.Store(expected)).To(Succeed())
+
+ actual, err := repo.RetrieveWithMediaFiles("userid")
+ Expect(err).ToNot(HaveOccurred())
+ AssertPlayQueue(expected, actual)
+ Expect(countPlayQueues(repo, "userid")).To(Equal(1))
+ })
+
+ It("replaces existing playqueue when storing without column names", func() {
+ By("Storing initial playqueue")
+ initial := aPlayQueue("userid", 0, 100, songComeTogether)
+ Expect(repo.Store(initial)).To(Succeed())
+
+ By("Storing replacement playqueue")
+ replacement := aPlayQueue("userid", 1, 200, songDayInALife, songAntenna)
+ Expect(repo.Store(replacement)).To(Succeed())
+
+ actual, err := repo.RetrieveWithMediaFiles("userid")
+ Expect(err).ToNot(HaveOccurred())
+ AssertPlayQueue(replacement, actual)
+ Expect(countPlayQueues(repo, "userid")).To(Equal(1))
+ })
+
+ It("clears playqueue when storing empty items", func() {
+ By("Storing initial playqueue")
+ initial := aPlayQueue("userid", 0, 100, songComeTogether)
+ Expect(repo.Store(initial)).To(Succeed())
+
+ By("Storing empty playqueue")
+ empty := aPlayQueue("userid", 0, 0)
+ Expect(repo.Store(empty)).To(Succeed())
+
+ By("Verifying playqueue is cleared")
+ _, err := repo.Retrieve("userid")
+ Expect(err).To(MatchError(model.ErrNotFound))
+ })
+
+ It("updates only current field when specified", func() {
+ By("Storing initial playqueue")
+ initial := aPlayQueue("userid", 0, 100, songComeTogether, songDayInALife)
+ Expect(repo.Store(initial)).To(Succeed())
+
+ By("Getting the existing playqueue to obtain its ID")
+ existing, err := repo.Retrieve("userid")
+ Expect(err).ToNot(HaveOccurred())
+
+ By("Updating only current field")
+ update := &model.PlayQueue{
+ ID: existing.ID, // Use existing ID for partial update
+ UserID: "userid",
+ Current: 1,
+ ChangedBy: "test-update",
+ }
+ Expect(repo.Store(update, "current")).To(Succeed())
+
+ By("Verifying only current was updated")
+ actual, err := repo.Retrieve("userid")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(actual.Current).To(Equal(1))
+ Expect(actual.Position).To(Equal(int64(100))) // Should remain unchanged
+ Expect(actual.Items).To(HaveLen(2)) // Should remain unchanged
+ })
+
+ It("updates only position field when specified", func() {
+ By("Storing initial playqueue")
+ initial := aPlayQueue("userid", 1, 100, songComeTogether, songDayInALife)
+ Expect(repo.Store(initial)).To(Succeed())
+
+ By("Getting the existing playqueue to obtain its ID")
+ existing, err := repo.Retrieve("userid")
+ Expect(err).ToNot(HaveOccurred())
+
+ By("Updating only position field")
+ update := &model.PlayQueue{
+ ID: existing.ID, // Use existing ID for partial update
+ UserID: "userid",
+ Position: 500,
+ ChangedBy: "test-update",
+ }
+ Expect(repo.Store(update, "position")).To(Succeed())
+
+ By("Verifying only position was updated")
+ actual, err := repo.Retrieve("userid")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(actual.Position).To(Equal(int64(500)))
+ Expect(actual.Current).To(Equal(1)) // Should remain unchanged
+ Expect(actual.Items).To(HaveLen(2)) // Should remain unchanged
+ })
+
+ It("updates multiple specified fields", func() {
+ By("Storing initial playqueue")
+ initial := aPlayQueue("userid", 0, 100, songComeTogether)
+ Expect(repo.Store(initial)).To(Succeed())
+
+ By("Getting the existing playqueue to obtain its ID")
+ existing, err := repo.Retrieve("userid")
+ Expect(err).ToNot(HaveOccurred())
+
+ By("Updating current and position fields")
+ update := &model.PlayQueue{
+ ID: existing.ID, // Use existing ID for partial update
+ UserID: "userid",
+ Current: 1,
+ Position: 300,
+ ChangedBy: "test-update",
+ }
+ Expect(repo.Store(update, "current", "position")).To(Succeed())
+
+ By("Verifying both fields were updated")
+ actual, err := repo.Retrieve("userid")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(actual.Current).To(Equal(1))
+ Expect(actual.Position).To(Equal(int64(300)))
+ Expect(actual.Items).To(HaveLen(1)) // Should remain unchanged
+ })
+
+ It("preserves existing data when updating with empty items list and column names", func() {
+ By("Storing initial playqueue")
+ initial := aPlayQueue("userid", 0, 100, songComeTogether, songDayInALife)
+ Expect(repo.Store(initial)).To(Succeed())
+
+ By("Getting the existing playqueue to obtain its ID")
+ existing, err := repo.Retrieve("userid")
+ Expect(err).ToNot(HaveOccurred())
+
+ By("Updating only position with empty items")
+ update := &model.PlayQueue{
+ ID: existing.ID, // Use existing ID for partial update
+ UserID: "userid",
+ Position: 200,
+ ChangedBy: "test-update",
+ Items: []model.MediaFile{}, // Empty items
+ }
+ Expect(repo.Store(update, "position")).To(Succeed())
+
+ By("Verifying items are preserved")
+ actual, err := repo.Retrieve("userid")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(actual.Position).To(Equal(int64(200)))
+ Expect(actual.Items).To(HaveLen(2)) // Should remain unchanged
+ })
+
+ It("ensures only one record per user by reusing existing record ID", func() {
+ By("Storing initial playqueue")
+ initial := aPlayQueue("userid", 0, 100, songComeTogether)
+ Expect(repo.Store(initial)).To(Succeed())
+ initialCount := countPlayQueues(repo, "userid")
+ Expect(initialCount).To(Equal(1))
+
+ By("Storing another playqueue with different ID but same user")
+ different := aPlayQueue("userid", 1, 200, songDayInALife)
+ different.ID = "different-id" // Force a different ID
+ Expect(repo.Store(different)).To(Succeed())
+
+ By("Verifying only one record exists for the user")
+ finalCount := countPlayQueues(repo, "userid")
+ Expect(finalCount).To(Equal(1))
+
+ By("Verifying the record was updated, not duplicated")
+ actual, err := repo.Retrieve("userid")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(actual.Current).To(Equal(1)) // Should be updated value
+ Expect(actual.Position).To(Equal(int64(200))) // Should be updated value
+ Expect(actual.Items).To(HaveLen(1)) // Should be new items
+ Expect(actual.Items[0].ID).To(Equal(songDayInALife.ID))
+ })
+
+ It("ensures only one record per user even with partial updates", func() {
+ By("Storing initial playqueue")
+ initial := aPlayQueue("userid", 0, 100, songComeTogether, songDayInALife)
+ Expect(repo.Store(initial)).To(Succeed())
+ initialCount := countPlayQueues(repo, "userid")
+ Expect(initialCount).To(Equal(1))
+
+ By("Storing partial update with different ID but same user")
+ partialUpdate := &model.PlayQueue{
+ ID: "completely-different-id", // Use a completely different ID
+ UserID: "userid",
+ Current: 1,
+ ChangedBy: "test-partial",
+ }
+ Expect(repo.Store(partialUpdate, "current")).To(Succeed())
+
+ By("Verifying only one record still exists for the user")
+ finalCount := countPlayQueues(repo, "userid")
+ Expect(finalCount).To(Equal(1))
+
+ By("Verifying the existing record was updated with new current value")
+ actual, err := repo.Retrieve("userid")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(actual.Current).To(Equal(1)) // Should be updated value
+ Expect(actual.Position).To(Equal(int64(100))) // Should remain unchanged
+ Expect(actual.Items).To(HaveLen(2)) // Should remain unchanged
+ })
+ })
+
+ Describe("Retrieve", func() {
It("returns notfound error if there's no playqueue for the user", func() {
_, err := repo.Retrieve("user999")
Expect(err).To(MatchError(model.ErrNotFound))
})
- It("stores and retrieves the playqueue for the user", func() {
+ It("retrieves the playqueue with only track IDs (no full MediaFile data)", func() {
By("Storing a playqueue for the user")
- expected := aPlayQueue("userid", songDayInALife.ID, 123, songComeTogether, songDayInALife)
+ expected := aPlayQueue("userid", 1, 123, songComeTogether, songDayInALife)
Expect(repo.Store(expected)).To(Succeed())
actual, err := repo.Retrieve("userid")
Expect(err).ToNot(HaveOccurred())
- AssertPlayQueue(expected, actual)
+ // Basic playqueue properties should match
+ Expect(actual.ID).To(Equal(expected.ID))
+ Expect(actual.UserID).To(Equal(expected.UserID))
+ Expect(actual.Current).To(Equal(expected.Current))
+ Expect(actual.Position).To(Equal(expected.Position))
+ Expect(actual.ChangedBy).To(Equal(expected.ChangedBy))
+ Expect(actual.Items).To(HaveLen(len(expected.Items)))
- By("Storing a new playqueue for the same user")
+ // Items should only contain IDs, not full MediaFile data
+ for i, item := range actual.Items {
+ Expect(item.ID).To(Equal(expected.Items[i].ID))
+ // These fields should be empty since we're not loading full MediaFiles
+ Expect(item.Title).To(BeEmpty())
+ Expect(item.Path).To(BeEmpty())
+ Expect(item.Album).To(BeEmpty())
+ Expect(item.Artist).To(BeEmpty())
+ }
+ })
- another := aPlayQueue("userid", songRadioactivity.ID, 321, songAntenna, songRadioactivity)
- Expect(repo.Store(another)).To(Succeed())
+ It("returns items with IDs even when some tracks don't exist in the DB", func() {
+ // Add a new song to the DB
+ newSong := songRadioactivity
+ newSong.ID = "temp-track"
+ newSong.Path = "/new-path"
+ mfRepo := NewMediaFileRepository(ctx, GetDBXBuilder())
- actual, err = repo.Retrieve("userid")
+ Expect(mfRepo.Put(&newSong)).To(Succeed())
+
+ // Create a playqueue with the new song
+ pq := aPlayQueue("userid", 0, 0, newSong, songAntenna)
+ Expect(repo.Store(pq)).To(Succeed())
+
+ // Delete the new song from the database
+ Expect(mfRepo.Delete("temp-track")).To(Succeed())
+
+ // Retrieve the playqueue with Retrieve method
+ actual, err := repo.Retrieve("userid")
Expect(err).ToNot(HaveOccurred())
- AssertPlayQueue(another, actual)
- Expect(countPlayQueues(repo, "userid")).To(Equal(1))
+ // The playqueue should still contain both track IDs (including the deleted one)
+ Expect(actual.Items).To(HaveLen(2))
+ Expect(actual.Items[0].ID).To(Equal("temp-track"))
+ Expect(actual.Items[1].ID).To(Equal(songAntenna.ID))
+
+ // Items should only contain IDs, no other data
+ for _, item := range actual.Items {
+ Expect(item.Title).To(BeEmpty())
+ Expect(item.Path).To(BeEmpty())
+ Expect(item.Album).To(BeEmpty())
+ Expect(item.Artist).To(BeEmpty())
+ }
+ })
+ })
+
+ Describe("RetrieveWithMediaFiles", func() {
+ It("returns notfound error if there's no playqueue for the user", func() {
+ _, err := repo.RetrieveWithMediaFiles("user999")
+ Expect(err).To(MatchError(model.ErrNotFound))
+ })
+
+ It("retrieves the playqueue with full MediaFile data", func() {
+ By("Storing a playqueue for the user")
+
+ expected := aPlayQueue("userid", 1, 123, songComeTogether, songDayInALife)
+ Expect(repo.Store(expected)).To(Succeed())
+
+ actual, err := repo.RetrieveWithMediaFiles("userid")
+ Expect(err).ToNot(HaveOccurred())
+
+ AssertPlayQueue(expected, actual)
})
It("does not return tracks if they don't exist in the DB", func() {
@@ -62,11 +321,11 @@ var _ = Describe("PlayQueueRepository", func() {
Expect(mfRepo.Put(&newSong)).To(Succeed())
// Create a playqueue with the new song
- pq := aPlayQueue("userid", newSong.ID, 0, newSong, songAntenna)
+ pq := aPlayQueue("userid", 0, 0, newSong, songAntenna)
Expect(repo.Store(pq)).To(Succeed())
// Retrieve the playqueue
- actual, err := repo.Retrieve("userid")
+ actual, err := repo.RetrieveWithMediaFiles("userid")
Expect(err).ToNot(HaveOccurred())
// The playqueue should contain both tracks
@@ -76,7 +335,7 @@ var _ = Describe("PlayQueueRepository", func() {
Expect(mfRepo.Delete("temp-track")).To(Succeed())
// Retrieve the playqueue
- actual, err = repo.Retrieve("userid")
+ actual, err = repo.RetrieveWithMediaFiles("userid")
Expect(err).ToNot(HaveOccurred())
// The playqueue should not contain the deleted track
@@ -84,6 +343,59 @@ var _ = Describe("PlayQueueRepository", func() {
Expect(actual.Items[0].ID).To(Equal(songAntenna.ID))
})
})
+
+ Describe("Clear", func() {
+ It("clears an existing playqueue", func() {
+ By("Storing a playqueue")
+ expected := aPlayQueue("userid", 1, 123, songComeTogether, songDayInALife)
+ Expect(repo.Store(expected)).To(Succeed())
+
+ By("Verifying playqueue exists")
+ _, err := repo.Retrieve("userid")
+ Expect(err).ToNot(HaveOccurred())
+
+ By("Clearing the playqueue")
+ Expect(repo.Clear("userid")).To(Succeed())
+
+ By("Verifying playqueue is cleared")
+ _, err = repo.Retrieve("userid")
+ Expect(err).To(MatchError(model.ErrNotFound))
+ })
+
+ It("does not error when clearing non-existent playqueue", func() {
+ // Clear should not error even if no playqueue exists
+ Expect(repo.Clear("nonexistent-user")).To(Succeed())
+ })
+
+ It("only clears the specified user's playqueue", func() {
+ By("Creating users in the database to avoid foreign key constraints")
+ userRepo := NewUserRepository(ctx, GetDBXBuilder())
+ user1 := &model.User{ID: "user1", UserName: "user1", Name: "User 1", Email: "user1@test.com"}
+ user2 := &model.User{ID: "user2", UserName: "user2", Name: "User 2", Email: "user2@test.com"}
+ Expect(userRepo.Put(user1)).To(Succeed())
+ Expect(userRepo.Put(user2)).To(Succeed())
+
+ By("Storing playqueues for two users")
+ user1Queue := aPlayQueue("user1", 0, 100, songComeTogether)
+ user2Queue := aPlayQueue("user2", 1, 200, songDayInALife)
+ Expect(repo.Store(user1Queue)).To(Succeed())
+ Expect(repo.Store(user2Queue)).To(Succeed())
+
+ By("Clearing only user1's playqueue")
+ Expect(repo.Clear("user1")).To(Succeed())
+
+ By("Verifying user1's playqueue is cleared")
+ _, err := repo.Retrieve("user1")
+ Expect(err).To(MatchError(model.ErrNotFound))
+
+ By("Verifying user2's playqueue still exists")
+ actual, err := repo.Retrieve("user2")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(actual.UserID).To(Equal("user2"))
+ Expect(actual.Current).To(Equal(1))
+ Expect(actual.Position).To(Equal(int64(200)))
+ })
+ })
})
func countPlayQueues(repo model.PlayQueueRepository, userId string) int {
@@ -107,7 +419,7 @@ func AssertPlayQueue(expected, actual *model.PlayQueue) {
}
}
-func aPlayQueue(userId, current string, position int64, items ...model.MediaFile) *model.PlayQueue {
+func aPlayQueue(userId string, current int, position int64, items ...model.MediaFile) *model.PlayQueue {
createdAt := time.Now()
updatedAt := createdAt.Add(time.Minute)
return &model.PlayQueue{
diff --git a/persistence/scrobble_buffer_repository_test.go b/persistence/scrobble_buffer_repository_test.go
index 6962ea7c6..62423ff45 100644
--- a/persistence/scrobble_buffer_repository_test.go
+++ b/persistence/scrobble_buffer_repository_test.go
@@ -152,7 +152,7 @@ var _ = Describe("ScrobbleBufferRepository", func() {
Expect(err).ToNot(HaveOccurred())
Expect(entry).ToNot(BeNil())
- Expect(entry.EnqueueTime).To(BeTemporally("~", now))
+ Expect(entry.EnqueueTime).To(BeTemporally("~", now, 100*time.Millisecond))
Expect(entry.MediaFileID).To(Equal(fileId))
Expect(entry.PlayTime).To(BeTemporally("==", playTime))
},
diff --git a/persistence/share_repository.go b/persistence/share_repository.go
index abe1ea6e6..d943943e0 100644
--- a/persistence/share_repository.go
+++ b/persistence/share_repository.go
@@ -95,7 +95,7 @@ func (r *shareRepository) loadMedia(share *model.Share) error {
return err
case "album":
albumRepo := NewAlbumRepository(r.ctx, r.db)
- share.Albums, err = albumRepo.GetAll(model.QueryOptions{Filters: noMissing(Eq{"id": ids})})
+ share.Albums, err = albumRepo.GetAll(model.QueryOptions{Filters: noMissing(Eq{"album.id": ids})})
if err != nil {
return err
}
diff --git a/persistence/share_repository_test.go b/persistence/share_repository_test.go
new file mode 100644
index 000000000..252115175
--- /dev/null
+++ b/persistence/share_repository_test.go
@@ -0,0 +1,133 @@
+package persistence
+
+import (
+ "context"
+ "time"
+
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("ShareRepository", func() {
+ var repo model.ShareRepository
+ var ctx context.Context
+ var adminUser = model.User{ID: "admin", UserName: "admin", IsAdmin: true}
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ ctx = request.WithUser(log.NewContext(context.TODO()), adminUser)
+ repo = NewShareRepository(ctx, GetDBXBuilder())
+
+ // Insert the admin user into the database (required for foreign key constraint)
+ ur := NewUserRepository(ctx, GetDBXBuilder())
+ err := ur.Put(&adminUser)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Clean up shares
+ db := GetDBXBuilder()
+ _, err = db.NewQuery("DELETE FROM share").Execute()
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ Describe("Headless Access", func() {
+ Context("Repository creation and basic operations", func() {
+ It("should create repository successfully with no user context", func() {
+ // Create repository with no user context (headless)
+ headlessRepo := NewShareRepository(context.Background(), GetDBXBuilder())
+ Expect(headlessRepo).ToNot(BeNil())
+ })
+
+ It("should handle GetAll for headless processes", func() {
+ // Create a simple share directly in database
+ shareID := "headless-test-share"
+ _, err := GetDBXBuilder().NewQuery(`
+ INSERT INTO share (id, user_id, description, resource_type, resource_ids, created_at, updated_at)
+ VALUES ({:id}, {:user}, {:desc}, {:type}, {:ids}, {:created}, {:updated})
+ `).Bind(map[string]interface{}{
+ "id": shareID,
+ "user": adminUser.ID,
+ "desc": "Headless Test Share",
+ "type": "song",
+ "ids": "song-1",
+ "created": time.Now(),
+ "updated": time.Now(),
+ }).Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Headless process should see all shares
+ headlessRepo := NewShareRepository(context.Background(), GetDBXBuilder())
+ shares, err := headlessRepo.GetAll()
+ Expect(err).ToNot(HaveOccurred())
+
+ found := false
+ for _, s := range shares {
+ if s.ID == shareID {
+ found = true
+ break
+ }
+ }
+ Expect(found).To(BeTrue(), "Headless process should see all shares")
+ })
+
+ It("should handle individual share retrieval for headless processes", func() {
+ // Create a simple share
+ shareID := "headless-get-share"
+ _, err := GetDBXBuilder().NewQuery(`
+ INSERT INTO share (id, user_id, description, resource_type, resource_ids, created_at, updated_at)
+ VALUES ({:id}, {:user}, {:desc}, {:type}, {:ids}, {:created}, {:updated})
+ `).Bind(map[string]interface{}{
+ "id": shareID,
+ "user": adminUser.ID,
+ "desc": "Headless Get Share",
+ "type": "song",
+ "ids": "song-2",
+ "created": time.Now(),
+ "updated": time.Now(),
+ }).Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Headless process should be able to get the share
+ headlessRepo := NewShareRepository(context.Background(), GetDBXBuilder())
+ share, err := headlessRepo.Get(shareID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(share.ID).To(Equal(shareID))
+ Expect(share.Description).To(Equal("Headless Get Share"))
+ })
+ })
+ })
+
+ Describe("SQL ambiguity fix verification", func() {
+ It("should handle share operations without SQL ambiguity errors", func() {
+ // This test verifies that the loadMedia function doesn't cause SQL ambiguity
+ // The key fix was using "album.id" instead of "id" in the album query filters
+
+ // Create a share that would trigger the loadMedia function
+ shareID := "sql-test-share"
+ _, err := GetDBXBuilder().NewQuery(`
+ INSERT INTO share (id, user_id, description, resource_type, resource_ids, created_at, updated_at)
+ VALUES ({:id}, {:user}, {:desc}, {:type}, {:ids}, {:created}, {:updated})
+ `).Bind(map[string]interface{}{
+ "id": shareID,
+ "user": adminUser.ID,
+ "desc": "SQL Test Share",
+ "type": "album",
+ "ids": "non-existent-album", // Won't find albums, but shouldn't cause SQL errors
+ "created": time.Now(),
+ "updated": time.Now(),
+ }).Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // The Get operation should work without SQL ambiguity errors
+ // even if no albums are found
+ share, err := repo.Get(shareID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(share.ID).To(Equal(shareID))
+ // Albums array should be empty since we used non-existent album ID
+ Expect(share.Albums).To(BeEmpty())
+ })
+ })
+})
diff --git a/persistence/sql_annotations.go b/persistence/sql_annotations.go
index daf621ffe..98ade6e21 100644
--- a/persistence/sql_annotations.go
+++ b/persistence/sql_annotations.go
@@ -15,15 +15,14 @@ import (
const annotationTable = "annotation"
func (r sqlRepository) withAnnotation(query SelectBuilder, idField string) SelectBuilder {
- if userId(r.ctx) == invalidUserId {
+ userID := loggedUser(r.ctx).ID
+ if userID == invalidUserId {
return query
}
query = query.
LeftJoin("annotation on ("+
"annotation.item_id = "+idField+
- // item_ids are unique across different item_types, so the clause below is not needed
- //" AND annotation.item_type = '"+r.tableName+"'"+
- " AND annotation.user_id = '"+userId(r.ctx)+"')").
+ " AND annotation.user_id = '"+userID+"')").
Columns(
"coalesce(starred, 0) as starred",
"coalesce(rating, 0) as rating",
@@ -42,8 +41,9 @@ func (r sqlRepository) withAnnotation(query SelectBuilder, idField string) Selec
}
func (r sqlRepository) annId(itemID ...string) And {
+ userID := loggedUser(r.ctx).ID
return And{
- Eq{annotationTable + ".user_id": userId(r.ctx)},
+ Eq{annotationTable + ".user_id": userID},
Eq{annotationTable + ".item_type": r.tableName},
Eq{annotationTable + ".item_id": itemID},
}
@@ -56,8 +56,9 @@ func (r sqlRepository) annUpsert(values map[string]interface{}, itemIDs ...strin
}
c, err := r.executeSQL(upd)
if c == 0 || errors.Is(err, sql.ErrNoRows) {
+ userID := loggedUser(r.ctx).ID
for _, itemID := range itemIDs {
- values["user_id"] = userId(r.ctx)
+ values["user_id"] = userID
values["item_type"] = r.tableName
values["item_id"] = itemID
ins := Insert(annotationTable).SetMap(values)
@@ -86,8 +87,9 @@ func (r sqlRepository) IncPlayCount(itemID string, ts time.Time) error {
c, err := r.executeSQL(upd)
if c == 0 || errors.Is(err, sql.ErrNoRows) {
+ userID := loggedUser(r.ctx).ID
values := map[string]interface{}{}
- values["user_id"] = userId(r.ctx)
+ values["user_id"] = userID
values["item_type"] = r.tableName
values["item_id"] = itemID
values["play_count"] = 1
@@ -117,7 +119,7 @@ func (r sqlRepository) cleanAnnotations() error {
del := Delete(annotationTable).Where(Eq{"item_type": r.tableName}).Where("item_id not in (select id from " + r.tableName + ")")
c, err := r.executeSQL(del)
if err != nil {
- return fmt.Errorf("error cleaning up annotations: %w", err)
+ return fmt.Errorf("error cleaning up %s annotations: %w", r.tableName, err)
}
if c > 0 {
log.Debug(r.ctx, "Clean-up annotations", "table", r.tableName, "totalDeleted", c)
diff --git a/persistence/sql_base_repository.go b/persistence/sql_base_repository.go
index 7cc24b6c4..ce026a3c3 100644
--- a/persistence/sql_base_repository.go
+++ b/persistence/sql_base_repository.go
@@ -49,27 +49,14 @@ type sqlRepository struct {
const invalidUserId = "-1"
-func userId(ctx context.Context) string {
- if user, ok := request.UserFrom(ctx); !ok {
- return invalidUserId
- } else {
- return user.ID
- }
-}
-
func loggedUser(ctx context.Context) *model.User {
if user, ok := request.UserFrom(ctx); !ok {
- return &model.User{}
+ return &model.User{ID: invalidUserId}
} else {
return &user
}
}
-func isAdmin(ctx context.Context) bool {
- user := loggedUser(ctx)
- return user.IsAdmin
-}
-
func (r *sqlRepository) registerModel(instance any, filters map[string]filterFunc) {
if r.tableName == "" {
r.tableName = strings.TrimPrefix(reflect.TypeOf(instance).String(), "*model.")
@@ -86,6 +73,10 @@ func (r *sqlRepository) registerModel(instance any, filters map[string]filterFun
// which gives precedence to sort tags.
// Ex: order_title => (coalesce(nullif(sort_title,”),order_title) collate nocase)
// To avoid performance issues, indexes should be created for these sort expressions
+//
+// NOTE: if an individual item has spaces, it should be wrapped in parentheses. For example,
+// you should write "(lyrics != '[]')". This prevents the item being split unexpectedly.
+// Without parentheses, "lyrics != '[]'" would be mapped as simply "lyrics"
func (r *sqlRepository) setSortMappings(mappings map[string]string, tableName ...string) {
tn := r.tableName
if len(tableName) > 0 {
@@ -195,10 +186,45 @@ func (r sqlRepository) applyFilters(sq SelectBuilder, options ...model.QueryOpti
return sq
}
+func (r *sqlRepository) withTableName(filter filterFunc) filterFunc {
+ return func(field string, value any) Sqlizer {
+ if r.tableName != "" {
+ field = r.tableName + "." + field
+ }
+ return filter(field, value)
+ }
+}
+
+// libraryIdFilter is a filter function to be added to resources that have a library_id column.
+func libraryIdFilter(_ string, value interface{}) Sqlizer {
+ return Eq{"library_id": value}
+}
+
+// applyLibraryFilter adds library filtering to queries for tables that have a library_id column
+// This ensures users only see content from libraries they have access to
+func (r sqlRepository) applyLibraryFilter(sq SelectBuilder, tableName ...string) SelectBuilder {
+ user := loggedUser(r.ctx)
+
+ // If the user is an admin, or the user ID is invalid (e.g., when no user is logged in), skip the library filter
+ if user.IsAdmin || user.ID == invalidUserId {
+ return sq
+ }
+
+ table := r.tableName
+ if len(tableName) > 0 {
+ table = tableName[0]
+ }
+
+ // Get user's accessible library IDs
+ // Use subquery to filter by user's library access
+ return sq.Where(Expr(table+".library_id IN ("+
+ "SELECT ul.library_id FROM user_library ul WHERE ul.user_id = ?)", user.ID))
+}
+
func (r sqlRepository) seedKey() string {
// Seed keys must be all lowercase, or else SQLite3 will encode it, making it not match the seed
// used in the query. Hashing the user ID and converting it to a hex string will do the trick
- userIDHash := md5.Sum([]byte(userId(r.ctx)))
+ userIDHash := md5.Sum([]byte(loggedUser(r.ctx).ID))
return fmt.Sprintf("%s|%x", r.tableName, userIDHash)
}
diff --git a/persistence/sql_base_repository_test.go b/persistence/sql_base_repository_test.go
index 4b380e298..b46e2066b 100644
--- a/persistence/sql_base_repository_test.go
+++ b/persistence/sql_base_repository_test.go
@@ -136,6 +136,10 @@ var _ = Describe("sqlRepository", func() {
})
Describe("buildSortOrder", func() {
+ BeforeEach(func() {
+ r.sortMappings = map[string]string{}
+ })
+
Context("single field", func() {
It("sorts by specified field", func() {
sql := r.buildSortOrder("name", "desc")
@@ -163,6 +167,14 @@ var _ = Describe("sqlRepository", func() {
sql := r.buildSortOrder("name desc, age, status asc", "desc")
Expect(sql).To(Equal("name asc, age desc, status desc"))
})
+ It("handles spaces in mapped field", func() {
+ r.sortMappings = map[string]string{
+ "has_lyrics": "(lyrics != '[]'), updated_at",
+ }
+ sql := r.buildSortOrder("has_lyrics", "desc")
+ Expect(sql).To(Equal("(lyrics != '[]') desc, updated_at desc"))
+ })
+
})
Context("function fields", func() {
It("handles functions with multiple params", func() {
@@ -211,4 +223,62 @@ var _ = Describe("sqlRepository", func() {
Expect(hasher.CurrentSeed(id)).To(Equal("seed"))
})
})
+
+ Describe("applyLibraryFilter", func() {
+ var sq squirrel.SelectBuilder
+
+ BeforeEach(func() {
+ sq = squirrel.Select("*").From("test_table")
+ })
+
+ Context("Admin User", func() {
+ BeforeEach(func() {
+ r.ctx = request.WithUser(context.Background(), model.User{ID: "admin", IsAdmin: true})
+ })
+
+ It("should not apply library filter for admin users", func() {
+ result := r.applyLibraryFilter(sq)
+ sql, _, _ := result.ToSql()
+ Expect(sql).To(Equal("SELECT * FROM test_table"))
+ })
+ })
+
+ Context("Regular User", func() {
+ BeforeEach(func() {
+ r.ctx = request.WithUser(context.Background(), model.User{ID: "user123", IsAdmin: false})
+ })
+
+ It("should apply library filter for regular users", func() {
+ result := r.applyLibraryFilter(sq)
+ sql, args, _ := result.ToSql()
+ Expect(sql).To(ContainSubstring("IN (SELECT ul.library_id FROM user_library ul WHERE ul.user_id = ?)"))
+ Expect(args).To(ContainElement("user123"))
+ })
+
+ It("should use custom table name when provided", func() {
+ result := r.applyLibraryFilter(sq, "custom_table")
+ sql, args, _ := result.ToSql()
+ Expect(sql).To(ContainSubstring("custom_table.library_id IN"))
+ Expect(args).To(ContainElement("user123"))
+ })
+ })
+
+ Context("Headless Process (No User Context)", func() {
+ BeforeEach(func() {
+ r.ctx = context.Background() // No user context
+ })
+
+ It("should not apply library filter for headless processes", func() {
+ result := r.applyLibraryFilter(sq)
+ sql, _, _ := result.ToSql()
+ Expect(sql).To(Equal("SELECT * FROM test_table"))
+ })
+
+ It("should not apply library filter even with custom table name", func() {
+ result := r.applyLibraryFilter(sq, "custom_table")
+ sql, _, _ := result.ToSql()
+ Expect(sql).To(Equal("SELECT * FROM test_table"))
+ })
+ })
+ })
})
diff --git a/persistence/sql_bookmarks.go b/persistence/sql_bookmarks.go
index 56645ea21..9164aed9d 100644
--- a/persistence/sql_bookmarks.go
+++ b/persistence/sql_bookmarks.go
@@ -15,21 +15,20 @@ import (
const bookmarkTable = "bookmark"
func (r sqlRepository) withBookmark(query SelectBuilder, idField string) SelectBuilder {
- if userId(r.ctx) == invalidUserId {
+ userID := loggedUser(r.ctx).ID
+ if userID == invalidUserId {
return query
}
return query.
LeftJoin("bookmark on (" +
"bookmark.item_id = " + idField +
- // item_ids are unique across different item_types, so the clause below is not needed
- //" AND bookmark.item_type = '" + r.tableName + "'" +
- " AND bookmark.user_id = '" + userId(r.ctx) + "')").
+ " AND bookmark.user_id = '" + userID + "')").
Columns("coalesce(position, 0) as bookmark_position")
}
func (r sqlRepository) bmkID(itemID ...string) And {
return And{
- Eq{bookmarkTable + ".user_id": userId(r.ctx)},
+ Eq{bookmarkTable + ".user_id": loggedUser(r.ctx).ID},
Eq{bookmarkTable + ".item_type": r.tableName},
Eq{bookmarkTable + ".item_id": itemID},
}
@@ -149,10 +148,10 @@ func (r sqlRepository) cleanBookmarks() error {
del := Delete(bookmarkTable).Where(Eq{"item_type": r.tableName}).Where("item_id not in (select id from " + r.tableName + ")")
c, err := r.executeSQL(del)
if err != nil {
- return fmt.Errorf("error cleaning up bookmarks: %w", err)
+ return fmt.Errorf("error cleaning up %s bookmarks: %w", r.tableName, err)
}
if c > 0 {
- log.Debug(r.ctx, "Clean-up bookmarks", "totalDeleted", c)
+ log.Debug(r.ctx, "Clean-up bookmarks", "totalDeleted", c, "itemType", r.tableName)
}
return nil
}
diff --git a/persistence/sql_participations.go b/persistence/sql_participations.go
index 006b7063b..d88eca45e 100644
--- a/persistence/sql_participations.go
+++ b/persistence/sql_participations.go
@@ -15,6 +15,13 @@ type participant struct {
SubRole string `json:"subRole,omitempty"`
}
+// flatParticipant represents a flattened participant structure for SQL processing
+type flatParticipant struct {
+ ArtistID string `json:"artist_id"`
+ Role string `json:"role"`
+ SubRole string `json:"sub_role,omitempty"`
+}
+
func marshalParticipants(participants model.Participants) string {
dbParticipants := make(map[model.Role][]participant)
for role, artists := range participants {
@@ -53,22 +60,47 @@ func (r sqlRepository) updateParticipants(itemID string, participants model.Part
if len(participants) == 0 {
return nil
}
- sqi := Insert(r.tableName+"_artists").
- Columns(r.tableName+"_id", "artist_id", "role", "sub_role").
- Suffix(fmt.Sprintf("on conflict (artist_id, %s_id, role, sub_role) do nothing", r.tableName))
+
+ var flatParticipants []flatParticipant
for role, artists := range participants {
for _, artist := range artists {
- sqi = sqi.Values(itemID, artist.ID, role.String(), artist.SubRole)
+ flatParticipants = append(flatParticipants, flatParticipant{
+ ArtistID: artist.ID,
+ Role: role.String(),
+ SubRole: artist.SubRole,
+ })
}
}
- _, err = r.executeSQL(sqi)
+
+ participantsJSON, err := json.Marshal(flatParticipants)
+ if err != nil {
+ return fmt.Errorf("marshaling participants: %w", err)
+ }
+
+ // Build the INSERT query using json_each and INNER JOIN to artist table
+ // to automatically filter out non-existent artist IDs
+ query := fmt.Sprintf(`
+ INSERT INTO %[1]s_artists (%[1]s_id, artist_id, role, sub_role)
+ SELECT ?,
+ json_extract(value, '$.artist_id') as artist_id,
+ json_extract(value, '$.role') as role,
+ COALESCE(json_extract(value, '$.sub_role'), '') as sub_role
+ -- Parse the flat JSON array: [{"artist_id": "id", "role": "role", "sub_role": "subRole"}]
+ FROM json_each(?) -- Iterate through each array element
+ -- CRITICAL: Only insert records for artists that actually exist in the database
+ JOIN artist ON artist.id = json_extract(value, '$.artist_id') -- Filter out non-existent artist IDs via INNER JOIN
+ -- Handle duplicate insertions gracefully (e.g., if called multiple times)
+ ON CONFLICT (artist_id, %[1]s_id, role, sub_role) DO NOTHING -- Ignore duplicates
+ `, r.tableName)
+
+ _, err = r.executeSQL(Expr(query, itemID, string(participantsJSON)))
return err
}
func (r *sqlRepository) getParticipants(m *model.MediaFile) (model.Participants, error) {
ar := NewArtistRepository(r.ctx, r.db)
ids := m.Participants.AllIDs()
- artists, err := ar.GetAll(model.QueryOptions{Filters: Eq{"id": ids}})
+ artists, err := ar.GetAll(model.QueryOptions{Filters: Eq{"artist.id": ids}})
if err != nil {
return nil, fmt.Errorf("getting participants: %w", err)
}
diff --git a/persistence/sql_restful.go b/persistence/sql_restful.go
index 6be368b00..ff0d06a8b 100644
--- a/persistence/sql_restful.go
+++ b/persistence/sql_restful.go
@@ -1,6 +1,7 @@
package persistence
import (
+ "cmp"
"context"
"fmt"
"reflect"
@@ -105,8 +106,15 @@ func booleanFilter(field string, value any) Sqlizer {
return Eq{field: v == "true"}
}
-func fullTextFilter(tableName string) func(string, any) Sqlizer {
- return func(field string, value any) Sqlizer { return fullTextExpr(tableName, value.(string)) }
+func fullTextFilter(tableName string, mbidFields ...string) func(string, any) Sqlizer {
+ return func(field string, value any) Sqlizer {
+ v := strings.ToLower(value.(string))
+ cond := cmp.Or(
+ mbidExpr(tableName, v, mbidFields...),
+ fullTextExpr(tableName, v),
+ )
+ return cond
+ }
}
func substringFilter(field string, value any) Sqlizer {
diff --git a/persistence/sql_restful_test.go b/persistence/sql_restful_test.go
index 20cc31a36..fd95fbb31 100644
--- a/persistence/sql_restful_test.go
+++ b/persistence/sql_restful_test.go
@@ -2,9 +2,12 @@ package persistence
import (
"context"
+ "strings"
"github.com/Masterminds/squirrel"
"github.com/deluan/rest"
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
)
@@ -66,4 +69,167 @@ var _ = Describe("sqlRestful", func() {
Expect(r.parseRestFilters(context.Background(), options)).To(Equal(squirrel.And{squirrel.Gt{"test": 100}}))
})
})
+
+ Describe("fullTextFilter function", func() {
+ var filter filterFunc
+ var tableName string
+ var mbidFields []string
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ tableName = "test_table"
+ mbidFields = []string{"mbid", "artist_mbid"}
+ filter = fullTextFilter(tableName, mbidFields...)
+ })
+
+ Context("when value is a valid UUID", func() {
+ It("returns only the mbid filter (precedence over full text)", func() {
+ uuid := "550e8400-e29b-41d4-a716-446655440000"
+ result := filter("search", uuid)
+
+ expected := squirrel.Or{
+ squirrel.Eq{"test_table.mbid": uuid},
+ squirrel.Eq{"test_table.artist_mbid": uuid},
+ }
+ Expect(result).To(Equal(expected))
+ })
+
+ It("falls back to full text when no mbid fields are provided", func() {
+ noMbidFilter := fullTextFilter(tableName)
+ uuid := "550e8400-e29b-41d4-a716-446655440000"
+ result := noMbidFilter("search", uuid)
+
+ // mbidExpr with no fields returns nil, so cmp.Or falls back to fullTextExpr
+ expected := squirrel.And{
+ squirrel.Like{"test_table.full_text": "% 550e8400-e29b-41d4-a716-446655440000%"},
+ }
+ Expect(result).To(Equal(expected))
+ })
+ })
+
+ Context("when value is not a valid UUID", func() {
+ It("returns full text search condition only", func() {
+ result := filter("search", "beatles")
+
+ // mbidExpr returns nil for non-UUIDs, so fullTextExpr result is returned directly
+ expected := squirrel.And{
+ squirrel.Like{"test_table.full_text": "% beatles%"},
+ }
+ Expect(result).To(Equal(expected))
+ })
+
+ It("handles multi-word search terms", func() {
+ result := filter("search", "the beatles abbey road")
+
+ // Should return And condition directly
+ andCondition, ok := result.(squirrel.And)
+ Expect(ok).To(BeTrue())
+ Expect(andCondition).To(HaveLen(4))
+
+ // Check that all words are present (order may vary)
+ Expect(andCondition).To(ContainElement(squirrel.Like{"test_table.full_text": "% the%"}))
+ Expect(andCondition).To(ContainElement(squirrel.Like{"test_table.full_text": "% beatles%"}))
+ Expect(andCondition).To(ContainElement(squirrel.Like{"test_table.full_text": "% abbey%"}))
+ Expect(andCondition).To(ContainElement(squirrel.Like{"test_table.full_text": "% road%"}))
+ })
+ })
+
+ Context("when SearchFullString config changes behavior", func() {
+ It("uses different separator with SearchFullString=false", func() {
+ conf.Server.SearchFullString = false
+ result := filter("search", "test query")
+
+ andCondition, ok := result.(squirrel.And)
+ Expect(ok).To(BeTrue())
+ Expect(andCondition).To(HaveLen(2))
+
+ // Check that all words are present with leading space (order may vary)
+ Expect(andCondition).To(ContainElement(squirrel.Like{"test_table.full_text": "% test%"}))
+ Expect(andCondition).To(ContainElement(squirrel.Like{"test_table.full_text": "% query%"}))
+ })
+
+ It("uses no separator with SearchFullString=true", func() {
+ conf.Server.SearchFullString = true
+ result := filter("search", "test query")
+
+ andCondition, ok := result.(squirrel.And)
+ Expect(ok).To(BeTrue())
+ Expect(andCondition).To(HaveLen(2))
+
+ // Check that all words are present without leading space (order may vary)
+ Expect(andCondition).To(ContainElement(squirrel.Like{"test_table.full_text": "%test%"}))
+ Expect(andCondition).To(ContainElement(squirrel.Like{"test_table.full_text": "%query%"}))
+ })
+ })
+
+ Context("edge cases", func() {
+ It("returns nil for empty string", func() {
+ result := filter("search", "")
+ Expect(result).To(BeNil())
+ })
+
+ It("returns nil for string with only whitespace", func() {
+ result := filter("search", " ")
+ Expect(result).To(BeNil())
+ })
+
+ It("handles special characters that are sanitized", func() {
+ result := filter("search", "don't")
+
+ expected := squirrel.And{
+ squirrel.Like{"test_table.full_text": "% dont%"}, // str.SanitizeStrings removes quotes
+ }
+ Expect(result).To(Equal(expected))
+ })
+
+ It("returns nil for single quote (SQL injection protection)", func() {
+ result := filter("search", "'")
+ Expect(result).To(BeNil())
+ })
+
+ It("handles mixed case UUIDs", func() {
+ uuid := "550E8400-E29B-41D4-A716-446655440000"
+ result := filter("search", uuid)
+
+ // Should return only mbid filter (uppercase UUID should work)
+ expected := squirrel.Or{
+ squirrel.Eq{"test_table.mbid": strings.ToLower(uuid)},
+ squirrel.Eq{"test_table.artist_mbid": strings.ToLower(uuid)},
+ }
+ Expect(result).To(Equal(expected))
+ })
+
+ It("handles invalid UUID format gracefully", func() {
+ result := filter("search", "550e8400-invalid-uuid")
+
+ // Should return full text filter since UUID is invalid
+ expected := squirrel.And{
+ squirrel.Like{"test_table.full_text": "% 550e8400-invalid-uuid%"},
+ }
+ Expect(result).To(Equal(expected))
+ })
+
+ It("handles empty mbid fields array", func() {
+ emptyMbidFilter := fullTextFilter(tableName, []string{}...)
+ result := emptyMbidFilter("search", "test")
+
+ // mbidExpr with empty fields returns nil, so cmp.Or falls back to fullTextExpr
+ expected := squirrel.And{
+ squirrel.Like{"test_table.full_text": "% test%"},
+ }
+ Expect(result).To(Equal(expected))
+ })
+
+ It("converts value to lowercase before processing", func() {
+ result := filter("search", "TEST")
+
+ // The function converts to lowercase internally
+ expected := squirrel.And{
+ squirrel.Like{"test_table.full_text": "% test%"},
+ }
+ Expect(result).To(Equal(expected))
+ })
+ })
+ })
+
})
diff --git a/persistence/sql_search.go b/persistence/sql_search.go
index 9ac171263..0d3bfb743 100644
--- a/persistence/sql_search.go
+++ b/persistence/sql_search.go
@@ -4,6 +4,7 @@ import (
"strings"
. "github.com/Masterminds/squirrel"
+ "github.com/google/uuid"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/utils/str"
@@ -14,32 +15,50 @@ func formatFullText(text ...string) string {
return " " + fullText
}
-func (r sqlRepository) doSearch(sq SelectBuilder, q string, offset, size int, includeMissing bool, results any, orderBys ...string) error {
+// doSearch performs a full-text search with the specified parameters.
+// The naturalOrder is used to sort results when no full-text filter is applied. It is useful for cases like
+// OpenSubsonic, where an empty search query should return all results in a natural order. Normally the parameter
+// should be `tableName + ".rowid"`, but some repositories (ex: artist) may use a different natural order.
+func (r sqlRepository) doSearch(sq SelectBuilder, q string, offset, size int, results any, naturalOrder string, orderBys ...string) error {
q = strings.TrimSpace(q)
q = strings.TrimSuffix(q, "*")
if len(q) < 2 {
return nil
}
- //sq := r.newSelect().Columns(r.tableName + ".*")
- //sq = r.withAnnotation(sq, r.tableName+".id")
- //sq = r.withBookmark(sq, r.tableName+".id")
filter := fullTextExpr(r.tableName, q)
if filter != nil {
sq = sq.Where(filter)
sq = sq.OrderBy(orderBys...)
} else {
- // If the filter is empty, we sort by rowid.
// This is to speed up the results of `search3?query=""`, for OpenSubsonic
- sq = sq.OrderBy(r.tableName + ".rowid")
- }
- if !includeMissing {
- sq = sq.Where(Eq{r.tableName + ".missing": false})
+ // If the filter is empty, we sort by the specified natural order.
+ sq = sq.OrderBy(naturalOrder)
}
+ sq = sq.Where(Eq{r.tableName + ".missing": false})
sq = sq.Limit(uint64(size)).Offset(uint64(offset))
return r.queryAll(sq, results, model.QueryOptions{Offset: offset})
}
+func (r sqlRepository) searchByMBID(sq SelectBuilder, mbid string, mbidFields []string, results any) error {
+ sq = sq.Where(mbidExpr(r.tableName, mbid, mbidFields...))
+ sq = sq.Where(Eq{r.tableName + ".missing": false})
+
+ return r.queryAll(sq, results)
+}
+
+func mbidExpr(tableName, mbid string, mbidFields ...string) Sqlizer {
+ if uuid.Validate(mbid) != nil || len(mbidFields) == 0 {
+ return nil
+ }
+ mbid = strings.ToLower(mbid)
+ var cond []Sqlizer
+ for _, mbidField := range mbidFields {
+ cond = append(cond, Eq{tableName + "." + mbidField: mbid})
+ }
+ return Or(cond)
+}
+
func fullTextExpr(tableName string, s string) Sqlizer {
q := str.SanitizeStrings(s)
if q == "" {
diff --git a/persistence/sql_tags.go b/persistence/sql_tags.go
index d7b48f23e..8c3c1e89d 100644
--- a/persistence/sql_tags.go
+++ b/persistence/sql_tags.go
@@ -1,12 +1,15 @@
package persistence
import (
+ "context"
"encoding/json"
"fmt"
"strings"
. "github.com/Masterminds/squirrel"
+ "github.com/deluan/rest"
"github.com/navidrome/navidrome/model"
+ "github.com/pocketbase/dbx"
)
// Format of a tag in the DB
@@ -55,3 +58,111 @@ func tagIDFilter(name string, idValue any) Sqlizer {
},
)
}
+
+// tagLibraryIdFilter filters tags based on library access through the library_tag table
+func tagLibraryIdFilter(_ string, value interface{}) Sqlizer {
+ return Eq{"library_tag.library_id": value}
+}
+
+// baseTagRepository provides common functionality for all tag-based repositories.
+// It handles CRUD operations with optional filtering by tag name.
+type baseTagRepository struct {
+ sqlRepository
+ tagFilter *model.TagName // nil = no filter (all tags), non-nil = filter by specific tag name
+}
+
+// newBaseTagRepository creates a new base tag repository with optional tag filtering.
+// If tagFilter is nil, the repository will work with all tags.
+// If tagFilter is provided, the repository will only work with tags of that specific name.
+func newBaseTagRepository(ctx context.Context, db dbx.Builder, tagFilter *model.TagName) *baseTagRepository {
+ r := &baseTagRepository{
+ tagFilter: tagFilter,
+ }
+ r.ctx = ctx
+ r.db = db
+ r.tableName = "tag"
+ r.registerModel(&model.Tag{}, map[string]filterFunc{
+ "name": containsFilter("tag_value"),
+ "library_id": tagLibraryIdFilter,
+ })
+ r.setSortMappings(map[string]string{
+ "name": "tag_value",
+ })
+ return r
+}
+
+// applyLibraryFiltering adds the appropriate library joins based on user context
+func (r *baseTagRepository) applyLibraryFiltering(sq SelectBuilder) SelectBuilder {
+ // Add library_tag join
+ sq = sq.LeftJoin("library_tag on library_tag.tag_id = tag.id")
+
+ // For authenticated users, also join with user_library to filter by accessible libraries
+ user := loggedUser(r.ctx)
+ if user.ID != invalidUserId {
+ sq = sq.Join("user_library on user_library.library_id = library_tag.library_id AND user_library.user_id = ?", user.ID)
+ }
+
+ return sq
+}
+
+// newSelect overrides the base implementation to apply tag name filtering and library filtering.
+func (r *baseTagRepository) newSelect(options ...model.QueryOptions) SelectBuilder {
+ sq := r.sqlRepository.newSelect(options...)
+
+ // Apply tag name filtering if specified
+ if r.tagFilter != nil {
+ sq = sq.Where(Eq{"tag.tag_name": *r.tagFilter})
+ }
+
+ // Apply library filtering and set up aggregation columns
+ sq = r.applyLibraryFiltering(sq).Columns(
+ "tag.id",
+ "tag.tag_name",
+ "tag.tag_value",
+ "COALESCE(SUM(library_tag.album_count), 0) as album_count",
+ "COALESCE(SUM(library_tag.media_file_count), 0) as song_count",
+ ).GroupBy("tag.id", "tag.tag_name", "tag.tag_value")
+
+ return sq
+}
+
+// ResourceRepository interface implementation
+
+func (r *baseTagRepository) Count(options ...rest.QueryOptions) (int64, error) {
+ sq := Select("COUNT(DISTINCT tag.id)").From("tag")
+
+ // Apply tag name filtering if specified
+ if r.tagFilter != nil {
+ sq = sq.Where(Eq{"tag.tag_name": *r.tagFilter})
+ }
+
+ // Apply library filtering
+ sq = r.applyLibraryFiltering(sq)
+
+ return r.count(sq, r.parseRestOptions(r.ctx, options...))
+}
+
+func (r *baseTagRepository) Read(id string) (interface{}, error) {
+ query := r.newSelect().Where(Eq{"id": id})
+ var res model.Tag
+ err := r.queryOne(query, &res)
+ return &res, err
+}
+
+func (r *baseTagRepository) ReadAll(options ...rest.QueryOptions) (interface{}, error) {
+ query := r.newSelect(r.parseRestOptions(r.ctx, options...))
+ var res model.TagList
+ err := r.queryAll(query, &res)
+ return res, err
+}
+
+func (r *baseTagRepository) EntityName() string {
+ return "tag"
+}
+
+func (r *baseTagRepository) NewInstance() interface{} {
+ return model.Tag{}
+}
+
+// Interface compliance check
+var _ model.ResourceRepository = (*baseTagRepository)(nil)
diff --git a/persistence/tag_library_filtering_test.go b/persistence/tag_library_filtering_test.go
new file mode 100644
index 000000000..ab0d57d52
--- /dev/null
+++ b/persistence/tag_library_filtering_test.go
@@ -0,0 +1,259 @@
+package persistence
+
+import (
+ "context"
+
+ "github.com/deluan/rest"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/id"
+ "github.com/navidrome/navidrome/model/request"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+ "github.com/pocketbase/dbx"
+)
+
+const (
+ adminUserID = "userid"
+ regularUserID = "2222"
+ libraryID1 = 1
+ libraryID2 = 2
+ libraryID3 = 3
+
+ tagNameGenre = "genre"
+ tagValueRock = "rock"
+ tagValuePop = "pop"
+ tagValueJazz = "jazz"
+)
+
+var _ = Describe("Tag Library Filtering", func() {
+ var (
+ tagRockID = id.NewTagID(tagNameGenre, tagValueRock)
+ tagPopID = id.NewTagID(tagNameGenre, tagValuePop)
+ tagJazzID = id.NewTagID(tagNameGenre, tagValueJazz)
+ )
+
+ expectTagValues := func(tagList model.TagList, expected []string) {
+ tagValues := make([]string, len(tagList))
+ for i, tag := range tagList {
+ tagValues[i] = tag.TagValue
+ }
+ Expect(tagValues).To(ContainElements(expected))
+ }
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+
+ // Clean up database
+ db := GetDBXBuilder()
+ _, err := db.NewQuery("DELETE FROM library_tag").Execute()
+ Expect(err).ToNot(HaveOccurred())
+ _, err = db.NewQuery("DELETE FROM tag").Execute()
+ Expect(err).ToNot(HaveOccurred())
+ _, err = db.NewQuery("DELETE FROM user_library WHERE user_id != {:admin} AND user_id != {:regular}").
+ Bind(dbx.Params{"admin": adminUserID, "regular": regularUserID}).Execute()
+ Expect(err).ToNot(HaveOccurred())
+ _, err = db.NewQuery("DELETE FROM library WHERE id > 1").Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Create test libraries
+ _, err = db.NewQuery("INSERT INTO library (id, name, path) VALUES ({:id}, {:name}, {:path})").
+ Bind(dbx.Params{"id": libraryID2, "name": "Library 2", "path": "/music/lib2"}).Execute()
+ Expect(err).ToNot(HaveOccurred())
+ _, err = db.NewQuery("INSERT INTO library (id, name, path) VALUES ({:id}, {:name}, {:path})").
+ Bind(dbx.Params{"id": libraryID3, "name": "Library 3", "path": "/music/lib3"}).Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Give admin access to all libraries
+ for _, libID := range []int{libraryID1, libraryID2, libraryID3} {
+ _, err = db.NewQuery("INSERT OR IGNORE INTO user_library (user_id, library_id) VALUES ({:user}, {:lib})").
+ Bind(dbx.Params{"user": adminUserID, "lib": libID}).Execute()
+ Expect(err).ToNot(HaveOccurred())
+ }
+
+ // Create test tags
+ adminCtx := request.WithUser(log.NewContext(context.TODO()), adminUser)
+ tagRepo := NewTagRepository(adminCtx, GetDBXBuilder())
+
+ createTag := func(libraryID int, name, value string) {
+ tag := model.Tag{ID: id.NewTagID(name, value), TagName: model.TagName(name), TagValue: value}
+ err := tagRepo.Add(libraryID, tag)
+ Expect(err).ToNot(HaveOccurred())
+ }
+
+ createTag(libraryID1, tagNameGenre, tagValueRock)
+ createTag(libraryID2, tagNameGenre, tagValuePop)
+ createTag(libraryID3, tagNameGenre, tagValueJazz)
+ createTag(libraryID2, tagNameGenre, tagValueRock) // Rock appears in both lib1 and lib2
+
+ // Set tag counts (manually for testing)
+ setCounts := func(tagID string, libID, albums, songs int) {
+ _, err := db.NewQuery("UPDATE library_tag SET album_count = {:albums}, media_file_count = {:songs} WHERE tag_id = {:tag} AND library_id = {:lib}").
+ Bind(dbx.Params{"albums": albums, "songs": songs, "tag": tagID, "lib": libID}).Execute()
+ Expect(err).ToNot(HaveOccurred())
+ }
+
+ setCounts(tagRockID, libraryID1, 5, 20)
+ setCounts(tagPopID, libraryID2, 3, 10)
+ setCounts(tagJazzID, libraryID3, 2, 8)
+ setCounts(tagRockID, libraryID2, 1, 4)
+
+ // Give regular user access to library 2 only
+ _, err = db.NewQuery("INSERT INTO user_library (user_id, library_id) VALUES ({:user}, {:lib})").
+ Bind(dbx.Params{"user": regularUserID, "lib": libraryID2}).Execute()
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ Describe("TagRepository Library Filtering", func() {
+ // Helper to create repository and read all tags
+ readAllTags := func(user *model.User, filters ...rest.QueryOptions) model.TagList {
+ var ctx context.Context
+ if user != nil {
+ ctx = request.WithUser(log.NewContext(context.TODO()), *user)
+ } else {
+ ctx = context.Background() // Headless context
+ }
+
+ tagRepo := NewTagRepository(ctx, GetDBXBuilder())
+ repo := tagRepo.(model.ResourceRepository)
+
+ var opts rest.QueryOptions
+ if len(filters) > 0 {
+ opts = filters[0]
+ }
+
+ tags, err := repo.ReadAll(opts)
+ Expect(err).ToNot(HaveOccurred())
+ return tags.(model.TagList)
+ }
+
+ // Helper to count tags
+ countTags := func(user *model.User) int64 {
+ var ctx context.Context
+ if user != nil {
+ ctx = request.WithUser(log.NewContext(context.TODO()), *user)
+ } else {
+ ctx = context.Background()
+ }
+
+ tagRepo := NewTagRepository(ctx, GetDBXBuilder())
+ repo := tagRepo.(model.ResourceRepository)
+
+ count, err := repo.Count()
+ Expect(err).ToNot(HaveOccurred())
+ return count
+ }
+
+ Context("Admin User", func() {
+ It("should see all tags regardless of library", func() {
+ tags := readAllTags(&adminUser)
+ Expect(tags).To(HaveLen(3))
+ })
+ })
+
+ Context("Regular User with Limited Library Access", func() {
+ It("should only see tags from accessible libraries", func() {
+ tags := readAllTags(®ularUser)
+ // Should see rock (libraries 1,2) and pop (library 2), but not jazz (library 3)
+ Expect(tags).To(HaveLen(2))
+ })
+
+ It("should respect explicit library_id filters within accessible libraries", func() {
+ tags := readAllTags(®ularUser, rest.QueryOptions{
+ Filters: map[string]interface{}{"library_id": libraryID2},
+ })
+ // Should see only tags from library 2: pop and rock(lib2)
+ Expect(tags).To(HaveLen(2))
+ expectTagValues(tags, []string{tagValuePop, tagValueRock})
+ })
+
+ It("should not return tags when filtering by inaccessible library", func() {
+ tags := readAllTags(®ularUser, rest.QueryOptions{
+ Filters: map[string]interface{}{"library_id": libraryID3},
+ })
+ // Should return no tags since user can't access library 3
+ Expect(tags).To(HaveLen(0))
+ })
+
+ It("should filter by library 1 correctly", func() {
+ tags := readAllTags(®ularUser, rest.QueryOptions{
+ Filters: map[string]interface{}{"library_id": libraryID1},
+ })
+ // Should see only rock from library 1
+ Expect(tags).To(HaveLen(1))
+ Expect(tags[0].TagValue).To(Equal(tagValueRock))
+ })
+ })
+
+ Context("Headless Processes (No User Context)", func() {
+ It("should see all tags from all libraries when no user is in context", func() {
+ tags := readAllTags(nil) // nil = headless context
+ // Should see all tags from all libraries (no filtering applied)
+ Expect(tags).To(HaveLen(3))
+ expectTagValues(tags, []string{tagValueRock, tagValuePop, tagValueJazz})
+ })
+
+ It("should count all tags from all libraries when no user is in context", func() {
+ count := countTags(nil)
+ // Should count all tags from all libraries
+ Expect(count).To(Equal(int64(3)))
+ })
+
+ It("should calculate proper statistics from all libraries for headless processes", func() {
+ tags := readAllTags(nil)
+
+ // Find the rock tag (appears in libraries 1 and 2)
+ var rockTag *model.Tag
+ for _, tag := range tags {
+ if tag.TagValue == tagValueRock {
+ rockTag = &tag
+ break
+ }
+ }
+ Expect(rockTag).ToNot(BeNil())
+
+ // Should have stats from all libraries where rock appears
+ // Library 1: 5 albums, 20 songs
+ // Library 2: 1 album, 4 songs
+ // Total: 6 albums, 24 songs
+ Expect(rockTag.AlbumCount).To(Equal(6))
+ Expect(rockTag.SongCount).To(Equal(24))
+ })
+
+ It("should allow headless processes to apply explicit library_id filters", func() {
+ tags := readAllTags(nil, rest.QueryOptions{
+ Filters: map[string]interface{}{"library_id": libraryID3},
+ })
+ // Should see only jazz from library 3
+ Expect(tags).To(HaveLen(1))
+ Expect(tags[0].TagValue).To(Equal(tagValueJazz))
+ })
+ })
+
+ Context("Admin User with Explicit Library Filtering", func() {
+ It("should see all tags when no filter is applied", func() {
+ tags := readAllTags(&adminUser)
+ Expect(tags).To(HaveLen(3))
+ })
+
+ It("should respect explicit library_id filters", func() {
+ tags := readAllTags(&adminUser, rest.QueryOptions{
+ Filters: map[string]interface{}{"library_id": libraryID3},
+ })
+ // Should see only jazz from library 3
+ Expect(tags).To(HaveLen(1))
+ Expect(tags[0].TagValue).To(Equal(tagValueJazz))
+ })
+
+ It("should filter by library 2 correctly", func() {
+ tags := readAllTags(&adminUser, rest.QueryOptions{
+ Filters: map[string]interface{}{"library_id": libraryID2},
+ })
+ // Should see pop and rock from library 2
+ Expect(tags).To(HaveLen(2))
+ expectTagValues(tags, []string{tagValuePop, tagValueRock})
+ })
+ })
+ })
+})
diff --git a/persistence/tag_repository.go b/persistence/tag_repository.go
index d63584af0..5bb8b3832 100644
--- a/persistence/tag_repository.go
+++ b/persistence/tag_repository.go
@@ -7,26 +7,22 @@ import (
"time"
. "github.com/Masterminds/squirrel"
- "github.com/deluan/rest"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
"github.com/pocketbase/dbx"
)
type tagRepository struct {
- sqlRepository
+ *baseTagRepository
}
func NewTagRepository(ctx context.Context, db dbx.Builder) model.TagRepository {
- r := &tagRepository{}
- r.ctx = ctx
- r.db = db
- r.tableName = "tag"
- r.registerModel(&model.Tag{}, nil)
- return r
+ return &tagRepository{
+ baseTagRepository: newBaseTagRepository(ctx, db, nil), // nil = no filter, works with all tags
+ }
}
-func (r *tagRepository) Add(tags ...model.Tag) error {
+func (r *tagRepository) Add(libraryID int, tags ...model.Tag) error {
for chunk := range slices.Chunk(tags, 200) {
sq := Insert(r.tableName).Columns("id", "tag_name", "tag_value").
Suffix("on conflict (id) do nothing")
@@ -37,34 +33,42 @@ func (r *tagRepository) Add(tags ...model.Tag) error {
if err != nil {
return err
}
+
+ // Create library_tag entries for library filtering
+ libSq := Insert("library_tag").Columns("tag_id", "library_id", "album_count", "media_file_count").
+ Suffix("on conflict (tag_id, library_id) do nothing")
+ for _, t := range chunk {
+ libSq = libSq.Values(t.ID, libraryID, 0, 0)
+ }
+ _, err = r.executeSQL(libSq)
+ if err != nil {
+ return fmt.Errorf("adding library_tag entries: %w", err)
+ }
}
return nil
}
-// UpdateCounts updates the album_count and media_file_count columns in the tag_counts table.
+// UpdateCounts updates the library_tag table with per-library statistics.
// Only genres are being updated for now.
func (r *tagRepository) UpdateCounts() error {
template := `
-with updated_values as (
- select jt.value as id, count(distinct %[1]s.id) as %[1]s_count
- from %[1]s
- join json_tree(tags, '$.genre') as jt
- where atom is not null
- and key = 'id'
- group by jt.value
-)
-update tag
-set %[1]s_count = updated_values.%[1]s_count
-from updated_values
-where tag.id = updated_values.id;
+INSERT INTO library_tag (tag_id, library_id, %[1]s_count)
+SELECT jt.value as tag_id, %[1]s.library_id, count(distinct %[1]s.id) as %[1]s_count
+FROM %[1]s
+JOIN json_tree(%[1]s.tags, '$.genre') as jt ON jt.atom IS NOT NULL AND jt.key = 'id'
+JOIN tag ON tag.id = jt.value
+GROUP BY jt.value, %[1]s.library_id
+ON CONFLICT (tag_id, library_id)
+DO UPDATE SET %[1]s_count = excluded.%[1]s_count;
`
+
for _, table := range []string{"album", "media_file"} {
start := time.Now()
query := Expr(fmt.Sprintf(template, table))
c, err := r.executeSQL(query)
- log.Debug(r.ctx, "Updated tag counts", "table", table, "elapsed", time.Since(start), "updated", c)
+ log.Debug(r.ctx, "Updated library tag counts", "table", table, "elapsed", time.Since(start), "updated", c)
if err != nil {
- return fmt.Errorf("updating %s tag counts: %w", table, err)
+ return fmt.Errorf("updating %s library tag counts: %w", table, err)
}
}
return nil
@@ -74,43 +78,22 @@ func (r *tagRepository) purgeUnused() error {
del := Delete(r.tableName).Where(`
id not in (select jt.value
from album left join json_tree(album.tags, '$') as jt
+ where atom is not null
+ and key = 'id'
+ UNION
+ select jt.value
+ from media_file left join json_tree(media_file.tags, '$') as jt
where atom is not null
and key = 'id')
`)
c, err := r.executeSQL(del)
if err != nil {
- return fmt.Errorf("error purging unused tags: %w", err)
+ return fmt.Errorf("error purging %s unused tags: %w", r.tableName, err)
}
if c > 0 {
- log.Debug(r.ctx, "Purged unused tags", "totalDeleted", c)
+ log.Debug(r.ctx, "Purged unused tags", "totalDeleted", c, "table", r.tableName)
}
return err
}
-func (r *tagRepository) Count(options ...rest.QueryOptions) (int64, error) {
- return r.count(r.newSelect(), r.parseRestOptions(r.ctx, options...))
-}
-
-func (r *tagRepository) Read(id string) (interface{}, error) {
- query := r.newSelect().Columns("*").Where(Eq{"id": id})
- var res model.Tag
- err := r.queryOne(query, &res)
- return &res, err
-}
-
-func (r *tagRepository) ReadAll(options ...rest.QueryOptions) (interface{}, error) {
- query := r.newSelect(r.parseRestOptions(r.ctx, options...)).Columns("*")
- var res model.TagList
- err := r.queryAll(query, &res)
- return res, err
-}
-
-func (r *tagRepository) EntityName() string {
- return "tag"
-}
-
-func (r *tagRepository) NewInstance() interface{} {
- return model.Tag{}
-}
-
var _ model.ResourceRepository = &tagRepository{}
diff --git a/persistence/tag_repository_test.go b/persistence/tag_repository_test.go
new file mode 100644
index 000000000..c3947a9f7
--- /dev/null
+++ b/persistence/tag_repository_test.go
@@ -0,0 +1,311 @@
+package persistence
+
+import (
+ "context"
+ "slices"
+ "strings"
+
+ "github.com/deluan/rest"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/id"
+ "github.com/navidrome/navidrome/model/request"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+ "github.com/pocketbase/dbx"
+)
+
+var _ = Describe("TagRepository", func() {
+ var repo model.TagRepository
+ var restRepo model.ResourceRepository
+ var ctx context.Context
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ ctx = request.WithUser(log.NewContext(context.TODO()), model.User{ID: "userid", UserName: "johndoe", IsAdmin: true})
+ tagRepo := NewTagRepository(ctx, GetDBXBuilder())
+ repo = tagRepo
+ restRepo = tagRepo.(model.ResourceRepository)
+
+ // Clean the database before each test to ensure isolation
+ db := GetDBXBuilder()
+ _, err := db.NewQuery("DELETE FROM tag").Execute()
+ Expect(err).ToNot(HaveOccurred())
+ _, err = db.NewQuery("DELETE FROM library_tag").Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Ensure library 1 exists (if it doesn't already)
+ _, err = db.NewQuery("INSERT OR IGNORE INTO library (id, name, path, default_new_users) VALUES (1, 'Test Library', '/test', true)").Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Ensure the admin user has access to library 1
+ _, err = db.NewQuery("INSERT OR IGNORE INTO user_library (user_id, library_id) VALUES ('userid', 1)").Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Add comprehensive test data that covers all test scenarios
+ newTag := func(name, value string) model.Tag {
+ return model.Tag{ID: id.NewTagID(name, value), TagName: model.TagName(name), TagValue: value}
+ }
+
+ err = repo.Add(1,
+ // Genre tags
+ newTag("genre", "rock"),
+ newTag("genre", "pop"),
+ newTag("genre", "jazz"),
+ newTag("genre", "electronic"),
+ newTag("genre", "classical"),
+ newTag("genre", "ambient"),
+ newTag("genre", "techno"),
+ newTag("genre", "house"),
+ newTag("genre", "trance"),
+ newTag("genre", "Alternative Rock"),
+ newTag("genre", "Blues"),
+ newTag("genre", "Country"),
+ // Mood tags
+ newTag("mood", "happy"),
+ newTag("mood", "sad"),
+ newTag("mood", "energetic"),
+ newTag("mood", "calm"),
+ // Other tag types
+ newTag("instrument", "guitar"),
+ newTag("instrument", "piano"),
+ newTag("decade", "1980s"),
+ newTag("decade", "1990s"),
+ )
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ Describe("Add", func() {
+ It("should handle adding new tags", func() {
+ newTag := model.Tag{
+ ID: id.NewTagID("genre", "experimental"),
+ TagName: "genre",
+ TagValue: "experimental",
+ }
+
+ err := repo.Add(1, newTag)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify tag was added
+ result, err := restRepo.Read(newTag.ID)
+ Expect(err).ToNot(HaveOccurred())
+ resultTag := result.(*model.Tag)
+ Expect(resultTag.TagValue).To(Equal("experimental"))
+
+ // Check count increased
+ count, err := restRepo.Count()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(count).To(Equal(int64(21))) // 20 from dataset + 1 new
+ })
+
+ It("should handle duplicate tags gracefully", func() {
+ // Try to add a duplicate tag
+ duplicateTag := model.Tag{
+ ID: id.NewTagID("genre", "rock"), // This already exists
+ TagName: "genre",
+ TagValue: "rock",
+ }
+
+ count, err := restRepo.Count()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(count).To(Equal(int64(20))) // Still 20 tags
+
+ err = repo.Add(1, duplicateTag)
+ Expect(err).ToNot(HaveOccurred()) // Should not error
+
+ // Count should remain the same
+ count, err = restRepo.Count()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(count).To(Equal(int64(20))) // Still 20 tags
+ })
+ })
+
+ Describe("UpdateCounts", func() {
+ It("should update tag counts successfully", func() {
+ err := repo.UpdateCounts()
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ It("should handle empty database gracefully", func() {
+ // Clear the database first
+ db := GetDBXBuilder()
+ _, err := db.NewQuery("DELETE FROM tag").Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ err = repo.UpdateCounts()
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ It("should handle albums with non-existent tag IDs in JSON gracefully", func() {
+ // Regression test for foreign key constraint error
+ // Create an album with tag IDs in JSON that don't exist in tag table
+ db := GetDBXBuilder()
+
+ // First, create a non-existent tag ID (this simulates tags in JSON that aren't in tag table)
+ nonExistentTagID := id.NewTagID("genre", "nonexistent-genre")
+
+ // Create album with JSON containing the non-existent tag ID
+ albumWithBadTags := `{"genre":[{"id":"` + nonExistentTagID + `","value":"nonexistent-genre"}]}`
+
+ // Insert album directly into database with the problematic JSON
+ _, err := db.NewQuery("INSERT INTO album (id, name, library_id, tags) VALUES ({:id}, {:name}, {:lib}, {:tags})").
+ Bind(dbx.Params{
+ "id": "test-album-bad-tags",
+ "name": "Album With Bad Tags",
+ "lib": 1,
+ "tags": albumWithBadTags,
+ }).Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // This should not fail with foreign key constraint error
+ err = repo.UpdateCounts()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Cleanup
+ _, err = db.NewQuery("DELETE FROM album WHERE id = {:id}").
+ Bind(dbx.Params{"id": "test-album-bad-tags"}).Execute()
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ It("should handle media files with non-existent tag IDs in JSON gracefully", func() {
+ // Regression test for foreign key constraint error with media files
+ db := GetDBXBuilder()
+
+ // Create a non-existent tag ID
+ nonExistentTagID := id.NewTagID("genre", "another-nonexistent-genre")
+
+ // Create media file with JSON containing the non-existent tag ID
+ mediaFileWithBadTags := `{"genre":[{"id":"` + nonExistentTagID + `","value":"another-nonexistent-genre"}]}`
+
+ // Insert media file directly into database with the problematic JSON
+ _, err := db.NewQuery("INSERT INTO media_file (id, title, library_id, tags) VALUES ({:id}, {:title}, {:lib}, {:tags})").
+ Bind(dbx.Params{
+ "id": "test-media-bad-tags",
+ "title": "Media File With Bad Tags",
+ "lib": 1,
+ "tags": mediaFileWithBadTags,
+ }).Execute()
+ Expect(err).ToNot(HaveOccurred())
+
+ // This should not fail with foreign key constraint error
+ err = repo.UpdateCounts()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Cleanup
+ _, err = db.NewQuery("DELETE FROM media_file WHERE id = {:id}").
+ Bind(dbx.Params{"id": "test-media-bad-tags"}).Execute()
+ Expect(err).ToNot(HaveOccurred())
+ })
+ })
+
+ Describe("Count", func() {
+ It("should return correct count of tags", func() {
+ count, err := restRepo.Count()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(count).To(Equal(int64(20))) // From the test dataset
+ })
+ })
+
+ Describe("Read", func() {
+ It("should return existing tag", func() {
+ rockID := id.NewTagID("genre", "rock")
+ result, err := restRepo.Read(rockID)
+ Expect(err).ToNot(HaveOccurred())
+ resultTag := result.(*model.Tag)
+ Expect(resultTag.ID).To(Equal(rockID))
+ Expect(resultTag.TagName).To(Equal(model.TagName("genre")))
+ Expect(resultTag.TagValue).To(Equal("rock"))
+ })
+
+ It("should return error for non-existent tag", func() {
+ _, err := restRepo.Read("non-existent-id")
+ Expect(err).To(HaveOccurred())
+ })
+ })
+
+ Describe("ReadAll", func() {
+ It("should return all tags from dataset", func() {
+ result, err := restRepo.ReadAll()
+ Expect(err).ToNot(HaveOccurred())
+ tags := result.(model.TagList)
+ Expect(tags).To(HaveLen(20))
+ })
+
+ It("should filter tags by partial value correctly", func() {
+ options := rest.QueryOptions{
+ Filters: map[string]interface{}{"name": "%rock%"}, // Tags containing 'rock'
+ }
+ result, err := restRepo.ReadAll(options)
+ Expect(err).ToNot(HaveOccurred())
+ tags := result.(model.TagList)
+ Expect(tags).To(HaveLen(2)) // "rock" and "Alternative Rock"
+
+ // Verify all returned tags contain 'rock' in their value
+ for _, tag := range tags {
+ Expect(strings.ToLower(tag.TagValue)).To(ContainSubstring("rock"))
+ }
+ })
+
+ It("should filter tags by partial value using LIKE", func() {
+ options := rest.QueryOptions{
+ Filters: map[string]interface{}{"name": "%e%"}, // Tags containing 'e'
+ }
+ result, err := restRepo.ReadAll(options)
+ Expect(err).ToNot(HaveOccurred())
+ tags := result.(model.TagList)
+ Expect(tags).To(HaveLen(8)) // electronic, house, trance, energetic, Blues, decade x2, Alternative Rock
+
+ // Verify all returned tags contain 'e' in their value
+ for _, tag := range tags {
+ Expect(strings.ToLower(tag.TagValue)).To(ContainSubstring("e"))
+ }
+ })
+
+ It("should sort tags by value ascending", func() {
+ options := rest.QueryOptions{
+ Filters: map[string]interface{}{"name": "%r%"}, // Tags containing 'r'
+ Sort: "name",
+ Order: "asc",
+ }
+ result, err := restRepo.ReadAll(options)
+ Expect(err).ToNot(HaveOccurred())
+ tags := result.(model.TagList)
+ Expect(tags).To(HaveLen(7))
+
+ Expect(slices.IsSortedFunc(tags, func(a, b model.Tag) int {
+ return strings.Compare(strings.ToLower(a.TagValue), strings.ToLower(b.TagValue))
+ }))
+ })
+
+ It("should sort tags by value descending", func() {
+ options := rest.QueryOptions{
+ Filters: map[string]interface{}{"name": "%r%"}, // Tags containing 'r'
+ Sort: "name",
+ Order: "desc",
+ }
+ result, err := restRepo.ReadAll(options)
+ Expect(err).ToNot(HaveOccurred())
+ tags := result.(model.TagList)
+ Expect(tags).To(HaveLen(7))
+
+ Expect(slices.IsSortedFunc(tags, func(a, b model.Tag) int {
+ return strings.Compare(strings.ToLower(b.TagValue), strings.ToLower(a.TagValue)) // Descending order
+ }))
+ })
+ })
+
+ Describe("EntityName", func() {
+ It("should return correct entity name", func() {
+ name := restRepo.EntityName()
+ Expect(name).To(Equal("tag"))
+ })
+ })
+
+ Describe("NewInstance", func() {
+ It("should return new tag instance", func() {
+ instance := restRepo.NewInstance()
+ Expect(instance).To(BeAssignableToTypeOf(model.Tag{}))
+ })
+ })
+})
diff --git a/persistence/transcoding_repository.go b/persistence/transcoding_repository.go
index bdcbe7262..125f57541 100644
--- a/persistence/transcoding_repository.go
+++ b/persistence/transcoding_repository.go
@@ -41,7 +41,7 @@ func (r *transcodingRepository) FindByFormat(format string) (*model.Transcoding,
}
func (r *transcodingRepository) Put(t *model.Transcoding) error {
- if !isAdmin(r.ctx) {
+ if !loggedUser(r.ctx).IsAdmin {
return rest.ErrPermissionDenied
}
_, err := r.put(t.ID, t)
@@ -72,7 +72,7 @@ func (r *transcodingRepository) NewInstance() interface{} {
}
func (r *transcodingRepository) Save(entity interface{}) (string, error) {
- if !isAdmin(r.ctx) {
+ if !loggedUser(r.ctx).IsAdmin {
return "", rest.ErrPermissionDenied
}
t := entity.(*model.Transcoding)
@@ -84,7 +84,7 @@ func (r *transcodingRepository) Save(entity interface{}) (string, error) {
}
func (r *transcodingRepository) Update(id string, entity interface{}, cols ...string) error {
- if !isAdmin(r.ctx) {
+ if !loggedUser(r.ctx).IsAdmin {
return rest.ErrPermissionDenied
}
t := entity.(*model.Transcoding)
@@ -97,7 +97,7 @@ func (r *transcodingRepository) Update(id string, entity interface{}, cols ...st
}
func (r *transcodingRepository) Delete(id string) error {
- if !isAdmin(r.ctx) {
+ if !loggedUser(r.ctx).IsAdmin {
return rest.ErrPermissionDenied
}
err := r.delete(Eq{"id": id})
diff --git a/persistence/user_repository.go b/persistence/user_repository.go
index 073e32963..7baa8f6a8 100644
--- a/persistence/user_repository.go
+++ b/persistence/user_repository.go
@@ -3,6 +3,7 @@ package persistence
import (
"context"
"crypto/sha256"
+ "encoding/json"
"errors"
"fmt"
"strings"
@@ -17,6 +18,7 @@ import (
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/model/id"
"github.com/navidrome/navidrome/utils"
+ "github.com/navidrome/navidrome/utils/slice"
"github.com/pocketbase/dbx"
)
@@ -24,6 +26,26 @@ type userRepository struct {
sqlRepository
}
+type dbUser struct {
+ *model.User `structs:",flatten"`
+ LibrariesJSON string `structs:"-" json:"-"`
+}
+
+func (u *dbUser) PostScan() error {
+ if u.LibrariesJSON != "" {
+ if err := json.Unmarshal([]byte(u.LibrariesJSON), &u.User.Libraries); err != nil {
+ return fmt.Errorf("parsing user libraries from db: %w", err)
+ }
+ }
+ return nil
+}
+
+type dbUsers []dbUser
+
+func (us dbUsers) toModels() model.Users {
+ return slice.Map(us, func(u dbUser) model.User { return *u.User })
+}
+
var (
once sync.Once
encKey []byte
@@ -33,8 +55,11 @@ func NewUserRepository(ctx context.Context, db dbx.Builder) model.UserRepository
r := &userRepository{}
r.ctx = ctx
r.db = db
+ r.tableName = "user"
r.registerModel(&model.User{}, map[string]filterFunc{
+ "id": idFilter(r.tableName),
"password": invalidFilter(ctx),
+ "name": r.withTableName(startsWithFilter),
})
once.Do(func() {
_ = r.initPasswordEncryptionKey()
@@ -42,28 +67,48 @@ func NewUserRepository(ctx context.Context, db dbx.Builder) model.UserRepository
return r
}
+// selectUserWithLibraries returns a SelectBuilder that includes library information
+func (r *userRepository) selectUserWithLibraries(options ...model.QueryOptions) SelectBuilder {
+ return r.newSelect(options...).
+ Columns(`user.*`,
+ `COALESCE(json_group_array(json_object(
+ 'id', library.id,
+ 'name', library.name,
+ 'path', library.path,
+ 'remote_path', library.remote_path,
+ 'last_scan_at', library.last_scan_at,
+ 'last_scan_started_at', library.last_scan_started_at,
+ 'full_scan_in_progress', library.full_scan_in_progress,
+ 'updated_at', library.updated_at,
+ 'created_at', library.created_at
+ )) FILTER (WHERE library.id IS NOT NULL), '[]') AS libraries_json`).
+ LeftJoin("user_library ul ON user.id = ul.user_id").
+ LeftJoin("library ON ul.library_id = library.id").
+ GroupBy("user.id")
+}
+
func (r *userRepository) CountAll(qo ...model.QueryOptions) (int64, error) {
return r.count(Select(), qo...)
}
func (r *userRepository) Get(id string) (*model.User, error) {
- sel := r.newSelect().Columns("*").Where(Eq{"id": id})
- var res model.User
+ sel := r.selectUserWithLibraries().Where(Eq{"user.id": id})
+ var res dbUser
err := r.queryOne(sel, &res)
if err != nil {
return nil, err
}
- return &res, nil
+ return res.User, nil
}
func (r *userRepository) GetAll(options ...model.QueryOptions) (model.Users, error) {
- sel := r.newSelect(options...).Columns("*")
- res := model.Users{}
+ sel := r.selectUserWithLibraries(options...)
+ var res dbUsers
err := r.queryAll(sel, &res)
if err != nil {
return nil, err
}
- return res, nil
+ return res.toModels(), nil
}
func (r *userRepository) Put(u *model.User) error {
@@ -79,38 +124,65 @@ func (r *userRepository) Put(u *model.User) error {
return fmt.Errorf("error converting user to SQL args: %w", err)
}
delete(values, "current_password")
+
+ // Save/update the user
update := Update(r.tableName).Where(Eq{"id": u.ID}).SetMap(values)
count, err := r.executeSQL(update)
if err != nil {
return err
}
- if count > 0 {
- return nil
+
+ isNewUser := count == 0
+ if isNewUser {
+ values["created_at"] = time.Now()
+ insert := Insert(r.tableName).SetMap(values)
+ _, err = r.executeSQL(insert)
+ if err != nil {
+ return err
+ }
}
- values["created_at"] = time.Now()
- insert := Insert(r.tableName).SetMap(values)
- _, err = r.executeSQL(insert)
- return err
+
+ // Auto-assign all libraries to admin users in a single SQL operation
+ if u.IsAdmin {
+ sql := Expr(
+ "INSERT OR IGNORE INTO user_library (user_id, library_id) SELECT ?, id FROM library",
+ u.ID,
+ )
+ if _, err := r.executeSQL(sql); err != nil {
+ return fmt.Errorf("failed to assign all libraries to admin user: %w", err)
+ }
+ } else if isNewUser { // Only for new regular users
+ // Auto-assign default libraries to new regular users
+ sql := Expr(
+ "INSERT OR IGNORE INTO user_library (user_id, library_id) SELECT ?, id FROM library WHERE default_new_users = true",
+ u.ID,
+ )
+ if _, err := r.executeSQL(sql); err != nil {
+ return fmt.Errorf("failed to assign default libraries to new user: %w", err)
+ }
+ }
+
+ return nil
}
func (r *userRepository) FindFirstAdmin() (*model.User, error) {
- sel := r.newSelect(model.QueryOptions{Sort: "updated_at", Max: 1}).Columns("*").Where(Eq{"is_admin": true})
- var usr model.User
+ sel := r.selectUserWithLibraries(model.QueryOptions{Sort: "updated_at", Max: 1}).Where(Eq{"user.is_admin": true})
+ var usr dbUser
err := r.queryOne(sel, &usr)
if err != nil {
return nil, err
}
- return &usr, nil
+ return usr.User, nil
}
func (r *userRepository) FindByUsername(username string) (*model.User, error) {
- sel := r.newSelect().Columns("*").Where(Expr("user_name = ? COLLATE NOCASE", username))
- var usr model.User
+ sel := r.selectUserWithLibraries().Where(Expr("user.user_name = ? COLLATE NOCASE", username))
+ var usr dbUser
err := r.queryOne(sel, &usr)
if err != nil {
return nil, err
}
- return &usr, nil
+ return usr.User, nil
}
func (r *userRepository) FindByUsernameWithPassword(username string) (*model.User, error) {
@@ -365,6 +437,39 @@ func (r *userRepository) decryptAllPasswords(users model.Users) error {
return nil
}
+// Library association methods
+
+func (r *userRepository) GetUserLibraries(userID string) (model.Libraries, error) {
+ sel := Select("l.*").
+ From("library l").
+ Join("user_library ul ON l.id = ul.library_id").
+ Where(Eq{"ul.user_id": userID}).
+ OrderBy("l.name")
+
+ var res model.Libraries
+ err := r.queryAll(sel, &res)
+ return res, err
+}
+
+func (r *userRepository) SetUserLibraries(userID string, libraryIDs []int) error {
+ // Remove existing associations
+ delSql := Delete("user_library").Where(Eq{"user_id": userID})
+ if _, err := r.executeSQL(delSql); err != nil {
+ return err
+ }
+
+ // Add new associations
+ if len(libraryIDs) > 0 {
+ insert := Insert("user_library").Columns("user_id", "library_id")
+ for _, libID := range libraryIDs {
+ insert = insert.Values(userID, libID)
+ }
+ _, err := r.executeSQL(insert)
+ return err
+ }
+ return nil
+}
+
var _ model.UserRepository = (*userRepository)(nil)
var _ rest.Repository = (*userRepository)(nil)
var _ rest.Persistable = (*userRepository)(nil)
diff --git a/persistence/user_repository_test.go b/persistence/user_repository_test.go
index 7b1ad79d7..8abbf76a9 100644
--- a/persistence/user_repository_test.go
+++ b/persistence/user_repository_test.go
@@ -3,7 +3,9 @@ package persistence
import (
"context"
"errors"
+ "slices"
+ "github.com/Masterminds/squirrel"
"github.com/deluan/rest"
"github.com/navidrome/navidrome/consts"
"github.com/navidrome/navidrome/log"
@@ -18,7 +20,7 @@ var _ = Describe("UserRepository", func() {
var repo model.UserRepository
BeforeEach(func() {
- repo = NewUserRepository(log.NewContext(context.TODO()), GetDBXBuilder())
+ repo = NewUserRepository(log.NewContext(GinkgoT().Context()), GetDBXBuilder())
})
Describe("Put/Get/FindByUsername", func() {
@@ -79,7 +81,7 @@ var _ = Describe("UserRepository", func() {
It("does nothing if passwords are not specified", func() {
user := &model.User{ID: "2", UserName: "johndoe"}
err := validatePasswordChange(user, loggedUser)
- Expect(err).To(BeNil())
+ Expect(err).ToNot(HaveOccurred())
})
Context("Autogenerated password (used with Reverse Proxy Authentication)", func() {
@@ -91,7 +93,7 @@ var _ = Describe("UserRepository", func() {
It("does nothing if passwords are not specified", func() {
user = *loggedUser
err := validatePasswordChange(&user, loggedUser)
- Expect(err).To(BeNil())
+ Expect(err).ToNot(HaveOccurred())
})
It("does not requires currentPassword for regular user", func() {
user = *loggedUser
@@ -118,7 +120,7 @@ var _ = Describe("UserRepository", func() {
user := &model.User{ID: "2", UserName: "johndoe"}
user.NewPassword = "new"
err := validatePasswordChange(user, loggedUser)
- Expect(err).To(BeNil())
+ Expect(err).ToNot(HaveOccurred())
})
It("requires currentPassword to change its own", func() {
user := *loggedUser
@@ -156,7 +158,7 @@ var _ = Describe("UserRepository", func() {
user.CurrentPassword = "abc123"
user.NewPassword = "new"
err := validatePasswordChange(&user, loggedUser)
- Expect(err).To(BeNil())
+ Expect(err).ToNot(HaveOccurred())
})
})
@@ -200,10 +202,11 @@ var _ = Describe("UserRepository", func() {
user.CurrentPassword = "abc123"
user.NewPassword = "new"
err := validatePasswordChange(&user, loggedUser)
- Expect(err).To(BeNil())
+ Expect(err).ToNot(HaveOccurred())
})
})
})
+
Describe("validateUsernameUnique", func() {
var repo *tests.MockedUserRepo
var existingUser *model.User
@@ -235,4 +238,336 @@ var _ = Describe("UserRepository", func() {
Expect(err).To(MatchError("fake error"))
})
})
+
+ Describe("Library Association Methods", func() {
+ var userID string
+ var library1, library2 model.Library
+
+ BeforeEach(func() {
+ // Create a test user first to satisfy foreign key constraints
+ testUser := model.User{
+ ID: "test-user-id",
+ UserName: "testuser",
+ Name: "Test User",
+ Email: "test@example.com",
+ NewPassword: "password",
+ IsAdmin: false,
+ }
+ Expect(repo.Put(&testUser)).To(BeNil())
+ userID = testUser.ID
+
+ library1 = model.Library{ID: 0, Name: "Library 500", Path: "/path/500"}
+ library2 = model.Library{ID: 0, Name: "Library 501", Path: "/path/501"}
+
+ // Create test libraries
+ libRepo := NewLibraryRepository(log.NewContext(context.TODO()), GetDBXBuilder())
+ Expect(libRepo.Put(&library1)).To(BeNil())
+ Expect(libRepo.Put(&library2)).To(BeNil())
+ })
+
+ AfterEach(func() {
+ // Clean up user-library associations to ensure test isolation
+ _ = repo.SetUserLibraries(userID, []int{})
+
+ // Clean up test libraries to ensure isolation between test groups
+ libRepo := NewLibraryRepository(log.NewContext(context.TODO()), GetDBXBuilder())
+ _ = libRepo.(*libraryRepository).delete(squirrel.Eq{"id": []int{library1.ID, library2.ID}})
+ })
+
+ Describe("GetUserLibraries", func() {
+ It("returns empty list when user has no library associations", func() {
+ libraries, err := repo.GetUserLibraries("non-existent-user")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(libraries).To(HaveLen(0))
+ })
+
+ It("returns user's associated libraries", func() {
+ err := repo.SetUserLibraries(userID, []int{library1.ID, library2.ID})
+ Expect(err).ToNot(HaveOccurred())
+
+ libraries, err := repo.GetUserLibraries(userID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(libraries).To(HaveLen(2))
+
+ libIDs := []int{libraries[0].ID, libraries[1].ID}
+ Expect(libIDs).To(ContainElements(library1.ID, library2.ID))
+ })
+ })
+
+ Describe("SetUserLibraries", func() {
+ It("sets user's library associations", func() {
+ libraryIDs := []int{library1.ID, library2.ID}
+ err := repo.SetUserLibraries(userID, libraryIDs)
+ Expect(err).ToNot(HaveOccurred())
+
+ libraries, err := repo.GetUserLibraries(userID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(libraries).To(HaveLen(2))
+ })
+
+ It("replaces existing associations", func() {
+ // Set initial associations
+ err := repo.SetUserLibraries(userID, []int{library1.ID, library2.ID})
+ Expect(err).ToNot(HaveOccurred())
+
+ // Replace with just one library
+ err = repo.SetUserLibraries(userID, []int{library1.ID})
+ Expect(err).ToNot(HaveOccurred())
+
+ libraries, err := repo.GetUserLibraries(userID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(libraries).To(HaveLen(1))
+ Expect(libraries[0].ID).To(Equal(library1.ID))
+ })
+
+ It("removes all associations when passed empty slice", func() {
+ // Set initial associations
+ err := repo.SetUserLibraries(userID, []int{library1.ID, library2.ID})
+ Expect(err).ToNot(HaveOccurred())
+
+ // Remove all
+ err = repo.SetUserLibraries(userID, []int{})
+ Expect(err).ToNot(HaveOccurred())
+
+ libraries, err := repo.GetUserLibraries(userID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(libraries).To(HaveLen(0))
+ })
+ })
+ })
+
+ Describe("Admin User Auto-Assignment", func() {
+ var (
+ libRepo model.LibraryRepository
+ library1 model.Library
+ library2 model.Library
+ initialLibCount int
+ )
+
+ BeforeEach(func() {
+ libRepo = NewLibraryRepository(log.NewContext(context.TODO()), GetDBXBuilder())
+
+ // Count initial libraries
+ existingLibs, err := libRepo.GetAll()
+ Expect(err).ToNot(HaveOccurred())
+ initialLibCount = len(existingLibs)
+
+ library1 = model.Library{ID: 0, Name: "Admin Test Library 1", Path: "/admin/test/path1"}
+ library2 = model.Library{ID: 0, Name: "Admin Test Library 2", Path: "/admin/test/path2"}
+
+ // Create test libraries
+ Expect(libRepo.Put(&library1)).To(BeNil())
+ Expect(libRepo.Put(&library2)).To(BeNil())
+ })
+
+ AfterEach(func() {
+ // Clean up test libraries and their associations
+ _ = libRepo.(*libraryRepository).delete(squirrel.Eq{"id": []int{library1.ID, library2.ID}})
+
+ // Clean up user-library associations for these test libraries
+ _, _ = repo.(*userRepository).executeSQL(squirrel.Delete("user_library").Where(squirrel.Eq{"library_id": []int{library1.ID, library2.ID}}))
+ })
+
+ It("automatically assigns all libraries to admin users when created", func() {
+ adminUser := model.User{
+ ID: "admin-user-id-1",
+ UserName: "adminuser1",
+ Name: "Admin User",
+ Email: "admin1@example.com",
+ NewPassword: "password",
+ IsAdmin: true,
+ }
+
+ err := repo.Put(&adminUser)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Admin should automatically have access to all libraries (including existing ones)
+ libraries, err := repo.GetUserLibraries(adminUser.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(libraries).To(HaveLen(initialLibCount + 2)) // Initial libraries + our 2 test libraries
+
+ libIDs := make([]int, len(libraries))
+ for i, lib := range libraries {
+ libIDs[i] = lib.ID
+ }
+ Expect(libIDs).To(ContainElements(library1.ID, library2.ID))
+ })
+
+ It("automatically assigns all libraries to admin users when updated", func() {
+ // Create regular user first
+ regularUser := model.User{
+ ID: "regular-user-id-1",
+ UserName: "regularuser1",
+ Name: "Regular User",
+ Email: "regular1@example.com",
+ NewPassword: "password",
+ IsAdmin: false,
+ }
+
+ err := repo.Put(®ularUser)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Give them access to just one library
+ err = repo.SetUserLibraries(regularUser.ID, []int{library1.ID})
+ Expect(err).ToNot(HaveOccurred())
+
+ // Promote to admin
+ regularUser.IsAdmin = true
+ err = repo.Put(®ularUser)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Should now have access to all libraries (including existing ones)
+ libraries, err := repo.GetUserLibraries(regularUser.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(libraries).To(HaveLen(initialLibCount + 2)) // Initial libraries + our 2 test libraries
+
+ libIDs := make([]int, len(libraries))
+ for i, lib := range libraries {
+ libIDs[i] = lib.ID
+ }
+ // Should include our test libraries plus all existing ones
+ Expect(libIDs).To(ContainElements(library1.ID, library2.ID))
+ })
+
+ It("assigns default libraries to regular users", func() {
+ regularUser := model.User{
+ ID: "regular-user-id-2",
+ UserName: "regularuser2",
+ Name: "Regular User",
+ Email: "regular2@example.com",
+ NewPassword: "password",
+ IsAdmin: false,
+ }
+
+ err := repo.Put(®ularUser)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Regular user should be assigned to default libraries (library ID 1 from migration)
+ libraries, err := repo.GetUserLibraries(regularUser.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(libraries).To(HaveLen(1))
+ Expect(libraries[0].ID).To(Equal(1))
+ Expect(libraries[0].DefaultNewUsers).To(BeTrue())
+ })
+ })
+
+ Describe("Libraries Field Population", func() {
+ var (
+ libRepo model.LibraryRepository
+ library1 model.Library
+ library2 model.Library
+ testUser model.User
+ )
+
+ BeforeEach(func() {
+ libRepo = NewLibraryRepository(log.NewContext(context.TODO()), GetDBXBuilder())
+ library1 = model.Library{ID: 0, Name: "Field Test Library 1", Path: "/field/test/path1"}
+ library2 = model.Library{ID: 0, Name: "Field Test Library 2", Path: "/field/test/path2"}
+
+ // Create test libraries
+ Expect(libRepo.Put(&library1)).To(BeNil())
+ Expect(libRepo.Put(&library2)).To(BeNil())
+
+ // Create test user
+ testUser = model.User{
+ ID: "field-test-user",
+ UserName: "fieldtestuser",
+ Name: "Field Test User",
+ Email: "fieldtest@example.com",
+ NewPassword: "password",
+ IsAdmin: false,
+ }
+ Expect(repo.Put(&testUser)).To(BeNil())
+
+ // Assign libraries to user
+ Expect(repo.SetUserLibraries(testUser.ID, []int{library1.ID, library2.ID})).To(BeNil())
+ })
+
+ AfterEach(func() {
+ // Clean up test libraries and their associations
+ _ = libRepo.(*libraryRepository).delete(squirrel.Eq{"id": []int{library1.ID, library2.ID}})
+ _ = repo.(*userRepository).delete(squirrel.Eq{"id": testUser.ID})
+
+ // Clean up user-library associations for these test libraries
+ _, _ = repo.(*userRepository).executeSQL(squirrel.Delete("user_library").Where(squirrel.Eq{"library_id": []int{library1.ID, library2.ID}}))
+ })
+
+ It("populates Libraries field when getting a single user", func() {
+ user, err := repo.Get(testUser.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(user.Libraries).To(HaveLen(2))
+
+ libIDs := []int{user.Libraries[0].ID, user.Libraries[1].ID}
+ Expect(libIDs).To(ContainElements(library1.ID, library2.ID))
+
+ // Check that library details are properly populated
+ for _, lib := range user.Libraries {
+ switch lib.ID {
+ case library1.ID:
+ Expect(lib.Name).To(Equal("Field Test Library 1"))
+ Expect(lib.Path).To(Equal("/field/test/path1"))
+ case library2.ID:
+ Expect(lib.Name).To(Equal("Field Test Library 2"))
+ Expect(lib.Path).To(Equal("/field/test/path2"))
+ }
+ }
+ })
+
+ It("populates Libraries field when getting all users", func() {
+ users, err := repo.(*userRepository).GetAll()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Find our test user in the results
+ found := slices.IndexFunc(users, func(u model.User) bool { return u.ID == testUser.ID })
+ Expect(found).ToNot(Equal(-1))
+
+ foundUser := users[found]
+ Expect(foundUser).ToNot(BeNil())
+ Expect(foundUser.Libraries).To(HaveLen(2))
+
+ libIDs := []int{foundUser.Libraries[0].ID, foundUser.Libraries[1].ID}
+ Expect(libIDs).To(ContainElements(library1.ID, library2.ID))
+ })
+
+ It("populates Libraries field when finding user by username", func() {
+ user, err := repo.FindByUsername(testUser.UserName)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(user.Libraries).To(HaveLen(2))
+
+ libIDs := []int{user.Libraries[0].ID, user.Libraries[1].ID}
+ Expect(libIDs).To(ContainElements(library1.ID, library2.ID))
+ })
+
+ It("returns default Libraries array for new regular users", func() {
+ // Create a user with no explicit library associations - should get default libraries
+ userWithoutLibs := model.User{
+ ID: "no-libs-user",
+ UserName: "nolibsuser",
+ Name: "No Libs User",
+ Email: "nolibs@example.com",
+ NewPassword: "password",
+ IsAdmin: false,
+ }
+ Expect(repo.Put(&userWithoutLibs)).To(BeNil())
+ defer func() { _ = repo.(*userRepository).delete(squirrel.Eq{"id": userWithoutLibs.ID}) }()
+
+ user, err := repo.Get(userWithoutLibs.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(user.Libraries).ToNot(BeNil())
+ // Regular users should be assigned to default libraries (library ID 1 from migration)
+ Expect(user.Libraries).To(HaveLen(1))
+ Expect(user.Libraries[0].ID).To(Equal(1))
+ })
+ })
+
+ Describe("filters", func() {
+ It("qualifies id filter with table name", func() {
+ r := repo.(*userRepository)
+ qo := r.parseRestOptions(r.ctx, rest.QueryOptions{Filters: map[string]any{"id": "123"}})
+ sel := r.selectUserWithLibraries(qo)
+ query, _, err := r.toSQL(sel)
+ Expect(err).NotTo(HaveOccurred())
+ Expect(query).To(ContainSubstring("user.id = {:p0}"))
+ })
+ })
})
diff --git a/plugins/README.md b/plugins/README.md
new file mode 100644
index 000000000..100230cbf
--- /dev/null
+++ b/plugins/README.md
@@ -0,0 +1,1760 @@
+# Navidrome Plugin System
+
+## Overview
+
+Navidrome's plugin system is a WebAssembly (WASM) based extension mechanism that enables developers to expand Navidrome's functionality without modifying the core codebase. The plugin system supports several capabilities that can be implemented by plugins:
+
+1. **MetadataAgent** - For fetching artist and album information, images, etc.
+2. **Scrobbler** - For implementing scrobbling functionality with external services
+3. **SchedulerCallback** - For executing code after a specified delay or on a recurring schedule
+4. **WebSocketCallback** - For interacting with WebSocket endpoints and handling WebSocket events
+5. **LifecycleManagement** - For plugin initialization and configuration (one-time `OnInit` only; not invoked per-request)
+
+## Plugin Architecture
+
+The plugin system is built on the following key components:
+
+### 1. Plugin Manager
+
+The `Manager` (implemented in `plugins/manager.go`) is the core component that:
+
+- Scans for plugins in the configured plugins directory
+- Loads and compiles plugins
+- Provides access to loaded plugins through capability-specific interfaces
+
+### 2. Plugin Protocol
+
+Plugins communicate with Navidrome using Protocol Buffers (protobuf) over a WASM runtime. The protocol is defined in `plugins/api/api.proto` which specifies the capabilities and messages that plugins can implement.
+
+### 3. Plugin Adapters
+
+Adapters bridge between the plugin API and Navidrome's internal interfaces:
+
+- `wasmMediaAgent` adapts `MetadataAgent` to the internal `agents.Interface`
+- `wasmScrobblerPlugin` adapts `Scrobbler` to the internal `scrobbler.Scrobbler`
+- `wasmSchedulerCallback` adapts `SchedulerCallback` to the internal `SchedulerCallback`
+
+* **Plugin Instance Pooling**: Instances are managed in an internal pool (default 8 max, 1m TTL).
+* **WASM Compilation & Caching**: Modules are pre-compiled concurrently (max 2) and cached in `[CacheFolder]/plugins`, reducing startup time. The compilation timeout can be configured via `DevPluginCompilationTimeout` in development.
+
+### 4. Host Services
+
+Navidrome provides host services that plugins can call to access functionality like HTTP requests and scheduling.
+These services are defined in `plugins/host/` and implemented in corresponding host files:
+
+- HTTP service (in `plugins/host_http.go`) for making external requests
+- Scheduler service (in `plugins/host_scheduler.go`) for scheduling timed events
+- Config service (in `plugins/host_config.go`) for accessing plugin-specific configuration
+- WebSocket service (in `plugins/host_websocket.go`) for WebSocket communication
+- Cache service (in `plugins/host_cache.go`) for TTL-based plugin caching
+- Artwork service (in `plugins/host_artwork.go`) for generating public artwork URLs
+- SubsonicAPI service (in `plugins/host_subsonicapi.go`) for accessing Navidrome's Subsonic API
+
+### Available Host Services
+
+The following host services are available to plugins:
+
+#### HttpService
+
+```protobuf
+// HTTP methods available to plugins
+service HttpService {
+ rpc Get(HttpRequest) returns (HttpResponse);
+ rpc Post(HttpRequest) returns (HttpResponse);
+ rpc Put(HttpRequest) returns (HttpResponse);
+ rpc Delete(HttpRequest) returns (HttpResponse);
+ rpc Patch(HttpRequest) returns (HttpResponse);
+ rpc Head(HttpRequest) returns (HttpResponse);
+ rpc Options(HttpRequest) returns (HttpResponse);
+}
+```
+
+#### ConfigService
+
+```protobuf
+service ConfigService {
+ rpc GetPluginConfig(GetPluginConfigRequest) returns (GetPluginConfigResponse);
+}
+```
+
+The ConfigService allows plugins to access plugin-specific configuration. See the [config.proto](host/config/config.proto) file for the full API.
+
+#### ArtworkService
+
+```protobuf
+service ArtworkService {
+ rpc GetArtistUrl(GetArtworkUrlRequest) returns (GetArtworkUrlResponse);
+ rpc GetAlbumUrl(GetArtworkUrlRequest) returns (GetArtworkUrlResponse);
+ rpc GetTrackUrl(GetArtworkUrlRequest) returns (GetArtworkUrlResponse);
+}
+```
+
+Provides methods to get public URLs for artwork images:
+
+- `GetArtistUrl(id string, size int) string`: Returns a public URL for an artist's artwork
+- `GetAlbumUrl(id string, size int) string`: Returns a public URL for an album's artwork
+- `GetTrackUrl(id string, size int) string`: Returns a public URL for a track's artwork
+
+The `size` parameter is optional (use 0 for original size). The URLs returned are based on the server's ShareURL configuration.
+
+Example:
+
+```go
+url := artwork.GetArtistUrl("123", 300) // Get artist artwork URL with size 300px
+url := artwork.GetAlbumUrl("456", 0) // Get album artwork URL in original size
+```
+
+#### CacheService
+
+```protobuf
+service CacheService {
+ // Set a string value in the cache
+ rpc SetString(SetStringRequest) returns (SetResponse);
+
+ // Get a string value from the cache
+ rpc GetString(GetRequest) returns (GetStringResponse);
+
+ // Set an integer value in the cache
+ rpc SetInt(SetIntRequest) returns (SetResponse);
+
+ // Get an integer value from the cache
+ rpc GetInt(GetRequest) returns (GetIntResponse);
+
+ // Set a float value in the cache
+ rpc SetFloat(SetFloatRequest) returns (SetResponse);
+
+ // Get a float value from the cache
+ rpc GetFloat(GetRequest) returns (GetFloatResponse);
+
+ // Set a byte slice value in the cache
+ rpc SetBytes(SetBytesRequest) returns (SetResponse);
+
+ // Get a byte slice value from the cache
+ rpc GetBytes(GetRequest) returns (GetBytesResponse);
+
+ // Remove a value from the cache
+ rpc Remove(RemoveRequest) returns (RemoveResponse);
+
+ // Check if a key exists in the cache
+ rpc Has(HasRequest) returns (HasResponse);
+}
+```
+
+The CacheService provides a TTL-based cache for plugins. Each plugin gets its own isolated cache instance. By default, cached items expire after 24 hours unless a custom TTL is specified.
+
+Key features:
+
+- **Isolated Caches**: Each plugin has its own cache namespace, so different plugins can use the same key names without conflicts
+- **Typed Values**: Store and retrieve values with their proper types (string, int64, float64, or byte slice)
+- **Configurable TTL**: Set custom expiration times per item, or use the default 24-hour TTL
+- **Type Safety**: The system handles type checking, returning "not exists" if there's a type mismatch
+
+Example usage:
+
+```go
+// Store a string value with default TTL (24 hours)
+cacheService.SetString(ctx, &cache.SetStringRequest{
+ Key: "user_preference",
+ Value: "dark_mode",
+})
+
+// Store an integer with custom TTL (5 minutes)
+cacheService.SetInt(ctx, &cache.SetIntRequest{
+ Key: "api_call_count",
+ Value: 42,
+ TtlSeconds: 300, // 5 minutes
+})
+
+// Retrieve a value
+resp, err := cacheService.GetString(ctx, &cache.GetRequest{
+ Key: "user_preference",
+})
+if err != nil {
+ // Handle error
+}
+if resp.Exists {
+ // Use resp.Value
+} else {
+ // Key doesn't exist or has expired
+}
+
+// Check if a key exists
+hasResp, err := cacheService.Has(ctx, &cache.HasRequest{
+ Key: "api_call_count",
+})
+if hasResp.Exists {
+ // Key exists and hasn't expired
+}
+
+// Remove a value
+cacheService.Remove(ctx, &cache.RemoveRequest{
+ Key: "user_preference",
+})
+```
+
+See the [cache.proto](host/cache/cache.proto) file for the full API definition.
+
+#### SchedulerService
+
+The SchedulerService provides a unified interface for scheduling both one-time and recurring tasks, as well as accessing current time information. See the [scheduler.proto](host/scheduler/scheduler.proto) file for the full API.
+
+```protobuf
+service SchedulerService {
+ // One-time event scheduling
+ rpc ScheduleOneTime(ScheduleOneTimeRequest) returns (ScheduleResponse);
+
+ // Recurring event scheduling
+ rpc ScheduleRecurring(ScheduleRecurringRequest) returns (ScheduleResponse);
+
+ // Cancel any scheduled job
+ rpc CancelSchedule(CancelRequest) returns (CancelResponse);
+
+ // Get current time in multiple formats
+ rpc TimeNow(TimeNowRequest) returns (TimeNowResponse);
+}
+```
+
+**Key Features:**
+
+- **One-time scheduling**: Schedule a callback to be executed once after a specified delay.
+- **Recurring scheduling**: Schedule a callback to be executed repeatedly according to a cron expression.
+- **Current time access**: Get the current time in standardized formats for time-based operations.
+
+**TimeNow Function:**
+
+The `TimeNow` function returns the current time in three formats:
+
+```protobuf
+message TimeNowResponse {
+ string rfc3339_nano = 1; // RFC3339 format with nanosecond precision
+ int64 unix_milli = 2; // Unix timestamp in milliseconds
+ string local_time_zone = 3; // Local timezone name (e.g., "UTC", "America/New_York")
+}
+```
+
+This allows plugins to:
+
+- Get high-precision timestamps for logging and event correlation
+- Perform time-based calculations using Unix timestamps
+- Handle timezone-aware operations by knowing the server's local timezone
+
+Example usage:
+
+```go
+// Get current time information
+timeResp, err := scheduler.TimeNow(ctx, &scheduler.TimeNowRequest{})
+if err != nil {
+ return err
+}
+
+// Use the different time formats
+timestamp := timeResp.Rfc3339Nano // "2024-01-15T10:30:45.123456789Z"
+unixMs := timeResp.UnixMilli // 1705312245123
+timezone := timeResp.LocalTimeZone // "UTC"
+```
+
+Plugins using this service must implement the `SchedulerCallback` interface:
+
+```protobuf
+service SchedulerCallback {
+ rpc OnSchedulerCallback(SchedulerCallbackRequest) returns (SchedulerCallbackResponse);
+}
+```
+
+The `IsRecurring` field in the request allows plugins to differentiate between one-time and recurring callbacks.
+
+#### WebSocketService
+
+The WebSocketService enables plugins to connect to and interact with WebSocket endpoints. See the [websocket.proto](host/websocket/websocket.proto) file for the full API.
+
+```protobuf
+service WebSocketService {
+ // Connect to a WebSocket endpoint
+ rpc Connect(ConnectRequest) returns (ConnectResponse);
+
+ // Send a text message
+ rpc SendText(SendTextRequest) returns (SendTextResponse);
+
+ // Send binary data
+ rpc SendBinary(SendBinaryRequest) returns (SendBinaryResponse);
+
+ // Close a connection
+ rpc Close(CloseRequest) returns (CloseResponse);
+}
+```
+
+- **Connect**: Establish a WebSocket connection to a specified URL with optional headers
+- **SendText**: Send text messages over an established connection
+- **SendBinary**: Send binary data over an established connection
+- **Close**: Close a WebSocket connection with optional close code and reason
+
+Plugins using this service must implement the `WebSocketCallback` interface to handle incoming messages and connection events:
+
+```protobuf
+service WebSocketCallback {
+ rpc OnTextMessage(OnTextMessageRequest) returns (OnTextMessageResponse);
+ rpc OnBinaryMessage(OnBinaryMessageRequest) returns (OnBinaryMessageResponse);
+ rpc OnError(OnErrorRequest) returns (OnErrorResponse);
+ rpc OnClose(OnCloseRequest) returns (OnCloseResponse);
+}
+```
+
+Example usage:
+
+```go
+// Connect to a WebSocket server
+connectResp, err := websocket.Connect(ctx, &websocket.ConnectRequest{
+ Url: "wss://example.com/ws",
+ Headers: map[string]string{"Authorization": "Bearer token"},
+ ConnectionId: "my-connection-id",
+})
+if err != nil {
+ return err
+}
+
+// Send a text message
+_, err = websocket.SendText(ctx, &websocket.SendTextRequest{
+ ConnectionId: "my-connection-id",
+ Message: "Hello WebSocket",
+})
+
+// Send binary data
+_, err = websocket.SendBinary(ctx, &websocket.SendBinaryRequest{
+ ConnectionId: "my-connection-id",
+ Data: []byte{0x01, 0x02, 0x03},
+})
+
+// Close the connection when done
+_, err = websocket.Close(ctx, &websocket.CloseRequest{
+ ConnectionId: "my-connection-id",
+ Code: 1000, // Normal closure
+ Reason: "Done",
+})
+```
+
+#### SubsonicAPIService
+
+```protobuf
+service SubsonicAPIService {
+ rpc Call(CallRequest) returns (CallResponse);
+}
+```
+
+The SubsonicAPIService provides plugins with access to Navidrome's Subsonic API endpoints. This allows plugins to query and interact with Navidrome's music library data using the same API that external Subsonic clients use.
+
+Key features:
+
+- **Library Access**: Query artists, albums, tracks, playlists, and other music library data
+- **Search Functionality**: Search across the music library using various criteria
+- **Metadata Retrieval**: Get detailed information about music items including ratings, play counts, etc.
+- **Authentication Handled**: The service automatically handles authentication using internal auth context
+- **JSON Responses**: All responses are returned as JSON strings for easy parsing
+
+**Important Security Notes:**
+
+- Plugins must specify a username via the `u` parameter in the URL - this determines which user's library view and permissions apply
+- The service uses internal authentication, so plugins don't need to provide passwords or API keys
+- All Subsonic API security and access controls apply based on the specified user
+
+Example usage:
+
+```go
+// Get ping response to test connectivity
+resp, err := subsonicAPI.Call(ctx, &subsonicapi.CallRequest{
+ Url: "/rest/ping?u=admin",
+})
+if err != nil {
+ return err
+}
+// resp.Json contains the JSON response
+
+// Search for artists
+resp, err = subsonicAPI.Call(ctx, &subsonicapi.CallRequest{
+ Url: "/rest/search3?u=admin&query=Beatles&artistCount=10",
+})
+
+// Get album details
+resp, err = subsonicAPI.Call(ctx, &subsonicapi.CallRequest{
+ Url: "/rest/getAlbum?u=admin&id=123",
+})
+
+// Check for errors
+if resp.Error != "" {
+ // Handle error - could be missing parameters, invalid user, etc.
+ log.Printf("SubsonicAPI error: %s", resp.Error)
+}
+```
+
+**Common URL Patterns:**
+
+- `/rest/ping?u=USERNAME` - Test API connectivity
+- `/rest/search3?u=USERNAME&query=TERM` - Search library
+- `/rest/getArtists?u=USERNAME` - Get all artists
+- `/rest/getAlbum?u=USERNAME&id=ID` - Get album details
+- `/rest/getPlaylists?u=USERNAME` - Get user playlists
+
+**Required Parameters:**
+
+- `u` (username): Required for all requests - determines user context and permissions
+- `f=json`: Recommended to get JSON responses (easier to parse than XML)
+
+The service accepts standard Subsonic API endpoints and parameters. Refer to the [Subsonic API documentation](http://www.subsonic.org/pages/api.jsp) for complete endpoint details, but note that authentication parameters (`p`, `t`, `s`, `c`, `v`) are handled automatically.
+
+See the [subsonicapi.proto](host/subsonicapi/subsonicapi.proto) file for the full API definition.
+
+## Plugin Permission System
+
+Navidrome implements a permission-based security system that controls which host services plugins can access. This system enforces security at load-time by only making authorized services available to plugins in their WebAssembly runtime environment.
+
+### How Permissions Work
+
+The permission system follows a **secure-by-default** approach:
+
+1. **Default Behavior**: Plugins have access to **no host services** unless explicitly declared
+2. **Load-time Enforcement**: Only services listed in a plugin's permissions are loaded into its WASM runtime
+3. **Runtime Security**: Unauthorized services are completely unavailable - attempts to call them result in "function not exported" errors
+
+This design ensures that even if malicious code tries to access unauthorized services, the calls will fail because the functions simply don't exist in the plugin's runtime environment.
+
+### Permission Syntax
+
+Permissions are declared in the plugin's `manifest.json` file using the `permissions` field as an object:
+
+```json
+{
+ "name": "my-plugin",
+ "author": "Plugin Developer",
+ "version": "1.0.0",
+ "description": "A plugin that fetches data and caches results",
+ "website": "https://github.com/plugindeveloper/my-plugin",
+ "capabilities": ["MetadataAgent"],
+ "permissions": {
+ "http": {
+ "reason": "To fetch metadata from external APIs",
+ "allowedUrls": {
+ "https://api.musicbrainz.org": ["GET"],
+ "https://coverartarchive.org": ["GET"]
+ },
+ "allowLocalNetwork": false
+ },
+ "cache": {
+ "reason": "To cache API responses and reduce rate limiting"
+ },
+ "subsonicapi": {
+ "reason": "To query music library for artist and album information",
+ "allowedUsernames": ["metadata-user"],
+ "allowAdmins": false
+ }
+ }
+}
+```
+
+Each permission is represented as a key in the permissions object. The value must be an object containing a `reason` field that explains why the permission is needed.
+
+**Important**: Some permissions require additional configuration fields:
+
+- **`http`**: Requires `allowedUrls` object mapping URL patterns to allowed HTTP methods, and optional `allowLocalNetwork` boolean
+- **`websocket`**: Requires `allowedUrls` array of WebSocket URL patterns, and optional `allowLocalNetwork` boolean
+- **`subsonicapi`**: Requires `reason` field, with optional `allowedUsernames` array and `allowAdmins` boolean for fine-grained access control
+- **`config`**, **`cache`**, **`scheduler`**, **`artwork`**: Only require the `reason` field
+
+**Security Benefits of Required Reasons:**
+
+- **Transparency**: Users can see exactly what each plugin will do with its permissions
+- **Security Auditing**: Makes it easier to identify suspicious or overly broad permission requests
+- **Developer Accountability**: Forces plugin authors to justify each permission they request
+- **Trust Building**: Clear explanations help users make informed decisions about plugin installation
+
+If no permissions are needed, use an empty permissions object: `"permissions": {}`.
+
+### Available Permissions
+
+The following permission keys correspond to host services:
+
+| Permission | Host Service | Description | Required Fields |
+| ------------- | ------------------ | -------------------------------------------------- | ----------------------------------------------------- |
+| `http` | HttpService | Make HTTP requests (GET, POST, PUT, DELETE, etc..) | `reason`, `allowedUrls` |
+| `websocket` | WebSocketService | Connect to and communicate via WebSockets | `reason`, `allowedUrls` |
+| `cache` | CacheService | Store and retrieve cached data with TTL | `reason` |
+| `config` | ConfigService | Access Navidrome configuration values | `reason` |
+| `scheduler` | SchedulerService | Schedule one-time and recurring tasks | `reason` |
+| `artwork` | ArtworkService | Generate public URLs for artwork images | `reason` |
+| `subsonicapi` | SubsonicAPIService | Access Navidrome's Subsonic API endpoints | `reason`, optional: `allowedUsernames`, `allowAdmins` |
+
+#### HTTP Permission Structure
+
+HTTP permissions require explicit URL whitelisting for security:
+
+```json
+{
+ "http": {
+ "reason": "To fetch artist data from MusicBrainz and album covers from Cover Art Archive",
+ "allowedUrls": {
+ "https://musicbrainz.org/ws/2/*": ["GET"],
+ "https://coverartarchive.org/*": ["GET"],
+ "https://api.example.com/submit": ["POST"]
+ },
+ "allowLocalNetwork": false
+ }
+}
+```
+
+**Fields:**
+
+- `reason` (required): Explanation of why HTTP access is needed
+- `allowedUrls` (required): Object mapping URL patterns to allowed HTTP methods
+- `allowLocalNetwork` (optional, default false): Whether to allow requests to localhost/private IPs
+
+**URL Pattern Matching:**
+
+- Exact URLs: `"https://api.example.com/endpoint": ["GET"]`
+- Wildcard paths: `"https://api.example.com/*": ["GET", "POST"]`
+- Subdomain wildcards: `"https://*.example.com": ["GET"]`
+
+**Important**: Redirect destinations must also be included in `allowedUrls` if you want to follow redirects.
+
+#### WebSocket Permission Structure
+
+WebSocket permissions require explicit URL whitelisting:
+
+```json
+{
+ "websocket": {
+ "reason": "To connect to Discord gateway for real-time Rich Presence updates",
+ "allowedUrls": ["wss://gateway.discord.gg", "wss://*.discord.gg"],
+ "allowLocalNetwork": false
+ }
+}
+```
+
+**Fields:**
+
+- `reason` (required): Explanation of why WebSocket access is needed
+- `allowedUrls` (required): Array of WebSocket URL patterns (must start with `ws://` or `wss://`)
+- `allowLocalNetwork` (optional, default false): Whether to allow connections to localhost/private IPs
+
+#### SubsonicAPI Permission Structure
+
+SubsonicAPI permissions control which users plugins can access Navidrome's Subsonic API as, providing fine-grained security controls:
+
+```json
+{
+ "subsonicapi": {
+ "reason": "To query music library data for recommendation engine",
+ "allowedUsernames": ["plugin-user", "readonly-user"],
+ "allowAdmins": false
+ }
+}
+```
+
+**Fields:**
+
+- `reason` (required): Explanation of why SubsonicAPI access is needed
+- `allowedUsernames` (optional): Array of specific usernames the plugin is allowed to use. If empty or omitted, any username can be used
+- `allowAdmins` (optional, default false): Whether the plugin can make API calls using admin user accounts
+
+**Security Model:**
+
+The SubsonicAPI service enforces strict user-based access controls:
+
+- **Username Validation**: The plugin must provide a valid `u` (username) parameter in all API calls
+- **User Context**: All API responses are filtered based on the specified user's permissions and library access
+- **Admin Protection**: By default, plugins cannot use admin accounts for API calls to prevent privilege escalation
+- **Username Restrictions**: When `allowedUsernames` is specified, only those users can be used
+
+**Common Permission Patterns:**
+
+```jsonc
+// Allow any non-admin user (most permissive)
+{
+ "subsonicapi": {
+ "reason": "To search music library for metadata enhancement",
+ "allowAdmins": false
+ }
+}
+
+// Allow only specific users (most secure)
+{
+ "subsonicapi": {
+ "reason": "To access playlists for synchronization with external service",
+ "allowedUsernames": ["sync-user"],
+ "allowAdmins": false
+ }
+}
+
+// Allow admin users (use with caution)
+{
+ "subsonicapi": {
+ "reason": "To perform administrative tasks like library statistics",
+ "allowAdmins": true
+ }
+}
+
+// Restrict to specific users but allow admins
+{
+ "subsonicapi": {
+ "reason": "To backup playlists for authorized users only",
+ "allowedUsernames": ["backup-admin", "user1", "user2"],
+ "allowAdmins": true
+ }
+}
+```
+
+**Important Notes:**
+
+- Username matching is case-insensitive
+- If `allowedUsernames` is empty or omitted, any username can be used (subject to `allowAdmins` setting)
+- Admin restriction (`allowAdmins: false`) is checked after username validation
+- Invalid or non-existent usernames will result in API call errors
+
+### Permission Validation
+
+The plugin system validates permissions during loading:
+
+1. **Schema Validation**: The manifest is validated against the JSON schema
+2. **Permission Recognition**: Unknown permission keys are silently accepted for forward compatibility
+3. **Service Loading**: Only services with corresponding permissions are made available to the plugin
+
+### Security Model
+
+The permission system provides multiple layers of security:
+
+#### 1. Principle of Least Privilege
+
+- Plugins start with zero permissions
+- Only explicitly requested services are available
+- No way to escalate privileges at runtime
+
+#### 2. Load-time Enforcement
+
+- Unauthorized services are not loaded into the WASM runtime
+- No performance overhead for permission checks during execution
+- Impossible to bypass restrictions through code manipulation
+
+#### 3. Service Isolation
+
+- Each plugin gets its own isolated service instances
+- Plugins cannot interfere with each other's service usage
+- Host services are sandboxed within the WASM environment
+
+### Best Practices for Plugin Developers
+
+#### Request Minimal Permissions
+
+```jsonc
+// Good: No permissions if none needed
+{
+ "permissions": {}
+}
+
+// Good: Only request what you need with clear reasoning
+{
+ "permissions": {
+ "http": {
+ "reason": "To fetch artist biography from MusicBrainz database",
+ "allowedUrls": {
+ "https://musicbrainz.org/ws/2/artist/*": ["GET"]
+ },
+ "allowLocalNetwork": false
+ }
+ }
+}
+
+// Avoid: Requesting unnecessary permissions
+{
+ "permissions": {
+ "http": {
+ "reason": "To fetch data",
+ "allowedUrls": {
+ "https://*": ["*"]
+ },
+ "allowLocalNetwork": true
+ },
+ "cache": {
+ "reason": "For caching"
+ },
+ "scheduler": {
+ "reason": "For scheduling"
+ },
+ "websocket": {
+ "reason": "For real-time updates",
+ "allowedUrls": ["wss://*"],
+ "allowLocalNetwork": true
+ }
+ }
+}
+```
+
+#### Write Clear Permission Reasons
+
+Provide specific, descriptive reasons for each permission that explain exactly what the plugin does. Good reasons should:
+
+- Specify **what data** will be accessed/fetched
+- Mention **which external services** will be contacted (if applicable)
+- Explain **why** the permission is necessary for the plugin's functionality
+- Use clear, non-technical language that users can understand
+
+```jsonc
+// Good: Specific and informative
+{
+ "http": {
+ "reason": "To fetch album reviews from AllMusic API and artist biographies from MusicBrainz",
+ "allowedUrls": {
+ "https://www.allmusic.com/api/*": ["GET"],
+ "https://musicbrainz.org/ws/2/*": ["GET"]
+ },
+ "allowLocalNetwork": false
+ },
+ "cache": {
+ "reason": "To cache API responses for 24 hours to respect rate limits and improve performance"
+ }
+}
+
+// Bad: Vague and unhelpful
+{
+ "http": {
+ "reason": "To make requests",
+ "allowedUrls": {
+ "https://*": ["*"]
+ },
+ "allowLocalNetwork": true
+ },
+ "cache": {
+ "reason": "For caching"
+ }
+}
+```
+
+#### Handle Missing Permissions Gracefully
+
+Your plugin should provide clear error messages when permissions are missing:
+
+```go
+func (p *Plugin) GetArtistInfo(ctx context.Context, req *api.ArtistInfoRequest) (*api.ArtistInfoResponse, error) {
+ // This will fail with "function not exported" if http permission is missing
+ resp, err := p.httpClient.Get(ctx, &http.HttpRequest{Url: apiURL})
+ if err != nil {
+ // Check if it's a permission error
+ if strings.Contains(err.Error(), "not exported") {
+ return &api.ArtistInfoResponse{
+ Error: "Plugin requires 'http' permission (reason: 'To fetch artist metadata from external APIs') - please add to manifest.json",
+ }, nil
+ }
+ return &api.ArtistInfoResponse{Error: err.Error()}, nil
+ }
+ // ... process response
+}
+```
+
+### Troubleshooting Permissions
+
+#### Common Error Messages
+
+**"function not exported in module env"**
+
+- Cause: Plugin trying to call a service without proper permission
+- Solution: Add the required permission to your manifest.json
+
+**"manifest validation failed" or "missing required field"**
+
+- Cause: Plugin manifest is missing required fields (e.g., `allowedUrls` for HTTP/WebSocket permissions)
+- Solution: Ensure your manifest includes all required fields for each permission type
+
+**Permission silently ignored**
+
+- Cause: Using a permission key not recognized by current Navidrome version
+- Effect: The unknown permission is silently ignored (no error or warning)
+- Solution: This is actually normal behavior for forward compatibility
+
+#### Debugging Permission Issues
+
+1. **Check the manifest**: Ensure required permissions are spelled correctly and present
+2. **Verify required fields**: Check that HTTP and WebSocket permissions include `allowedUrls` and other required fields
+3. **Review logs**: Check for plugin loading errors, manifest validation errors, and WASM runtime errors
+4. **Test incrementally**: Add permissions one at a time to identify which services your plugin needs
+5. **Verify service names**: Ensure permission keys match exactly: `http`, `cache`, `config`, `scheduler`, `websocket`, `artwork`, `subsonicapi`
+6. **Validate manifest**: Use a JSON schema validator to check your manifest against the schema
+
+### Future Considerations
+
+The permission system is designed for extensibility:
+
+- **Unknown permissions** are allowed in manifests for forward compatibility
+- **New services** can be added with corresponding permission keys
+- **Permission scoping** could be added in the future (e.g., read-only vs. read-write access)
+
+This ensures that plugins developed today will continue to work as the system evolves, while maintaining strong security boundaries.
+
+## Plugin System Implementation
+
+Navidrome's plugin system is built using the following key libraries:
+
+### 1. WebAssembly Runtime (Wazero)
+
+The plugin system uses [Wazero](https://github.com/tetratelabs/wazero), a WebAssembly runtime written in pure Go. Wazero was chosen for several reasons:
+
+- **No CGO dependency**: Unlike other WebAssembly runtimes, Wazero is implemented in pure Go, which simplifies cross-compilation and deployment.
+- **Performance**: It provides efficient compilation and caching of WebAssembly modules.
+- **Security**: Wazero enforces strict sandboxing, which is important for running third-party plugin code safely.
+
+The plugin manager uses Wazero to:
+
+- Compile and cache WebAssembly modules
+- Create isolated runtime environments for each plugin
+- Instantiate plugin modules when they're called
+- Provide host functions that plugins can call
+
+### 2. Go-plugin Framework
+
+Navidrome builds on [go-plugin](https://github.com/knqyf263/go-plugin), a Go plugin system over WebAssembly that provides:
+
+- **Code generation**: Custom Protocol Buffer compiler plugin (`protoc-gen-go-plugin`) that generates Go code for both the host and WebAssembly plugins
+- **Host function system**: Framework for exposing host functionality to plugins safely
+- **Interface versioning**: Built-in mechanism for handling API compatibility between the host and plugins
+- **Type conversion**: Utilities for marshaling and unmarshaling data between Go and WebAssembly
+
+This framework significantly simplifies plugin development by handling the low-level details of WebAssembly communication, allowing plugin developers to focus on implementing capabilities interfaces.
+
+### 3. Protocol Buffers (Protobuf)
+
+[Protocol Buffers](https://developers.google.com/protocol-buffers) serve as the interface definition language for the plugin system. Navidrome uses:
+
+- **protoc-gen-go-plugin**: A custom protobuf compiler plugin that generates Go code for both the Navidrome host and WebAssembly plugins
+- Protobuf messages for structured data exchange between the host and plugins
+
+The protobuf definitions are located in:
+
+- `plugins/api/api.proto`: Core plugin capability interfaces
+- `plugins/host/http/http.proto`: HTTP service interface
+- `plugins/host/scheduler/scheduler.proto`: Scheduler service interface
+- `plugins/host/config/config.proto`: Config service interface
+- `plugins/host/websocket/websocket.proto`: WebSocket service interface
+- `plugins/host/cache/cache.proto`: Cache service interface
+- `plugins/host/artwork/artwork.proto`: Artwork service interface
+- `plugins/host/subsonicapi/subsonicapi.proto`: SubsonicAPI service interface
+
+### 4. Integration Architecture
+
+The plugin system integrates these libraries through several key components:
+
+- **Plugin Manager**: Manages the lifecycle of plugins, from discovery to loading
+- **Compilation Cache**: Improves performance by caching compiled WebAssembly modules
+- **Host Function Bridge**: Exposes Navidrome functionality to plugins through WebAssembly imports
+- **Capability Adapters**: Convert between the plugin API and Navidrome's internal interfaces
+
+Each plugin method call:
+
+1. Creates a new isolated plugin instance using Wazero
+2. Executes the method in the sandboxed environment
+3. Converts data between Go and WebAssembly formats using the protobuf-generated code
+4. Cleans up the instance after the call completes
+
+This stateless design ensures that plugins remain isolated and can't interfere with Navidrome's core functionality or each other.
+
+## Configuration
+
+Plugins are configured in Navidrome's main configuration via the `Plugins` section:
+
+```toml
+[Plugins]
+# Enable or disable plugin support
+Enabled = true
+
+# Directory where plugins are stored (defaults to [DataFolder]/plugins)
+Folder = "/path/to/plugins"
+```
+
+By default, the plugins folder is created under `[DataFolder]/plugins` with restrictive permissions (`0700`) to limit access to the Navidrome user.
+
+### Plugin-specific Configuration
+
+You can also provide plugin-specific configuration using the `PluginConfig` section. Each plugin can have its own configuration map using the **folder name** as the key:
+
+```toml
+[PluginConfig.my-plugin-folder]
+api_key = "your-api-key"
+user_id = "your-user-id"
+enable_feature = "true"
+
+[PluginConfig.another-plugin-folder]
+server_url = "https://example.com/api"
+timeout = "30"
+```
+
+These configuration values are passed to plugins during initialization through the `OnInit` method in the `LifecycleManagement` capability.
+Plugins that implement the `LifecycleManagement` capability will receive their configuration as a map of string keys and values.
+
+## Plugin Directory Structure
+
+Each plugin must be located in its own directory under the plugins folder:
+
+```
+plugins/
+├── my-plugin/
+│ ├── plugin.wasm # Compiled WebAssembly module
+│ └── manifest.json # Plugin manifest defining metadata and capabilities
+├── another-plugin/
+│ ├── plugin.wasm
+│ └── manifest.json
+```
+
+**Note**: Plugin identification has changed! Navidrome now uses the **folder name** as the unique identifier for plugins, not the `name` field in `manifest.json`. This means:
+
+- **Multiple plugins can have the same `name` in their manifest**, as long as they are in different folders
+- **Plugin loading and commands use the folder name**, not the manifest name
+- **Folder names must be unique** across all plugins in your plugins directory
+
+This change allows you to have multiple versions or variants of the same plugin (e.g., `lastfm-official`, `lastfm-custom`, `lastfm-dev`) that all have the same manifest name but coexist peacefully.
+
+### Example: Multiple Plugin Variants
+
+```
+plugins/
+├── lastfm-official/
+│ ├── plugin.wasm
+│ └── manifest.json # {"name": "LastFM Agent", ...}
+├── lastfm-custom/
+│ ├── plugin.wasm
+│ └── manifest.json # {"name": "LastFM Agent", ...}
+└── lastfm-dev/
+ ├── plugin.wasm
+ └── manifest.json # {"name": "LastFM Agent", ...}
+```
+
+All three plugins can have the same `"name": "LastFM Agent"` in their manifest, but they are identified and loaded by their folder names:
+
+```bash
+# Load specific variants
+navidrome plugin refresh lastfm-official
+navidrome plugin refresh lastfm-custom
+navidrome plugin refresh lastfm-dev
+
+# Configure each variant separately
+[PluginConfig.lastfm-official]
+api_key = "production-key"
+
+[PluginConfig.lastfm-dev]
+api_key = "development-key"
+```
+
+### Using Symlinks for Plugin Variants
+
+Symlinks provide a powerful way to create multiple configurations for the same plugin without duplicating files. When you create a symlink to a plugin directory, Navidrome treats the symlink as a separate plugin with its own configuration.
+
+**Example: Discord Rich Presence with Multiple Configurations**
+
+```bash
+# Create symlinks for different environments
+cd /path/to/navidrome/plugins
+ln -s /path/to/discord-rich-presence-plugin drp-prod
+ln -s /path/to/discord-rich-presence-plugin drp-dev
+ln -s /path/to/discord-rich-presence-plugin drp-test
+```
+
+Directory structure:
+
+```
+plugins/
+├── drp-prod -> /path/to/discord-rich-presence-plugin/
+├── drp-dev -> /path/to/discord-rich-presence-plugin/
+├── drp-test -> /path/to/discord-rich-presence-plugin/
+```
+
+Each symlink can have its own configuration:
+
+```toml
+[PluginConfig.drp-prod]
+clientid = "production-client-id"
+users = "admin:prod-token"
+
+[PluginConfig.drp-dev]
+clientid = "development-client-id"
+users = "admin:dev-token,testuser:test-token"
+
+[PluginConfig.drp-test]
+clientid = "test-client-id"
+users = "testuser:test-token"
+```
+
+**Key Benefits:**
+
+- **Single Source**: One plugin implementation serves multiple use cases
+- **Independent Configuration**: Each symlink has its own configuration namespace
+- **Development Workflow**: Easy to test different configurations without code changes
+- **Resource Sharing**: All symlinks share the same compiled WASM binary
+
+**Important Notes:**
+
+- The **symlink name** (not the target folder name) is used as the plugin ID
+- Configuration keys use the symlink name: `PluginConfig.`
+- Each symlink appears as a separate plugin in `navidrome plugin list`
+- CLI commands use the symlink name: `navidrome plugin refresh drp-dev`
+
+## Plugin Package Format (.ndp)
+
+Navidrome Plugin Packages (.ndp) are ZIP archives that bundle all files needed for a plugin. They can be installed using the `navidrome plugin install` command.
+
+### Package Structure
+
+A valid .ndp file must contain:
+
+```
+plugin-name.ndp (ZIP file)
+├── plugin.wasm # Required: The compiled WebAssembly module
+├── manifest.json # Required: Plugin manifest with metadata
+├── README.md # Optional: Documentation
+└── LICENSE # Optional: License information
+```
+
+### Creating a Plugin Package
+
+To create a plugin package:
+
+1. Compile your plugin to WebAssembly (plugin.wasm)
+2. Create a manifest.json file with required fields
+3. Include any documentation files you want to bundle
+4. Create a ZIP archive of all files
+5. Rename the ZIP file to have a .ndp extension
+
+### Installing a Plugin Package
+
+Use the Navidrome CLI to install plugins:
+
+```bash
+navidrome plugin install /path/to/plugin-name.ndp
+```
+
+This will extract the plugin to a directory in your configured plugins folder.
+
+## Plugin Management
+
+Navidrome provides a command-line interface for managing plugins. To use these commands, the plugin system must be enabled in your configuration.
+
+### Available Commands
+
+```bash
+# List all installed plugins
+navidrome plugin list
+
+# Show detailed information about a plugin package or installed plugin
+navidrome plugin info plugin-name-or-package.ndp
+
+# Install a plugin from a .ndp file
+navidrome plugin install /path/to/plugin.ndp
+
+# Remove an installed plugin (use folder name)
+navidrome plugin remove plugin-folder-name
+
+# Update an existing plugin
+navidrome plugin update /path/to/updated-plugin.ndp
+
+# Reload a plugin without restarting Navidrome (use folder name)
+navidrome plugin refresh plugin-folder-name
+
+# Create a symlink to a plugin development folder
+navidrome plugin dev /path/to/dev/folder
+```
+
+### Plugin Development
+
+The `dev` and `refresh` commands are particularly useful for plugin development:
+
+#### Development Workflow
+
+1. Create a plugin development folder with required files (`manifest.json` and `plugin.wasm`)
+2. Run `navidrome plugin dev /path/to/your/plugin` to create a symlink in the plugins directory
+3. Make changes to your plugin code
+4. Recompile the WebAssembly module
+5. Run `navidrome plugin refresh your-plugin-folder-name` to reload the plugin without restarting Navidrome
+
+The `dev` command creates a symlink from your development folder to the plugins directory, allowing you to edit the plugin files directly in your development environment without copying them to the plugins directory after each change.
+
+The refresh process:
+
+- Reloads the plugin manifest
+- Recompiles the WebAssembly module
+- Updates the plugin registration
+- Makes the updated plugin immediately available to Navidrome
+
+### Plugin Security
+
+Navidrome provides multiple layers of security for plugin execution:
+
+1. **WebAssembly Sandbox**: Plugins run in isolated WebAssembly environments with no direct system access
+2. **Permission System**: Plugins can only access host services they explicitly request in their manifest (see [Plugin Permission System](#plugin-permission-system))
+3. **File System Security**: The plugins folder is configured with restricted permissions (0700) accessible only by the user running Navidrome
+4. **Resource Isolation**: Each plugin instance is isolated and cannot interfere with other plugins or core Navidrome functionality
+
+The permission system ensures that plugins follow the principle of least privilege - they start with no access to host services and must explicitly declare what they need. This prevents malicious or poorly written plugins from accessing unauthorized functionality.
+
+Always ensure you trust the source of any plugins you install, and review their requested permissions before installation.
+
+## Plugin Manifest
+
+**Capability Names Are Case-Sensitive**: Entries in the `capabilities` array must exactly match one of the supported capabilities: `MetadataAgent`, `Scrobbler`, `SchedulerCallback`, `WebSocketCallback`, or `LifecycleManagement`.
+**Manifest Validation**: The `manifest.json` is validated against the embedded JSON schema (`plugins/schema/manifest.schema.json`). Invalid manifests will be rejected during plugin discovery.
+
+Every plugin must provide a `manifest.json` file that declares metadata, capabilities, and permissions:
+
+```json
+{
+ "name": "my-awesome-plugin",
+ "author": "Your Name",
+ "version": "1.0.0",
+ "description": "A plugin that does awesome things",
+ "website": "https://github.com/yourname/my-awesome-plugin",
+ "capabilities": [
+ "MetadataAgent",
+ "Scrobbler",
+ "SchedulerCallback",
+ "WebSocketCallback",
+ "LifecycleManagement"
+ ],
+ "permissions": {
+ "http": {
+ "reason": "To fetch metadata from external music APIs"
+ },
+ "cache": {
+ "reason": "To cache API responses and reduce rate limiting"
+ },
+ "config": {
+ "reason": "To read API keys and service configuration"
+ },
+ "scheduler": {
+ "reason": "To schedule periodic data refresh tasks"
+ }
+ }
+}
+```
+
+Required fields:
+
+- `name`: Display name of the plugin (used for documentation/display purposes; folder name is used for identification)
+- `author`: The creator or organization behind the plugin
+- `version`: Version identifier (recommended to follow semantic versioning)
+- `description`: A brief description of what the plugin does
+- `website`: Website URL for the plugin documentation, source code, or homepage (must be a valid URI)
+- `capabilities`: Array of capability types the plugin implements
+- `permissions`: Object mapping host service names to their configurations (use empty object `{}` for no permissions)
+
+Currently supported capabilities:
+
+- `MetadataAgent` - For implementing media metadata providers
+- `Scrobbler` - For implementing scrobbling plugins
+- `SchedulerCallback` - For implementing timed callbacks
+- `WebSocketCallback` - For interacting with WebSocket endpoints and handling WebSocket events
+- `LifecycleManagement` - For handling plugin initialization and configuration
+
+## Plugin Loading Process
+
+1. The Plugin Manager scans the plugins directory and all subdirectories
+2. For each subdirectory containing a `plugin.wasm` file and valid `manifest.json`, the manager:
+ - Validates the manifest and checks for supported capabilities
+ - Pre-compiles the WASM module in the background
+ - Registers the plugin using the **folder name** as the unique identifier in the plugin registry
+3. Plugins can be loaded on-demand by folder name or all at once, depending on the manager's method calls
+
+## Writing a Plugin
+
+### Requirements
+
+1. Your plugin must be compiled to WebAssembly (WASM)
+2. Your plugin must implement at least one of the capability interfaces defined in `api.proto`
+3. Your plugin must be placed in its own directory with a proper `manifest.json`
+
+### Plugin Registration Functions
+
+The plugin API provides several registration functions that plugins can call during initialization to register capabilities and obtain host services. These functions should typically be called in your plugin's `init()` function.
+
+#### Standard Registration Functions
+
+```go
+func RegisterMetadataAgent(agent MetadataAgent)
+func RegisterScrobbler(scrobbler Scrobbler)
+func RegisterSchedulerCallback(callback SchedulerCallback)
+func RegisterLifecycleManagement(lifecycle LifecycleManagement)
+func RegisterWebSocketCallback(callback WebSocketCallback)
+```
+
+These functions register plugins for the standard capability interfaces:
+
+- **RegisterMetadataAgent**: Register a plugin that provides artist/album metadata and images
+- **RegisterScrobbler**: Register a plugin that handles scrobbling to external services
+- **RegisterSchedulerCallback**: Register a plugin that handles scheduled callbacks (single callback per plugin)
+- **RegisterLifecycleManagement**: Register a plugin that handles initialization and configuration
+- **RegisterWebSocketCallback**: Register a plugin that handles WebSocket events
+
+**Basic Usage Example:**
+
+```go
+type MyPlugin struct {
+ // plugin implementation
+}
+
+func init() {
+ plugin := &MyPlugin{}
+
+ // Register capabilities your plugin implements
+ api.RegisterScrobbler(plugin)
+ api.RegisterLifecycleManagement(plugin)
+}
+```
+
+#### RegisterNamedSchedulerCallback
+
+```go
+func RegisterNamedSchedulerCallback(name string, cb SchedulerCallback) scheduler.SchedulerService
+```
+
+This function registers a named scheduler callback and returns a scheduler service instance. Named callbacks allow a single plugin to register multiple scheduler callbacks for different purposes, each with its own identifier.
+
+**Parameters:**
+
+- `name` (string): A unique identifier for this scheduler callback within the plugin. This name is used to route scheduled events to the correct callback handler.
+- `cb` (SchedulerCallback): An object that implements the `SchedulerCallback` interface
+
+**Returns:**
+
+- `scheduler.SchedulerService`: A scheduler service instance that can be used to schedule one-time or recurring tasks for this specific callback
+
+**Usage Example** (from Discord Rich Presence plugin):
+
+```go
+func init() {
+ // Register multiple named scheduler callbacks for different purposes
+ plugin.sched = api.RegisterNamedSchedulerCallback("close-activity", plugin)
+ plugin.rpc.sched = api.RegisterNamedSchedulerCallback("heartbeat", plugin.rpc)
+}
+
+// The plugin implements SchedulerCallback to handle "close-activity" events
+func (d *DiscordRPPlugin) OnSchedulerCallback(ctx context.Context, req *api.SchedulerCallbackRequest) (*api.SchedulerCallbackResponse, error) {
+ log.Printf("Removing presence for user %s", req.ScheduleId)
+ // Handle close-activity scheduling events
+ return nil, d.rpc.clearActivity(ctx, req.ScheduleId)
+}
+
+// The rpc component implements SchedulerCallback to handle "heartbeat" events
+func (r *discordRPC) OnSchedulerCallback(ctx context.Context, req *api.SchedulerCallbackRequest) (*api.SchedulerCallbackResponse, error) {
+ // Handle heartbeat scheduling events
+ return nil, r.sendHeartbeat(ctx, req.ScheduleId)
+}
+
+// Use the returned scheduler service to schedule tasks
+func (d *DiscordRPPlugin) NowPlaying(ctx context.Context, request *api.ScrobblerNowPlayingRequest) (*api.ScrobblerNowPlayingResponse, error) {
+ // Schedule a one-time callback to clear activity when track ends
+ _, err = d.sched.ScheduleOneTime(ctx, &scheduler.ScheduleOneTimeRequest{
+ ScheduleId: request.Username,
+ DelaySeconds: request.Track.Length - request.Track.Position + 5,
+ })
+ return nil, err
+}
+
+func (r *discordRPC) connect(ctx context.Context, username string, token string) error {
+ // Schedule recurring heartbeats for Discord connection
+ _, err := r.sched.ScheduleRecurring(ctx, &scheduler.ScheduleRecurringRequest{
+ CronExpression: "@every 41s",
+ ScheduleId: username,
+ })
+ return err
+}
+```
+
+**Key Benefits:**
+
+- **Multiple Schedulers**: A single plugin can have multiple named scheduler callbacks for different purposes (e.g., "heartbeat", "cleanup", "refresh")
+- **Isolated Scheduling**: Each named callback gets its own scheduler service, allowing independent scheduling management
+- **Clear Separation**: Different callback handlers can be implemented on different objects within your plugin
+- **Flexible Routing**: The scheduler automatically routes callbacks to the correct handler based on the registration name
+
+**Important Notes:**
+
+- The `name` parameter must be unique within your plugin, but can be the same across different plugins
+- The returned scheduler service is specifically tied to the named callback you registered
+- Scheduled events will call the `OnSchedulerCallback` method on the object you provided during registration
+- You must implement the `SchedulerCallback` interface on the object you register
+
+#### RegisterSchedulerCallback vs RegisterNamedSchedulerCallback
+
+- **Use `RegisterSchedulerCallback`** when your plugin only needs a single scheduler callback
+- **Use `RegisterNamedSchedulerCallback`** when your plugin needs multiple scheduler callbacks for different purposes (like the Discord plugin's "heartbeat" and "close-activity" callbacks)
+
+The named version allows better organization and separation of concerns when you have complex scheduling requirements.
+
+### Capability Interfaces
+
+#### Metadata Agent
+
+A capability fetches metadata about artists and albums. Implement this interface to add support for fetching data from external sources.
+
+```protobuf
+service MetadataAgent {
+ // Artist metadata methods
+ rpc GetArtistMBID(ArtistMBIDRequest) returns (ArtistMBIDResponse);
+ rpc GetArtistURL(ArtistURLRequest) returns (ArtistURLResponse);
+ rpc GetArtistBiography(ArtistBiographyRequest) returns (ArtistBiographyResponse);
+ rpc GetSimilarArtists(ArtistSimilarRequest) returns (ArtistSimilarResponse);
+ rpc GetArtistImages(ArtistImageRequest) returns (ArtistImageResponse);
+ rpc GetArtistTopSongs(ArtistTopSongsRequest) returns (ArtistTopSongsResponse);
+
+ // Album metadata methods
+ rpc GetAlbumInfo(AlbumInfoRequest) returns (AlbumInfoResponse);
+ rpc GetAlbumImages(AlbumImagesRequest) returns (AlbumImagesResponse);
+}
+```
+
+#### Scrobbler
+
+This capability enables scrobbling to external services. Implement this interface to add support for custom scrobblers.
+
+```protobuf
+service Scrobbler {
+ rpc IsAuthorized(ScrobblerIsAuthorizedRequest) returns (ScrobblerIsAuthorizedResponse);
+ rpc NowPlaying(ScrobblerNowPlayingRequest) returns (ScrobblerNowPlayingResponse);
+ rpc Scrobble(ScrobblerScrobbleRequest) returns (ScrobblerScrobbleResponse);
+}
+```
+
+#### Scheduler Callback
+
+This capability allows plugins to receive one-time or recurring scheduled callbacks. Implement this interface to add
+support for scheduled tasks. See the [SchedulerService](#scheduler-service) for more information.
+
+```protobuf
+service SchedulerCallback {
+ rpc OnSchedulerCallback(SchedulerCallbackRequest) returns (SchedulerCallbackResponse);
+}
+```
+
+#### WebSocket Callback
+
+This capability allows plugins to interact with WebSocket endpoints and handle WebSocket events. Implement this interface to add support for WebSocket-based communication.
+
+```protobuf
+service WebSocketCallback {
+ // Called when a text message is received
+ rpc OnTextMessage(OnTextMessageRequest) returns (OnTextMessageResponse);
+
+ // Called when a binary message is received
+ rpc OnBinaryMessage(OnBinaryMessageRequest) returns (OnBinaryMessageResponse);
+
+ // Called when an error occurs
+ rpc OnError(OnErrorRequest) returns (OnErrorResponse);
+
+ // Called when the connection is closed
+ rpc OnClose(OnCloseRequest) returns (OnCloseResponse);
+}
+```
+
+Plugins can use the WebSocket host service to connect to WebSocket endpoints, send messages, and handle responses:
+
+```go
+// Define a connection ID first
+connectionID := "my-connection-id"
+
+// Connect to a WebSocket server
+connectResp, err := websocket.Connect(ctx, &websocket.ConnectRequest{
+ Url: "wss://example.com/ws",
+ Headers: map[string]string{"Authorization": "Bearer token"},
+ ConnectionId: connectionID,
+})
+if err != nil {
+ return err
+}
+
+// Send a text message
+_, err = websocket.SendText(ctx, &websocket.SendTextRequest{
+ ConnectionId: connectionID,
+ Message: "Hello WebSocket",
+})
+
+// Close the connection when done
+_, err = websocket.Close(ctx, &websocket.CloseRequest{
+ ConnectionId: connectionID,
+ Code: 1000, // Normal closure
+ Reason: "Done",
+})
+```
+
+## Host Services
+
+Navidrome provides several host services that plugins can use to interact with external systems and access functionality. Plugins must declare permissions for each service they want to use in their `manifest.json`.
+
+### HTTP Service
+
+The HTTP service allows plugins to make HTTP requests to external APIs and services. To use this service, declare the `http` permission in your manifest.
+
+#### Basic Usage
+
+```json
+{
+ "permissions": {
+ "http": {
+ "reason": "To fetch artist metadata from external music APIs"
+ }
+ }
+}
+```
+
+#### Granular Permissions
+
+For enhanced security, you can specify granular HTTP permissions that restrict which URLs and HTTP methods your plugin can access:
+
+```json
+{
+ "permissions": {
+ "http": {
+ "reason": "To fetch album reviews from AllMusic and artist data from MusicBrainz",
+ "allowedUrls": {
+ "https://api.allmusic.com": ["GET", "POST"],
+ "https://*.musicbrainz.org": ["GET"],
+ "https://coverartarchive.org": ["GET"],
+ "*": ["GET"]
+ },
+ "allowLocalNetwork": false
+ }
+ }
+}
+```
+
+**Permission Fields:**
+
+- `reason` (required): Clear explanation of why HTTP access is needed
+- `allowedUrls` (required): Map of URL patterns to allowed HTTP methods
+
+ - Must contain at least one URL pattern
+ - For unrestricted access, use: `{"*": ["*"]}`
+ - Keys can be exact URLs, wildcard patterns, or `*` for any URL
+ - Values are arrays of HTTP methods: `GET`, `POST`, `PUT`, `DELETE`, `PATCH`, `HEAD`, `OPTIONS`, or `*` for any method
+ - **Important**: Redirect destinations must also be included in this list. If a URL redirects to another URL not in `allowedUrls`, the redirect will be blocked.
+
+- `allowLocalNetwork` (optional, default: `false`): Whether to allow requests to localhost/private IPs
+
+**URL Pattern Matching:**
+
+- Exact URLs: `https://api.example.com`
+- Wildcard subdomains: `https://*.example.com` (matches any subdomain)
+- Wildcard paths: `https://example.com/api/*` (matches any path under /api/)
+- Global wildcard: `*` (matches any URL - use with caution)
+
+**Examples:**
+
+```json
+// Allow only GET requests to specific APIs
+{
+ "allowedUrls": {
+ "https://api.last.fm": ["GET"],
+ "https://ws.audioscrobbler.com": ["GET"]
+ }
+}
+
+// Allow any method to a trusted domain, GET everywhere else
+{
+ "allowedUrls": {
+ "https://my-trusted-api.com": ["*"],
+ "*": ["GET"]
+ }
+}
+
+// Handle redirects by including redirect destinations
+{
+ "allowedUrls": {
+ "https://short.ly/api123": ["GET"], // Original URL
+ "https://api.actual-service.com": ["GET"] // Redirect destination
+ }
+}
+
+// Strict permissions for a secure plugin (blocks redirects by not including redirect destinations)
+{
+ "allowedUrls": {
+ "https://api.musicbrainz.org/ws/2": ["GET"]
+ },
+ "allowLocalNetwork": false
+}
+```
+
+#### Security Considerations
+
+The HTTP service implements several security features:
+
+1. **Local Network Protection**: By default, requests to localhost and private IP ranges are blocked
+2. **URL Filtering**: Only URLs matching `allowedUrls` patterns are allowed
+3. **Method Restrictions**: HTTP methods are validated against the allowed list for each URL pattern
+4. **Redirect Security**:
+ - Redirect destinations must also match `allowedUrls` patterns and methods
+ - Maximum of 5 redirects per request to prevent redirect loops
+ - To block all redirects, simply don't include any redirect destinations in `allowedUrls`
+
+**Private IP Ranges Blocked (when `allowLocalNetwork: false`):**
+
+- IPv4: `10.0.0.0/8`, `172.16.0.0/12`, `192.168.0.0/16`, `127.0.0.0/8`, `169.254.0.0/16`
+- IPv6: `::1`, `fe80::/10`, `fc00::/7`
+- Hostnames: `localhost`
+
+#### Making HTTP Requests
+
+```go
+import "github.com/navidrome/navidrome/plugins/host/http"
+
+// GET request
+resp, err := httpClient.Get(ctx, &http.HttpRequest{
+ Url: "https://api.example.com/data",
+ Headers: map[string]string{
+ "Authorization": "Bearer " + token,
+ "User-Agent": "MyPlugin/1.0",
+ },
+ TimeoutMs: 5000,
+})
+
+// POST request with body
+resp, err := httpClient.Post(ctx, &http.HttpRequest{
+ Url: "https://api.example.com/submit",
+ Headers: map[string]string{
+ "Content-Type": "application/json",
+ },
+ Body: []byte(`{"key": "value"}`),
+ TimeoutMs: 10000,
+})
+
+// Handle response
+if err != nil {
+ return &api.Response{Error: "HTTP request failed: " + err.Error()}, nil
+}
+
+if resp.Error != "" {
+ return &api.Response{Error: "HTTP error: " + resp.Error}, nil
+}
+
+if resp.Status != 200 {
+ return &api.Response{Error: fmt.Sprintf("HTTP %d: %s", resp.Status, string(resp.Body))}, nil
+}
+
+// Use response data
+data := resp.Body
+headers := resp.Headers
+```
+
+### Other Host Services
+
+#### Config Service
+
+Access plugin-specific configuration:
+
+```json
+{
+ "permissions": {
+ "config": {
+ "reason": "To read API keys and service endpoints from plugin configuration"
+ }
+ }
+}
+```
+
+#### Cache Service
+
+Store and retrieve data to improve performance:
+
+```json
+{
+ "permissions": {
+ "cache": {
+ "reason": "To cache API responses and reduce external service calls"
+ }
+ }
+}
+```
+
+#### Scheduler Service
+
+Schedule recurring or one-time tasks:
+
+```json
+{
+ "permissions": {
+ "scheduler": {
+ "reason": "To schedule periodic metadata refresh and cleanup tasks"
+ }
+ }
+}
+```
+
+#### WebSocket Service
+
+Connect to WebSocket endpoints:
+
+```json
+{
+ "permissions": {
+ "websocket": {
+ "reason": "To connect to real-time music service APIs for live data",
+ "allowedUrls": [
+ "wss://api.musicservice.com/ws",
+ "wss://realtime.example.com"
+ ],
+ "allowLocalNetwork": false
+ }
+ }
+}
+```
+
+#### Artwork Service
+
+Generate public URLs for artwork:
+
+```json
+{
+ "permissions": {
+ "artwork": {
+ "reason": "To generate public URLs for album and artist images"
+ }
+ }
+}
+```
+
+### Error Handling
+
+Plugins should use the standard error values (`plugin:not_found`, `plugin:not_implemented`) to indicate resource-not-found and unimplemented-method scenarios. All other errors will be propagated directly to the caller. Ensure your capability methods return errors via the response message `error` fields rather than panicking or relying on transport errors.
+
+## Plugin Lifecycle and Statelessness
+
+**Important**: Navidrome plugins are stateless. Each method call creates a new plugin instance which is destroyed afterward. This has several important implications:
+
+1. **No in-memory persistence**: Plugins cannot store state between method calls in memory
+2. **Each call is isolated**: Variables, configurations, and runtime state don't persist between calls
+3. **No shared resources**: Each plugin instance has its own memory space
+
+This stateless design is crucial for security and stability:
+
+- Memory leaks in one call won't affect subsequent operations
+- A crashed plugin instance won't bring down the entire system
+- Resource usage is more predictable and contained
+
+When developing plugins, keep these guidelines in mind:
+
+- Don't try to cache data in memory between calls
+- Don't store authentication tokens or session data in variables
+- If persistence is needed, use external storage or the host's HTTP interface
+- Performance optimizations should focus on efficient per-call execution
+
+### Using Plugin Configuration
+
+Since plugins are stateless, you can use the `LifecycleManagement` interface to read configuration when your plugin is loaded and perform any necessary setup:
+
+```go
+func (p *myPlugin) OnInit(ctx context.Context, req *api.InitRequest) (*api.InitResponse, error) {
+ // Access plugin configuration
+ apiKey := req.Config["api_key"]
+ if apiKey == "" {
+ return &api.InitResponse{Error: "Missing API key in configuration"}, nil
+ }
+
+ // Validate configuration
+ serverURL := req.Config["server_url"]
+ if serverURL == "" {
+ serverURL = "https://default-api.example.com" // Use default if not specified
+ }
+
+ // Perform initialization tasks (e.g., validate API key)
+ httpClient := &http.HttpServiceClient{}
+ resp, err := httpClient.Get(ctx, &http.HttpRequest{
+ Url: serverURL + "/validate?key=" + apiKey,
+ })
+ if err != nil {
+ return &api.InitResponse{Error: "Failed to validate API key: " + err.Error()}, nil
+ }
+
+ if resp.StatusCode != 200 {
+ return &api.InitResponse{Error: "Invalid API key"}, nil
+ }
+
+ return &api.InitResponse{}, nil
+}
+```
+
+Remember, the `OnInit` method is called only once when the plugin is loaded. It cannot store any state that needs to persist between method calls. It's primarily useful for:
+
+1. Validating required configuration
+2. Checking API credentials
+3. Verifying connectivity to external services
+4. Initializing any external resources
+
+## Caching
+
+The plugin system implements a compilation cache to improve performance:
+
+1. Compiled WASM modules are cached in `[CacheFolder]/plugins`
+2. This reduces startup time for plugins that have already been compiled
+3. The cache has a automatic cleanup mechanism to remove old modules.
+ - when the cache folder exceeds `Plugins.CacheSize` (default 100MB),
+ the oldest modules are removed
+
+### WASM Loading Optimization
+
+To improve performance during plugin instance creation, the system implements an optimization that avoids repeated file reads and compilation:
+
+1. **Precompilation**: During plugin discovery, WASM files are read and compiled in the background, with both the MD5 hash of the file bytes and compiled modules cached in memory.
+
+2. **Optimized Runtime**: After precompilation completes, plugins use an `optimizedRuntime` wrapper that overrides `CompileModule` to detect when the same WASM bytes are being compiled by comparing MD5 hashes.
+
+3. **Cache Hit**: When the generated plugin code calls `os.ReadFile()` and `CompileModule()`, the optimization calculates the MD5 hash of the incoming bytes and compares it with the cached hash. If they match, it returns the pre-compiled module directly.
+
+4. **Performance Benefit**: This eliminates repeated compilation while using minimal memory (16 bytes per plugin for the MD5 hash vs potentially MB of WASM bytes), significantly improving plugin instance creation speed while maintaining full compatibility with the generated API code.
+
+5. **Memory Efficiency**: By storing only MD5 hashes instead of full WASM bytes, the optimization scales efficiently regardless of plugin size or count.
+
+The optimization is transparent to plugin developers and automatically activates when plugins are successfully precompiled.
+
+## Best Practices
+
+1. **Resource Management**:
+
+ - The host handles HTTP response cleanup, so no need to close response objects
+ - Keep plugin instances lightweight as they are created and destroyed frequently
+
+2. **Error Handling**:
+
+ - Use the standard error types when appropriate
+ - Return descriptive error messages for debugging
+ - Custom errors are supported and will be propagated to the caller
+
+3. **Performance**:
+
+ - Remember plugins are stateless, so don't rely on local variables for caching. Use the CacheService for caching data.
+ - Use efficient algorithms that work well in single-call scenarios
+
+4. **Security**:
+ - Only request permissions you actually need (see [Plugin Permission System](#plugin-permission-system))
+ - Validate inputs to prevent injection attacks
+ - Don't store sensitive credentials in the plugin code
+ - Use configuration for API keys and sensitive data
+
+## Limitations
+
+1. WASM plugins have limited access to system resources
+2. Plugin compilation has an initial overhead on first load, as it needs to be compiled to WebAssembly
+ - Subsequent calls are faster due to caching
+3. New plugin capabilities types require changes to the core codebase
+4. Stateless nature prevents certain optimizations
+
+## Troubleshooting
+
+1. **Plugin not detected**:
+
+ - Ensure `plugin.wasm` and `manifest.json` exist in the plugin directory
+ - Check that the manifest contains valid capabilities names
+ - Verify the manifest schema is valid (see [Plugin Permission System](#plugin-permission-system))
+
+2. **Permission errors**:
+
+ - **"function not exported in module env"**: Plugin trying to use a service without proper permission
+ - Check that required permissions are declared in `manifest.json`
+ - See [Troubleshooting Permissions](#troubleshooting-permissions) for detailed guidance
+
+3. **Compilation errors**:
+
+ - Check logs for WASM compilation errors
+ - Verify the plugin is compatible with the current API version
+
+4. **Runtime errors**:
+ - Look for error messages in the Navidrome logs
+ - Add debug logging to your plugin
+ - Check if the error is permission-related before debugging plugin logic
diff --git a/plugins/adapter_media_agent.go b/plugins/adapter_media_agent.go
new file mode 100644
index 000000000..eca891275
--- /dev/null
+++ b/plugins/adapter_media_agent.go
@@ -0,0 +1,166 @@
+package plugins
+
+import (
+ "context"
+
+ "github.com/navidrome/navidrome/core/agents"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/tetratelabs/wazero"
+)
+
+// NewWasmMediaAgent creates a new adapter for a MetadataAgent plugin
+func newWasmMediaAgent(wasmPath, pluginID string, m *managerImpl, runtime api.WazeroNewRuntime, mc wazero.ModuleConfig) WasmPlugin {
+ loader, err := api.NewMetadataAgentPlugin(context.Background(), api.WazeroRuntime(runtime), api.WazeroModuleConfig(mc))
+ if err != nil {
+ log.Error("Error creating media metadata service plugin", "plugin", pluginID, "path", wasmPath, err)
+ return nil
+ }
+ return &wasmMediaAgent{
+ baseCapability: newBaseCapability[api.MetadataAgent, *api.MetadataAgentPlugin](
+ wasmPath,
+ pluginID,
+ CapabilityMetadataAgent,
+ m.metrics,
+ loader,
+ func(ctx context.Context, l *api.MetadataAgentPlugin, path string) (api.MetadataAgent, error) {
+ return l.Load(ctx, path)
+ },
+ ),
+ }
+}
+
+// wasmMediaAgent adapts a MetadataAgent plugin to implement the agents.Interface
+type wasmMediaAgent struct {
+ *baseCapability[api.MetadataAgent, *api.MetadataAgentPlugin]
+}
+
+func (w *wasmMediaAgent) AgentName() string {
+ return w.id
+}
+
+func (w *wasmMediaAgent) mapError(err error) error {
+ if err != nil && (err.Error() == api.ErrNotFound.Error() || err.Error() == api.ErrNotImplemented.Error()) {
+ return agents.ErrNotFound
+ }
+ return err
+}
+
+// Album-related methods
+
+func (w *wasmMediaAgent) GetAlbumInfo(ctx context.Context, name, artist, mbid string) (*agents.AlbumInfo, error) {
+ res, err := callMethod(ctx, w, "GetAlbumInfo", func(inst api.MetadataAgent) (*api.AlbumInfoResponse, error) {
+ return inst.GetAlbumInfo(ctx, &api.AlbumInfoRequest{Name: name, Artist: artist, Mbid: mbid})
+ })
+ if err != nil {
+ return nil, w.mapError(err)
+ }
+ if res == nil || res.Info == nil {
+ return nil, agents.ErrNotFound
+ }
+ info := res.Info
+ return &agents.AlbumInfo{
+ Name: info.Name,
+ MBID: info.Mbid,
+ Description: info.Description,
+ URL: info.Url,
+ }, nil
+}
+
+func (w *wasmMediaAgent) GetAlbumImages(ctx context.Context, name, artist, mbid string) ([]agents.ExternalImage, error) {
+ res, err := callMethod(ctx, w, "GetAlbumImages", func(inst api.MetadataAgent) (*api.AlbumImagesResponse, error) {
+ return inst.GetAlbumImages(ctx, &api.AlbumImagesRequest{Name: name, Artist: artist, Mbid: mbid})
+ })
+ if err != nil {
+ return nil, w.mapError(err)
+ }
+ return convertExternalImages(res.Images), nil
+}
+
+// Artist-related methods
+
+func (w *wasmMediaAgent) GetArtistMBID(ctx context.Context, id string, name string) (string, error) {
+ res, err := callMethod(ctx, w, "GetArtistMBID", func(inst api.MetadataAgent) (*api.ArtistMBIDResponse, error) {
+ return inst.GetArtistMBID(ctx, &api.ArtistMBIDRequest{Id: id, Name: name})
+ })
+ if err != nil {
+ return "", w.mapError(err)
+ }
+ return res.GetMbid(), nil
+}
+
+func (w *wasmMediaAgent) GetArtistURL(ctx context.Context, id, name, mbid string) (string, error) {
+ res, err := callMethod(ctx, w, "GetArtistURL", func(inst api.MetadataAgent) (*api.ArtistURLResponse, error) {
+ return inst.GetArtistURL(ctx, &api.ArtistURLRequest{Id: id, Name: name, Mbid: mbid})
+ })
+ if err != nil {
+ return "", w.mapError(err)
+ }
+ return res.GetUrl(), nil
+}
+
+func (w *wasmMediaAgent) GetArtistBiography(ctx context.Context, id, name, mbid string) (string, error) {
+ res, err := callMethod(ctx, w, "GetArtistBiography", func(inst api.MetadataAgent) (*api.ArtistBiographyResponse, error) {
+ return inst.GetArtistBiography(ctx, &api.ArtistBiographyRequest{Id: id, Name: name, Mbid: mbid})
+ })
+ if err != nil {
+ return "", w.mapError(err)
+ }
+ return res.GetBiography(), nil
+}
+
+func (w *wasmMediaAgent) GetSimilarArtists(ctx context.Context, id, name, mbid string, limit int) ([]agents.Artist, error) {
+ resp, err := callMethod(ctx, w, "GetSimilarArtists", func(inst api.MetadataAgent) (*api.ArtistSimilarResponse, error) {
+ return inst.GetSimilarArtists(ctx, &api.ArtistSimilarRequest{Id: id, Name: name, Mbid: mbid, Limit: int32(limit)})
+ })
+ if err != nil {
+ return nil, w.mapError(err)
+ }
+ artists := make([]agents.Artist, 0, len(resp.GetArtists()))
+ for _, a := range resp.GetArtists() {
+ artists = append(artists, agents.Artist{
+ Name: a.GetName(),
+ MBID: a.GetMbid(),
+ })
+ }
+ return artists, nil
+}
+
+func (w *wasmMediaAgent) GetArtistImages(ctx context.Context, id, name, mbid string) ([]agents.ExternalImage, error) {
+ resp, err := callMethod(ctx, w, "GetArtistImages", func(inst api.MetadataAgent) (*api.ArtistImageResponse, error) {
+ return inst.GetArtistImages(ctx, &api.ArtistImageRequest{Id: id, Name: name, Mbid: mbid})
+ })
+ if err != nil {
+ return nil, w.mapError(err)
+ }
+ return convertExternalImages(resp.Images), nil
+}
+
+func (w *wasmMediaAgent) GetArtistTopSongs(ctx context.Context, id, artistName, mbid string, count int) ([]agents.Song, error) {
+ resp, err := callMethod(ctx, w, "GetArtistTopSongs", func(inst api.MetadataAgent) (*api.ArtistTopSongsResponse, error) {
+ return inst.GetArtistTopSongs(ctx, &api.ArtistTopSongsRequest{Id: id, ArtistName: artistName, Mbid: mbid, Count: int32(count)})
+ })
+ if err != nil {
+ return nil, w.mapError(err)
+ }
+ songs := make([]agents.Song, 0, len(resp.GetSongs()))
+ for _, s := range resp.GetSongs() {
+ songs = append(songs, agents.Song{
+ Name: s.GetName(),
+ MBID: s.GetMbid(),
+ })
+ }
+ return songs, nil
+}
+
+// Helper function to convert ExternalImage objects from the API to the agents package
+func convertExternalImages(images []*api.ExternalImage) []agents.ExternalImage {
+ result := make([]agents.ExternalImage, 0, len(images))
+ for _, img := range images {
+ result = append(result, agents.ExternalImage{
+ URL: img.GetUrl(),
+ Size: int(img.GetSize()),
+ })
+ }
+ return result
+}
diff --git a/plugins/adapter_media_agent_test.go b/plugins/adapter_media_agent_test.go
new file mode 100644
index 000000000..70b5d275a
--- /dev/null
+++ b/plugins/adapter_media_agent_test.go
@@ -0,0 +1,227 @@
+package plugins
+
+import (
+ "context"
+ "errors"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/core/agents"
+ "github.com/navidrome/navidrome/core/metrics"
+ "github.com/navidrome/navidrome/plugins/api"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("Adapter Media Agent", func() {
+ var ctx context.Context
+ var mgr *managerImpl
+
+ BeforeEach(func() {
+ ctx = GinkgoT().Context()
+
+ // Ensure plugins folder is set to testdata
+ DeferCleanup(configtest.SetupConfig())
+ conf.Server.Plugins.Folder = testDataDir
+
+ mgr = createManager(nil, metrics.NewNoopInstance())
+ mgr.ScanPlugins()
+
+ // Wait for all plugins to compile to avoid race conditions
+ err := mgr.EnsureCompiled("multi_plugin")
+ Expect(err).NotTo(HaveOccurred(), "multi_plugin should compile successfully")
+ err = mgr.EnsureCompiled("fake_album_agent")
+ Expect(err).NotTo(HaveOccurred(), "fake_album_agent should compile successfully")
+ })
+
+ Describe("AgentName and PluginName", func() {
+ It("should return the plugin name", func() {
+ agent := mgr.LoadPlugin("multi_plugin", "MetadataAgent")
+ Expect(agent).NotTo(BeNil(), "multi_plugin should be loaded")
+ Expect(agent.PluginID()).To(Equal("multi_plugin"))
+ })
+ It("should return the agent name", func() {
+ agent, ok := mgr.LoadMediaAgent("multi_plugin")
+ Expect(ok).To(BeTrue(), "multi_plugin should be loaded as media agent")
+ Expect(agent.AgentName()).To(Equal("multi_plugin"))
+ })
+ })
+
+ Describe("Album methods", func() {
+ var agent *wasmMediaAgent
+
+ BeforeEach(func() {
+ a, ok := mgr.LoadMediaAgent("fake_album_agent")
+ Expect(ok).To(BeTrue(), "fake_album_agent should be loaded")
+ agent = a.(*wasmMediaAgent)
+ })
+
+ Context("GetAlbumInfo", func() {
+ It("should return album information", func() {
+ info, err := agent.GetAlbumInfo(ctx, "Test Album", "Test Artist", "mbid")
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(info).NotTo(BeNil())
+ Expect(info.Name).To(Equal("Test Album"))
+ Expect(info.MBID).To(Equal("album-mbid-123"))
+ Expect(info.Description).To(Equal("This is a test album description"))
+ Expect(info.URL).To(Equal("https://example.com/album"))
+ })
+
+ It("should return ErrNotFound when plugin returns not found", func() {
+ _, err := agent.GetAlbumInfo(ctx, "Test Album", "", "mbid")
+
+ Expect(err).To(Equal(agents.ErrNotFound))
+ })
+
+ It("should return ErrNotFound when plugin returns nil response", func() {
+ _, err := agent.GetAlbumInfo(ctx, "", "", "")
+
+ Expect(err).To(Equal(agents.ErrNotFound))
+ })
+ })
+
+ Context("GetAlbumImages", func() {
+ It("should return album images", func() {
+ images, err := agent.GetAlbumImages(ctx, "Test Album", "Test Artist", "mbid")
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(images).To(Equal([]agents.ExternalImage{
+ {URL: "https://example.com/album1.jpg", Size: 300},
+ {URL: "https://example.com/album2.jpg", Size: 400},
+ }))
+ })
+ })
+ })
+
+ Describe("Artist methods", func() {
+ var agent *wasmMediaAgent
+
+ BeforeEach(func() {
+ a, ok := mgr.LoadMediaAgent("fake_artist_agent")
+ Expect(ok).To(BeTrue(), "fake_artist_agent should be loaded")
+ agent = a.(*wasmMediaAgent)
+ })
+
+ Context("GetArtistMBID", func() {
+ It("should return artist MBID", func() {
+ mbid, err := agent.GetArtistMBID(ctx, "artist-id", "Test Artist")
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(mbid).To(Equal("1234567890"))
+ })
+
+ It("should return ErrNotFound when plugin returns not found", func() {
+ _, err := agent.GetArtistMBID(ctx, "artist-id", "")
+
+ Expect(err).To(Equal(agents.ErrNotFound))
+ })
+ })
+
+ Context("GetArtistURL", func() {
+ It("should return artist URL", func() {
+ url, err := agent.GetArtistURL(ctx, "artist-id", "Test Artist", "mbid")
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(url).To(Equal("https://example.com"))
+ })
+ })
+
+ Context("GetArtistBiography", func() {
+ It("should return artist biography", func() {
+ bio, err := agent.GetArtistBiography(ctx, "artist-id", "Test Artist", "mbid")
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(bio).To(Equal("This is a test biography"))
+ })
+ })
+
+ Context("GetSimilarArtists", func() {
+ It("should return similar artists", func() {
+ artists, err := agent.GetSimilarArtists(ctx, "artist-id", "Test Artist", "mbid", 10)
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(artists).To(Equal([]agents.Artist{
+ {Name: "Similar Artist 1", MBID: "mbid1"},
+ {Name: "Similar Artist 2", MBID: "mbid2"},
+ }))
+ })
+ })
+
+ Context("GetArtistImages", func() {
+ It("should return artist images", func() {
+ images, err := agent.GetArtistImages(ctx, "artist-id", "Test Artist", "mbid")
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(images).To(Equal([]agents.ExternalImage{
+ {URL: "https://example.com/image1.jpg", Size: 100},
+ {URL: "https://example.com/image2.jpg", Size: 200},
+ }))
+ })
+ })
+
+ Context("GetArtistTopSongs", func() {
+ It("should return artist top songs", func() {
+ songs, err := agent.GetArtistTopSongs(ctx, "artist-id", "Test Artist", "mbid", 10)
+
+ Expect(err).NotTo(HaveOccurred())
+ Expect(songs).To(Equal([]agents.Song{
+ {Name: "Song 1", MBID: "mbid1"},
+ {Name: "Song 2", MBID: "mbid2"},
+ }))
+ })
+ })
+ })
+
+ Describe("Helper functions", func() {
+ It("convertExternalImages should convert API image objects to agent image objects", func() {
+ apiImages := []*api.ExternalImage{
+ {Url: "https://example.com/image1.jpg", Size: 100},
+ {Url: "https://example.com/image2.jpg", Size: 200},
+ }
+
+ agentImages := convertExternalImages(apiImages)
+ Expect(agentImages).To(HaveLen(2))
+
+ for i, img := range agentImages {
+ Expect(img.URL).To(Equal(apiImages[i].Url))
+ Expect(img.Size).To(Equal(int(apiImages[i].Size)))
+ }
+ })
+
+ It("convertExternalImages should handle empty slice", func() {
+ agentImages := convertExternalImages([]*api.ExternalImage{})
+ Expect(agentImages).To(BeEmpty())
+ })
+
+ It("convertExternalImages should handle nil", func() {
+ agentImages := convertExternalImages(nil)
+ Expect(agentImages).To(BeEmpty())
+ })
+ })
+
+ Describe("Error mapping", func() {
+ var agent wasmMediaAgent
+
+ It("should map API ErrNotFound to agents.ErrNotFound", func() {
+ err := agent.mapError(api.ErrNotFound)
+ Expect(err).To(Equal(agents.ErrNotFound))
+ })
+
+ It("should map API ErrNotImplemented to agents.ErrNotFound", func() {
+ err := agent.mapError(api.ErrNotImplemented)
+ Expect(err).To(Equal(agents.ErrNotFound))
+ })
+
+ It("should pass through other errors", func() {
+ testErr := errors.New("test error")
+ err := agent.mapError(testErr)
+ Expect(err).To(Equal(testErr))
+ })
+
+ It("should handle nil error", func() {
+ err := agent.mapError(nil)
+ Expect(err).To(BeNil())
+ })
+ })
+})
diff --git a/plugins/adapter_scheduler_callback.go b/plugins/adapter_scheduler_callback.go
new file mode 100644
index 000000000..64b7eefff
--- /dev/null
+++ b/plugins/adapter_scheduler_callback.go
@@ -0,0 +1,46 @@
+package plugins
+
+import (
+ "context"
+
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/tetratelabs/wazero"
+)
+
+// newWasmSchedulerCallback creates a new adapter for a SchedulerCallback plugin
+func newWasmSchedulerCallback(wasmPath, pluginID string, m *managerImpl, runtime api.WazeroNewRuntime, mc wazero.ModuleConfig) WasmPlugin {
+ loader, err := api.NewSchedulerCallbackPlugin(context.Background(), api.WazeroRuntime(runtime), api.WazeroModuleConfig(mc))
+ if err != nil {
+ log.Error("Error creating scheduler callback plugin", "plugin", pluginID, "path", wasmPath, err)
+ return nil
+ }
+ return &wasmSchedulerCallback{
+ baseCapability: newBaseCapability[api.SchedulerCallback, *api.SchedulerCallbackPlugin](
+ wasmPath,
+ pluginID,
+ CapabilitySchedulerCallback,
+ m.metrics,
+ loader,
+ func(ctx context.Context, l *api.SchedulerCallbackPlugin, path string) (api.SchedulerCallback, error) {
+ return l.Load(ctx, path)
+ },
+ ),
+ }
+}
+
+// wasmSchedulerCallback adapts a SchedulerCallback plugin
+type wasmSchedulerCallback struct {
+ *baseCapability[api.SchedulerCallback, *api.SchedulerCallbackPlugin]
+}
+
+func (w *wasmSchedulerCallback) OnSchedulerCallback(ctx context.Context, scheduleID string, payload []byte, isRecurring bool) error {
+ _, err := callMethod(ctx, w, "OnSchedulerCallback", func(inst api.SchedulerCallback) (*api.SchedulerCallbackResponse, error) {
+ return inst.OnSchedulerCallback(ctx, &api.SchedulerCallbackRequest{
+ ScheduleId: scheduleID,
+ Payload: payload,
+ IsRecurring: isRecurring,
+ })
+ })
+ return err
+}
diff --git a/plugins/adapter_scrobbler.go b/plugins/adapter_scrobbler.go
new file mode 100644
index 000000000..54c6af127
--- /dev/null
+++ b/plugins/adapter_scrobbler.go
@@ -0,0 +1,136 @@
+package plugins
+
+import (
+ "context"
+ "time"
+
+ "github.com/navidrome/navidrome/core/scrobbler"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/tetratelabs/wazero"
+)
+
+func newWasmScrobblerPlugin(wasmPath, pluginID string, m *managerImpl, runtime api.WazeroNewRuntime, mc wazero.ModuleConfig) WasmPlugin {
+ loader, err := api.NewScrobblerPlugin(context.Background(), api.WazeroRuntime(runtime), api.WazeroModuleConfig(mc))
+ if err != nil {
+ log.Error("Error creating scrobbler service plugin", "plugin", pluginID, "path", wasmPath, err)
+ return nil
+ }
+ return &wasmScrobblerPlugin{
+ baseCapability: newBaseCapability[api.Scrobbler, *api.ScrobblerPlugin](
+ wasmPath,
+ pluginID,
+ CapabilityScrobbler,
+ m.metrics,
+ loader,
+ func(ctx context.Context, l *api.ScrobblerPlugin, path string) (api.Scrobbler, error) {
+ return l.Load(ctx, path)
+ },
+ ),
+ }
+}
+
+type wasmScrobblerPlugin struct {
+ *baseCapability[api.Scrobbler, *api.ScrobblerPlugin]
+}
+
+func (w *wasmScrobblerPlugin) IsAuthorized(ctx context.Context, userId string) bool {
+ username, _ := request.UsernameFrom(ctx)
+ if username == "" {
+ u, ok := request.UserFrom(ctx)
+ if ok {
+ username = u.UserName
+ }
+ }
+ resp, err := callMethod(ctx, w, "IsAuthorized", func(inst api.Scrobbler) (*api.ScrobblerIsAuthorizedResponse, error) {
+ return inst.IsAuthorized(ctx, &api.ScrobblerIsAuthorizedRequest{
+ UserId: userId,
+ Username: username,
+ })
+ })
+ if err != nil {
+ log.Warn("Error calling IsAuthorized", "userId", userId, "pluginID", w.id, err)
+ }
+ return err == nil && resp.Authorized
+}
+
+func (w *wasmScrobblerPlugin) NowPlaying(ctx context.Context, userId string, track *model.MediaFile, position int) error {
+ username, _ := request.UsernameFrom(ctx)
+ if username == "" {
+ u, ok := request.UserFrom(ctx)
+ if ok {
+ username = u.UserName
+ }
+ }
+
+ trackInfo := w.toTrackInfo(track, position)
+ _, err := callMethod(ctx, w, "NowPlaying", func(inst api.Scrobbler) (struct{}, error) {
+ resp, err := inst.NowPlaying(ctx, &api.ScrobblerNowPlayingRequest{
+ UserId: userId,
+ Username: username,
+ Track: trackInfo,
+ Timestamp: time.Now().Unix(),
+ })
+ if err != nil {
+ return struct{}{}, err
+ }
+ if resp.Error != "" {
+ return struct{}{}, nil
+ }
+ return struct{}{}, nil
+ })
+ return err
+}
+
+func (w *wasmScrobblerPlugin) Scrobble(ctx context.Context, userId string, s scrobbler.Scrobble) error {
+ username, _ := request.UsernameFrom(ctx)
+ if username == "" {
+ u, ok := request.UserFrom(ctx)
+ if ok {
+ username = u.UserName
+ }
+ }
+ trackInfo := w.toTrackInfo(&s.MediaFile, 0)
+ _, err := callMethod(ctx, w, "Scrobble", func(inst api.Scrobbler) (struct{}, error) {
+ resp, err := inst.Scrobble(ctx, &api.ScrobblerScrobbleRequest{
+ UserId: userId,
+ Username: username,
+ Track: trackInfo,
+ Timestamp: s.TimeStamp.Unix(),
+ })
+ if err != nil {
+ return struct{}{}, err
+ }
+ if resp.Error != "" {
+ return struct{}{}, nil
+ }
+ return struct{}{}, nil
+ })
+ return err
+}
+
+func (w *wasmScrobblerPlugin) toTrackInfo(track *model.MediaFile, position int) *api.TrackInfo {
+ artists := make([]*api.Artist, 0, len(track.Participants[model.RoleArtist]))
+
+ for _, a := range track.Participants[model.RoleArtist] {
+ artists = append(artists, &api.Artist{Name: a.Name, Mbid: a.MbzArtistID})
+ }
+ albumArtists := make([]*api.Artist, 0, len(track.Participants[model.RoleAlbumArtist]))
+ for _, a := range track.Participants[model.RoleAlbumArtist] {
+ albumArtists = append(albumArtists, &api.Artist{Name: a.Name, Mbid: a.MbzArtistID})
+ }
+ trackInfo := &api.TrackInfo{
+ Id: track.ID,
+ Mbid: track.MbzRecordingID,
+ Name: track.Title,
+ Album: track.Album,
+ AlbumMbid: track.MbzAlbumID,
+ Artists: artists,
+ AlbumArtists: albumArtists,
+ Length: int32(track.Duration),
+ Position: int32(position),
+ }
+ return trackInfo
+}
diff --git a/plugins/adapter_websocket_callback.go b/plugins/adapter_websocket_callback.go
new file mode 100644
index 000000000..83b8dd567
--- /dev/null
+++ b/plugins/adapter_websocket_callback.go
@@ -0,0 +1,35 @@
+package plugins
+
+import (
+ "context"
+
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/tetratelabs/wazero"
+)
+
+// newWasmWebSocketCallback creates a new adapter for a WebSocketCallback plugin
+func newWasmWebSocketCallback(wasmPath, pluginID string, m *managerImpl, runtime api.WazeroNewRuntime, mc wazero.ModuleConfig) WasmPlugin {
+ loader, err := api.NewWebSocketCallbackPlugin(context.Background(), api.WazeroRuntime(runtime), api.WazeroModuleConfig(mc))
+ if err != nil {
+ log.Error("Error creating WebSocket callback plugin", "plugin", pluginID, "path", wasmPath, err)
+ return nil
+ }
+ return &wasmWebSocketCallback{
+ baseCapability: newBaseCapability[api.WebSocketCallback, *api.WebSocketCallbackPlugin](
+ wasmPath,
+ pluginID,
+ CapabilityWebSocketCallback,
+ m.metrics,
+ loader,
+ func(ctx context.Context, l *api.WebSocketCallbackPlugin, path string) (api.WebSocketCallback, error) {
+ return l.Load(ctx, path)
+ },
+ ),
+ }
+}
+
+// wasmWebSocketCallback adapts a WebSocketCallback plugin
+type wasmWebSocketCallback struct {
+ *baseCapability[api.WebSocketCallback, *api.WebSocketCallbackPlugin]
+}
diff --git a/plugins/api/api.pb.go b/plugins/api/api.pb.go
new file mode 100644
index 000000000..b570d5c61
--- /dev/null
+++ b/plugins/api/api.pb.go
@@ -0,0 +1,1136 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: api/api.proto
+
+package api
+
+import (
+ context "context"
+ protoreflect "google.golang.org/protobuf/reflect/protoreflect"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+type ArtistMBIDRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Id string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"`
+ Name string `protobuf:"bytes,2,opt,name=name,proto3" json:"name,omitempty"`
+}
+
+func (x *ArtistMBIDRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ArtistMBIDRequest) GetId() string {
+ if x != nil {
+ return x.Id
+ }
+ return ""
+}
+
+func (x *ArtistMBIDRequest) GetName() string {
+ if x != nil {
+ return x.Name
+ }
+ return ""
+}
+
+type ArtistMBIDResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Mbid string `protobuf:"bytes,1,opt,name=mbid,proto3" json:"mbid,omitempty"`
+}
+
+func (x *ArtistMBIDResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ArtistMBIDResponse) GetMbid() string {
+ if x != nil {
+ return x.Mbid
+ }
+ return ""
+}
+
+type ArtistURLRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Id string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"`
+ Name string `protobuf:"bytes,2,opt,name=name,proto3" json:"name,omitempty"`
+ Mbid string `protobuf:"bytes,3,opt,name=mbid,proto3" json:"mbid,omitempty"`
+}
+
+func (x *ArtistURLRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ArtistURLRequest) GetId() string {
+ if x != nil {
+ return x.Id
+ }
+ return ""
+}
+
+func (x *ArtistURLRequest) GetName() string {
+ if x != nil {
+ return x.Name
+ }
+ return ""
+}
+
+func (x *ArtistURLRequest) GetMbid() string {
+ if x != nil {
+ return x.Mbid
+ }
+ return ""
+}
+
+type ArtistURLResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Url string `protobuf:"bytes,1,opt,name=url,proto3" json:"url,omitempty"`
+}
+
+func (x *ArtistURLResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ArtistURLResponse) GetUrl() string {
+ if x != nil {
+ return x.Url
+ }
+ return ""
+}
+
+type ArtistBiographyRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Id string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"`
+ Name string `protobuf:"bytes,2,opt,name=name,proto3" json:"name,omitempty"`
+ Mbid string `protobuf:"bytes,3,opt,name=mbid,proto3" json:"mbid,omitempty"`
+}
+
+func (x *ArtistBiographyRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ArtistBiographyRequest) GetId() string {
+ if x != nil {
+ return x.Id
+ }
+ return ""
+}
+
+func (x *ArtistBiographyRequest) GetName() string {
+ if x != nil {
+ return x.Name
+ }
+ return ""
+}
+
+func (x *ArtistBiographyRequest) GetMbid() string {
+ if x != nil {
+ return x.Mbid
+ }
+ return ""
+}
+
+type ArtistBiographyResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Biography string `protobuf:"bytes,1,opt,name=biography,proto3" json:"biography,omitempty"`
+}
+
+func (x *ArtistBiographyResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ArtistBiographyResponse) GetBiography() string {
+ if x != nil {
+ return x.Biography
+ }
+ return ""
+}
+
+type ArtistSimilarRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Id string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"`
+ Name string `protobuf:"bytes,2,opt,name=name,proto3" json:"name,omitempty"`
+ Mbid string `protobuf:"bytes,3,opt,name=mbid,proto3" json:"mbid,omitempty"`
+ Limit int32 `protobuf:"varint,4,opt,name=limit,proto3" json:"limit,omitempty"`
+}
+
+func (x *ArtistSimilarRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ArtistSimilarRequest) GetId() string {
+ if x != nil {
+ return x.Id
+ }
+ return ""
+}
+
+func (x *ArtistSimilarRequest) GetName() string {
+ if x != nil {
+ return x.Name
+ }
+ return ""
+}
+
+func (x *ArtistSimilarRequest) GetMbid() string {
+ if x != nil {
+ return x.Mbid
+ }
+ return ""
+}
+
+func (x *ArtistSimilarRequest) GetLimit() int32 {
+ if x != nil {
+ return x.Limit
+ }
+ return 0
+}
+
+type Artist struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Name string `protobuf:"bytes,1,opt,name=name,proto3" json:"name,omitempty"`
+ Mbid string `protobuf:"bytes,2,opt,name=mbid,proto3" json:"mbid,omitempty"`
+}
+
+func (x *Artist) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *Artist) GetName() string {
+ if x != nil {
+ return x.Name
+ }
+ return ""
+}
+
+func (x *Artist) GetMbid() string {
+ if x != nil {
+ return x.Mbid
+ }
+ return ""
+}
+
+type ArtistSimilarResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Artists []*Artist `protobuf:"bytes,1,rep,name=artists,proto3" json:"artists,omitempty"`
+}
+
+func (x *ArtistSimilarResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ArtistSimilarResponse) GetArtists() []*Artist {
+ if x != nil {
+ return x.Artists
+ }
+ return nil
+}
+
+type ArtistImageRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Id string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"`
+ Name string `protobuf:"bytes,2,opt,name=name,proto3" json:"name,omitempty"`
+ Mbid string `protobuf:"bytes,3,opt,name=mbid,proto3" json:"mbid,omitempty"`
+}
+
+func (x *ArtistImageRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ArtistImageRequest) GetId() string {
+ if x != nil {
+ return x.Id
+ }
+ return ""
+}
+
+func (x *ArtistImageRequest) GetName() string {
+ if x != nil {
+ return x.Name
+ }
+ return ""
+}
+
+func (x *ArtistImageRequest) GetMbid() string {
+ if x != nil {
+ return x.Mbid
+ }
+ return ""
+}
+
+type ExternalImage struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Url string `protobuf:"bytes,1,opt,name=url,proto3" json:"url,omitempty"`
+ Size int32 `protobuf:"varint,2,opt,name=size,proto3" json:"size,omitempty"`
+}
+
+func (x *ExternalImage) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ExternalImage) GetUrl() string {
+ if x != nil {
+ return x.Url
+ }
+ return ""
+}
+
+func (x *ExternalImage) GetSize() int32 {
+ if x != nil {
+ return x.Size
+ }
+ return 0
+}
+
+type ArtistImageResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Images []*ExternalImage `protobuf:"bytes,1,rep,name=images,proto3" json:"images,omitempty"`
+}
+
+func (x *ArtistImageResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ArtistImageResponse) GetImages() []*ExternalImage {
+ if x != nil {
+ return x.Images
+ }
+ return nil
+}
+
+type ArtistTopSongsRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Id string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"`
+ ArtistName string `protobuf:"bytes,2,opt,name=artistName,proto3" json:"artistName,omitempty"`
+ Mbid string `protobuf:"bytes,3,opt,name=mbid,proto3" json:"mbid,omitempty"`
+ Count int32 `protobuf:"varint,4,opt,name=count,proto3" json:"count,omitempty"`
+}
+
+func (x *ArtistTopSongsRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ArtistTopSongsRequest) GetId() string {
+ if x != nil {
+ return x.Id
+ }
+ return ""
+}
+
+func (x *ArtistTopSongsRequest) GetArtistName() string {
+ if x != nil {
+ return x.ArtistName
+ }
+ return ""
+}
+
+func (x *ArtistTopSongsRequest) GetMbid() string {
+ if x != nil {
+ return x.Mbid
+ }
+ return ""
+}
+
+func (x *ArtistTopSongsRequest) GetCount() int32 {
+ if x != nil {
+ return x.Count
+ }
+ return 0
+}
+
+type Song struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Name string `protobuf:"bytes,1,opt,name=name,proto3" json:"name,omitempty"`
+ Mbid string `protobuf:"bytes,2,opt,name=mbid,proto3" json:"mbid,omitempty"`
+}
+
+func (x *Song) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *Song) GetName() string {
+ if x != nil {
+ return x.Name
+ }
+ return ""
+}
+
+func (x *Song) GetMbid() string {
+ if x != nil {
+ return x.Mbid
+ }
+ return ""
+}
+
+type ArtistTopSongsResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Songs []*Song `protobuf:"bytes,1,rep,name=songs,proto3" json:"songs,omitempty"`
+}
+
+func (x *ArtistTopSongsResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ArtistTopSongsResponse) GetSongs() []*Song {
+ if x != nil {
+ return x.Songs
+ }
+ return nil
+}
+
+type AlbumInfoRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Name string `protobuf:"bytes,1,opt,name=name,proto3" json:"name,omitempty"`
+ Artist string `protobuf:"bytes,2,opt,name=artist,proto3" json:"artist,omitempty"`
+ Mbid string `protobuf:"bytes,3,opt,name=mbid,proto3" json:"mbid,omitempty"`
+}
+
+func (x *AlbumInfoRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *AlbumInfoRequest) GetName() string {
+ if x != nil {
+ return x.Name
+ }
+ return ""
+}
+
+func (x *AlbumInfoRequest) GetArtist() string {
+ if x != nil {
+ return x.Artist
+ }
+ return ""
+}
+
+func (x *AlbumInfoRequest) GetMbid() string {
+ if x != nil {
+ return x.Mbid
+ }
+ return ""
+}
+
+type AlbumInfo struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Name string `protobuf:"bytes,1,opt,name=name,proto3" json:"name,omitempty"`
+ Mbid string `protobuf:"bytes,2,opt,name=mbid,proto3" json:"mbid,omitempty"`
+ Description string `protobuf:"bytes,3,opt,name=description,proto3" json:"description,omitempty"`
+ Url string `protobuf:"bytes,4,opt,name=url,proto3" json:"url,omitempty"`
+}
+
+func (x *AlbumInfo) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *AlbumInfo) GetName() string {
+ if x != nil {
+ return x.Name
+ }
+ return ""
+}
+
+func (x *AlbumInfo) GetMbid() string {
+ if x != nil {
+ return x.Mbid
+ }
+ return ""
+}
+
+func (x *AlbumInfo) GetDescription() string {
+ if x != nil {
+ return x.Description
+ }
+ return ""
+}
+
+func (x *AlbumInfo) GetUrl() string {
+ if x != nil {
+ return x.Url
+ }
+ return ""
+}
+
+type AlbumInfoResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Info *AlbumInfo `protobuf:"bytes,1,opt,name=info,proto3" json:"info,omitempty"`
+}
+
+func (x *AlbumInfoResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *AlbumInfoResponse) GetInfo() *AlbumInfo {
+ if x != nil {
+ return x.Info
+ }
+ return nil
+}
+
+type AlbumImagesRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Name string `protobuf:"bytes,1,opt,name=name,proto3" json:"name,omitempty"`
+ Artist string `protobuf:"bytes,2,opt,name=artist,proto3" json:"artist,omitempty"`
+ Mbid string `protobuf:"bytes,3,opt,name=mbid,proto3" json:"mbid,omitempty"`
+}
+
+func (x *AlbumImagesRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *AlbumImagesRequest) GetName() string {
+ if x != nil {
+ return x.Name
+ }
+ return ""
+}
+
+func (x *AlbumImagesRequest) GetArtist() string {
+ if x != nil {
+ return x.Artist
+ }
+ return ""
+}
+
+func (x *AlbumImagesRequest) GetMbid() string {
+ if x != nil {
+ return x.Mbid
+ }
+ return ""
+}
+
+type AlbumImagesResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Images []*ExternalImage `protobuf:"bytes,1,rep,name=images,proto3" json:"images,omitempty"`
+}
+
+func (x *AlbumImagesResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *AlbumImagesResponse) GetImages() []*ExternalImage {
+ if x != nil {
+ return x.Images
+ }
+ return nil
+}
+
+type ScrobblerIsAuthorizedRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ UserId string `protobuf:"bytes,1,opt,name=user_id,json=userId,proto3" json:"user_id,omitempty"`
+ Username string `protobuf:"bytes,2,opt,name=username,proto3" json:"username,omitempty"`
+}
+
+func (x *ScrobblerIsAuthorizedRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ScrobblerIsAuthorizedRequest) GetUserId() string {
+ if x != nil {
+ return x.UserId
+ }
+ return ""
+}
+
+func (x *ScrobblerIsAuthorizedRequest) GetUsername() string {
+ if x != nil {
+ return x.Username
+ }
+ return ""
+}
+
+type ScrobblerIsAuthorizedResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Authorized bool `protobuf:"varint,1,opt,name=authorized,proto3" json:"authorized,omitempty"`
+ Error string `protobuf:"bytes,2,opt,name=error,proto3" json:"error,omitempty"`
+}
+
+func (x *ScrobblerIsAuthorizedResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ScrobblerIsAuthorizedResponse) GetAuthorized() bool {
+ if x != nil {
+ return x.Authorized
+ }
+ return false
+}
+
+func (x *ScrobblerIsAuthorizedResponse) GetError() string {
+ if x != nil {
+ return x.Error
+ }
+ return ""
+}
+
+type TrackInfo struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Id string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"`
+ Mbid string `protobuf:"bytes,2,opt,name=mbid,proto3" json:"mbid,omitempty"`
+ Name string `protobuf:"bytes,3,opt,name=name,proto3" json:"name,omitempty"`
+ Album string `protobuf:"bytes,4,opt,name=album,proto3" json:"album,omitempty"`
+ AlbumMbid string `protobuf:"bytes,5,opt,name=album_mbid,json=albumMbid,proto3" json:"album_mbid,omitempty"`
+ Artists []*Artist `protobuf:"bytes,6,rep,name=artists,proto3" json:"artists,omitempty"`
+ AlbumArtists []*Artist `protobuf:"bytes,7,rep,name=album_artists,json=albumArtists,proto3" json:"album_artists,omitempty"`
+ Length int32 `protobuf:"varint,8,opt,name=length,proto3" json:"length,omitempty"` // seconds
+ Position int32 `protobuf:"varint,9,opt,name=position,proto3" json:"position,omitempty"` // seconds
+}
+
+func (x *TrackInfo) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *TrackInfo) GetId() string {
+ if x != nil {
+ return x.Id
+ }
+ return ""
+}
+
+func (x *TrackInfo) GetMbid() string {
+ if x != nil {
+ return x.Mbid
+ }
+ return ""
+}
+
+func (x *TrackInfo) GetName() string {
+ if x != nil {
+ return x.Name
+ }
+ return ""
+}
+
+func (x *TrackInfo) GetAlbum() string {
+ if x != nil {
+ return x.Album
+ }
+ return ""
+}
+
+func (x *TrackInfo) GetAlbumMbid() string {
+ if x != nil {
+ return x.AlbumMbid
+ }
+ return ""
+}
+
+func (x *TrackInfo) GetArtists() []*Artist {
+ if x != nil {
+ return x.Artists
+ }
+ return nil
+}
+
+func (x *TrackInfo) GetAlbumArtists() []*Artist {
+ if x != nil {
+ return x.AlbumArtists
+ }
+ return nil
+}
+
+func (x *TrackInfo) GetLength() int32 {
+ if x != nil {
+ return x.Length
+ }
+ return 0
+}
+
+func (x *TrackInfo) GetPosition() int32 {
+ if x != nil {
+ return x.Position
+ }
+ return 0
+}
+
+type ScrobblerNowPlayingRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ UserId string `protobuf:"bytes,1,opt,name=user_id,json=userId,proto3" json:"user_id,omitempty"`
+ Username string `protobuf:"bytes,2,opt,name=username,proto3" json:"username,omitempty"`
+ Track *TrackInfo `protobuf:"bytes,3,opt,name=track,proto3" json:"track,omitempty"`
+ Timestamp int64 `protobuf:"varint,4,opt,name=timestamp,proto3" json:"timestamp,omitempty"`
+}
+
+func (x *ScrobblerNowPlayingRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ScrobblerNowPlayingRequest) GetUserId() string {
+ if x != nil {
+ return x.UserId
+ }
+ return ""
+}
+
+func (x *ScrobblerNowPlayingRequest) GetUsername() string {
+ if x != nil {
+ return x.Username
+ }
+ return ""
+}
+
+func (x *ScrobblerNowPlayingRequest) GetTrack() *TrackInfo {
+ if x != nil {
+ return x.Track
+ }
+ return nil
+}
+
+func (x *ScrobblerNowPlayingRequest) GetTimestamp() int64 {
+ if x != nil {
+ return x.Timestamp
+ }
+ return 0
+}
+
+type ScrobblerNowPlayingResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Error string `protobuf:"bytes,1,opt,name=error,proto3" json:"error,omitempty"`
+}
+
+func (x *ScrobblerNowPlayingResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ScrobblerNowPlayingResponse) GetError() string {
+ if x != nil {
+ return x.Error
+ }
+ return ""
+}
+
+type ScrobblerScrobbleRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ UserId string `protobuf:"bytes,1,opt,name=user_id,json=userId,proto3" json:"user_id,omitempty"`
+ Username string `protobuf:"bytes,2,opt,name=username,proto3" json:"username,omitempty"`
+ Track *TrackInfo `protobuf:"bytes,3,opt,name=track,proto3" json:"track,omitempty"`
+ Timestamp int64 `protobuf:"varint,4,opt,name=timestamp,proto3" json:"timestamp,omitempty"`
+}
+
+func (x *ScrobblerScrobbleRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ScrobblerScrobbleRequest) GetUserId() string {
+ if x != nil {
+ return x.UserId
+ }
+ return ""
+}
+
+func (x *ScrobblerScrobbleRequest) GetUsername() string {
+ if x != nil {
+ return x.Username
+ }
+ return ""
+}
+
+func (x *ScrobblerScrobbleRequest) GetTrack() *TrackInfo {
+ if x != nil {
+ return x.Track
+ }
+ return nil
+}
+
+func (x *ScrobblerScrobbleRequest) GetTimestamp() int64 {
+ if x != nil {
+ return x.Timestamp
+ }
+ return 0
+}
+
+type ScrobblerScrobbleResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Error string `protobuf:"bytes,1,opt,name=error,proto3" json:"error,omitempty"`
+}
+
+func (x *ScrobblerScrobbleResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ScrobblerScrobbleResponse) GetError() string {
+ if x != nil {
+ return x.Error
+ }
+ return ""
+}
+
+type SchedulerCallbackRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ ScheduleId string `protobuf:"bytes,1,opt,name=schedule_id,json=scheduleId,proto3" json:"schedule_id,omitempty"` // ID of the scheduled job that triggered this callback
+ Payload []byte `protobuf:"bytes,2,opt,name=payload,proto3" json:"payload,omitempty"` // The data passed when the job was scheduled
+ IsRecurring bool `protobuf:"varint,3,opt,name=is_recurring,json=isRecurring,proto3" json:"is_recurring,omitempty"` // Whether this is from a recurring schedule (cron job)
+}
+
+func (x *SchedulerCallbackRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *SchedulerCallbackRequest) GetScheduleId() string {
+ if x != nil {
+ return x.ScheduleId
+ }
+ return ""
+}
+
+func (x *SchedulerCallbackRequest) GetPayload() []byte {
+ if x != nil {
+ return x.Payload
+ }
+ return nil
+}
+
+func (x *SchedulerCallbackRequest) GetIsRecurring() bool {
+ if x != nil {
+ return x.IsRecurring
+ }
+ return false
+}
+
+type SchedulerCallbackResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Error string `protobuf:"bytes,1,opt,name=error,proto3" json:"error,omitempty"` // Error message if the callback failed
+}
+
+func (x *SchedulerCallbackResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *SchedulerCallbackResponse) GetError() string {
+ if x != nil {
+ return x.Error
+ }
+ return ""
+}
+
+type InitRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Config map[string]string `protobuf:"bytes,1,rep,name=config,proto3" json:"config,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` // Configuration specific to this plugin
+}
+
+func (x *InitRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *InitRequest) GetConfig() map[string]string {
+ if x != nil {
+ return x.Config
+ }
+ return nil
+}
+
+type InitResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Error string `protobuf:"bytes,1,opt,name=error,proto3" json:"error,omitempty"` // Error message if initialization failed
+}
+
+func (x *InitResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *InitResponse) GetError() string {
+ if x != nil {
+ return x.Error
+ }
+ return ""
+}
+
+type OnTextMessageRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ ConnectionId string `protobuf:"bytes,1,opt,name=connection_id,json=connectionId,proto3" json:"connection_id,omitempty"`
+ Message string `protobuf:"bytes,2,opt,name=message,proto3" json:"message,omitempty"`
+}
+
+func (x *OnTextMessageRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *OnTextMessageRequest) GetConnectionId() string {
+ if x != nil {
+ return x.ConnectionId
+ }
+ return ""
+}
+
+func (x *OnTextMessageRequest) GetMessage() string {
+ if x != nil {
+ return x.Message
+ }
+ return ""
+}
+
+type OnTextMessageResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+}
+
+func (x *OnTextMessageResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+type OnBinaryMessageRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ ConnectionId string `protobuf:"bytes,1,opt,name=connection_id,json=connectionId,proto3" json:"connection_id,omitempty"`
+ Data []byte `protobuf:"bytes,2,opt,name=data,proto3" json:"data,omitempty"`
+}
+
+func (x *OnBinaryMessageRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *OnBinaryMessageRequest) GetConnectionId() string {
+ if x != nil {
+ return x.ConnectionId
+ }
+ return ""
+}
+
+func (x *OnBinaryMessageRequest) GetData() []byte {
+ if x != nil {
+ return x.Data
+ }
+ return nil
+}
+
+type OnBinaryMessageResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+}
+
+func (x *OnBinaryMessageResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+type OnErrorRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ ConnectionId string `protobuf:"bytes,1,opt,name=connection_id,json=connectionId,proto3" json:"connection_id,omitempty"`
+ Error string `protobuf:"bytes,2,opt,name=error,proto3" json:"error,omitempty"`
+}
+
+func (x *OnErrorRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *OnErrorRequest) GetConnectionId() string {
+ if x != nil {
+ return x.ConnectionId
+ }
+ return ""
+}
+
+func (x *OnErrorRequest) GetError() string {
+ if x != nil {
+ return x.Error
+ }
+ return ""
+}
+
+type OnErrorResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+}
+
+func (x *OnErrorResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+type OnCloseRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ ConnectionId string `protobuf:"bytes,1,opt,name=connection_id,json=connectionId,proto3" json:"connection_id,omitempty"`
+ Code int32 `protobuf:"varint,2,opt,name=code,proto3" json:"code,omitempty"`
+ Reason string `protobuf:"bytes,3,opt,name=reason,proto3" json:"reason,omitempty"`
+}
+
+func (x *OnCloseRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *OnCloseRequest) GetConnectionId() string {
+ if x != nil {
+ return x.ConnectionId
+ }
+ return ""
+}
+
+func (x *OnCloseRequest) GetCode() int32 {
+ if x != nil {
+ return x.Code
+ }
+ return 0
+}
+
+func (x *OnCloseRequest) GetReason() string {
+ if x != nil {
+ return x.Reason
+ }
+ return ""
+}
+
+type OnCloseResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+}
+
+func (x *OnCloseResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+// go:plugin type=plugin version=1
+type MetadataAgent interface {
+ // Artist metadata methods
+ GetArtistMBID(context.Context, *ArtistMBIDRequest) (*ArtistMBIDResponse, error)
+ GetArtistURL(context.Context, *ArtistURLRequest) (*ArtistURLResponse, error)
+ GetArtistBiography(context.Context, *ArtistBiographyRequest) (*ArtistBiographyResponse, error)
+ GetSimilarArtists(context.Context, *ArtistSimilarRequest) (*ArtistSimilarResponse, error)
+ GetArtistImages(context.Context, *ArtistImageRequest) (*ArtistImageResponse, error)
+ GetArtistTopSongs(context.Context, *ArtistTopSongsRequest) (*ArtistTopSongsResponse, error)
+ // Album metadata methods
+ GetAlbumInfo(context.Context, *AlbumInfoRequest) (*AlbumInfoResponse, error)
+ GetAlbumImages(context.Context, *AlbumImagesRequest) (*AlbumImagesResponse, error)
+}
+
+// go:plugin type=plugin version=1
+type Scrobbler interface {
+ IsAuthorized(context.Context, *ScrobblerIsAuthorizedRequest) (*ScrobblerIsAuthorizedResponse, error)
+ NowPlaying(context.Context, *ScrobblerNowPlayingRequest) (*ScrobblerNowPlayingResponse, error)
+ Scrobble(context.Context, *ScrobblerScrobbleRequest) (*ScrobblerScrobbleResponse, error)
+}
+
+// go:plugin type=plugin version=1
+type SchedulerCallback interface {
+ OnSchedulerCallback(context.Context, *SchedulerCallbackRequest) (*SchedulerCallbackResponse, error)
+}
+
+// go:plugin type=plugin version=1
+type LifecycleManagement interface {
+ OnInit(context.Context, *InitRequest) (*InitResponse, error)
+}
+
+// go:plugin type=plugin version=1
+type WebSocketCallback interface {
+ // Called when a text message is received
+ OnTextMessage(context.Context, *OnTextMessageRequest) (*OnTextMessageResponse, error)
+ // Called when a binary message is received
+ OnBinaryMessage(context.Context, *OnBinaryMessageRequest) (*OnBinaryMessageResponse, error)
+ // Called when an error occurs
+ OnError(context.Context, *OnErrorRequest) (*OnErrorResponse, error)
+ // Called when the connection is closed
+ OnClose(context.Context, *OnCloseRequest) (*OnCloseResponse, error)
+}
diff --git a/plugins/api/api.proto b/plugins/api/api.proto
new file mode 100644
index 000000000..7929ff9e6
--- /dev/null
+++ b/plugins/api/api.proto
@@ -0,0 +1,246 @@
+syntax = "proto3";
+
+package api;
+
+option go_package = "github.com/navidrome/navidrome/plugins/api;api";
+
+// go:plugin type=plugin version=1
+service MetadataAgent {
+ // Artist metadata methods
+ rpc GetArtistMBID(ArtistMBIDRequest) returns (ArtistMBIDResponse);
+ rpc GetArtistURL(ArtistURLRequest) returns (ArtistURLResponse);
+ rpc GetArtistBiography(ArtistBiographyRequest) returns (ArtistBiographyResponse);
+ rpc GetSimilarArtists(ArtistSimilarRequest) returns (ArtistSimilarResponse);
+ rpc GetArtistImages(ArtistImageRequest) returns (ArtistImageResponse);
+ rpc GetArtistTopSongs(ArtistTopSongsRequest) returns (ArtistTopSongsResponse);
+
+ // Album metadata methods
+ rpc GetAlbumInfo(AlbumInfoRequest) returns (AlbumInfoResponse);
+ rpc GetAlbumImages(AlbumImagesRequest) returns (AlbumImagesResponse);
+}
+
+message ArtistMBIDRequest {
+ string id = 1;
+ string name = 2;
+}
+
+message ArtistMBIDResponse {
+ string mbid = 1;
+}
+
+message ArtistURLRequest {
+ string id = 1;
+ string name = 2;
+ string mbid = 3;
+}
+
+message ArtistURLResponse {
+ string url = 1;
+}
+
+message ArtistBiographyRequest {
+ string id = 1;
+ string name = 2;
+ string mbid = 3;
+}
+
+message ArtistBiographyResponse {
+ string biography = 1;
+}
+
+message ArtistSimilarRequest {
+ string id = 1;
+ string name = 2;
+ string mbid = 3;
+ int32 limit = 4;
+}
+
+message Artist {
+ string name = 1;
+ string mbid = 2;
+}
+
+message ArtistSimilarResponse {
+ repeated Artist artists = 1;
+}
+
+message ArtistImageRequest {
+ string id = 1;
+ string name = 2;
+ string mbid = 3;
+}
+
+message ExternalImage {
+ string url = 1;
+ int32 size = 2;
+}
+
+message ArtistImageResponse {
+ repeated ExternalImage images = 1;
+}
+
+message ArtistTopSongsRequest {
+ string id = 1;
+ string artistName = 2;
+ string mbid = 3;
+ int32 count = 4;
+}
+
+message Song {
+ string name = 1;
+ string mbid = 2;
+}
+
+message ArtistTopSongsResponse {
+ repeated Song songs = 1;
+}
+
+message AlbumInfoRequest {
+ string name = 1;
+ string artist = 2;
+ string mbid = 3;
+}
+
+message AlbumInfo {
+ string name = 1;
+ string mbid = 2;
+ string description = 3;
+ string url = 4;
+}
+
+message AlbumInfoResponse {
+ AlbumInfo info = 1;
+}
+
+message AlbumImagesRequest {
+ string name = 1;
+ string artist = 2;
+ string mbid = 3;
+}
+
+message AlbumImagesResponse {
+ repeated ExternalImage images = 1;
+}
+
+// go:plugin type=plugin version=1
+service Scrobbler {
+ rpc IsAuthorized(ScrobblerIsAuthorizedRequest) returns (ScrobblerIsAuthorizedResponse);
+ rpc NowPlaying(ScrobblerNowPlayingRequest) returns (ScrobblerNowPlayingResponse);
+ rpc Scrobble(ScrobblerScrobbleRequest) returns (ScrobblerScrobbleResponse);
+}
+
+message ScrobblerIsAuthorizedRequest {
+ string user_id = 1;
+ string username = 2;
+}
+
+message ScrobblerIsAuthorizedResponse {
+ bool authorized = 1;
+ string error = 2;
+}
+
+message TrackInfo {
+ string id = 1;
+ string mbid = 2;
+ string name = 3;
+ string album = 4;
+ string album_mbid = 5;
+ repeated Artist artists = 6;
+ repeated Artist album_artists = 7;
+ int32 length = 8; // seconds
+ int32 position = 9; // seconds
+}
+
+message ScrobblerNowPlayingRequest {
+ string user_id = 1;
+ string username = 2;
+ TrackInfo track = 3;
+ int64 timestamp = 4;
+}
+
+message ScrobblerNowPlayingResponse {
+ string error = 1;
+}
+
+message ScrobblerScrobbleRequest {
+ string user_id = 1;
+ string username = 2;
+ TrackInfo track = 3;
+ int64 timestamp = 4;
+}
+
+message ScrobblerScrobbleResponse {
+ string error = 1;
+}
+
+// go:plugin type=plugin version=1
+service SchedulerCallback {
+ rpc OnSchedulerCallback(SchedulerCallbackRequest) returns (SchedulerCallbackResponse);
+}
+
+message SchedulerCallbackRequest {
+ string schedule_id = 1; // ID of the scheduled job that triggered this callback
+ bytes payload = 2; // The data passed when the job was scheduled
+ bool is_recurring = 3; // Whether this is from a recurring schedule (cron job)
+}
+
+message SchedulerCallbackResponse {
+ string error = 1; // Error message if the callback failed
+}
+
+// go:plugin type=plugin version=1
+service LifecycleManagement {
+ rpc OnInit(InitRequest) returns (InitResponse);
+}
+
+message InitRequest {
+ map config = 1; // Configuration specific to this plugin
+}
+
+message InitResponse {
+ string error = 1; // Error message if initialization failed
+}
+
+// go:plugin type=plugin version=1
+service WebSocketCallback {
+ // Called when a text message is received
+ rpc OnTextMessage(OnTextMessageRequest) returns (OnTextMessageResponse);
+
+ // Called when a binary message is received
+ rpc OnBinaryMessage(OnBinaryMessageRequest) returns (OnBinaryMessageResponse);
+
+ // Called when an error occurs
+ rpc OnError(OnErrorRequest) returns (OnErrorResponse);
+
+ // Called when the connection is closed
+ rpc OnClose(OnCloseRequest) returns (OnCloseResponse);
+}
+
+message OnTextMessageRequest {
+ string connection_id = 1;
+ string message = 2;
+}
+
+message OnTextMessageResponse {}
+
+message OnBinaryMessageRequest {
+ string connection_id = 1;
+ bytes data = 2;
+}
+
+message OnBinaryMessageResponse {}
+
+message OnErrorRequest {
+ string connection_id = 1;
+ string error = 2;
+}
+
+message OnErrorResponse {}
+
+message OnCloseRequest {
+ string connection_id = 1;
+ int32 code = 2;
+ string reason = 3;
+}
+
+message OnCloseResponse {}
\ No newline at end of file
diff --git a/plugins/api/api_host.pb.go b/plugins/api/api_host.pb.go
new file mode 100644
index 000000000..55e648c6c
--- /dev/null
+++ b/plugins/api/api_host.pb.go
@@ -0,0 +1,1688 @@
+//go:build !wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: api/api.proto
+
+package api
+
+import (
+ context "context"
+ errors "errors"
+ fmt "fmt"
+ wazero "github.com/tetratelabs/wazero"
+ api "github.com/tetratelabs/wazero/api"
+ sys "github.com/tetratelabs/wazero/sys"
+ os "os"
+)
+
+const MetadataAgentPluginAPIVersion = 1
+
+type MetadataAgentPlugin struct {
+ newRuntime func(context.Context) (wazero.Runtime, error)
+ moduleConfig wazero.ModuleConfig
+}
+
+func NewMetadataAgentPlugin(ctx context.Context, opts ...wazeroConfigOption) (*MetadataAgentPlugin, error) {
+ o := &WazeroConfig{
+ newRuntime: DefaultWazeroRuntime(),
+ moduleConfig: wazero.NewModuleConfig().WithStartFunctions("_initialize"),
+ }
+
+ for _, opt := range opts {
+ opt(o)
+ }
+
+ return &MetadataAgentPlugin{
+ newRuntime: o.newRuntime,
+ moduleConfig: o.moduleConfig,
+ }, nil
+}
+
+type metadataAgent interface {
+ Close(ctx context.Context) error
+ MetadataAgent
+}
+
+func (p *MetadataAgentPlugin) Load(ctx context.Context, pluginPath string) (metadataAgent, error) {
+ b, err := os.ReadFile(pluginPath)
+ if err != nil {
+ return nil, err
+ }
+
+ // Create a new runtime so that multiple modules will not conflict
+ r, err := p.newRuntime(ctx)
+ if err != nil {
+ return nil, err
+ }
+
+ // Compile the WebAssembly module using the default configuration.
+ code, err := r.CompileModule(ctx, b)
+ if err != nil {
+ return nil, err
+ }
+
+ // InstantiateModule runs the "_start" function, WASI's "main".
+ module, err := r.InstantiateModule(ctx, code, p.moduleConfig)
+ if err != nil {
+ // Note: Most compilers do not exit the module after running "_start",
+ // unless there was an Error. This allows you to call exported functions.
+ if exitErr, ok := err.(*sys.ExitError); ok && exitErr.ExitCode() != 0 {
+ return nil, fmt.Errorf("unexpected exit_code: %d", exitErr.ExitCode())
+ } else if !ok {
+ return nil, err
+ }
+ }
+
+ // Compare API versions with the loading plugin
+ apiVersion := module.ExportedFunction("metadata_agent_api_version")
+ if apiVersion == nil {
+ return nil, errors.New("metadata_agent_api_version is not exported")
+ }
+ results, err := apiVersion.Call(ctx)
+ if err != nil {
+ return nil, err
+ } else if len(results) != 1 {
+ return nil, errors.New("invalid metadata_agent_api_version signature")
+ }
+ if results[0] != MetadataAgentPluginAPIVersion {
+ return nil, fmt.Errorf("API version mismatch, host: %d, plugin: %d", MetadataAgentPluginAPIVersion, results[0])
+ }
+
+ getartistmbid := module.ExportedFunction("metadata_agent_get_artist_mbid")
+ if getartistmbid == nil {
+ return nil, errors.New("metadata_agent_get_artist_mbid is not exported")
+ }
+ getartisturl := module.ExportedFunction("metadata_agent_get_artist_url")
+ if getartisturl == nil {
+ return nil, errors.New("metadata_agent_get_artist_url is not exported")
+ }
+ getartistbiography := module.ExportedFunction("metadata_agent_get_artist_biography")
+ if getartistbiography == nil {
+ return nil, errors.New("metadata_agent_get_artist_biography is not exported")
+ }
+ getsimilarartists := module.ExportedFunction("metadata_agent_get_similar_artists")
+ if getsimilarartists == nil {
+ return nil, errors.New("metadata_agent_get_similar_artists is not exported")
+ }
+ getartistimages := module.ExportedFunction("metadata_agent_get_artist_images")
+ if getartistimages == nil {
+ return nil, errors.New("metadata_agent_get_artist_images is not exported")
+ }
+ getartisttopsongs := module.ExportedFunction("metadata_agent_get_artist_top_songs")
+ if getartisttopsongs == nil {
+ return nil, errors.New("metadata_agent_get_artist_top_songs is not exported")
+ }
+ getalbuminfo := module.ExportedFunction("metadata_agent_get_album_info")
+ if getalbuminfo == nil {
+ return nil, errors.New("metadata_agent_get_album_info is not exported")
+ }
+ getalbumimages := module.ExportedFunction("metadata_agent_get_album_images")
+ if getalbumimages == nil {
+ return nil, errors.New("metadata_agent_get_album_images is not exported")
+ }
+
+ malloc := module.ExportedFunction("malloc")
+ if malloc == nil {
+ return nil, errors.New("malloc is not exported")
+ }
+
+ free := module.ExportedFunction("free")
+ if free == nil {
+ return nil, errors.New("free is not exported")
+ }
+ return &metadataAgentPlugin{
+ runtime: r,
+ module: module,
+ malloc: malloc,
+ free: free,
+ getartistmbid: getartistmbid,
+ getartisturl: getartisturl,
+ getartistbiography: getartistbiography,
+ getsimilarartists: getsimilarartists,
+ getartistimages: getartistimages,
+ getartisttopsongs: getartisttopsongs,
+ getalbuminfo: getalbuminfo,
+ getalbumimages: getalbumimages,
+ }, nil
+}
+
+func (p *metadataAgentPlugin) Close(ctx context.Context) (err error) {
+ if r := p.runtime; r != nil {
+ r.Close(ctx)
+ }
+ return
+}
+
+type metadataAgentPlugin struct {
+ runtime wazero.Runtime
+ module api.Module
+ malloc api.Function
+ free api.Function
+ getartistmbid api.Function
+ getartisturl api.Function
+ getartistbiography api.Function
+ getsimilarartists api.Function
+ getartistimages api.Function
+ getartisttopsongs api.Function
+ getalbuminfo api.Function
+ getalbumimages api.Function
+}
+
+func (p *metadataAgentPlugin) GetArtistMBID(ctx context.Context, request *ArtistMBIDRequest) (*ArtistMBIDResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.getartistmbid.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(ArtistMBIDResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+func (p *metadataAgentPlugin) GetArtistURL(ctx context.Context, request *ArtistURLRequest) (*ArtistURLResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.getartisturl.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(ArtistURLResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+func (p *metadataAgentPlugin) GetArtistBiography(ctx context.Context, request *ArtistBiographyRequest) (*ArtistBiographyResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.getartistbiography.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(ArtistBiographyResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+func (p *metadataAgentPlugin) GetSimilarArtists(ctx context.Context, request *ArtistSimilarRequest) (*ArtistSimilarResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.getsimilarartists.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(ArtistSimilarResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+func (p *metadataAgentPlugin) GetArtistImages(ctx context.Context, request *ArtistImageRequest) (*ArtistImageResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.getartistimages.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(ArtistImageResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+func (p *metadataAgentPlugin) GetArtistTopSongs(ctx context.Context, request *ArtistTopSongsRequest) (*ArtistTopSongsResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.getartisttopsongs.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(ArtistTopSongsResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+func (p *metadataAgentPlugin) GetAlbumInfo(ctx context.Context, request *AlbumInfoRequest) (*AlbumInfoResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.getalbuminfo.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(AlbumInfoResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+func (p *metadataAgentPlugin) GetAlbumImages(ctx context.Context, request *AlbumImagesRequest) (*AlbumImagesResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.getalbumimages.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(AlbumImagesResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+
+const ScrobblerPluginAPIVersion = 1
+
+type ScrobblerPlugin struct {
+ newRuntime func(context.Context) (wazero.Runtime, error)
+ moduleConfig wazero.ModuleConfig
+}
+
+func NewScrobblerPlugin(ctx context.Context, opts ...wazeroConfigOption) (*ScrobblerPlugin, error) {
+ o := &WazeroConfig{
+ newRuntime: DefaultWazeroRuntime(),
+ moduleConfig: wazero.NewModuleConfig().WithStartFunctions("_initialize"),
+ }
+
+ for _, opt := range opts {
+ opt(o)
+ }
+
+ return &ScrobblerPlugin{
+ newRuntime: o.newRuntime,
+ moduleConfig: o.moduleConfig,
+ }, nil
+}
+
+type scrobbler interface {
+ Close(ctx context.Context) error
+ Scrobbler
+}
+
+func (p *ScrobblerPlugin) Load(ctx context.Context, pluginPath string) (scrobbler, error) {
+ b, err := os.ReadFile(pluginPath)
+ if err != nil {
+ return nil, err
+ }
+
+ // Create a new runtime so that multiple modules will not conflict
+ r, err := p.newRuntime(ctx)
+ if err != nil {
+ return nil, err
+ }
+
+ // Compile the WebAssembly module using the default configuration.
+ code, err := r.CompileModule(ctx, b)
+ if err != nil {
+ return nil, err
+ }
+
+ // InstantiateModule runs the "_start" function, WASI's "main".
+ module, err := r.InstantiateModule(ctx, code, p.moduleConfig)
+ if err != nil {
+ // Note: Most compilers do not exit the module after running "_start",
+ // unless there was an Error. This allows you to call exported functions.
+ if exitErr, ok := err.(*sys.ExitError); ok && exitErr.ExitCode() != 0 {
+ return nil, fmt.Errorf("unexpected exit_code: %d", exitErr.ExitCode())
+ } else if !ok {
+ return nil, err
+ }
+ }
+
+ // Compare API versions with the loading plugin
+ apiVersion := module.ExportedFunction("scrobbler_api_version")
+ if apiVersion == nil {
+ return nil, errors.New("scrobbler_api_version is not exported")
+ }
+ results, err := apiVersion.Call(ctx)
+ if err != nil {
+ return nil, err
+ } else if len(results) != 1 {
+ return nil, errors.New("invalid scrobbler_api_version signature")
+ }
+ if results[0] != ScrobblerPluginAPIVersion {
+ return nil, fmt.Errorf("API version mismatch, host: %d, plugin: %d", ScrobblerPluginAPIVersion, results[0])
+ }
+
+ isauthorized := module.ExportedFunction("scrobbler_is_authorized")
+ if isauthorized == nil {
+ return nil, errors.New("scrobbler_is_authorized is not exported")
+ }
+ nowplaying := module.ExportedFunction("scrobbler_now_playing")
+ if nowplaying == nil {
+ return nil, errors.New("scrobbler_now_playing is not exported")
+ }
+ scrobble := module.ExportedFunction("scrobbler_scrobble")
+ if scrobble == nil {
+ return nil, errors.New("scrobbler_scrobble is not exported")
+ }
+
+ malloc := module.ExportedFunction("malloc")
+ if malloc == nil {
+ return nil, errors.New("malloc is not exported")
+ }
+
+ free := module.ExportedFunction("free")
+ if free == nil {
+ return nil, errors.New("free is not exported")
+ }
+ return &scrobblerPlugin{
+ runtime: r,
+ module: module,
+ malloc: malloc,
+ free: free,
+ isauthorized: isauthorized,
+ nowplaying: nowplaying,
+ scrobble: scrobble,
+ }, nil
+}
+
+func (p *scrobblerPlugin) Close(ctx context.Context) (err error) {
+ if r := p.runtime; r != nil {
+ r.Close(ctx)
+ }
+ return
+}
+
+type scrobblerPlugin struct {
+ runtime wazero.Runtime
+ module api.Module
+ malloc api.Function
+ free api.Function
+ isauthorized api.Function
+ nowplaying api.Function
+ scrobble api.Function
+}
+
+func (p *scrobblerPlugin) IsAuthorized(ctx context.Context, request *ScrobblerIsAuthorizedRequest) (*ScrobblerIsAuthorizedResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.isauthorized.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(ScrobblerIsAuthorizedResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+func (p *scrobblerPlugin) NowPlaying(ctx context.Context, request *ScrobblerNowPlayingRequest) (*ScrobblerNowPlayingResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.nowplaying.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(ScrobblerNowPlayingResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+func (p *scrobblerPlugin) Scrobble(ctx context.Context, request *ScrobblerScrobbleRequest) (*ScrobblerScrobbleResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.scrobble.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(ScrobblerScrobbleResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+
+const SchedulerCallbackPluginAPIVersion = 1
+
+type SchedulerCallbackPlugin struct {
+ newRuntime func(context.Context) (wazero.Runtime, error)
+ moduleConfig wazero.ModuleConfig
+}
+
+func NewSchedulerCallbackPlugin(ctx context.Context, opts ...wazeroConfigOption) (*SchedulerCallbackPlugin, error) {
+ o := &WazeroConfig{
+ newRuntime: DefaultWazeroRuntime(),
+ moduleConfig: wazero.NewModuleConfig().WithStartFunctions("_initialize"),
+ }
+
+ for _, opt := range opts {
+ opt(o)
+ }
+
+ return &SchedulerCallbackPlugin{
+ newRuntime: o.newRuntime,
+ moduleConfig: o.moduleConfig,
+ }, nil
+}
+
+type schedulerCallback interface {
+ Close(ctx context.Context) error
+ SchedulerCallback
+}
+
+func (p *SchedulerCallbackPlugin) Load(ctx context.Context, pluginPath string) (schedulerCallback, error) {
+ b, err := os.ReadFile(pluginPath)
+ if err != nil {
+ return nil, err
+ }
+
+ // Create a new runtime so that multiple modules will not conflict
+ r, err := p.newRuntime(ctx)
+ if err != nil {
+ return nil, err
+ }
+
+ // Compile the WebAssembly module using the default configuration.
+ code, err := r.CompileModule(ctx, b)
+ if err != nil {
+ return nil, err
+ }
+
+ // InstantiateModule runs the "_start" function, WASI's "main".
+ module, err := r.InstantiateModule(ctx, code, p.moduleConfig)
+ if err != nil {
+ // Note: Most compilers do not exit the module after running "_start",
+ // unless there was an Error. This allows you to call exported functions.
+ if exitErr, ok := err.(*sys.ExitError); ok && exitErr.ExitCode() != 0 {
+ return nil, fmt.Errorf("unexpected exit_code: %d", exitErr.ExitCode())
+ } else if !ok {
+ return nil, err
+ }
+ }
+
+ // Compare API versions with the loading plugin
+ apiVersion := module.ExportedFunction("scheduler_callback_api_version")
+ if apiVersion == nil {
+ return nil, errors.New("scheduler_callback_api_version is not exported")
+ }
+ results, err := apiVersion.Call(ctx)
+ if err != nil {
+ return nil, err
+ } else if len(results) != 1 {
+ return nil, errors.New("invalid scheduler_callback_api_version signature")
+ }
+ if results[0] != SchedulerCallbackPluginAPIVersion {
+ return nil, fmt.Errorf("API version mismatch, host: %d, plugin: %d", SchedulerCallbackPluginAPIVersion, results[0])
+ }
+
+ onschedulercallback := module.ExportedFunction("scheduler_callback_on_scheduler_callback")
+ if onschedulercallback == nil {
+ return nil, errors.New("scheduler_callback_on_scheduler_callback is not exported")
+ }
+
+ malloc := module.ExportedFunction("malloc")
+ if malloc == nil {
+ return nil, errors.New("malloc is not exported")
+ }
+
+ free := module.ExportedFunction("free")
+ if free == nil {
+ return nil, errors.New("free is not exported")
+ }
+ return &schedulerCallbackPlugin{
+ runtime: r,
+ module: module,
+ malloc: malloc,
+ free: free,
+ onschedulercallback: onschedulercallback,
+ }, nil
+}
+
+func (p *schedulerCallbackPlugin) Close(ctx context.Context) (err error) {
+ if r := p.runtime; r != nil {
+ r.Close(ctx)
+ }
+ return
+}
+
+type schedulerCallbackPlugin struct {
+ runtime wazero.Runtime
+ module api.Module
+ malloc api.Function
+ free api.Function
+ onschedulercallback api.Function
+}
+
+func (p *schedulerCallbackPlugin) OnSchedulerCallback(ctx context.Context, request *SchedulerCallbackRequest) (*SchedulerCallbackResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.onschedulercallback.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(SchedulerCallbackResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+
+const LifecycleManagementPluginAPIVersion = 1
+
+type LifecycleManagementPlugin struct {
+ newRuntime func(context.Context) (wazero.Runtime, error)
+ moduleConfig wazero.ModuleConfig
+}
+
+func NewLifecycleManagementPlugin(ctx context.Context, opts ...wazeroConfigOption) (*LifecycleManagementPlugin, error) {
+ o := &WazeroConfig{
+ newRuntime: DefaultWazeroRuntime(),
+ moduleConfig: wazero.NewModuleConfig().WithStartFunctions("_initialize"),
+ }
+
+ for _, opt := range opts {
+ opt(o)
+ }
+
+ return &LifecycleManagementPlugin{
+ newRuntime: o.newRuntime,
+ moduleConfig: o.moduleConfig,
+ }, nil
+}
+
+type lifecycleManagement interface {
+ Close(ctx context.Context) error
+ LifecycleManagement
+}
+
+func (p *LifecycleManagementPlugin) Load(ctx context.Context, pluginPath string) (lifecycleManagement, error) {
+ b, err := os.ReadFile(pluginPath)
+ if err != nil {
+ return nil, err
+ }
+
+ // Create a new runtime so that multiple modules will not conflict
+ r, err := p.newRuntime(ctx)
+ if err != nil {
+ return nil, err
+ }
+
+ // Compile the WebAssembly module using the default configuration.
+ code, err := r.CompileModule(ctx, b)
+ if err != nil {
+ return nil, err
+ }
+
+ // InstantiateModule runs the "_start" function, WASI's "main".
+ module, err := r.InstantiateModule(ctx, code, p.moduleConfig)
+ if err != nil {
+ // Note: Most compilers do not exit the module after running "_start",
+ // unless there was an Error. This allows you to call exported functions.
+ if exitErr, ok := err.(*sys.ExitError); ok && exitErr.ExitCode() != 0 {
+ return nil, fmt.Errorf("unexpected exit_code: %d", exitErr.ExitCode())
+ } else if !ok {
+ return nil, err
+ }
+ }
+
+ // Compare API versions with the loading plugin
+ apiVersion := module.ExportedFunction("lifecycle_management_api_version")
+ if apiVersion == nil {
+ return nil, errors.New("lifecycle_management_api_version is not exported")
+ }
+ results, err := apiVersion.Call(ctx)
+ if err != nil {
+ return nil, err
+ } else if len(results) != 1 {
+ return nil, errors.New("invalid lifecycle_management_api_version signature")
+ }
+ if results[0] != LifecycleManagementPluginAPIVersion {
+ return nil, fmt.Errorf("API version mismatch, host: %d, plugin: %d", LifecycleManagementPluginAPIVersion, results[0])
+ }
+
+ oninit := module.ExportedFunction("lifecycle_management_on_init")
+ if oninit == nil {
+ return nil, errors.New("lifecycle_management_on_init is not exported")
+ }
+
+ malloc := module.ExportedFunction("malloc")
+ if malloc == nil {
+ return nil, errors.New("malloc is not exported")
+ }
+
+ free := module.ExportedFunction("free")
+ if free == nil {
+ return nil, errors.New("free is not exported")
+ }
+ return &lifecycleManagementPlugin{
+ runtime: r,
+ module: module,
+ malloc: malloc,
+ free: free,
+ oninit: oninit,
+ }, nil
+}
+
+func (p *lifecycleManagementPlugin) Close(ctx context.Context) (err error) {
+ if r := p.runtime; r != nil {
+ r.Close(ctx)
+ }
+ return
+}
+
+type lifecycleManagementPlugin struct {
+ runtime wazero.Runtime
+ module api.Module
+ malloc api.Function
+ free api.Function
+ oninit api.Function
+}
+
+func (p *lifecycleManagementPlugin) OnInit(ctx context.Context, request *InitRequest) (*InitResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.oninit.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(InitResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+
+const WebSocketCallbackPluginAPIVersion = 1
+
+type WebSocketCallbackPlugin struct {
+ newRuntime func(context.Context) (wazero.Runtime, error)
+ moduleConfig wazero.ModuleConfig
+}
+
+func NewWebSocketCallbackPlugin(ctx context.Context, opts ...wazeroConfigOption) (*WebSocketCallbackPlugin, error) {
+ o := &WazeroConfig{
+ newRuntime: DefaultWazeroRuntime(),
+ moduleConfig: wazero.NewModuleConfig().WithStartFunctions("_initialize"),
+ }
+
+ for _, opt := range opts {
+ opt(o)
+ }
+
+ return &WebSocketCallbackPlugin{
+ newRuntime: o.newRuntime,
+ moduleConfig: o.moduleConfig,
+ }, nil
+}
+
+type webSocketCallback interface {
+ Close(ctx context.Context) error
+ WebSocketCallback
+}
+
+func (p *WebSocketCallbackPlugin) Load(ctx context.Context, pluginPath string) (webSocketCallback, error) {
+ b, err := os.ReadFile(pluginPath)
+ if err != nil {
+ return nil, err
+ }
+
+ // Create a new runtime so that multiple modules will not conflict
+ r, err := p.newRuntime(ctx)
+ if err != nil {
+ return nil, err
+ }
+
+ // Compile the WebAssembly module using the default configuration.
+ code, err := r.CompileModule(ctx, b)
+ if err != nil {
+ return nil, err
+ }
+
+ // InstantiateModule runs the "_start" function, WASI's "main".
+ module, err := r.InstantiateModule(ctx, code, p.moduleConfig)
+ if err != nil {
+ // Note: Most compilers do not exit the module after running "_start",
+ // unless there was an Error. This allows you to call exported functions.
+ if exitErr, ok := err.(*sys.ExitError); ok && exitErr.ExitCode() != 0 {
+ return nil, fmt.Errorf("unexpected exit_code: %d", exitErr.ExitCode())
+ } else if !ok {
+ return nil, err
+ }
+ }
+
+ // Compare API versions with the loading plugin
+ apiVersion := module.ExportedFunction("web_socket_callback_api_version")
+ if apiVersion == nil {
+ return nil, errors.New("web_socket_callback_api_version is not exported")
+ }
+ results, err := apiVersion.Call(ctx)
+ if err != nil {
+ return nil, err
+ } else if len(results) != 1 {
+ return nil, errors.New("invalid web_socket_callback_api_version signature")
+ }
+ if results[0] != WebSocketCallbackPluginAPIVersion {
+ return nil, fmt.Errorf("API version mismatch, host: %d, plugin: %d", WebSocketCallbackPluginAPIVersion, results[0])
+ }
+
+ ontextmessage := module.ExportedFunction("web_socket_callback_on_text_message")
+ if ontextmessage == nil {
+ return nil, errors.New("web_socket_callback_on_text_message is not exported")
+ }
+ onbinarymessage := module.ExportedFunction("web_socket_callback_on_binary_message")
+ if onbinarymessage == nil {
+ return nil, errors.New("web_socket_callback_on_binary_message is not exported")
+ }
+ onerror := module.ExportedFunction("web_socket_callback_on_error")
+ if onerror == nil {
+ return nil, errors.New("web_socket_callback_on_error is not exported")
+ }
+ onclose := module.ExportedFunction("web_socket_callback_on_close")
+ if onclose == nil {
+ return nil, errors.New("web_socket_callback_on_close is not exported")
+ }
+
+ malloc := module.ExportedFunction("malloc")
+ if malloc == nil {
+ return nil, errors.New("malloc is not exported")
+ }
+
+ free := module.ExportedFunction("free")
+ if free == nil {
+ return nil, errors.New("free is not exported")
+ }
+ return &webSocketCallbackPlugin{
+ runtime: r,
+ module: module,
+ malloc: malloc,
+ free: free,
+ ontextmessage: ontextmessage,
+ onbinarymessage: onbinarymessage,
+ onerror: onerror,
+ onclose: onclose,
+ }, nil
+}
+
+func (p *webSocketCallbackPlugin) Close(ctx context.Context) (err error) {
+ if r := p.runtime; r != nil {
+ r.Close(ctx)
+ }
+ return
+}
+
+type webSocketCallbackPlugin struct {
+ runtime wazero.Runtime
+ module api.Module
+ malloc api.Function
+ free api.Function
+ ontextmessage api.Function
+ onbinarymessage api.Function
+ onerror api.Function
+ onclose api.Function
+}
+
+func (p *webSocketCallbackPlugin) OnTextMessage(ctx context.Context, request *OnTextMessageRequest) (*OnTextMessageResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.ontextmessage.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(OnTextMessageResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+func (p *webSocketCallbackPlugin) OnBinaryMessage(ctx context.Context, request *OnBinaryMessageRequest) (*OnBinaryMessageResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.onbinarymessage.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(OnBinaryMessageResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+func (p *webSocketCallbackPlugin) OnError(ctx context.Context, request *OnErrorRequest) (*OnErrorResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.onerror.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(OnErrorResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
+func (p *webSocketCallbackPlugin) OnClose(ctx context.Context, request *OnCloseRequest) (*OnCloseResponse, error) {
+ data, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ dataSize := uint64(len(data))
+
+ var dataPtr uint64
+ // If the input data is not empty, we must allocate the in-Wasm memory to store it, and pass to the plugin.
+ if dataSize != 0 {
+ results, err := p.malloc.Call(ctx, dataSize)
+ if err != nil {
+ return nil, err
+ }
+ dataPtr = results[0]
+ // This pointer is managed by the Wasm module, which is unaware of external usage.
+ // So, we have to free it when finished
+ defer p.free.Call(ctx, dataPtr)
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ if !p.module.Memory().Write(uint32(dataPtr), data) {
+ return nil, fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d", dataPtr, dataSize, p.module.Memory().Size())
+ }
+ }
+
+ ptrSize, err := p.onclose.Call(ctx, dataPtr, dataSize)
+ if err != nil {
+ return nil, err
+ }
+
+ resPtr := uint32(ptrSize[0] >> 32)
+ resSize := uint32(ptrSize[0])
+ var isErrResponse bool
+ if (resSize & (1 << 31)) > 0 {
+ isErrResponse = true
+ resSize &^= (1 << 31)
+ }
+
+ // We don't need the memory after deserialization: make sure it is freed.
+ if resPtr != 0 {
+ defer p.free.Call(ctx, uint64(resPtr))
+ }
+
+ // The pointer is a linear memory offset, which is where we write the name.
+ bytes, ok := p.module.Memory().Read(resPtr, resSize)
+ if !ok {
+ return nil, fmt.Errorf("Memory.Read(%d, %d) out of range of memory size %d",
+ resPtr, resSize, p.module.Memory().Size())
+ }
+
+ if isErrResponse {
+ return nil, errors.New(string(bytes))
+ }
+
+ response := new(OnCloseResponse)
+ if err = response.UnmarshalVT(bytes); err != nil {
+ return nil, err
+ }
+
+ return response, nil
+}
diff --git a/plugins/api/api_options.pb.go b/plugins/api/api_options.pb.go
new file mode 100644
index 000000000..430bf0a5c
--- /dev/null
+++ b/plugins/api/api_options.pb.go
@@ -0,0 +1,47 @@
+//go:build !wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: api/api.proto
+
+package api
+
+import (
+ context "context"
+ wazero "github.com/tetratelabs/wazero"
+ wasi_snapshot_preview1 "github.com/tetratelabs/wazero/imports/wasi_snapshot_preview1"
+)
+
+type wazeroConfigOption func(plugin *WazeroConfig)
+
+type WazeroNewRuntime func(context.Context) (wazero.Runtime, error)
+
+type WazeroConfig struct {
+ newRuntime func(context.Context) (wazero.Runtime, error)
+ moduleConfig wazero.ModuleConfig
+}
+
+func WazeroRuntime(newRuntime WazeroNewRuntime) wazeroConfigOption {
+ return func(h *WazeroConfig) {
+ h.newRuntime = newRuntime
+ }
+}
+
+func DefaultWazeroRuntime() WazeroNewRuntime {
+ return func(ctx context.Context) (wazero.Runtime, error) {
+ r := wazero.NewRuntime(ctx)
+ if _, err := wasi_snapshot_preview1.Instantiate(ctx, r); err != nil {
+ return nil, err
+ }
+
+ return r, nil
+ }
+}
+
+func WazeroModuleConfig(moduleConfig wazero.ModuleConfig) wazeroConfigOption {
+ return func(h *WazeroConfig) {
+ h.moduleConfig = moduleConfig
+ }
+}
diff --git a/plugins/api/api_plugin.pb.go b/plugins/api/api_plugin.pb.go
new file mode 100644
index 000000000..0a022be9b
--- /dev/null
+++ b/plugins/api/api_plugin.pb.go
@@ -0,0 +1,487 @@
+//go:build wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: api/api.proto
+
+package api
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+)
+
+const MetadataAgentPluginAPIVersion = 1
+
+//go:wasmexport metadata_agent_api_version
+func _metadata_agent_api_version() uint64 {
+ return MetadataAgentPluginAPIVersion
+}
+
+var metadataAgent MetadataAgent
+
+func RegisterMetadataAgent(p MetadataAgent) {
+ metadataAgent = p
+}
+
+//go:wasmexport metadata_agent_get_artist_mbid
+func _metadata_agent_get_artist_mbid(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(ArtistMBIDRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := metadataAgent.GetArtistMBID(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+//go:wasmexport metadata_agent_get_artist_url
+func _metadata_agent_get_artist_url(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(ArtistURLRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := metadataAgent.GetArtistURL(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+//go:wasmexport metadata_agent_get_artist_biography
+func _metadata_agent_get_artist_biography(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(ArtistBiographyRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := metadataAgent.GetArtistBiography(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+//go:wasmexport metadata_agent_get_similar_artists
+func _metadata_agent_get_similar_artists(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(ArtistSimilarRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := metadataAgent.GetSimilarArtists(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+//go:wasmexport metadata_agent_get_artist_images
+func _metadata_agent_get_artist_images(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(ArtistImageRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := metadataAgent.GetArtistImages(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+//go:wasmexport metadata_agent_get_artist_top_songs
+func _metadata_agent_get_artist_top_songs(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(ArtistTopSongsRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := metadataAgent.GetArtistTopSongs(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+//go:wasmexport metadata_agent_get_album_info
+func _metadata_agent_get_album_info(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(AlbumInfoRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := metadataAgent.GetAlbumInfo(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+//go:wasmexport metadata_agent_get_album_images
+func _metadata_agent_get_album_images(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(AlbumImagesRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := metadataAgent.GetAlbumImages(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+const ScrobblerPluginAPIVersion = 1
+
+//go:wasmexport scrobbler_api_version
+func _scrobbler_api_version() uint64 {
+ return ScrobblerPluginAPIVersion
+}
+
+var scrobbler Scrobbler
+
+func RegisterScrobbler(p Scrobbler) {
+ scrobbler = p
+}
+
+//go:wasmexport scrobbler_is_authorized
+func _scrobbler_is_authorized(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(ScrobblerIsAuthorizedRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := scrobbler.IsAuthorized(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+//go:wasmexport scrobbler_now_playing
+func _scrobbler_now_playing(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(ScrobblerNowPlayingRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := scrobbler.NowPlaying(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+//go:wasmexport scrobbler_scrobble
+func _scrobbler_scrobble(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(ScrobblerScrobbleRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := scrobbler.Scrobble(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+const SchedulerCallbackPluginAPIVersion = 1
+
+//go:wasmexport scheduler_callback_api_version
+func _scheduler_callback_api_version() uint64 {
+ return SchedulerCallbackPluginAPIVersion
+}
+
+var schedulerCallback SchedulerCallback
+
+func RegisterSchedulerCallback(p SchedulerCallback) {
+ schedulerCallback = p
+}
+
+//go:wasmexport scheduler_callback_on_scheduler_callback
+func _scheduler_callback_on_scheduler_callback(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(SchedulerCallbackRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := schedulerCallback.OnSchedulerCallback(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+const LifecycleManagementPluginAPIVersion = 1
+
+//go:wasmexport lifecycle_management_api_version
+func _lifecycle_management_api_version() uint64 {
+ return LifecycleManagementPluginAPIVersion
+}
+
+var lifecycleManagement LifecycleManagement
+
+func RegisterLifecycleManagement(p LifecycleManagement) {
+ lifecycleManagement = p
+}
+
+//go:wasmexport lifecycle_management_on_init
+func _lifecycle_management_on_init(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(InitRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := lifecycleManagement.OnInit(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+const WebSocketCallbackPluginAPIVersion = 1
+
+//go:wasmexport web_socket_callback_api_version
+func _web_socket_callback_api_version() uint64 {
+ return WebSocketCallbackPluginAPIVersion
+}
+
+var webSocketCallback WebSocketCallback
+
+func RegisterWebSocketCallback(p WebSocketCallback) {
+ webSocketCallback = p
+}
+
+//go:wasmexport web_socket_callback_on_text_message
+func _web_socket_callback_on_text_message(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(OnTextMessageRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := webSocketCallback.OnTextMessage(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+//go:wasmexport web_socket_callback_on_binary_message
+func _web_socket_callback_on_binary_message(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(OnBinaryMessageRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := webSocketCallback.OnBinaryMessage(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+//go:wasmexport web_socket_callback_on_error
+func _web_socket_callback_on_error(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(OnErrorRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := webSocketCallback.OnError(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
+
+//go:wasmexport web_socket_callback_on_close
+func _web_socket_callback_on_close(ptr, size uint32) uint64 {
+ b := wasm.PtrToByte(ptr, size)
+ req := new(OnCloseRequest)
+ if err := req.UnmarshalVT(b); err != nil {
+ return 0
+ }
+ response, err := webSocketCallback.OnClose(context.Background(), req)
+ if err != nil {
+ ptr, size = wasm.ByteToPtr([]byte(err.Error()))
+ return (uint64(ptr) << uint64(32)) | uint64(size) |
+ // Indicate that this is the error string by setting the 32-th bit, assuming that
+ // no data exceeds 31-bit size (2 GiB).
+ (1 << 31)
+ }
+
+ b, err = response.MarshalVT()
+ if err != nil {
+ return 0
+ }
+ ptr, size = wasm.ByteToPtr(b)
+ return (uint64(ptr) << uint64(32)) | uint64(size)
+}
diff --git a/plugins/api/api_plugin_dev.go b/plugins/api/api_plugin_dev.go
new file mode 100644
index 000000000..ed5a064b2
--- /dev/null
+++ b/plugins/api/api_plugin_dev.go
@@ -0,0 +1,34 @@
+//go:build !wasip1
+
+package api
+
+import "github.com/navidrome/navidrome/plugins/host/scheduler"
+
+// This file exists to provide stubs for the plugin registration functions when building for non-WASM targets.
+// This is useful for testing and development purposes, as it allows you to build and run your plugin code
+// without having to compile it to WASM.
+// In a real-world scenario, you would compile your plugin to WASM and use the generated registration functions.
+
+func RegisterMetadataAgent(MetadataAgent) {
+ panic("not implemented")
+}
+
+func RegisterScrobbler(Scrobbler) {
+ panic("not implemented")
+}
+
+func RegisterSchedulerCallback(SchedulerCallback) {
+ panic("not implemented")
+}
+
+func RegisterLifecycleManagement(LifecycleManagement) {
+ panic("not implemented")
+}
+
+func RegisterWebSocketCallback(WebSocketCallback) {
+ panic("not implemented")
+}
+
+func RegisterNamedSchedulerCallback(name string, cb SchedulerCallback) scheduler.SchedulerService {
+ panic("not implemented")
+}
diff --git a/plugins/api/api_plugin_dev_named_registry.go b/plugins/api/api_plugin_dev_named_registry.go
new file mode 100644
index 000000000..2ddb68779
--- /dev/null
+++ b/plugins/api/api_plugin_dev_named_registry.go
@@ -0,0 +1,94 @@
+//go:build wasip1
+
+package api
+
+import (
+ "context"
+ "strings"
+
+ "github.com/navidrome/navidrome/plugins/host/scheduler"
+)
+
+var callbacks = make(namedCallbacks)
+
+// RegisterNamedSchedulerCallback registers a named scheduler callback. Named callbacks allow multiple callbacks to be registered
+// within the same plugin, and for the schedules to be scoped to the named callback. If you only need a single callback, you can use
+// the default (unnamed) callback registration function, RegisterSchedulerCallback.
+// It returns a scheduler.SchedulerService that can be used to schedule jobs for the named callback.
+//
+// Notes:
+//
+// - You can't mix named and unnamed callbacks within the same plugin.
+// - The name should be unique within the plugin, and it's recommended to use a short, descriptive name.
+// - The name is case-sensitive.
+func RegisterNamedSchedulerCallback(name string, cb SchedulerCallback) scheduler.SchedulerService {
+ callbacks[name] = cb
+ RegisterSchedulerCallback(&callbacks)
+ return &namedSchedulerService{name: name, svc: scheduler.NewSchedulerService()}
+}
+
+const zwsp = string('\u200b')
+
+// namedCallbacks is a map of named scheduler callbacks. The key is the name of the callback, and the value is the callback itself.
+type namedCallbacks map[string]SchedulerCallback
+
+func parseKey(key string) (string, string) {
+ parts := strings.SplitN(key, zwsp, 2)
+ if len(parts) != 2 {
+ return "", ""
+ }
+ return parts[0], parts[1]
+}
+
+func (n *namedCallbacks) OnSchedulerCallback(ctx context.Context, req *SchedulerCallbackRequest) (*SchedulerCallbackResponse, error) {
+ name, scheduleId := parseKey(req.ScheduleId)
+ cb, exists := callbacks[name]
+ if !exists {
+ return nil, nil
+ }
+ req.ScheduleId = scheduleId
+ return cb.OnSchedulerCallback(ctx, req)
+}
+
+// namedSchedulerService is a wrapper around the host scheduler service that prefixes the schedule IDs with the
+// callback name. It is returned by RegisterNamedSchedulerCallback, and should be used by the plugin to schedule
+// jobs for the named callback.
+type namedSchedulerService struct {
+ name string
+ cb SchedulerCallback
+ svc scheduler.SchedulerService
+}
+
+func (n *namedSchedulerService) makeKey(id string) string {
+ return n.name + zwsp + id
+}
+
+func (n *namedSchedulerService) mapResponse(resp *scheduler.ScheduleResponse, err error) (*scheduler.ScheduleResponse, error) {
+ if err != nil {
+ return nil, err
+ }
+ _, resp.ScheduleId = parseKey(resp.ScheduleId)
+ return resp, nil
+}
+
+func (n *namedSchedulerService) ScheduleOneTime(ctx context.Context, request *scheduler.ScheduleOneTimeRequest) (*scheduler.ScheduleResponse, error) {
+ key := n.makeKey(request.ScheduleId)
+ request.ScheduleId = key
+ return n.mapResponse(n.svc.ScheduleOneTime(ctx, request))
+}
+
+func (n *namedSchedulerService) ScheduleRecurring(ctx context.Context, request *scheduler.ScheduleRecurringRequest) (*scheduler.ScheduleResponse, error) {
+ key := n.makeKey(request.ScheduleId)
+ request.ScheduleId = key
+ return n.mapResponse(n.svc.ScheduleRecurring(ctx, request))
+}
+
+func (n *namedSchedulerService) CancelSchedule(ctx context.Context, request *scheduler.CancelRequest) (*scheduler.CancelResponse, error) {
+ key := n.makeKey(request.ScheduleId)
+ request.ScheduleId = key
+ return n.svc.CancelSchedule(ctx, request)
+}
+
+func (n *namedSchedulerService) TimeNow(ctx context.Context, request *scheduler.TimeNowRequest) (*scheduler.TimeNowResponse, error) {
+ return n.svc.TimeNow(ctx, request)
+}
diff --git a/plugins/api/api_vtproto.pb.go b/plugins/api/api_vtproto.pb.go
new file mode 100644
index 000000000..11caa1946
--- /dev/null
+++ b/plugins/api/api_vtproto.pb.go
@@ -0,0 +1,7315 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: api/api.proto
+
+package api
+
+import (
+ fmt "fmt"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+ io "io"
+ bits "math/bits"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+func (m *ArtistMBIDRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ArtistMBIDRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ArtistMBIDRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Name) > 0 {
+ i -= len(m.Name)
+ copy(dAtA[i:], m.Name)
+ i = encodeVarint(dAtA, i, uint64(len(m.Name)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Id) > 0 {
+ i -= len(m.Id)
+ copy(dAtA[i:], m.Id)
+ i = encodeVarint(dAtA, i, uint64(len(m.Id)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ArtistMBIDResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ArtistMBIDResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ArtistMBIDResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Mbid) > 0 {
+ i -= len(m.Mbid)
+ copy(dAtA[i:], m.Mbid)
+ i = encodeVarint(dAtA, i, uint64(len(m.Mbid)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ArtistURLRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ArtistURLRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ArtistURLRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Mbid) > 0 {
+ i -= len(m.Mbid)
+ copy(dAtA[i:], m.Mbid)
+ i = encodeVarint(dAtA, i, uint64(len(m.Mbid)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.Name) > 0 {
+ i -= len(m.Name)
+ copy(dAtA[i:], m.Name)
+ i = encodeVarint(dAtA, i, uint64(len(m.Name)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Id) > 0 {
+ i -= len(m.Id)
+ copy(dAtA[i:], m.Id)
+ i = encodeVarint(dAtA, i, uint64(len(m.Id)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ArtistURLResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ArtistURLResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ArtistURLResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Url) > 0 {
+ i -= len(m.Url)
+ copy(dAtA[i:], m.Url)
+ i = encodeVarint(dAtA, i, uint64(len(m.Url)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ArtistBiographyRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ArtistBiographyRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ArtistBiographyRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Mbid) > 0 {
+ i -= len(m.Mbid)
+ copy(dAtA[i:], m.Mbid)
+ i = encodeVarint(dAtA, i, uint64(len(m.Mbid)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.Name) > 0 {
+ i -= len(m.Name)
+ copy(dAtA[i:], m.Name)
+ i = encodeVarint(dAtA, i, uint64(len(m.Name)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Id) > 0 {
+ i -= len(m.Id)
+ copy(dAtA[i:], m.Id)
+ i = encodeVarint(dAtA, i, uint64(len(m.Id)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ArtistBiographyResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ArtistBiographyResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ArtistBiographyResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Biography) > 0 {
+ i -= len(m.Biography)
+ copy(dAtA[i:], m.Biography)
+ i = encodeVarint(dAtA, i, uint64(len(m.Biography)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ArtistSimilarRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ArtistSimilarRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ArtistSimilarRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.Limit != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.Limit))
+ i--
+ dAtA[i] = 0x20
+ }
+ if len(m.Mbid) > 0 {
+ i -= len(m.Mbid)
+ copy(dAtA[i:], m.Mbid)
+ i = encodeVarint(dAtA, i, uint64(len(m.Mbid)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.Name) > 0 {
+ i -= len(m.Name)
+ copy(dAtA[i:], m.Name)
+ i = encodeVarint(dAtA, i, uint64(len(m.Name)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Id) > 0 {
+ i -= len(m.Id)
+ copy(dAtA[i:], m.Id)
+ i = encodeVarint(dAtA, i, uint64(len(m.Id)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *Artist) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *Artist) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *Artist) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Mbid) > 0 {
+ i -= len(m.Mbid)
+ copy(dAtA[i:], m.Mbid)
+ i = encodeVarint(dAtA, i, uint64(len(m.Mbid)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Name) > 0 {
+ i -= len(m.Name)
+ copy(dAtA[i:], m.Name)
+ i = encodeVarint(dAtA, i, uint64(len(m.Name)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ArtistSimilarResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ArtistSimilarResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ArtistSimilarResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Artists) > 0 {
+ for iNdEx := len(m.Artists) - 1; iNdEx >= 0; iNdEx-- {
+ size, err := m.Artists[iNdEx].MarshalToSizedBufferVT(dAtA[:i])
+ if err != nil {
+ return 0, err
+ }
+ i -= size
+ i = encodeVarint(dAtA, i, uint64(size))
+ i--
+ dAtA[i] = 0xa
+ }
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ArtistImageRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ArtistImageRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ArtistImageRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Mbid) > 0 {
+ i -= len(m.Mbid)
+ copy(dAtA[i:], m.Mbid)
+ i = encodeVarint(dAtA, i, uint64(len(m.Mbid)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.Name) > 0 {
+ i -= len(m.Name)
+ copy(dAtA[i:], m.Name)
+ i = encodeVarint(dAtA, i, uint64(len(m.Name)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Id) > 0 {
+ i -= len(m.Id)
+ copy(dAtA[i:], m.Id)
+ i = encodeVarint(dAtA, i, uint64(len(m.Id)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ExternalImage) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ExternalImage) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ExternalImage) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.Size != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.Size))
+ i--
+ dAtA[i] = 0x10
+ }
+ if len(m.Url) > 0 {
+ i -= len(m.Url)
+ copy(dAtA[i:], m.Url)
+ i = encodeVarint(dAtA, i, uint64(len(m.Url)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ArtistImageResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ArtistImageResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ArtistImageResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Images) > 0 {
+ for iNdEx := len(m.Images) - 1; iNdEx >= 0; iNdEx-- {
+ size, err := m.Images[iNdEx].MarshalToSizedBufferVT(dAtA[:i])
+ if err != nil {
+ return 0, err
+ }
+ i -= size
+ i = encodeVarint(dAtA, i, uint64(size))
+ i--
+ dAtA[i] = 0xa
+ }
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ArtistTopSongsRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ArtistTopSongsRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ArtistTopSongsRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.Count != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.Count))
+ i--
+ dAtA[i] = 0x20
+ }
+ if len(m.Mbid) > 0 {
+ i -= len(m.Mbid)
+ copy(dAtA[i:], m.Mbid)
+ i = encodeVarint(dAtA, i, uint64(len(m.Mbid)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.ArtistName) > 0 {
+ i -= len(m.ArtistName)
+ copy(dAtA[i:], m.ArtistName)
+ i = encodeVarint(dAtA, i, uint64(len(m.ArtistName)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Id) > 0 {
+ i -= len(m.Id)
+ copy(dAtA[i:], m.Id)
+ i = encodeVarint(dAtA, i, uint64(len(m.Id)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *Song) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *Song) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *Song) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Mbid) > 0 {
+ i -= len(m.Mbid)
+ copy(dAtA[i:], m.Mbid)
+ i = encodeVarint(dAtA, i, uint64(len(m.Mbid)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Name) > 0 {
+ i -= len(m.Name)
+ copy(dAtA[i:], m.Name)
+ i = encodeVarint(dAtA, i, uint64(len(m.Name)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ArtistTopSongsResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ArtistTopSongsResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ArtistTopSongsResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Songs) > 0 {
+ for iNdEx := len(m.Songs) - 1; iNdEx >= 0; iNdEx-- {
+ size, err := m.Songs[iNdEx].MarshalToSizedBufferVT(dAtA[:i])
+ if err != nil {
+ return 0, err
+ }
+ i -= size
+ i = encodeVarint(dAtA, i, uint64(size))
+ i--
+ dAtA[i] = 0xa
+ }
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *AlbumInfoRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *AlbumInfoRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *AlbumInfoRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Mbid) > 0 {
+ i -= len(m.Mbid)
+ copy(dAtA[i:], m.Mbid)
+ i = encodeVarint(dAtA, i, uint64(len(m.Mbid)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.Artist) > 0 {
+ i -= len(m.Artist)
+ copy(dAtA[i:], m.Artist)
+ i = encodeVarint(dAtA, i, uint64(len(m.Artist)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Name) > 0 {
+ i -= len(m.Name)
+ copy(dAtA[i:], m.Name)
+ i = encodeVarint(dAtA, i, uint64(len(m.Name)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *AlbumInfo) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *AlbumInfo) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *AlbumInfo) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Url) > 0 {
+ i -= len(m.Url)
+ copy(dAtA[i:], m.Url)
+ i = encodeVarint(dAtA, i, uint64(len(m.Url)))
+ i--
+ dAtA[i] = 0x22
+ }
+ if len(m.Description) > 0 {
+ i -= len(m.Description)
+ copy(dAtA[i:], m.Description)
+ i = encodeVarint(dAtA, i, uint64(len(m.Description)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.Mbid) > 0 {
+ i -= len(m.Mbid)
+ copy(dAtA[i:], m.Mbid)
+ i = encodeVarint(dAtA, i, uint64(len(m.Mbid)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Name) > 0 {
+ i -= len(m.Name)
+ copy(dAtA[i:], m.Name)
+ i = encodeVarint(dAtA, i, uint64(len(m.Name)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *AlbumInfoResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *AlbumInfoResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *AlbumInfoResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.Info != nil {
+ size, err := m.Info.MarshalToSizedBufferVT(dAtA[:i])
+ if err != nil {
+ return 0, err
+ }
+ i -= size
+ i = encodeVarint(dAtA, i, uint64(size))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *AlbumImagesRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *AlbumImagesRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *AlbumImagesRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Mbid) > 0 {
+ i -= len(m.Mbid)
+ copy(dAtA[i:], m.Mbid)
+ i = encodeVarint(dAtA, i, uint64(len(m.Mbid)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.Artist) > 0 {
+ i -= len(m.Artist)
+ copy(dAtA[i:], m.Artist)
+ i = encodeVarint(dAtA, i, uint64(len(m.Artist)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Name) > 0 {
+ i -= len(m.Name)
+ copy(dAtA[i:], m.Name)
+ i = encodeVarint(dAtA, i, uint64(len(m.Name)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *AlbumImagesResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *AlbumImagesResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *AlbumImagesResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Images) > 0 {
+ for iNdEx := len(m.Images) - 1; iNdEx >= 0; iNdEx-- {
+ size, err := m.Images[iNdEx].MarshalToSizedBufferVT(dAtA[:i])
+ if err != nil {
+ return 0, err
+ }
+ i -= size
+ i = encodeVarint(dAtA, i, uint64(size))
+ i--
+ dAtA[i] = 0xa
+ }
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ScrobblerIsAuthorizedRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ScrobblerIsAuthorizedRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ScrobblerIsAuthorizedRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Username) > 0 {
+ i -= len(m.Username)
+ copy(dAtA[i:], m.Username)
+ i = encodeVarint(dAtA, i, uint64(len(m.Username)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.UserId) > 0 {
+ i -= len(m.UserId)
+ copy(dAtA[i:], m.UserId)
+ i = encodeVarint(dAtA, i, uint64(len(m.UserId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ScrobblerIsAuthorizedResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ScrobblerIsAuthorizedResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ScrobblerIsAuthorizedResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Error) > 0 {
+ i -= len(m.Error)
+ copy(dAtA[i:], m.Error)
+ i = encodeVarint(dAtA, i, uint64(len(m.Error)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if m.Authorized {
+ i--
+ if m.Authorized {
+ dAtA[i] = 1
+ } else {
+ dAtA[i] = 0
+ }
+ i--
+ dAtA[i] = 0x8
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *TrackInfo) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *TrackInfo) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *TrackInfo) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.Position != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.Position))
+ i--
+ dAtA[i] = 0x48
+ }
+ if m.Length != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.Length))
+ i--
+ dAtA[i] = 0x40
+ }
+ if len(m.AlbumArtists) > 0 {
+ for iNdEx := len(m.AlbumArtists) - 1; iNdEx >= 0; iNdEx-- {
+ size, err := m.AlbumArtists[iNdEx].MarshalToSizedBufferVT(dAtA[:i])
+ if err != nil {
+ return 0, err
+ }
+ i -= size
+ i = encodeVarint(dAtA, i, uint64(size))
+ i--
+ dAtA[i] = 0x3a
+ }
+ }
+ if len(m.Artists) > 0 {
+ for iNdEx := len(m.Artists) - 1; iNdEx >= 0; iNdEx-- {
+ size, err := m.Artists[iNdEx].MarshalToSizedBufferVT(dAtA[:i])
+ if err != nil {
+ return 0, err
+ }
+ i -= size
+ i = encodeVarint(dAtA, i, uint64(size))
+ i--
+ dAtA[i] = 0x32
+ }
+ }
+ if len(m.AlbumMbid) > 0 {
+ i -= len(m.AlbumMbid)
+ copy(dAtA[i:], m.AlbumMbid)
+ i = encodeVarint(dAtA, i, uint64(len(m.AlbumMbid)))
+ i--
+ dAtA[i] = 0x2a
+ }
+ if len(m.Album) > 0 {
+ i -= len(m.Album)
+ copy(dAtA[i:], m.Album)
+ i = encodeVarint(dAtA, i, uint64(len(m.Album)))
+ i--
+ dAtA[i] = 0x22
+ }
+ if len(m.Name) > 0 {
+ i -= len(m.Name)
+ copy(dAtA[i:], m.Name)
+ i = encodeVarint(dAtA, i, uint64(len(m.Name)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.Mbid) > 0 {
+ i -= len(m.Mbid)
+ copy(dAtA[i:], m.Mbid)
+ i = encodeVarint(dAtA, i, uint64(len(m.Mbid)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Id) > 0 {
+ i -= len(m.Id)
+ copy(dAtA[i:], m.Id)
+ i = encodeVarint(dAtA, i, uint64(len(m.Id)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ScrobblerNowPlayingRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ScrobblerNowPlayingRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ScrobblerNowPlayingRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.Timestamp != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.Timestamp))
+ i--
+ dAtA[i] = 0x20
+ }
+ if m.Track != nil {
+ size, err := m.Track.MarshalToSizedBufferVT(dAtA[:i])
+ if err != nil {
+ return 0, err
+ }
+ i -= size
+ i = encodeVarint(dAtA, i, uint64(size))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.Username) > 0 {
+ i -= len(m.Username)
+ copy(dAtA[i:], m.Username)
+ i = encodeVarint(dAtA, i, uint64(len(m.Username)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.UserId) > 0 {
+ i -= len(m.UserId)
+ copy(dAtA[i:], m.UserId)
+ i = encodeVarint(dAtA, i, uint64(len(m.UserId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ScrobblerNowPlayingResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ScrobblerNowPlayingResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ScrobblerNowPlayingResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Error) > 0 {
+ i -= len(m.Error)
+ copy(dAtA[i:], m.Error)
+ i = encodeVarint(dAtA, i, uint64(len(m.Error)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ScrobblerScrobbleRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ScrobblerScrobbleRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ScrobblerScrobbleRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.Timestamp != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.Timestamp))
+ i--
+ dAtA[i] = 0x20
+ }
+ if m.Track != nil {
+ size, err := m.Track.MarshalToSizedBufferVT(dAtA[:i])
+ if err != nil {
+ return 0, err
+ }
+ i -= size
+ i = encodeVarint(dAtA, i, uint64(size))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.Username) > 0 {
+ i -= len(m.Username)
+ copy(dAtA[i:], m.Username)
+ i = encodeVarint(dAtA, i, uint64(len(m.Username)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.UserId) > 0 {
+ i -= len(m.UserId)
+ copy(dAtA[i:], m.UserId)
+ i = encodeVarint(dAtA, i, uint64(len(m.UserId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ScrobblerScrobbleResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ScrobblerScrobbleResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ScrobblerScrobbleResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Error) > 0 {
+ i -= len(m.Error)
+ copy(dAtA[i:], m.Error)
+ i = encodeVarint(dAtA, i, uint64(len(m.Error)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *SchedulerCallbackRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *SchedulerCallbackRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *SchedulerCallbackRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.IsRecurring {
+ i--
+ if m.IsRecurring {
+ dAtA[i] = 1
+ } else {
+ dAtA[i] = 0
+ }
+ i--
+ dAtA[i] = 0x18
+ }
+ if len(m.Payload) > 0 {
+ i -= len(m.Payload)
+ copy(dAtA[i:], m.Payload)
+ i = encodeVarint(dAtA, i, uint64(len(m.Payload)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.ScheduleId) > 0 {
+ i -= len(m.ScheduleId)
+ copy(dAtA[i:], m.ScheduleId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ScheduleId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *SchedulerCallbackResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *SchedulerCallbackResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *SchedulerCallbackResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Error) > 0 {
+ i -= len(m.Error)
+ copy(dAtA[i:], m.Error)
+ i = encodeVarint(dAtA, i, uint64(len(m.Error)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *InitRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *InitRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *InitRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Config) > 0 {
+ for k := range m.Config {
+ v := m.Config[k]
+ baseI := i
+ i -= len(v)
+ copy(dAtA[i:], v)
+ i = encodeVarint(dAtA, i, uint64(len(v)))
+ i--
+ dAtA[i] = 0x12
+ i -= len(k)
+ copy(dAtA[i:], k)
+ i = encodeVarint(dAtA, i, uint64(len(k)))
+ i--
+ dAtA[i] = 0xa
+ i = encodeVarint(dAtA, i, uint64(baseI-i))
+ i--
+ dAtA[i] = 0xa
+ }
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *InitResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *InitResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *InitResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Error) > 0 {
+ i -= len(m.Error)
+ copy(dAtA[i:], m.Error)
+ i = encodeVarint(dAtA, i, uint64(len(m.Error)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *OnTextMessageRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *OnTextMessageRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *OnTextMessageRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Message) > 0 {
+ i -= len(m.Message)
+ copy(dAtA[i:], m.Message)
+ i = encodeVarint(dAtA, i, uint64(len(m.Message)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.ConnectionId) > 0 {
+ i -= len(m.ConnectionId)
+ copy(dAtA[i:], m.ConnectionId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ConnectionId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *OnTextMessageResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *OnTextMessageResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *OnTextMessageResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *OnBinaryMessageRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *OnBinaryMessageRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *OnBinaryMessageRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Data) > 0 {
+ i -= len(m.Data)
+ copy(dAtA[i:], m.Data)
+ i = encodeVarint(dAtA, i, uint64(len(m.Data)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.ConnectionId) > 0 {
+ i -= len(m.ConnectionId)
+ copy(dAtA[i:], m.ConnectionId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ConnectionId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *OnBinaryMessageResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *OnBinaryMessageResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *OnBinaryMessageResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *OnErrorRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *OnErrorRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *OnErrorRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Error) > 0 {
+ i -= len(m.Error)
+ copy(dAtA[i:], m.Error)
+ i = encodeVarint(dAtA, i, uint64(len(m.Error)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.ConnectionId) > 0 {
+ i -= len(m.ConnectionId)
+ copy(dAtA[i:], m.ConnectionId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ConnectionId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *OnErrorResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *OnErrorResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *OnErrorResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *OnCloseRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *OnCloseRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *OnCloseRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Reason) > 0 {
+ i -= len(m.Reason)
+ copy(dAtA[i:], m.Reason)
+ i = encodeVarint(dAtA, i, uint64(len(m.Reason)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if m.Code != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.Code))
+ i--
+ dAtA[i] = 0x10
+ }
+ if len(m.ConnectionId) > 0 {
+ i -= len(m.ConnectionId)
+ copy(dAtA[i:], m.ConnectionId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ConnectionId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *OnCloseResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *OnCloseResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *OnCloseResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ return len(dAtA) - i, nil
+}
+
+func encodeVarint(dAtA []byte, offset int, v uint64) int {
+ offset -= sov(v)
+ base := offset
+ for v >= 1<<7 {
+ dAtA[offset] = uint8(v&0x7f | 0x80)
+ v >>= 7
+ offset++
+ }
+ dAtA[offset] = uint8(v)
+ return base
+}
+func (m *ArtistMBIDRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Id)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Name)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ArtistMBIDResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Mbid)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ArtistURLRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Id)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Name)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Mbid)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ArtistURLResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Url)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ArtistBiographyRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Id)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Name)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Mbid)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ArtistBiographyResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Biography)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ArtistSimilarRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Id)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Name)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Mbid)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.Limit != 0 {
+ n += 1 + sov(uint64(m.Limit))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *Artist) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Name)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Mbid)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ArtistSimilarResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if len(m.Artists) > 0 {
+ for _, e := range m.Artists {
+ l = e.SizeVT()
+ n += 1 + l + sov(uint64(l))
+ }
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ArtistImageRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Id)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Name)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Mbid)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ExternalImage) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Url)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.Size != 0 {
+ n += 1 + sov(uint64(m.Size))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ArtistImageResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if len(m.Images) > 0 {
+ for _, e := range m.Images {
+ l = e.SizeVT()
+ n += 1 + l + sov(uint64(l))
+ }
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ArtistTopSongsRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Id)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.ArtistName)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Mbid)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.Count != 0 {
+ n += 1 + sov(uint64(m.Count))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *Song) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Name)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Mbid)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ArtistTopSongsResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if len(m.Songs) > 0 {
+ for _, e := range m.Songs {
+ l = e.SizeVT()
+ n += 1 + l + sov(uint64(l))
+ }
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *AlbumInfoRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Name)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Artist)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Mbid)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *AlbumInfo) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Name)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Mbid)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Description)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Url)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *AlbumInfoResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if m.Info != nil {
+ l = m.Info.SizeVT()
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *AlbumImagesRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Name)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Artist)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Mbid)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *AlbumImagesResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if len(m.Images) > 0 {
+ for _, e := range m.Images {
+ l = e.SizeVT()
+ n += 1 + l + sov(uint64(l))
+ }
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ScrobblerIsAuthorizedRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.UserId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Username)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ScrobblerIsAuthorizedResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if m.Authorized {
+ n += 2
+ }
+ l = len(m.Error)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *TrackInfo) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Id)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Mbid)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Name)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Album)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.AlbumMbid)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if len(m.Artists) > 0 {
+ for _, e := range m.Artists {
+ l = e.SizeVT()
+ n += 1 + l + sov(uint64(l))
+ }
+ }
+ if len(m.AlbumArtists) > 0 {
+ for _, e := range m.AlbumArtists {
+ l = e.SizeVT()
+ n += 1 + l + sov(uint64(l))
+ }
+ }
+ if m.Length != 0 {
+ n += 1 + sov(uint64(m.Length))
+ }
+ if m.Position != 0 {
+ n += 1 + sov(uint64(m.Position))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ScrobblerNowPlayingRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.UserId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Username)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.Track != nil {
+ l = m.Track.SizeVT()
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.Timestamp != 0 {
+ n += 1 + sov(uint64(m.Timestamp))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ScrobblerNowPlayingResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Error)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ScrobblerScrobbleRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.UserId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Username)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.Track != nil {
+ l = m.Track.SizeVT()
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.Timestamp != 0 {
+ n += 1 + sov(uint64(m.Timestamp))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ScrobblerScrobbleResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Error)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *SchedulerCallbackRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.ScheduleId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Payload)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.IsRecurring {
+ n += 2
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *SchedulerCallbackResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Error)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *InitRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if len(m.Config) > 0 {
+ for k, v := range m.Config {
+ _ = k
+ _ = v
+ mapEntrySize := 1 + len(k) + sov(uint64(len(k))) + 1 + len(v) + sov(uint64(len(v)))
+ n += mapEntrySize + 1 + sov(uint64(mapEntrySize))
+ }
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *InitResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Error)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *OnTextMessageRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.ConnectionId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Message)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *OnTextMessageResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *OnBinaryMessageRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.ConnectionId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Data)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *OnBinaryMessageResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *OnErrorRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.ConnectionId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Error)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *OnErrorResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *OnCloseRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.ConnectionId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.Code != 0 {
+ n += 1 + sov(uint64(m.Code))
+ }
+ l = len(m.Reason)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *OnCloseResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ n += len(m.unknownFields)
+ return n
+}
+
+func sov(x uint64) (n int) {
+ return (bits.Len64(x|1) + 6) / 7
+}
+func soz(x uint64) (n int) {
+ return sov(uint64((x << 1) ^ uint64((int64(x) >> 63))))
+}
+func (m *ArtistMBIDRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ArtistMBIDRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ArtistMBIDRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Id", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Id = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Name = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ArtistMBIDResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ArtistMBIDResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ArtistMBIDResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Mbid", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Mbid = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ArtistURLRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ArtistURLRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ArtistURLRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Id", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Id = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Name = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Mbid", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Mbid = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ArtistURLResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ArtistURLResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ArtistURLResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Url", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Url = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ArtistBiographyRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ArtistBiographyRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ArtistBiographyRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Id", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Id = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Name = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Mbid", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Mbid = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ArtistBiographyResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ArtistBiographyResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ArtistBiographyResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Biography", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Biography = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ArtistSimilarRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ArtistSimilarRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ArtistSimilarRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Id", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Id = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Name = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Mbid", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Mbid = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 4:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Limit", wireType)
+ }
+ m.Limit = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.Limit |= int32(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *Artist) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: Artist: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: Artist: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Name = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Mbid", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Mbid = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ArtistSimilarResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ArtistSimilarResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ArtistSimilarResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Artists", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Artists = append(m.Artists, &Artist{})
+ if err := m.Artists[len(m.Artists)-1].UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil {
+ return err
+ }
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ArtistImageRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ArtistImageRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ArtistImageRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Id", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Id = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Name = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Mbid", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Mbid = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ExternalImage) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ExternalImage: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ExternalImage: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Url", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Url = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Size", wireType)
+ }
+ m.Size = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.Size |= int32(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ArtistImageResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ArtistImageResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ArtistImageResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Images", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Images = append(m.Images, &ExternalImage{})
+ if err := m.Images[len(m.Images)-1].UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil {
+ return err
+ }
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ArtistTopSongsRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ArtistTopSongsRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ArtistTopSongsRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Id", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Id = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ArtistName", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ArtistName = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Mbid", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Mbid = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 4:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Count", wireType)
+ }
+ m.Count = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.Count |= int32(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *Song) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: Song: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: Song: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Name = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Mbid", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Mbid = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ArtistTopSongsResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ArtistTopSongsResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ArtistTopSongsResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Songs", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Songs = append(m.Songs, &Song{})
+ if err := m.Songs[len(m.Songs)-1].UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil {
+ return err
+ }
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *AlbumInfoRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: AlbumInfoRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: AlbumInfoRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Name = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Artist", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Artist = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Mbid", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Mbid = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *AlbumInfo) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: AlbumInfo: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: AlbumInfo: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Name = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Mbid", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Mbid = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Description", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Description = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 4:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Url", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Url = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *AlbumInfoResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: AlbumInfoResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: AlbumInfoResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Info", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ if m.Info == nil {
+ m.Info = &AlbumInfo{}
+ }
+ if err := m.Info.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil {
+ return err
+ }
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *AlbumImagesRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: AlbumImagesRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: AlbumImagesRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Name = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Artist", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Artist = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Mbid", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Mbid = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *AlbumImagesResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: AlbumImagesResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: AlbumImagesResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Images", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Images = append(m.Images, &ExternalImage{})
+ if err := m.Images[len(m.Images)-1].UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil {
+ return err
+ }
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ScrobblerIsAuthorizedRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ScrobblerIsAuthorizedRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ScrobblerIsAuthorizedRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field UserId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.UserId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Username", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Username = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ScrobblerIsAuthorizedResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ScrobblerIsAuthorizedResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ScrobblerIsAuthorizedResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Authorized", wireType)
+ }
+ var v int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ v |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ m.Authorized = bool(v != 0)
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Error", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Error = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *TrackInfo) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: TrackInfo: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: TrackInfo: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Id", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Id = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Mbid", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Mbid = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Name = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 4:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Album", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Album = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 5:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field AlbumMbid", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.AlbumMbid = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 6:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Artists", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Artists = append(m.Artists, &Artist{})
+ if err := m.Artists[len(m.Artists)-1].UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil {
+ return err
+ }
+ iNdEx = postIndex
+ case 7:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field AlbumArtists", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.AlbumArtists = append(m.AlbumArtists, &Artist{})
+ if err := m.AlbumArtists[len(m.AlbumArtists)-1].UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil {
+ return err
+ }
+ iNdEx = postIndex
+ case 8:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Length", wireType)
+ }
+ m.Length = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.Length |= int32(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ case 9:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Position", wireType)
+ }
+ m.Position = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.Position |= int32(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ScrobblerNowPlayingRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ScrobblerNowPlayingRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ScrobblerNowPlayingRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field UserId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.UserId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Username", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Username = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Track", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ if m.Track == nil {
+ m.Track = &TrackInfo{}
+ }
+ if err := m.Track.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil {
+ return err
+ }
+ iNdEx = postIndex
+ case 4:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Timestamp", wireType)
+ }
+ m.Timestamp = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.Timestamp |= int64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ScrobblerNowPlayingResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ScrobblerNowPlayingResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ScrobblerNowPlayingResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Error", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Error = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ScrobblerScrobbleRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ScrobblerScrobbleRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ScrobblerScrobbleRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field UserId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.UserId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Username", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Username = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Track", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ if m.Track == nil {
+ m.Track = &TrackInfo{}
+ }
+ if err := m.Track.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil {
+ return err
+ }
+ iNdEx = postIndex
+ case 4:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Timestamp", wireType)
+ }
+ m.Timestamp = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.Timestamp |= int64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ScrobblerScrobbleResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ScrobblerScrobbleResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ScrobblerScrobbleResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Error", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Error = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *SchedulerCallbackRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: SchedulerCallbackRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: SchedulerCallbackRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ScheduleId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ScheduleId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Payload", wireType)
+ }
+ var byteLen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ byteLen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if byteLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + byteLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Payload = append(m.Payload[:0], dAtA[iNdEx:postIndex]...)
+ if m.Payload == nil {
+ m.Payload = []byte{}
+ }
+ iNdEx = postIndex
+ case 3:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field IsRecurring", wireType)
+ }
+ var v int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ v |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ m.IsRecurring = bool(v != 0)
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *SchedulerCallbackResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: SchedulerCallbackResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: SchedulerCallbackResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Error", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Error = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *InitRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: InitRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: InitRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Config", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ if m.Config == nil {
+ m.Config = make(map[string]string)
+ }
+ var mapkey string
+ var mapvalue string
+ for iNdEx < postIndex {
+ entryPreIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ if fieldNum == 1 {
+ var stringLenmapkey uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLenmapkey |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLenmapkey := int(stringLenmapkey)
+ if intStringLenmapkey < 0 {
+ return ErrInvalidLength
+ }
+ postStringIndexmapkey := iNdEx + intStringLenmapkey
+ if postStringIndexmapkey < 0 {
+ return ErrInvalidLength
+ }
+ if postStringIndexmapkey > l {
+ return io.ErrUnexpectedEOF
+ }
+ mapkey = string(dAtA[iNdEx:postStringIndexmapkey])
+ iNdEx = postStringIndexmapkey
+ } else if fieldNum == 2 {
+ var stringLenmapvalue uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLenmapvalue |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLenmapvalue := int(stringLenmapvalue)
+ if intStringLenmapvalue < 0 {
+ return ErrInvalidLength
+ }
+ postStringIndexmapvalue := iNdEx + intStringLenmapvalue
+ if postStringIndexmapvalue < 0 {
+ return ErrInvalidLength
+ }
+ if postStringIndexmapvalue > l {
+ return io.ErrUnexpectedEOF
+ }
+ mapvalue = string(dAtA[iNdEx:postStringIndexmapvalue])
+ iNdEx = postStringIndexmapvalue
+ } else {
+ iNdEx = entryPreIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > postIndex {
+ return io.ErrUnexpectedEOF
+ }
+ iNdEx += skippy
+ }
+ }
+ m.Config[mapkey] = mapvalue
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *InitResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: InitResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: InitResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Error", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Error = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *OnTextMessageRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: OnTextMessageRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: OnTextMessageRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ConnectionId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ConnectionId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Message", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Message = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *OnTextMessageResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: OnTextMessageResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: OnTextMessageResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *OnBinaryMessageRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: OnBinaryMessageRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: OnBinaryMessageRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ConnectionId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ConnectionId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Data", wireType)
+ }
+ var byteLen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ byteLen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if byteLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + byteLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Data = append(m.Data[:0], dAtA[iNdEx:postIndex]...)
+ if m.Data == nil {
+ m.Data = []byte{}
+ }
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *OnBinaryMessageResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: OnBinaryMessageResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: OnBinaryMessageResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *OnErrorRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: OnErrorRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: OnErrorRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ConnectionId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ConnectionId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Error", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Error = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *OnErrorResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: OnErrorResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: OnErrorResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *OnCloseRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: OnCloseRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: OnCloseRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ConnectionId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ConnectionId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Code", wireType)
+ }
+ m.Code = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.Code |= int32(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Reason", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Reason = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *OnCloseResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: OnCloseResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: OnCloseResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+
+func skip(dAtA []byte) (n int, err error) {
+ l := len(dAtA)
+ iNdEx := 0
+ depth := 0
+ for iNdEx < l {
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= (uint64(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ wireType := int(wire & 0x7)
+ switch wireType {
+ case 0:
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ iNdEx++
+ if dAtA[iNdEx-1] < 0x80 {
+ break
+ }
+ }
+ case 1:
+ iNdEx += 8
+ case 2:
+ var length int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ length |= (int(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if length < 0 {
+ return 0, ErrInvalidLength
+ }
+ iNdEx += length
+ case 3:
+ depth++
+ case 4:
+ if depth == 0 {
+ return 0, ErrUnexpectedEndOfGroup
+ }
+ depth--
+ case 5:
+ iNdEx += 4
+ default:
+ return 0, fmt.Errorf("proto: illegal wireType %d", wireType)
+ }
+ if iNdEx < 0 {
+ return 0, ErrInvalidLength
+ }
+ if depth == 0 {
+ return iNdEx, nil
+ }
+ }
+ return 0, io.ErrUnexpectedEOF
+}
+
+var (
+ ErrInvalidLength = fmt.Errorf("proto: negative length found during unmarshaling")
+ ErrIntOverflow = fmt.Errorf("proto: integer overflow")
+ ErrUnexpectedEndOfGroup = fmt.Errorf("proto: unexpected end of group")
+)
diff --git a/plugins/api/errors.go b/plugins/api/errors.go
new file mode 100644
index 000000000..796774b15
--- /dev/null
+++ b/plugins/api/errors.go
@@ -0,0 +1,12 @@
+package api
+
+import "errors"
+
+var (
+ // ErrNotImplemented indicates that the plugin does not implement the requested method.
+ // No logic should be executed by the plugin.
+ ErrNotImplemented = errors.New("plugin:not_implemented")
+
+ // ErrNotFound indicates that the requested resource was not found by the plugin.
+ ErrNotFound = errors.New("plugin:not_found")
+)
diff --git a/plugins/base_capability.go b/plugins/base_capability.go
new file mode 100644
index 000000000..6572a25ec
--- /dev/null
+++ b/plugins/base_capability.go
@@ -0,0 +1,159 @@
+package plugins
+
+import (
+ "context"
+ "errors"
+ "fmt"
+ "time"
+
+ "github.com/navidrome/navidrome/core/metrics"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model/id"
+ "github.com/navidrome/navidrome/plugins/api"
+)
+
+// newBaseCapability creates a new instance of baseCapability with the required parameters.
+func newBaseCapability[S any, P any](wasmPath, id, capability string, m metrics.Metrics, loader P, loadFunc loaderFunc[S, P]) *baseCapability[S, P] {
+ return &baseCapability[S, P]{
+ wasmPath: wasmPath,
+ id: id,
+ capability: capability,
+ loader: loader,
+ loadFunc: loadFunc,
+ metrics: m,
+ }
+}
+
+// LoaderFunc is a generic function type that loads a plugin instance.
+type loaderFunc[S any, P any] func(ctx context.Context, loader P, path string) (S, error)
+
+// baseCapability is a generic base implementation for WASM plugins.
+// S is the capability interface type and P is the plugin loader type.
+type baseCapability[S any, P any] struct {
+ wasmPath string
+ id string
+ capability string
+ loader P
+ loadFunc loaderFunc[S, P]
+ metrics metrics.Metrics
+}
+
+func (w *baseCapability[S, P]) PluginID() string {
+ return w.id
+}
+
+func (w *baseCapability[S, P]) serviceName() string {
+ return w.id + "_" + w.capability
+}
+
+func (w *baseCapability[S, P]) getMetrics() metrics.Metrics {
+ return w.metrics
+}
+
+// getInstance loads a new plugin instance and returns a cleanup function.
+func (w *baseCapability[S, P]) getInstance(ctx context.Context, methodName string) (S, func(), error) {
+ start := time.Now()
+ // Add context metadata for tracing
+ ctx = log.NewContext(ctx, "capability", w.serviceName(), "method", methodName)
+
+ inst, err := w.loadFunc(ctx, w.loader, w.wasmPath)
+ if err != nil {
+ var zero S
+ return zero, func() {}, fmt.Errorf("baseCapability: failed to load instance for %s: %w", w.serviceName(), err)
+ }
+ // Add context metadata for tracing
+ ctx = log.NewContext(ctx, "instanceID", getInstanceID(inst))
+ log.Trace(ctx, "baseCapability: loaded instance", "elapsed", time.Since(start))
+ return inst, func() {
+ log.Trace(ctx, "baseCapability: finished using instance", "elapsed", time.Since(start))
+ if closer, ok := any(inst).(interface{ Close(context.Context) error }); ok {
+ _ = closer.Close(ctx)
+ }
+ }, nil
+}
+
+type wasmPlugin[S any] interface {
+ PluginID() string
+ getInstance(ctx context.Context, methodName string) (S, func(), error)
+ getMetrics() metrics.Metrics
+}
+
+func callMethod[S any, R any](ctx context.Context, wp WasmPlugin, methodName string, fn func(inst S) (R, error)) (R, error) {
+ // Add a unique call ID to the context for tracing
+ ctx = log.NewContext(ctx, "callID", id.NewRandom())
+ var r R
+
+ p, ok := wp.(wasmPlugin[S])
+ if !ok {
+ log.Error(ctx, "callMethod: not a wasm plugin", "method", methodName, "pluginID", wp.PluginID())
+ return r, fmt.Errorf("wasm plugin: not a wasm plugin: %s", wp.PluginID())
+ }
+
+ inst, done, err := p.getInstance(ctx, methodName)
+ if err != nil {
+ return r, err
+ }
+ start := time.Now()
+ defer done()
+ r, err = checkErr(fn(inst))
+ elapsed := time.Since(start)
+
+ if !errors.Is(err, api.ErrNotImplemented) {
+ id := p.PluginID()
+ isOk := err == nil
+ metrics := p.getMetrics()
+ if metrics != nil {
+ metrics.RecordPluginRequest(ctx, id, methodName, isOk, elapsed.Milliseconds())
+ log.Trace(ctx, "callMethod: sending metrics", "plugin", id, "method", methodName, "ok", isOk, "elapsed", elapsed)
+ }
+ }
+
+ return r, err
+}
+
+// errorResponse is an interface that defines a method to retrieve an error message.
+// It is automatically implemented (generated) by all plugin responses that have an Error field
+type errorResponse interface {
+ GetError() string
+}
+
+// checkErr returns an updated error if the response implements errorResponse and contains an error message.
+// If the response is nil, it returns the original error. Otherwise, it wraps or creates an error as needed.
+// It also maps error strings to their corresponding api.Err* constants.
+func checkErr[T any](resp T, err error) (T, error) {
+ if any(resp) == nil {
+ return resp, mapAPIError(err)
+ }
+ respErr, ok := any(resp).(errorResponse)
+ if ok && respErr.GetError() != "" {
+ respErrMsg := respErr.GetError()
+ respErrErr := errors.New(respErrMsg)
+ mappedErr := mapAPIError(respErrErr)
+ // Check if the error was mapped to an API error (different from the temp error)
+ if errors.Is(mappedErr, api.ErrNotImplemented) || errors.Is(mappedErr, api.ErrNotFound) {
+ // Return the mapped API error instead of wrapping
+ return resp, mappedErr
+ }
+ // For non-API errors, use wrap the original error if it is not nil
+ return resp, errors.Join(respErrErr, err)
+ }
+ return resp, mapAPIError(err)
+}
+
+// mapAPIError maps error strings to their corresponding api.Err* constants.
+// This is needed as errors from plugins may not be of type api.Error, due to serialization/deserialization.
+func mapAPIError(err error) error {
+ if err == nil {
+ return nil
+ }
+
+ errStr := err.Error()
+ switch errStr {
+ case api.ErrNotImplemented.Error():
+ return api.ErrNotImplemented
+ case api.ErrNotFound.Error():
+ return api.ErrNotFound
+ default:
+ return err
+ }
+}
diff --git a/plugins/base_capability_test.go b/plugins/base_capability_test.go
new file mode 100644
index 000000000..3bece8dcd
--- /dev/null
+++ b/plugins/base_capability_test.go
@@ -0,0 +1,285 @@
+package plugins
+
+import (
+ "context"
+ "errors"
+
+ "github.com/navidrome/navidrome/plugins/api"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+type nilInstance struct{}
+
+var _ = Describe("baseCapability", func() {
+ var ctx = context.Background()
+
+ It("should load instance using loadFunc", func() {
+ called := false
+ plugin := &baseCapability[*nilInstance, any]{
+ wasmPath: "",
+ id: "test",
+ capability: "test",
+ loadFunc: func(ctx context.Context, _ any, path string) (*nilInstance, error) {
+ called = true
+ return &nilInstance{}, nil
+ },
+ }
+ inst, done, err := plugin.getInstance(ctx, "test")
+ defer done()
+ Expect(err).To(BeNil())
+ Expect(inst).ToNot(BeNil())
+ Expect(called).To(BeTrue())
+ })
+})
+
+var _ = Describe("checkErr", func() {
+ Context("when resp is nil", func() {
+ It("should return nil error when both resp and err are nil", func() {
+ var resp *testErrorResponse
+
+ result, err := checkErr(resp, nil)
+
+ Expect(result).To(BeNil())
+ Expect(err).To(BeNil())
+ })
+
+ It("should return original error unchanged for non-API errors", func() {
+ var resp *testErrorResponse
+ originalErr := errors.New("original error")
+
+ result, err := checkErr(resp, originalErr)
+
+ Expect(result).To(BeNil())
+ Expect(err).To(Equal(originalErr))
+ })
+
+ It("should return mapped API error for ErrNotImplemented", func() {
+ var resp *testErrorResponse
+ err := errors.New("plugin:not_implemented")
+
+ result, mappedErr := checkErr(resp, err)
+
+ Expect(result).To(BeNil())
+ Expect(mappedErr).To(Equal(api.ErrNotImplemented))
+ })
+
+ It("should return mapped API error for ErrNotFound", func() {
+ var resp *testErrorResponse
+ err := errors.New("plugin:not_found")
+
+ result, mappedErr := checkErr(resp, err)
+
+ Expect(result).To(BeNil())
+ Expect(mappedErr).To(Equal(api.ErrNotFound))
+ })
+ })
+
+ Context("when resp is a typed nil that implements errorResponse", func() {
+ It("should not panic and return original error", func() {
+ var resp *testErrorResponse // typed nil
+ originalErr := errors.New("original error")
+
+ // This should not panic
+ result, err := checkErr(resp, originalErr)
+
+ Expect(result).To(BeNil())
+ Expect(err).To(Equal(originalErr))
+ })
+
+ It("should handle typed nil with nil error gracefully", func() {
+ var resp *testErrorResponse // typed nil
+
+ // This should not panic
+ result, err := checkErr(resp, nil)
+
+ Expect(result).To(BeNil())
+ Expect(err).To(BeNil())
+ })
+ })
+
+ Context("when resp implements errorResponse with non-empty error", func() {
+ It("should create new error when original error is nil", func() {
+ resp := &testErrorResponse{errorMsg: "plugin error"}
+
+ result, err := checkErr(resp, nil)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(MatchError("plugin error"))
+ })
+
+ It("should wrap original error when both exist", func() {
+ resp := &testErrorResponse{errorMsg: "plugin error"}
+ originalErr := errors.New("original error")
+
+ result, err := checkErr(resp, originalErr)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(HaveOccurred())
+ // Check that both error messages are present in the joined error
+ errStr := err.Error()
+ Expect(errStr).To(ContainSubstring("plugin error"))
+ Expect(errStr).To(ContainSubstring("original error"))
+ })
+
+ It("should return mapped API error for ErrNotImplemented when no original error", func() {
+ resp := &testErrorResponse{errorMsg: "plugin:not_implemented"}
+
+ result, err := checkErr(resp, nil)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(MatchError(api.ErrNotImplemented))
+ })
+
+ It("should return mapped API error for ErrNotFound when no original error", func() {
+ resp := &testErrorResponse{errorMsg: "plugin:not_found"}
+
+ result, err := checkErr(resp, nil)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(MatchError(api.ErrNotFound))
+ })
+
+ It("should return mapped API error for ErrNotImplemented even with original error", func() {
+ resp := &testErrorResponse{errorMsg: "plugin:not_implemented"}
+ originalErr := errors.New("original error")
+
+ result, err := checkErr(resp, originalErr)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(MatchError(api.ErrNotImplemented))
+ })
+
+ It("should return mapped API error for ErrNotFound even with original error", func() {
+ resp := &testErrorResponse{errorMsg: "plugin:not_found"}
+ originalErr := errors.New("original error")
+
+ result, err := checkErr(resp, originalErr)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(MatchError(api.ErrNotFound))
+ })
+ })
+
+ Context("when resp implements errorResponse with empty error", func() {
+ It("should return original error unchanged", func() {
+ resp := &testErrorResponse{errorMsg: ""}
+ originalErr := errors.New("original error")
+
+ result, err := checkErr(resp, originalErr)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(MatchError(originalErr))
+ })
+
+ It("should return nil error when both are empty/nil", func() {
+ resp := &testErrorResponse{errorMsg: ""}
+
+ result, err := checkErr(resp, nil)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(BeNil())
+ })
+
+ It("should map original API error when response error is empty", func() {
+ resp := &testErrorResponse{errorMsg: ""}
+ originalErr := errors.New("plugin:not_implemented")
+
+ result, err := checkErr(resp, originalErr)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(MatchError(api.ErrNotImplemented))
+ })
+ })
+
+ Context("when resp does not implement errorResponse", func() {
+ It("should return original error unchanged", func() {
+ resp := &testNonErrorResponse{data: "some data"}
+ originalErr := errors.New("original error")
+
+ result, err := checkErr(resp, originalErr)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(Equal(originalErr))
+ })
+
+ It("should return nil error when original error is nil", func() {
+ resp := &testNonErrorResponse{data: "some data"}
+
+ result, err := checkErr(resp, nil)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(BeNil())
+ })
+
+ It("should map original API error when response doesn't implement errorResponse", func() {
+ resp := &testNonErrorResponse{data: "some data"}
+ originalErr := errors.New("plugin:not_found")
+
+ result, err := checkErr(resp, originalErr)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(MatchError(api.ErrNotFound))
+ })
+ })
+
+ Context("when resp is a value type (not pointer)", func() {
+ It("should handle value types that implement errorResponse", func() {
+ resp := testValueErrorResponse{errorMsg: "value error"}
+ originalErr := errors.New("original error")
+
+ result, err := checkErr(resp, originalErr)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(HaveOccurred())
+ // Check that both error messages are present in the joined error
+ errStr := err.Error()
+ Expect(errStr).To(ContainSubstring("value error"))
+ Expect(errStr).To(ContainSubstring("original error"))
+ })
+
+ It("should handle value types with empty error", func() {
+ resp := testValueErrorResponse{errorMsg: ""}
+ originalErr := errors.New("original error")
+
+ result, err := checkErr(resp, originalErr)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(MatchError(originalErr))
+ })
+
+ It("should handle value types with API error", func() {
+ resp := testValueErrorResponse{errorMsg: "plugin:not_implemented"}
+ originalErr := errors.New("original error")
+
+ result, err := checkErr(resp, originalErr)
+
+ Expect(result).To(Equal(resp))
+ Expect(err).To(MatchError(api.ErrNotImplemented))
+ })
+ })
+})
+
+// Test helper types
+type testErrorResponse struct {
+ errorMsg string
+}
+
+func (t *testErrorResponse) GetError() string {
+ if t == nil {
+ return "" // This is what would typically happen with a typed nil
+ }
+ return t.errorMsg
+}
+
+type testNonErrorResponse struct {
+ data string
+}
+
+type testValueErrorResponse struct {
+ errorMsg string
+}
+
+func (t testValueErrorResponse) GetError() string {
+ return t.errorMsg
+}
diff --git a/plugins/discovery.go b/plugins/discovery.go
new file mode 100644
index 000000000..4125da322
--- /dev/null
+++ b/plugins/discovery.go
@@ -0,0 +1,145 @@
+package plugins
+
+import (
+ "fmt"
+ "os"
+ "path/filepath"
+
+ "github.com/navidrome/navidrome/plugins/schema"
+)
+
+// PluginDiscoveryEntry represents the result of plugin discovery
+type PluginDiscoveryEntry struct {
+ ID string // Plugin ID (directory name)
+ Path string // Resolved plugin directory path
+ WasmPath string // Path to the WASM file
+ Manifest *schema.PluginManifest // Loaded manifest (nil if failed)
+ IsSymlink bool // Whether the plugin is a development symlink
+ Error error // Error encountered during discovery
+}
+
+// DiscoverPlugins scans the plugins directory and returns information about all discoverable plugins
+// This shared function eliminates duplication between ScanPlugins and plugin list commands
+func DiscoverPlugins(pluginsDir string) []PluginDiscoveryEntry {
+ var discoveries []PluginDiscoveryEntry
+
+ entries, err := os.ReadDir(pluginsDir)
+ if err != nil {
+ // Return a single entry with the error
+ return []PluginDiscoveryEntry{{
+ Error: fmt.Errorf("failed to read plugins directory %s: %w", pluginsDir, err),
+ }}
+ }
+
+ for _, entry := range entries {
+ name := entry.Name()
+ pluginPath := filepath.Join(pluginsDir, name)
+
+ // Skip hidden files
+ if name[0] == '.' {
+ continue
+ }
+
+ // Check if it's a directory or symlink
+ info, err := os.Lstat(pluginPath)
+ if err != nil {
+ discoveries = append(discoveries, PluginDiscoveryEntry{
+ ID: name,
+ Error: fmt.Errorf("failed to stat entry %s: %w", pluginPath, err),
+ })
+ continue
+ }
+
+ isSymlink := info.Mode()&os.ModeSymlink != 0
+ isDir := info.IsDir()
+
+ // Skip if not a directory or symlink
+ if !isDir && !isSymlink {
+ continue
+ }
+
+ // Resolve symlinks
+ pluginDir := pluginPath
+ if isSymlink {
+ targetDir, err := os.Readlink(pluginPath)
+ if err != nil {
+ discoveries = append(discoveries, PluginDiscoveryEntry{
+ ID: name,
+ IsSymlink: true,
+ Error: fmt.Errorf("failed to resolve symlink %s: %w", pluginPath, err),
+ })
+ continue
+ }
+
+ // If target is a relative path, make it absolute
+ if !filepath.IsAbs(targetDir) {
+ targetDir = filepath.Join(filepath.Dir(pluginPath), targetDir)
+ }
+
+ // Verify that the target is a directory
+ targetInfo, err := os.Stat(targetDir)
+ if err != nil {
+ discoveries = append(discoveries, PluginDiscoveryEntry{
+ ID: name,
+ IsSymlink: true,
+ Error: fmt.Errorf("failed to stat symlink target %s: %w", targetDir, err),
+ })
+ continue
+ }
+
+ if !targetInfo.IsDir() {
+ discoveries = append(discoveries, PluginDiscoveryEntry{
+ ID: name,
+ IsSymlink: true,
+ Error: fmt.Errorf("symlink target is not a directory: %s", targetDir),
+ })
+ continue
+ }
+
+ pluginDir = targetDir
+ }
+
+ // Check for WASM file
+ wasmPath := filepath.Join(pluginDir, "plugin.wasm")
+ if _, err := os.Stat(wasmPath); err != nil {
+ discoveries = append(discoveries, PluginDiscoveryEntry{
+ ID: name,
+ Path: pluginDir,
+ Error: fmt.Errorf("no plugin.wasm found: %w", err),
+ })
+ continue
+ }
+
+ // Load manifest
+ manifest, err := LoadManifest(pluginDir)
+ if err != nil {
+ discoveries = append(discoveries, PluginDiscoveryEntry{
+ ID: name,
+ Path: pluginDir,
+ Error: fmt.Errorf("failed to load manifest: %w", err),
+ })
+ continue
+ }
+
+ // Check for capabilities
+ if len(manifest.Capabilities) == 0 {
+ discoveries = append(discoveries, PluginDiscoveryEntry{
+ ID: name,
+ Path: pluginDir,
+ Error: fmt.Errorf("no capabilities found in manifest"),
+ })
+ continue
+ }
+
+ // Success!
+ discoveries = append(discoveries, PluginDiscoveryEntry{
+ ID: name,
+ Path: pluginDir,
+ WasmPath: wasmPath,
+ Manifest: manifest,
+ IsSymlink: isSymlink,
+ })
+ }
+
+ return discoveries
+}
diff --git a/plugins/discovery_test.go b/plugins/discovery_test.go
new file mode 100644
index 000000000..a5fd34516
--- /dev/null
+++ b/plugins/discovery_test.go
@@ -0,0 +1,402 @@
+package plugins
+
+import (
+ "os"
+ "path/filepath"
+
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("DiscoverPlugins", func() {
+ var tempPluginsDir string
+
+ // Helper to create a valid plugin for discovery testing
+ createValidPlugin := func(name, manifestName, author, version string, capabilities []string) {
+ pluginDir := filepath.Join(tempPluginsDir, name)
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+
+ // Copy real WASM file from testdata
+ sourceWasmPath := filepath.Join(testDataDir, "fake_artist_agent", "plugin.wasm")
+ targetWasmPath := filepath.Join(pluginDir, "plugin.wasm")
+ sourceWasm, err := os.ReadFile(sourceWasmPath)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(os.WriteFile(targetWasmPath, sourceWasm, 0600)).To(Succeed())
+
+ manifest := `{
+ "name": "` + manifestName + `",
+ "version": "` + version + `",
+ "capabilities": [`
+ for i, cap := range capabilities {
+ if i > 0 {
+ manifest += `, `
+ }
+ manifest += `"` + cap + `"`
+ }
+ manifest += `],
+ "author": "` + author + `",
+ "description": "Test Plugin",
+ "website": "https://test.navidrome.org/` + manifestName + `",
+ "permissions": {}
+ }`
+ Expect(os.WriteFile(filepath.Join(pluginDir, "manifest.json"), []byte(manifest), 0600)).To(Succeed())
+ }
+
+ createManifestOnlyPlugin := func(name string) {
+ pluginDir := filepath.Join(tempPluginsDir, name)
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+
+ manifest := `{
+ "name": "manifest-only",
+ "version": "1.0.0",
+ "capabilities": ["MetadataAgent"],
+ "author": "Test Author",
+ "description": "Test Plugin",
+ "website": "https://test.navidrome.org/manifest-only",
+ "permissions": {}
+ }`
+ Expect(os.WriteFile(filepath.Join(pluginDir, "manifest.json"), []byte(manifest), 0600)).To(Succeed())
+ }
+
+ createWasmOnlyPlugin := func(name string) {
+ pluginDir := filepath.Join(tempPluginsDir, name)
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+
+ // Copy real WASM file from testdata
+ sourceWasmPath := filepath.Join(testDataDir, "fake_artist_agent", "plugin.wasm")
+ targetWasmPath := filepath.Join(pluginDir, "plugin.wasm")
+ sourceWasm, err := os.ReadFile(sourceWasmPath)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(os.WriteFile(targetWasmPath, sourceWasm, 0600)).To(Succeed())
+ }
+
+ createInvalidManifestPlugin := func(name string) {
+ pluginDir := filepath.Join(tempPluginsDir, name)
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+
+ // Copy real WASM file from testdata
+ sourceWasmPath := filepath.Join(testDataDir, "fake_artist_agent", "plugin.wasm")
+ targetWasmPath := filepath.Join(pluginDir, "plugin.wasm")
+ sourceWasm, err := os.ReadFile(sourceWasmPath)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(os.WriteFile(targetWasmPath, sourceWasm, 0600)).To(Succeed())
+
+ invalidManifest := `{ "invalid": "json" }`
+ Expect(os.WriteFile(filepath.Join(pluginDir, "manifest.json"), []byte(invalidManifest), 0600)).To(Succeed())
+ }
+
+ createEmptyCapabilitiesPlugin := func(name string) {
+ pluginDir := filepath.Join(tempPluginsDir, name)
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+
+ // Copy real WASM file from testdata
+ sourceWasmPath := filepath.Join(testDataDir, "fake_artist_agent", "plugin.wasm")
+ targetWasmPath := filepath.Join(pluginDir, "plugin.wasm")
+ sourceWasm, err := os.ReadFile(sourceWasmPath)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(os.WriteFile(targetWasmPath, sourceWasm, 0600)).To(Succeed())
+
+ manifest := `{
+ "name": "empty-capabilities",
+ "version": "1.0.0",
+ "capabilities": [],
+ "author": "Test Author",
+ "description": "Test Plugin",
+ "website": "https://test.navidrome.org/empty-capabilities",
+ "permissions": {}
+ }`
+ Expect(os.WriteFile(filepath.Join(pluginDir, "manifest.json"), []byte(manifest), 0600)).To(Succeed())
+ }
+
+ BeforeEach(func() {
+ tempPluginsDir, _ = os.MkdirTemp("", "navidrome-plugins-discovery-test-*")
+ DeferCleanup(func() {
+ _ = os.RemoveAll(tempPluginsDir)
+ })
+ })
+
+ Context("Valid plugins", func() {
+ It("should discover valid plugins with all required files", func() {
+ createValidPlugin("test-plugin", "Test Plugin", "Test Author", "1.0.0", []string{"MetadataAgent"})
+ createValidPlugin("another-plugin", "Another Plugin", "Another Author", "2.0.0", []string{"Scrobbler"})
+
+ discoveries := DiscoverPlugins(tempPluginsDir)
+
+ Expect(discoveries).To(HaveLen(2))
+
+ // Find each plugin by ID
+ var testPlugin, anotherPlugin *PluginDiscoveryEntry
+ for i := range discoveries {
+ switch discoveries[i].ID {
+ case "test-plugin":
+ testPlugin = &discoveries[i]
+ case "another-plugin":
+ anotherPlugin = &discoveries[i]
+ }
+ }
+
+ Expect(testPlugin).NotTo(BeNil())
+ Expect(testPlugin.Error).To(BeNil())
+ Expect(testPlugin.Manifest.Name).To(Equal("Test Plugin"))
+ Expect(string(testPlugin.Manifest.Capabilities[0])).To(Equal("MetadataAgent"))
+
+ Expect(anotherPlugin).NotTo(BeNil())
+ Expect(anotherPlugin.Error).To(BeNil())
+ Expect(anotherPlugin.Manifest.Name).To(Equal("Another Plugin"))
+ Expect(string(anotherPlugin.Manifest.Capabilities[0])).To(Equal("Scrobbler"))
+ })
+
+ It("should handle plugins with same manifest name in different directories", func() {
+ createValidPlugin("lastfm-official", "lastfm", "Official Author", "1.0.0", []string{"MetadataAgent"})
+ createValidPlugin("lastfm-custom", "lastfm", "Custom Author", "2.0.0", []string{"MetadataAgent"})
+
+ discoveries := DiscoverPlugins(tempPluginsDir)
+
+ Expect(discoveries).To(HaveLen(2))
+
+ // Find each plugin by ID
+ var officialPlugin, customPlugin *PluginDiscoveryEntry
+ for i := range discoveries {
+ switch discoveries[i].ID {
+ case "lastfm-official":
+ officialPlugin = &discoveries[i]
+ case "lastfm-custom":
+ customPlugin = &discoveries[i]
+ }
+ }
+
+ Expect(officialPlugin).NotTo(BeNil())
+ Expect(officialPlugin.Error).To(BeNil())
+ Expect(officialPlugin.Manifest.Name).To(Equal("lastfm"))
+ Expect(officialPlugin.Manifest.Author).To(Equal("Official Author"))
+
+ Expect(customPlugin).NotTo(BeNil())
+ Expect(customPlugin.Error).To(BeNil())
+ Expect(customPlugin.Manifest.Name).To(Equal("lastfm"))
+ Expect(customPlugin.Manifest.Author).To(Equal("Custom Author"))
+ })
+ })
+
+ Context("Missing files", func() {
+ It("should report error for plugins missing WASM files", func() {
+ createManifestOnlyPlugin("manifest-only")
+
+ discoveries := DiscoverPlugins(tempPluginsDir)
+
+ Expect(discoveries).To(HaveLen(1))
+ Expect(discoveries[0].ID).To(Equal("manifest-only"))
+ Expect(discoveries[0].Error).To(HaveOccurred())
+ Expect(discoveries[0].Error.Error()).To(ContainSubstring("no plugin.wasm found"))
+ })
+
+ It("should skip directories missing manifest files", func() {
+ createWasmOnlyPlugin("wasm-only")
+
+ discoveries := DiscoverPlugins(tempPluginsDir)
+
+ Expect(discoveries).To(HaveLen(1))
+ Expect(discoveries[0].ID).To(Equal("wasm-only"))
+ Expect(discoveries[0].Error).To(HaveOccurred())
+ Expect(discoveries[0].Error.Error()).To(ContainSubstring("failed to load manifest"))
+ })
+ })
+
+ Context("Invalid content", func() {
+ It("should report error for invalid manifest JSON", func() {
+ createInvalidManifestPlugin("invalid-manifest")
+
+ discoveries := DiscoverPlugins(tempPluginsDir)
+
+ Expect(discoveries).To(HaveLen(1))
+ Expect(discoveries[0].ID).To(Equal("invalid-manifest"))
+ Expect(discoveries[0].Error).To(HaveOccurred())
+ Expect(discoveries[0].Error.Error()).To(ContainSubstring("failed to load manifest"))
+ })
+
+ It("should report error for plugins with empty capabilities", func() {
+ createEmptyCapabilitiesPlugin("empty-capabilities")
+
+ discoveries := DiscoverPlugins(tempPluginsDir)
+
+ Expect(discoveries).To(HaveLen(1))
+ Expect(discoveries[0].ID).To(Equal("empty-capabilities"))
+ Expect(discoveries[0].Error).To(HaveOccurred())
+ Expect(discoveries[0].Error.Error()).To(ContainSubstring("field capabilities length: must be >= 1"))
+ })
+ })
+
+ Context("Symlinks", func() {
+ It("should discover symlinked plugins correctly", func() {
+ // Create a real plugin directory outside tempPluginsDir
+ realPluginDir, err := os.MkdirTemp("", "navidrome-real-plugin-*")
+ Expect(err).ToNot(HaveOccurred())
+ DeferCleanup(func() {
+ _ = os.RemoveAll(realPluginDir)
+ })
+
+ // Create plugin files in the real directory
+ sourceWasmPath := filepath.Join(testDataDir, "fake_artist_agent", "plugin.wasm")
+ targetWasmPath := filepath.Join(realPluginDir, "plugin.wasm")
+ sourceWasm, err := os.ReadFile(sourceWasmPath)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(os.WriteFile(targetWasmPath, sourceWasm, 0600)).To(Succeed())
+
+ manifest := `{
+ "name": "symlinked-plugin",
+ "version": "1.0.0",
+ "capabilities": ["MetadataAgent"],
+ "author": "Test Author",
+ "description": "Test Plugin",
+ "website": "https://test.navidrome.org/symlinked-plugin",
+ "permissions": {}
+ }`
+ Expect(os.WriteFile(filepath.Join(realPluginDir, "manifest.json"), []byte(manifest), 0600)).To(Succeed())
+
+ // Create symlink
+ symlinkPath := filepath.Join(tempPluginsDir, "symlinked-plugin")
+ Expect(os.Symlink(realPluginDir, symlinkPath)).To(Succeed())
+
+ discoveries := DiscoverPlugins(tempPluginsDir)
+
+ Expect(discoveries).To(HaveLen(1))
+ Expect(discoveries[0].ID).To(Equal("symlinked-plugin"))
+ Expect(discoveries[0].Error).To(BeNil())
+ Expect(discoveries[0].IsSymlink).To(BeTrue())
+ Expect(discoveries[0].Path).To(Equal(realPluginDir))
+ Expect(discoveries[0].Manifest.Name).To(Equal("symlinked-plugin"))
+ })
+
+ It("should handle relative symlinks", func() {
+ // Create a real plugin directory in the same parent as tempPluginsDir
+ parentDir := filepath.Dir(tempPluginsDir)
+ realPluginDir := filepath.Join(parentDir, "real-plugin-dir")
+ Expect(os.MkdirAll(realPluginDir, 0755)).To(Succeed())
+ DeferCleanup(func() {
+ _ = os.RemoveAll(realPluginDir)
+ })
+
+ // Create plugin files in the real directory
+ sourceWasmPath := filepath.Join(testDataDir, "fake_artist_agent", "plugin.wasm")
+ targetWasmPath := filepath.Join(realPluginDir, "plugin.wasm")
+ sourceWasm, err := os.ReadFile(sourceWasmPath)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(os.WriteFile(targetWasmPath, sourceWasm, 0600)).To(Succeed())
+
+ manifest := `{
+ "name": "relative-symlinked-plugin",
+ "version": "1.0.0",
+ "capabilities": ["MetadataAgent"],
+ "author": "Test Author",
+ "description": "Test Plugin",
+ "website": "https://test.navidrome.org/relative-symlinked-plugin",
+ "permissions": {}
+ }`
+ Expect(os.WriteFile(filepath.Join(realPluginDir, "manifest.json"), []byte(manifest), 0600)).To(Succeed())
+
+ // Create relative symlink
+ symlinkPath := filepath.Join(tempPluginsDir, "relative-symlinked-plugin")
+ relativeTarget := "../real-plugin-dir"
+ Expect(os.Symlink(relativeTarget, symlinkPath)).To(Succeed())
+
+ discoveries := DiscoverPlugins(tempPluginsDir)
+
+ Expect(discoveries).To(HaveLen(1))
+ Expect(discoveries[0].ID).To(Equal("relative-symlinked-plugin"))
+ Expect(discoveries[0].Error).To(BeNil())
+ Expect(discoveries[0].IsSymlink).To(BeTrue())
+ Expect(discoveries[0].Path).To(Equal(realPluginDir))
+ Expect(discoveries[0].Manifest.Name).To(Equal("relative-symlinked-plugin"))
+ })
+
+ It("should report error for broken symlinks", func() {
+ symlinkPath := filepath.Join(tempPluginsDir, "broken-symlink")
+ nonExistentTarget := "/non/existent/path"
+ Expect(os.Symlink(nonExistentTarget, symlinkPath)).To(Succeed())
+
+ discoveries := DiscoverPlugins(tempPluginsDir)
+
+ Expect(discoveries).To(HaveLen(1))
+ Expect(discoveries[0].ID).To(Equal("broken-symlink"))
+ Expect(discoveries[0].Error).To(HaveOccurred())
+ Expect(discoveries[0].Error.Error()).To(ContainSubstring("failed to stat symlink target"))
+ Expect(discoveries[0].IsSymlink).To(BeTrue())
+ })
+
+ It("should report error for symlinks pointing to files", func() {
+ // Create a regular file
+ regularFile := filepath.Join(tempPluginsDir, "regular-file.txt")
+ Expect(os.WriteFile(regularFile, []byte("content"), 0600)).To(Succeed())
+
+ // Create symlink pointing to the file
+ symlinkPath := filepath.Join(tempPluginsDir, "symlink-to-file")
+ Expect(os.Symlink(regularFile, symlinkPath)).To(Succeed())
+
+ discoveries := DiscoverPlugins(tempPluginsDir)
+
+ Expect(discoveries).To(HaveLen(1))
+ Expect(discoveries[0].ID).To(Equal("symlink-to-file"))
+ Expect(discoveries[0].Error).To(HaveOccurred())
+ Expect(discoveries[0].Error.Error()).To(ContainSubstring("symlink target is not a directory"))
+ Expect(discoveries[0].IsSymlink).To(BeTrue())
+ })
+ })
+
+ Context("Directory filtering", func() {
+ It("should ignore hidden directories", func() {
+ createValidPlugin(".hidden-plugin", "Hidden Plugin", "Test Author", "1.0.0", []string{"MetadataAgent"})
+ createValidPlugin("visible-plugin", "Visible Plugin", "Test Author", "1.0.0", []string{"MetadataAgent"})
+
+ discoveries := DiscoverPlugins(tempPluginsDir)
+
+ Expect(discoveries).To(HaveLen(1))
+ Expect(discoveries[0].ID).To(Equal("visible-plugin"))
+ })
+
+ It("should ignore regular files", func() {
+ // Create a regular file
+ Expect(os.WriteFile(filepath.Join(tempPluginsDir, "regular-file.txt"), []byte("content"), 0600)).To(Succeed())
+ createValidPlugin("valid-plugin", "Valid Plugin", "Test Author", "1.0.0", []string{"MetadataAgent"})
+
+ discoveries := DiscoverPlugins(tempPluginsDir)
+
+ Expect(discoveries).To(HaveLen(1))
+ Expect(discoveries[0].ID).To(Equal("valid-plugin"))
+ })
+
+ It("should handle mixed valid and invalid plugins", func() {
+ createValidPlugin("valid-plugin", "Valid Plugin", "Test Author", "1.0.0", []string{"MetadataAgent"})
+ createManifestOnlyPlugin("manifest-only")
+ createInvalidManifestPlugin("invalid-manifest")
+ createValidPlugin("another-valid", "Another Valid", "Test Author", "1.0.0", []string{"Scrobbler"})
+
+ discoveries := DiscoverPlugins(tempPluginsDir)
+
+ Expect(discoveries).To(HaveLen(4))
+
+ var validCount int
+ var errorCount int
+ for _, discovery := range discoveries {
+ if discovery.Error == nil {
+ validCount++
+ } else {
+ errorCount++
+ }
+ }
+
+ Expect(validCount).To(Equal(2))
+ Expect(errorCount).To(Equal(2))
+ })
+ })
+
+ Context("Error handling", func() {
+ It("should handle non-existent plugins directory", func() {
+ nonExistentDir := "/non/existent/plugins/dir"
+
+ discoveries := DiscoverPlugins(nonExistentDir)
+
+ Expect(discoveries).To(HaveLen(1))
+ Expect(discoveries[0].Error).To(HaveOccurred())
+ Expect(discoveries[0].Error.Error()).To(ContainSubstring("failed to read plugins directory"))
+ })
+ })
+})
diff --git a/plugins/examples/Makefile b/plugins/examples/Makefile
new file mode 100644
index 000000000..e2acc2ff8
--- /dev/null
+++ b/plugins/examples/Makefile
@@ -0,0 +1,27 @@
+all: wikimedia coverartarchive crypto-ticker discord-rich-presence subsonicapi-demo
+
+wikimedia: wikimedia/plugin.wasm
+coverartarchive: coverartarchive/plugin.wasm
+crypto-ticker: crypto-ticker/plugin.wasm
+discord-rich-presence: discord-rich-presence/plugin.wasm
+subsonicapi-demo: subsonicapi-demo/plugin.wasm
+
+wikimedia/plugin.wasm: wikimedia/plugin.go
+ GOOS=wasip1 GOARCH=wasm go build -buildmode=c-shared -o $@ ./wikimedia
+
+coverartarchive/plugin.wasm: coverartarchive/plugin.go
+ GOOS=wasip1 GOARCH=wasm go build -buildmode=c-shared -o $@ ./coverartarchive
+
+crypto-ticker/plugin.wasm: crypto-ticker/plugin.go
+ GOOS=wasip1 GOARCH=wasm go build -buildmode=c-shared -o $@ ./crypto-ticker
+
+DISCORD_RP_FILES=$(shell find discord-rich-presence -type f -name "*.go")
+discord-rich-presence/plugin.wasm: $(DISCORD_RP_FILES)
+ GOOS=wasip1 GOARCH=wasm go build -buildmode=c-shared -o $@ ./discord-rich-presence/...
+
+subsonicapi-demo/plugin.wasm: subsonicapi-demo/plugin.go
+ GOOS=wasip1 GOARCH=wasm go build -buildmode=c-shared -o $@ ./subsonicapi-demo
+
+clean:
+ rm -f wikimedia/plugin.wasm coverartarchive/plugin.wasm crypto-ticker/plugin.wasm \
+ discord-rich-presence/plugin.wasm subsonicapi-demo/plugin.wasm
\ No newline at end of file
diff --git a/plugins/examples/README.md b/plugins/examples/README.md
new file mode 100644
index 000000000..61d6b2ef9
--- /dev/null
+++ b/plugins/examples/README.md
@@ -0,0 +1,31 @@
+# Plugin Examples
+
+This directory contains example plugins for Navidrome, intended for demonstration and reference purposes. These plugins are not used in automated tests.
+
+## Contents
+
+- `wikimedia/`: Retrieves artist information from Wikidata.
+- `coverartarchive/`: Fetches album cover images from the Cover Art Archive.
+- `crypto-ticker/`: Uses websockets to log real-time cryptocurrency prices.
+- `discord-rich-presence/`: Integrates with Discord Rich Presence to display currently playing tracks on Discord profiles.
+- `subsonicapi-demo/`: Demonstrates interaction with Navidrome's Subsonic API from a plugin.
+
+## Building
+
+To build all example plugins, run:
+
+```
+make
+```
+
+Or to build a specific plugin:
+
+```
+make wikimedia
+make coverartarchive
+make crypto-ticker
+make discord-rich-presence
+make subsonicapi-demo
+```
+
+This will produce the corresponding `plugin.wasm` files in each plugin's directory.
diff --git a/plugins/examples/coverartarchive/README.md b/plugins/examples/coverartarchive/README.md
new file mode 100644
index 000000000..e886f6871
--- /dev/null
+++ b/plugins/examples/coverartarchive/README.md
@@ -0,0 +1,34 @@
+# Cover Art Archive AlbumMetadataService Plugin
+
+This plugin provides album cover images for Navidrome by querying the [Cover Art Archive](https://coverartarchive.org/) API using the MusicBrainz Release Group MBID.
+
+## Features
+
+- Implements only the `GetAlbumImages` method of the AlbumMetadataService plugin interface.
+- Returns front cover images for a given release-group MBID.
+- Returns `not found` if no MBID is provided or no images are found.
+
+## Requirements
+
+- Go 1.24 or newer (with WASI support)
+- The Navidrome repository (with generated plugin API code in `plugins/api`)
+
+## How to Compile
+
+To build the WASM plugin, run the following command from the project root:
+
+```sh
+GOOS=wasip1 GOARCH=wasm go build -buildmode=c-shared -o plugins/testdata/coverartarchive/plugin.wasm ./plugins/testdata/coverartarchive
+```
+
+This will produce `plugin.wasm` in this directory.
+
+## Usage
+
+- The plugin can be loaded by Navidrome for integration and end-to-end tests of the plugin system.
+- It is intended for testing and development purposes only.
+
+## API Reference
+
+- [Cover Art Archive API](https://musicbrainz.org/doc/Cover_Art_Archive/API)
+- This plugin uses the endpoint: `https://coverartarchive.org/release-group/{mbid}`
diff --git a/plugins/examples/coverartarchive/manifest.json b/plugins/examples/coverartarchive/manifest.json
new file mode 100644
index 000000000..4049fc358
--- /dev/null
+++ b/plugins/examples/coverartarchive/manifest.json
@@ -0,0 +1,19 @@
+{
+ "$schema": "https://raw.githubusercontent.com/navidrome/navidrome/refs/heads/master/plugins/schema/manifest.schema.json",
+ "name": "coverartarchive",
+ "author": "Navidrome",
+ "version": "1.0.0",
+ "description": "Album cover art from the Cover Art Archive",
+ "website": "https://coverartarchive.org",
+ "capabilities": ["MetadataAgent"],
+ "permissions": {
+ "http": {
+ "reason": "To fetch album cover art from the Cover Art Archive API",
+ "allowedUrls": {
+ "https://coverartarchive.org": ["GET"],
+ "https://*.archive.org": ["GET"]
+ },
+ "allowLocalNetwork": false
+ }
+ }
+}
diff --git a/plugins/examples/coverartarchive/plugin.go b/plugins/examples/coverartarchive/plugin.go
new file mode 100644
index 000000000..ee612c31c
--- /dev/null
+++ b/plugins/examples/coverartarchive/plugin.go
@@ -0,0 +1,151 @@
+//go:build wasip1
+
+package main
+
+import (
+ "context"
+ "encoding/json"
+ "fmt"
+ "log"
+
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/navidrome/navidrome/plugins/host/http"
+)
+
+type CoverArtArchiveAgent struct{}
+
+var ErrNotFound = api.ErrNotFound
+
+type caaImage struct {
+ Image string `json:"image"`
+ Front bool `json:"front"`
+ Types []string `json:"types"`
+ Thumbnails map[string]string `json:"thumbnails"`
+}
+
+var client = http.NewHttpService()
+
+func (CoverArtArchiveAgent) GetAlbumImages(ctx context.Context, req *api.AlbumImagesRequest) (*api.AlbumImagesResponse, error) {
+ if req.Mbid == "" {
+ return nil, ErrNotFound
+ }
+
+ url := "https://coverartarchive.org/release/" + req.Mbid
+ resp, err := client.Get(ctx, &http.HttpRequest{Url: url, TimeoutMs: 5000})
+ if err != nil || resp.Status != 200 {
+ log.Printf("[CAA] Error getting album images from CoverArtArchive (status: %d): %v", resp.Status, err)
+ return nil, ErrNotFound
+ }
+
+ images, err := extractFrontImages(resp.Body)
+ if err != nil || len(images) == 0 {
+ return nil, ErrNotFound
+ }
+ return &api.AlbumImagesResponse{Images: images}, nil
+}
+
+func extractFrontImages(body []byte) ([]*api.ExternalImage, error) {
+ var data struct {
+ Images []caaImage `json:"images"`
+ }
+ if err := json.Unmarshal(body, &data); err != nil {
+ return nil, err
+ }
+ img := findFrontImage(data.Images)
+ if img == nil {
+ return nil, ErrNotFound
+ }
+ return buildImageList(img), nil
+}
+
+func findFrontImage(images []caaImage) *caaImage {
+ for i, img := range images {
+ if img.Front {
+ return &images[i]
+ }
+ }
+ for i, img := range images {
+ for _, t := range img.Types {
+ if t == "Front" {
+ return &images[i]
+ }
+ }
+ }
+ if len(images) > 0 {
+ return &images[0]
+ }
+ return nil
+}
+
+func buildImageList(img *caaImage) []*api.ExternalImage {
+ var images []*api.ExternalImage
+ // First, try numeric sizes only
+ for sizeStr, url := range img.Thumbnails {
+ if url == "" {
+ continue
+ }
+ size := 0
+ if _, err := fmt.Sscanf(sizeStr, "%d", &size); err == nil {
+ images = append(images, &api.ExternalImage{Url: url, Size: int32(size)})
+ }
+ }
+ // If no numeric sizes, fallback to large/small
+ if len(images) == 0 {
+ for sizeStr, url := range img.Thumbnails {
+ if url == "" {
+ continue
+ }
+ var size int
+ switch sizeStr {
+ case "large":
+ size = 500
+ case "small":
+ size = 250
+ default:
+ continue
+ }
+ images = append(images, &api.ExternalImage{Url: url, Size: int32(size)})
+ }
+ }
+ if len(images) == 0 && img.Image != "" {
+ images = append(images, &api.ExternalImage{Url: img.Image, Size: 0})
+ }
+ return images
+}
+
+func (CoverArtArchiveAgent) GetAlbumInfo(ctx context.Context, req *api.AlbumInfoRequest) (*api.AlbumInfoResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+func (CoverArtArchiveAgent) GetArtistMBID(ctx context.Context, req *api.ArtistMBIDRequest) (*api.ArtistMBIDResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (CoverArtArchiveAgent) GetArtistURL(ctx context.Context, req *api.ArtistURLRequest) (*api.ArtistURLResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (CoverArtArchiveAgent) GetArtistBiography(ctx context.Context, req *api.ArtistBiographyRequest) (*api.ArtistBiographyResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (CoverArtArchiveAgent) GetSimilarArtists(ctx context.Context, req *api.ArtistSimilarRequest) (*api.ArtistSimilarResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (CoverArtArchiveAgent) GetArtistImages(ctx context.Context, req *api.ArtistImageRequest) (*api.ArtistImageResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (CoverArtArchiveAgent) GetArtistTopSongs(ctx context.Context, req *api.ArtistTopSongsRequest) (*api.ArtistTopSongsResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func main() {}
+
+func init() {
+ // Configure logging: No timestamps, no source file/line
+ log.SetFlags(0)
+ log.SetPrefix("[CAA] ")
+
+ api.RegisterMetadataAgent(CoverArtArchiveAgent{})
+}
diff --git a/plugins/examples/crypto-ticker/README.md b/plugins/examples/crypto-ticker/README.md
new file mode 100644
index 000000000..ca6d2c44a
--- /dev/null
+++ b/plugins/examples/crypto-ticker/README.md
@@ -0,0 +1,53 @@
+# Crypto Ticker Plugin
+
+This is a WebSocket-based WASM plugin for Navidrome that displays real-time cryptocurrency prices from Coinbase.
+
+## Features
+
+- Connects to Coinbase WebSocket API to receive real-time ticker updates
+- Configurable to track multiple cryptocurrency pairs
+- Implements WebSocketCallback and LifecycleManagement interfaces
+- Automatically reconnects on connection loss
+- Displays price, best bid, best ask, and 24-hour percentage change
+
+## Configuration
+
+In your `navidrome.toml` file, add:
+
+```toml
+[PluginConfig.crypto-ticker]
+tickers = "BTC,ETH,SOL,MATIC"
+```
+
+- `tickers` is a comma-separated list of cryptocurrency symbols
+- The plugin will append `-USD` to any symbol without a trading pair specified
+
+## How it Works
+
+- The plugin connects to Coinbase's WebSocket API upon initialization
+- It subscribes to ticker updates for the configured cryptocurrencies
+- Incoming ticker data is processed and logged
+- On connection loss, it automatically attempts to reconnect (TODO)
+
+## Building
+
+To build the plugin to WASM:
+
+```
+GOOS=wasip1 GOARCH=wasm go build -buildmode=c-shared -o plugin.wasm plugin.go
+```
+
+## Installation
+
+Copy the resulting `plugin.wasm` and create a `manifest.json` file in your Navidrome plugins folder under a `crypto-ticker` directory.
+
+## Example Output
+
+```
+CRYPTO TICKER: BTC-USD Price: 65432.50 Best Bid: 65431.25 Best Ask: 65433.75 24h Change: 2.75%
+CRYPTO TICKER: ETH-USD Price: 3456.78 Best Bid: 3455.90 Best Ask: 3457.80 24h Change: 1.25%
+```
+
+---
+
+For more details, see the source code in `plugin.go`.
diff --git a/plugins/examples/crypto-ticker/manifest.json b/plugins/examples/crypto-ticker/manifest.json
new file mode 100644
index 000000000..482731684
--- /dev/null
+++ b/plugins/examples/crypto-ticker/manifest.json
@@ -0,0 +1,25 @@
+{
+ "name": "crypto-ticker",
+ "author": "Navidrome Plugin",
+ "version": "1.0.0",
+ "description": "A plugin that tracks crypto currency prices using Coinbase WebSocket API",
+ "website": "https://github.com/navidrome/navidrome/tree/master/plugins/examples/crypto-ticker",
+ "capabilities": [
+ "WebSocketCallback",
+ "LifecycleManagement",
+ "SchedulerCallback"
+ ],
+ "permissions": {
+ "config": {
+ "reason": "To read API configuration and WebSocket endpoint settings"
+ },
+ "scheduler": {
+ "reason": "To schedule periodic reconnection attempts and status updates"
+ },
+ "websocket": {
+ "reason": "To connect to Coinbase WebSocket API for real-time cryptocurrency prices",
+ "allowedUrls": ["wss://ws-feed.exchange.coinbase.com"],
+ "allowLocalNetwork": false
+ }
+ }
+}
diff --git a/plugins/examples/crypto-ticker/plugin.go b/plugins/examples/crypto-ticker/plugin.go
new file mode 100644
index 000000000..3fced6d5c
--- /dev/null
+++ b/plugins/examples/crypto-ticker/plugin.go
@@ -0,0 +1,304 @@
+//go:build wasip1
+
+package main
+
+import (
+ "context"
+ "encoding/json"
+ "fmt"
+ "log"
+ "strings"
+
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/navidrome/navidrome/plugins/host/config"
+ "github.com/navidrome/navidrome/plugins/host/scheduler"
+ "github.com/navidrome/navidrome/plugins/host/websocket"
+)
+
+const (
+ // Coinbase WebSocket API endpoint
+ coinbaseWSEndpoint = "wss://ws-feed.exchange.coinbase.com"
+
+ // Connection ID for our WebSocket connection
+ connectionID = "crypto-ticker-connection"
+
+ // ID for the reconnection schedule
+ reconnectScheduleID = "crypto-ticker-reconnect"
+)
+
+var (
+ // Store ticker symbols from the configuration
+ tickers []string
+)
+
+// WebSocketService instance used to manage WebSocket connections and communication.
+var wsService = websocket.NewWebSocketService()
+
+// ConfigService instance for accessing plugin configuration.
+var configService = config.NewConfigService()
+
+// SchedulerService instance for scheduling tasks.
+var schedService = scheduler.NewSchedulerService()
+
+// CryptoTickerPlugin implements WebSocketCallback, LifecycleManagement, and SchedulerCallback interfaces
+type CryptoTickerPlugin struct{}
+
+// Coinbase subscription message structure
+type CoinbaseSubscription struct {
+ Type string `json:"type"`
+ ProductIDs []string `json:"product_ids"`
+ Channels []string `json:"channels"`
+}
+
+// Coinbase ticker message structure
+type CoinbaseTicker struct {
+ Type string `json:"type"`
+ Sequence int64 `json:"sequence"`
+ ProductID string `json:"product_id"`
+ Price string `json:"price"`
+ Open24h string `json:"open_24h"`
+ Volume24h string `json:"volume_24h"`
+ Low24h string `json:"low_24h"`
+ High24h string `json:"high_24h"`
+ Volume30d string `json:"volume_30d"`
+ BestBid string `json:"best_bid"`
+ BestAsk string `json:"best_ask"`
+ Side string `json:"side"`
+ Time string `json:"time"`
+ TradeID int `json:"trade_id"`
+ LastSize string `json:"last_size"`
+}
+
+// OnInit is called when the plugin is loaded
+func (CryptoTickerPlugin) OnInit(ctx context.Context, req *api.InitRequest) (*api.InitResponse, error) {
+ log.Printf("Crypto Ticker Plugin initializing...")
+
+ // Check if ticker configuration exists
+ tickerConfig, ok := req.Config["tickers"]
+ if !ok {
+ return &api.InitResponse{Error: "Missing 'tickers' configuration"}, nil
+ }
+
+ // Parse ticker symbols
+ tickers := parseTickerSymbols(tickerConfig)
+ log.Printf("Configured tickers: %v", tickers)
+
+ // Connect to WebSocket and subscribe to tickers
+ err := connectAndSubscribe(ctx, tickers)
+ if err != nil {
+ return &api.InitResponse{Error: err.Error()}, nil
+ }
+
+ return &api.InitResponse{}, nil
+}
+
+// Helper function to parse ticker symbols from a comma-separated string
+func parseTickerSymbols(tickerConfig string) []string {
+ tickers := strings.Split(tickerConfig, ",")
+ for i, ticker := range tickers {
+ tickers[i] = strings.TrimSpace(ticker)
+
+ // Add -USD suffix if not present
+ if !strings.Contains(tickers[i], "-") {
+ tickers[i] = tickers[i] + "-USD"
+ }
+ }
+ return tickers
+}
+
+// Helper function to connect to WebSocket and subscribe to tickers
+func connectAndSubscribe(ctx context.Context, tickers []string) error {
+ // Connect to the WebSocket API
+ _, err := wsService.Connect(ctx, &websocket.ConnectRequest{
+ Url: coinbaseWSEndpoint,
+ ConnectionId: connectionID,
+ })
+
+ if err != nil {
+ log.Printf("Failed to connect to Coinbase WebSocket API: %v", err)
+ return fmt.Errorf("WebSocket connection error: %v", err)
+ }
+
+ log.Printf("Connected to Coinbase WebSocket API")
+
+ // Subscribe to ticker channel for the configured symbols
+ subscription := CoinbaseSubscription{
+ Type: "subscribe",
+ ProductIDs: tickers,
+ Channels: []string{"ticker"},
+ }
+
+ subscriptionJSON, err := json.Marshal(subscription)
+ if err != nil {
+ log.Printf("Failed to marshal subscription message: %v", err)
+ return fmt.Errorf("JSON marshal error: %v", err)
+ }
+
+ // Send subscription message
+ _, err = wsService.SendText(ctx, &websocket.SendTextRequest{
+ ConnectionId: connectionID,
+ Message: string(subscriptionJSON),
+ })
+
+ if err != nil {
+ log.Printf("Failed to send subscription message: %v", err)
+ return fmt.Errorf("WebSocket send error: %v", err)
+ }
+
+ log.Printf("Subscription message sent to Coinbase WebSocket API")
+ return nil
+}
+
+// OnTextMessage is called when a text message is received from the WebSocket
+func (CryptoTickerPlugin) OnTextMessage(ctx context.Context, req *api.OnTextMessageRequest) (*api.OnTextMessageResponse, error) {
+ // Only process messages from our connection
+ if req.ConnectionId != connectionID {
+ log.Printf("Received message from unexpected connection: %s", req.ConnectionId)
+ return &api.OnTextMessageResponse{}, nil
+ }
+
+ // Try to parse as a ticker message
+ var ticker CoinbaseTicker
+ err := json.Unmarshal([]byte(req.Message), &ticker)
+ if err != nil {
+ log.Printf("Failed to parse ticker message: %v", err)
+ return &api.OnTextMessageResponse{}, nil
+ }
+
+ // If the message is not a ticker or has an error, just log it
+ if ticker.Type != "ticker" {
+ // This could be subscription confirmation or other messages
+ log.Printf("Received non-ticker message: %s", req.Message)
+ return &api.OnTextMessageResponse{}, nil
+ }
+
+ // Format and print ticker information
+ log.Printf("CRYPTO TICKER: %s Price: %s Best Bid: %s Best Ask: %s 24h Change: %s%%\n",
+ ticker.ProductID,
+ ticker.Price,
+ ticker.BestBid,
+ ticker.BestAsk,
+ calculatePercentChange(ticker.Open24h, ticker.Price),
+ )
+
+ return &api.OnTextMessageResponse{}, nil
+}
+
+// OnBinaryMessage is called when a binary message is received
+func (CryptoTickerPlugin) OnBinaryMessage(ctx context.Context, req *api.OnBinaryMessageRequest) (*api.OnBinaryMessageResponse, error) {
+ // Not expected from Coinbase WebSocket API
+ return &api.OnBinaryMessageResponse{}, nil
+}
+
+// OnError is called when an error occurs on the WebSocket connection
+func (CryptoTickerPlugin) OnError(ctx context.Context, req *api.OnErrorRequest) (*api.OnErrorResponse, error) {
+ log.Printf("WebSocket error: %s", req.Error)
+ return &api.OnErrorResponse{}, nil
+}
+
+// OnClose is called when the WebSocket connection is closed
+func (CryptoTickerPlugin) OnClose(ctx context.Context, req *api.OnCloseRequest) (*api.OnCloseResponse, error) {
+ log.Printf("WebSocket connection closed with code %d: %s", req.Code, req.Reason)
+
+ // Try to reconnect if this is our connection
+ if req.ConnectionId == connectionID {
+ log.Printf("Scheduling reconnection attempts every 2 seconds...")
+
+ // Create a recurring schedule to attempt reconnection every 2 seconds
+ resp, err := schedService.ScheduleRecurring(ctx, &scheduler.ScheduleRecurringRequest{
+ // Run every 2 seconds using cron expression
+ CronExpression: "*/2 * * * * *",
+ ScheduleId: reconnectScheduleID,
+ })
+
+ if err != nil {
+ log.Printf("Failed to schedule reconnection attempts: %v", err)
+ } else {
+ log.Printf("Reconnection schedule created with ID: %s", resp.ScheduleId)
+ }
+ }
+
+ return &api.OnCloseResponse{}, nil
+}
+
+// OnSchedulerCallback is called when a scheduled event triggers
+func (CryptoTickerPlugin) OnSchedulerCallback(ctx context.Context, req *api.SchedulerCallbackRequest) (*api.SchedulerCallbackResponse, error) {
+ // Only handle our reconnection schedule
+ if req.ScheduleId != reconnectScheduleID {
+ log.Printf("Received callback for unknown schedule: %s", req.ScheduleId)
+ return &api.SchedulerCallbackResponse{}, nil
+ }
+
+ log.Printf("Attempting to reconnect to Coinbase WebSocket API...")
+
+ // Get the current ticker configuration
+ configResp, err := configService.GetPluginConfig(ctx, &config.GetPluginConfigRequest{})
+ if err != nil {
+ log.Printf("Failed to get plugin configuration: %v", err)
+ return &api.SchedulerCallbackResponse{Error: fmt.Sprintf("Config error: %v", err)}, nil
+ }
+
+ // Check if ticker configuration exists
+ tickerConfig, ok := configResp.Config["tickers"]
+ if !ok {
+ log.Printf("Missing 'tickers' configuration")
+ return &api.SchedulerCallbackResponse{Error: "Missing 'tickers' configuration"}, nil
+ }
+
+ // Parse ticker symbols
+ tickers := parseTickerSymbols(tickerConfig)
+ log.Printf("Reconnecting with tickers: %v", tickers)
+
+ // Try to connect and subscribe
+ err = connectAndSubscribe(ctx, tickers)
+ if err != nil {
+ log.Printf("Reconnection attempt failed: %v", err)
+ return &api.SchedulerCallbackResponse{Error: err.Error()}, nil
+ }
+
+ // Successfully reconnected, cancel the reconnection schedule
+ _, err = schedService.CancelSchedule(ctx, &scheduler.CancelRequest{
+ ScheduleId: reconnectScheduleID,
+ })
+
+ if err != nil {
+ log.Printf("Failed to cancel reconnection schedule: %v", err)
+ } else {
+ log.Printf("Reconnection schedule canceled after successful reconnection")
+ }
+
+ return &api.SchedulerCallbackResponse{}, nil
+}
+
+// Helper function to calculate percent change
+func calculatePercentChange(open, current string) string {
+ var openFloat, currentFloat float64
+ _, err := fmt.Sscanf(open, "%f", &openFloat)
+ if err != nil {
+ return "N/A"
+ }
+ _, err = fmt.Sscanf(current, "%f", ¤tFloat)
+ if err != nil {
+ return "N/A"
+ }
+
+ if openFloat == 0 {
+ return "N/A"
+ }
+
+ change := ((currentFloat - openFloat) / openFloat) * 100
+ return fmt.Sprintf("%.2f", change)
+}
+
+// Required by Go WASI build
+func main() {}
+
+func init() {
+ // Configure logging: No timestamps, no source file/line, prepend [Crypto]
+ log.SetFlags(0)
+ log.SetPrefix("[Crypto] ")
+
+ api.RegisterWebSocketCallback(CryptoTickerPlugin{})
+ api.RegisterLifecycleManagement(CryptoTickerPlugin{})
+ api.RegisterSchedulerCallback(CryptoTickerPlugin{})
+}
diff --git a/plugins/examples/discord-rich-presence/README.md b/plugins/examples/discord-rich-presence/README.md
new file mode 100644
index 000000000..80b12166f
--- /dev/null
+++ b/plugins/examples/discord-rich-presence/README.md
@@ -0,0 +1,88 @@
+# Discord Rich Presence Plugin
+
+This example plugin integrates Navidrome with Discord Rich Presence. It shows how a plugin can keep a real-time
+connection to an external service while remaining completely stateless. This plugin is based on the
+[Navicord](https://github.com/logixism/navicord) project, which provides a similar functionality.
+
+**NOTE: This plugin is for demonstration purposes only. It relies on the user's Discord token being stored in the
+Navidrome configuration file, which is not secure, and may be against Discord's terms of service.
+Use it at your own risk.**
+
+## Overview
+
+The plugin exposes three capabilities:
+
+- **Scrobbler** – receives `NowPlaying` notifications from Navidrome
+- **WebSocketCallback** – handles Discord gateway messages
+- **SchedulerCallback** – used to clear presence and send periodic heartbeats
+
+It relies on several host services declared in `manifest.json`:
+
+- `http` – queries Discord API endpoints
+- `websocket` – maintains gateway connections
+- `scheduler` – schedules heartbeats and presence cleanup
+- `cache` – stores sequence numbers for heartbeats
+- `config` – retrieves the plugin configuration on each call
+- `artwork` – resolves track artwork URLs
+
+## Architecture
+
+Each call from Navidrome creates a new plugin instance. The `init` function registers the capabilities and obtains the
+scheduler service:
+
+```go
+api.RegisterScrobbler(plugin)
+api.RegisterWebSocketCallback(plugin.rpc)
+plugin.sched = api.RegisterNamedSchedulerCallback("close-activity", plugin)
+plugin.rpc.sched = api.RegisterNamedSchedulerCallback("heartbeat", plugin.rpc)
+```
+
+When `NowPlaying` is invoked the plugin:
+
+1. Loads `clientid` and user tokens from the configuration (because plugins are stateless).
+2. Connects to Discord using `WebSocketService` if no connection exists.
+3. Sends the activity payload with track details and artwork.
+4. Schedules a one‑time callback to clear the presence after the track finishes.
+
+Heartbeat messages are sent by a recurring scheduler job. Sequence numbers received from Discord are stored in
+`CacheService` to remain available across plugin instances.
+
+The `OnSchedulerCallback` method clears the presence and closes the connection when the scheduled time is reached.
+
+```go
+// The plugin is stateless, we need to load the configuration every time
+clientID, users, err := d.getConfig(ctx)
+```
+
+## Configuration
+
+Add the following to `navidrome.toml` and adjust for your tokens:
+
+```toml
+[PluginConfig.discord-rich-presence]
+ClientID = "123456789012345678"
+Users = "alice:token123,bob:token456"
+```
+
+- `clientid` is your Discord application ID
+- `users` is a comma‑separated list of `username:token` pairs used for authorization
+
+## Building
+
+```sh
+GOOS=wasip1 GOARCH=wasm go build -buildmode=c-shared -o plugin.wasm ./discord-rich-presence/...
+```
+
+Place the resulting `plugin.wasm` and `manifest.json` in a `discord-rich-presence` folder under your Navidrome plugins
+directory.
+
+## Stateless Operation
+
+Navidrome plugins are completely stateless – each method call instantiates a new plugin instance and discards it
+afterwards.
+
+To work within this model the plugin stores no in-memory state. Connections are keyed by user name inside the host
+services and any transient data (like Discord sequence numbers) is kept in the cache. Configuration is reloaded on every
+method call.
+
+For more implementation details see `plugin.go` and `rpc.go`.
diff --git a/plugins/examples/discord-rich-presence/manifest.json b/plugins/examples/discord-rich-presence/manifest.json
new file mode 100644
index 000000000..c6fa9c283
--- /dev/null
+++ b/plugins/examples/discord-rich-presence/manifest.json
@@ -0,0 +1,35 @@
+{
+ "$schema": "https://raw.githubusercontent.com/navidrome/navidrome/refs/heads/master/plugins/schema/manifest.schema.json",
+ "name": "discord-rich-presence",
+ "author": "Navidrome Team",
+ "version": "1.0.0",
+ "description": "Discord Rich Presence integration for Navidrome",
+ "website": "https://github.com/navidrome/navidrome/tree/master/plugins/examples/discord-rich-presence",
+ "capabilities": ["Scrobbler", "SchedulerCallback", "WebSocketCallback"],
+ "permissions": {
+ "http": {
+ "reason": "To communicate with Discord API for gateway discovery and image uploads",
+ "allowedUrls": {
+ "https://discord.com/api/*": ["GET", "POST"]
+ },
+ "allowLocalNetwork": false
+ },
+ "websocket": {
+ "reason": "To maintain real-time connection with Discord gateway",
+ "allowedUrls": ["wss://gateway.discord.gg"],
+ "allowLocalNetwork": false
+ },
+ "config": {
+ "reason": "To access plugin configuration (client ID and user tokens)"
+ },
+ "cache": {
+ "reason": "To store connection state and sequence numbers"
+ },
+ "scheduler": {
+ "reason": "To schedule heartbeat messages and activity clearing"
+ },
+ "artwork": {
+ "reason": "To get track artwork URLs for rich presence display"
+ }
+ }
+}
diff --git a/plugins/examples/discord-rich-presence/plugin.go b/plugins/examples/discord-rich-presence/plugin.go
new file mode 100644
index 000000000..c93ccf35d
--- /dev/null
+++ b/plugins/examples/discord-rich-presence/plugin.go
@@ -0,0 +1,186 @@
+package main
+
+import (
+ "context"
+ "fmt"
+ "log"
+ "strings"
+
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/navidrome/navidrome/plugins/host/artwork"
+ "github.com/navidrome/navidrome/plugins/host/cache"
+ "github.com/navidrome/navidrome/plugins/host/config"
+ "github.com/navidrome/navidrome/plugins/host/http"
+ "github.com/navidrome/navidrome/plugins/host/scheduler"
+ "github.com/navidrome/navidrome/plugins/host/websocket"
+ "github.com/navidrome/navidrome/utils/slice"
+)
+
+type DiscordRPPlugin struct {
+ rpc *discordRPC
+ cfg config.ConfigService
+ artwork artwork.ArtworkService
+ sched scheduler.SchedulerService
+}
+
+func (d *DiscordRPPlugin) IsAuthorized(ctx context.Context, req *api.ScrobblerIsAuthorizedRequest) (*api.ScrobblerIsAuthorizedResponse, error) {
+ // Get plugin configuration
+ _, users, err := d.getConfig(ctx)
+ if err != nil {
+ return nil, fmt.Errorf("failed to check user authorization: %w", err)
+ }
+
+ // Check if the user has a Discord token configured
+ _, authorized := users[req.Username]
+ log.Printf("IsAuthorized for user %s: %v", req.Username, authorized)
+ return &api.ScrobblerIsAuthorizedResponse{
+ Authorized: authorized,
+ }, nil
+}
+
+func (d *DiscordRPPlugin) NowPlaying(ctx context.Context, request *api.ScrobblerNowPlayingRequest) (*api.ScrobblerNowPlayingResponse, error) {
+ log.Printf("Setting presence for user %s, track: %s", request.Username, request.Track.Name)
+
+ // The plugin is stateless, we need to load the configuration every time
+ clientID, users, err := d.getConfig(ctx)
+ if err != nil {
+ return nil, fmt.Errorf("failed to get config: %w", err)
+ }
+
+ // Check if the user has a Discord token configured
+ userToken, authorized := users[request.Username]
+ if !authorized {
+ return nil, fmt.Errorf("user '%s' not authorized", request.Username)
+ }
+
+ // Make sure we have a connection
+ if err := d.rpc.connect(ctx, request.Username, userToken); err != nil {
+ return nil, fmt.Errorf("failed to connect to Discord: %w", err)
+ }
+
+ // Cancel any existing completion schedule
+ if resp, _ := d.sched.CancelSchedule(ctx, &scheduler.CancelRequest{ScheduleId: request.Username}); resp.Error != "" {
+ log.Printf("Ignoring failure to cancel schedule: %s", resp.Error)
+ }
+
+ // Send activity update
+ if err := d.rpc.sendActivity(ctx, clientID, request.Username, userToken, activity{
+ Application: clientID,
+ Name: "Navidrome",
+ Type: 2,
+ Details: request.Track.Name,
+ State: d.getArtistList(request.Track),
+ Timestamps: activityTimestamps{
+ Start: (request.Timestamp - int64(request.Track.Position)) * 1000,
+ End: (request.Timestamp - int64(request.Track.Position) + int64(request.Track.Length)) * 1000,
+ },
+ Assets: activityAssets{
+ LargeImage: d.imageURL(ctx, request),
+ LargeText: request.Track.Album,
+ },
+ }); err != nil {
+ return nil, fmt.Errorf("failed to send activity: %w", err)
+ }
+
+ // Schedule a timer to clear the activity after the track completes
+ _, err = d.sched.ScheduleOneTime(ctx, &scheduler.ScheduleOneTimeRequest{
+ ScheduleId: request.Username,
+ DelaySeconds: request.Track.Length - request.Track.Position + 5,
+ })
+ if err != nil {
+ return nil, fmt.Errorf("failed to schedule completion timer: %w", err)
+ }
+
+ return nil, nil
+}
+
+func (d *DiscordRPPlugin) imageURL(ctx context.Context, request *api.ScrobblerNowPlayingRequest) string {
+ imageResp, _ := d.artwork.GetTrackUrl(ctx, &artwork.GetArtworkUrlRequest{Id: request.Track.Id, Size: 300})
+ imageURL := imageResp.Url
+ if strings.HasPrefix(imageURL, "http://localhost") {
+ return ""
+ }
+ return imageURL
+}
+
+func (d *DiscordRPPlugin) getArtistList(track *api.TrackInfo) string {
+ return strings.Join(slice.Map(track.Artists, func(a *api.Artist) string { return a.Name }), " • ")
+}
+
+func (d *DiscordRPPlugin) Scrobble(context.Context, *api.ScrobblerScrobbleRequest) (*api.ScrobblerScrobbleResponse, error) {
+ return nil, nil
+}
+
+func (d *DiscordRPPlugin) getConfig(ctx context.Context) (string, map[string]string, error) {
+ const (
+ clientIDKey = "clientid"
+ usersKey = "users"
+ )
+ confResp, err := d.cfg.GetPluginConfig(ctx, &config.GetPluginConfigRequest{})
+ if err != nil {
+ return "", nil, fmt.Errorf("unable to load config: %w", err)
+ }
+ conf := confResp.GetConfig()
+ if len(conf) < 1 {
+ log.Print("missing configuration")
+ return "", nil, nil
+ }
+ clientID := conf[clientIDKey]
+ if clientID == "" {
+ log.Printf("missing ClientID: %v", conf)
+ return "", nil, nil
+ }
+ cfgUsers := conf[usersKey]
+ if len(cfgUsers) == 0 {
+ log.Print("no users configured")
+ return "", nil, nil
+ }
+ users := map[string]string{}
+ for _, user := range strings.Split(cfgUsers, ",") {
+ tuple := strings.Split(user, ":")
+ if len(tuple) != 2 {
+ return clientID, nil, fmt.Errorf("invalid user config: %s", user)
+ }
+ users[tuple[0]] = tuple[1]
+ }
+ return clientID, users, nil
+}
+
+func (d *DiscordRPPlugin) OnSchedulerCallback(ctx context.Context, req *api.SchedulerCallbackRequest) (*api.SchedulerCallbackResponse, error) {
+ log.Printf("Removing presence for user %s", req.ScheduleId)
+ if err := d.rpc.clearActivity(ctx, req.ScheduleId); err != nil {
+ return nil, fmt.Errorf("failed to clear activity: %w", err)
+ }
+ log.Printf("Disconnecting user %s", req.ScheduleId)
+ if err := d.rpc.disconnect(ctx, req.ScheduleId); err != nil {
+ return nil, fmt.Errorf("failed to disconnect from Discord: %w", err)
+ }
+ return nil, nil
+}
+
+// Creates a new instance of the DiscordRPPlugin, with all host services as dependencies
+var plugin = &DiscordRPPlugin{
+ cfg: config.NewConfigService(),
+ artwork: artwork.NewArtworkService(),
+ rpc: &discordRPC{
+ ws: websocket.NewWebSocketService(),
+ web: http.NewHttpService(),
+ mem: cache.NewCacheService(),
+ },
+}
+
+func init() {
+ // Configure logging: No timestamps, no source file/line, prepend [Discord]
+ log.SetFlags(0)
+ log.SetPrefix("[Discord] ")
+
+ // Register plugin capabilities
+ api.RegisterScrobbler(plugin)
+ api.RegisterWebSocketCallback(plugin.rpc)
+
+ // Register named scheduler callbacks, and get the scheduler service for each
+ plugin.sched = api.RegisterNamedSchedulerCallback("close-activity", plugin)
+ plugin.rpc.sched = api.RegisterNamedSchedulerCallback("heartbeat", plugin.rpc)
+}
+
+func main() {}
diff --git a/plugins/examples/discord-rich-presence/rpc.go b/plugins/examples/discord-rich-presence/rpc.go
new file mode 100644
index 000000000..4fab42f41
--- /dev/null
+++ b/plugins/examples/discord-rich-presence/rpc.go
@@ -0,0 +1,402 @@
+package main
+
+import (
+ "context"
+ "encoding/json"
+ "fmt"
+ "log"
+ "strings"
+ "time"
+
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/navidrome/navidrome/plugins/host/cache"
+ "github.com/navidrome/navidrome/plugins/host/http"
+ "github.com/navidrome/navidrome/plugins/host/scheduler"
+ "github.com/navidrome/navidrome/plugins/host/websocket"
+)
+
+type discordRPC struct {
+ ws websocket.WebSocketService
+ web http.HttpService
+ mem cache.CacheService
+ sched scheduler.SchedulerService
+}
+
+// Discord WebSocket Gateway constants
+const (
+ heartbeatOpCode = 1 // Heartbeat operation code
+ gateOpCode = 2 // Identify operation code
+ presenceOpCode = 3 // Presence update operation code
+)
+
+const (
+ heartbeatInterval = 41 // Heartbeat interval in seconds
+ defaultImage = "https://i.imgur.com/hb3XPzA.png"
+)
+
+// Activity is a struct that represents an activity in Discord.
+type activity struct {
+ Name string `json:"name"`
+ Type int `json:"type"`
+ Details string `json:"details"`
+ State string `json:"state"`
+ Application string `json:"application_id"`
+ Timestamps activityTimestamps `json:"timestamps"`
+ Assets activityAssets `json:"assets"`
+}
+
+type activityTimestamps struct {
+ Start int64 `json:"start"`
+ End int64 `json:"end"`
+}
+
+type activityAssets struct {
+ LargeImage string `json:"large_image"`
+ LargeText string `json:"large_text"`
+}
+
+// PresencePayload is a struct that represents a presence update in Discord.
+type presencePayload struct {
+ Activities []activity `json:"activities"`
+ Since int64 `json:"since"`
+ Status string `json:"status"`
+ Afk bool `json:"afk"`
+}
+
+// IdentifyPayload is a struct that represents an identify payload in Discord.
+type identifyPayload struct {
+ Token string `json:"token"`
+ Intents int `json:"intents"`
+ Properties identifyProperties `json:"properties"`
+}
+
+type identifyProperties struct {
+ OS string `json:"os"`
+ Browser string `json:"browser"`
+ Device string `json:"device"`
+}
+
+func (r *discordRPC) processImage(ctx context.Context, imageURL string, clientID string, token string) (string, error) {
+ return r.processImageWithFallback(ctx, imageURL, clientID, token, false)
+}
+
+func (r *discordRPC) processImageWithFallback(ctx context.Context, imageURL string, clientID string, token string, isDefaultImage bool) (string, error) {
+ // Check if context is canceled
+ if err := ctx.Err(); err != nil {
+ return "", fmt.Errorf("context canceled: %w", err)
+ }
+
+ if imageURL == "" {
+ if isDefaultImage {
+ // We're already processing the default image and it's empty, return error
+ return "", fmt.Errorf("default image URL is empty")
+ }
+ return r.processImageWithFallback(ctx, defaultImage, clientID, token, true)
+ }
+
+ if strings.HasPrefix(imageURL, "mp:") {
+ return imageURL, nil
+ }
+
+ // Check cache first
+ cacheKey := fmt.Sprintf("discord.image.%x", imageURL)
+ cacheResp, _ := r.mem.GetString(ctx, &cache.GetRequest{Key: cacheKey})
+ if cacheResp.Exists {
+ log.Printf("Cache hit for image URL: %s", imageURL)
+ return cacheResp.Value, nil
+ }
+
+ resp, _ := r.web.Post(ctx, &http.HttpRequest{
+ Url: fmt.Sprintf("https://discord.com/api/v9/applications/%s/external-assets", clientID),
+ Headers: map[string]string{
+ "Authorization": token,
+ "Content-Type": "application/json",
+ },
+ Body: fmt.Appendf(nil, `{"urls":[%q]}`, imageURL),
+ })
+
+ // Handle HTTP error responses
+ if resp.Status >= 400 {
+ if isDefaultImage {
+ return "", fmt.Errorf("failed to process default image: HTTP %d %s", resp.Status, resp.Error)
+ }
+ return r.processImageWithFallback(ctx, defaultImage, clientID, token, true)
+ }
+ if resp.Error != "" {
+ if isDefaultImage {
+ // If we're already processing the default image and it fails, return error
+ return "", fmt.Errorf("failed to process default image: %s", resp.Error)
+ }
+ // Try with default image
+ return r.processImageWithFallback(ctx, defaultImage, clientID, token, true)
+ }
+
+ var data []map[string]string
+ if err := json.Unmarshal(resp.Body, &data); err != nil {
+ if isDefaultImage {
+ // If we're already processing the default image and it fails, return error
+ return "", fmt.Errorf("failed to unmarshal default image response: %w", err)
+ }
+ // Try with default image
+ return r.processImageWithFallback(ctx, defaultImage, clientID, token, true)
+ }
+
+ if len(data) == 0 {
+ if isDefaultImage {
+ // If we're already processing the default image and it fails, return error
+ return "", fmt.Errorf("no data returned for default image")
+ }
+ // Try with default image
+ return r.processImageWithFallback(ctx, defaultImage, clientID, token, true)
+ }
+
+ image := data[0]["external_asset_path"]
+ if image == "" {
+ if isDefaultImage {
+ // If we're already processing the default image and it fails, return error
+ return "", fmt.Errorf("empty external_asset_path for default image")
+ }
+ // Try with default image
+ return r.processImageWithFallback(ctx, defaultImage, clientID, token, true)
+ }
+
+ processedImage := fmt.Sprintf("mp:%s", image)
+
+ // Cache the processed image URL
+ var ttl = 4 * time.Hour // 4 hours for regular images
+ if isDefaultImage {
+ ttl = 48 * time.Hour // 48 hours for default image
+ }
+
+ _, _ = r.mem.SetString(ctx, &cache.SetStringRequest{
+ Key: cacheKey,
+ Value: processedImage,
+ TtlSeconds: int64(ttl.Seconds()),
+ })
+
+ log.Printf("Cached processed image URL for %s (TTL: %s seconds)", imageURL, ttl)
+
+ return processedImage, nil
+}
+
+func (r *discordRPC) sendActivity(ctx context.Context, clientID, username, token string, data activity) error {
+ log.Printf("Sending activity to for user %s: %#v", username, data)
+
+ processedImage, err := r.processImage(ctx, data.Assets.LargeImage, clientID, token)
+ if err != nil {
+ log.Printf("Failed to process image for user %s, continuing without image: %v", username, err)
+ // Clear the image and continue without it
+ data.Assets.LargeImage = ""
+ } else {
+ log.Printf("Processed image for URL %s: %s", data.Assets.LargeImage, processedImage)
+ data.Assets.LargeImage = processedImage
+ }
+
+ presence := presencePayload{
+ Activities: []activity{data},
+ Status: "dnd",
+ Afk: false,
+ }
+ return r.sendMessage(ctx, username, presenceOpCode, presence)
+}
+
+func (r *discordRPC) clearActivity(ctx context.Context, username string) error {
+ log.Printf("Clearing activity for user %s", username)
+ return r.sendMessage(ctx, username, presenceOpCode, presencePayload{})
+}
+
+func (r *discordRPC) sendMessage(ctx context.Context, username string, opCode int, payload any) error {
+ message := map[string]any{
+ "op": opCode,
+ "d": payload,
+ }
+ b, err := json.Marshal(message)
+ if err != nil {
+ return fmt.Errorf("failed to marshal presence update: %w", err)
+ }
+
+ resp, _ := r.ws.SendText(ctx, &websocket.SendTextRequest{
+ ConnectionId: username,
+ Message: string(b),
+ })
+ if resp.Error != "" {
+ return fmt.Errorf("failed to send presence update: %s", resp.Error)
+ }
+ return nil
+}
+
+func (r *discordRPC) getDiscordGateway(ctx context.Context) (string, error) {
+ resp, _ := r.web.Get(ctx, &http.HttpRequest{
+ Url: "https://discord.com/api/gateway",
+ })
+ if resp.Error != "" {
+ return "", fmt.Errorf("failed to get Discord gateway: %s", resp.Error)
+ }
+ var result map[string]string
+ err := json.Unmarshal(resp.Body, &result)
+ if err != nil {
+ return "", fmt.Errorf("failed to parse Discord gateway response: %w", err)
+ }
+ return result["url"], nil
+}
+
+func (r *discordRPC) sendHeartbeat(ctx context.Context, username string) error {
+ resp, _ := r.mem.GetInt(ctx, &cache.GetRequest{
+ Key: fmt.Sprintf("discord.seq.%s", username),
+ })
+ log.Printf("Sending heartbeat for user %s: %d", username, resp.Value)
+ return r.sendMessage(ctx, username, heartbeatOpCode, resp.Value)
+}
+
+func (r *discordRPC) cleanupFailedConnection(ctx context.Context, username string) {
+ log.Printf("Cleaning up failed connection for user %s", username)
+
+ // Cancel the heartbeat schedule
+ if resp, _ := r.sched.CancelSchedule(ctx, &scheduler.CancelRequest{ScheduleId: username}); resp.Error != "" {
+ log.Printf("Failed to cancel heartbeat schedule for user %s: %s", username, resp.Error)
+ }
+
+ // Close the WebSocket connection
+ if resp, _ := r.ws.Close(ctx, &websocket.CloseRequest{
+ ConnectionId: username,
+ Code: 1000,
+ Reason: "Connection lost",
+ }); resp.Error != "" {
+ log.Printf("Failed to close WebSocket connection for user %s: %s", username, resp.Error)
+ }
+
+ // Clean up cache entries (just the sequence number, no failure tracking needed)
+ _, _ = r.mem.Remove(ctx, &cache.RemoveRequest{Key: fmt.Sprintf("discord.seq.%s", username)})
+
+ log.Printf("Cleaned up connection for user %s", username)
+}
+
+func (r *discordRPC) isConnected(ctx context.Context, username string) bool {
+ // Try to send a heartbeat to test the connection
+ err := r.sendHeartbeat(ctx, username)
+ if err != nil {
+ log.Printf("Heartbeat test failed for user %s: %v", username, err)
+ return false
+ }
+ return true
+}
+
+func (r *discordRPC) connect(ctx context.Context, username string, token string) error {
+ if r.isConnected(ctx, username) {
+ log.Printf("Reusing existing connection for user %s", username)
+ return nil
+ }
+ log.Printf("Creating new connection for user %s", username)
+
+ // Get Discord Gateway URL
+ gateway, err := r.getDiscordGateway(ctx)
+ if err != nil {
+ return fmt.Errorf("failed to get Discord gateway: %w", err)
+ }
+ log.Printf("Using gateway: %s", gateway)
+
+ // Connect to Discord Gateway
+ resp, _ := r.ws.Connect(ctx, &websocket.ConnectRequest{
+ ConnectionId: username,
+ Url: gateway,
+ })
+ if resp.Error != "" {
+ return fmt.Errorf("failed to connect to WebSocket: %s", resp.Error)
+ }
+
+ // Send identify payload
+ payload := identifyPayload{
+ Token: token,
+ Intents: 0,
+ Properties: identifyProperties{
+ OS: "Windows 10",
+ Browser: "Discord Client",
+ Device: "Discord Client",
+ },
+ }
+ err = r.sendMessage(ctx, username, gateOpCode, payload)
+ if err != nil {
+ return fmt.Errorf("failed to send identify payload: %w", err)
+ }
+
+ // Schedule heartbeats for this user/connection
+ cronResp, _ := r.sched.ScheduleRecurring(ctx, &scheduler.ScheduleRecurringRequest{
+ CronExpression: fmt.Sprintf("@every %ds", heartbeatInterval),
+ ScheduleId: username,
+ })
+ log.Printf("Scheduled heartbeat for user %s with ID %s", username, cronResp.ScheduleId)
+
+ log.Printf("Successfully authenticated user %s", username)
+ return nil
+}
+
+func (r *discordRPC) disconnect(ctx context.Context, username string) error {
+ if resp, _ := r.sched.CancelSchedule(ctx, &scheduler.CancelRequest{ScheduleId: username}); resp.Error != "" {
+ return fmt.Errorf("failed to cancel schedule: %s", resp.Error)
+ }
+ resp, _ := r.ws.Close(ctx, &websocket.CloseRequest{
+ ConnectionId: username,
+ Code: 1000,
+ Reason: "Navidrome disconnect",
+ })
+ if resp.Error != "" {
+ return fmt.Errorf("failed to close WebSocket connection: %s", resp.Error)
+ }
+ return nil
+}
+
+func (r *discordRPC) OnTextMessage(ctx context.Context, req *api.OnTextMessageRequest) (*api.OnTextMessageResponse, error) {
+ if len(req.Message) < 1024 {
+ log.Printf("Received WebSocket message for connection '%s': %s", req.ConnectionId, req.Message)
+ } else {
+ log.Printf("Received WebSocket message for connection '%s' (truncated): %s...", req.ConnectionId, req.Message[:1021])
+ }
+
+ // Parse the message. If it's a heartbeat_ack, store the sequence number.
+ message := map[string]any{}
+ err := json.Unmarshal([]byte(req.Message), &message)
+ if err != nil {
+ return nil, fmt.Errorf("failed to parse WebSocket message: %w", err)
+ }
+ if v := message["s"]; v != nil {
+ seq := int64(v.(float64))
+ log.Printf("Received heartbeat_ack for connection '%s': %d", req.ConnectionId, seq)
+ resp, _ := r.mem.SetInt(ctx, &cache.SetIntRequest{
+ Key: fmt.Sprintf("discord.seq.%s", req.ConnectionId),
+ Value: seq,
+ TtlSeconds: heartbeatInterval * 2,
+ })
+ if !resp.Success {
+ return nil, fmt.Errorf("failed to store sequence number for user %s", req.ConnectionId)
+ }
+ }
+ return nil, nil
+}
+
+func (r *discordRPC) OnBinaryMessage(_ context.Context, req *api.OnBinaryMessageRequest) (*api.OnBinaryMessageResponse, error) {
+ log.Printf("Received unexpected binary message for connection '%s'", req.ConnectionId)
+ return nil, nil
+}
+
+func (r *discordRPC) OnError(_ context.Context, req *api.OnErrorRequest) (*api.OnErrorResponse, error) {
+ log.Printf("WebSocket error for connection '%s': %s", req.ConnectionId, req.Error)
+ return nil, nil
+}
+
+func (r *discordRPC) OnClose(_ context.Context, req *api.OnCloseRequest) (*api.OnCloseResponse, error) {
+ log.Printf("WebSocket connection '%s' closed with code %d: %s", req.ConnectionId, req.Code, req.Reason)
+ return nil, nil
+}
+
+func (r *discordRPC) OnSchedulerCallback(ctx context.Context, req *api.SchedulerCallbackRequest) (*api.SchedulerCallbackResponse, error) {
+ err := r.sendHeartbeat(ctx, req.ScheduleId)
+ if err != nil {
+ // On first heartbeat failure, immediately clean up the connection
+ // The next NowPlaying call will reconnect if needed
+ log.Printf("Heartbeat failed for user %s, cleaning up connection: %v", req.ScheduleId, err)
+ r.cleanupFailedConnection(ctx, req.ScheduleId)
+ return nil, fmt.Errorf("heartbeat failed, connection cleaned up: %w", err)
+ }
+
+ return nil, nil
+}
diff --git a/plugins/examples/subsonicapi-demo/README.md b/plugins/examples/subsonicapi-demo/README.md
new file mode 100644
index 000000000..b5ac9f784
--- /dev/null
+++ b/plugins/examples/subsonicapi-demo/README.md
@@ -0,0 +1,88 @@
+# SubsonicAPI Demo Plugin
+
+This example plugin demonstrates how to use the SubsonicAPI host service to access Navidrome's Subsonic API from within a plugin.
+
+## What it does
+
+The plugin performs the following operations during initialization:
+
+1. **Ping the server**: Calls `/rest/ping` to check if the Subsonic API is responding
+2. **Get license info**: Calls `/rest/getLicense` to retrieve server license information
+
+## Key Features
+
+- Shows how to request `subsonicapi` permission in the manifest
+- Demonstrates making Subsonic API calls using the `subsonicapi.Call()` method
+- Handles both successful responses and errors
+- Uses proper lifecycle management with `OnInit`
+
+## Usage
+
+### Manifest Configuration
+
+```json
+{
+ "permissions": {
+ "subsonicapi": {
+ "reason": "Demonstrate accessing Navidrome's Subsonic API from within plugins",
+ "allowAdmins": true
+ }
+ }
+}
+```
+
+### Plugin Implementation
+
+```go
+import "github.com/navidrome/navidrome/plugins/host/subsonicapi"
+
+var subsonicService = subsonicapi.NewSubsonicAPIService()
+
+// OnInit is called when the plugin is loaded
+func (SubsonicAPIDemoPlugin) OnInit(ctx context.Context, req *api.InitRequest) (*api.InitResponse, error) {
+ // Make API calls
+ response, err := subsonicService.Call(ctx, &subsonicapi.CallRequest{
+ Url: "/rest/ping?u=admin",
+ })
+ // Handle response...
+}
+```
+
+When running Navidrome with this plugin installed, it will automatically call the Subsonic API endpoints during the
+server startup, and you can see the results in the logs:
+
+```agsl
+INFO[0000] 2022/01/01 00:00:00 SubsonicAPI Demo Plugin initializing...
+DEBU[0000] API: New request /ping client=subsonicapi-demo username=admin version=1.16.1
+DEBU[0000] API: Successful response endpoint=/ping status=OK
+DEBU[0000] API: New request /getLicense client=subsonicapi-demo username=admin version=1.16.1
+INFO[0000] 2022/01/01 00:00:00 SubsonicAPI ping response: {"subsonic-response":{"status":"ok","version":"1.16.1","type":"navidrome","serverVersion":"dev","openSubsonic":true}}
+DEBU[0000] API: Successful response endpoint=/getLicense status=OK
+DEBU[0000] Plugin initialized successfully elapsed=41.9ms plugin=subsonicapi-demo
+INFO[0000] 2022/01/01 00:00:00 SubsonicAPI license info: {"subsonic-response":{"status":"ok","version":"1.16.1","type":"navidrome","serverVersion":"dev","openSubsonic":true,"license":{"valid":true}}}
+```
+
+## Important Notes
+
+1. **Authentication**: The plugin must provide valid authentication parameters in the URL:
+ - **Required**: `u` (username) - The service validates this parameter is present
+ - Example: `"/rest/ping?u=admin"`
+2. **URL Format**: Only the path and query parameters from the URL are used - host, protocol, and method are ignored
+3. **Automatic Parameters**: The service automatically adds:
+ - `c`: Plugin name (client identifier)
+ - `v`: Subsonic API version (1.16.1)
+ - `f`: Response format (json)
+4. **Internal Authentication**: The service sets up internal authentication using the `u` parameter
+5. **Lifecycle**: This plugin uses `LifecycleManagement` with only the `OnInit` method
+
+## Building
+
+This plugin uses the `wasip1` build constraint and must be compiled for WebAssembly:
+
+```bash
+# Using the project's make target (recommended)
+make plugin-examples
+
+# Manual compilation (when using the proper toolchain)
+GOOS=wasip1 GOARCH=wasm go build -buildmode=c-shared -o plugin.wasm plugin.go
+```
diff --git a/plugins/examples/subsonicapi-demo/manifest.json b/plugins/examples/subsonicapi-demo/manifest.json
new file mode 100644
index 000000000..d26c33181
--- /dev/null
+++ b/plugins/examples/subsonicapi-demo/manifest.json
@@ -0,0 +1,16 @@
+{
+ "$schema": "https://raw.githubusercontent.com/navidrome/navidrome/refs/heads/master/plugins/schema/manifest.schema.json",
+ "name": "subsonicapi-demo",
+ "author": "Navidrome Team",
+ "version": "1.0.0",
+ "description": "Example plugin demonstrating SubsonicAPI host service usage",
+ "website": "https://github.com/navidrome/navidrome",
+ "capabilities": ["LifecycleManagement"],
+ "permissions": {
+ "subsonicapi": {
+ "reason": "Demonstrate accessing Navidrome's Subsonic API from within plugins",
+ "allowAdmins": true,
+ "allowedUsernames": ["admin"]
+ }
+ }
+}
diff --git a/plugins/examples/subsonicapi-demo/plugin.go b/plugins/examples/subsonicapi-demo/plugin.go
new file mode 100644
index 000000000..4ca087ac7
--- /dev/null
+++ b/plugins/examples/subsonicapi-demo/plugin.go
@@ -0,0 +1,68 @@
+//go:build wasip1
+
+package main
+
+import (
+ "context"
+ "log"
+
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/navidrome/navidrome/plugins/host/subsonicapi"
+)
+
+// SubsonicAPIService instance for making API calls
+var subsonicService = subsonicapi.NewSubsonicAPIService()
+
+// SubsonicAPIDemoPlugin implements LifecycleManagement interface
+type SubsonicAPIDemoPlugin struct{}
+
+// OnInit is called when the plugin is loaded
+func (SubsonicAPIDemoPlugin) OnInit(ctx context.Context, req *api.InitRequest) (*api.InitResponse, error) {
+ log.Printf("SubsonicAPI Demo Plugin initializing...")
+
+ // Example: Call the ping endpoint to check if the server is alive
+ response, err := subsonicService.Call(ctx, &subsonicapi.CallRequest{
+ Url: "/rest/ping?u=admin",
+ })
+
+ if err != nil {
+ log.Printf("SubsonicAPI call failed: %v", err)
+ return &api.InitResponse{Error: err.Error()}, nil
+ }
+
+ if response.Error != "" {
+ log.Printf("SubsonicAPI returned error: %s", response.Error)
+ return &api.InitResponse{Error: response.Error}, nil
+ }
+
+ log.Printf("SubsonicAPI ping response: %s", response.Json)
+
+ // Example: Get server info
+ infoResponse, err := subsonicService.Call(ctx, &subsonicapi.CallRequest{
+ Url: "/rest/getLicense?u=admin",
+ })
+
+ if err != nil {
+ log.Printf("SubsonicAPI getLicense call failed: %v", err)
+ return &api.InitResponse{Error: err.Error()}, nil
+ }
+
+ if infoResponse.Error != "" {
+ log.Printf("SubsonicAPI getLicense returned error: %s", infoResponse.Error)
+ return &api.InitResponse{Error: infoResponse.Error}, nil
+ }
+
+ log.Printf("SubsonicAPI license info: %s", infoResponse.Json)
+
+ return &api.InitResponse{}, nil
+}
+
+func main() {}
+
+func init() {
+ // Configure logging: No timestamps, no source file/line
+ log.SetFlags(0)
+ log.SetPrefix("[Subsonic Plugin] ")
+
+ api.RegisterLifecycleManagement(&SubsonicAPIDemoPlugin{})
+}
diff --git a/plugins/examples/wikimedia/README.md b/plugins/examples/wikimedia/README.md
new file mode 100644
index 000000000..15feed2d3
--- /dev/null
+++ b/plugins/examples/wikimedia/README.md
@@ -0,0 +1,32 @@
+# Wikimedia Artist Metadata Plugin
+
+This is a WASM plugin for Navidrome that retrieves artist information from Wikidata/DBpedia using the Wikidata SPARQL endpoint.
+
+## Implemented Methods
+
+- `GetArtistBiography`: Returns the artist's English biography/description from Wikidata.
+- `GetArtistURL`: Returns the artist's official website (if available) from Wikidata.
+- `GetArtistImages`: Returns the artist's main image (Wikimedia Commons) from Wikidata.
+
+All other methods (`GetArtistMBID`, `GetSimilarArtists`, `GetArtistTopSongs`) return a "not implemented" error, as this data is not available from Wikidata/DBpedia.
+
+## How it Works
+
+- The plugin uses the host-provided HTTP service (`HttpService`) to make SPARQL queries to the Wikidata endpoint.
+- No network requests are made directly from the plugin; all HTTP is routed through the host.
+
+## Building
+
+To build the plugin to WASM:
+
+```
+GOOS=wasip1 GOARCH=wasm go build -buildmode=c-shared -o plugin.wasm plugin.go
+```
+
+## Usage
+
+Copy the resulting `plugin.wasm` to your Navidrome plugins folder under a `wikimedia` directory.
+
+---
+
+For more details, see the source code in `plugin.go`.
diff --git a/plugins/examples/wikimedia/manifest.json b/plugins/examples/wikimedia/manifest.json
new file mode 100644
index 000000000..5d0196e0a
--- /dev/null
+++ b/plugins/examples/wikimedia/manifest.json
@@ -0,0 +1,20 @@
+{
+ "$schema": "https://raw.githubusercontent.com/navidrome/navidrome/refs/heads/master/plugins/schema/manifest.schema.json",
+ "name": "wikimedia",
+ "author": "Navidrome",
+ "version": "1.0.0",
+ "description": "Artist information and images from Wikimedia Commons",
+ "website": "https://commons.wikimedia.org",
+ "capabilities": ["MetadataAgent"],
+ "permissions": {
+ "http": {
+ "reason": "To fetch artist information and images from Wikimedia Commons API",
+ "allowedUrls": {
+ "https://*.wikimedia.org": ["GET"],
+ "https://*.wikipedia.org": ["GET"],
+ "https://commons.wikimedia.org": ["GET"]
+ },
+ "allowLocalNetwork": false
+ }
+ }
+}
diff --git a/plugins/examples/wikimedia/plugin.go b/plugins/examples/wikimedia/plugin.go
new file mode 100644
index 000000000..6b60e69da
--- /dev/null
+++ b/plugins/examples/wikimedia/plugin.go
@@ -0,0 +1,391 @@
+//go:build wasip1
+
+package main
+
+import (
+ "context"
+ "encoding/json"
+ "errors"
+ "fmt"
+ "log"
+ "net/url"
+ "strings"
+
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/navidrome/navidrome/plugins/host/http"
+)
+
+const (
+ wikidataEndpoint = "https://query.wikidata.org/sparql"
+ dbpediaEndpoint = "https://dbpedia.org/sparql"
+ mediawikiAPIEndpoint = "https://en.wikipedia.org/w/api.php"
+ requestTimeoutMs = 5000
+)
+
+var (
+ ErrNotFound = api.ErrNotFound
+ ErrNotImplemented = api.ErrNotImplemented
+
+ client = http.NewHttpService()
+)
+
+// SPARQLResult struct for all possible fields
+// Only the needed field will be non-nil in each context
+// (Sitelink, Wiki, Comment, Img)
+type SPARQLResult struct {
+ Results struct {
+ Bindings []struct {
+ Sitelink *struct{ Value string } `json:"sitelink,omitempty"`
+ Wiki *struct{ Value string } `json:"wiki,omitempty"`
+ Comment *struct{ Value string } `json:"comment,omitempty"`
+ Img *struct{ Value string } `json:"img,omitempty"`
+ } `json:"bindings"`
+ } `json:"results"`
+}
+
+// MediaWikiExtractResult is used to unmarshal MediaWiki API extract responses
+// (for getWikipediaExtract)
+type MediaWikiExtractResult struct {
+ Query struct {
+ Pages map[string]struct {
+ PageID int `json:"pageid"`
+ Ns int `json:"ns"`
+ Title string `json:"title"`
+ Extract string `json:"extract"`
+ Missing bool `json:"missing"`
+ } `json:"pages"`
+ } `json:"query"`
+}
+
+// --- SPARQL Query Helper ---
+func sparqlQuery(ctx context.Context, client http.HttpService, endpoint, query string) (*SPARQLResult, error) {
+ form := url.Values{}
+ form.Set("query", query)
+
+ req := &http.HttpRequest{
+ Url: endpoint,
+ Headers: map[string]string{
+ "Accept": "application/sparql-results+json",
+ "Content-Type": "application/x-www-form-urlencoded", // Required by SPARQL endpoints
+ "User-Agent": "NavidromeWikimediaPlugin/0.1",
+ },
+ Body: []byte(form.Encode()), // Send encoded form data
+ TimeoutMs: requestTimeoutMs,
+ }
+ log.Printf("[Wikimedia Query] Attempting SPARQL query to %s (query length: %d):\n%s", endpoint, len(query), query)
+ resp, err := client.Post(ctx, req)
+ if err != nil {
+ return nil, fmt.Errorf("SPARQL request error: %w", err)
+ }
+ if resp.Status != 200 {
+ log.Printf("[Wikimedia Query] SPARQL HTTP error %d for query to %s. Body: %s", resp.Status, endpoint, string(resp.Body))
+ return nil, fmt.Errorf("SPARQL HTTP error: status %d", resp.Status)
+ }
+ var result SPARQLResult
+ if err := json.Unmarshal(resp.Body, &result); err != nil {
+ return nil, fmt.Errorf("failed to parse SPARQL response: %w", err)
+ }
+ if len(result.Results.Bindings) == 0 {
+ return nil, ErrNotFound
+ }
+ return &result, nil
+}
+
+// --- MediaWiki API Helper ---
+func mediawikiQuery(ctx context.Context, client http.HttpService, params url.Values) ([]byte, error) {
+ apiURL := fmt.Sprintf("%s?%s", mediawikiAPIEndpoint, params.Encode())
+ req := &http.HttpRequest{
+ Url: apiURL,
+ Headers: map[string]string{
+ "Accept": "application/json",
+ "User-Agent": "NavidromeWikimediaPlugin/0.1",
+ },
+ TimeoutMs: requestTimeoutMs,
+ }
+ resp, err := client.Get(ctx, req)
+ if err != nil {
+ return nil, fmt.Errorf("MediaWiki request error: %w", err)
+ }
+ if resp.Status != 200 {
+ return nil, fmt.Errorf("MediaWiki HTTP error: status %d, body: %s", resp.Status, string(resp.Body))
+ }
+ return resp.Body, nil
+}
+
+// --- Wikidata Fetch Functions ---
+func getWikidataWikipediaURL(ctx context.Context, client http.HttpService, mbid, name string) (string, error) {
+ var q string
+ if mbid != "" {
+ // Using property chain: ?sitelink schema:about ?artist; schema:isPartOf .
+ q = fmt.Sprintf(`SELECT ?sitelink WHERE { ?artist wdt:P434 "%s". ?sitelink schema:about ?artist; schema:isPartOf . } LIMIT 1`, mbid)
+ } else if name != "" {
+ escapedName := strings.ReplaceAll(name, "\"", "\\\"")
+ // Using property chain: ?sitelink schema:about ?artist; schema:isPartOf .
+ q = fmt.Sprintf(`SELECT ?sitelink WHERE { ?artist rdfs:label "%s"@en. ?sitelink schema:about ?artist; schema:isPartOf . } LIMIT 1`, escapedName)
+ } else {
+ return "", errors.New("MBID or Name required for Wikidata URL lookup")
+ }
+
+ result, err := sparqlQuery(ctx, client, wikidataEndpoint, q)
+ if err != nil {
+ return "", fmt.Errorf("Wikidata SPARQL query failed: %w", err)
+ }
+ if result.Results.Bindings[0].Sitelink != nil {
+ return result.Results.Bindings[0].Sitelink.Value, nil
+ }
+ return "", ErrNotFound
+}
+
+// --- DBpedia Fetch Functions ---
+func getDBpediaWikipediaURL(ctx context.Context, client http.HttpService, name string) (string, error) {
+ if name == "" {
+ return "", ErrNotFound
+ }
+ escapedName := strings.ReplaceAll(name, "\"", "\\\"")
+ q := fmt.Sprintf(`SELECT ?wiki WHERE { ?artist foaf:name "%s"@en; foaf:isPrimaryTopicOf ?wiki. FILTER regex(str(?wiki), "^https://en.wikipedia.org/") } LIMIT 1`, escapedName)
+ result, err := sparqlQuery(ctx, client, dbpediaEndpoint, q)
+ if err != nil {
+ return "", fmt.Errorf("DBpedia SPARQL query failed: %w", err)
+ }
+ if result.Results.Bindings[0].Wiki != nil {
+ return result.Results.Bindings[0].Wiki.Value, nil
+ }
+ return "", ErrNotFound
+}
+
+func getDBpediaComment(ctx context.Context, client http.HttpService, name string) (string, error) {
+ if name == "" {
+ return "", ErrNotFound
+ }
+ escapedName := strings.ReplaceAll(name, "\"", "\\\"")
+ q := fmt.Sprintf(`SELECT ?comment WHERE { ?artist foaf:name "%s"@en; rdfs:comment ?comment. FILTER (lang(?comment) = 'en') } LIMIT 1`, escapedName)
+ result, err := sparqlQuery(ctx, client, dbpediaEndpoint, q)
+ if err != nil {
+ return "", fmt.Errorf("DBpedia comment SPARQL query failed: %w", err)
+ }
+ if result.Results.Bindings[0].Comment != nil {
+ return result.Results.Bindings[0].Comment.Value, nil
+ }
+ return "", ErrNotFound
+}
+
+// --- Wikipedia API Fetch Function ---
+func getWikipediaExtract(ctx context.Context, client http.HttpService, pageTitle string) (string, error) {
+ if pageTitle == "" {
+ return "", errors.New("page title required for Wikipedia API lookup")
+ }
+ params := url.Values{}
+ params.Set("action", "query")
+ params.Set("format", "json")
+ params.Set("prop", "extracts")
+ params.Set("exintro", "true") // Intro section only
+ params.Set("explaintext", "true") // Plain text
+ params.Set("titles", pageTitle)
+ params.Set("redirects", "1") // Follow redirects
+
+ body, err := mediawikiQuery(ctx, client, params)
+ if err != nil {
+ return "", fmt.Errorf("MediaWiki query failed: %w", err)
+ }
+
+ var result MediaWikiExtractResult
+ if err := json.Unmarshal(body, &result); err != nil {
+ return "", fmt.Errorf("failed to parse MediaWiki response: %w", err)
+ }
+
+ // Iterate through the pages map (usually only one page)
+ for _, page := range result.Query.Pages {
+ if page.Missing {
+ continue // Skip missing pages
+ }
+ if page.Extract != "" {
+ return strings.TrimSpace(page.Extract), nil
+ }
+ }
+
+ return "", ErrNotFound
+}
+
+// --- Helper to get Wikipedia Page Title from URL ---
+func extractPageTitleFromURL(wikiURL string) (string, error) {
+ parsedURL, err := url.Parse(wikiURL)
+ if err != nil {
+ return "", err
+ }
+ if parsedURL.Host != "en.wikipedia.org" {
+ return "", fmt.Errorf("URL host is not en.wikipedia.org: %s", parsedURL.Host)
+ }
+ pathParts := strings.Split(strings.TrimPrefix(parsedURL.Path, "/"), "/")
+ if len(pathParts) < 2 || pathParts[0] != "wiki" {
+ return "", fmt.Errorf("URL path does not match /wiki/ format: %s", parsedURL.Path)
+ }
+ title := pathParts[1]
+ if title == "" {
+ return "", errors.New("extracted title is empty")
+ }
+ decodedTitle, err := url.PathUnescape(title)
+ if err != nil {
+ return "", fmt.Errorf("failed to decode title '%s': %w", title, err)
+ }
+ return decodedTitle, nil
+}
+
+// --- Agent Implementation ---
+type WikimediaAgent struct{}
+
+// GetArtistURL fetches the Wikipedia URL.
+// Order: Wikidata(MBID/Name) -> DBpedia(Name) -> Search URL
+func (WikimediaAgent) GetArtistURL(ctx context.Context, req *api.ArtistURLRequest) (*api.ArtistURLResponse, error) {
+ var wikiURL string
+ var err error
+
+ // 1. Try Wikidata (MBID first, then name)
+ wikiURL, err = getWikidataWikipediaURL(ctx, client, req.Mbid, req.Name)
+ if err == nil && wikiURL != "" {
+ return &api.ArtistURLResponse{Url: wikiURL}, nil
+ }
+ if err != nil && err != ErrNotFound {
+ log.Printf("[Wikimedia] Error fetching Wikidata URL: %v\n", err)
+ // Don't stop, try DBpedia
+ }
+
+ // 2. Try DBpedia (Name only)
+ if req.Name != "" {
+ wikiURL, err = getDBpediaWikipediaURL(ctx, client, req.Name)
+ if err == nil && wikiURL != "" {
+ return &api.ArtistURLResponse{Url: wikiURL}, nil
+ }
+ if err != nil && err != ErrNotFound {
+ log.Printf("[Wikimedia] Error fetching DBpedia URL: %v\n", err)
+ // Don't stop, generate search URL
+ }
+ }
+
+ // 3. Fallback to search URL
+ if req.Name != "" {
+ searchURL := fmt.Sprintf("https://en.wikipedia.org/w/index.php?search=%s", url.QueryEscape(req.Name))
+ log.Printf("[Wikimedia] URL not found, falling back to search URL: %s\n", searchURL)
+ return &api.ArtistURLResponse{Url: searchURL}, nil
+ }
+
+ log.Printf("[Wikimedia] Could not determine Wikipedia URL for: %s (%s)\n", req.Name, req.Mbid)
+ return nil, ErrNotFound
+}
+
+// GetArtistBiography fetches the long biography.
+// Order: Wikipedia API (via Wikidata/DBpedia URL) -> DBpedia Comment (Name)
+func (WikimediaAgent) GetArtistBiography(ctx context.Context, req *api.ArtistBiographyRequest) (*api.ArtistBiographyResponse, error) {
+ var bio string
+ var err error
+
+ log.Printf("[Wikimedia Bio] Fetching for Name: %s, MBID: %s", req.Name, req.Mbid)
+
+ // 1. Get Wikipedia URL (using the logic from GetArtistURL)
+ wikiURL := ""
+ // Try Wikidata first
+ tempURL, wdErr := getWikidataWikipediaURL(ctx, client, req.Mbid, req.Name)
+ if wdErr == nil && tempURL != "" {
+ log.Printf("[Wikimedia Bio] Found Wikidata URL: %s", tempURL)
+ wikiURL = tempURL
+ } else if req.Name != "" {
+ // Try DBpedia if Wikidata failed or returned not found
+ log.Printf("[Wikimedia Bio] Wikidata URL failed (%v), trying DBpedia URL", wdErr)
+ tempURL, dbErr := getDBpediaWikipediaURL(ctx, client, req.Name)
+ if dbErr == nil && tempURL != "" {
+ log.Printf("[Wikimedia Bio] Found DBpedia URL: %s", tempURL)
+ wikiURL = tempURL
+ } else {
+ log.Printf("[Wikimedia Bio] DBpedia URL failed (%v)", dbErr)
+ }
+ }
+
+ // 2. If Wikipedia URL found, try MediaWiki API
+ if wikiURL != "" {
+ pageTitle, err := extractPageTitleFromURL(wikiURL)
+ if err == nil {
+ log.Printf("[Wikimedia Bio] Extracted page title: %s", pageTitle)
+ bio, err = getWikipediaExtract(ctx, client, pageTitle)
+ if err == nil && bio != "" {
+ log.Printf("[Wikimedia Bio] Found Wikipedia extract.")
+ return &api.ArtistBiographyResponse{Biography: bio}, nil
+ }
+ log.Printf("[Wikimedia Bio] Wikipedia extract failed: %v", err)
+ if err != nil && err != ErrNotFound {
+ log.Printf("[Wikimedia Bio] Error fetching Wikipedia extract for '%s': %v", pageTitle, err)
+ // Don't stop, try DBpedia comment
+ }
+ } else {
+ log.Printf("[Wikimedia Bio] Error extracting page title from URL '%s': %v", wikiURL, err)
+ // Don't stop, try DBpedia comment
+ }
+ }
+
+ // 3. Fallback to DBpedia Comment (Name only)
+ if req.Name != "" {
+ log.Printf("[Wikimedia Bio] Falling back to DBpedia comment for name: %s", req.Name)
+ bio, err = getDBpediaComment(ctx, client, req.Name)
+ if err == nil && bio != "" {
+ log.Printf("[Wikimedia Bio] Found DBpedia comment.")
+ return &api.ArtistBiographyResponse{Biography: bio}, nil
+ }
+ log.Printf("[Wikimedia Bio] DBpedia comment failed: %v", err)
+ if err != nil && err != ErrNotFound {
+ log.Printf("[Wikimedia Bio] Error fetching DBpedia comment for '%s': %v", req.Name, err)
+ }
+ }
+
+ log.Printf("[Wikimedia Bio] Final: Biography not found for: %s (%s)", req.Name, req.Mbid)
+ return nil, ErrNotFound
+}
+
+// GetArtistImages fetches images (Wikidata only for now)
+func (WikimediaAgent) GetArtistImages(ctx context.Context, req *api.ArtistImageRequest) (*api.ArtistImageResponse, error) {
+ var q string
+ if req.Mbid != "" {
+ q = fmt.Sprintf(`SELECT ?img WHERE { ?artist wdt:P434 "%s"; wdt:P18 ?img } LIMIT 1`, req.Mbid)
+ } else if req.Name != "" {
+ escapedName := strings.ReplaceAll(req.Name, "\"", "\\\"")
+ q = fmt.Sprintf(`SELECT ?img WHERE { ?artist rdfs:label "%s"@en; wdt:P18 ?img } LIMIT 1`, escapedName)
+ } else {
+ return nil, errors.New("MBID or Name required for Wikidata Image lookup")
+ }
+
+ result, err := sparqlQuery(ctx, client, wikidataEndpoint, q)
+ if err != nil {
+ log.Printf("[Wikimedia] Image not found for: %s (%s)\n", req.Name, req.Mbid)
+ return nil, ErrNotFound
+ }
+ if result.Results.Bindings[0].Img != nil {
+ return &api.ArtistImageResponse{Images: []*api.ExternalImage{{Url: result.Results.Bindings[0].Img.Value, Size: 0}}}, nil
+ }
+ log.Printf("[Wikimedia] Image not found for: %s (%s)\n", req.Name, req.Mbid)
+ return nil, ErrNotFound
+}
+
+// Not implemented methods
+func (WikimediaAgent) GetArtistMBID(context.Context, *api.ArtistMBIDRequest) (*api.ArtistMBIDResponse, error) {
+ return nil, ErrNotImplemented
+}
+func (WikimediaAgent) GetSimilarArtists(context.Context, *api.ArtistSimilarRequest) (*api.ArtistSimilarResponse, error) {
+ return nil, ErrNotImplemented
+}
+func (WikimediaAgent) GetArtistTopSongs(context.Context, *api.ArtistTopSongsRequest) (*api.ArtistTopSongsResponse, error) {
+ return nil, ErrNotImplemented
+}
+func (WikimediaAgent) GetAlbumInfo(context.Context, *api.AlbumInfoRequest) (*api.AlbumInfoResponse, error) {
+ return nil, ErrNotImplemented
+}
+
+func (WikimediaAgent) GetAlbumImages(context.Context, *api.AlbumImagesRequest) (*api.AlbumImagesResponse, error) {
+ return nil, ErrNotImplemented
+}
+
+func main() {}
+
+func init() {
+ // Configure logging: No timestamps, no source file/line
+ log.SetFlags(0)
+ log.SetPrefix("[Wikimedia] ")
+
+ api.RegisterMetadataAgent(WikimediaAgent{})
+}
diff --git a/plugins/host/artwork/artwork.pb.go b/plugins/host/artwork/artwork.pb.go
new file mode 100644
index 000000000..228eced22
--- /dev/null
+++ b/plugins/host/artwork/artwork.pb.go
@@ -0,0 +1,73 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/artwork/artwork.proto
+
+package artwork
+
+import (
+ context "context"
+ protoreflect "google.golang.org/protobuf/reflect/protoreflect"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+type GetArtworkUrlRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Id string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"`
+ Size int32 `protobuf:"varint,2,opt,name=size,proto3" json:"size,omitempty"` // Optional, 0 means original size
+}
+
+func (x *GetArtworkUrlRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *GetArtworkUrlRequest) GetId() string {
+ if x != nil {
+ return x.Id
+ }
+ return ""
+}
+
+func (x *GetArtworkUrlRequest) GetSize() int32 {
+ if x != nil {
+ return x.Size
+ }
+ return 0
+}
+
+type GetArtworkUrlResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Url string `protobuf:"bytes,1,opt,name=url,proto3" json:"url,omitempty"`
+}
+
+func (x *GetArtworkUrlResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *GetArtworkUrlResponse) GetUrl() string {
+ if x != nil {
+ return x.Url
+ }
+ return ""
+}
+
+// go:plugin type=host version=1
+type ArtworkService interface {
+ GetArtistUrl(context.Context, *GetArtworkUrlRequest) (*GetArtworkUrlResponse, error)
+ GetAlbumUrl(context.Context, *GetArtworkUrlRequest) (*GetArtworkUrlResponse, error)
+ GetTrackUrl(context.Context, *GetArtworkUrlRequest) (*GetArtworkUrlResponse, error)
+}
diff --git a/plugins/host/artwork/artwork.proto b/plugins/host/artwork/artwork.proto
new file mode 100644
index 000000000..cb562e536
--- /dev/null
+++ b/plugins/host/artwork/artwork.proto
@@ -0,0 +1,21 @@
+syntax = "proto3";
+
+package artwork;
+
+option go_package = "github.com/navidrome/navidrome/plugins/host/artwork;artwork";
+
+// go:plugin type=host version=1
+service ArtworkService {
+ rpc GetArtistUrl(GetArtworkUrlRequest) returns (GetArtworkUrlResponse);
+ rpc GetAlbumUrl(GetArtworkUrlRequest) returns (GetArtworkUrlResponse);
+ rpc GetTrackUrl(GetArtworkUrlRequest) returns (GetArtworkUrlResponse);
+}
+
+message GetArtworkUrlRequest {
+ string id = 1;
+ int32 size = 2; // Optional, 0 means original size
+}
+
+message GetArtworkUrlResponse {
+ string url = 1;
+}
\ No newline at end of file
diff --git a/plugins/host/artwork/artwork_host.pb.go b/plugins/host/artwork/artwork_host.pb.go
new file mode 100644
index 000000000..346fe1449
--- /dev/null
+++ b/plugins/host/artwork/artwork_host.pb.go
@@ -0,0 +1,130 @@
+//go:build !wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/artwork/artwork.proto
+
+package artwork
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ wazero "github.com/tetratelabs/wazero"
+ api "github.com/tetratelabs/wazero/api"
+)
+
+const (
+ i32 = api.ValueTypeI32
+ i64 = api.ValueTypeI64
+)
+
+type _artworkService struct {
+ ArtworkService
+}
+
+// Instantiate a Go-defined module named "env" that exports host functions.
+func Instantiate(ctx context.Context, r wazero.Runtime, hostFunctions ArtworkService) error {
+ envBuilder := r.NewHostModuleBuilder("env")
+ h := _artworkService{hostFunctions}
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._GetArtistUrl), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("get_artist_url")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._GetAlbumUrl), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("get_album_url")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._GetTrackUrl), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("get_track_url")
+
+ _, err := envBuilder.Instantiate(ctx)
+ return err
+}
+
+func (h _artworkService) _GetArtistUrl(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(GetArtworkUrlRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.GetArtistUrl(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+func (h _artworkService) _GetAlbumUrl(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(GetArtworkUrlRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.GetAlbumUrl(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+func (h _artworkService) _GetTrackUrl(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(GetArtworkUrlRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.GetTrackUrl(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
diff --git a/plugins/host/artwork/artwork_plugin.pb.go b/plugins/host/artwork/artwork_plugin.pb.go
new file mode 100644
index 000000000..f54aac0b9
--- /dev/null
+++ b/plugins/host/artwork/artwork_plugin.pb.go
@@ -0,0 +1,90 @@
+//go:build wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/artwork/artwork.proto
+
+package artwork
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ _ "unsafe"
+)
+
+type artworkService struct{}
+
+func NewArtworkService() ArtworkService {
+ return artworkService{}
+}
+
+//go:wasmimport env get_artist_url
+func _get_artist_url(ptr uint32, size uint32) uint64
+
+func (h artworkService) GetArtistUrl(ctx context.Context, request *GetArtworkUrlRequest) (*GetArtworkUrlResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _get_artist_url(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(GetArtworkUrlResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env get_album_url
+func _get_album_url(ptr uint32, size uint32) uint64
+
+func (h artworkService) GetAlbumUrl(ctx context.Context, request *GetArtworkUrlRequest) (*GetArtworkUrlResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _get_album_url(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(GetArtworkUrlResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env get_track_url
+func _get_track_url(ptr uint32, size uint32) uint64
+
+func (h artworkService) GetTrackUrl(ctx context.Context, request *GetArtworkUrlRequest) (*GetArtworkUrlResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _get_track_url(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(GetArtworkUrlResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
diff --git a/plugins/host/artwork/artwork_plugin_dev.go b/plugins/host/artwork/artwork_plugin_dev.go
new file mode 100644
index 000000000..0071f5726
--- /dev/null
+++ b/plugins/host/artwork/artwork_plugin_dev.go
@@ -0,0 +1,7 @@
+//go:build !wasip1
+
+package artwork
+
+func NewArtworkService() ArtworkService {
+ panic("not implemented")
+}
diff --git a/plugins/host/artwork/artwork_vtproto.pb.go b/plugins/host/artwork/artwork_vtproto.pb.go
new file mode 100644
index 000000000..6a1c0ba4e
--- /dev/null
+++ b/plugins/host/artwork/artwork_vtproto.pb.go
@@ -0,0 +1,425 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/artwork/artwork.proto
+
+package artwork
+
+import (
+ fmt "fmt"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+ io "io"
+ bits "math/bits"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+func (m *GetArtworkUrlRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *GetArtworkUrlRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *GetArtworkUrlRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.Size != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.Size))
+ i--
+ dAtA[i] = 0x10
+ }
+ if len(m.Id) > 0 {
+ i -= len(m.Id)
+ copy(dAtA[i:], m.Id)
+ i = encodeVarint(dAtA, i, uint64(len(m.Id)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *GetArtworkUrlResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *GetArtworkUrlResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *GetArtworkUrlResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Url) > 0 {
+ i -= len(m.Url)
+ copy(dAtA[i:], m.Url)
+ i = encodeVarint(dAtA, i, uint64(len(m.Url)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func encodeVarint(dAtA []byte, offset int, v uint64) int {
+ offset -= sov(v)
+ base := offset
+ for v >= 1<<7 {
+ dAtA[offset] = uint8(v&0x7f | 0x80)
+ v >>= 7
+ offset++
+ }
+ dAtA[offset] = uint8(v)
+ return base
+}
+func (m *GetArtworkUrlRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Id)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.Size != 0 {
+ n += 1 + sov(uint64(m.Size))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *GetArtworkUrlResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Url)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func sov(x uint64) (n int) {
+ return (bits.Len64(x|1) + 6) / 7
+}
+func soz(x uint64) (n int) {
+ return sov(uint64((x << 1) ^ uint64((int64(x) >> 63))))
+}
+func (m *GetArtworkUrlRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: GetArtworkUrlRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: GetArtworkUrlRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Id", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Id = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Size", wireType)
+ }
+ m.Size = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.Size |= int32(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *GetArtworkUrlResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: GetArtworkUrlResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: GetArtworkUrlResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Url", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Url = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+
+func skip(dAtA []byte) (n int, err error) {
+ l := len(dAtA)
+ iNdEx := 0
+ depth := 0
+ for iNdEx < l {
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= (uint64(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ wireType := int(wire & 0x7)
+ switch wireType {
+ case 0:
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ iNdEx++
+ if dAtA[iNdEx-1] < 0x80 {
+ break
+ }
+ }
+ case 1:
+ iNdEx += 8
+ case 2:
+ var length int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ length |= (int(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if length < 0 {
+ return 0, ErrInvalidLength
+ }
+ iNdEx += length
+ case 3:
+ depth++
+ case 4:
+ if depth == 0 {
+ return 0, ErrUnexpectedEndOfGroup
+ }
+ depth--
+ case 5:
+ iNdEx += 4
+ default:
+ return 0, fmt.Errorf("proto: illegal wireType %d", wireType)
+ }
+ if iNdEx < 0 {
+ return 0, ErrInvalidLength
+ }
+ if depth == 0 {
+ return iNdEx, nil
+ }
+ }
+ return 0, io.ErrUnexpectedEOF
+}
+
+var (
+ ErrInvalidLength = fmt.Errorf("proto: negative length found during unmarshaling")
+ ErrIntOverflow = fmt.Errorf("proto: integer overflow")
+ ErrUnexpectedEndOfGroup = fmt.Errorf("proto: unexpected end of group")
+)
diff --git a/plugins/host/cache/cache.pb.go b/plugins/host/cache/cache.pb.go
new file mode 100644
index 000000000..6113a89b4
--- /dev/null
+++ b/plugins/host/cache/cache.pb.go
@@ -0,0 +1,420 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/cache/cache.proto
+
+package cache
+
+import (
+ context "context"
+ protoreflect "google.golang.org/protobuf/reflect/protoreflect"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+// Request to store a string value
+type SetStringRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Key string `protobuf:"bytes,1,opt,name=key,proto3" json:"key,omitempty"` // Cache key
+ Value string `protobuf:"bytes,2,opt,name=value,proto3" json:"value,omitempty"` // String value to store
+ TtlSeconds int64 `protobuf:"varint,3,opt,name=ttl_seconds,json=ttlSeconds,proto3" json:"ttl_seconds,omitempty"` // TTL in seconds, 0 means use default
+}
+
+func (x *SetStringRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *SetStringRequest) GetKey() string {
+ if x != nil {
+ return x.Key
+ }
+ return ""
+}
+
+func (x *SetStringRequest) GetValue() string {
+ if x != nil {
+ return x.Value
+ }
+ return ""
+}
+
+func (x *SetStringRequest) GetTtlSeconds() int64 {
+ if x != nil {
+ return x.TtlSeconds
+ }
+ return 0
+}
+
+// Request to store an integer value
+type SetIntRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Key string `protobuf:"bytes,1,opt,name=key,proto3" json:"key,omitempty"` // Cache key
+ Value int64 `protobuf:"varint,2,opt,name=value,proto3" json:"value,omitempty"` // Integer value to store
+ TtlSeconds int64 `protobuf:"varint,3,opt,name=ttl_seconds,json=ttlSeconds,proto3" json:"ttl_seconds,omitempty"` // TTL in seconds, 0 means use default
+}
+
+func (x *SetIntRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *SetIntRequest) GetKey() string {
+ if x != nil {
+ return x.Key
+ }
+ return ""
+}
+
+func (x *SetIntRequest) GetValue() int64 {
+ if x != nil {
+ return x.Value
+ }
+ return 0
+}
+
+func (x *SetIntRequest) GetTtlSeconds() int64 {
+ if x != nil {
+ return x.TtlSeconds
+ }
+ return 0
+}
+
+// Request to store a float value
+type SetFloatRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Key string `protobuf:"bytes,1,opt,name=key,proto3" json:"key,omitempty"` // Cache key
+ Value float64 `protobuf:"fixed64,2,opt,name=value,proto3" json:"value,omitempty"` // Float value to store
+ TtlSeconds int64 `protobuf:"varint,3,opt,name=ttl_seconds,json=ttlSeconds,proto3" json:"ttl_seconds,omitempty"` // TTL in seconds, 0 means use default
+}
+
+func (x *SetFloatRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *SetFloatRequest) GetKey() string {
+ if x != nil {
+ return x.Key
+ }
+ return ""
+}
+
+func (x *SetFloatRequest) GetValue() float64 {
+ if x != nil {
+ return x.Value
+ }
+ return 0
+}
+
+func (x *SetFloatRequest) GetTtlSeconds() int64 {
+ if x != nil {
+ return x.TtlSeconds
+ }
+ return 0
+}
+
+// Request to store a byte slice value
+type SetBytesRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Key string `protobuf:"bytes,1,opt,name=key,proto3" json:"key,omitempty"` // Cache key
+ Value []byte `protobuf:"bytes,2,opt,name=value,proto3" json:"value,omitempty"` // Byte slice value to store
+ TtlSeconds int64 `protobuf:"varint,3,opt,name=ttl_seconds,json=ttlSeconds,proto3" json:"ttl_seconds,omitempty"` // TTL in seconds, 0 means use default
+}
+
+func (x *SetBytesRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *SetBytesRequest) GetKey() string {
+ if x != nil {
+ return x.Key
+ }
+ return ""
+}
+
+func (x *SetBytesRequest) GetValue() []byte {
+ if x != nil {
+ return x.Value
+ }
+ return nil
+}
+
+func (x *SetBytesRequest) GetTtlSeconds() int64 {
+ if x != nil {
+ return x.TtlSeconds
+ }
+ return 0
+}
+
+// Response after setting a value
+type SetResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Success bool `protobuf:"varint,1,opt,name=success,proto3" json:"success,omitempty"` // Whether the operation was successful
+}
+
+func (x *SetResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *SetResponse) GetSuccess() bool {
+ if x != nil {
+ return x.Success
+ }
+ return false
+}
+
+// Request to get a value
+type GetRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Key string `protobuf:"bytes,1,opt,name=key,proto3" json:"key,omitempty"` // Cache key
+}
+
+func (x *GetRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *GetRequest) GetKey() string {
+ if x != nil {
+ return x.Key
+ }
+ return ""
+}
+
+// Response containing a string value
+type GetStringResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Exists bool `protobuf:"varint,1,opt,name=exists,proto3" json:"exists,omitempty"` // Whether the key exists
+ Value string `protobuf:"bytes,2,opt,name=value,proto3" json:"value,omitempty"` // The string value (if exists is true)
+}
+
+func (x *GetStringResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *GetStringResponse) GetExists() bool {
+ if x != nil {
+ return x.Exists
+ }
+ return false
+}
+
+func (x *GetStringResponse) GetValue() string {
+ if x != nil {
+ return x.Value
+ }
+ return ""
+}
+
+// Response containing an integer value
+type GetIntResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Exists bool `protobuf:"varint,1,opt,name=exists,proto3" json:"exists,omitempty"` // Whether the key exists
+ Value int64 `protobuf:"varint,2,opt,name=value,proto3" json:"value,omitempty"` // The integer value (if exists is true)
+}
+
+func (x *GetIntResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *GetIntResponse) GetExists() bool {
+ if x != nil {
+ return x.Exists
+ }
+ return false
+}
+
+func (x *GetIntResponse) GetValue() int64 {
+ if x != nil {
+ return x.Value
+ }
+ return 0
+}
+
+// Response containing a float value
+type GetFloatResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Exists bool `protobuf:"varint,1,opt,name=exists,proto3" json:"exists,omitempty"` // Whether the key exists
+ Value float64 `protobuf:"fixed64,2,opt,name=value,proto3" json:"value,omitempty"` // The float value (if exists is true)
+}
+
+func (x *GetFloatResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *GetFloatResponse) GetExists() bool {
+ if x != nil {
+ return x.Exists
+ }
+ return false
+}
+
+func (x *GetFloatResponse) GetValue() float64 {
+ if x != nil {
+ return x.Value
+ }
+ return 0
+}
+
+// Response containing a byte slice value
+type GetBytesResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Exists bool `protobuf:"varint,1,opt,name=exists,proto3" json:"exists,omitempty"` // Whether the key exists
+ Value []byte `protobuf:"bytes,2,opt,name=value,proto3" json:"value,omitempty"` // The byte slice value (if exists is true)
+}
+
+func (x *GetBytesResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *GetBytesResponse) GetExists() bool {
+ if x != nil {
+ return x.Exists
+ }
+ return false
+}
+
+func (x *GetBytesResponse) GetValue() []byte {
+ if x != nil {
+ return x.Value
+ }
+ return nil
+}
+
+// Request to remove a value
+type RemoveRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Key string `protobuf:"bytes,1,opt,name=key,proto3" json:"key,omitempty"` // Cache key
+}
+
+func (x *RemoveRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *RemoveRequest) GetKey() string {
+ if x != nil {
+ return x.Key
+ }
+ return ""
+}
+
+// Response after removing a value
+type RemoveResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Success bool `protobuf:"varint,1,opt,name=success,proto3" json:"success,omitempty"` // Whether the operation was successful
+}
+
+func (x *RemoveResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *RemoveResponse) GetSuccess() bool {
+ if x != nil {
+ return x.Success
+ }
+ return false
+}
+
+// Request to check if a key exists
+type HasRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Key string `protobuf:"bytes,1,opt,name=key,proto3" json:"key,omitempty"` // Cache key
+}
+
+func (x *HasRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *HasRequest) GetKey() string {
+ if x != nil {
+ return x.Key
+ }
+ return ""
+}
+
+// Response indicating if a key exists
+type HasResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Exists bool `protobuf:"varint,1,opt,name=exists,proto3" json:"exists,omitempty"` // Whether the key exists
+}
+
+func (x *HasResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *HasResponse) GetExists() bool {
+ if x != nil {
+ return x.Exists
+ }
+ return false
+}
+
+// go:plugin type=host version=1
+type CacheService interface {
+ // Set a string value in the cache
+ SetString(context.Context, *SetStringRequest) (*SetResponse, error)
+ // Get a string value from the cache
+ GetString(context.Context, *GetRequest) (*GetStringResponse, error)
+ // Set an integer value in the cache
+ SetInt(context.Context, *SetIntRequest) (*SetResponse, error)
+ // Get an integer value from the cache
+ GetInt(context.Context, *GetRequest) (*GetIntResponse, error)
+ // Set a float value in the cache
+ SetFloat(context.Context, *SetFloatRequest) (*SetResponse, error)
+ // Get a float value from the cache
+ GetFloat(context.Context, *GetRequest) (*GetFloatResponse, error)
+ // Set a byte slice value in the cache
+ SetBytes(context.Context, *SetBytesRequest) (*SetResponse, error)
+ // Get a byte slice value from the cache
+ GetBytes(context.Context, *GetRequest) (*GetBytesResponse, error)
+ // Remove a value from the cache
+ Remove(context.Context, *RemoveRequest) (*RemoveResponse, error)
+ // Check if a key exists in the cache
+ Has(context.Context, *HasRequest) (*HasResponse, error)
+}
diff --git a/plugins/host/cache/cache.proto b/plugins/host/cache/cache.proto
new file mode 100644
index 000000000..8081eca3d
--- /dev/null
+++ b/plugins/host/cache/cache.proto
@@ -0,0 +1,120 @@
+syntax = "proto3";
+
+package cache;
+
+option go_package = "github.com/navidrome/navidrome/plugins/host/cache;cache";
+
+// go:plugin type=host version=1
+service CacheService {
+ // Set a string value in the cache
+ rpc SetString(SetStringRequest) returns (SetResponse);
+
+ // Get a string value from the cache
+ rpc GetString(GetRequest) returns (GetStringResponse);
+
+ // Set an integer value in the cache
+ rpc SetInt(SetIntRequest) returns (SetResponse);
+
+ // Get an integer value from the cache
+ rpc GetInt(GetRequest) returns (GetIntResponse);
+
+ // Set a float value in the cache
+ rpc SetFloat(SetFloatRequest) returns (SetResponse);
+
+ // Get a float value from the cache
+ rpc GetFloat(GetRequest) returns (GetFloatResponse);
+
+ // Set a byte slice value in the cache
+ rpc SetBytes(SetBytesRequest) returns (SetResponse);
+
+ // Get a byte slice value from the cache
+ rpc GetBytes(GetRequest) returns (GetBytesResponse);
+
+ // Remove a value from the cache
+ rpc Remove(RemoveRequest) returns (RemoveResponse);
+
+ // Check if a key exists in the cache
+ rpc Has(HasRequest) returns (HasResponse);
+}
+
+// Request to store a string value
+message SetStringRequest {
+ string key = 1; // Cache key
+ string value = 2; // String value to store
+ int64 ttl_seconds = 3; // TTL in seconds, 0 means use default
+}
+
+// Request to store an integer value
+message SetIntRequest {
+ string key = 1; // Cache key
+ int64 value = 2; // Integer value to store
+ int64 ttl_seconds = 3; // TTL in seconds, 0 means use default
+}
+
+// Request to store a float value
+message SetFloatRequest {
+ string key = 1; // Cache key
+ double value = 2; // Float value to store
+ int64 ttl_seconds = 3; // TTL in seconds, 0 means use default
+}
+
+// Request to store a byte slice value
+message SetBytesRequest {
+ string key = 1; // Cache key
+ bytes value = 2; // Byte slice value to store
+ int64 ttl_seconds = 3; // TTL in seconds, 0 means use default
+}
+
+// Response after setting a value
+message SetResponse {
+ bool success = 1; // Whether the operation was successful
+}
+
+// Request to get a value
+message GetRequest {
+ string key = 1; // Cache key
+}
+
+// Response containing a string value
+message GetStringResponse {
+ bool exists = 1; // Whether the key exists
+ string value = 2; // The string value (if exists is true)
+}
+
+// Response containing an integer value
+message GetIntResponse {
+ bool exists = 1; // Whether the key exists
+ int64 value = 2; // The integer value (if exists is true)
+}
+
+// Response containing a float value
+message GetFloatResponse {
+ bool exists = 1; // Whether the key exists
+ double value = 2; // The float value (if exists is true)
+}
+
+// Response containing a byte slice value
+message GetBytesResponse {
+ bool exists = 1; // Whether the key exists
+ bytes value = 2; // The byte slice value (if exists is true)
+}
+
+// Request to remove a value
+message RemoveRequest {
+ string key = 1; // Cache key
+}
+
+// Response after removing a value
+message RemoveResponse {
+ bool success = 1; // Whether the operation was successful
+}
+
+// Request to check if a key exists
+message HasRequest {
+ string key = 1; // Cache key
+}
+
+// Response indicating if a key exists
+message HasResponse {
+ bool exists = 1; // Whether the key exists
+}
\ No newline at end of file
diff --git a/plugins/host/cache/cache_host.pb.go b/plugins/host/cache/cache_host.pb.go
new file mode 100644
index 000000000..479473fa8
--- /dev/null
+++ b/plugins/host/cache/cache_host.pb.go
@@ -0,0 +1,374 @@
+//go:build !wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/cache/cache.proto
+
+package cache
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ wazero "github.com/tetratelabs/wazero"
+ api "github.com/tetratelabs/wazero/api"
+)
+
+const (
+ i32 = api.ValueTypeI32
+ i64 = api.ValueTypeI64
+)
+
+type _cacheService struct {
+ CacheService
+}
+
+// Instantiate a Go-defined module named "env" that exports host functions.
+func Instantiate(ctx context.Context, r wazero.Runtime, hostFunctions CacheService) error {
+ envBuilder := r.NewHostModuleBuilder("env")
+ h := _cacheService{hostFunctions}
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._SetString), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("set_string")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._GetString), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("get_string")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._SetInt), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("set_int")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._GetInt), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("get_int")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._SetFloat), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("set_float")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._GetFloat), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("get_float")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._SetBytes), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("set_bytes")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._GetBytes), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("get_bytes")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._Remove), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("remove")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._Has), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("has")
+
+ _, err := envBuilder.Instantiate(ctx)
+ return err
+}
+
+// Set a string value in the cache
+
+func (h _cacheService) _SetString(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(SetStringRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.SetString(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Get a string value from the cache
+
+func (h _cacheService) _GetString(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(GetRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.GetString(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Set an integer value in the cache
+
+func (h _cacheService) _SetInt(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(SetIntRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.SetInt(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Get an integer value from the cache
+
+func (h _cacheService) _GetInt(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(GetRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.GetInt(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Set a float value in the cache
+
+func (h _cacheService) _SetFloat(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(SetFloatRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.SetFloat(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Get a float value from the cache
+
+func (h _cacheService) _GetFloat(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(GetRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.GetFloat(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Set a byte slice value in the cache
+
+func (h _cacheService) _SetBytes(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(SetBytesRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.SetBytes(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Get a byte slice value from the cache
+
+func (h _cacheService) _GetBytes(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(GetRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.GetBytes(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Remove a value from the cache
+
+func (h _cacheService) _Remove(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(RemoveRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.Remove(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Check if a key exists in the cache
+
+func (h _cacheService) _Has(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(HasRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.Has(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
diff --git a/plugins/host/cache/cache_plugin.pb.go b/plugins/host/cache/cache_plugin.pb.go
new file mode 100644
index 000000000..6e3bdcd44
--- /dev/null
+++ b/plugins/host/cache/cache_plugin.pb.go
@@ -0,0 +1,251 @@
+//go:build wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/cache/cache.proto
+
+package cache
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ _ "unsafe"
+)
+
+type cacheService struct{}
+
+func NewCacheService() CacheService {
+ return cacheService{}
+}
+
+//go:wasmimport env set_string
+func _set_string(ptr uint32, size uint32) uint64
+
+func (h cacheService) SetString(ctx context.Context, request *SetStringRequest) (*SetResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _set_string(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(SetResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env get_string
+func _get_string(ptr uint32, size uint32) uint64
+
+func (h cacheService) GetString(ctx context.Context, request *GetRequest) (*GetStringResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _get_string(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(GetStringResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env set_int
+func _set_int(ptr uint32, size uint32) uint64
+
+func (h cacheService) SetInt(ctx context.Context, request *SetIntRequest) (*SetResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _set_int(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(SetResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env get_int
+func _get_int(ptr uint32, size uint32) uint64
+
+func (h cacheService) GetInt(ctx context.Context, request *GetRequest) (*GetIntResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _get_int(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(GetIntResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env set_float
+func _set_float(ptr uint32, size uint32) uint64
+
+func (h cacheService) SetFloat(ctx context.Context, request *SetFloatRequest) (*SetResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _set_float(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(SetResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env get_float
+func _get_float(ptr uint32, size uint32) uint64
+
+func (h cacheService) GetFloat(ctx context.Context, request *GetRequest) (*GetFloatResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _get_float(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(GetFloatResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env set_bytes
+func _set_bytes(ptr uint32, size uint32) uint64
+
+func (h cacheService) SetBytes(ctx context.Context, request *SetBytesRequest) (*SetResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _set_bytes(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(SetResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env get_bytes
+func _get_bytes(ptr uint32, size uint32) uint64
+
+func (h cacheService) GetBytes(ctx context.Context, request *GetRequest) (*GetBytesResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _get_bytes(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(GetBytesResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env remove
+func _remove(ptr uint32, size uint32) uint64
+
+func (h cacheService) Remove(ctx context.Context, request *RemoveRequest) (*RemoveResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _remove(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(RemoveResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env has
+func _has(ptr uint32, size uint32) uint64
+
+func (h cacheService) Has(ctx context.Context, request *HasRequest) (*HasResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _has(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(HasResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
diff --git a/plugins/host/cache/cache_plugin_dev.go b/plugins/host/cache/cache_plugin_dev.go
new file mode 100644
index 000000000..824dcc71d
--- /dev/null
+++ b/plugins/host/cache/cache_plugin_dev.go
@@ -0,0 +1,7 @@
+//go:build !wasip1
+
+package cache
+
+func NewCacheService() CacheService {
+ panic("not implemented")
+}
diff --git a/plugins/host/cache/cache_vtproto.pb.go b/plugins/host/cache/cache_vtproto.pb.go
new file mode 100644
index 000000000..0ee3d9f22
--- /dev/null
+++ b/plugins/host/cache/cache_vtproto.pb.go
@@ -0,0 +1,2352 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/cache/cache.proto
+
+package cache
+
+import (
+ binary "encoding/binary"
+ fmt "fmt"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+ io "io"
+ math "math"
+ bits "math/bits"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+func (m *SetStringRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *SetStringRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *SetStringRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.TtlSeconds != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.TtlSeconds))
+ i--
+ dAtA[i] = 0x18
+ }
+ if len(m.Value) > 0 {
+ i -= len(m.Value)
+ copy(dAtA[i:], m.Value)
+ i = encodeVarint(dAtA, i, uint64(len(m.Value)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Key) > 0 {
+ i -= len(m.Key)
+ copy(dAtA[i:], m.Key)
+ i = encodeVarint(dAtA, i, uint64(len(m.Key)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *SetIntRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *SetIntRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *SetIntRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.TtlSeconds != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.TtlSeconds))
+ i--
+ dAtA[i] = 0x18
+ }
+ if m.Value != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.Value))
+ i--
+ dAtA[i] = 0x10
+ }
+ if len(m.Key) > 0 {
+ i -= len(m.Key)
+ copy(dAtA[i:], m.Key)
+ i = encodeVarint(dAtA, i, uint64(len(m.Key)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *SetFloatRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *SetFloatRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *SetFloatRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.TtlSeconds != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.TtlSeconds))
+ i--
+ dAtA[i] = 0x18
+ }
+ if m.Value != 0 {
+ i -= 8
+ binary.LittleEndian.PutUint64(dAtA[i:], uint64(math.Float64bits(float64(m.Value))))
+ i--
+ dAtA[i] = 0x11
+ }
+ if len(m.Key) > 0 {
+ i -= len(m.Key)
+ copy(dAtA[i:], m.Key)
+ i = encodeVarint(dAtA, i, uint64(len(m.Key)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *SetBytesRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *SetBytesRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *SetBytesRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.TtlSeconds != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.TtlSeconds))
+ i--
+ dAtA[i] = 0x18
+ }
+ if len(m.Value) > 0 {
+ i -= len(m.Value)
+ copy(dAtA[i:], m.Value)
+ i = encodeVarint(dAtA, i, uint64(len(m.Value)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Key) > 0 {
+ i -= len(m.Key)
+ copy(dAtA[i:], m.Key)
+ i = encodeVarint(dAtA, i, uint64(len(m.Key)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *SetResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *SetResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *SetResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.Success {
+ i--
+ if m.Success {
+ dAtA[i] = 1
+ } else {
+ dAtA[i] = 0
+ }
+ i--
+ dAtA[i] = 0x8
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *GetRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *GetRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *GetRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Key) > 0 {
+ i -= len(m.Key)
+ copy(dAtA[i:], m.Key)
+ i = encodeVarint(dAtA, i, uint64(len(m.Key)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *GetStringResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *GetStringResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *GetStringResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Value) > 0 {
+ i -= len(m.Value)
+ copy(dAtA[i:], m.Value)
+ i = encodeVarint(dAtA, i, uint64(len(m.Value)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if m.Exists {
+ i--
+ if m.Exists {
+ dAtA[i] = 1
+ } else {
+ dAtA[i] = 0
+ }
+ i--
+ dAtA[i] = 0x8
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *GetIntResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *GetIntResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *GetIntResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.Value != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.Value))
+ i--
+ dAtA[i] = 0x10
+ }
+ if m.Exists {
+ i--
+ if m.Exists {
+ dAtA[i] = 1
+ } else {
+ dAtA[i] = 0
+ }
+ i--
+ dAtA[i] = 0x8
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *GetFloatResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *GetFloatResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *GetFloatResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.Value != 0 {
+ i -= 8
+ binary.LittleEndian.PutUint64(dAtA[i:], uint64(math.Float64bits(float64(m.Value))))
+ i--
+ dAtA[i] = 0x11
+ }
+ if m.Exists {
+ i--
+ if m.Exists {
+ dAtA[i] = 1
+ } else {
+ dAtA[i] = 0
+ }
+ i--
+ dAtA[i] = 0x8
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *GetBytesResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *GetBytesResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *GetBytesResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Value) > 0 {
+ i -= len(m.Value)
+ copy(dAtA[i:], m.Value)
+ i = encodeVarint(dAtA, i, uint64(len(m.Value)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if m.Exists {
+ i--
+ if m.Exists {
+ dAtA[i] = 1
+ } else {
+ dAtA[i] = 0
+ }
+ i--
+ dAtA[i] = 0x8
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *RemoveRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *RemoveRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *RemoveRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Key) > 0 {
+ i -= len(m.Key)
+ copy(dAtA[i:], m.Key)
+ i = encodeVarint(dAtA, i, uint64(len(m.Key)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *RemoveResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *RemoveResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *RemoveResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.Success {
+ i--
+ if m.Success {
+ dAtA[i] = 1
+ } else {
+ dAtA[i] = 0
+ }
+ i--
+ dAtA[i] = 0x8
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *HasRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *HasRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *HasRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Key) > 0 {
+ i -= len(m.Key)
+ copy(dAtA[i:], m.Key)
+ i = encodeVarint(dAtA, i, uint64(len(m.Key)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *HasResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *HasResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *HasResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if m.Exists {
+ i--
+ if m.Exists {
+ dAtA[i] = 1
+ } else {
+ dAtA[i] = 0
+ }
+ i--
+ dAtA[i] = 0x8
+ }
+ return len(dAtA) - i, nil
+}
+
+func encodeVarint(dAtA []byte, offset int, v uint64) int {
+ offset -= sov(v)
+ base := offset
+ for v >= 1<<7 {
+ dAtA[offset] = uint8(v&0x7f | 0x80)
+ v >>= 7
+ offset++
+ }
+ dAtA[offset] = uint8(v)
+ return base
+}
+func (m *SetStringRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Key)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Value)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.TtlSeconds != 0 {
+ n += 1 + sov(uint64(m.TtlSeconds))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *SetIntRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Key)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.Value != 0 {
+ n += 1 + sov(uint64(m.Value))
+ }
+ if m.TtlSeconds != 0 {
+ n += 1 + sov(uint64(m.TtlSeconds))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *SetFloatRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Key)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.Value != 0 {
+ n += 9
+ }
+ if m.TtlSeconds != 0 {
+ n += 1 + sov(uint64(m.TtlSeconds))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *SetBytesRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Key)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Value)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.TtlSeconds != 0 {
+ n += 1 + sov(uint64(m.TtlSeconds))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *SetResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if m.Success {
+ n += 2
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *GetRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Key)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *GetStringResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if m.Exists {
+ n += 2
+ }
+ l = len(m.Value)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *GetIntResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if m.Exists {
+ n += 2
+ }
+ if m.Value != 0 {
+ n += 1 + sov(uint64(m.Value))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *GetFloatResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if m.Exists {
+ n += 2
+ }
+ if m.Value != 0 {
+ n += 9
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *GetBytesResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if m.Exists {
+ n += 2
+ }
+ l = len(m.Value)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *RemoveRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Key)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *RemoveResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if m.Success {
+ n += 2
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *HasRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Key)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *HasResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if m.Exists {
+ n += 2
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func sov(x uint64) (n int) {
+ return (bits.Len64(x|1) + 6) / 7
+}
+func soz(x uint64) (n int) {
+ return sov(uint64((x << 1) ^ uint64((int64(x) >> 63))))
+}
+func (m *SetStringRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: SetStringRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: SetStringRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Key", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Key = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Value", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Value = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 3:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field TtlSeconds", wireType)
+ }
+ m.TtlSeconds = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.TtlSeconds |= int64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *SetIntRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: SetIntRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: SetIntRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Key", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Key = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Value", wireType)
+ }
+ m.Value = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.Value |= int64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ case 3:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field TtlSeconds", wireType)
+ }
+ m.TtlSeconds = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.TtlSeconds |= int64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *SetFloatRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: SetFloatRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: SetFloatRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Key", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Key = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 1 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Value", wireType)
+ }
+ var v uint64
+ if (iNdEx + 8) > l {
+ return io.ErrUnexpectedEOF
+ }
+ v = uint64(binary.LittleEndian.Uint64(dAtA[iNdEx:]))
+ iNdEx += 8
+ m.Value = float64(math.Float64frombits(v))
+ case 3:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field TtlSeconds", wireType)
+ }
+ m.TtlSeconds = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.TtlSeconds |= int64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *SetBytesRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: SetBytesRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: SetBytesRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Key", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Key = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Value", wireType)
+ }
+ var byteLen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ byteLen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if byteLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + byteLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Value = append(m.Value[:0], dAtA[iNdEx:postIndex]...)
+ if m.Value == nil {
+ m.Value = []byte{}
+ }
+ iNdEx = postIndex
+ case 3:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field TtlSeconds", wireType)
+ }
+ m.TtlSeconds = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.TtlSeconds |= int64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *SetResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: SetResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: SetResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Success", wireType)
+ }
+ var v int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ v |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ m.Success = bool(v != 0)
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *GetRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: GetRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: GetRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Key", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Key = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *GetStringResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: GetStringResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: GetStringResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Exists", wireType)
+ }
+ var v int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ v |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ m.Exists = bool(v != 0)
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Value", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Value = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *GetIntResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: GetIntResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: GetIntResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Exists", wireType)
+ }
+ var v int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ v |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ m.Exists = bool(v != 0)
+ case 2:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Value", wireType)
+ }
+ m.Value = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.Value |= int64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *GetFloatResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: GetFloatResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: GetFloatResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Exists", wireType)
+ }
+ var v int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ v |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ m.Exists = bool(v != 0)
+ case 2:
+ if wireType != 1 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Value", wireType)
+ }
+ var v uint64
+ if (iNdEx + 8) > l {
+ return io.ErrUnexpectedEOF
+ }
+ v = uint64(binary.LittleEndian.Uint64(dAtA[iNdEx:]))
+ iNdEx += 8
+ m.Value = float64(math.Float64frombits(v))
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *GetBytesResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: GetBytesResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: GetBytesResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Exists", wireType)
+ }
+ var v int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ v |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ m.Exists = bool(v != 0)
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Value", wireType)
+ }
+ var byteLen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ byteLen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if byteLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + byteLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Value = append(m.Value[:0], dAtA[iNdEx:postIndex]...)
+ if m.Value == nil {
+ m.Value = []byte{}
+ }
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *RemoveRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: RemoveRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: RemoveRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Key", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Key = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *RemoveResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: RemoveResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: RemoveResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Success", wireType)
+ }
+ var v int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ v |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ m.Success = bool(v != 0)
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *HasRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: HasRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: HasRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Key", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Key = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *HasResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: HasResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: HasResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Exists", wireType)
+ }
+ var v int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ v |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ m.Exists = bool(v != 0)
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+
+func skip(dAtA []byte) (n int, err error) {
+ l := len(dAtA)
+ iNdEx := 0
+ depth := 0
+ for iNdEx < l {
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= (uint64(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ wireType := int(wire & 0x7)
+ switch wireType {
+ case 0:
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ iNdEx++
+ if dAtA[iNdEx-1] < 0x80 {
+ break
+ }
+ }
+ case 1:
+ iNdEx += 8
+ case 2:
+ var length int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ length |= (int(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if length < 0 {
+ return 0, ErrInvalidLength
+ }
+ iNdEx += length
+ case 3:
+ depth++
+ case 4:
+ if depth == 0 {
+ return 0, ErrUnexpectedEndOfGroup
+ }
+ depth--
+ case 5:
+ iNdEx += 4
+ default:
+ return 0, fmt.Errorf("proto: illegal wireType %d", wireType)
+ }
+ if iNdEx < 0 {
+ return 0, ErrInvalidLength
+ }
+ if depth == 0 {
+ return iNdEx, nil
+ }
+ }
+ return 0, io.ErrUnexpectedEOF
+}
+
+var (
+ ErrInvalidLength = fmt.Errorf("proto: negative length found during unmarshaling")
+ ErrIntOverflow = fmt.Errorf("proto: integer overflow")
+ ErrUnexpectedEndOfGroup = fmt.Errorf("proto: unexpected end of group")
+)
diff --git a/plugins/host/config/config.pb.go b/plugins/host/config/config.pb.go
new file mode 100644
index 000000000..dfc70af19
--- /dev/null
+++ b/plugins/host/config/config.pb.go
@@ -0,0 +1,54 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/config/config.proto
+
+package config
+
+import (
+ context "context"
+ protoreflect "google.golang.org/protobuf/reflect/protoreflect"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+type GetPluginConfigRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+}
+
+func (x *GetPluginConfigRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+type GetPluginConfigResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Config map[string]string `protobuf:"bytes,1,rep,name=config,proto3" json:"config,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"`
+}
+
+func (x *GetPluginConfigResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *GetPluginConfigResponse) GetConfig() map[string]string {
+ if x != nil {
+ return x.Config
+ }
+ return nil
+}
+
+// go:plugin type=host version=1
+type ConfigService interface {
+ GetPluginConfig(context.Context, *GetPluginConfigRequest) (*GetPluginConfigResponse, error)
+}
diff --git a/plugins/host/config/config.proto b/plugins/host/config/config.proto
new file mode 100644
index 000000000..76076b47b
--- /dev/null
+++ b/plugins/host/config/config.proto
@@ -0,0 +1,18 @@
+syntax = "proto3";
+
+package config;
+
+option go_package = "github.com/navidrome/navidrome/plugins/host/config;config";
+
+// go:plugin type=host version=1
+service ConfigService {
+ rpc GetPluginConfig(GetPluginConfigRequest) returns (GetPluginConfigResponse);
+}
+
+message GetPluginConfigRequest {
+ // No fields needed; plugin name is inferred from context
+}
+
+message GetPluginConfigResponse {
+ map config = 1;
+}
\ No newline at end of file
diff --git a/plugins/host/config/config_host.pb.go b/plugins/host/config/config_host.pb.go
new file mode 100644
index 000000000..87894f1a2
--- /dev/null
+++ b/plugins/host/config/config_host.pb.go
@@ -0,0 +1,66 @@
+//go:build !wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/config/config.proto
+
+package config
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ wazero "github.com/tetratelabs/wazero"
+ api "github.com/tetratelabs/wazero/api"
+)
+
+const (
+ i32 = api.ValueTypeI32
+ i64 = api.ValueTypeI64
+)
+
+type _configService struct {
+ ConfigService
+}
+
+// Instantiate a Go-defined module named "env" that exports host functions.
+func Instantiate(ctx context.Context, r wazero.Runtime, hostFunctions ConfigService) error {
+ envBuilder := r.NewHostModuleBuilder("env")
+ h := _configService{hostFunctions}
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._GetPluginConfig), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("get_plugin_config")
+
+ _, err := envBuilder.Instantiate(ctx)
+ return err
+}
+
+func (h _configService) _GetPluginConfig(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(GetPluginConfigRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.GetPluginConfig(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
diff --git a/plugins/host/config/config_plugin.pb.go b/plugins/host/config/config_plugin.pb.go
new file mode 100644
index 000000000..45c60d13a
--- /dev/null
+++ b/plugins/host/config/config_plugin.pb.go
@@ -0,0 +1,44 @@
+//go:build wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/config/config.proto
+
+package config
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ _ "unsafe"
+)
+
+type configService struct{}
+
+func NewConfigService() ConfigService {
+ return configService{}
+}
+
+//go:wasmimport env get_plugin_config
+func _get_plugin_config(ptr uint32, size uint32) uint64
+
+func (h configService) GetPluginConfig(ctx context.Context, request *GetPluginConfigRequest) (*GetPluginConfigResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _get_plugin_config(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(GetPluginConfigResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
diff --git a/plugins/host/config/config_plugin_dev.go b/plugins/host/config/config_plugin_dev.go
new file mode 100644
index 000000000..dddbc9ceb
--- /dev/null
+++ b/plugins/host/config/config_plugin_dev.go
@@ -0,0 +1,7 @@
+//go:build !wasip1
+
+package config
+
+func NewConfigService() ConfigService {
+ panic("not implemented")
+}
diff --git a/plugins/host/config/config_vtproto.pb.go b/plugins/host/config/config_vtproto.pb.go
new file mode 100644
index 000000000..295da164d
--- /dev/null
+++ b/plugins/host/config/config_vtproto.pb.go
@@ -0,0 +1,466 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/config/config.proto
+
+package config
+
+import (
+ fmt "fmt"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+ io "io"
+ bits "math/bits"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+func (m *GetPluginConfigRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *GetPluginConfigRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *GetPluginConfigRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *GetPluginConfigResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *GetPluginConfigResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *GetPluginConfigResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Config) > 0 {
+ for k := range m.Config {
+ v := m.Config[k]
+ baseI := i
+ i -= len(v)
+ copy(dAtA[i:], v)
+ i = encodeVarint(dAtA, i, uint64(len(v)))
+ i--
+ dAtA[i] = 0x12
+ i -= len(k)
+ copy(dAtA[i:], k)
+ i = encodeVarint(dAtA, i, uint64(len(k)))
+ i--
+ dAtA[i] = 0xa
+ i = encodeVarint(dAtA, i, uint64(baseI-i))
+ i--
+ dAtA[i] = 0xa
+ }
+ }
+ return len(dAtA) - i, nil
+}
+
+func encodeVarint(dAtA []byte, offset int, v uint64) int {
+ offset -= sov(v)
+ base := offset
+ for v >= 1<<7 {
+ dAtA[offset] = uint8(v&0x7f | 0x80)
+ v >>= 7
+ offset++
+ }
+ dAtA[offset] = uint8(v)
+ return base
+}
+func (m *GetPluginConfigRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *GetPluginConfigResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if len(m.Config) > 0 {
+ for k, v := range m.Config {
+ _ = k
+ _ = v
+ mapEntrySize := 1 + len(k) + sov(uint64(len(k))) + 1 + len(v) + sov(uint64(len(v)))
+ n += mapEntrySize + 1 + sov(uint64(mapEntrySize))
+ }
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func sov(x uint64) (n int) {
+ return (bits.Len64(x|1) + 6) / 7
+}
+func soz(x uint64) (n int) {
+ return sov(uint64((x << 1) ^ uint64((int64(x) >> 63))))
+}
+func (m *GetPluginConfigRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: GetPluginConfigRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: GetPluginConfigRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *GetPluginConfigResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: GetPluginConfigResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: GetPluginConfigResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Config", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ if m.Config == nil {
+ m.Config = make(map[string]string)
+ }
+ var mapkey string
+ var mapvalue string
+ for iNdEx < postIndex {
+ entryPreIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ if fieldNum == 1 {
+ var stringLenmapkey uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLenmapkey |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLenmapkey := int(stringLenmapkey)
+ if intStringLenmapkey < 0 {
+ return ErrInvalidLength
+ }
+ postStringIndexmapkey := iNdEx + intStringLenmapkey
+ if postStringIndexmapkey < 0 {
+ return ErrInvalidLength
+ }
+ if postStringIndexmapkey > l {
+ return io.ErrUnexpectedEOF
+ }
+ mapkey = string(dAtA[iNdEx:postStringIndexmapkey])
+ iNdEx = postStringIndexmapkey
+ } else if fieldNum == 2 {
+ var stringLenmapvalue uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLenmapvalue |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLenmapvalue := int(stringLenmapvalue)
+ if intStringLenmapvalue < 0 {
+ return ErrInvalidLength
+ }
+ postStringIndexmapvalue := iNdEx + intStringLenmapvalue
+ if postStringIndexmapvalue < 0 {
+ return ErrInvalidLength
+ }
+ if postStringIndexmapvalue > l {
+ return io.ErrUnexpectedEOF
+ }
+ mapvalue = string(dAtA[iNdEx:postStringIndexmapvalue])
+ iNdEx = postStringIndexmapvalue
+ } else {
+ iNdEx = entryPreIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > postIndex {
+ return io.ErrUnexpectedEOF
+ }
+ iNdEx += skippy
+ }
+ }
+ m.Config[mapkey] = mapvalue
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+
+func skip(dAtA []byte) (n int, err error) {
+ l := len(dAtA)
+ iNdEx := 0
+ depth := 0
+ for iNdEx < l {
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= (uint64(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ wireType := int(wire & 0x7)
+ switch wireType {
+ case 0:
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ iNdEx++
+ if dAtA[iNdEx-1] < 0x80 {
+ break
+ }
+ }
+ case 1:
+ iNdEx += 8
+ case 2:
+ var length int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ length |= (int(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if length < 0 {
+ return 0, ErrInvalidLength
+ }
+ iNdEx += length
+ case 3:
+ depth++
+ case 4:
+ if depth == 0 {
+ return 0, ErrUnexpectedEndOfGroup
+ }
+ depth--
+ case 5:
+ iNdEx += 4
+ default:
+ return 0, fmt.Errorf("proto: illegal wireType %d", wireType)
+ }
+ if iNdEx < 0 {
+ return 0, ErrInvalidLength
+ }
+ if depth == 0 {
+ return iNdEx, nil
+ }
+ }
+ return 0, io.ErrUnexpectedEOF
+}
+
+var (
+ ErrInvalidLength = fmt.Errorf("proto: negative length found during unmarshaling")
+ ErrIntOverflow = fmt.Errorf("proto: integer overflow")
+ ErrUnexpectedEndOfGroup = fmt.Errorf("proto: unexpected end of group")
+)
diff --git a/plugins/host/http/http.pb.go b/plugins/host/http/http.pb.go
new file mode 100644
index 000000000..0bc2c5040
--- /dev/null
+++ b/plugins/host/http/http.pb.go
@@ -0,0 +1,117 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/http/http.proto
+
+package http
+
+import (
+ context "context"
+ protoreflect "google.golang.org/protobuf/reflect/protoreflect"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+type HttpRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Url string `protobuf:"bytes,1,opt,name=url,proto3" json:"url,omitempty"`
+ Headers map[string]string `protobuf:"bytes,2,rep,name=headers,proto3" json:"headers,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"`
+ TimeoutMs int32 `protobuf:"varint,3,opt,name=timeout_ms,json=timeoutMs,proto3" json:"timeout_ms,omitempty"`
+ Body []byte `protobuf:"bytes,4,opt,name=body,proto3" json:"body,omitempty"` // Ignored for GET/DELETE/HEAD/OPTIONS
+}
+
+func (x *HttpRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *HttpRequest) GetUrl() string {
+ if x != nil {
+ return x.Url
+ }
+ return ""
+}
+
+func (x *HttpRequest) GetHeaders() map[string]string {
+ if x != nil {
+ return x.Headers
+ }
+ return nil
+}
+
+func (x *HttpRequest) GetTimeoutMs() int32 {
+ if x != nil {
+ return x.TimeoutMs
+ }
+ return 0
+}
+
+func (x *HttpRequest) GetBody() []byte {
+ if x != nil {
+ return x.Body
+ }
+ return nil
+}
+
+type HttpResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Status int32 `protobuf:"varint,1,opt,name=status,proto3" json:"status,omitempty"`
+ Body []byte `protobuf:"bytes,2,opt,name=body,proto3" json:"body,omitempty"`
+ Headers map[string]string `protobuf:"bytes,3,rep,name=headers,proto3" json:"headers,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"`
+ Error string `protobuf:"bytes,4,opt,name=error,proto3" json:"error,omitempty"` // Non-empty if network/protocol error
+}
+
+func (x *HttpResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *HttpResponse) GetStatus() int32 {
+ if x != nil {
+ return x.Status
+ }
+ return 0
+}
+
+func (x *HttpResponse) GetBody() []byte {
+ if x != nil {
+ return x.Body
+ }
+ return nil
+}
+
+func (x *HttpResponse) GetHeaders() map[string]string {
+ if x != nil {
+ return x.Headers
+ }
+ return nil
+}
+
+func (x *HttpResponse) GetError() string {
+ if x != nil {
+ return x.Error
+ }
+ return ""
+}
+
+// go:plugin type=host version=1
+type HttpService interface {
+ Get(context.Context, *HttpRequest) (*HttpResponse, error)
+ Post(context.Context, *HttpRequest) (*HttpResponse, error)
+ Put(context.Context, *HttpRequest) (*HttpResponse, error)
+ Delete(context.Context, *HttpRequest) (*HttpResponse, error)
+ Patch(context.Context, *HttpRequest) (*HttpResponse, error)
+ Head(context.Context, *HttpRequest) (*HttpResponse, error)
+ Options(context.Context, *HttpRequest) (*HttpResponse, error)
+}
diff --git a/plugins/host/http/http.proto b/plugins/host/http/http.proto
new file mode 100644
index 000000000..2ed7a4262
--- /dev/null
+++ b/plugins/host/http/http.proto
@@ -0,0 +1,30 @@
+syntax = "proto3";
+
+package http;
+
+option go_package = "github.com/navidrome/navidrome/plugins/host/http;http";
+
+// go:plugin type=host version=1
+service HttpService {
+ rpc Get(HttpRequest) returns (HttpResponse);
+ rpc Post(HttpRequest) returns (HttpResponse);
+ rpc Put(HttpRequest) returns (HttpResponse);
+ rpc Delete(HttpRequest) returns (HttpResponse);
+ rpc Patch(HttpRequest) returns (HttpResponse);
+ rpc Head(HttpRequest) returns (HttpResponse);
+ rpc Options(HttpRequest) returns (HttpResponse);
+}
+
+message HttpRequest {
+ string url = 1;
+ map headers = 2;
+ int32 timeout_ms = 3;
+ bytes body = 4; // Ignored for GET/DELETE/HEAD/OPTIONS
+}
+
+message HttpResponse {
+ int32 status = 1;
+ bytes body = 2;
+ map headers = 3;
+ string error = 4; // Non-empty if network/protocol error
+}
\ No newline at end of file
diff --git a/plugins/host/http/http_host.pb.go b/plugins/host/http/http_host.pb.go
new file mode 100644
index 000000000..326aba508
--- /dev/null
+++ b/plugins/host/http/http_host.pb.go
@@ -0,0 +1,258 @@
+//go:build !wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/http/http.proto
+
+package http
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ wazero "github.com/tetratelabs/wazero"
+ api "github.com/tetratelabs/wazero/api"
+)
+
+const (
+ i32 = api.ValueTypeI32
+ i64 = api.ValueTypeI64
+)
+
+type _httpService struct {
+ HttpService
+}
+
+// Instantiate a Go-defined module named "env" that exports host functions.
+func Instantiate(ctx context.Context, r wazero.Runtime, hostFunctions HttpService) error {
+ envBuilder := r.NewHostModuleBuilder("env")
+ h := _httpService{hostFunctions}
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._Get), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("get")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._Post), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("post")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._Put), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("put")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._Delete), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("delete")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._Patch), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("patch")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._Head), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("head")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._Options), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("options")
+
+ _, err := envBuilder.Instantiate(ctx)
+ return err
+}
+
+func (h _httpService) _Get(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(HttpRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.Get(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+func (h _httpService) _Post(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(HttpRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.Post(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+func (h _httpService) _Put(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(HttpRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.Put(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+func (h _httpService) _Delete(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(HttpRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.Delete(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+func (h _httpService) _Patch(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(HttpRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.Patch(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+func (h _httpService) _Head(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(HttpRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.Head(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+func (h _httpService) _Options(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(HttpRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.Options(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
diff --git a/plugins/host/http/http_plugin.pb.go b/plugins/host/http/http_plugin.pb.go
new file mode 100644
index 000000000..2e8c21891
--- /dev/null
+++ b/plugins/host/http/http_plugin.pb.go
@@ -0,0 +1,182 @@
+//go:build wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/http/http.proto
+
+package http
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ _ "unsafe"
+)
+
+type httpService struct{}
+
+func NewHttpService() HttpService {
+ return httpService{}
+}
+
+//go:wasmimport env get
+func _get(ptr uint32, size uint32) uint64
+
+func (h httpService) Get(ctx context.Context, request *HttpRequest) (*HttpResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _get(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(HttpResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env post
+func _post(ptr uint32, size uint32) uint64
+
+func (h httpService) Post(ctx context.Context, request *HttpRequest) (*HttpResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _post(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(HttpResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env put
+func _put(ptr uint32, size uint32) uint64
+
+func (h httpService) Put(ctx context.Context, request *HttpRequest) (*HttpResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _put(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(HttpResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env delete
+func _delete(ptr uint32, size uint32) uint64
+
+func (h httpService) Delete(ctx context.Context, request *HttpRequest) (*HttpResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _delete(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(HttpResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env patch
+func _patch(ptr uint32, size uint32) uint64
+
+func (h httpService) Patch(ctx context.Context, request *HttpRequest) (*HttpResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _patch(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(HttpResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env head
+func _head(ptr uint32, size uint32) uint64
+
+func (h httpService) Head(ctx context.Context, request *HttpRequest) (*HttpResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _head(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(HttpResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env options
+func _options(ptr uint32, size uint32) uint64
+
+func (h httpService) Options(ctx context.Context, request *HttpRequest) (*HttpResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _options(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(HttpResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
diff --git a/plugins/host/http/http_plugin_dev.go b/plugins/host/http/http_plugin_dev.go
new file mode 100644
index 000000000..04e3c2508
--- /dev/null
+++ b/plugins/host/http/http_plugin_dev.go
@@ -0,0 +1,7 @@
+//go:build !wasip1
+
+package http
+
+func NewHttpService() HttpService {
+ panic("not implemented")
+}
diff --git a/plugins/host/http/http_vtproto.pb.go b/plugins/host/http/http_vtproto.pb.go
new file mode 100644
index 000000000..064fdb08a
--- /dev/null
+++ b/plugins/host/http/http_vtproto.pb.go
@@ -0,0 +1,850 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/http/http.proto
+
+package http
+
+import (
+ fmt "fmt"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+ io "io"
+ bits "math/bits"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+func (m *HttpRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *HttpRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *HttpRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Body) > 0 {
+ i -= len(m.Body)
+ copy(dAtA[i:], m.Body)
+ i = encodeVarint(dAtA, i, uint64(len(m.Body)))
+ i--
+ dAtA[i] = 0x22
+ }
+ if m.TimeoutMs != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.TimeoutMs))
+ i--
+ dAtA[i] = 0x18
+ }
+ if len(m.Headers) > 0 {
+ for k := range m.Headers {
+ v := m.Headers[k]
+ baseI := i
+ i -= len(v)
+ copy(dAtA[i:], v)
+ i = encodeVarint(dAtA, i, uint64(len(v)))
+ i--
+ dAtA[i] = 0x12
+ i -= len(k)
+ copy(dAtA[i:], k)
+ i = encodeVarint(dAtA, i, uint64(len(k)))
+ i--
+ dAtA[i] = 0xa
+ i = encodeVarint(dAtA, i, uint64(baseI-i))
+ i--
+ dAtA[i] = 0x12
+ }
+ }
+ if len(m.Url) > 0 {
+ i -= len(m.Url)
+ copy(dAtA[i:], m.Url)
+ i = encodeVarint(dAtA, i, uint64(len(m.Url)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *HttpResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *HttpResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *HttpResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Error) > 0 {
+ i -= len(m.Error)
+ copy(dAtA[i:], m.Error)
+ i = encodeVarint(dAtA, i, uint64(len(m.Error)))
+ i--
+ dAtA[i] = 0x22
+ }
+ if len(m.Headers) > 0 {
+ for k := range m.Headers {
+ v := m.Headers[k]
+ baseI := i
+ i -= len(v)
+ copy(dAtA[i:], v)
+ i = encodeVarint(dAtA, i, uint64(len(v)))
+ i--
+ dAtA[i] = 0x12
+ i -= len(k)
+ copy(dAtA[i:], k)
+ i = encodeVarint(dAtA, i, uint64(len(k)))
+ i--
+ dAtA[i] = 0xa
+ i = encodeVarint(dAtA, i, uint64(baseI-i))
+ i--
+ dAtA[i] = 0x1a
+ }
+ }
+ if len(m.Body) > 0 {
+ i -= len(m.Body)
+ copy(dAtA[i:], m.Body)
+ i = encodeVarint(dAtA, i, uint64(len(m.Body)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if m.Status != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.Status))
+ i--
+ dAtA[i] = 0x8
+ }
+ return len(dAtA) - i, nil
+}
+
+func encodeVarint(dAtA []byte, offset int, v uint64) int {
+ offset -= sov(v)
+ base := offset
+ for v >= 1<<7 {
+ dAtA[offset] = uint8(v&0x7f | 0x80)
+ v >>= 7
+ offset++
+ }
+ dAtA[offset] = uint8(v)
+ return base
+}
+func (m *HttpRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Url)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if len(m.Headers) > 0 {
+ for k, v := range m.Headers {
+ _ = k
+ _ = v
+ mapEntrySize := 1 + len(k) + sov(uint64(len(k))) + 1 + len(v) + sov(uint64(len(v)))
+ n += mapEntrySize + 1 + sov(uint64(mapEntrySize))
+ }
+ }
+ if m.TimeoutMs != 0 {
+ n += 1 + sov(uint64(m.TimeoutMs))
+ }
+ l = len(m.Body)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *HttpResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if m.Status != 0 {
+ n += 1 + sov(uint64(m.Status))
+ }
+ l = len(m.Body)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if len(m.Headers) > 0 {
+ for k, v := range m.Headers {
+ _ = k
+ _ = v
+ mapEntrySize := 1 + len(k) + sov(uint64(len(k))) + 1 + len(v) + sov(uint64(len(v)))
+ n += mapEntrySize + 1 + sov(uint64(mapEntrySize))
+ }
+ }
+ l = len(m.Error)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func sov(x uint64) (n int) {
+ return (bits.Len64(x|1) + 6) / 7
+}
+func soz(x uint64) (n int) {
+ return sov(uint64((x << 1) ^ uint64((int64(x) >> 63))))
+}
+func (m *HttpRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: HttpRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: HttpRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Url", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Url = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Headers", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ if m.Headers == nil {
+ m.Headers = make(map[string]string)
+ }
+ var mapkey string
+ var mapvalue string
+ for iNdEx < postIndex {
+ entryPreIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ if fieldNum == 1 {
+ var stringLenmapkey uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLenmapkey |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLenmapkey := int(stringLenmapkey)
+ if intStringLenmapkey < 0 {
+ return ErrInvalidLength
+ }
+ postStringIndexmapkey := iNdEx + intStringLenmapkey
+ if postStringIndexmapkey < 0 {
+ return ErrInvalidLength
+ }
+ if postStringIndexmapkey > l {
+ return io.ErrUnexpectedEOF
+ }
+ mapkey = string(dAtA[iNdEx:postStringIndexmapkey])
+ iNdEx = postStringIndexmapkey
+ } else if fieldNum == 2 {
+ var stringLenmapvalue uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLenmapvalue |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLenmapvalue := int(stringLenmapvalue)
+ if intStringLenmapvalue < 0 {
+ return ErrInvalidLength
+ }
+ postStringIndexmapvalue := iNdEx + intStringLenmapvalue
+ if postStringIndexmapvalue < 0 {
+ return ErrInvalidLength
+ }
+ if postStringIndexmapvalue > l {
+ return io.ErrUnexpectedEOF
+ }
+ mapvalue = string(dAtA[iNdEx:postStringIndexmapvalue])
+ iNdEx = postStringIndexmapvalue
+ } else {
+ iNdEx = entryPreIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > postIndex {
+ return io.ErrUnexpectedEOF
+ }
+ iNdEx += skippy
+ }
+ }
+ m.Headers[mapkey] = mapvalue
+ iNdEx = postIndex
+ case 3:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field TimeoutMs", wireType)
+ }
+ m.TimeoutMs = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.TimeoutMs |= int32(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ case 4:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Body", wireType)
+ }
+ var byteLen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ byteLen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if byteLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + byteLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Body = append(m.Body[:0], dAtA[iNdEx:postIndex]...)
+ if m.Body == nil {
+ m.Body = []byte{}
+ }
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *HttpResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: HttpResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: HttpResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Status", wireType)
+ }
+ m.Status = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.Status |= int32(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Body", wireType)
+ }
+ var byteLen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ byteLen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if byteLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + byteLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Body = append(m.Body[:0], dAtA[iNdEx:postIndex]...)
+ if m.Body == nil {
+ m.Body = []byte{}
+ }
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Headers", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ if m.Headers == nil {
+ m.Headers = make(map[string]string)
+ }
+ var mapkey string
+ var mapvalue string
+ for iNdEx < postIndex {
+ entryPreIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ if fieldNum == 1 {
+ var stringLenmapkey uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLenmapkey |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLenmapkey := int(stringLenmapkey)
+ if intStringLenmapkey < 0 {
+ return ErrInvalidLength
+ }
+ postStringIndexmapkey := iNdEx + intStringLenmapkey
+ if postStringIndexmapkey < 0 {
+ return ErrInvalidLength
+ }
+ if postStringIndexmapkey > l {
+ return io.ErrUnexpectedEOF
+ }
+ mapkey = string(dAtA[iNdEx:postStringIndexmapkey])
+ iNdEx = postStringIndexmapkey
+ } else if fieldNum == 2 {
+ var stringLenmapvalue uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLenmapvalue |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLenmapvalue := int(stringLenmapvalue)
+ if intStringLenmapvalue < 0 {
+ return ErrInvalidLength
+ }
+ postStringIndexmapvalue := iNdEx + intStringLenmapvalue
+ if postStringIndexmapvalue < 0 {
+ return ErrInvalidLength
+ }
+ if postStringIndexmapvalue > l {
+ return io.ErrUnexpectedEOF
+ }
+ mapvalue = string(dAtA[iNdEx:postStringIndexmapvalue])
+ iNdEx = postStringIndexmapvalue
+ } else {
+ iNdEx = entryPreIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > postIndex {
+ return io.ErrUnexpectedEOF
+ }
+ iNdEx += skippy
+ }
+ }
+ m.Headers[mapkey] = mapvalue
+ iNdEx = postIndex
+ case 4:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Error", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Error = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+
+func skip(dAtA []byte) (n int, err error) {
+ l := len(dAtA)
+ iNdEx := 0
+ depth := 0
+ for iNdEx < l {
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= (uint64(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ wireType := int(wire & 0x7)
+ switch wireType {
+ case 0:
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ iNdEx++
+ if dAtA[iNdEx-1] < 0x80 {
+ break
+ }
+ }
+ case 1:
+ iNdEx += 8
+ case 2:
+ var length int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ length |= (int(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if length < 0 {
+ return 0, ErrInvalidLength
+ }
+ iNdEx += length
+ case 3:
+ depth++
+ case 4:
+ if depth == 0 {
+ return 0, ErrUnexpectedEndOfGroup
+ }
+ depth--
+ case 5:
+ iNdEx += 4
+ default:
+ return 0, fmt.Errorf("proto: illegal wireType %d", wireType)
+ }
+ if iNdEx < 0 {
+ return 0, ErrInvalidLength
+ }
+ if depth == 0 {
+ return iNdEx, nil
+ }
+ }
+ return 0, io.ErrUnexpectedEOF
+}
+
+var (
+ ErrInvalidLength = fmt.Errorf("proto: negative length found during unmarshaling")
+ ErrIntOverflow = fmt.Errorf("proto: integer overflow")
+ ErrUnexpectedEndOfGroup = fmt.Errorf("proto: unexpected end of group")
+)
diff --git a/plugins/host/scheduler/scheduler.pb.go b/plugins/host/scheduler/scheduler.pb.go
new file mode 100644
index 000000000..07d250cc5
--- /dev/null
+++ b/plugins/host/scheduler/scheduler.pb.go
@@ -0,0 +1,212 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/scheduler/scheduler.proto
+
+package scheduler
+
+import (
+ context "context"
+ protoreflect "google.golang.org/protobuf/reflect/protoreflect"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+type ScheduleOneTimeRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ DelaySeconds int32 `protobuf:"varint,1,opt,name=delay_seconds,json=delaySeconds,proto3" json:"delay_seconds,omitempty"` // Delay in seconds
+ Payload []byte `protobuf:"bytes,2,opt,name=payload,proto3" json:"payload,omitempty"` // Serialized data to pass to the callback
+ ScheduleId string `protobuf:"bytes,3,opt,name=schedule_id,json=scheduleId,proto3" json:"schedule_id,omitempty"` // Optional custom ID (if not provided, one will be generated)
+}
+
+func (x *ScheduleOneTimeRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ScheduleOneTimeRequest) GetDelaySeconds() int32 {
+ if x != nil {
+ return x.DelaySeconds
+ }
+ return 0
+}
+
+func (x *ScheduleOneTimeRequest) GetPayload() []byte {
+ if x != nil {
+ return x.Payload
+ }
+ return nil
+}
+
+func (x *ScheduleOneTimeRequest) GetScheduleId() string {
+ if x != nil {
+ return x.ScheduleId
+ }
+ return ""
+}
+
+type ScheduleRecurringRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ CronExpression string `protobuf:"bytes,1,opt,name=cron_expression,json=cronExpression,proto3" json:"cron_expression,omitempty"` // Cron expression (e.g. "0 0 * * *" for daily at midnight)
+ Payload []byte `protobuf:"bytes,2,opt,name=payload,proto3" json:"payload,omitempty"` // Serialized data to pass to the callback
+ ScheduleId string `protobuf:"bytes,3,opt,name=schedule_id,json=scheduleId,proto3" json:"schedule_id,omitempty"` // Optional custom ID (if not provided, one will be generated)
+}
+
+func (x *ScheduleRecurringRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ScheduleRecurringRequest) GetCronExpression() string {
+ if x != nil {
+ return x.CronExpression
+ }
+ return ""
+}
+
+func (x *ScheduleRecurringRequest) GetPayload() []byte {
+ if x != nil {
+ return x.Payload
+ }
+ return nil
+}
+
+func (x *ScheduleRecurringRequest) GetScheduleId() string {
+ if x != nil {
+ return x.ScheduleId
+ }
+ return ""
+}
+
+type ScheduleResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ ScheduleId string `protobuf:"bytes,1,opt,name=schedule_id,json=scheduleId,proto3" json:"schedule_id,omitempty"` // ID to reference this scheduled job
+}
+
+func (x *ScheduleResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ScheduleResponse) GetScheduleId() string {
+ if x != nil {
+ return x.ScheduleId
+ }
+ return ""
+}
+
+type CancelRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ ScheduleId string `protobuf:"bytes,1,opt,name=schedule_id,json=scheduleId,proto3" json:"schedule_id,omitempty"` // ID of the schedule to cancel
+}
+
+func (x *CancelRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *CancelRequest) GetScheduleId() string {
+ if x != nil {
+ return x.ScheduleId
+ }
+ return ""
+}
+
+type CancelResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Success bool `protobuf:"varint,1,opt,name=success,proto3" json:"success,omitempty"` // Whether cancellation was successful
+ Error string `protobuf:"bytes,2,opt,name=error,proto3" json:"error,omitempty"` // Error message if cancellation failed
+}
+
+func (x *CancelResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *CancelResponse) GetSuccess() bool {
+ if x != nil {
+ return x.Success
+ }
+ return false
+}
+
+func (x *CancelResponse) GetError() string {
+ if x != nil {
+ return x.Error
+ }
+ return ""
+}
+
+type TimeNowRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+}
+
+func (x *TimeNowRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+type TimeNowResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Rfc3339Nano string `protobuf:"bytes,1,opt,name=rfc3339_nano,json=rfc3339Nano,proto3" json:"rfc3339_nano,omitempty"` // Current time in RFC3339Nano format
+ UnixMilli int64 `protobuf:"varint,2,opt,name=unix_milli,json=unixMilli,proto3" json:"unix_milli,omitempty"` // Current time as Unix milliseconds timestamp
+ LocalTimeZone string `protobuf:"bytes,3,opt,name=local_time_zone,json=localTimeZone,proto3" json:"local_time_zone,omitempty"` // Local timezone name (e.g., "America/New_York", "UTC")
+}
+
+func (x *TimeNowResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *TimeNowResponse) GetRfc3339Nano() string {
+ if x != nil {
+ return x.Rfc3339Nano
+ }
+ return ""
+}
+
+func (x *TimeNowResponse) GetUnixMilli() int64 {
+ if x != nil {
+ return x.UnixMilli
+ }
+ return 0
+}
+
+func (x *TimeNowResponse) GetLocalTimeZone() string {
+ if x != nil {
+ return x.LocalTimeZone
+ }
+ return ""
+}
+
+// go:plugin type=host version=1
+type SchedulerService interface {
+ // One-time event scheduling
+ ScheduleOneTime(context.Context, *ScheduleOneTimeRequest) (*ScheduleResponse, error)
+ // Recurring event scheduling
+ ScheduleRecurring(context.Context, *ScheduleRecurringRequest) (*ScheduleResponse, error)
+ // Cancel any scheduled job
+ CancelSchedule(context.Context, *CancelRequest) (*CancelResponse, error)
+ // Get current time in multiple formats
+ TimeNow(context.Context, *TimeNowRequest) (*TimeNowResponse, error)
+}
diff --git a/plugins/host/scheduler/scheduler.proto b/plugins/host/scheduler/scheduler.proto
new file mode 100644
index 000000000..d164b4f90
--- /dev/null
+++ b/plugins/host/scheduler/scheduler.proto
@@ -0,0 +1,55 @@
+syntax = "proto3";
+
+package scheduler;
+
+option go_package = "github.com/navidrome/navidrome/plugins/host/scheduler;scheduler";
+
+// go:plugin type=host version=1
+service SchedulerService {
+ // One-time event scheduling
+ rpc ScheduleOneTime(ScheduleOneTimeRequest) returns (ScheduleResponse);
+
+ // Recurring event scheduling
+ rpc ScheduleRecurring(ScheduleRecurringRequest) returns (ScheduleResponse);
+
+ // Cancel any scheduled job
+ rpc CancelSchedule(CancelRequest) returns (CancelResponse);
+
+ // Get current time in multiple formats
+ rpc TimeNow(TimeNowRequest) returns (TimeNowResponse);
+}
+
+message ScheduleOneTimeRequest {
+ int32 delay_seconds = 1; // Delay in seconds
+ bytes payload = 2; // Serialized data to pass to the callback
+ string schedule_id = 3; // Optional custom ID (if not provided, one will be generated)
+}
+
+message ScheduleRecurringRequest {
+ string cron_expression = 1; // Cron expression (e.g. "0 0 * * *" for daily at midnight)
+ bytes payload = 2; // Serialized data to pass to the callback
+ string schedule_id = 3; // Optional custom ID (if not provided, one will be generated)
+}
+
+message ScheduleResponse {
+ string schedule_id = 1; // ID to reference this scheduled job
+}
+
+message CancelRequest {
+ string schedule_id = 1; // ID of the schedule to cancel
+}
+
+message CancelResponse {
+ bool success = 1; // Whether cancellation was successful
+ string error = 2; // Error message if cancellation failed
+}
+
+message TimeNowRequest {
+ // Empty request - no parameters needed
+}
+
+message TimeNowResponse {
+ string rfc3339_nano = 1; // Current time in RFC3339Nano format
+ int64 unix_milli = 2; // Current time as Unix milliseconds timestamp
+ string local_time_zone = 3; // Local timezone name (e.g., "America/New_York", "UTC")
+}
\ No newline at end of file
diff --git a/plugins/host/scheduler/scheduler_host.pb.go b/plugins/host/scheduler/scheduler_host.pb.go
new file mode 100644
index 000000000..714603a3b
--- /dev/null
+++ b/plugins/host/scheduler/scheduler_host.pb.go
@@ -0,0 +1,170 @@
+//go:build !wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/scheduler/scheduler.proto
+
+package scheduler
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ wazero "github.com/tetratelabs/wazero"
+ api "github.com/tetratelabs/wazero/api"
+)
+
+const (
+ i32 = api.ValueTypeI32
+ i64 = api.ValueTypeI64
+)
+
+type _schedulerService struct {
+ SchedulerService
+}
+
+// Instantiate a Go-defined module named "env" that exports host functions.
+func Instantiate(ctx context.Context, r wazero.Runtime, hostFunctions SchedulerService) error {
+ envBuilder := r.NewHostModuleBuilder("env")
+ h := _schedulerService{hostFunctions}
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._ScheduleOneTime), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("schedule_one_time")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._ScheduleRecurring), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("schedule_recurring")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._CancelSchedule), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("cancel_schedule")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._TimeNow), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("time_now")
+
+ _, err := envBuilder.Instantiate(ctx)
+ return err
+}
+
+// One-time event scheduling
+
+func (h _schedulerService) _ScheduleOneTime(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(ScheduleOneTimeRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.ScheduleOneTime(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Recurring event scheduling
+
+func (h _schedulerService) _ScheduleRecurring(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(ScheduleRecurringRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.ScheduleRecurring(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Cancel any scheduled job
+
+func (h _schedulerService) _CancelSchedule(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(CancelRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.CancelSchedule(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Get current time in multiple formats
+
+func (h _schedulerService) _TimeNow(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(TimeNowRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.TimeNow(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
diff --git a/plugins/host/scheduler/scheduler_plugin.pb.go b/plugins/host/scheduler/scheduler_plugin.pb.go
new file mode 100644
index 000000000..ab7f8cd48
--- /dev/null
+++ b/plugins/host/scheduler/scheduler_plugin.pb.go
@@ -0,0 +1,113 @@
+//go:build wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/scheduler/scheduler.proto
+
+package scheduler
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ _ "unsafe"
+)
+
+type schedulerService struct{}
+
+func NewSchedulerService() SchedulerService {
+ return schedulerService{}
+}
+
+//go:wasmimport env schedule_one_time
+func _schedule_one_time(ptr uint32, size uint32) uint64
+
+func (h schedulerService) ScheduleOneTime(ctx context.Context, request *ScheduleOneTimeRequest) (*ScheduleResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _schedule_one_time(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(ScheduleResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env schedule_recurring
+func _schedule_recurring(ptr uint32, size uint32) uint64
+
+func (h schedulerService) ScheduleRecurring(ctx context.Context, request *ScheduleRecurringRequest) (*ScheduleResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _schedule_recurring(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(ScheduleResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env cancel_schedule
+func _cancel_schedule(ptr uint32, size uint32) uint64
+
+func (h schedulerService) CancelSchedule(ctx context.Context, request *CancelRequest) (*CancelResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _cancel_schedule(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(CancelResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env time_now
+func _time_now(ptr uint32, size uint32) uint64
+
+func (h schedulerService) TimeNow(ctx context.Context, request *TimeNowRequest) (*TimeNowResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _time_now(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(TimeNowResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
diff --git a/plugins/host/scheduler/scheduler_plugin_dev.go b/plugins/host/scheduler/scheduler_plugin_dev.go
new file mode 100644
index 000000000..b6feaa8e4
--- /dev/null
+++ b/plugins/host/scheduler/scheduler_plugin_dev.go
@@ -0,0 +1,7 @@
+//go:build !wasip1
+
+package scheduler
+
+func NewSchedulerService() SchedulerService {
+ panic("not implemented")
+}
diff --git a/plugins/host/scheduler/scheduler_vtproto.pb.go b/plugins/host/scheduler/scheduler_vtproto.pb.go
new file mode 100644
index 000000000..ee6421783
--- /dev/null
+++ b/plugins/host/scheduler/scheduler_vtproto.pb.go
@@ -0,0 +1,1303 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/scheduler/scheduler.proto
+
+package scheduler
+
+import (
+ fmt "fmt"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+ io "io"
+ bits "math/bits"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+func (m *ScheduleOneTimeRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ScheduleOneTimeRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ScheduleOneTimeRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.ScheduleId) > 0 {
+ i -= len(m.ScheduleId)
+ copy(dAtA[i:], m.ScheduleId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ScheduleId)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.Payload) > 0 {
+ i -= len(m.Payload)
+ copy(dAtA[i:], m.Payload)
+ i = encodeVarint(dAtA, i, uint64(len(m.Payload)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if m.DelaySeconds != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.DelaySeconds))
+ i--
+ dAtA[i] = 0x8
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ScheduleRecurringRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ScheduleRecurringRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ScheduleRecurringRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.ScheduleId) > 0 {
+ i -= len(m.ScheduleId)
+ copy(dAtA[i:], m.ScheduleId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ScheduleId)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.Payload) > 0 {
+ i -= len(m.Payload)
+ copy(dAtA[i:], m.Payload)
+ i = encodeVarint(dAtA, i, uint64(len(m.Payload)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.CronExpression) > 0 {
+ i -= len(m.CronExpression)
+ copy(dAtA[i:], m.CronExpression)
+ i = encodeVarint(dAtA, i, uint64(len(m.CronExpression)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ScheduleResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ScheduleResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ScheduleResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.ScheduleId) > 0 {
+ i -= len(m.ScheduleId)
+ copy(dAtA[i:], m.ScheduleId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ScheduleId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *CancelRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *CancelRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *CancelRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.ScheduleId) > 0 {
+ i -= len(m.ScheduleId)
+ copy(dAtA[i:], m.ScheduleId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ScheduleId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *CancelResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *CancelResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *CancelResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Error) > 0 {
+ i -= len(m.Error)
+ copy(dAtA[i:], m.Error)
+ i = encodeVarint(dAtA, i, uint64(len(m.Error)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if m.Success {
+ i--
+ if m.Success {
+ dAtA[i] = 1
+ } else {
+ dAtA[i] = 0
+ }
+ i--
+ dAtA[i] = 0x8
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *TimeNowRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *TimeNowRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *TimeNowRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *TimeNowResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *TimeNowResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *TimeNowResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.LocalTimeZone) > 0 {
+ i -= len(m.LocalTimeZone)
+ copy(dAtA[i:], m.LocalTimeZone)
+ i = encodeVarint(dAtA, i, uint64(len(m.LocalTimeZone)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if m.UnixMilli != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.UnixMilli))
+ i--
+ dAtA[i] = 0x10
+ }
+ if len(m.Rfc3339Nano) > 0 {
+ i -= len(m.Rfc3339Nano)
+ copy(dAtA[i:], m.Rfc3339Nano)
+ i = encodeVarint(dAtA, i, uint64(len(m.Rfc3339Nano)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func encodeVarint(dAtA []byte, offset int, v uint64) int {
+ offset -= sov(v)
+ base := offset
+ for v >= 1<<7 {
+ dAtA[offset] = uint8(v&0x7f | 0x80)
+ v >>= 7
+ offset++
+ }
+ dAtA[offset] = uint8(v)
+ return base
+}
+func (m *ScheduleOneTimeRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if m.DelaySeconds != 0 {
+ n += 1 + sov(uint64(m.DelaySeconds))
+ }
+ l = len(m.Payload)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.ScheduleId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ScheduleRecurringRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.CronExpression)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Payload)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.ScheduleId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ScheduleResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.ScheduleId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *CancelRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.ScheduleId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *CancelResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ if m.Success {
+ n += 2
+ }
+ l = len(m.Error)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *TimeNowRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *TimeNowResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Rfc3339Nano)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.UnixMilli != 0 {
+ n += 1 + sov(uint64(m.UnixMilli))
+ }
+ l = len(m.LocalTimeZone)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func sov(x uint64) (n int) {
+ return (bits.Len64(x|1) + 6) / 7
+}
+func soz(x uint64) (n int) {
+ return sov(uint64((x << 1) ^ uint64((int64(x) >> 63))))
+}
+func (m *ScheduleOneTimeRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ScheduleOneTimeRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ScheduleOneTimeRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field DelaySeconds", wireType)
+ }
+ m.DelaySeconds = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.DelaySeconds |= int32(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Payload", wireType)
+ }
+ var byteLen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ byteLen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if byteLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + byteLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Payload = append(m.Payload[:0], dAtA[iNdEx:postIndex]...)
+ if m.Payload == nil {
+ m.Payload = []byte{}
+ }
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ScheduleId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ScheduleId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ScheduleRecurringRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ScheduleRecurringRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ScheduleRecurringRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field CronExpression", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.CronExpression = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Payload", wireType)
+ }
+ var byteLen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ byteLen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if byteLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + byteLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Payload = append(m.Payload[:0], dAtA[iNdEx:postIndex]...)
+ if m.Payload == nil {
+ m.Payload = []byte{}
+ }
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ScheduleId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ScheduleId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ScheduleResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ScheduleResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ScheduleResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ScheduleId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ScheduleId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *CancelRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: CancelRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: CancelRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ScheduleId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ScheduleId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *CancelResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: CancelResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: CancelResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Success", wireType)
+ }
+ var v int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ v |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ m.Success = bool(v != 0)
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Error", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Error = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *TimeNowRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: TimeNowRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: TimeNowRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *TimeNowResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: TimeNowResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: TimeNowResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Rfc3339Nano", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Rfc3339Nano = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field UnixMilli", wireType)
+ }
+ m.UnixMilli = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.UnixMilli |= int64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field LocalTimeZone", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.LocalTimeZone = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+
+func skip(dAtA []byte) (n int, err error) {
+ l := len(dAtA)
+ iNdEx := 0
+ depth := 0
+ for iNdEx < l {
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= (uint64(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ wireType := int(wire & 0x7)
+ switch wireType {
+ case 0:
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ iNdEx++
+ if dAtA[iNdEx-1] < 0x80 {
+ break
+ }
+ }
+ case 1:
+ iNdEx += 8
+ case 2:
+ var length int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ length |= (int(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if length < 0 {
+ return 0, ErrInvalidLength
+ }
+ iNdEx += length
+ case 3:
+ depth++
+ case 4:
+ if depth == 0 {
+ return 0, ErrUnexpectedEndOfGroup
+ }
+ depth--
+ case 5:
+ iNdEx += 4
+ default:
+ return 0, fmt.Errorf("proto: illegal wireType %d", wireType)
+ }
+ if iNdEx < 0 {
+ return 0, ErrInvalidLength
+ }
+ if depth == 0 {
+ return iNdEx, nil
+ }
+ }
+ return 0, io.ErrUnexpectedEOF
+}
+
+var (
+ ErrInvalidLength = fmt.Errorf("proto: negative length found during unmarshaling")
+ ErrIntOverflow = fmt.Errorf("proto: integer overflow")
+ ErrUnexpectedEndOfGroup = fmt.Errorf("proto: unexpected end of group")
+)
diff --git a/plugins/host/subsonicapi/subsonicapi.pb.go b/plugins/host/subsonicapi/subsonicapi.pb.go
new file mode 100644
index 000000000..0dbd9054f
--- /dev/null
+++ b/plugins/host/subsonicapi/subsonicapi.pb.go
@@ -0,0 +1,71 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/subsonicapi/subsonicapi.proto
+
+package subsonicapi
+
+import (
+ context "context"
+ protoreflect "google.golang.org/protobuf/reflect/protoreflect"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+type CallRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Url string `protobuf:"bytes,1,opt,name=url,proto3" json:"url,omitempty"`
+}
+
+func (x *CallRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *CallRequest) GetUrl() string {
+ if x != nil {
+ return x.Url
+ }
+ return ""
+}
+
+type CallResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Json string `protobuf:"bytes,1,opt,name=json,proto3" json:"json,omitempty"`
+ Error string `protobuf:"bytes,2,opt,name=error,proto3" json:"error,omitempty"` // Non-empty if operation failed
+}
+
+func (x *CallResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *CallResponse) GetJson() string {
+ if x != nil {
+ return x.Json
+ }
+ return ""
+}
+
+func (x *CallResponse) GetError() string {
+ if x != nil {
+ return x.Error
+ }
+ return ""
+}
+
+// go:plugin type=host version=1
+type SubsonicAPIService interface {
+ Call(context.Context, *CallRequest) (*CallResponse, error)
+}
diff --git a/plugins/host/subsonicapi/subsonicapi.proto b/plugins/host/subsonicapi/subsonicapi.proto
new file mode 100644
index 000000000..29dc365ca
--- /dev/null
+++ b/plugins/host/subsonicapi/subsonicapi.proto
@@ -0,0 +1,19 @@
+syntax = "proto3";
+
+package subsonicapi;
+
+option go_package = "github.com/navidrome/navidrome/plugins/host/subsonicapi;subsonicapi";
+
+// go:plugin type=host version=1
+service SubsonicAPIService {
+ rpc Call(CallRequest) returns (CallResponse);
+}
+
+message CallRequest {
+ string url = 1;
+}
+
+message CallResponse {
+ string json = 1;
+ string error = 2; // Non-empty if operation failed
+}
\ No newline at end of file
diff --git a/plugins/host/subsonicapi/subsonicapi_host.pb.go b/plugins/host/subsonicapi/subsonicapi_host.pb.go
new file mode 100644
index 000000000..b7c0f042e
--- /dev/null
+++ b/plugins/host/subsonicapi/subsonicapi_host.pb.go
@@ -0,0 +1,66 @@
+//go:build !wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/subsonicapi/subsonicapi.proto
+
+package subsonicapi
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ wazero "github.com/tetratelabs/wazero"
+ api "github.com/tetratelabs/wazero/api"
+)
+
+const (
+ i32 = api.ValueTypeI32
+ i64 = api.ValueTypeI64
+)
+
+type _subsonicAPIService struct {
+ SubsonicAPIService
+}
+
+// Instantiate a Go-defined module named "env" that exports host functions.
+func Instantiate(ctx context.Context, r wazero.Runtime, hostFunctions SubsonicAPIService) error {
+ envBuilder := r.NewHostModuleBuilder("env")
+ h := _subsonicAPIService{hostFunctions}
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._Call), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("call")
+
+ _, err := envBuilder.Instantiate(ctx)
+ return err
+}
+
+func (h _subsonicAPIService) _Call(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(CallRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.Call(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
diff --git a/plugins/host/subsonicapi/subsonicapi_plugin.pb.go b/plugins/host/subsonicapi/subsonicapi_plugin.pb.go
new file mode 100644
index 000000000..1ffdbf526
--- /dev/null
+++ b/plugins/host/subsonicapi/subsonicapi_plugin.pb.go
@@ -0,0 +1,44 @@
+//go:build wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/subsonicapi/subsonicapi.proto
+
+package subsonicapi
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ _ "unsafe"
+)
+
+type subsonicAPIService struct{}
+
+func NewSubsonicAPIService() SubsonicAPIService {
+ return subsonicAPIService{}
+}
+
+//go:wasmimport env call
+func _call(ptr uint32, size uint32) uint64
+
+func (h subsonicAPIService) Call(ctx context.Context, request *CallRequest) (*CallResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _call(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(CallResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
diff --git a/plugins/host/subsonicapi/subsonicapi_vtproto.pb.go b/plugins/host/subsonicapi/subsonicapi_vtproto.pb.go
new file mode 100644
index 000000000..05403216b
--- /dev/null
+++ b/plugins/host/subsonicapi/subsonicapi_vtproto.pb.go
@@ -0,0 +1,441 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/subsonicapi/subsonicapi.proto
+
+package subsonicapi
+
+import (
+ fmt "fmt"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+ io "io"
+ bits "math/bits"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+func (m *CallRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *CallRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *CallRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Url) > 0 {
+ i -= len(m.Url)
+ copy(dAtA[i:], m.Url)
+ i = encodeVarint(dAtA, i, uint64(len(m.Url)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *CallResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *CallResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *CallResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Error) > 0 {
+ i -= len(m.Error)
+ copy(dAtA[i:], m.Error)
+ i = encodeVarint(dAtA, i, uint64(len(m.Error)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.Json) > 0 {
+ i -= len(m.Json)
+ copy(dAtA[i:], m.Json)
+ i = encodeVarint(dAtA, i, uint64(len(m.Json)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func encodeVarint(dAtA []byte, offset int, v uint64) int {
+ offset -= sov(v)
+ base := offset
+ for v >= 1<<7 {
+ dAtA[offset] = uint8(v&0x7f | 0x80)
+ v >>= 7
+ offset++
+ }
+ dAtA[offset] = uint8(v)
+ return base
+}
+func (m *CallRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Url)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *CallResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Json)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Error)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func sov(x uint64) (n int) {
+ return (bits.Len64(x|1) + 6) / 7
+}
+func soz(x uint64) (n int) {
+ return sov(uint64((x << 1) ^ uint64((int64(x) >> 63))))
+}
+func (m *CallRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: CallRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: CallRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Url", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Url = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *CallResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: CallResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: CallResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Json", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Json = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Error", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Error = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+
+func skip(dAtA []byte) (n int, err error) {
+ l := len(dAtA)
+ iNdEx := 0
+ depth := 0
+ for iNdEx < l {
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= (uint64(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ wireType := int(wire & 0x7)
+ switch wireType {
+ case 0:
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ iNdEx++
+ if dAtA[iNdEx-1] < 0x80 {
+ break
+ }
+ }
+ case 1:
+ iNdEx += 8
+ case 2:
+ var length int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ length |= (int(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if length < 0 {
+ return 0, ErrInvalidLength
+ }
+ iNdEx += length
+ case 3:
+ depth++
+ case 4:
+ if depth == 0 {
+ return 0, ErrUnexpectedEndOfGroup
+ }
+ depth--
+ case 5:
+ iNdEx += 4
+ default:
+ return 0, fmt.Errorf("proto: illegal wireType %d", wireType)
+ }
+ if iNdEx < 0 {
+ return 0, ErrInvalidLength
+ }
+ if depth == 0 {
+ return iNdEx, nil
+ }
+ }
+ return 0, io.ErrUnexpectedEOF
+}
+
+var (
+ ErrInvalidLength = fmt.Errorf("proto: negative length found during unmarshaling")
+ ErrIntOverflow = fmt.Errorf("proto: integer overflow")
+ ErrUnexpectedEndOfGroup = fmt.Errorf("proto: unexpected end of group")
+)
diff --git a/plugins/host/websocket/websocket.pb.go b/plugins/host/websocket/websocket.pb.go
new file mode 100644
index 000000000..f3ab68963
--- /dev/null
+++ b/plugins/host/websocket/websocket.pb.go
@@ -0,0 +1,240 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/websocket/websocket.proto
+
+package websocket
+
+import (
+ context "context"
+ protoreflect "google.golang.org/protobuf/reflect/protoreflect"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+type ConnectRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Url string `protobuf:"bytes,1,opt,name=url,proto3" json:"url,omitempty"`
+ Headers map[string]string `protobuf:"bytes,2,rep,name=headers,proto3" json:"headers,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"`
+ ConnectionId string `protobuf:"bytes,3,opt,name=connection_id,json=connectionId,proto3" json:"connection_id,omitempty"`
+}
+
+func (x *ConnectRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ConnectRequest) GetUrl() string {
+ if x != nil {
+ return x.Url
+ }
+ return ""
+}
+
+func (x *ConnectRequest) GetHeaders() map[string]string {
+ if x != nil {
+ return x.Headers
+ }
+ return nil
+}
+
+func (x *ConnectRequest) GetConnectionId() string {
+ if x != nil {
+ return x.ConnectionId
+ }
+ return ""
+}
+
+type ConnectResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ ConnectionId string `protobuf:"bytes,1,opt,name=connection_id,json=connectionId,proto3" json:"connection_id,omitempty"`
+ Error string `protobuf:"bytes,2,opt,name=error,proto3" json:"error,omitempty"`
+}
+
+func (x *ConnectResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *ConnectResponse) GetConnectionId() string {
+ if x != nil {
+ return x.ConnectionId
+ }
+ return ""
+}
+
+func (x *ConnectResponse) GetError() string {
+ if x != nil {
+ return x.Error
+ }
+ return ""
+}
+
+type SendTextRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ ConnectionId string `protobuf:"bytes,1,opt,name=connection_id,json=connectionId,proto3" json:"connection_id,omitempty"`
+ Message string `protobuf:"bytes,2,opt,name=message,proto3" json:"message,omitempty"`
+}
+
+func (x *SendTextRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *SendTextRequest) GetConnectionId() string {
+ if x != nil {
+ return x.ConnectionId
+ }
+ return ""
+}
+
+func (x *SendTextRequest) GetMessage() string {
+ if x != nil {
+ return x.Message
+ }
+ return ""
+}
+
+type SendTextResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Error string `protobuf:"bytes,1,opt,name=error,proto3" json:"error,omitempty"`
+}
+
+func (x *SendTextResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *SendTextResponse) GetError() string {
+ if x != nil {
+ return x.Error
+ }
+ return ""
+}
+
+type SendBinaryRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ ConnectionId string `protobuf:"bytes,1,opt,name=connection_id,json=connectionId,proto3" json:"connection_id,omitempty"`
+ Data []byte `protobuf:"bytes,2,opt,name=data,proto3" json:"data,omitempty"`
+}
+
+func (x *SendBinaryRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *SendBinaryRequest) GetConnectionId() string {
+ if x != nil {
+ return x.ConnectionId
+ }
+ return ""
+}
+
+func (x *SendBinaryRequest) GetData() []byte {
+ if x != nil {
+ return x.Data
+ }
+ return nil
+}
+
+type SendBinaryResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Error string `protobuf:"bytes,1,opt,name=error,proto3" json:"error,omitempty"`
+}
+
+func (x *SendBinaryResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *SendBinaryResponse) GetError() string {
+ if x != nil {
+ return x.Error
+ }
+ return ""
+}
+
+type CloseRequest struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ ConnectionId string `protobuf:"bytes,1,opt,name=connection_id,json=connectionId,proto3" json:"connection_id,omitempty"`
+ Code int32 `protobuf:"varint,2,opt,name=code,proto3" json:"code,omitempty"`
+ Reason string `protobuf:"bytes,3,opt,name=reason,proto3" json:"reason,omitempty"`
+}
+
+func (x *CloseRequest) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *CloseRequest) GetConnectionId() string {
+ if x != nil {
+ return x.ConnectionId
+ }
+ return ""
+}
+
+func (x *CloseRequest) GetCode() int32 {
+ if x != nil {
+ return x.Code
+ }
+ return 0
+}
+
+func (x *CloseRequest) GetReason() string {
+ if x != nil {
+ return x.Reason
+ }
+ return ""
+}
+
+type CloseResponse struct {
+ state protoimpl.MessageState
+ sizeCache protoimpl.SizeCache
+ unknownFields protoimpl.UnknownFields
+
+ Error string `protobuf:"bytes,1,opt,name=error,proto3" json:"error,omitempty"`
+}
+
+func (x *CloseResponse) ProtoReflect() protoreflect.Message {
+ panic(`not implemented`)
+}
+
+func (x *CloseResponse) GetError() string {
+ if x != nil {
+ return x.Error
+ }
+ return ""
+}
+
+// go:plugin type=host version=1
+type WebSocketService interface {
+ // Connect to a WebSocket endpoint
+ Connect(context.Context, *ConnectRequest) (*ConnectResponse, error)
+ // Send a text message
+ SendText(context.Context, *SendTextRequest) (*SendTextResponse, error)
+ // Send binary data
+ SendBinary(context.Context, *SendBinaryRequest) (*SendBinaryResponse, error)
+ // Close a connection
+ Close(context.Context, *CloseRequest) (*CloseResponse, error)
+}
diff --git a/plugins/host/websocket/websocket.proto b/plugins/host/websocket/websocket.proto
new file mode 100644
index 000000000..53adaca95
--- /dev/null
+++ b/plugins/host/websocket/websocket.proto
@@ -0,0 +1,57 @@
+syntax = "proto3";
+package websocket;
+option go_package = "github.com/navidrome/navidrome/plugins/host/websocket";
+
+// go:plugin type=host version=1
+service WebSocketService {
+ // Connect to a WebSocket endpoint
+ rpc Connect(ConnectRequest) returns (ConnectResponse);
+
+ // Send a text message
+ rpc SendText(SendTextRequest) returns (SendTextResponse);
+
+ // Send binary data
+ rpc SendBinary(SendBinaryRequest) returns (SendBinaryResponse);
+
+ // Close a connection
+ rpc Close(CloseRequest) returns (CloseResponse);
+}
+
+message ConnectRequest {
+ string url = 1;
+ map headers = 2;
+ string connection_id = 3;
+}
+
+message ConnectResponse {
+ string connection_id = 1;
+ string error = 2;
+}
+
+message SendTextRequest {
+ string connection_id = 1;
+ string message = 2;
+}
+
+message SendTextResponse {
+ string error = 1;
+}
+
+message SendBinaryRequest {
+ string connection_id = 1;
+ bytes data = 2;
+}
+
+message SendBinaryResponse {
+ string error = 1;
+}
+
+message CloseRequest {
+ string connection_id = 1;
+ int32 code = 2;
+ string reason = 3;
+}
+
+message CloseResponse {
+ string error = 1;
+}
\ No newline at end of file
diff --git a/plugins/host/websocket/websocket_host.pb.go b/plugins/host/websocket/websocket_host.pb.go
new file mode 100644
index 000000000..b95eb451c
--- /dev/null
+++ b/plugins/host/websocket/websocket_host.pb.go
@@ -0,0 +1,170 @@
+//go:build !wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/websocket/websocket.proto
+
+package websocket
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ wazero "github.com/tetratelabs/wazero"
+ api "github.com/tetratelabs/wazero/api"
+)
+
+const (
+ i32 = api.ValueTypeI32
+ i64 = api.ValueTypeI64
+)
+
+type _webSocketService struct {
+ WebSocketService
+}
+
+// Instantiate a Go-defined module named "env" that exports host functions.
+func Instantiate(ctx context.Context, r wazero.Runtime, hostFunctions WebSocketService) error {
+ envBuilder := r.NewHostModuleBuilder("env")
+ h := _webSocketService{hostFunctions}
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._Connect), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("connect")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._SendText), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("send_text")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._SendBinary), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("send_binary")
+
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(api.GoModuleFunc(h._Close), []api.ValueType{i32, i32}, []api.ValueType{i64}).
+ WithParameterNames("offset", "size").
+ Export("close")
+
+ _, err := envBuilder.Instantiate(ctx)
+ return err
+}
+
+// Connect to a WebSocket endpoint
+
+func (h _webSocketService) _Connect(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(ConnectRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.Connect(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Send a text message
+
+func (h _webSocketService) _SendText(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(SendTextRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.SendText(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Send binary data
+
+func (h _webSocketService) _SendBinary(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(SendBinaryRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.SendBinary(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
+
+// Close a connection
+
+func (h _webSocketService) _Close(ctx context.Context, m api.Module, stack []uint64) {
+ offset, size := uint32(stack[0]), uint32(stack[1])
+ buf, err := wasm.ReadMemory(m.Memory(), offset, size)
+ if err != nil {
+ panic(err)
+ }
+ request := new(CloseRequest)
+ err = request.UnmarshalVT(buf)
+ if err != nil {
+ panic(err)
+ }
+ resp, err := h.Close(ctx, request)
+ if err != nil {
+ panic(err)
+ }
+ buf, err = resp.MarshalVT()
+ if err != nil {
+ panic(err)
+ }
+ ptr, err := wasm.WriteMemory(ctx, m, buf)
+ if err != nil {
+ panic(err)
+ }
+ ptrLen := (ptr << uint64(32)) | uint64(len(buf))
+ stack[0] = ptrLen
+}
diff --git a/plugins/host/websocket/websocket_plugin.pb.go b/plugins/host/websocket/websocket_plugin.pb.go
new file mode 100644
index 000000000..e7d5c3fe0
--- /dev/null
+++ b/plugins/host/websocket/websocket_plugin.pb.go
@@ -0,0 +1,113 @@
+//go:build wasip1
+
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/websocket/websocket.proto
+
+package websocket
+
+import (
+ context "context"
+ wasm "github.com/knqyf263/go-plugin/wasm"
+ _ "unsafe"
+)
+
+type webSocketService struct{}
+
+func NewWebSocketService() WebSocketService {
+ return webSocketService{}
+}
+
+//go:wasmimport env connect
+func _connect(ptr uint32, size uint32) uint64
+
+func (h webSocketService) Connect(ctx context.Context, request *ConnectRequest) (*ConnectResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _connect(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(ConnectResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env send_text
+func _send_text(ptr uint32, size uint32) uint64
+
+func (h webSocketService) SendText(ctx context.Context, request *SendTextRequest) (*SendTextResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _send_text(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(SendTextResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env send_binary
+func _send_binary(ptr uint32, size uint32) uint64
+
+func (h webSocketService) SendBinary(ctx context.Context, request *SendBinaryRequest) (*SendBinaryResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _send_binary(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(SendBinaryResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
+
+//go:wasmimport env close
+func _close(ptr uint32, size uint32) uint64
+
+func (h webSocketService) Close(ctx context.Context, request *CloseRequest) (*CloseResponse, error) {
+ buf, err := request.MarshalVT()
+ if err != nil {
+ return nil, err
+ }
+ ptr, size := wasm.ByteToPtr(buf)
+ ptrSize := _close(ptr, size)
+ wasm.Free(ptr)
+
+ ptr = uint32(ptrSize >> 32)
+ size = uint32(ptrSize)
+ buf = wasm.PtrToByte(ptr, size)
+
+ response := new(CloseResponse)
+ if err = response.UnmarshalVT(buf); err != nil {
+ return nil, err
+ }
+ return response, nil
+}
diff --git a/plugins/host/websocket/websocket_plugin_dev.go b/plugins/host/websocket/websocket_plugin_dev.go
new file mode 100644
index 000000000..cfb72462a
--- /dev/null
+++ b/plugins/host/websocket/websocket_plugin_dev.go
@@ -0,0 +1,7 @@
+//go:build !wasip1
+
+package websocket
+
+func NewWebSocketService() WebSocketService {
+ panic("not implemented")
+}
diff --git a/plugins/host/websocket/websocket_vtproto.pb.go b/plugins/host/websocket/websocket_vtproto.pb.go
new file mode 100644
index 000000000..fb15a22b7
--- /dev/null
+++ b/plugins/host/websocket/websocket_vtproto.pb.go
@@ -0,0 +1,1618 @@
+// Code generated by protoc-gen-go-plugin. DO NOT EDIT.
+// versions:
+// protoc-gen-go-plugin v0.1.0
+// protoc v5.29.3
+// source: host/websocket/websocket.proto
+
+package websocket
+
+import (
+ fmt "fmt"
+ protoimpl "google.golang.org/protobuf/runtime/protoimpl"
+ io "io"
+ bits "math/bits"
+)
+
+const (
+ // Verify that this generated code is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
+ // Verify that runtime/protoimpl is sufficiently up-to-date.
+ _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
+)
+
+func (m *ConnectRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ConnectRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ConnectRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.ConnectionId) > 0 {
+ i -= len(m.ConnectionId)
+ copy(dAtA[i:], m.ConnectionId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ConnectionId)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if len(m.Headers) > 0 {
+ for k := range m.Headers {
+ v := m.Headers[k]
+ baseI := i
+ i -= len(v)
+ copy(dAtA[i:], v)
+ i = encodeVarint(dAtA, i, uint64(len(v)))
+ i--
+ dAtA[i] = 0x12
+ i -= len(k)
+ copy(dAtA[i:], k)
+ i = encodeVarint(dAtA, i, uint64(len(k)))
+ i--
+ dAtA[i] = 0xa
+ i = encodeVarint(dAtA, i, uint64(baseI-i))
+ i--
+ dAtA[i] = 0x12
+ }
+ }
+ if len(m.Url) > 0 {
+ i -= len(m.Url)
+ copy(dAtA[i:], m.Url)
+ i = encodeVarint(dAtA, i, uint64(len(m.Url)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *ConnectResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *ConnectResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *ConnectResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Error) > 0 {
+ i -= len(m.Error)
+ copy(dAtA[i:], m.Error)
+ i = encodeVarint(dAtA, i, uint64(len(m.Error)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.ConnectionId) > 0 {
+ i -= len(m.ConnectionId)
+ copy(dAtA[i:], m.ConnectionId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ConnectionId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *SendTextRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *SendTextRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *SendTextRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Message) > 0 {
+ i -= len(m.Message)
+ copy(dAtA[i:], m.Message)
+ i = encodeVarint(dAtA, i, uint64(len(m.Message)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.ConnectionId) > 0 {
+ i -= len(m.ConnectionId)
+ copy(dAtA[i:], m.ConnectionId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ConnectionId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *SendTextResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *SendTextResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *SendTextResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Error) > 0 {
+ i -= len(m.Error)
+ copy(dAtA[i:], m.Error)
+ i = encodeVarint(dAtA, i, uint64(len(m.Error)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *SendBinaryRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *SendBinaryRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *SendBinaryRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Data) > 0 {
+ i -= len(m.Data)
+ copy(dAtA[i:], m.Data)
+ i = encodeVarint(dAtA, i, uint64(len(m.Data)))
+ i--
+ dAtA[i] = 0x12
+ }
+ if len(m.ConnectionId) > 0 {
+ i -= len(m.ConnectionId)
+ copy(dAtA[i:], m.ConnectionId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ConnectionId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *SendBinaryResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *SendBinaryResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *SendBinaryResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Error) > 0 {
+ i -= len(m.Error)
+ copy(dAtA[i:], m.Error)
+ i = encodeVarint(dAtA, i, uint64(len(m.Error)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *CloseRequest) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *CloseRequest) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *CloseRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Reason) > 0 {
+ i -= len(m.Reason)
+ copy(dAtA[i:], m.Reason)
+ i = encodeVarint(dAtA, i, uint64(len(m.Reason)))
+ i--
+ dAtA[i] = 0x1a
+ }
+ if m.Code != 0 {
+ i = encodeVarint(dAtA, i, uint64(m.Code))
+ i--
+ dAtA[i] = 0x10
+ }
+ if len(m.ConnectionId) > 0 {
+ i -= len(m.ConnectionId)
+ copy(dAtA[i:], m.ConnectionId)
+ i = encodeVarint(dAtA, i, uint64(len(m.ConnectionId)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func (m *CloseResponse) MarshalVT() (dAtA []byte, err error) {
+ if m == nil {
+ return nil, nil
+ }
+ size := m.SizeVT()
+ dAtA = make([]byte, size)
+ n, err := m.MarshalToSizedBufferVT(dAtA[:size])
+ if err != nil {
+ return nil, err
+ }
+ return dAtA[:n], nil
+}
+
+func (m *CloseResponse) MarshalToVT(dAtA []byte) (int, error) {
+ size := m.SizeVT()
+ return m.MarshalToSizedBufferVT(dAtA[:size])
+}
+
+func (m *CloseResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) {
+ if m == nil {
+ return 0, nil
+ }
+ i := len(dAtA)
+ _ = i
+ var l int
+ _ = l
+ if m.unknownFields != nil {
+ i -= len(m.unknownFields)
+ copy(dAtA[i:], m.unknownFields)
+ }
+ if len(m.Error) > 0 {
+ i -= len(m.Error)
+ copy(dAtA[i:], m.Error)
+ i = encodeVarint(dAtA, i, uint64(len(m.Error)))
+ i--
+ dAtA[i] = 0xa
+ }
+ return len(dAtA) - i, nil
+}
+
+func encodeVarint(dAtA []byte, offset int, v uint64) int {
+ offset -= sov(v)
+ base := offset
+ for v >= 1<<7 {
+ dAtA[offset] = uint8(v&0x7f | 0x80)
+ v >>= 7
+ offset++
+ }
+ dAtA[offset] = uint8(v)
+ return base
+}
+func (m *ConnectRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Url)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if len(m.Headers) > 0 {
+ for k, v := range m.Headers {
+ _ = k
+ _ = v
+ mapEntrySize := 1 + len(k) + sov(uint64(len(k))) + 1 + len(v) + sov(uint64(len(v)))
+ n += mapEntrySize + 1 + sov(uint64(mapEntrySize))
+ }
+ }
+ l = len(m.ConnectionId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *ConnectResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.ConnectionId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Error)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *SendTextRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.ConnectionId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Message)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *SendTextResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Error)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *SendBinaryRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.ConnectionId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ l = len(m.Data)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *SendBinaryResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Error)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *CloseRequest) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.ConnectionId)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ if m.Code != 0 {
+ n += 1 + sov(uint64(m.Code))
+ }
+ l = len(m.Reason)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func (m *CloseResponse) SizeVT() (n int) {
+ if m == nil {
+ return 0
+ }
+ var l int
+ _ = l
+ l = len(m.Error)
+ if l > 0 {
+ n += 1 + l + sov(uint64(l))
+ }
+ n += len(m.unknownFields)
+ return n
+}
+
+func sov(x uint64) (n int) {
+ return (bits.Len64(x|1) + 6) / 7
+}
+func soz(x uint64) (n int) {
+ return sov(uint64((x << 1) ^ uint64((int64(x) >> 63))))
+}
+func (m *ConnectRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ConnectRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ConnectRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Url", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Url = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Headers", wireType)
+ }
+ var msglen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ msglen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if msglen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + msglen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ if m.Headers == nil {
+ m.Headers = make(map[string]string)
+ }
+ var mapkey string
+ var mapvalue string
+ for iNdEx < postIndex {
+ entryPreIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ if fieldNum == 1 {
+ var stringLenmapkey uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLenmapkey |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLenmapkey := int(stringLenmapkey)
+ if intStringLenmapkey < 0 {
+ return ErrInvalidLength
+ }
+ postStringIndexmapkey := iNdEx + intStringLenmapkey
+ if postStringIndexmapkey < 0 {
+ return ErrInvalidLength
+ }
+ if postStringIndexmapkey > l {
+ return io.ErrUnexpectedEOF
+ }
+ mapkey = string(dAtA[iNdEx:postStringIndexmapkey])
+ iNdEx = postStringIndexmapkey
+ } else if fieldNum == 2 {
+ var stringLenmapvalue uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLenmapvalue |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLenmapvalue := int(stringLenmapvalue)
+ if intStringLenmapvalue < 0 {
+ return ErrInvalidLength
+ }
+ postStringIndexmapvalue := iNdEx + intStringLenmapvalue
+ if postStringIndexmapvalue < 0 {
+ return ErrInvalidLength
+ }
+ if postStringIndexmapvalue > l {
+ return io.ErrUnexpectedEOF
+ }
+ mapvalue = string(dAtA[iNdEx:postStringIndexmapvalue])
+ iNdEx = postStringIndexmapvalue
+ } else {
+ iNdEx = entryPreIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > postIndex {
+ return io.ErrUnexpectedEOF
+ }
+ iNdEx += skippy
+ }
+ }
+ m.Headers[mapkey] = mapvalue
+ iNdEx = postIndex
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ConnectionId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ConnectionId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *ConnectResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: ConnectResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: ConnectResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ConnectionId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ConnectionId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Error", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Error = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *SendTextRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: SendTextRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: SendTextRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ConnectionId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ConnectionId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Message", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Message = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *SendTextResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: SendTextResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: SendTextResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Error", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Error = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *SendBinaryRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: SendBinaryRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: SendBinaryRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ConnectionId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ConnectionId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Data", wireType)
+ }
+ var byteLen int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ byteLen |= int(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if byteLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + byteLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Data = append(m.Data[:0], dAtA[iNdEx:postIndex]...)
+ if m.Data == nil {
+ m.Data = []byte{}
+ }
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *SendBinaryResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: SendBinaryResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: SendBinaryResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Error", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Error = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *CloseRequest) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: CloseRequest: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: CloseRequest: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field ConnectionId", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.ConnectionId = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ case 2:
+ if wireType != 0 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Code", wireType)
+ }
+ m.Code = 0
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ m.Code |= int32(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ case 3:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Reason", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Reason = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+func (m *CloseResponse) UnmarshalVT(dAtA []byte) error {
+ l := len(dAtA)
+ iNdEx := 0
+ for iNdEx < l {
+ preIndex := iNdEx
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ fieldNum := int32(wire >> 3)
+ wireType := int(wire & 0x7)
+ if wireType == 4 {
+ return fmt.Errorf("proto: CloseResponse: wiretype end group for non-group")
+ }
+ if fieldNum <= 0 {
+ return fmt.Errorf("proto: CloseResponse: illegal tag %d (wire type %d)", fieldNum, wire)
+ }
+ switch fieldNum {
+ case 1:
+ if wireType != 2 {
+ return fmt.Errorf("proto: wrong wireType = %d for field Error", wireType)
+ }
+ var stringLen uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ stringLen |= uint64(b&0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ intStringLen := int(stringLen)
+ if intStringLen < 0 {
+ return ErrInvalidLength
+ }
+ postIndex := iNdEx + intStringLen
+ if postIndex < 0 {
+ return ErrInvalidLength
+ }
+ if postIndex > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.Error = string(dAtA[iNdEx:postIndex])
+ iNdEx = postIndex
+ default:
+ iNdEx = preIndex
+ skippy, err := skip(dAtA[iNdEx:])
+ if err != nil {
+ return err
+ }
+ if (skippy < 0) || (iNdEx+skippy) < 0 {
+ return ErrInvalidLength
+ }
+ if (iNdEx + skippy) > l {
+ return io.ErrUnexpectedEOF
+ }
+ m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)
+ iNdEx += skippy
+ }
+ }
+
+ if iNdEx > l {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
+
+func skip(dAtA []byte) (n int, err error) {
+ l := len(dAtA)
+ iNdEx := 0
+ depth := 0
+ for iNdEx < l {
+ var wire uint64
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ wire |= (uint64(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ wireType := int(wire & 0x7)
+ switch wireType {
+ case 0:
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ iNdEx++
+ if dAtA[iNdEx-1] < 0x80 {
+ break
+ }
+ }
+ case 1:
+ iNdEx += 8
+ case 2:
+ var length int
+ for shift := uint(0); ; shift += 7 {
+ if shift >= 64 {
+ return 0, ErrIntOverflow
+ }
+ if iNdEx >= l {
+ return 0, io.ErrUnexpectedEOF
+ }
+ b := dAtA[iNdEx]
+ iNdEx++
+ length |= (int(b) & 0x7F) << shift
+ if b < 0x80 {
+ break
+ }
+ }
+ if length < 0 {
+ return 0, ErrInvalidLength
+ }
+ iNdEx += length
+ case 3:
+ depth++
+ case 4:
+ if depth == 0 {
+ return 0, ErrUnexpectedEndOfGroup
+ }
+ depth--
+ case 5:
+ iNdEx += 4
+ default:
+ return 0, fmt.Errorf("proto: illegal wireType %d", wireType)
+ }
+ if iNdEx < 0 {
+ return 0, ErrInvalidLength
+ }
+ if depth == 0 {
+ return iNdEx, nil
+ }
+ }
+ return 0, io.ErrUnexpectedEOF
+}
+
+var (
+ ErrInvalidLength = fmt.Errorf("proto: negative length found during unmarshaling")
+ ErrIntOverflow = fmt.Errorf("proto: integer overflow")
+ ErrUnexpectedEndOfGroup = fmt.Errorf("proto: unexpected end of group")
+)
diff --git a/plugins/host_artwork.go b/plugins/host_artwork.go
new file mode 100644
index 000000000..dac622206
--- /dev/null
+++ b/plugins/host_artwork.go
@@ -0,0 +1,47 @@
+package plugins
+
+import (
+ "context"
+ "fmt"
+ "net/http"
+ "net/url"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/plugins/host/artwork"
+ "github.com/navidrome/navidrome/server/public"
+)
+
+type artworkServiceImpl struct{}
+
+func (a *artworkServiceImpl) GetArtistUrl(_ context.Context, req *artwork.GetArtworkUrlRequest) (*artwork.GetArtworkUrlResponse, error) {
+ artID := model.ArtworkID{Kind: model.KindArtistArtwork, ID: req.Id}
+ imageURL := public.ImageURL(a.createRequest(), artID, int(req.Size))
+ return &artwork.GetArtworkUrlResponse{Url: imageURL}, nil
+}
+
+func (a *artworkServiceImpl) GetAlbumUrl(_ context.Context, req *artwork.GetArtworkUrlRequest) (*artwork.GetArtworkUrlResponse, error) {
+ artID := model.ArtworkID{Kind: model.KindAlbumArtwork, ID: req.Id}
+ imageURL := public.ImageURL(a.createRequest(), artID, int(req.Size))
+ return &artwork.GetArtworkUrlResponse{Url: imageURL}, nil
+}
+
+func (a *artworkServiceImpl) GetTrackUrl(_ context.Context, req *artwork.GetArtworkUrlRequest) (*artwork.GetArtworkUrlResponse, error) {
+ artID := model.ArtworkID{Kind: model.KindMediaFileArtwork, ID: req.Id}
+ imageURL := public.ImageURL(a.createRequest(), artID, int(req.Size))
+ return &artwork.GetArtworkUrlResponse{Url: imageURL}, nil
+}
+
+func (a *artworkServiceImpl) createRequest() *http.Request {
+ var scheme, host string
+ if conf.Server.ShareURL != "" {
+ shareURL, _ := url.Parse(conf.Server.ShareURL)
+ scheme = shareURL.Scheme
+ host = shareURL.Host
+ } else {
+ scheme = "http"
+ host = "localhost"
+ }
+ r, _ := http.NewRequest("GET", fmt.Sprintf("%s://%s", scheme, host), nil)
+ return r
+}
diff --git a/plugins/host_artwork_test.go b/plugins/host_artwork_test.go
new file mode 100644
index 000000000..b6667bde3
--- /dev/null
+++ b/plugins/host_artwork_test.go
@@ -0,0 +1,58 @@
+package plugins
+
+import (
+ "context"
+
+ "github.com/go-chi/jwtauth/v5"
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/core/auth"
+ "github.com/navidrome/navidrome/plugins/host/artwork"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("ArtworkService", func() {
+ var svc *artworkServiceImpl
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ // Setup auth for tests
+ auth.TokenAuth = jwtauth.New("HS256", []byte("super secret"), nil)
+ svc = &artworkServiceImpl{}
+ })
+
+ Context("with ShareURL configured", func() {
+ BeforeEach(func() {
+ conf.Server.ShareURL = "https://music.example.com"
+ })
+
+ It("returns artist artwork URL", func() {
+ resp, err := svc.GetArtistUrl(context.Background(), &artwork.GetArtworkUrlRequest{Id: "123", Size: 300})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.Url).To(ContainSubstring("https://music.example.com"))
+ Expect(resp.Url).To(ContainSubstring("size=300"))
+ })
+
+ It("returns album artwork URL", func() {
+ resp, err := svc.GetAlbumUrl(context.Background(), &artwork.GetArtworkUrlRequest{Id: "456"})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.Url).To(ContainSubstring("https://music.example.com"))
+ })
+
+ It("returns track artwork URL", func() {
+ resp, err := svc.GetTrackUrl(context.Background(), &artwork.GetArtworkUrlRequest{Id: "789", Size: 150})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.Url).To(ContainSubstring("https://music.example.com"))
+ Expect(resp.Url).To(ContainSubstring("size=150"))
+ })
+ })
+
+ Context("without ShareURL configured", func() {
+ It("returns localhost URLs", func() {
+ resp, err := svc.GetArtistUrl(context.Background(), &artwork.GetArtworkUrlRequest{Id: "123"})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.Url).To(ContainSubstring("http://localhost"))
+ })
+ })
+})
diff --git a/plugins/host_cache.go b/plugins/host_cache.go
new file mode 100644
index 000000000..291a17870
--- /dev/null
+++ b/plugins/host_cache.go
@@ -0,0 +1,152 @@
+package plugins
+
+import (
+ "context"
+ "sync"
+ "time"
+
+ "github.com/jellydator/ttlcache/v3"
+ "github.com/navidrome/navidrome/log"
+ cacheproto "github.com/navidrome/navidrome/plugins/host/cache"
+)
+
+const (
+ defaultCacheTTL = 24 * time.Hour
+)
+
+// cacheServiceImpl implements the cache.CacheService interface
+type cacheServiceImpl struct {
+ pluginID string
+ defaultTTL time.Duration
+}
+
+var (
+ _cache *ttlcache.Cache[string, any]
+ initCacheOnce sync.Once
+)
+
+// newCacheService creates a new cacheServiceImpl instance
+func newCacheService(pluginID string) *cacheServiceImpl {
+ initCacheOnce.Do(func() {
+ opts := []ttlcache.Option[string, any]{
+ ttlcache.WithTTL[string, any](defaultCacheTTL),
+ }
+ _cache = ttlcache.New[string, any](opts...)
+
+ // Start the janitor goroutine to clean up expired entries
+ go _cache.Start()
+ })
+
+ return &cacheServiceImpl{
+ pluginID: pluginID,
+ defaultTTL: defaultCacheTTL,
+ }
+}
+
+// mapKey combines the plugin name and a provided key to create a unique cache key.
+func (s *cacheServiceImpl) mapKey(key string) string {
+ return s.pluginID + ":" + key
+}
+
+// getTTL converts seconds to a duration, using default if 0
+func (s *cacheServiceImpl) getTTL(seconds int64) time.Duration {
+ if seconds <= 0 {
+ return s.defaultTTL
+ }
+ return time.Duration(seconds) * time.Second
+}
+
+// setCacheValue is a generic function to set a value in the cache
+func setCacheValue[T any](ctx context.Context, cs *cacheServiceImpl, key string, value T, ttlSeconds int64) (*cacheproto.SetResponse, error) {
+ ttl := cs.getTTL(ttlSeconds)
+ key = cs.mapKey(key)
+ _cache.Set(key, value, ttl)
+ return &cacheproto.SetResponse{Success: true}, nil
+}
+
+// getCacheValue is a generic function to get a value from the cache
+func getCacheValue[T any](ctx context.Context, cs *cacheServiceImpl, key string, typeName string) (T, bool, error) {
+ key = cs.mapKey(key)
+ var zero T
+ item := _cache.Get(key)
+ if item == nil {
+ return zero, false, nil
+ }
+
+ value, ok := item.Value().(T)
+ if !ok {
+ log.Debug(ctx, "Type mismatch in cache", "plugin", cs.pluginID, "key", key, "expected", typeName)
+ return zero, false, nil
+ }
+ return value, true, nil
+}
+
+// SetString sets a string value in the cache
+func (s *cacheServiceImpl) SetString(ctx context.Context, req *cacheproto.SetStringRequest) (*cacheproto.SetResponse, error) {
+ return setCacheValue(ctx, s, req.Key, req.Value, req.TtlSeconds)
+}
+
+// GetString gets a string value from the cache
+func (s *cacheServiceImpl) GetString(ctx context.Context, req *cacheproto.GetRequest) (*cacheproto.GetStringResponse, error) {
+ value, exists, err := getCacheValue[string](ctx, s, req.Key, "string")
+ if err != nil {
+ return nil, err
+ }
+ return &cacheproto.GetStringResponse{Exists: exists, Value: value}, nil
+}
+
+// SetInt sets an integer value in the cache
+func (s *cacheServiceImpl) SetInt(ctx context.Context, req *cacheproto.SetIntRequest) (*cacheproto.SetResponse, error) {
+ return setCacheValue(ctx, s, req.Key, req.Value, req.TtlSeconds)
+}
+
+// GetInt gets an integer value from the cache
+func (s *cacheServiceImpl) GetInt(ctx context.Context, req *cacheproto.GetRequest) (*cacheproto.GetIntResponse, error) {
+ value, exists, err := getCacheValue[int64](ctx, s, req.Key, "int64")
+ if err != nil {
+ return nil, err
+ }
+ return &cacheproto.GetIntResponse{Exists: exists, Value: value}, nil
+}
+
+// SetFloat sets a float value in the cache
+func (s *cacheServiceImpl) SetFloat(ctx context.Context, req *cacheproto.SetFloatRequest) (*cacheproto.SetResponse, error) {
+ return setCacheValue(ctx, s, req.Key, req.Value, req.TtlSeconds)
+}
+
+// GetFloat gets a float value from the cache
+func (s *cacheServiceImpl) GetFloat(ctx context.Context, req *cacheproto.GetRequest) (*cacheproto.GetFloatResponse, error) {
+ value, exists, err := getCacheValue[float64](ctx, s, req.Key, "float64")
+ if err != nil {
+ return nil, err
+ }
+ return &cacheproto.GetFloatResponse{Exists: exists, Value: value}, nil
+}
+
+// SetBytes sets a byte slice value in the cache
+func (s *cacheServiceImpl) SetBytes(ctx context.Context, req *cacheproto.SetBytesRequest) (*cacheproto.SetResponse, error) {
+ return setCacheValue(ctx, s, req.Key, req.Value, req.TtlSeconds)
+}
+
+// GetBytes gets a byte slice value from the cache
+func (s *cacheServiceImpl) GetBytes(ctx context.Context, req *cacheproto.GetRequest) (*cacheproto.GetBytesResponse, error) {
+ value, exists, err := getCacheValue[[]byte](ctx, s, req.Key, "[]byte")
+ if err != nil {
+ return nil, err
+ }
+ return &cacheproto.GetBytesResponse{Exists: exists, Value: value}, nil
+}
+
+// Remove removes a value from the cache
+func (s *cacheServiceImpl) Remove(ctx context.Context, req *cacheproto.RemoveRequest) (*cacheproto.RemoveResponse, error) {
+ key := s.mapKey(req.Key)
+ _cache.Delete(key)
+ return &cacheproto.RemoveResponse{Success: true}, nil
+}
+
+// Has checks if a key exists in the cache
+func (s *cacheServiceImpl) Has(ctx context.Context, req *cacheproto.HasRequest) (*cacheproto.HasResponse, error) {
+ key := s.mapKey(req.Key)
+ item := _cache.Get(key)
+ return &cacheproto.HasResponse{Exists: item != nil}, nil
+}
diff --git a/plugins/host_cache_test.go b/plugins/host_cache_test.go
new file mode 100644
index 000000000..efb03e289
--- /dev/null
+++ b/plugins/host_cache_test.go
@@ -0,0 +1,171 @@
+package plugins
+
+import (
+ "context"
+ "time"
+
+ "github.com/navidrome/navidrome/plugins/host/cache"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("CacheService", func() {
+ var service *cacheServiceImpl
+ var ctx context.Context
+
+ BeforeEach(func() {
+ ctx = context.Background()
+ service = newCacheService("test_plugin")
+ })
+
+ Describe("getTTL", func() {
+ It("returns default TTL when seconds is 0", func() {
+ ttl := service.getTTL(0)
+ Expect(ttl).To(Equal(defaultCacheTTL))
+ })
+
+ It("returns default TTL when seconds is negative", func() {
+ ttl := service.getTTL(-10)
+ Expect(ttl).To(Equal(defaultCacheTTL))
+ })
+
+ It("returns correct duration when seconds is positive", func() {
+ ttl := service.getTTL(60)
+ Expect(ttl).To(Equal(time.Minute))
+ })
+ })
+
+ Describe("String Operations", func() {
+ It("sets and gets a string value", func() {
+ _, err := service.SetString(ctx, &cache.SetStringRequest{
+ Key: "string_key",
+ Value: "test_value",
+ TtlSeconds: 300,
+ })
+ Expect(err).NotTo(HaveOccurred())
+
+ res, err := service.GetString(ctx, &cache.GetRequest{Key: "string_key"})
+ Expect(err).NotTo(HaveOccurred())
+ Expect(res.Exists).To(BeTrue())
+ Expect(res.Value).To(Equal("test_value"))
+ })
+
+ It("returns not exists for missing key", func() {
+ res, err := service.GetString(ctx, &cache.GetRequest{Key: "missing_key"})
+ Expect(err).NotTo(HaveOccurred())
+ Expect(res.Exists).To(BeFalse())
+ })
+ })
+
+ Describe("Integer Operations", func() {
+ It("sets and gets an integer value", func() {
+ _, err := service.SetInt(ctx, &cache.SetIntRequest{
+ Key: "int_key",
+ Value: 42,
+ TtlSeconds: 300,
+ })
+ Expect(err).NotTo(HaveOccurred())
+
+ res, err := service.GetInt(ctx, &cache.GetRequest{Key: "int_key"})
+ Expect(err).NotTo(HaveOccurred())
+ Expect(res.Exists).To(BeTrue())
+ Expect(res.Value).To(Equal(int64(42)))
+ })
+ })
+
+ Describe("Float Operations", func() {
+ It("sets and gets a float value", func() {
+ _, err := service.SetFloat(ctx, &cache.SetFloatRequest{
+ Key: "float_key",
+ Value: 3.14,
+ TtlSeconds: 300,
+ })
+ Expect(err).NotTo(HaveOccurred())
+
+ res, err := service.GetFloat(ctx, &cache.GetRequest{Key: "float_key"})
+ Expect(err).NotTo(HaveOccurred())
+ Expect(res.Exists).To(BeTrue())
+ Expect(res.Value).To(Equal(3.14))
+ })
+ })
+
+ Describe("Bytes Operations", func() {
+ It("sets and gets a bytes value", func() {
+ byteData := []byte("hello world")
+ _, err := service.SetBytes(ctx, &cache.SetBytesRequest{
+ Key: "bytes_key",
+ Value: byteData,
+ TtlSeconds: 300,
+ })
+ Expect(err).NotTo(HaveOccurred())
+
+ res, err := service.GetBytes(ctx, &cache.GetRequest{Key: "bytes_key"})
+ Expect(err).NotTo(HaveOccurred())
+ Expect(res.Exists).To(BeTrue())
+ Expect(res.Value).To(Equal(byteData))
+ })
+ })
+
+ Describe("Type mismatch handling", func() {
+ It("returns not exists when type doesn't match the getter", func() {
+ // Set string
+ _, err := service.SetString(ctx, &cache.SetStringRequest{
+ Key: "mixed_key",
+ Value: "string value",
+ })
+ Expect(err).NotTo(HaveOccurred())
+
+ // Try to get as int
+ res, err := service.GetInt(ctx, &cache.GetRequest{Key: "mixed_key"})
+ Expect(err).NotTo(HaveOccurred())
+ Expect(res.Exists).To(BeFalse())
+ })
+ })
+
+ Describe("Remove Operation", func() {
+ It("removes a value from the cache", func() {
+ // Set a value
+ _, err := service.SetString(ctx, &cache.SetStringRequest{
+ Key: "remove_key",
+ Value: "to be removed",
+ })
+ Expect(err).NotTo(HaveOccurred())
+
+ // Verify it exists
+ res, err := service.Has(ctx, &cache.HasRequest{Key: "remove_key"})
+ Expect(err).NotTo(HaveOccurred())
+ Expect(res.Exists).To(BeTrue())
+
+ // Remove it
+ _, err = service.Remove(ctx, &cache.RemoveRequest{Key: "remove_key"})
+ Expect(err).NotTo(HaveOccurred())
+
+ // Verify it's gone
+ res, err = service.Has(ctx, &cache.HasRequest{Key: "remove_key"})
+ Expect(err).NotTo(HaveOccurred())
+ Expect(res.Exists).To(BeFalse())
+ })
+ })
+
+ Describe("Has Operation", func() {
+ It("returns true for existing key", func() {
+ // Set a value
+ _, err := service.SetString(ctx, &cache.SetStringRequest{
+ Key: "existing_key",
+ Value: "exists",
+ })
+ Expect(err).NotTo(HaveOccurred())
+
+ // Check if it exists
+ res, err := service.Has(ctx, &cache.HasRequest{Key: "existing_key"})
+ Expect(err).NotTo(HaveOccurred())
+ Expect(res.Exists).To(BeTrue())
+ })
+
+ It("returns false for non-existing key", func() {
+ res, err := service.Has(ctx, &cache.HasRequest{Key: "non_existing_key"})
+ Expect(err).NotTo(HaveOccurred())
+ Expect(res.Exists).To(BeFalse())
+ })
+ })
+})
diff --git a/plugins/host_config.go b/plugins/host_config.go
new file mode 100644
index 000000000..baee6a00c
--- /dev/null
+++ b/plugins/host_config.go
@@ -0,0 +1,22 @@
+package plugins
+
+import (
+ "context"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/plugins/host/config"
+)
+
+type configServiceImpl struct {
+ pluginID string
+}
+
+func (c *configServiceImpl) GetPluginConfig(ctx context.Context, req *config.GetPluginConfigRequest) (*config.GetPluginConfigResponse, error) {
+ cfg, ok := conf.Server.PluginConfig[c.pluginID]
+ if !ok {
+ cfg = map[string]string{}
+ }
+ return &config.GetPluginConfigResponse{
+ Config: cfg,
+ }, nil
+}
diff --git a/plugins/host_config_test.go b/plugins/host_config_test.go
new file mode 100644
index 000000000..bae7043be
--- /dev/null
+++ b/plugins/host_config_test.go
@@ -0,0 +1,46 @@
+package plugins
+
+import (
+ "context"
+
+ "github.com/navidrome/navidrome/conf"
+ hostconfig "github.com/navidrome/navidrome/plugins/host/config"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("configServiceImpl", func() {
+ var (
+ svc *configServiceImpl
+ pluginName string
+ )
+
+ BeforeEach(func() {
+ pluginName = "testplugin"
+ svc = &configServiceImpl{pluginID: pluginName}
+ conf.Server.PluginConfig = map[string]map[string]string{
+ pluginName: {"foo": "bar", "baz": "qux"},
+ }
+ })
+
+ It("returns config for known plugin", func() {
+ resp, err := svc.GetPluginConfig(context.Background(), &hostconfig.GetPluginConfigRequest{})
+ Expect(err).To(BeNil())
+ Expect(resp.Config).To(HaveKeyWithValue("foo", "bar"))
+ Expect(resp.Config).To(HaveKeyWithValue("baz", "qux"))
+ })
+
+ It("returns error for unknown plugin", func() {
+ svc.pluginID = "unknown"
+ resp, err := svc.GetPluginConfig(context.Background(), &hostconfig.GetPluginConfigRequest{})
+ Expect(err).To(BeNil())
+ Expect(resp.Config).To(BeEmpty())
+ })
+
+ It("returns empty config if plugin config is empty", func() {
+ conf.Server.PluginConfig[pluginName] = map[string]string{}
+ resp, err := svc.GetPluginConfig(context.Background(), &hostconfig.GetPluginConfigRequest{})
+ Expect(err).To(BeNil())
+ Expect(resp.Config).To(BeEmpty())
+ })
+})
diff --git a/plugins/host_http.go b/plugins/host_http.go
new file mode 100644
index 000000000..24fc77b18
--- /dev/null
+++ b/plugins/host_http.go
@@ -0,0 +1,114 @@
+package plugins
+
+import (
+ "bytes"
+ "cmp"
+ "context"
+ "io"
+ "net/http"
+ "time"
+
+ "github.com/navidrome/navidrome/log"
+ hosthttp "github.com/navidrome/navidrome/plugins/host/http"
+)
+
+type httpServiceImpl struct {
+ pluginID string
+ permissions *httpPermissions
+}
+
+const defaultTimeout = 10 * time.Second
+
+func (s *httpServiceImpl) Get(ctx context.Context, req *hosthttp.HttpRequest) (*hosthttp.HttpResponse, error) {
+ return s.doHttp(ctx, http.MethodGet, req)
+}
+
+func (s *httpServiceImpl) Post(ctx context.Context, req *hosthttp.HttpRequest) (*hosthttp.HttpResponse, error) {
+ return s.doHttp(ctx, http.MethodPost, req)
+}
+
+func (s *httpServiceImpl) Put(ctx context.Context, req *hosthttp.HttpRequest) (*hosthttp.HttpResponse, error) {
+ return s.doHttp(ctx, http.MethodPut, req)
+}
+
+func (s *httpServiceImpl) Delete(ctx context.Context, req *hosthttp.HttpRequest) (*hosthttp.HttpResponse, error) {
+ return s.doHttp(ctx, http.MethodDelete, req)
+}
+
+func (s *httpServiceImpl) Patch(ctx context.Context, req *hosthttp.HttpRequest) (*hosthttp.HttpResponse, error) {
+ return s.doHttp(ctx, http.MethodPatch, req)
+}
+
+func (s *httpServiceImpl) Head(ctx context.Context, req *hosthttp.HttpRequest) (*hosthttp.HttpResponse, error) {
+ return s.doHttp(ctx, http.MethodHead, req)
+}
+
+func (s *httpServiceImpl) Options(ctx context.Context, req *hosthttp.HttpRequest) (*hosthttp.HttpResponse, error) {
+ return s.doHttp(ctx, http.MethodOptions, req)
+}
+
+func (s *httpServiceImpl) doHttp(ctx context.Context, method string, req *hosthttp.HttpRequest) (*hosthttp.HttpResponse, error) {
+ // Check permissions if they exist
+ if s.permissions != nil {
+ if err := s.permissions.IsRequestAllowed(req.Url, method); err != nil {
+ log.Warn(ctx, "HTTP request blocked by permissions", "plugin", s.pluginID, "url", req.Url, "method", method, err)
+ return &hosthttp.HttpResponse{Error: "Request blocked by plugin permissions: " + err.Error()}, nil
+ }
+ }
+ client := &http.Client{
+ Timeout: cmp.Or(time.Duration(req.TimeoutMs)*time.Millisecond, defaultTimeout),
+ }
+
+ // Configure redirect policy based on permissions
+ if s.permissions != nil {
+ client.CheckRedirect = func(req *http.Request, via []*http.Request) error {
+ // Enforce maximum redirect limit
+ if len(via) >= httpMaxRedirects {
+ log.Warn(ctx, "HTTP redirect limit exceeded", "plugin", s.pluginID, "url", req.URL.String(), "redirectCount", len(via))
+ return http.ErrUseLastResponse
+ }
+
+ // Check if redirect destination is allowed
+ if err := s.permissions.IsRequestAllowed(req.URL.String(), req.Method); err != nil {
+ log.Warn(ctx, "HTTP redirect blocked by permissions", "plugin", s.pluginID, "url", req.URL.String(), "method", req.Method, err)
+ return http.ErrUseLastResponse
+ }
+
+ return nil // Allow redirect
+ }
+ }
+ var body io.Reader
+ if method == http.MethodPost || method == http.MethodPut || method == http.MethodPatch {
+ body = bytes.NewReader(req.Body)
+ }
+ httpReq, err := http.NewRequestWithContext(ctx, method, req.Url, body)
+ if err != nil {
+ return nil, err
+ }
+ for k, v := range req.Headers {
+ httpReq.Header.Set(k, v)
+ }
+ resp, err := client.Do(httpReq)
+ if err != nil {
+ log.Trace(ctx, "HttpService request error", "method", method, "url", req.Url, "headers", req.Headers, err)
+ return &hosthttp.HttpResponse{Error: err.Error()}, nil
+ }
+ log.Trace(ctx, "HttpService request", "method", method, "url", req.Url, "headers", req.Headers, "resp.status", resp.StatusCode)
+ defer resp.Body.Close()
+ respBody, err := io.ReadAll(resp.Body)
+ if err != nil {
+ log.Trace(ctx, "HttpService request error", "method", method, "url", req.Url, "headers", req.Headers, "resp.status", resp.StatusCode, err)
+ return &hosthttp.HttpResponse{Error: err.Error()}, nil
+ }
+ headers := map[string]string{}
+ for k, v := range resp.Header {
+ if len(v) > 0 {
+ headers[k] = v[0]
+ }
+ }
+ return &hosthttp.HttpResponse{
+ Status: int32(resp.StatusCode),
+ Body: respBody,
+ Headers: headers,
+ }, nil
+}
diff --git a/plugins/host_http_permissions.go b/plugins/host_http_permissions.go
new file mode 100644
index 000000000..158bdb105
--- /dev/null
+++ b/plugins/host_http_permissions.go
@@ -0,0 +1,90 @@
+package plugins
+
+import (
+ "fmt"
+ "strings"
+
+ "github.com/navidrome/navidrome/plugins/schema"
+)
+
+// Maximum number of HTTP redirects allowed for plugin requests
+const httpMaxRedirects = 5
+
+// HTTPPermissions represents granular HTTP access permissions for plugins
+type httpPermissions struct {
+ *networkPermissionsBase
+ AllowedUrls map[string][]string `json:"allowedUrls"`
+ matcher *urlMatcher
+}
+
+// parseHTTPPermissions extracts HTTP permissions from the schema
+func parseHTTPPermissions(permData *schema.PluginManifestPermissionsHttp) (*httpPermissions, error) {
+ base := &networkPermissionsBase{
+ AllowLocalNetwork: permData.AllowLocalNetwork,
+ }
+
+ if len(permData.AllowedUrls) == 0 {
+ return nil, fmt.Errorf("allowedUrls must contain at least one URL pattern")
+ }
+
+ allowedUrls := make(map[string][]string)
+ for urlPattern, methodEnums := range permData.AllowedUrls {
+ methods := make([]string, len(methodEnums))
+ for i, methodEnum := range methodEnums {
+ methods[i] = string(methodEnum)
+ }
+ allowedUrls[urlPattern] = methods
+ }
+
+ return &httpPermissions{
+ networkPermissionsBase: base,
+ AllowedUrls: allowedUrls,
+ matcher: newURLMatcher(),
+ }, nil
+}
+
+// IsRequestAllowed checks if a specific network request is allowed by the permissions
+func (p *httpPermissions) IsRequestAllowed(requestURL, operation string) error {
+ if _, err := checkURLPolicy(requestURL, p.AllowLocalNetwork); err != nil {
+ return err
+ }
+
+ // allowedUrls is now required - no fallback to allow all URLs
+ if p.AllowedUrls == nil || len(p.AllowedUrls) == 0 {
+ return fmt.Errorf("no allowed URLs configured for plugin")
+ }
+
+ matcher := newURLMatcher()
+
+ // Check URL patterns and operations
+ // First try exact matches, then wildcard matches
+ operation = strings.ToUpper(operation)
+
+ // Phase 1: Check for exact matches first
+ for urlPattern, allowedOperations := range p.AllowedUrls {
+ if !strings.Contains(urlPattern, "*") && matcher.MatchesURLPattern(requestURL, urlPattern) {
+ // Check if operation is allowed
+ for _, allowedOperation := range allowedOperations {
+ if allowedOperation == "*" || allowedOperation == operation {
+ return nil
+ }
+ }
+ return fmt.Errorf("operation %s not allowed for URL pattern %s", operation, urlPattern)
+ }
+ }
+
+ // Phase 2: Check wildcard patterns
+ for urlPattern, allowedOperations := range p.AllowedUrls {
+ if strings.Contains(urlPattern, "*") && matcher.MatchesURLPattern(requestURL, urlPattern) {
+ // Check if operation is allowed
+ for _, allowedOperation := range allowedOperations {
+ if allowedOperation == "*" || allowedOperation == operation {
+ return nil
+ }
+ }
+ return fmt.Errorf("operation %s not allowed for URL pattern %s", operation, urlPattern)
+ }
+ }
+
+ return fmt.Errorf("URL %s does not match any allowed URL patterns", requestURL)
+}
diff --git a/plugins/host_http_permissions_test.go b/plugins/host_http_permissions_test.go
new file mode 100644
index 000000000..3385ffc03
--- /dev/null
+++ b/plugins/host_http_permissions_test.go
@@ -0,0 +1,187 @@
+package plugins
+
+import (
+ "github.com/navidrome/navidrome/plugins/schema"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("HTTP Permissions", func() {
+ Describe("parseHTTPPermissions", func() {
+ It("should parse valid HTTP permissions", func() {
+ permData := &schema.PluginManifestPermissionsHttp{
+ Reason: "Need to fetch album artwork",
+ AllowLocalNetwork: false,
+ AllowedUrls: map[string][]schema.PluginManifestPermissionsHttpAllowedUrlsValueElem{
+ "https://api.example.com/*": {
+ schema.PluginManifestPermissionsHttpAllowedUrlsValueElemGET,
+ schema.PluginManifestPermissionsHttpAllowedUrlsValueElemPOST,
+ },
+ "https://cdn.example.com/*": {
+ schema.PluginManifestPermissionsHttpAllowedUrlsValueElemGET,
+ },
+ },
+ }
+
+ perms, err := parseHTTPPermissions(permData)
+ Expect(err).To(BeNil())
+ Expect(perms).ToNot(BeNil())
+ Expect(perms.AllowLocalNetwork).To(BeFalse())
+ Expect(perms.AllowedUrls).To(HaveLen(2))
+ Expect(perms.AllowedUrls["https://api.example.com/*"]).To(Equal([]string{"GET", "POST"}))
+ Expect(perms.AllowedUrls["https://cdn.example.com/*"]).To(Equal([]string{"GET"}))
+ })
+
+ It("should fail if allowedUrls is empty", func() {
+ permData := &schema.PluginManifestPermissionsHttp{
+ Reason: "Need to fetch album artwork",
+ AllowLocalNetwork: false,
+ AllowedUrls: map[string][]schema.PluginManifestPermissionsHttpAllowedUrlsValueElem{},
+ }
+
+ _, err := parseHTTPPermissions(permData)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("allowedUrls must contain at least one URL pattern"))
+ })
+
+ It("should handle method enum types correctly", func() {
+ permData := &schema.PluginManifestPermissionsHttp{
+ Reason: "Need to fetch album artwork",
+ AllowLocalNetwork: false,
+ AllowedUrls: map[string][]schema.PluginManifestPermissionsHttpAllowedUrlsValueElem{
+ "https://api.example.com/*": {
+ schema.PluginManifestPermissionsHttpAllowedUrlsValueElemWildcard, // "*"
+ },
+ },
+ }
+
+ perms, err := parseHTTPPermissions(permData)
+ Expect(err).To(BeNil())
+ Expect(perms.AllowedUrls["https://api.example.com/*"]).To(Equal([]string{"*"}))
+ })
+ })
+
+ Describe("IsRequestAllowed", func() {
+ var perms *httpPermissions
+
+ Context("HTTP method-specific validation", func() {
+ BeforeEach(func() {
+ perms = &httpPermissions{
+ networkPermissionsBase: &networkPermissionsBase{
+ Reason: "Test permissions",
+ AllowLocalNetwork: false,
+ },
+ AllowedUrls: map[string][]string{
+ "https://api.example.com": {"GET", "POST"},
+ "https://upload.example.com": {"PUT", "PATCH"},
+ "https://admin.example.com": {"DELETE"},
+ "https://webhook.example.com": {"*"},
+ },
+ matcher: newURLMatcher(),
+ }
+ })
+
+ DescribeTable("method-specific access control",
+ func(url, method string, shouldSucceed bool) {
+ err := perms.IsRequestAllowed(url, method)
+ if shouldSucceed {
+ Expect(err).ToNot(HaveOccurred())
+ } else {
+ Expect(err).To(HaveOccurred())
+ }
+ },
+ // Allowed methods
+ Entry("GET to api", "https://api.example.com", "GET", true),
+ Entry("POST to api", "https://api.example.com", "POST", true),
+ Entry("PUT to upload", "https://upload.example.com", "PUT", true),
+ Entry("PATCH to upload", "https://upload.example.com", "PATCH", true),
+ Entry("DELETE to admin", "https://admin.example.com", "DELETE", true),
+ Entry("any method to webhook", "https://webhook.example.com", "OPTIONS", true),
+ Entry("any method to webhook", "https://webhook.example.com", "HEAD", true),
+
+ // Disallowed methods
+ Entry("DELETE to api", "https://api.example.com", "DELETE", false),
+ Entry("GET to upload", "https://upload.example.com", "GET", false),
+ Entry("POST to admin", "https://admin.example.com", "POST", false),
+ )
+ })
+
+ Context("case insensitive method handling", func() {
+ BeforeEach(func() {
+ perms = &httpPermissions{
+ networkPermissionsBase: &networkPermissionsBase{
+ Reason: "Test permissions",
+ AllowLocalNetwork: false,
+ },
+ AllowedUrls: map[string][]string{
+ "https://api.example.com": {"GET", "POST"}, // Both uppercase for consistency
+ },
+ matcher: newURLMatcher(),
+ }
+ })
+
+ DescribeTable("case insensitive method matching",
+ func(method string, shouldSucceed bool) {
+ err := perms.IsRequestAllowed("https://api.example.com", method)
+ if shouldSucceed {
+ Expect(err).ToNot(HaveOccurred())
+ } else {
+ Expect(err).To(HaveOccurred())
+ }
+ },
+ Entry("uppercase GET", "GET", true),
+ Entry("lowercase get", "get", true),
+ Entry("mixed case Get", "Get", true),
+ Entry("uppercase POST", "POST", true),
+ Entry("lowercase post", "post", true),
+ Entry("mixed case Post", "Post", true),
+ Entry("disallowed method", "DELETE", false),
+ )
+ })
+
+ Context("with complex URL patterns and HTTP methods", func() {
+ BeforeEach(func() {
+ perms = &httpPermissions{
+ networkPermissionsBase: &networkPermissionsBase{
+ Reason: "Test permissions",
+ AllowLocalNetwork: false,
+ },
+ AllowedUrls: map[string][]string{
+ "https://api.example.com/v1/*": {"GET"},
+ "https://api.example.com/v1/users": {"POST", "PUT"},
+ "https://*.example.com/public/*": {"GET", "HEAD"},
+ "https://admin.*.example.com": {"*"},
+ },
+ matcher: newURLMatcher(),
+ }
+ })
+
+ DescribeTable("complex pattern and method combinations",
+ func(url, method string, shouldSucceed bool) {
+ err := perms.IsRequestAllowed(url, method)
+ if shouldSucceed {
+ Expect(err).ToNot(HaveOccurred())
+ } else {
+ Expect(err).To(HaveOccurred())
+ }
+ },
+ // Path wildcards with specific methods
+ Entry("GET to v1 path", "https://api.example.com/v1/posts", "GET", true),
+ Entry("POST to v1 path", "https://api.example.com/v1/posts", "POST", false),
+ Entry("POST to specific users endpoint", "https://api.example.com/v1/users", "POST", true),
+ Entry("PUT to specific users endpoint", "https://api.example.com/v1/users", "PUT", true),
+ Entry("DELETE to specific users endpoint", "https://api.example.com/v1/users", "DELETE", false),
+
+ // Subdomain wildcards with specific methods
+ Entry("GET to public path on subdomain", "https://cdn.example.com/public/assets", "GET", true),
+ Entry("HEAD to public path on subdomain", "https://static.example.com/public/files", "HEAD", true),
+ Entry("POST to public path on subdomain", "https://api.example.com/public/upload", "POST", false),
+
+ // Admin subdomain with all methods
+ Entry("GET to admin subdomain", "https://admin.prod.example.com", "GET", true),
+ Entry("POST to admin subdomain", "https://admin.staging.example.com", "POST", true),
+ Entry("DELETE to admin subdomain", "https://admin.dev.example.com", "DELETE", true),
+ )
+ })
+ })
+})
diff --git a/plugins/host_http_test.go b/plugins/host_http_test.go
new file mode 100644
index 000000000..b6f823a07
--- /dev/null
+++ b/plugins/host_http_test.go
@@ -0,0 +1,190 @@
+package plugins
+
+import (
+ "context"
+ "net/http"
+ "net/http/httptest"
+ "time"
+
+ hosthttp "github.com/navidrome/navidrome/plugins/host/http"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("httpServiceImpl", func() {
+ var (
+ svc *httpServiceImpl
+ ts *httptest.Server
+ )
+
+ BeforeEach(func() {
+ svc = &httpServiceImpl{}
+ })
+
+ AfterEach(func() {
+ if ts != nil {
+ ts.Close()
+ }
+ })
+
+ It("should handle GET requests", func() {
+ ts = httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ w.Header().Set("X-Test", "ok")
+ w.WriteHeader(201)
+ _, _ = w.Write([]byte("hello"))
+ }))
+ resp, err := svc.Get(context.Background(), &hosthttp.HttpRequest{
+ Url: ts.URL,
+ Headers: map[string]string{"A": "B"},
+ TimeoutMs: 1000,
+ })
+ Expect(err).To(BeNil())
+ Expect(resp.Error).To(BeEmpty())
+ Expect(resp.Status).To(Equal(int32(201)))
+ Expect(string(resp.Body)).To(Equal("hello"))
+ Expect(resp.Headers["X-Test"]).To(Equal("ok"))
+ })
+
+ It("should handle POST requests with body", func() {
+ ts = httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ b := make([]byte, r.ContentLength)
+ _, _ = r.Body.Read(b)
+ _, _ = w.Write([]byte("got:" + string(b)))
+ }))
+ resp, err := svc.Post(context.Background(), &hosthttp.HttpRequest{
+ Url: ts.URL,
+ Body: []byte("abc"),
+ TimeoutMs: 1000,
+ })
+ Expect(err).To(BeNil())
+ Expect(resp.Error).To(BeEmpty())
+ Expect(string(resp.Body)).To(Equal("got:abc"))
+ })
+
+ It("should handle PUT requests with body", func() {
+ ts = httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ b := make([]byte, r.ContentLength)
+ _, _ = r.Body.Read(b)
+ _, _ = w.Write([]byte("put:" + string(b)))
+ }))
+ resp, err := svc.Put(context.Background(), &hosthttp.HttpRequest{
+ Url: ts.URL,
+ Body: []byte("xyz"),
+ TimeoutMs: 1000,
+ })
+ Expect(err).To(BeNil())
+ Expect(resp.Error).To(BeEmpty())
+ Expect(string(resp.Body)).To(Equal("put:xyz"))
+ })
+
+ It("should handle DELETE requests", func() {
+ ts = httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ w.WriteHeader(204)
+ }))
+ resp, err := svc.Delete(context.Background(), &hosthttp.HttpRequest{
+ Url: ts.URL,
+ TimeoutMs: 1000,
+ })
+ Expect(err).To(BeNil())
+ Expect(resp.Error).To(BeEmpty())
+ Expect(resp.Status).To(Equal(int32(204)))
+ })
+
+ It("should handle PATCH requests with body", func() {
+ ts = httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ b := make([]byte, r.ContentLength)
+ _, _ = r.Body.Read(b)
+ _, _ = w.Write([]byte("patch:" + string(b)))
+ }))
+ resp, err := svc.Patch(context.Background(), &hosthttp.HttpRequest{
+ Url: ts.URL,
+ Body: []byte("test-patch"),
+ TimeoutMs: 1000,
+ })
+ Expect(err).To(BeNil())
+ Expect(resp.Error).To(BeEmpty())
+ Expect(string(resp.Body)).To(Equal("patch:test-patch"))
+ })
+
+ It("should handle HEAD requests", func() {
+ ts = httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ w.Header().Set("Content-Type", "application/json")
+ w.Header().Set("Content-Length", "42")
+ w.WriteHeader(200)
+ // HEAD responses shouldn't have a body, but the headers should be present
+ }))
+ resp, err := svc.Head(context.Background(), &hosthttp.HttpRequest{
+ Url: ts.URL,
+ TimeoutMs: 1000,
+ })
+ Expect(err).To(BeNil())
+ Expect(resp.Error).To(BeEmpty())
+ Expect(resp.Status).To(Equal(int32(200)))
+ Expect(resp.Headers["Content-Type"]).To(Equal("application/json"))
+ Expect(resp.Headers["Content-Length"]).To(Equal("42"))
+ Expect(resp.Body).To(BeEmpty()) // HEAD responses have no body
+ })
+
+ It("should handle OPTIONS requests", func() {
+ ts = httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ w.Header().Set("Allow", "GET, POST, PUT, DELETE, PATCH, HEAD, OPTIONS")
+ w.Header().Set("Access-Control-Allow-Methods", "GET, POST, PUT, DELETE, PATCH, HEAD, OPTIONS")
+ w.WriteHeader(200)
+ }))
+ resp, err := svc.Options(context.Background(), &hosthttp.HttpRequest{
+ Url: ts.URL,
+ TimeoutMs: 1000,
+ })
+ Expect(err).To(BeNil())
+ Expect(resp.Error).To(BeEmpty())
+ Expect(resp.Status).To(Equal(int32(200)))
+ Expect(resp.Headers["Allow"]).To(Equal("GET, POST, PUT, DELETE, PATCH, HEAD, OPTIONS"))
+ Expect(resp.Headers["Access-Control-Allow-Methods"]).To(Equal("GET, POST, PUT, DELETE, PATCH, HEAD, OPTIONS"))
+ })
+
+ It("should handle timeouts and errors", func() {
+ ts = httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ time.Sleep(50 * time.Millisecond)
+ }))
+ resp, err := svc.Get(context.Background(), &hosthttp.HttpRequest{
+ Url: ts.URL,
+ TimeoutMs: 1,
+ })
+ Expect(err).To(BeNil())
+ Expect(resp).NotTo(BeNil())
+ Expect(resp.Error).To(ContainSubstring("deadline exceeded"))
+ })
+
+ It("should return error on context timeout", func() {
+ ts = httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ time.Sleep(50 * time.Millisecond)
+ }))
+ ctx, cancel := context.WithTimeout(context.Background(), 1*time.Millisecond)
+ defer cancel()
+ resp, err := svc.Get(ctx, &hosthttp.HttpRequest{
+ Url: ts.URL,
+ TimeoutMs: 1000,
+ })
+ Expect(err).To(BeNil())
+ Expect(resp).NotTo(BeNil())
+ Expect(resp.Error).To(ContainSubstring("context deadline exceeded"))
+ })
+
+ It("should return error on context cancellation", func() {
+ ts = httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ time.Sleep(50 * time.Millisecond)
+ }))
+ ctx, cancel := context.WithCancel(context.Background())
+ go func() {
+ time.Sleep(1 * time.Millisecond)
+ cancel()
+ }()
+ resp, err := svc.Get(ctx, &hosthttp.HttpRequest{
+ Url: ts.URL,
+ TimeoutMs: 1000,
+ })
+ Expect(err).To(BeNil())
+ Expect(resp).NotTo(BeNil())
+ Expect(resp.Error).To(ContainSubstring("context canceled"))
+ })
+})
diff --git a/plugins/host_network_permissions_base.go b/plugins/host_network_permissions_base.go
new file mode 100644
index 000000000..c3224fe2a
--- /dev/null
+++ b/plugins/host_network_permissions_base.go
@@ -0,0 +1,192 @@
+package plugins
+
+import (
+ "fmt"
+ "net"
+ "net/url"
+ "regexp"
+ "strings"
+)
+
+// NetworkPermissionsBase contains common functionality for network-based permissions
+type networkPermissionsBase struct {
+ Reason string `json:"reason"`
+ AllowLocalNetwork bool `json:"allowLocalNetwork,omitempty"`
+}
+
+// URLMatcher provides URL pattern matching functionality
+type urlMatcher struct{}
+
+// newURLMatcher creates a new URL matcher instance
+func newURLMatcher() *urlMatcher {
+ return &urlMatcher{}
+}
+
+// checkURLPolicy performs common checks for a URL against network policies.
+func checkURLPolicy(requestURL string, allowLocalNetwork bool) (*url.URL, error) {
+ parsedURL, err := url.Parse(requestURL)
+ if err != nil {
+ return nil, fmt.Errorf("invalid URL: %w", err)
+ }
+
+ // Check local network restrictions
+ if !allowLocalNetwork {
+ if err := checkLocalNetwork(parsedURL); err != nil {
+ return nil, err
+ }
+ }
+ return parsedURL, nil
+}
+
+// MatchesURLPattern checks if a URL matches a given pattern
+func (m *urlMatcher) MatchesURLPattern(requestURL, pattern string) bool {
+ // Handle wildcard pattern
+ if pattern == "*" {
+ return true
+ }
+
+ // Parse both URLs to handle path matching correctly
+ reqURL, err := url.Parse(requestURL)
+ if err != nil {
+ return false
+ }
+
+ patternURL, err := url.Parse(pattern)
+ if err != nil {
+ // If pattern is not a valid URL, treat it as a simple string pattern
+ regexPattern := m.urlPatternToRegex(pattern)
+ matched, err := regexp.MatchString(regexPattern, requestURL)
+ if err != nil {
+ return false
+ }
+ return matched
+ }
+
+ // Match scheme
+ if patternURL.Scheme != "" && patternURL.Scheme != reqURL.Scheme {
+ return false
+ }
+
+ // Match host with wildcard support
+ if !m.matchesHost(reqURL.Host, patternURL.Host) {
+ return false
+ }
+
+ // Match path with wildcard support
+ // Special case: if pattern URL has empty path and contains wildcards, allow any path (domain-only wildcard matching)
+ if (patternURL.Path == "" || patternURL.Path == "/") && strings.Contains(pattern, "*") {
+ // This is a domain-only wildcard pattern, allow any path
+ return true
+ }
+ if !m.matchesPath(reqURL.Path, patternURL.Path) {
+ return false
+ }
+
+ return true
+}
+
+// urlPatternToRegex converts a URL pattern with wildcards to a regex pattern
+func (m *urlMatcher) urlPatternToRegex(pattern string) string {
+ // Escape special regex characters except *
+ escaped := regexp.QuoteMeta(pattern)
+
+ // Replace escaped \* with regex pattern for wildcard matching
+ // For subdomain: *.example.com -> [^.]*\.example\.com
+ // For path: /api/* -> /api/.*
+ escaped = strings.ReplaceAll(escaped, "\\*", ".*")
+
+ // Anchor the pattern to match the full URL
+ return "^" + escaped + "$"
+}
+
+// matchesHost checks if a host matches a pattern with wildcard support
+func (m *urlMatcher) matchesHost(host, pattern string) bool {
+ if pattern == "" {
+ return true
+ }
+
+ if pattern == "*" {
+ return true
+ }
+
+ // Handle wildcard patterns anywhere in the host
+ if strings.Contains(pattern, "*") {
+ patterns := []string{
+ strings.ReplaceAll(regexp.QuoteMeta(pattern), "\\*", "[0-9.]+"), // IP pattern
+ strings.ReplaceAll(regexp.QuoteMeta(pattern), "\\*", "[^.]*"), // Domain pattern
+ }
+
+ for _, regexPattern := range patterns {
+ fullPattern := "^" + regexPattern + "$"
+ if matched, err := regexp.MatchString(fullPattern, host); err == nil && matched {
+ return true
+ }
+ }
+ return false
+ }
+
+ return host == pattern
+}
+
+// matchesPath checks if a path matches a pattern with wildcard support
+func (m *urlMatcher) matchesPath(path, pattern string) bool {
+ // Normalize empty paths to "/"
+ if path == "" {
+ path = "/"
+ }
+ if pattern == "" {
+ pattern = "/"
+ }
+
+ if pattern == "*" {
+ return true
+ }
+
+ // Handle wildcard paths
+ if strings.HasSuffix(pattern, "/*") {
+ prefix := pattern[:len(pattern)-2] // Remove "/*"
+ if prefix == "" {
+ prefix = "/"
+ }
+ return strings.HasPrefix(path, prefix)
+ }
+
+ return path == pattern
+}
+
+// CheckLocalNetwork checks if the URL is accessing local network resources
+func checkLocalNetwork(parsedURL *url.URL) error {
+ host := parsedURL.Hostname()
+
+ // Check for localhost variants
+ if host == "localhost" || host == "127.0.0.1" || host == "::1" {
+ return fmt.Errorf("requests to localhost are not allowed")
+ }
+
+ // Try to parse as IP address
+ ip := net.ParseIP(host)
+ if ip != nil && isPrivateIP(ip) {
+ return fmt.Errorf("requests to private IP addresses are not allowed")
+ }
+
+ return nil
+}
+
+// IsPrivateIP checks if an IP is loopback, private, or link-local (IPv4/IPv6).
+func isPrivateIP(ip net.IP) bool {
+ if ip == nil {
+ return false
+ }
+ if ip.IsLoopback() || ip.IsPrivate() {
+ return true
+ }
+ // IPv4 link-local: 169.254.0.0/16
+ if ip4 := ip.To4(); ip4 != nil {
+ return ip4[0] == 169 && ip4[1] == 254
+ }
+ // IPv6 link-local: fe80::/10
+ if ip16 := ip.To16(); ip16 != nil && ip.To4() == nil {
+ return ip16[0] == 0xfe && (ip16[1]&0xc0) == 0x80
+ }
+ return false
+}
diff --git a/plugins/host_network_permissions_base_test.go b/plugins/host_network_permissions_base_test.go
new file mode 100644
index 000000000..9147e99ac
--- /dev/null
+++ b/plugins/host_network_permissions_base_test.go
@@ -0,0 +1,119 @@
+package plugins
+
+import (
+ "net"
+ "net/url"
+
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("networkPermissionsBase", func() {
+ Describe("urlMatcher", func() {
+ var matcher *urlMatcher
+
+ BeforeEach(func() {
+ matcher = newURLMatcher()
+ })
+
+ Describe("MatchesURLPattern", func() {
+ DescribeTable("exact URL matching",
+ func(requestURL, pattern string, expected bool) {
+ result := matcher.MatchesURLPattern(requestURL, pattern)
+ Expect(result).To(Equal(expected))
+ },
+ Entry("exact match", "https://api.example.com", "https://api.example.com", true),
+ Entry("different domain", "https://api.example.com", "https://api.other.com", false),
+ Entry("different scheme", "http://api.example.com", "https://api.example.com", false),
+ Entry("different path", "https://api.example.com/v1", "https://api.example.com/v2", false),
+ )
+
+ DescribeTable("wildcard pattern matching",
+ func(requestURL, pattern string, expected bool) {
+ result := matcher.MatchesURLPattern(requestURL, pattern)
+ Expect(result).To(Equal(expected))
+ },
+ Entry("universal wildcard", "https://api.example.com", "*", true),
+ Entry("subdomain wildcard match", "https://api.example.com", "https://*.example.com", true),
+ Entry("subdomain wildcard non-match", "https://api.other.com", "https://*.example.com", false),
+ Entry("path wildcard match", "https://api.example.com/v1/users", "https://api.example.com/*", true),
+ Entry("path wildcard non-match", "https://other.example.com/v1", "https://api.example.com/*", false),
+ Entry("port wildcard match", "https://api.example.com:8080", "https://api.example.com:*", true),
+ )
+ })
+ })
+
+ Describe("isPrivateIP", func() {
+ DescribeTable("IPv4 private IP detection",
+ func(ip string, expected bool) {
+ parsedIP := net.ParseIP(ip)
+ Expect(parsedIP).ToNot(BeNil(), "Failed to parse IP: %s", ip)
+ result := isPrivateIP(parsedIP)
+ Expect(result).To(Equal(expected))
+ },
+ // Private IPv4 ranges
+ Entry("10.0.0.1 (10.0.0.0/8)", "10.0.0.1", true),
+ Entry("10.255.255.255 (10.0.0.0/8)", "10.255.255.255", true),
+ Entry("172.16.0.1 (172.16.0.0/12)", "172.16.0.1", true),
+ Entry("172.31.255.255 (172.16.0.0/12)", "172.31.255.255", true),
+ Entry("192.168.1.1 (192.168.0.0/16)", "192.168.1.1", true),
+ Entry("192.168.255.255 (192.168.0.0/16)", "192.168.255.255", true),
+ Entry("127.0.0.1 (localhost)", "127.0.0.1", true),
+ Entry("127.255.255.255 (localhost)", "127.255.255.255", true),
+ Entry("169.254.1.1 (link-local)", "169.254.1.1", true),
+ Entry("169.254.255.255 (link-local)", "169.254.255.255", true),
+
+ // Public IPv4 addresses
+ Entry("8.8.8.8 (Google DNS)", "8.8.8.8", false),
+ Entry("1.1.1.1 (Cloudflare DNS)", "1.1.1.1", false),
+ Entry("208.67.222.222 (OpenDNS)", "208.67.222.222", false),
+ Entry("172.15.255.255 (just outside 172.16.0.0/12)", "172.15.255.255", false),
+ Entry("172.32.0.1 (just outside 172.16.0.0/12)", "172.32.0.1", false),
+ )
+
+ DescribeTable("IPv6 private IP detection",
+ func(ip string, expected bool) {
+ parsedIP := net.ParseIP(ip)
+ Expect(parsedIP).ToNot(BeNil(), "Failed to parse IP: %s", ip)
+ result := isPrivateIP(parsedIP)
+ Expect(result).To(Equal(expected))
+ },
+ // Private IPv6 ranges
+ Entry("::1 (IPv6 localhost)", "::1", true),
+ Entry("fe80::1 (link-local)", "fe80::1", true),
+ Entry("fc00::1 (unique local)", "fc00::1", true),
+ Entry("fd00::1 (unique local)", "fd00::1", true),
+
+ // Public IPv6 addresses
+ Entry("2001:4860:4860::8888 (Google DNS)", "2001:4860:4860::8888", false),
+ Entry("2606:4700:4700::1111 (Cloudflare DNS)", "2606:4700:4700::1111", false),
+ )
+ })
+
+ Describe("checkLocalNetwork", func() {
+ DescribeTable("local network detection",
+ func(urlStr string, shouldError bool, expectedErrorSubstring string) {
+ parsedURL, err := url.Parse(urlStr)
+ Expect(err).ToNot(HaveOccurred())
+
+ err = checkLocalNetwork(parsedURL)
+ if shouldError {
+ Expect(err).To(HaveOccurred())
+ if expectedErrorSubstring != "" {
+ Expect(err.Error()).To(ContainSubstring(expectedErrorSubstring))
+ }
+ } else {
+ Expect(err).ToNot(HaveOccurred())
+ }
+ },
+ Entry("localhost", "http://localhost:8080", true, "localhost"),
+ Entry("127.0.0.1", "http://127.0.0.1:3000", true, "localhost"),
+ Entry("::1", "http://[::1]:8080", true, "localhost"),
+ Entry("private IP 192.168.1.100", "http://192.168.1.100", true, "private IP"),
+ Entry("private IP 10.0.0.1", "http://10.0.0.1", true, "private IP"),
+ Entry("private IP 172.16.0.1", "http://172.16.0.1", true, "private IP"),
+ Entry("public IP 8.8.8.8", "http://8.8.8.8", false, ""),
+ Entry("public domain", "https://api.example.com", false, ""),
+ )
+ })
+})
diff --git a/plugins/host_scheduler.go b/plugins/host_scheduler.go
new file mode 100644
index 000000000..26c5e92f8
--- /dev/null
+++ b/plugins/host_scheduler.go
@@ -0,0 +1,338 @@
+package plugins
+
+import (
+ "context"
+ "fmt"
+ "sync"
+ "time"
+
+ gonanoid "github.com/matoous/go-nanoid/v2"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/plugins/host/scheduler"
+ navidsched "github.com/navidrome/navidrome/scheduler"
+)
+
+const (
+ ScheduleTypeOneTime = "one-time"
+ ScheduleTypeRecurring = "recurring"
+)
+
+// ScheduledCallback represents a registered schedule callback
+type ScheduledCallback struct {
+ ID string
+ PluginID string
+ Type string // "one-time" or "recurring"
+ Payload []byte
+ EntryID int // Used for recurring schedules via the scheduler
+ Cancel context.CancelFunc // Used for one-time schedules
+}
+
+// SchedulerHostFunctions implements the scheduler.SchedulerService interface
+type SchedulerHostFunctions struct {
+ ss *schedulerService
+ pluginID string
+}
+
+func (s SchedulerHostFunctions) ScheduleOneTime(ctx context.Context, req *scheduler.ScheduleOneTimeRequest) (*scheduler.ScheduleResponse, error) {
+ return s.ss.scheduleOneTime(ctx, s.pluginID, req)
+}
+
+func (s SchedulerHostFunctions) ScheduleRecurring(ctx context.Context, req *scheduler.ScheduleRecurringRequest) (*scheduler.ScheduleResponse, error) {
+ return s.ss.scheduleRecurring(ctx, s.pluginID, req)
+}
+
+func (s SchedulerHostFunctions) CancelSchedule(ctx context.Context, req *scheduler.CancelRequest) (*scheduler.CancelResponse, error) {
+ return s.ss.cancelSchedule(ctx, s.pluginID, req)
+}
+
+func (s SchedulerHostFunctions) TimeNow(ctx context.Context, req *scheduler.TimeNowRequest) (*scheduler.TimeNowResponse, error) {
+ return s.ss.timeNow(ctx, req)
+}
+
+type schedulerService struct {
+ // Map of schedule IDs to their callback info
+ schedules map[string]*ScheduledCallback
+ manager *managerImpl
+ navidSched navidsched.Scheduler // Navidrome scheduler for recurring jobs
+ mu sync.Mutex
+}
+
+// newSchedulerService creates a new schedulerService instance
+func newSchedulerService(manager *managerImpl) *schedulerService {
+ return &schedulerService{
+ schedules: make(map[string]*ScheduledCallback),
+ manager: manager,
+ navidSched: navidsched.GetInstance(),
+ }
+}
+
+func (s *schedulerService) HostFunctions(pluginID string) SchedulerHostFunctions {
+ return SchedulerHostFunctions{
+ ss: s,
+ pluginID: pluginID,
+ }
+}
+
+// Safe accessor methods for tests
+
+// hasSchedule safely checks if a schedule exists
+func (s *schedulerService) hasSchedule(id string) bool {
+ s.mu.Lock()
+ defer s.mu.Unlock()
+ _, exists := s.schedules[id]
+ return exists
+}
+
+// scheduleCount safely returns the number of schedules
+func (s *schedulerService) scheduleCount() int {
+ s.mu.Lock()
+ defer s.mu.Unlock()
+ return len(s.schedules)
+}
+
+// getScheduleType safely returns the type of a schedule
+func (s *schedulerService) getScheduleType(id string) string {
+ s.mu.Lock()
+ defer s.mu.Unlock()
+ if cb, exists := s.schedules[id]; exists {
+ return cb.Type
+ }
+ return ""
+}
+
+// scheduleJob is a helper function that handles the common logic for scheduling jobs
+func (s *schedulerService) scheduleJob(pluginID string, scheduleId string, jobType string, payload []byte) (string, *ScheduledCallback, context.CancelFunc, error) {
+ if s.manager == nil {
+ return "", nil, nil, fmt.Errorf("scheduler service not properly initialized")
+ }
+
+ // Original scheduleId (what the plugin will see)
+ originalScheduleId := scheduleId
+ if originalScheduleId == "" {
+ // Generate a random ID if one wasn't provided
+ originalScheduleId, _ = gonanoid.New(10)
+ }
+
+ // Internal scheduleId (prefixed with plugin name to avoid conflicts)
+ internalScheduleId := pluginID + ":" + originalScheduleId
+
+ // Store any existing cancellation function to call after we've updated the map
+ var cancelExisting context.CancelFunc
+
+ // Check if there's an existing schedule with the same ID, we'll cancel it after updating the map
+ if existingSchedule, ok := s.schedules[internalScheduleId]; ok {
+ log.Debug("Replacing existing schedule with same ID", "plugin", pluginID, "scheduleID", originalScheduleId)
+
+ // Store cancel information but don't call it yet
+ if existingSchedule.Type == ScheduleTypeOneTime && existingSchedule.Cancel != nil {
+ // We'll set the Cancel to nil to prevent the old job from removing the new one
+ cancelExisting = existingSchedule.Cancel
+ existingSchedule.Cancel = nil
+ } else if existingSchedule.Type == ScheduleTypeRecurring {
+ existingRecurringEntryID := existingSchedule.EntryID
+ if existingRecurringEntryID != 0 {
+ s.navidSched.Remove(existingRecurringEntryID)
+ }
+ }
+ }
+
+ // Create the callback object
+ callback := &ScheduledCallback{
+ ID: originalScheduleId,
+ PluginID: pluginID,
+ Type: jobType,
+ Payload: payload,
+ }
+
+ return internalScheduleId, callback, cancelExisting, nil
+}
+
+// scheduleOneTime registers a new one-time scheduled job
+func (s *schedulerService) scheduleOneTime(_ context.Context, pluginID string, req *scheduler.ScheduleOneTimeRequest) (*scheduler.ScheduleResponse, error) {
+ s.mu.Lock()
+ defer s.mu.Unlock()
+
+ internalScheduleId, callback, cancelExisting, err := s.scheduleJob(pluginID, req.ScheduleId, ScheduleTypeOneTime, req.Payload)
+ if err != nil {
+ return nil, err
+ }
+
+ // Create a context with cancel for this one-time schedule
+ scheduleCtx, cancel := context.WithCancel(context.Background())
+ callback.Cancel = cancel
+
+ // Store the callback info
+ s.schedules[internalScheduleId] = callback
+
+ // Now that the new job is in the map, we can safely cancel the old one
+ if cancelExisting != nil {
+ // Cancel in a goroutine to avoid deadlock since we're already holding the lock
+ go cancelExisting()
+ }
+
+ log.Debug("One-time schedule registered", "plugin", pluginID, "scheduleID", callback.ID, "internalID", internalScheduleId)
+
+ // Start the timer goroutine with the internal ID
+ go s.runOneTimeSchedule(scheduleCtx, internalScheduleId, time.Duration(req.DelaySeconds)*time.Second)
+
+ // Return the original ID to the plugin
+ return &scheduler.ScheduleResponse{
+ ScheduleId: callback.ID,
+ }, nil
+}
+
+// scheduleRecurring registers a new recurring scheduled job
+func (s *schedulerService) scheduleRecurring(_ context.Context, pluginID string, req *scheduler.ScheduleRecurringRequest) (*scheduler.ScheduleResponse, error) {
+ s.mu.Lock()
+ defer s.mu.Unlock()
+
+ internalScheduleId, callback, cancelExisting, err := s.scheduleJob(pluginID, req.ScheduleId, ScheduleTypeRecurring, req.Payload)
+ if err != nil {
+ return nil, err
+ }
+
+ // Schedule the job with the Navidrome scheduler
+ entryID, err := s.navidSched.Add(req.CronExpression, func() {
+ s.executeCallback(context.Background(), internalScheduleId, true)
+ })
+ if err != nil {
+ return nil, fmt.Errorf("failed to schedule recurring job: %w", err)
+ }
+
+ // Store the entry ID so we can cancel it later
+ callback.EntryID = entryID
+
+ // Store the callback info
+ s.schedules[internalScheduleId] = callback
+
+ // Now that the new job is in the map, we can safely cancel the old one
+ if cancelExisting != nil {
+ // Cancel in a goroutine to avoid deadlock since we're already holding the lock
+ go cancelExisting()
+ }
+
+ log.Debug("Recurring schedule registered", "plugin", pluginID, "scheduleID", callback.ID, "internalID", internalScheduleId, "cron", req.CronExpression)
+
+ // Return the original ID to the plugin
+ return &scheduler.ScheduleResponse{
+ ScheduleId: callback.ID,
+ }, nil
+}
+
+// cancelSchedule cancels a scheduled job (either one-time or recurring)
+func (s *schedulerService) cancelSchedule(_ context.Context, pluginID string, req *scheduler.CancelRequest) (*scheduler.CancelResponse, error) {
+ s.mu.Lock()
+ defer s.mu.Unlock()
+
+ internalScheduleId := pluginID + ":" + req.ScheduleId
+ callback, exists := s.schedules[internalScheduleId]
+ if !exists {
+ return &scheduler.CancelResponse{
+ Success: false,
+ Error: "schedule not found",
+ }, nil
+ }
+
+ // Store the cancel functions to call after we've updated the schedule map
+ var cancelFunc context.CancelFunc
+ var recurringEntryID int
+
+ // Store cancel information but don't call it yet
+ if callback.Type == ScheduleTypeOneTime && callback.Cancel != nil {
+ cancelFunc = callback.Cancel
+ callback.Cancel = nil // Set to nil to prevent the cancel handler from removing the job
+ } else if callback.Type == ScheduleTypeRecurring {
+ recurringEntryID = callback.EntryID
+ }
+
+ // First remove from the map
+ delete(s.schedules, internalScheduleId)
+
+ // Now perform the cancellation safely
+ if cancelFunc != nil {
+ // Execute in a goroutine to avoid deadlock since we're already holding the lock
+ go cancelFunc()
+ }
+ if recurringEntryID != 0 {
+ s.navidSched.Remove(recurringEntryID)
+ }
+
+ log.Debug("Schedule canceled", "plugin", pluginID, "scheduleID", req.ScheduleId, "internalID", internalScheduleId, "type", callback.Type)
+
+ return &scheduler.CancelResponse{
+ Success: true,
+ }, nil
+}
+
+// timeNow returns the current time in multiple formats
+func (s *schedulerService) timeNow(_ context.Context, req *scheduler.TimeNowRequest) (*scheduler.TimeNowResponse, error) {
+ now := time.Now()
+
+ return &scheduler.TimeNowResponse{
+ Rfc3339Nano: now.Format(time.RFC3339Nano),
+ UnixMilli: now.UnixMilli(),
+ LocalTimeZone: now.Location().String(),
+ }, nil
+}
+
+// runOneTimeSchedule handles the one-time schedule execution and callback
+func (s *schedulerService) runOneTimeSchedule(ctx context.Context, internalScheduleId string, delay time.Duration) {
+ tmr := time.NewTimer(delay)
+ defer tmr.Stop()
+
+ select {
+ case <-ctx.Done():
+ // Schedule was cancelled via its context
+ // We're no longer removing the schedule here because that's handled by the code that
+ // cancelled the context
+ log.Debug("One-time schedule context canceled", "internalID", internalScheduleId)
+ return
+
+ case <-tmr.C:
+ // Timer fired, execute the callback
+ s.executeCallback(ctx, internalScheduleId, false)
+ }
+}
+
+// executeCallback calls the plugin's OnSchedulerCallback method
+func (s *schedulerService) executeCallback(ctx context.Context, internalScheduleId string, isRecurring bool) {
+ s.mu.Lock()
+ callback := s.schedules[internalScheduleId]
+ // Only remove one-time schedules from the map after execution
+ if callback != nil && callback.Type == ScheduleTypeOneTime {
+ delete(s.schedules, internalScheduleId)
+ }
+ s.mu.Unlock()
+
+ if callback == nil {
+ log.Error("Schedule not found for callback", "internalID", internalScheduleId)
+ return
+ }
+
+ ctx = log.NewContext(ctx, "plugin", callback.PluginID, "scheduleID", callback.ID, "type", callback.Type)
+ log.Debug("Executing schedule callback")
+ start := time.Now()
+
+ // Get the plugin
+ p := s.manager.LoadPlugin(callback.PluginID, CapabilitySchedulerCallback)
+ if p == nil {
+ log.Error("Plugin not found for callback", "plugin", callback.PluginID)
+ return
+ }
+
+ // Type-check the plugin
+ plugin, ok := p.(*wasmSchedulerCallback)
+ if !ok {
+ log.Error("Plugin does not implement SchedulerCallback", "plugin", callback.PluginID)
+ return
+ }
+
+ // Call the plugin's OnSchedulerCallback method
+ log.Trace(ctx, "Executing schedule callback")
+ err := plugin.OnSchedulerCallback(ctx, callback.ID, callback.Payload, isRecurring)
+ if err != nil {
+ log.Error("Error executing schedule callback", "elapsed", time.Since(start), err)
+ return
+ }
+ log.Debug("Schedule callback executed", "elapsed", time.Since(start))
+}
diff --git a/plugins/host_scheduler_test.go b/plugins/host_scheduler_test.go
new file mode 100644
index 000000000..1a3efaae9
--- /dev/null
+++ b/plugins/host_scheduler_test.go
@@ -0,0 +1,192 @@
+package plugins
+
+import (
+ "context"
+ "time"
+
+ "github.com/navidrome/navidrome/core/metrics"
+ "github.com/navidrome/navidrome/plugins/host/scheduler"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("SchedulerService", func() {
+ var (
+ ss *schedulerService
+ manager *managerImpl
+ pluginName = "test_plugin"
+ )
+
+ BeforeEach(func() {
+ manager = createManager(nil, metrics.NewNoopInstance())
+ ss = manager.schedulerService
+ })
+
+ Describe("One-time scheduling", func() {
+ It("schedules one-time jobs successfully", func() {
+ req := &scheduler.ScheduleOneTimeRequest{
+ DelaySeconds: 1,
+ Payload: []byte("test payload"),
+ ScheduleId: "test-job",
+ }
+
+ resp, err := ss.scheduleOneTime(context.Background(), pluginName, req)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.ScheduleId).To(Equal("test-job"))
+ Expect(ss.hasSchedule(pluginName + ":" + "test-job")).To(BeTrue())
+ Expect(ss.getScheduleType(pluginName + ":" + "test-job")).To(Equal(ScheduleTypeOneTime))
+
+ // Test auto-generated ID
+ req.ScheduleId = ""
+ resp, err = ss.scheduleOneTime(context.Background(), pluginName, req)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.ScheduleId).ToNot(BeEmpty())
+ })
+
+ It("cancels one-time jobs successfully", func() {
+ req := &scheduler.ScheduleOneTimeRequest{
+ DelaySeconds: 10,
+ ScheduleId: "test-job",
+ }
+
+ _, err := ss.scheduleOneTime(context.Background(), pluginName, req)
+ Expect(err).ToNot(HaveOccurred())
+
+ cancelReq := &scheduler.CancelRequest{
+ ScheduleId: "test-job",
+ }
+
+ resp, err := ss.cancelSchedule(context.Background(), pluginName, cancelReq)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.Success).To(BeTrue())
+ Expect(ss.hasSchedule(pluginName + ":" + "test-job")).To(BeFalse())
+ })
+ })
+
+ Describe("Recurring scheduling", func() {
+ It("schedules recurring jobs successfully", func() {
+ req := &scheduler.ScheduleRecurringRequest{
+ CronExpression: "* * * * *", // Every minute
+ Payload: []byte("test payload"),
+ ScheduleId: "test-cron",
+ }
+
+ resp, err := ss.scheduleRecurring(context.Background(), pluginName, req)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.ScheduleId).To(Equal("test-cron"))
+ Expect(ss.hasSchedule(pluginName + ":" + "test-cron")).To(BeTrue())
+ Expect(ss.getScheduleType(pluginName + ":" + "test-cron")).To(Equal(ScheduleTypeRecurring))
+
+ // Test auto-generated ID
+ req.ScheduleId = ""
+ resp, err = ss.scheduleRecurring(context.Background(), pluginName, req)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.ScheduleId).ToNot(BeEmpty())
+ })
+
+ It("cancels recurring jobs successfully", func() {
+ req := &scheduler.ScheduleRecurringRequest{
+ CronExpression: "* * * * *", // Every minute
+ ScheduleId: "test-cron",
+ }
+
+ _, err := ss.scheduleRecurring(context.Background(), pluginName, req)
+ Expect(err).ToNot(HaveOccurred())
+
+ cancelReq := &scheduler.CancelRequest{
+ ScheduleId: "test-cron",
+ }
+
+ resp, err := ss.cancelSchedule(context.Background(), pluginName, cancelReq)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.Success).To(BeTrue())
+ Expect(ss.hasSchedule(pluginName + ":" + "test-cron")).To(BeFalse())
+ })
+ })
+
+ Describe("Replace existing schedules", func() {
+ It("replaces one-time jobs with new ones", func() {
+ // Create first job
+ req1 := &scheduler.ScheduleOneTimeRequest{
+ DelaySeconds: 10,
+ Payload: []byte("test payload 1"),
+ ScheduleId: "replace-job",
+ }
+ _, err := ss.scheduleOneTime(context.Background(), pluginName, req1)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify that the initial job exists
+ scheduleId := pluginName + ":" + "replace-job"
+ Expect(ss.hasSchedule(scheduleId)).To(BeTrue(), "Initial schedule should exist")
+
+ beforeCount := ss.scheduleCount()
+
+ // Replace with second job using same ID
+ req2 := &scheduler.ScheduleOneTimeRequest{
+ DelaySeconds: 60, // Use a longer delay to ensure it doesn't execute during the test
+ Payload: []byte("test payload 2"),
+ ScheduleId: "replace-job",
+ }
+
+ _, err = ss.scheduleOneTime(context.Background(), pluginName, req2)
+ Expect(err).ToNot(HaveOccurred())
+
+ Eventually(func() bool {
+ return ss.hasSchedule(scheduleId)
+ }).Should(BeTrue(), "Schedule should exist after replacement")
+ Expect(ss.scheduleCount()).To(Equal(beforeCount), "Job count should remain the same after replacement")
+ })
+
+ It("replaces recurring jobs with new ones", func() {
+ // Create first job
+ req1 := &scheduler.ScheduleRecurringRequest{
+ CronExpression: "0 * * * *",
+ Payload: []byte("test payload 1"),
+ ScheduleId: "replace-cron",
+ }
+ _, err := ss.scheduleRecurring(context.Background(), pluginName, req1)
+ Expect(err).ToNot(HaveOccurred())
+
+ beforeCount := ss.scheduleCount()
+
+ // Replace with second job using same ID
+ req2 := &scheduler.ScheduleRecurringRequest{
+ CronExpression: "*/5 * * * *",
+ Payload: []byte("test payload 2"),
+ ScheduleId: "replace-cron",
+ }
+
+ _, err = ss.scheduleRecurring(context.Background(), pluginName, req2)
+ Expect(err).ToNot(HaveOccurred())
+
+ Eventually(func() bool {
+ return ss.hasSchedule(pluginName + ":" + "replace-cron")
+ }).Should(BeTrue(), "Schedule should exist after replacement")
+ Expect(ss.scheduleCount()).To(Equal(beforeCount), "Job count should remain the same after replacement")
+ })
+ })
+
+ Describe("TimeNow", func() {
+ It("returns current time in RFC3339Nano, Unix milliseconds, and local timezone", func() {
+ now := time.Now()
+ req := &scheduler.TimeNowRequest{}
+ resp, err := ss.timeNow(context.Background(), req)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.UnixMilli).To(BeNumerically(">=", now.UnixMilli()))
+ Expect(resp.LocalTimeZone).ToNot(BeEmpty())
+
+ // Validate RFC3339Nano format can be parsed
+ parsedTime, parseErr := time.Parse(time.RFC3339Nano, resp.Rfc3339Nano)
+ Expect(parseErr).ToNot(HaveOccurred())
+
+ // Validate that Unix milliseconds is reasonably close to the RFC3339Nano time
+ expectedMillis := parsedTime.UnixMilli()
+ Expect(resp.UnixMilli).To(Equal(expectedMillis))
+
+ // Validate local timezone matches the current system timezone
+ expectedTimezone := now.Location().String()
+ Expect(resp.LocalTimeZone).To(Equal(expectedTimezone))
+ })
+ })
+})
diff --git a/plugins/host_subsonicapi.go b/plugins/host_subsonicapi.go
new file mode 100644
index 000000000..937dd044f
--- /dev/null
+++ b/plugins/host_subsonicapi.go
@@ -0,0 +1,170 @@
+package plugins
+
+import (
+ "context"
+ "errors"
+ "fmt"
+ "net/http"
+ "net/http/httptest"
+ "net/url"
+ "path"
+ "strings"
+
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/plugins/host/subsonicapi"
+ "github.com/navidrome/navidrome/plugins/schema"
+ "github.com/navidrome/navidrome/server/subsonic"
+)
+
+// SubsonicAPIService is the interface for the Subsonic API service
+//
+// Authentication: The plugin must provide valid authentication parameters in the URL:
+// - Required: `u` (username) - The service validates this parameter is present
+// - Example: `"/rest/ping?u=admin"`
+//
+// URL Format: Only the path and query parameters from the URL are used - host, protocol, and method are ignored
+//
+// Automatic Parameters: The service automatically adds:
+// - `c`: Plugin name (client identifier)
+// - `v`: Subsonic API version (1.16.1)
+// - `f`: Response format (json)
+//
+// See example usage in the `plugins/examples/subsonicapi-demo` plugin
+type subsonicAPIServiceImpl struct {
+ pluginID string
+ router SubsonicRouter
+ ds model.DataStore
+ permissions *subsonicAPIPermissions
+}
+
+func newSubsonicAPIService(pluginID string, router *SubsonicRouter, ds model.DataStore, permissions *schema.PluginManifestPermissionsSubsonicapi) subsonicapi.SubsonicAPIService {
+ return &subsonicAPIServiceImpl{
+ pluginID: pluginID,
+ router: *router,
+ ds: ds,
+ permissions: parseSubsonicAPIPermissions(permissions),
+ }
+}
+
+func (s *subsonicAPIServiceImpl) Call(ctx context.Context, req *subsonicapi.CallRequest) (*subsonicapi.CallResponse, error) {
+ if s.router == nil {
+ return &subsonicapi.CallResponse{
+ Error: "SubsonicAPI router not available",
+ }, nil
+ }
+
+ // Parse the input URL
+ parsedURL, err := url.Parse(req.Url)
+ if err != nil {
+ return &subsonicapi.CallResponse{
+ Error: fmt.Sprintf("invalid URL format: %v", err),
+ }, nil
+ }
+
+ // Extract query parameters
+ query := parsedURL.Query()
+
+ // Validate that 'u' (username) parameter is present
+ username := query.Get("u")
+ if username == "" {
+ return &subsonicapi.CallResponse{
+ Error: "missing required parameter 'u' (username)",
+ }, nil
+ }
+
+ if err := s.checkPermissions(ctx, username); err != nil {
+ log.Warn(ctx, "SubsonicAPI call blocked by permissions", "plugin", s.pluginID, "user", username, err)
+ return &subsonicapi.CallResponse{Error: err.Error()}, nil
+ }
+
+ // Add required Subsonic API parameters
+ query.Set("c", s.pluginID) // Client name (plugin ID)
+ query.Set("f", "json") // Response format
+ query.Set("v", subsonic.Version) // API version
+
+ // Extract the endpoint from the path
+ endpoint := path.Base(parsedURL.Path)
+
+ // Build the final URL with processed path and modified query parameters
+ finalURL := &url.URL{
+ Path: "/" + endpoint,
+ RawQuery: query.Encode(),
+ }
+
+ // Create HTTP request with a fresh context to avoid Chi RouteContext pollution.
+ // Using http.NewRequest (instead of http.NewRequestWithContext) ensures the internal
+ // SubsonicAPI call doesn't inherit routing information from the parent handler,
+ // which would cause Chi to invoke the wrong handler. Authentication context is
+ // explicitly added in the next step via request.WithInternalAuth.
+ httpReq, err := http.NewRequest("GET", finalURL.String(), nil)
+ if err != nil {
+ return &subsonicapi.CallResponse{
+ Error: fmt.Sprintf("failed to create HTTP request: %v", err),
+ }, nil
+ }
+
+ // Set internal authentication context using the username from the 'u' parameter
+ authCtx := request.WithInternalAuth(httpReq.Context(), username)
+ httpReq = httpReq.WithContext(authCtx)
+
+ // Use ResponseRecorder to capture the response
+ recorder := httptest.NewRecorder()
+
+ // Call the subsonic router
+ s.router.ServeHTTP(recorder, httpReq)
+
+ // Return the response body as JSON
+ return &subsonicapi.CallResponse{
+ Json: recorder.Body.String(),
+ }, nil
+}
+
+func (s *subsonicAPIServiceImpl) checkPermissions(ctx context.Context, username string) error {
+ if s.permissions == nil {
+ return nil
+ }
+ if len(s.permissions.AllowedUsernames) > 0 {
+ if _, ok := s.permissions.usernameMap[strings.ToLower(username)]; !ok {
+ return fmt.Errorf("username %s is not allowed", username)
+ }
+ }
+ if !s.permissions.AllowAdmins {
+ if s.router == nil {
+ return fmt.Errorf("permissions check failed: router not available")
+ }
+ usr, err := s.ds.User(ctx).FindByUsername(username)
+ if err != nil {
+ if errors.Is(err, model.ErrNotFound) {
+ return fmt.Errorf("username %s not found", username)
+ }
+ return err
+ }
+ if usr.IsAdmin {
+ return fmt.Errorf("calling SubsonicAPI as admin user is not allowed")
+ }
+ }
+ return nil
+}
+
+type subsonicAPIPermissions struct {
+ AllowedUsernames []string
+ AllowAdmins bool
+ usernameMap map[string]struct{}
+}
+
+func parseSubsonicAPIPermissions(data *schema.PluginManifestPermissionsSubsonicapi) *subsonicAPIPermissions {
+ if data == nil {
+ return &subsonicAPIPermissions{}
+ }
+ perms := &subsonicAPIPermissions{
+ AllowedUsernames: data.AllowedUsernames,
+ AllowAdmins: data.AllowAdmins,
+ usernameMap: make(map[string]struct{}),
+ }
+ for _, u := range data.AllowedUsernames {
+ perms.usernameMap[strings.ToLower(u)] = struct{}{}
+ }
+ return perms
+}
diff --git a/plugins/host_subsonicapi_test.go b/plugins/host_subsonicapi_test.go
new file mode 100644
index 000000000..a3161ff06
--- /dev/null
+++ b/plugins/host_subsonicapi_test.go
@@ -0,0 +1,218 @@
+package plugins
+
+import (
+ "context"
+ "net/http"
+
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/plugins/host/subsonicapi"
+ "github.com/navidrome/navidrome/plugins/schema"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("SubsonicAPI Host Service", func() {
+ var (
+ service *subsonicAPIServiceImpl
+ mockRouter http.Handler
+ userRepo *tests.MockedUserRepo
+ )
+
+ BeforeEach(func() {
+ // Setup mock datastore with users
+ userRepo = tests.CreateMockUserRepo()
+ _ = userRepo.Put(&model.User{UserName: "admin", IsAdmin: true})
+ _ = userRepo.Put(&model.User{UserName: "user", IsAdmin: false})
+ ds := &tests.MockDataStore{MockedUser: userRepo}
+
+ // Create a mock router
+ mockRouter = http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ w.Header().Set("Content-Type", "application/json")
+ w.WriteHeader(http.StatusOK)
+ _, _ = w.Write([]byte(`{"subsonic-response":{"status":"ok","version":"1.16.1"}}`))
+ })
+
+ // Create service implementation
+ service = &subsonicAPIServiceImpl{
+ pluginID: "test-plugin",
+ router: mockRouter,
+ ds: ds,
+ }
+ })
+
+ // Helper function to create a mock router that captures the request
+ setupRequestCapture := func() **http.Request {
+ var capturedRequest *http.Request
+ mockRouter = http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ capturedRequest = r
+ w.WriteHeader(http.StatusOK)
+ _, _ = w.Write([]byte(`{}`))
+ })
+ service.router = mockRouter
+ return &capturedRequest
+ }
+
+ Describe("Call", func() {
+ Context("when subsonic router is available", func() {
+ It("should process the request successfully", func() {
+ req := &subsonicapi.CallRequest{
+ Url: "/rest/ping?u=admin",
+ }
+
+ resp, err := service.Call(context.Background(), req)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp).ToNot(BeNil())
+ Expect(resp.Error).To(BeEmpty())
+ Expect(resp.Json).To(ContainSubstring("subsonic-response"))
+ Expect(resp.Json).To(ContainSubstring("ok"))
+ })
+
+ It("should add required parameters to the URL", func() {
+ capturedRequestPtr := setupRequestCapture()
+
+ req := &subsonicapi.CallRequest{
+ Url: "/rest/getAlbum.view?id=123&u=admin",
+ }
+
+ _, err := service.Call(context.Background(), req)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(*capturedRequestPtr).ToNot(BeNil())
+
+ query := (*capturedRequestPtr).URL.Query()
+ Expect(query.Get("c")).To(Equal("test-plugin"))
+ Expect(query.Get("f")).To(Equal("json"))
+ Expect(query.Get("v")).To(Equal("1.16.1"))
+ Expect(query.Get("id")).To(Equal("123"))
+ Expect(query.Get("u")).To(Equal("admin"))
+ })
+
+ It("should only use path and query from the input URL", func() {
+ capturedRequestPtr := setupRequestCapture()
+
+ req := &subsonicapi.CallRequest{
+ Url: "https://external.example.com:8080/rest/ping?u=admin",
+ }
+
+ _, err := service.Call(context.Background(), req)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(*capturedRequestPtr).ToNot(BeNil())
+ Expect((*capturedRequestPtr).URL.Path).To(Equal("/ping"))
+ Expect((*capturedRequestPtr).URL.Host).To(BeEmpty())
+ Expect((*capturedRequestPtr).URL.Scheme).To(BeEmpty())
+ })
+
+ It("ignores the path prefix in the URL", func() {
+ capturedRequestPtr := setupRequestCapture()
+
+ req := &subsonicapi.CallRequest{
+ Url: "/basepath/rest/ping?u=admin",
+ }
+
+ _, err := service.Call(context.Background(), req)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(*capturedRequestPtr).ToNot(BeNil())
+ Expect((*capturedRequestPtr).URL.Path).To(Equal("/ping"))
+ })
+
+ It("should set internal authentication with username from 'u' parameter", func() {
+ capturedRequestPtr := setupRequestCapture()
+
+ req := &subsonicapi.CallRequest{
+ Url: "/rest/ping?u=testuser",
+ }
+
+ _, err := service.Call(context.Background(), req)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(*capturedRequestPtr).ToNot(BeNil())
+
+ // Verify that internal authentication is set in the context
+ username, ok := request.InternalAuthFrom((*capturedRequestPtr).Context())
+ Expect(ok).To(BeTrue())
+ Expect(username).To(Equal("testuser"))
+ })
+ })
+
+ Context("when subsonic router is not available", func() {
+ BeforeEach(func() {
+ service.router = nil
+ })
+
+ It("should return an error", func() {
+ req := &subsonicapi.CallRequest{
+ Url: "/rest/ping?u=admin",
+ }
+
+ resp, err := service.Call(context.Background(), req)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp).ToNot(BeNil())
+ Expect(resp.Error).To(Equal("SubsonicAPI router not available"))
+ Expect(resp.Json).To(BeEmpty())
+ })
+ })
+
+ Context("when URL is invalid", func() {
+ It("should return an error for malformed URLs", func() {
+ req := &subsonicapi.CallRequest{
+ Url: "://invalid-url",
+ }
+
+ resp, err := service.Call(context.Background(), req)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp).ToNot(BeNil())
+ Expect(resp.Error).To(ContainSubstring("invalid URL format"))
+ Expect(resp.Json).To(BeEmpty())
+ })
+
+ It("should return an error when 'u' parameter is missing", func() {
+ req := &subsonicapi.CallRequest{
+ Url: "/rest/ping?p=password",
+ }
+
+ resp, err := service.Call(context.Background(), req)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp).ToNot(BeNil())
+ Expect(resp.Error).To(Equal("missing required parameter 'u' (username)"))
+ Expect(resp.Json).To(BeEmpty())
+ })
+ })
+
+ Context("permission checks", func() {
+ It("rejects disallowed username", func() {
+ service.permissions = parseSubsonicAPIPermissions(&schema.PluginManifestPermissionsSubsonicapi{
+ Reason: "test",
+ AllowedUsernames: []string{"user"},
+ })
+
+ resp, err := service.Call(context.Background(), &subsonicapi.CallRequest{Url: "/rest/ping?u=admin"})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.Error).To(ContainSubstring("not allowed"))
+ })
+
+ It("rejects admin when allowAdmins is false", func() {
+ service.permissions = parseSubsonicAPIPermissions(&schema.PluginManifestPermissionsSubsonicapi{Reason: "test"})
+
+ resp, err := service.Call(context.Background(), &subsonicapi.CallRequest{Url: "/rest/ping?u=admin"})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.Error).To(ContainSubstring("not allowed"))
+ })
+
+ It("allows admin when allowAdmins is true", func() {
+ service.permissions = parseSubsonicAPIPermissions(&schema.PluginManifestPermissionsSubsonicapi{Reason: "test", AllowAdmins: true})
+
+ resp, err := service.Call(context.Background(), &subsonicapi.CallRequest{Url: "/rest/ping?u=admin"})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.Error).To(BeEmpty())
+ })
+ })
+ })
+})
diff --git a/plugins/host_websocket.go b/plugins/host_websocket.go
new file mode 100644
index 000000000..e90d1363d
--- /dev/null
+++ b/plugins/host_websocket.go
@@ -0,0 +1,400 @@
+package plugins
+
+import (
+ "context"
+ "encoding/binary"
+ "fmt"
+ "strings"
+ "sync"
+ "time"
+
+ gorillaws "github.com/gorilla/websocket"
+ gonanoid "github.com/matoous/go-nanoid/v2"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/navidrome/navidrome/plugins/host/websocket"
+)
+
+// WebSocketConnection represents a WebSocket connection
+type WebSocketConnection struct {
+ Conn *gorillaws.Conn
+ PluginName string
+ ConnectionID string
+ Done chan struct{}
+ mu sync.Mutex
+}
+
+// WebSocketHostFunctions implements the websocket.WebSocketService interface
+type WebSocketHostFunctions struct {
+ ws *websocketService
+ pluginID string
+ permissions *webSocketPermissions
+}
+
+func (s WebSocketHostFunctions) Connect(ctx context.Context, req *websocket.ConnectRequest) (*websocket.ConnectResponse, error) {
+ return s.ws.connect(ctx, s.pluginID, req, s.permissions)
+}
+
+func (s WebSocketHostFunctions) SendText(ctx context.Context, req *websocket.SendTextRequest) (*websocket.SendTextResponse, error) {
+ return s.ws.sendText(ctx, s.pluginID, req)
+}
+
+func (s WebSocketHostFunctions) SendBinary(ctx context.Context, req *websocket.SendBinaryRequest) (*websocket.SendBinaryResponse, error) {
+ return s.ws.sendBinary(ctx, s.pluginID, req)
+}
+
+func (s WebSocketHostFunctions) Close(ctx context.Context, req *websocket.CloseRequest) (*websocket.CloseResponse, error) {
+ return s.ws.close(ctx, s.pluginID, req)
+}
+
+// websocketService implements the WebSocket service functionality
+type websocketService struct {
+ connections map[string]*WebSocketConnection
+ manager *managerImpl
+ mu sync.RWMutex
+}
+
+// newWebsocketService creates a new websocketService instance
+func newWebsocketService(manager *managerImpl) *websocketService {
+ return &websocketService{
+ connections: make(map[string]*WebSocketConnection),
+ manager: manager,
+ }
+}
+
+// HostFunctions returns the WebSocketHostFunctions for the given plugin
+func (s *websocketService) HostFunctions(pluginID string, permissions *webSocketPermissions) WebSocketHostFunctions {
+ return WebSocketHostFunctions{
+ ws: s,
+ pluginID: pluginID,
+ permissions: permissions,
+ }
+}
+
+// Safe accessor methods
+
+// hasConnection safely checks if a connection exists
+func (s *websocketService) hasConnection(id string) bool {
+ s.mu.RLock()
+ defer s.mu.RUnlock()
+ _, exists := s.connections[id]
+ return exists
+}
+
+// connectionCount safely returns the number of connections
+func (s *websocketService) connectionCount() int {
+ s.mu.RLock()
+ defer s.mu.RUnlock()
+ return len(s.connections)
+}
+
+// getConnection safely retrieves a connection by internal ID
+func (s *websocketService) getConnection(internalConnectionID string) (*WebSocketConnection, error) {
+ s.mu.RLock()
+ defer s.mu.RUnlock()
+ conn, exists := s.connections[internalConnectionID]
+
+ if !exists {
+ return nil, fmt.Errorf("connection not found")
+ }
+ return conn, nil
+}
+
+// internalConnectionID builds the internal connection ID from plugin and connection ID
+func internalConnectionID(pluginName, connectionID string) string {
+ return pluginName + ":" + connectionID
+}
+
+// extractConnectionID extracts the original connection ID from an internal ID
+func extractConnectionID(internalID string) (string, error) {
+ parts := strings.Split(internalID, ":")
+ if len(parts) != 2 {
+ return "", fmt.Errorf("invalid internal connection ID format: %s", internalID)
+ }
+ return parts[1], nil
+}
+
+// connect establishes a new WebSocket connection
+func (s *websocketService) connect(ctx context.Context, pluginID string, req *websocket.ConnectRequest, permissions *webSocketPermissions) (*websocket.ConnectResponse, error) {
+ if s.manager == nil {
+ return nil, fmt.Errorf("websocket service not properly initialized")
+ }
+
+ // Check permissions if they exist
+ if permissions != nil {
+ if err := permissions.IsConnectionAllowed(req.Url); err != nil {
+ log.Warn(ctx, "WebSocket connection blocked by permissions", "plugin", pluginID, "url", req.Url, err)
+ return &websocket.ConnectResponse{Error: "Connection blocked by plugin permissions: " + err.Error()}, nil
+ }
+ }
+
+ // Create websocket dialer with the headers
+ dialer := gorillaws.DefaultDialer
+ header := make(map[string][]string)
+ for k, v := range req.Headers {
+ header[k] = []string{v}
+ }
+
+ // Connect to the WebSocket server
+ conn, resp, err := dialer.DialContext(ctx, req.Url, header)
+ if err != nil {
+ return nil, fmt.Errorf("failed to connect to WebSocket server: %w", err)
+ }
+ defer resp.Body.Close()
+
+ // Generate a connection ID
+ if req.ConnectionId == "" {
+ req.ConnectionId, _ = gonanoid.New(10)
+ }
+ connectionID := req.ConnectionId
+ internal := internalConnectionID(pluginID, connectionID)
+
+ // Create the connection object
+ wsConn := &WebSocketConnection{
+ Conn: conn,
+ PluginName: pluginID,
+ ConnectionID: connectionID,
+ Done: make(chan struct{}),
+ }
+
+ // Store the connection
+ s.mu.Lock()
+ defer s.mu.Unlock()
+ s.connections[internal] = wsConn
+
+ log.Debug("WebSocket connection established", "plugin", pluginID, "connectionID", connectionID, "url", req.Url)
+
+ // Start the message handling goroutine
+ go s.handleMessages(internal, wsConn)
+
+ return &websocket.ConnectResponse{
+ ConnectionId: connectionID,
+ }, nil
+}
+
+// writeMessage is a helper to send messages to a websocket connection
+func (s *websocketService) writeMessage(pluginID string, connID string, messageType int, data []byte) error {
+ internal := internalConnectionID(pluginID, connID)
+
+ conn, err := s.getConnection(internal)
+ if err != nil {
+ return err
+ }
+
+ conn.mu.Lock()
+ defer conn.mu.Unlock()
+
+ if err := conn.Conn.WriteMessage(messageType, data); err != nil {
+ return fmt.Errorf("failed to send message: %w", err)
+ }
+
+ return nil
+}
+
+// sendText sends a text message over a WebSocket connection
+func (s *websocketService) sendText(ctx context.Context, pluginID string, req *websocket.SendTextRequest) (*websocket.SendTextResponse, error) {
+ if err := s.writeMessage(pluginID, req.ConnectionId, gorillaws.TextMessage, []byte(req.Message)); err != nil {
+ return &websocket.SendTextResponse{Error: err.Error()}, nil //nolint:nilerr
+ }
+ return &websocket.SendTextResponse{}, nil
+}
+
+// sendBinary sends binary data over a WebSocket connection
+func (s *websocketService) sendBinary(ctx context.Context, pluginID string, req *websocket.SendBinaryRequest) (*websocket.SendBinaryResponse, error) {
+ if err := s.writeMessage(pluginID, req.ConnectionId, gorillaws.BinaryMessage, req.Data); err != nil {
+ return &websocket.SendBinaryResponse{Error: err.Error()}, nil //nolint:nilerr
+ }
+ return &websocket.SendBinaryResponse{}, nil
+}
+
+// close closes a WebSocket connection
+func (s *websocketService) close(ctx context.Context, pluginID string, req *websocket.CloseRequest) (*websocket.CloseResponse, error) {
+ internal := internalConnectionID(pluginID, req.ConnectionId)
+
+ s.mu.Lock()
+ conn, exists := s.connections[internal]
+ if !exists {
+ s.mu.Unlock()
+ return &websocket.CloseResponse{Error: "connection not found"}, nil
+ }
+ delete(s.connections, internal)
+ s.mu.Unlock()
+
+ // Signal the message handling goroutine to stop
+ close(conn.Done)
+
+ // Close the connection with the specified code and reason
+ conn.mu.Lock()
+ defer conn.mu.Unlock()
+
+ err := conn.Conn.WriteControl(
+ gorillaws.CloseMessage,
+ gorillaws.FormatCloseMessage(int(req.Code), req.Reason),
+ time.Now().Add(time.Second),
+ )
+ if err != nil {
+ log.Error("Error sending close message", "plugin", pluginID, "error", err)
+ }
+
+ if err := conn.Conn.Close(); err != nil {
+ return nil, fmt.Errorf("error closing connection: %w", err)
+ }
+
+ log.Debug("WebSocket connection closed", "plugin", pluginID, "connectionID", req.ConnectionId)
+ return &websocket.CloseResponse{}, nil
+}
+
+// handleMessages processes incoming WebSocket messages
+func (s *websocketService) handleMessages(internalID string, conn *WebSocketConnection) {
+ // Get the original connection ID (without plugin prefix)
+ connectionID, err := extractConnectionID(internalID)
+ if err != nil {
+ log.Error("Invalid internal connection ID", "id", internalID, "error", err)
+ return
+ }
+
+ ctx, cancel := context.WithCancel(context.Background())
+ defer cancel()
+
+ defer func() {
+ // Ensure the connection is removed from the map if not already removed
+ s.mu.Lock()
+ defer s.mu.Unlock()
+ delete(s.connections, internalID)
+
+ log.Debug("WebSocket message handler stopped", "plugin", conn.PluginName, "connectionID", connectionID)
+ }()
+
+ // Add connection info to context
+ ctx = log.NewContext(ctx,
+ "connectionID", connectionID,
+ "plugin", conn.PluginName,
+ )
+
+ for {
+ select {
+ case <-conn.Done:
+ // Connection was closed by a Close call
+ return
+ default:
+ // Set a read deadline
+ _ = conn.Conn.SetReadDeadline(time.Now().Add(time.Second * 60))
+
+ // Read the next message
+ messageType, message, err := conn.Conn.ReadMessage()
+ if err != nil {
+ s.notifyErrorCallback(ctx, connectionID, conn, err.Error())
+ return
+ }
+
+ // Reset the read deadline
+ _ = conn.Conn.SetReadDeadline(time.Time{})
+
+ // Process the message based on its type
+ switch messageType {
+ case gorillaws.TextMessage:
+ s.notifyTextCallback(ctx, connectionID, conn, string(message))
+ case gorillaws.BinaryMessage:
+ s.notifyBinaryCallback(ctx, connectionID, conn, message)
+ case gorillaws.CloseMessage:
+ code := gorillaws.CloseNormalClosure
+ reason := ""
+ if len(message) >= 2 {
+ code = int(binary.BigEndian.Uint16(message[:2]))
+ if len(message) > 2 {
+ reason = string(message[2:])
+ }
+ }
+ s.notifyCloseCallback(ctx, connectionID, conn, code, reason)
+ return
+ }
+ }
+ }
+}
+
+// executeCallback is a common function that handles the plugin loading and execution
+// for all types of callbacks
+func (s *websocketService) executeCallback(ctx context.Context, pluginID, methodName string, fn func(context.Context, api.WebSocketCallback) error) {
+ log.Debug(ctx, "WebSocket received")
+
+ start := time.Now()
+
+ // Get the plugin
+ p := s.manager.LoadPlugin(pluginID, CapabilityWebSocketCallback)
+ if p == nil {
+ log.Error(ctx, "Plugin not found for WebSocket callback")
+ return
+ }
+
+ _, _ = callMethod(ctx, p, methodName, func(inst api.WebSocketCallback) (struct{}, error) {
+ // Call the appropriate callback function
+ log.Trace(ctx, "Executing WebSocket callback")
+ if err := fn(ctx, inst); err != nil {
+ log.Error(ctx, "Error executing WebSocket callback", "elapsed", time.Since(start), err)
+ return struct{}{}, fmt.Errorf("error executing WebSocket callback: %w", err)
+ }
+ log.Debug(ctx, "WebSocket callback executed", "elapsed", time.Since(start))
+ return struct{}{}, nil
+ })
+}
+
+// notifyTextCallback notifies the plugin of a text message
+func (s *websocketService) notifyTextCallback(ctx context.Context, connectionID string, conn *WebSocketConnection, message string) {
+ req := &api.OnTextMessageRequest{
+ ConnectionId: connectionID,
+ Message: message,
+ }
+
+ ctx = log.NewContext(ctx, "callback", "OnTextMessage", "size", len(message))
+
+ s.executeCallback(ctx, conn.PluginName, "OnTextMessage", func(ctx context.Context, plugin api.WebSocketCallback) error {
+ _, err := checkErr(plugin.OnTextMessage(ctx, req))
+ return err
+ })
+}
+
+// notifyBinaryCallback notifies the plugin of a binary message
+func (s *websocketService) notifyBinaryCallback(ctx context.Context, connectionID string, conn *WebSocketConnection, data []byte) {
+ req := &api.OnBinaryMessageRequest{
+ ConnectionId: connectionID,
+ Data: data,
+ }
+
+ ctx = log.NewContext(ctx, "callback", "OnBinaryMessage", "size", len(data))
+
+ s.executeCallback(ctx, conn.PluginName, "OnBinaryMessage", func(ctx context.Context, plugin api.WebSocketCallback) error {
+ _, err := checkErr(plugin.OnBinaryMessage(ctx, req))
+ return err
+ })
+}
+
+// notifyErrorCallback notifies the plugin of an error
+func (s *websocketService) notifyErrorCallback(ctx context.Context, connectionID string, conn *WebSocketConnection, errorMsg string) {
+ req := &api.OnErrorRequest{
+ ConnectionId: connectionID,
+ Error: errorMsg,
+ }
+
+ ctx = log.NewContext(ctx, "callback", "OnError", "error", errorMsg)
+
+ s.executeCallback(ctx, conn.PluginName, "OnError", func(ctx context.Context, plugin api.WebSocketCallback) error {
+ _, err := checkErr(plugin.OnError(ctx, req))
+ return err
+ })
+}
+
+// notifyCloseCallback notifies the plugin that the connection was closed
+func (s *websocketService) notifyCloseCallback(ctx context.Context, connectionID string, conn *WebSocketConnection, code int, reason string) {
+ req := &api.OnCloseRequest{
+ ConnectionId: connectionID,
+ Code: int32(code),
+ Reason: reason,
+ }
+
+ ctx = log.NewContext(ctx, "callback", "OnClose", "code", code, "reason", reason)
+
+ s.executeCallback(ctx, conn.PluginName, "OnClose", func(ctx context.Context, plugin api.WebSocketCallback) error {
+ _, err := checkErr(plugin.OnClose(ctx, req))
+ return err
+ })
+}
diff --git a/plugins/host_websocket_permissions.go b/plugins/host_websocket_permissions.go
new file mode 100644
index 000000000..53f6a127b
--- /dev/null
+++ b/plugins/host_websocket_permissions.go
@@ -0,0 +1,76 @@
+package plugins
+
+import (
+ "fmt"
+
+ "github.com/navidrome/navidrome/plugins/schema"
+)
+
+// WebSocketPermissions represents granular WebSocket access permissions for plugins
+type webSocketPermissions struct {
+ *networkPermissionsBase
+ AllowedUrls []string `json:"allowedUrls"`
+ matcher *urlMatcher
+}
+
+// parseWebSocketPermissions extracts WebSocket permissions from the schema
+func parseWebSocketPermissions(permData *schema.PluginManifestPermissionsWebsocket) (*webSocketPermissions, error) {
+ if len(permData.AllowedUrls) == 0 {
+ return nil, fmt.Errorf("allowedUrls must contain at least one URL pattern")
+ }
+
+ return &webSocketPermissions{
+ networkPermissionsBase: &networkPermissionsBase{
+ AllowLocalNetwork: permData.AllowLocalNetwork,
+ },
+ AllowedUrls: permData.AllowedUrls,
+ matcher: newURLMatcher(),
+ }, nil
+}
+
+// IsConnectionAllowed checks if a WebSocket connection is allowed
+func (w *webSocketPermissions) IsConnectionAllowed(requestURL string) error {
+ if _, err := checkURLPolicy(requestURL, w.AllowLocalNetwork); err != nil {
+ return err
+ }
+
+ // allowedUrls is required - no fallback to allow all URLs
+ if len(w.AllowedUrls) == 0 {
+ return fmt.Errorf("no allowed URLs configured for plugin")
+ }
+
+ // Check URL patterns
+ // First try exact matches, then wildcard matches
+
+ // Phase 1: Check for exact matches first
+ for _, urlPattern := range w.AllowedUrls {
+ if urlPattern == "*" || (!containsWildcard(urlPattern) && w.matcher.MatchesURLPattern(requestURL, urlPattern)) {
+ return nil
+ }
+ }
+
+ // Phase 2: Check wildcard patterns
+ for _, urlPattern := range w.AllowedUrls {
+ if containsWildcard(urlPattern) && w.matcher.MatchesURLPattern(requestURL, urlPattern) {
+ return nil
+ }
+ }
+
+ return fmt.Errorf("URL %s does not match any allowed URL patterns", requestURL)
+}
+
+// containsWildcard checks if a URL pattern contains wildcard characters
+func containsWildcard(pattern string) bool {
+ if pattern == "*" {
+ return true
+ }
+
+ // Check for wildcards anywhere in the pattern
+ for _, char := range pattern {
+ if char == '*' {
+ return true
+ }
+ }
+
+ return false
+}
diff --git a/plugins/host_websocket_permissions_test.go b/plugins/host_websocket_permissions_test.go
new file mode 100644
index 000000000..e794ca6ad
--- /dev/null
+++ b/plugins/host_websocket_permissions_test.go
@@ -0,0 +1,79 @@
+package plugins
+
+import (
+ "github.com/navidrome/navidrome/plugins/schema"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("WebSocket Permissions", func() {
+ Describe("parseWebSocketPermissions", func() {
+ It("should parse valid WebSocket permissions", func() {
+ permData := &schema.PluginManifestPermissionsWebsocket{
+ Reason: "Need to connect to WebSocket API",
+ AllowLocalNetwork: false,
+ AllowedUrls: []string{"wss://api.example.com/ws", "wss://cdn.example.com/*"},
+ }
+
+ perms, err := parseWebSocketPermissions(permData)
+ Expect(err).To(BeNil())
+ Expect(perms).ToNot(BeNil())
+ Expect(perms.AllowLocalNetwork).To(BeFalse())
+ Expect(perms.AllowedUrls).To(Equal([]string{"wss://api.example.com/ws", "wss://cdn.example.com/*"}))
+ })
+
+ It("should fail if allowedUrls is empty", func() {
+ permData := &schema.PluginManifestPermissionsWebsocket{
+ Reason: "Need to connect to WebSocket API",
+ AllowLocalNetwork: false,
+ AllowedUrls: []string{},
+ }
+
+ _, err := parseWebSocketPermissions(permData)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("allowedUrls must contain at least one URL pattern"))
+ })
+
+ It("should handle wildcard patterns", func() {
+ permData := &schema.PluginManifestPermissionsWebsocket{
+ Reason: "Need to connect to any WebSocket",
+ AllowLocalNetwork: true,
+ AllowedUrls: []string{"wss://*"},
+ }
+
+ perms, err := parseWebSocketPermissions(permData)
+ Expect(err).To(BeNil())
+ Expect(perms.AllowLocalNetwork).To(BeTrue())
+ Expect(perms.AllowedUrls).To(Equal([]string{"wss://*"}))
+ })
+
+ Context("URL matching", func() {
+ var perms *webSocketPermissions
+
+ BeforeEach(func() {
+ permData := &schema.PluginManifestPermissionsWebsocket{
+ Reason: "Need to connect to external services",
+ AllowLocalNetwork: true,
+ AllowedUrls: []string{"wss://api.example.com/*", "ws://localhost:8080"},
+ }
+ var err error
+ perms, err = parseWebSocketPermissions(permData)
+ Expect(err).To(BeNil())
+ })
+
+ It("should allow connections to URLs matching patterns", func() {
+ err := perms.IsConnectionAllowed("wss://api.example.com/v1/stream")
+ Expect(err).To(BeNil())
+
+ err = perms.IsConnectionAllowed("ws://localhost:8080")
+ Expect(err).To(BeNil())
+ })
+
+ It("should deny connections to URLs not matching patterns", func() {
+ err := perms.IsConnectionAllowed("wss://malicious.com/stream")
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("does not match any allowed URL patterns"))
+ })
+ })
+ })
+})
diff --git a/plugins/host_websocket_test.go b/plugins/host_websocket_test.go
new file mode 100644
index 000000000..ecadc6463
--- /dev/null
+++ b/plugins/host_websocket_test.go
@@ -0,0 +1,231 @@
+package plugins
+
+import (
+ "context"
+ "net/http"
+ "net/http/httptest"
+ "strings"
+ "sync"
+ "testing"
+ "time"
+
+ gorillaws "github.com/gorilla/websocket"
+ "github.com/navidrome/navidrome/core/metrics"
+ "github.com/navidrome/navidrome/plugins/host/websocket"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("WebSocket Host Service", func() {
+ var (
+ wsService *websocketService
+ manager *managerImpl
+ ctx context.Context
+ server *httptest.Server
+ upgrader gorillaws.Upgrader
+ serverMessages []string
+ serverMu sync.Mutex
+ )
+
+ // WebSocket echo server handler
+ echoHandler := func(w http.ResponseWriter, r *http.Request) {
+ // Check headers
+ if r.Header.Get("X-Test-Header") != "test-value" {
+ http.Error(w, "Missing or invalid X-Test-Header", http.StatusBadRequest)
+ return
+ }
+
+ // Upgrade connection to WebSocket
+ conn, err := upgrader.Upgrade(w, r, nil)
+ if err != nil {
+ return
+ }
+ defer conn.Close()
+
+ // Echo messages back
+ for {
+ mt, message, err := conn.ReadMessage()
+ if err != nil {
+ break
+ }
+
+ // Store the received message for verification
+ if mt == gorillaws.TextMessage {
+ msg := string(message)
+ serverMu.Lock()
+ serverMessages = append(serverMessages, msg)
+ serverMu.Unlock()
+ }
+
+ // Echo it back
+ err = conn.WriteMessage(mt, message)
+ if err != nil {
+ break
+ }
+
+ // If message is "close", close the connection
+ if mt == gorillaws.TextMessage && string(message) == "close" {
+ _ = conn.WriteControl(
+ gorillaws.CloseMessage,
+ gorillaws.FormatCloseMessage(gorillaws.CloseNormalClosure, "bye"),
+ time.Now().Add(time.Second),
+ )
+ break
+ }
+ }
+ }
+
+ BeforeEach(func() {
+ ctx = context.Background()
+ serverMessages = make([]string, 0)
+ serverMu = sync.Mutex{}
+
+ // Create a test WebSocket server
+ //upgrader = gorillaws.Upgrader{}
+ server = httptest.NewServer(http.HandlerFunc(echoHandler))
+ DeferCleanup(server.Close)
+
+ // Create a new manager and websocket service
+ manager = createManager(nil, metrics.NewNoopInstance())
+ wsService = newWebsocketService(manager)
+ })
+
+ Describe("WebSocket operations", func() {
+ var (
+ pluginName string
+ connectionID string
+ wsURL string
+ )
+
+ BeforeEach(func() {
+ pluginName = "test-plugin"
+ connectionID = "test-connection-id"
+ wsURL = "ws" + strings.TrimPrefix(server.URL, "http")
+ })
+
+ It("connects to a WebSocket server", func() {
+ // Connect to the WebSocket server
+ req := &websocket.ConnectRequest{
+ Url: wsURL,
+ Headers: map[string]string{
+ "X-Test-Header": "test-value",
+ },
+ ConnectionId: connectionID,
+ }
+
+ resp, err := wsService.connect(ctx, pluginName, req, nil)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.ConnectionId).ToNot(BeEmpty())
+ connectionID = resp.ConnectionId
+
+ // Verify that the connection was added to the service
+ internalID := pluginName + ":" + connectionID
+ Expect(wsService.hasConnection(internalID)).To(BeTrue())
+ })
+
+ It("sends and receives text messages", func() {
+ // Connect to the WebSocket server
+ req := &websocket.ConnectRequest{
+ Url: wsURL,
+ Headers: map[string]string{
+ "X-Test-Header": "test-value",
+ },
+ ConnectionId: connectionID,
+ }
+
+ resp, err := wsService.connect(ctx, pluginName, req, nil)
+ Expect(err).ToNot(HaveOccurred())
+ connectionID = resp.ConnectionId
+
+ // Send a text message
+ textReq := &websocket.SendTextRequest{
+ ConnectionId: connectionID,
+ Message: "hello websocket",
+ }
+
+ _, err = wsService.sendText(ctx, pluginName, textReq)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Wait a bit for the message to be processed
+ Eventually(func() []string {
+ serverMu.Lock()
+ defer serverMu.Unlock()
+ return serverMessages
+ }, "1s").Should(ContainElement("hello websocket"))
+ })
+
+ It("closes a WebSocket connection", func() {
+ // Connect to the WebSocket server
+ req := &websocket.ConnectRequest{
+ Url: wsURL,
+ Headers: map[string]string{
+ "X-Test-Header": "test-value",
+ },
+ ConnectionId: connectionID,
+ }
+
+ resp, err := wsService.connect(ctx, pluginName, req, nil)
+ Expect(err).ToNot(HaveOccurred())
+ connectionID = resp.ConnectionId
+
+ initialCount := wsService.connectionCount()
+
+ // Close the connection
+ closeReq := &websocket.CloseRequest{
+ ConnectionId: connectionID,
+ Code: 1000, // Normal closure
+ Reason: "test complete",
+ }
+
+ _, err = wsService.close(ctx, pluginName, closeReq)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify that the connection was removed
+ Eventually(func() int {
+ return wsService.connectionCount()
+ }, "1s").Should(Equal(initialCount - 1))
+
+ internalID := pluginName + ":" + connectionID
+ Expect(wsService.hasConnection(internalID)).To(BeFalse())
+ })
+
+ It("handles connection errors gracefully", func() {
+ if testing.Short() {
+ GinkgoT().Skip("skipping test in short mode.")
+ }
+
+ // Try to connect to an invalid URL
+ req := &websocket.ConnectRequest{
+ Url: "ws://invalid-url-that-does-not-exist",
+ Headers: map[string]string{},
+ ConnectionId: connectionID,
+ }
+
+ _, err := wsService.connect(ctx, pluginName, req, nil)
+ Expect(err).To(HaveOccurred())
+ })
+
+ It("returns error when attempting to use non-existent connection", func() {
+ // Try to send a message to a non-existent connection
+ textReq := &websocket.SendTextRequest{
+ ConnectionId: "non-existent-connection",
+ Message: "this should fail",
+ }
+
+ sendResp, err := wsService.sendText(ctx, pluginName, textReq)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(sendResp.Error).To(ContainSubstring("connection not found"))
+
+ // Try to close a non-existent connection
+ closeReq := &websocket.CloseRequest{
+ ConnectionId: "non-existent-connection",
+ Code: 1000,
+ Reason: "test complete",
+ }
+
+ closeResp, err := wsService.close(ctx, pluginName, closeReq)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(closeResp.Error).To(ContainSubstring("connection not found"))
+ })
+ })
+})
diff --git a/plugins/manager.go b/plugins/manager.go
new file mode 100644
index 000000000..35a1130fd
--- /dev/null
+++ b/plugins/manager.go
@@ -0,0 +1,421 @@
+package plugins
+
+//go:generate protoc --go-plugin_out=. --go-plugin_opt=paths=source_relative api/api.proto
+//go:generate protoc --go-plugin_out=. --go-plugin_opt=paths=source_relative host/http/http.proto
+//go:generate protoc --go-plugin_out=. --go-plugin_opt=paths=source_relative host/config/config.proto
+//go:generate protoc --go-plugin_out=. --go-plugin_opt=paths=source_relative host/websocket/websocket.proto
+//go:generate protoc --go-plugin_out=. --go-plugin_opt=paths=source_relative host/scheduler/scheduler.proto
+//go:generate protoc --go-plugin_out=. --go-plugin_opt=paths=source_relative host/cache/cache.proto
+//go:generate protoc --go-plugin_out=. --go-plugin_opt=paths=source_relative host/artwork/artwork.proto
+//go:generate protoc --go-plugin_out=. --go-plugin_opt=paths=source_relative host/subsonicapi/subsonicapi.proto
+
+import (
+ "fmt"
+ "net/http"
+ "os"
+ "slices"
+ "sync"
+ "sync/atomic"
+ "time"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/core/agents"
+ "github.com/navidrome/navidrome/core/metrics"
+ "github.com/navidrome/navidrome/core/scrobbler"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/navidrome/navidrome/plugins/schema"
+ "github.com/navidrome/navidrome/utils/singleton"
+ "github.com/navidrome/navidrome/utils/slice"
+ "github.com/tetratelabs/wazero"
+)
+
+const (
+ CapabilityMetadataAgent = "MetadataAgent"
+ CapabilityScrobbler = "Scrobbler"
+ CapabilitySchedulerCallback = "SchedulerCallback"
+ CapabilityWebSocketCallback = "WebSocketCallback"
+ CapabilityLifecycleManagement = "LifecycleManagement"
+)
+
+// pluginCreators maps capability types to their respective creator functions
+type pluginConstructor func(wasmPath, pluginID string, m *managerImpl, runtime api.WazeroNewRuntime, mc wazero.ModuleConfig) WasmPlugin
+
+var pluginCreators = map[string]pluginConstructor{
+ CapabilityMetadataAgent: newWasmMediaAgent,
+ CapabilityScrobbler: newWasmScrobblerPlugin,
+ CapabilitySchedulerCallback: newWasmSchedulerCallback,
+ CapabilityWebSocketCallback: newWasmWebSocketCallback,
+}
+
+// WasmPlugin is the base interface that all WASM plugins implement
+type WasmPlugin interface {
+ // PluginID returns the unique identifier of the plugin (folder name)
+ PluginID() string
+}
+
+type plugin struct {
+ ID string
+ Path string
+ Capabilities []string
+ WasmPath string
+ Manifest *schema.PluginManifest // Loaded manifest
+ Runtime api.WazeroNewRuntime
+ ModConfig wazero.ModuleConfig
+ compilationReady chan struct{}
+ compilationErr error
+}
+
+func (p *plugin) waitForCompilation() error {
+ timeout := pluginCompilationTimeout()
+ select {
+ case <-p.compilationReady:
+ case <-time.After(timeout):
+ err := fmt.Errorf("timed out waiting for plugin %s to compile", p.ID)
+ log.Error("Timed out waiting for plugin compilation", "name", p.ID, "path", p.WasmPath, "timeout", timeout, "err", err)
+ return err
+ }
+ if p.compilationErr != nil {
+ log.Error("Failed to compile plugin", "name", p.ID, "path", p.WasmPath, p.compilationErr)
+ }
+ return p.compilationErr
+}
+
+type SubsonicRouter http.Handler
+
+type Manager interface {
+ SetSubsonicRouter(router SubsonicRouter)
+ EnsureCompiled(name string) error
+ PluginList() map[string]schema.PluginManifest
+ PluginNames(capability string) []string
+ LoadPlugin(name string, capability string) WasmPlugin
+ LoadMediaAgent(name string) (agents.Interface, bool)
+ LoadScrobbler(name string) (scrobbler.Scrobbler, bool)
+ ScanPlugins()
+}
+
+// managerImpl is a singleton that manages plugins
+type managerImpl struct {
+ plugins map[string]*plugin // Map of plugin folder name to plugin info
+ pluginsMu sync.RWMutex // Protects plugins map
+ subsonicRouter atomic.Pointer[SubsonicRouter] // Subsonic API router
+ schedulerService *schedulerService // Service for handling scheduled tasks
+ websocketService *websocketService // Service for handling WebSocket connections
+ lifecycle *pluginLifecycleManager // Manages plugin lifecycle and initialization
+ adapters map[string]WasmPlugin // Map of plugin folder name + capability to adapter
+ ds model.DataStore // DataStore for accessing persistent data
+ metrics metrics.Metrics
+}
+
+// GetManager returns the singleton instance of managerImpl
+func GetManager(ds model.DataStore, metrics metrics.Metrics) Manager {
+ if !conf.Server.Plugins.Enabled {
+ return &noopManager{}
+ }
+ return singleton.GetInstance(func() *managerImpl {
+ return createManager(ds, metrics)
+ })
+}
+
+// createManager creates a new managerImpl instance. Used in tests
+func createManager(ds model.DataStore, metrics metrics.Metrics) *managerImpl {
+ m := &managerImpl{
+ plugins: make(map[string]*plugin),
+ lifecycle: newPluginLifecycleManager(metrics),
+ ds: ds,
+ metrics: metrics,
+ }
+
+ // Create the host services
+ m.schedulerService = newSchedulerService(m)
+ m.websocketService = newWebsocketService(m)
+
+ return m
+}
+
+// SetSubsonicRouter sets the SubsonicRouter after managerImpl initialization
+func (m *managerImpl) SetSubsonicRouter(router SubsonicRouter) {
+ m.subsonicRouter.Store(&router)
+}
+
+// registerPlugin adds a plugin to the registry with the given parameters
+// Used internally by ScanPlugins to register plugins
+func (m *managerImpl) registerPlugin(pluginID, pluginDir, wasmPath string, manifest *schema.PluginManifest) *plugin {
+ // Create custom runtime function
+ customRuntime := m.createRuntime(pluginID, manifest.Permissions)
+
+ // Configure module and determine plugin name
+ mc := newWazeroModuleConfig()
+
+ // Check if it's a symlink, indicating development mode
+ isSymlink := false
+ if fileInfo, err := os.Lstat(pluginDir); err == nil {
+ isSymlink = fileInfo.Mode()&os.ModeSymlink != 0
+ }
+
+ // Store plugin info
+ p := &plugin{
+ ID: pluginID,
+ Path: pluginDir,
+ Capabilities: slice.Map(manifest.Capabilities, func(cap schema.PluginManifestCapabilitiesElem) string { return string(cap) }),
+ WasmPath: wasmPath,
+ Manifest: manifest,
+ Runtime: customRuntime,
+ ModConfig: mc,
+ compilationReady: make(chan struct{}),
+ }
+
+ // Register the plugin first
+ m.pluginsMu.Lock()
+ m.plugins[pluginID] = p
+
+ // Register one plugin adapter for each capability
+ for _, capability := range manifest.Capabilities {
+ capabilityStr := string(capability)
+ constructor := pluginCreators[capabilityStr]
+ if constructor == nil {
+ // Warn about unknown capabilities, except for LifecycleManagement (it does not have an adapter)
+ if capability != CapabilityLifecycleManagement {
+ log.Warn("Unknown plugin capability type", "capability", capability, "plugin", pluginID)
+ }
+ continue
+ }
+ adapter := constructor(wasmPath, pluginID, m, customRuntime, mc)
+ if adapter == nil {
+ log.Error("Failed to create plugin adapter", "plugin", pluginID, "capability", capabilityStr, "path", wasmPath)
+ continue
+ }
+ m.adapters[pluginID+"_"+capabilityStr] = adapter
+ }
+ m.pluginsMu.Unlock()
+
+ log.Info("Discovered plugin", "folder", pluginID, "name", manifest.Name, "capabilities", manifest.Capabilities, "wasm", wasmPath, "dev_mode", isSymlink)
+ return m.plugins[pluginID]
+}
+
+// initializePluginIfNeeded calls OnInit on plugins that implement LifecycleManagement
+func (m *managerImpl) initializePluginIfNeeded(plugin *plugin) {
+ // Skip if already initialized
+ if m.lifecycle.isInitialized(plugin) {
+ return
+ }
+
+ // Check if the plugin implements LifecycleManagement
+ if slices.Contains(plugin.Manifest.Capabilities, CapabilityLifecycleManagement) {
+ if err := m.lifecycle.callOnInit(plugin); err != nil {
+ m.unregisterPlugin(plugin.ID)
+ }
+ }
+}
+
+// unregisterPlugin removes a plugin from the manager
+func (m *managerImpl) unregisterPlugin(pluginID string) {
+ m.pluginsMu.Lock()
+ defer m.pluginsMu.Unlock()
+
+ plugin, ok := m.plugins[pluginID]
+ if !ok {
+ return
+ }
+
+ // Clear initialization state from lifecycle manager
+ m.lifecycle.clearInitialized(plugin)
+
+ // Unregister plugin adapters
+ for _, capability := range plugin.Manifest.Capabilities {
+ delete(m.adapters, pluginID+"_"+string(capability))
+ }
+
+ // Unregister plugin
+ delete(m.plugins, pluginID)
+ log.Info("Unregistered plugin", "plugin", pluginID)
+}
+
+// ScanPlugins scans the plugins directory, discovers all valid plugins, and registers them for use.
+func (m *managerImpl) ScanPlugins() {
+ // Clear existing plugins
+ m.pluginsMu.Lock()
+ m.plugins = make(map[string]*plugin)
+ m.adapters = make(map[string]WasmPlugin)
+ m.pluginsMu.Unlock()
+
+ // Get plugins directory from config
+ root := conf.Server.Plugins.Folder
+ log.Debug("Scanning plugins folder", "root", root)
+
+ // Fail fast if the compilation cache cannot be initialized
+ _, err := getCompilationCache()
+ if err != nil {
+ log.Error("Failed to initialize plugins compilation cache. Disabling plugins", err)
+ return
+ }
+
+ // Discover all plugins using the shared discovery function
+ discoveries := DiscoverPlugins(root)
+
+ var validPluginNames []string
+ var registeredPlugins []*plugin
+ for _, discovery := range discoveries {
+ if discovery.Error != nil {
+ // Handle global errors (like directory read failure)
+ if discovery.ID == "" {
+ log.Error("Plugin discovery failed", discovery.Error)
+ return
+ }
+ // Handle individual plugin errors
+ log.Error("Failed to process plugin", "plugin", discovery.ID, discovery.Error)
+ continue
+ }
+
+ // Log discovery details
+ log.Debug("Processing entry", "name", discovery.ID, "isSymlink", discovery.IsSymlink)
+ if discovery.IsSymlink {
+ log.Debug("Processing symlinked plugin directory", "name", discovery.ID, "target", discovery.Path)
+ }
+ log.Debug("Checking for plugin.wasm", "wasmPath", discovery.WasmPath)
+ log.Debug("Manifest loaded successfully", "folder", discovery.ID, "name", discovery.Manifest.Name, "capabilities", discovery.Manifest.Capabilities)
+
+ validPluginNames = append(validPluginNames, discovery.ID)
+
+ // Register the plugin
+ plugin := m.registerPlugin(discovery.ID, discovery.Path, discovery.WasmPath, discovery.Manifest)
+ if plugin != nil {
+ registeredPlugins = append(registeredPlugins, plugin)
+ }
+ }
+
+ // Start background processing for all registered plugins after registration is complete
+ // This avoids race conditions between registration and goroutines that might unregister plugins
+ for _, p := range registeredPlugins {
+ go func(plugin *plugin) {
+ precompilePlugin(plugin)
+ // Check if this plugin implements InitService and hasn't been initialized yet
+ m.initializePluginIfNeeded(plugin)
+ }(p)
+ }
+
+ log.Debug("Found valid plugins", "count", len(validPluginNames), "plugins", validPluginNames)
+}
+
+// PluginList returns a map of all registered plugins with their manifests
+func (m *managerImpl) PluginList() map[string]schema.PluginManifest {
+ m.pluginsMu.RLock()
+ defer m.pluginsMu.RUnlock()
+
+ // Create a map to hold the plugin manifests
+ pluginList := make(map[string]schema.PluginManifest, len(m.plugins))
+ for name, plugin := range m.plugins {
+ // Use the plugin ID as the key and the manifest as the value
+ pluginList[name] = *plugin.Manifest
+ }
+ return pluginList
+}
+
+// PluginNames returns the folder names of all plugins that implement the specified capability
+func (m *managerImpl) PluginNames(capability string) []string {
+ m.pluginsMu.RLock()
+ defer m.pluginsMu.RUnlock()
+
+ var names []string
+ for name, plugin := range m.plugins {
+ for _, c := range plugin.Manifest.Capabilities {
+ if string(c) == capability {
+ names = append(names, name)
+ break
+ }
+ }
+ }
+ return names
+}
+
+func (m *managerImpl) getPlugin(name string, capability string) (*plugin, WasmPlugin, error) {
+ m.pluginsMu.RLock()
+ defer m.pluginsMu.RUnlock()
+ info, infoOk := m.plugins[name]
+ adapter, adapterOk := m.adapters[name+"_"+capability]
+
+ if !infoOk {
+ return nil, nil, fmt.Errorf("plugin not registered: %s", name)
+ }
+ if !adapterOk {
+ return nil, nil, fmt.Errorf("plugin adapter not registered: %s, capability: %s", name, capability)
+ }
+ return info, adapter, nil
+}
+
+// LoadPlugin instantiates and returns a plugin by folder name
+func (m *managerImpl) LoadPlugin(name string, capability string) WasmPlugin {
+ info, adapter, err := m.getPlugin(name, capability)
+ if err != nil {
+ log.Warn("Error loading plugin", err)
+ return nil
+ }
+
+ log.Debug("Loading plugin", "name", name, "path", info.Path)
+
+ // Wait for the plugin to be ready before using it.
+ if err := info.waitForCompilation(); err != nil {
+ log.Error("Plugin is not ready, cannot be loaded", "plugin", name, "capability", capability, "err", err)
+ return nil
+ }
+
+ if adapter == nil {
+ log.Warn("Plugin adapter not found", "name", name, "capability", capability)
+ return nil
+ }
+ return adapter
+}
+
+// EnsureCompiled waits for a plugin to finish compilation and returns any compilation error.
+// This is useful when you need to wait for compilation without loading a specific capability,
+// such as during plugin refresh operations or health checks.
+func (m *managerImpl) EnsureCompiled(name string) error {
+ m.pluginsMu.RLock()
+ plugin, ok := m.plugins[name]
+ m.pluginsMu.RUnlock()
+
+ if !ok {
+ return fmt.Errorf("plugin not found: %s", name)
+ }
+
+ return plugin.waitForCompilation()
+}
+
+// LoadMediaAgent instantiates and returns a media agent plugin by folder name
+func (m *managerImpl) LoadMediaAgent(name string) (agents.Interface, bool) {
+ plugin := m.LoadPlugin(name, CapabilityMetadataAgent)
+ if plugin == nil {
+ return nil, false
+ }
+ agent, ok := plugin.(*wasmMediaAgent)
+ return agent, ok
+}
+
+// LoadScrobbler instantiates and returns a scrobbler plugin by folder name
+func (m *managerImpl) LoadScrobbler(name string) (scrobbler.Scrobbler, bool) {
+ plugin := m.LoadPlugin(name, CapabilityScrobbler)
+ if plugin == nil {
+ return nil, false
+ }
+ s, ok := plugin.(scrobbler.Scrobbler)
+ return s, ok
+}
+
+type noopManager struct{}
+
+func (n noopManager) SetSubsonicRouter(router SubsonicRouter) {}
+
+func (n noopManager) EnsureCompiled(name string) error { return nil }
+
+func (n noopManager) PluginList() map[string]schema.PluginManifest { return nil }
+
+func (n noopManager) PluginNames(capability string) []string { return nil }
+
+func (n noopManager) LoadPlugin(name string, capability string) WasmPlugin { return nil }
+
+func (n noopManager) LoadMediaAgent(name string) (agents.Interface, bool) { return nil, false }
+
+func (n noopManager) LoadScrobbler(name string) (scrobbler.Scrobbler, bool) { return nil, false }
+
+func (n noopManager) ScanPlugins() {}
diff --git a/plugins/manager_test.go b/plugins/manager_test.go
new file mode 100644
index 000000000..8b361f8b3
--- /dev/null
+++ b/plugins/manager_test.go
@@ -0,0 +1,367 @@
+package plugins
+
+import (
+ "context"
+ "os"
+ "path/filepath"
+ "time"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/core/agents"
+ "github.com/navidrome/navidrome/core/metrics"
+ "github.com/navidrome/navidrome/plugins/schema"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("Plugin Manager", func() {
+ var mgr *managerImpl
+ var ctx context.Context
+
+ BeforeEach(func() {
+ // We change the plugins folder to random location to avoid conflicts with other tests,
+ // but, as this is an integration test, we can't use configtest.SetupConfig() as it causes
+ // data races.
+ originalPluginsFolder := conf.Server.Plugins.Folder
+ originalTimeout := conf.Server.DevPluginCompilationTimeout
+ conf.Server.DevPluginCompilationTimeout = 2 * time.Minute
+ DeferCleanup(func() {
+ conf.Server.Plugins.Folder = originalPluginsFolder
+ conf.Server.DevPluginCompilationTimeout = originalTimeout
+ })
+ conf.Server.Plugins.Enabled = true
+ conf.Server.Plugins.Folder = testDataDir
+
+ ctx = GinkgoT().Context()
+ mgr = createManager(nil, metrics.NewNoopInstance())
+ mgr.ScanPlugins()
+
+ // Wait for all plugins to compile to avoid race conditions
+ err := mgr.EnsureCompiled("fake_artist_agent")
+ Expect(err).NotTo(HaveOccurred(), "fake_artist_agent should compile successfully")
+ err = mgr.EnsureCompiled("fake_album_agent")
+ Expect(err).NotTo(HaveOccurred(), "fake_album_agent should compile successfully")
+ err = mgr.EnsureCompiled("multi_plugin")
+ Expect(err).NotTo(HaveOccurred(), "multi_plugin should compile successfully")
+ err = mgr.EnsureCompiled("unauthorized_plugin")
+ Expect(err).NotTo(HaveOccurred(), "unauthorized_plugin should compile successfully")
+ })
+
+ It("should scan and discover plugins from the testdata folder", func() {
+ Expect(mgr).NotTo(BeNil())
+
+ mediaAgentNames := mgr.PluginNames("MetadataAgent")
+ Expect(mediaAgentNames).To(HaveLen(4))
+ Expect(mediaAgentNames).To(ContainElements(
+ "fake_artist_agent",
+ "fake_album_agent",
+ "multi_plugin",
+ "unauthorized_plugin",
+ ))
+
+ scrobblerNames := mgr.PluginNames("Scrobbler")
+ Expect(scrobblerNames).To(ContainElement("fake_scrobbler"))
+
+ initServiceNames := mgr.PluginNames("LifecycleManagement")
+ Expect(initServiceNames).To(ContainElements("multi_plugin", "fake_init_service"))
+
+ schedulerCallbackNames := mgr.PluginNames("SchedulerCallback")
+ Expect(schedulerCallbackNames).To(ContainElement("multi_plugin"))
+ })
+
+ It("should load all plugins from folder", func() {
+ all := mgr.PluginList()
+ Expect(all).To(HaveLen(6))
+ Expect(all["fake_artist_agent"].Name).To(Equal("fake_artist_agent"))
+ Expect(all["unauthorized_plugin"].Capabilities).To(HaveExactElements(schema.PluginManifestCapabilitiesElem("MetadataAgent")))
+ })
+
+ It("should load a MetadataAgent plugin and invoke artist-related methods", func() {
+ plugin := mgr.LoadPlugin("fake_artist_agent", CapabilityMetadataAgent)
+ Expect(plugin).NotTo(BeNil())
+
+ agent, ok := plugin.(agents.Interface)
+ Expect(ok).To(BeTrue(), "plugin should implement agents.Interface")
+ Expect(agent.AgentName()).To(Equal("fake_artist_agent"))
+
+ mbidRetriever, ok := agent.(agents.ArtistMBIDRetriever)
+ Expect(ok).To(BeTrue())
+ mbid, err := mbidRetriever.GetArtistMBID(ctx, "123", "The Beatles")
+ Expect(err).NotTo(HaveOccurred())
+ Expect(mbid).To(Equal("1234567890"))
+ })
+
+ It("should load all MetadataAgent plugins", func() {
+ mediaAgentNames := mgr.PluginNames("MetadataAgent")
+ Expect(mediaAgentNames).To(HaveLen(4))
+
+ var agentNames []string
+ for _, name := range mediaAgentNames {
+ agent, ok := mgr.LoadMediaAgent(name)
+ if ok {
+ agentNames = append(agentNames, agent.AgentName())
+ }
+ }
+
+ Expect(agentNames).To(ContainElements("fake_artist_agent", "fake_album_agent", "multi_plugin", "unauthorized_plugin"))
+ })
+
+ Describe("ScanPlugins", func() {
+ var tempPluginsDir string
+ var m *managerImpl
+
+ BeforeEach(func() {
+ tempPluginsDir, _ = os.MkdirTemp("", "navidrome-plugins-test-*")
+ DeferCleanup(func() {
+ _ = os.RemoveAll(tempPluginsDir)
+ })
+
+ conf.Server.Plugins.Folder = tempPluginsDir
+ m = createManager(nil, metrics.NewNoopInstance())
+ })
+
+ // Helper to create a complete valid plugin for manager testing
+ createValidPlugin := func(folderName, manifestName string) {
+ pluginDir := filepath.Join(tempPluginsDir, folderName)
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+
+ // Copy real WASM file from testdata
+ sourceWasmPath := filepath.Join(testDataDir, "fake_artist_agent", "plugin.wasm")
+ targetWasmPath := filepath.Join(pluginDir, "plugin.wasm")
+ sourceWasm, err := os.ReadFile(sourceWasmPath)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(os.WriteFile(targetWasmPath, sourceWasm, 0600)).To(Succeed())
+
+ manifest := `{
+ "name": "` + manifestName + `",
+ "version": "1.0.0",
+ "capabilities": ["MetadataAgent"],
+ "author": "Test Author",
+ "description": "Test Plugin",
+ "website": "https://test.navidrome.org/` + manifestName + `",
+ "permissions": {}
+ }`
+ Expect(os.WriteFile(filepath.Join(pluginDir, "manifest.json"), []byte(manifest), 0600)).To(Succeed())
+ }
+
+ It("should register and compile discovered plugins", func() {
+ createValidPlugin("test-plugin", "test-plugin")
+
+ m.ScanPlugins()
+
+ // Focus on manager behavior: registration and compilation
+ Expect(m.plugins).To(HaveLen(1))
+ Expect(m.plugins).To(HaveKey("test-plugin"))
+
+ plugin := m.plugins["test-plugin"]
+ Expect(plugin.ID).To(Equal("test-plugin"))
+ Expect(plugin.Manifest.Name).To(Equal("test-plugin"))
+
+ // Verify plugin can be loaded (compilation successful)
+ loadedPlugin := m.LoadPlugin("test-plugin", CapabilityMetadataAgent)
+ Expect(loadedPlugin).NotTo(BeNil())
+ })
+
+ It("should handle multiple plugins with different IDs but same manifest names", func() {
+ // This tests manager-specific behavior: how it handles ID conflicts
+ createValidPlugin("lastfm-official", "lastfm")
+ createValidPlugin("lastfm-custom", "lastfm")
+
+ m.ScanPlugins()
+
+ // Both should be registered with their folder names as IDs
+ Expect(m.plugins).To(HaveLen(2))
+ Expect(m.plugins).To(HaveKey("lastfm-official"))
+ Expect(m.plugins).To(HaveKey("lastfm-custom"))
+
+ // Both should be loadable independently
+ official := m.LoadPlugin("lastfm-official", CapabilityMetadataAgent)
+ custom := m.LoadPlugin("lastfm-custom", CapabilityMetadataAgent)
+ Expect(official).NotTo(BeNil())
+ Expect(custom).NotTo(BeNil())
+ Expect(official.PluginID()).To(Equal("lastfm-official"))
+ Expect(custom.PluginID()).To(Equal("lastfm-custom"))
+ })
+ })
+
+ Describe("LoadPlugin", func() {
+ It("should load a MetadataAgent plugin and invoke artist-related methods", func() {
+ plugin := mgr.LoadPlugin("fake_artist_agent", CapabilityMetadataAgent)
+ Expect(plugin).NotTo(BeNil())
+
+ agent, ok := plugin.(agents.Interface)
+ Expect(ok).To(BeTrue(), "plugin should implement agents.Interface")
+ Expect(agent.AgentName()).To(Equal("fake_artist_agent"))
+
+ mbidRetriever, ok := agent.(agents.ArtistMBIDRetriever)
+ Expect(ok).To(BeTrue())
+ mbid, err := mbidRetriever.GetArtistMBID(ctx, "id", "Test Artist")
+ Expect(err).NotTo(HaveOccurred())
+ Expect(mbid).To(Equal("1234567890"))
+ })
+ })
+
+ Describe("EnsureCompiled", func() {
+ It("should successfully wait for plugin compilation", func() {
+ err := mgr.EnsureCompiled("fake_artist_agent")
+ Expect(err).NotTo(HaveOccurred())
+ })
+
+ It("should return error for non-existent plugin", func() {
+ err := mgr.EnsureCompiled("non-existent-plugin")
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("plugin not found: non-existent-plugin"))
+ })
+
+ It("should wait for compilation to complete for all valid plugins", func() {
+ pluginNames := []string{"fake_artist_agent", "fake_album_agent", "multi_plugin", "fake_scrobbler"}
+
+ for _, name := range pluginNames {
+ err := mgr.EnsureCompiled(name)
+ Expect(err).NotTo(HaveOccurred(), "plugin %s should compile successfully", name)
+ }
+ })
+ })
+
+ Describe("Invoke Methods", func() {
+ It("should load all MetadataAgent plugins and invoke methods", func() {
+ fakeAlbumPlugin, isMediaAgent := mgr.LoadMediaAgent("fake_album_agent")
+ Expect(isMediaAgent).To(BeTrue())
+
+ Expect(fakeAlbumPlugin).NotTo(BeNil(), "fake_album_agent should be loaded")
+
+ // Test GetAlbumInfo method - need to cast to the specific interface
+ albumRetriever, ok := fakeAlbumPlugin.(agents.AlbumInfoRetriever)
+ Expect(ok).To(BeTrue(), "fake_album_agent should implement AlbumInfoRetriever")
+
+ info, err := albumRetriever.GetAlbumInfo(ctx, "Test Album", "Test Artist", "123")
+ Expect(err).NotTo(HaveOccurred())
+ Expect(info).NotTo(BeNil())
+ Expect(info.Name).To(Equal("Test Album"))
+ })
+ })
+
+ Describe("Permission Enforcement Integration", func() {
+ It("should fail when plugin tries to access unauthorized services", func() {
+ // This plugin tries to access config service but has no permissions
+ plugin := mgr.LoadPlugin("unauthorized_plugin", CapabilityMetadataAgent)
+ Expect(plugin).NotTo(BeNil())
+
+ agent, ok := plugin.(agents.Interface)
+ Expect(ok).To(BeTrue())
+
+ // This should fail because the plugin tries to access unauthorized config service
+ // The exact behavior depends on the plugin implementation, but it should either:
+ // 1. Fail during instantiation, or
+ // 2. Return an error when trying to call config methods
+
+ // Try to use one of the available methods - let's test with GetArtistMBID
+ mbidRetriever, isMBIDRetriever := agent.(agents.ArtistMBIDRetriever)
+ if isMBIDRetriever {
+ _, err := mbidRetriever.GetArtistMBID(ctx, "id", "Test Artist")
+ if err == nil {
+ // If no error, the plugin should still be working
+ // but any config access should fail silently or return default values
+ Expect(agent.AgentName()).To(Equal("unauthorized_plugin"))
+ } else {
+ // If there's an error, it should be related to missing permissions
+ Expect(err.Error()).To(ContainSubstring(""))
+ }
+ } else {
+ // If the plugin doesn't implement the interface, that's also acceptable
+ Expect(agent.AgentName()).To(Equal("unauthorized_plugin"))
+ }
+ })
+ })
+
+ Describe("Plugin Initialization Lifecycle", func() {
+ BeforeEach(func() {
+ conf.Server.Plugins.Enabled = true
+ conf.Server.Plugins.Folder = testDataDir
+ })
+
+ Context("when OnInit is successful", func() {
+ It("should register and initialize the plugin", func() {
+ conf.Server.PluginConfig = nil
+ mgr = createManager(nil, metrics.NewNoopInstance()) // Create manager after setting config
+ mgr.ScanPlugins()
+
+ plugin := mgr.plugins["fake_init_service"]
+ Expect(plugin).NotTo(BeNil())
+
+ Eventually(func() bool {
+ return mgr.lifecycle.isInitialized(plugin)
+ }).Should(BeTrue())
+
+ // Check that the plugin is still registered
+ names := mgr.PluginNames(CapabilityLifecycleManagement)
+ Expect(names).To(ContainElement("fake_init_service"))
+ })
+ })
+
+ Context("when OnInit fails", func() {
+ It("should unregister the plugin if OnInit returns an error string", func() {
+ conf.Server.PluginConfig = map[string]map[string]string{
+ "fake_init_service": {
+ "returnError": "response_error",
+ },
+ }
+ mgr = createManager(nil, metrics.NewNoopInstance()) // Create manager after setting config
+ mgr.ScanPlugins()
+
+ Eventually(func() []string {
+ return mgr.PluginNames(CapabilityLifecycleManagement)
+ }).ShouldNot(ContainElement("fake_init_service"))
+ })
+
+ It("should unregister the plugin if OnInit returns a Go error", func() {
+ conf.Server.PluginConfig = map[string]map[string]string{
+ "fake_init_service": {
+ "returnError": "go_error",
+ },
+ }
+ mgr = createManager(nil, metrics.NewNoopInstance()) // Create manager after setting config
+ mgr.ScanPlugins()
+
+ Eventually(func() []string {
+ return mgr.PluginNames(CapabilityLifecycleManagement)
+ }).ShouldNot(ContainElement("fake_init_service"))
+ })
+ })
+
+ It("should clear lifecycle state when unregistering a plugin", func() {
+ // Create a manager and register a plugin
+ mgr := createManager(nil, metrics.NewNoopInstance())
+
+ // Create a mock plugin with LifecycleManagement capability
+ plugin := &plugin{
+ ID: "test-plugin",
+ Capabilities: []string{CapabilityLifecycleManagement},
+ Manifest: &schema.PluginManifest{
+ Version: "1.0.0",
+ },
+ }
+
+ // Register the plugin in the manager
+ mgr.pluginsMu.Lock()
+ mgr.plugins[plugin.ID] = plugin
+ mgr.pluginsMu.Unlock()
+
+ // Mark the plugin as initialized in the lifecycle manager
+ mgr.lifecycle.markInitialized(plugin)
+ Expect(mgr.lifecycle.isInitialized(plugin)).To(BeTrue())
+
+ // Unregister the plugin
+ mgr.unregisterPlugin(plugin.ID)
+
+ // Verify that the plugin is no longer in the manager
+ mgr.pluginsMu.RLock()
+ _, exists := mgr.plugins[plugin.ID]
+ mgr.pluginsMu.RUnlock()
+ Expect(exists).To(BeFalse())
+
+ // Verify that the lifecycle state has been cleared
+ Expect(mgr.lifecycle.isInitialized(plugin)).To(BeFalse())
+ })
+ })
+})
diff --git a/plugins/manifest.go b/plugins/manifest.go
new file mode 100644
index 000000000..b56187bcc
--- /dev/null
+++ b/plugins/manifest.go
@@ -0,0 +1,30 @@
+package plugins
+
+//go:generate go tool go-jsonschema --schema-root-type navidrome://plugins/manifest=PluginManifest -p schema --output schema/manifest_gen.go schema/manifest.schema.json
+
+import (
+ _ "embed"
+ "encoding/json"
+ "fmt"
+ "os"
+ "path/filepath"
+
+ "github.com/navidrome/navidrome/plugins/schema"
+)
+
+// LoadManifest loads and parses the manifest.json file from the given plugin directory.
+// Returns the generated schema.PluginManifest type with full validation and type safety.
+func LoadManifest(pluginDir string) (*schema.PluginManifest, error) {
+ manifestPath := filepath.Join(pluginDir, "manifest.json")
+ data, err := os.ReadFile(manifestPath)
+ if err != nil {
+ return nil, fmt.Errorf("failed to read manifest file: %w", err)
+ }
+
+ var manifest schema.PluginManifest
+ if err := json.Unmarshal(data, &manifest); err != nil {
+ return nil, fmt.Errorf("invalid manifest: %w", err)
+ }
+
+ return &manifest, nil
+}
diff --git a/plugins/manifest_permissions_test.go b/plugins/manifest_permissions_test.go
new file mode 100644
index 000000000..7a3df5f2d
--- /dev/null
+++ b/plugins/manifest_permissions_test.go
@@ -0,0 +1,526 @@
+package plugins
+
+import (
+ "context"
+ "encoding/json"
+ "os"
+ "path/filepath"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/core/metrics"
+ "github.com/navidrome/navidrome/plugins/schema"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+// Helper function to create test plugins with typed permissions
+func createTestPlugin(tempDir, name string, permissions schema.PluginManifestPermissions) string {
+ pluginDir := filepath.Join(tempDir, name)
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+
+ // Use the generated PluginManifest type directly - it handles JSON marshaling automatically
+ manifest := schema.PluginManifest{
+ Name: name,
+ Author: "Test Author",
+ Version: "1.0.0",
+ Description: "Test plugin for permissions",
+ Website: "https://test.navidrome.org/" + name,
+ Capabilities: []schema.PluginManifestCapabilitiesElem{
+ schema.PluginManifestCapabilitiesElemMetadataAgent,
+ },
+ Permissions: permissions,
+ }
+
+ // Marshal the typed manifest directly - gets all validation for free
+ manifestData, err := json.Marshal(manifest)
+ Expect(err).NotTo(HaveOccurred())
+
+ manifestPath := filepath.Join(pluginDir, "manifest.json")
+ Expect(os.WriteFile(manifestPath, manifestData, 0600)).To(Succeed())
+
+ // Create fake WASM file (since plugin discovery checks for it)
+ wasmPath := filepath.Join(pluginDir, "plugin.wasm")
+ Expect(os.WriteFile(wasmPath, []byte("fake wasm content"), 0600)).To(Succeed())
+
+ return pluginDir
+}
+
+var _ = Describe("Plugin Permissions", func() {
+ var (
+ mgr *managerImpl
+ tempDir string
+ ctx context.Context
+ )
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ ctx = context.Background()
+ mgr = createManager(nil, metrics.NewNoopInstance())
+ tempDir = GinkgoT().TempDir()
+ })
+
+ Describe("Permission Enforcement in createRuntime", func() {
+ It("should only load services specified in permissions", func() {
+ // Test with limited permissions using typed structs
+ permissions := schema.PluginManifestPermissions{
+ Http: &schema.PluginManifestPermissionsHttp{
+ Reason: "To fetch data from external APIs",
+ AllowedUrls: map[string][]schema.PluginManifestPermissionsHttpAllowedUrlsValueElem{
+ "*": {schema.PluginManifestPermissionsHttpAllowedUrlsValueElemWildcard},
+ },
+ AllowLocalNetwork: false,
+ },
+ Config: &schema.PluginManifestPermissionsConfig{
+ Reason: "To read configuration settings",
+ },
+ }
+
+ runtimeFunc := mgr.createRuntime("test-plugin", permissions)
+
+ // Create runtime to test service availability
+ runtime, err := runtimeFunc(ctx)
+ Expect(err).NotTo(HaveOccurred())
+ defer runtime.Close(ctx)
+
+ // The runtime was created successfully with the specified permissions
+ Expect(runtime).NotTo(BeNil())
+
+ // Note: The actual verification of which specific host functions are available
+ // would require introspecting the WASM runtime, which is complex.
+ // The key test is that the runtime creation succeeds with valid permissions.
+ })
+
+ It("should create runtime with empty permissions", func() {
+ permissions := schema.PluginManifestPermissions{}
+
+ runtimeFunc := mgr.createRuntime("empty-permissions-plugin", permissions)
+
+ runtime, err := runtimeFunc(ctx)
+ Expect(err).NotTo(HaveOccurred())
+ defer runtime.Close(ctx)
+
+ // Should succeed but with no host services available
+ Expect(runtime).NotTo(BeNil())
+ })
+
+ It("should handle all available permissions", func() {
+ // Test with all possible permissions using typed structs
+ permissions := schema.PluginManifestPermissions{
+ Http: &schema.PluginManifestPermissionsHttp{
+ Reason: "To fetch data from external APIs",
+ AllowedUrls: map[string][]schema.PluginManifestPermissionsHttpAllowedUrlsValueElem{
+ "*": {schema.PluginManifestPermissionsHttpAllowedUrlsValueElemWildcard},
+ },
+ AllowLocalNetwork: false,
+ },
+ Config: &schema.PluginManifestPermissionsConfig{
+ Reason: "To read configuration settings",
+ },
+ Scheduler: &schema.PluginManifestPermissionsScheduler{
+ Reason: "To schedule periodic tasks",
+ },
+ Websocket: &schema.PluginManifestPermissionsWebsocket{
+ Reason: "To handle real-time communication",
+ AllowedUrls: []string{"wss://api.example.com"},
+ AllowLocalNetwork: false,
+ },
+ Cache: &schema.PluginManifestPermissionsCache{
+ Reason: "To cache data and reduce API calls",
+ },
+ Artwork: &schema.PluginManifestPermissionsArtwork{
+ Reason: "To generate artwork URLs",
+ },
+ }
+
+ runtimeFunc := mgr.createRuntime("full-permissions-plugin", permissions)
+
+ runtime, err := runtimeFunc(ctx)
+ Expect(err).NotTo(HaveOccurred())
+ defer runtime.Close(ctx)
+
+ Expect(runtime).NotTo(BeNil())
+ })
+ })
+
+ Describe("Plugin Discovery with Permissions", func() {
+ BeforeEach(func() {
+ conf.Server.Plugins.Folder = tempDir
+ })
+
+ It("should discover plugin with valid permissions manifest", func() {
+ // Create plugin with http permission using typed structs
+ permissions := schema.PluginManifestPermissions{
+ Http: &schema.PluginManifestPermissionsHttp{
+ Reason: "To fetch metadata from external APIs",
+ AllowedUrls: map[string][]schema.PluginManifestPermissionsHttpAllowedUrlsValueElem{
+ "*": {schema.PluginManifestPermissionsHttpAllowedUrlsValueElemWildcard},
+ },
+ },
+ }
+ createTestPlugin(tempDir, "valid-plugin", permissions)
+
+ // Scan for plugins
+ mgr.ScanPlugins()
+
+ // Verify plugin was discovered (even without valid WASM)
+ pluginNames := mgr.PluginNames("MetadataAgent")
+ Expect(pluginNames).To(ContainElement("valid-plugin"))
+ })
+
+ It("should discover plugin with no permissions", func() {
+ // Create plugin with empty permissions using typed structs
+ permissions := schema.PluginManifestPermissions{}
+ createTestPlugin(tempDir, "no-perms-plugin", permissions)
+
+ mgr.ScanPlugins()
+
+ pluginNames := mgr.PluginNames("MetadataAgent")
+ Expect(pluginNames).To(ContainElement("no-perms-plugin"))
+ })
+
+ It("should discover plugin with multiple permissions", func() {
+ // Create plugin with multiple permissions using typed structs
+ permissions := schema.PluginManifestPermissions{
+ Http: &schema.PluginManifestPermissionsHttp{
+ Reason: "To fetch metadata from external APIs",
+ AllowedUrls: map[string][]schema.PluginManifestPermissionsHttpAllowedUrlsValueElem{
+ "*": {schema.PluginManifestPermissionsHttpAllowedUrlsValueElemWildcard},
+ },
+ },
+ Config: &schema.PluginManifestPermissionsConfig{
+ Reason: "To read plugin configuration settings",
+ },
+ Scheduler: &schema.PluginManifestPermissionsScheduler{
+ Reason: "To schedule periodic data updates",
+ },
+ }
+ createTestPlugin(tempDir, "multi-perms-plugin", permissions)
+
+ mgr.ScanPlugins()
+
+ pluginNames := mgr.PluginNames("MetadataAgent")
+ Expect(pluginNames).To(ContainElement("multi-perms-plugin"))
+ })
+ })
+
+ Describe("Existing Plugin Permissions", func() {
+ BeforeEach(func() {
+ // Use the testdata directory with updated plugins
+ conf.Server.Plugins.Folder = testDataDir
+ mgr.ScanPlugins()
+ })
+
+ It("should discover fake_scrobbler with empty permissions", func() {
+ scrobblerNames := mgr.PluginNames(CapabilityScrobbler)
+ Expect(scrobblerNames).To(ContainElement("fake_scrobbler"))
+ })
+
+ It("should discover multi_plugin with scheduler permissions", func() {
+ agentNames := mgr.PluginNames(CapabilityMetadataAgent)
+ Expect(agentNames).To(ContainElement("multi_plugin"))
+ })
+
+ It("should discover all test plugins successfully", func() {
+ // All test plugins should be discovered with their updated permissions
+ testPlugins := []struct {
+ name string
+ capability string
+ }{
+ {"fake_album_agent", CapabilityMetadataAgent},
+ {"fake_artist_agent", CapabilityMetadataAgent},
+ {"fake_scrobbler", CapabilityScrobbler},
+ {"multi_plugin", CapabilityMetadataAgent},
+ {"fake_init_service", CapabilityLifecycleManagement},
+ }
+
+ for _, testPlugin := range testPlugins {
+ pluginNames := mgr.PluginNames(testPlugin.capability)
+ Expect(pluginNames).To(ContainElement(testPlugin.name), "Plugin %s should be discovered", testPlugin.name)
+ }
+ })
+ })
+
+ Describe("Permission Validation", func() {
+ It("should enforce permissions are required in manifest", func() {
+ // Create a manifest JSON string without the permissions field
+ manifestContent := `{
+ "name": "test-plugin",
+ "author": "Test Author",
+ "version": "1.0.0",
+ "description": "A test plugin",
+ "website": "https://test.navidrome.org/test-plugin",
+ "capabilities": ["MetadataAgent"]
+ }`
+
+ manifestPath := filepath.Join(tempDir, "manifest.json")
+ err := os.WriteFile(manifestPath, []byte(manifestContent), 0600)
+ Expect(err).NotTo(HaveOccurred())
+
+ _, err = LoadManifest(tempDir)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("field permissions in PluginManifest: required"))
+ })
+
+ It("should allow unknown permission keys", func() {
+ // Create manifest with both known and unknown permission types
+ pluginDir := filepath.Join(tempDir, "unknown-perms")
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+
+ manifestContent := `{
+ "name": "unknown-perms",
+ "author": "Test Author",
+ "version": "1.0.0",
+ "description": "Manifest with unknown permissions",
+ "website": "https://test.navidrome.org/unknown-perms",
+ "capabilities": ["MetadataAgent"],
+ "permissions": {
+ "http": {
+ "reason": "To fetch data from external APIs",
+ "allowedUrls": {
+ "*": ["*"]
+ }
+ },
+ "unknown": {
+ "customField": "customValue"
+ }
+ }
+ }`
+
+ Expect(os.WriteFile(filepath.Join(pluginDir, "manifest.json"), []byte(manifestContent), 0600)).To(Succeed())
+
+ // Test manifest loading directly - should succeed even with unknown permissions
+ loadedManifest, err := LoadManifest(pluginDir)
+ Expect(err).NotTo(HaveOccurred())
+ Expect(loadedManifest).NotTo(BeNil())
+ // With typed permissions, we check the specific fields
+ Expect(loadedManifest.Permissions.Http).NotTo(BeNil())
+ Expect(loadedManifest.Permissions.Http.Reason).To(Equal("To fetch data from external APIs"))
+ // The key point is that the manifest loads successfully despite unknown permissions
+ // The actual handling of AdditionalProperties depends on the JSON schema implementation
+ })
+ })
+
+ Describe("Runtime Pool with Permissions", func() {
+ It("should create separate runtimes for different permission sets", func() {
+ // Create two different permission sets using typed structs
+ permissions1 := schema.PluginManifestPermissions{
+ Http: &schema.PluginManifestPermissionsHttp{
+ Reason: "To fetch data from external APIs",
+ AllowedUrls: map[string][]schema.PluginManifestPermissionsHttpAllowedUrlsValueElem{
+ "*": {schema.PluginManifestPermissionsHttpAllowedUrlsValueElemWildcard},
+ },
+ AllowLocalNetwork: false,
+ },
+ }
+ permissions2 := schema.PluginManifestPermissions{
+ Config: &schema.PluginManifestPermissionsConfig{
+ Reason: "To read configuration settings",
+ },
+ }
+
+ runtimeFunc1 := mgr.createRuntime("plugin1", permissions1)
+ runtimeFunc2 := mgr.createRuntime("plugin2", permissions2)
+
+ runtime1, err1 := runtimeFunc1(ctx)
+ Expect(err1).NotTo(HaveOccurred())
+ defer runtime1.Close(ctx)
+
+ runtime2, err2 := runtimeFunc2(ctx)
+ Expect(err2).NotTo(HaveOccurred())
+ defer runtime2.Close(ctx)
+
+ // Should be different runtime instances
+ Expect(runtime1).NotTo(BeIdenticalTo(runtime2))
+ })
+ })
+
+ Describe("Permission System Integration", func() {
+ It("should successfully validate manifests with permissions", func() {
+ // Create a valid manifest with permissions
+ pluginDir := filepath.Join(tempDir, "valid-manifest")
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+
+ manifestContent := `{
+ "name": "valid-manifest",
+ "author": "Test Author",
+ "version": "1.0.0",
+ "description": "Valid manifest with permissions",
+ "website": "https://test.navidrome.org/valid-manifest",
+ "capabilities": ["MetadataAgent"],
+ "permissions": {
+ "http": {
+ "reason": "To fetch metadata from external APIs",
+ "allowedUrls": {
+ "*": ["*"]
+ }
+ },
+ "config": {
+ "reason": "To read plugin configuration settings"
+ }
+ }
+ }`
+
+ Expect(os.WriteFile(filepath.Join(pluginDir, "manifest.json"), []byte(manifestContent), 0600)).To(Succeed())
+
+ // Load the manifest - should succeed
+ manifest, err := LoadManifest(pluginDir)
+ Expect(err).NotTo(HaveOccurred())
+ Expect(manifest).NotTo(BeNil())
+ // With typed permissions, check the specific permission fields
+ Expect(manifest.Permissions.Http).NotTo(BeNil())
+ Expect(manifest.Permissions.Http.Reason).To(Equal("To fetch metadata from external APIs"))
+ Expect(manifest.Permissions.Config).NotTo(BeNil())
+ Expect(manifest.Permissions.Config.Reason).To(Equal("To read plugin configuration settings"))
+ })
+
+ It("should track which services are requested per plugin", func() {
+ // Test that different plugins can have different permission sets
+ permissions1 := schema.PluginManifestPermissions{
+ Http: &schema.PluginManifestPermissionsHttp{
+ Reason: "To fetch data from external APIs",
+ AllowedUrls: map[string][]schema.PluginManifestPermissionsHttpAllowedUrlsValueElem{
+ "*": {schema.PluginManifestPermissionsHttpAllowedUrlsValueElemWildcard},
+ },
+ AllowLocalNetwork: false,
+ },
+ Config: &schema.PluginManifestPermissionsConfig{
+ Reason: "To read configuration settings",
+ },
+ }
+ permissions2 := schema.PluginManifestPermissions{
+ Scheduler: &schema.PluginManifestPermissionsScheduler{
+ Reason: "To schedule periodic tasks",
+ },
+ Config: &schema.PluginManifestPermissionsConfig{
+ Reason: "To read configuration for scheduler",
+ },
+ }
+ permissions3 := schema.PluginManifestPermissions{} // Empty permissions
+
+ createTestPlugin(tempDir, "plugin-with-http", permissions1)
+ createTestPlugin(tempDir, "plugin-with-scheduler", permissions2)
+ createTestPlugin(tempDir, "plugin-with-none", permissions3)
+
+ conf.Server.Plugins.Folder = tempDir
+ mgr.ScanPlugins()
+
+ // All should be discovered
+ pluginNames := mgr.PluginNames(CapabilityMetadataAgent)
+ Expect(pluginNames).To(ContainElement("plugin-with-http"))
+ Expect(pluginNames).To(ContainElement("plugin-with-scheduler"))
+ Expect(pluginNames).To(ContainElement("plugin-with-none"))
+ })
+ })
+
+ Describe("Runtime Service Access Control", func() {
+ It("should successfully create runtime with permitted services", func() {
+ // Create runtime with HTTP permission using typed struct
+ permissions := schema.PluginManifestPermissions{
+ Http: &schema.PluginManifestPermissionsHttp{
+ Reason: "To fetch data from external APIs",
+ AllowedUrls: map[string][]schema.PluginManifestPermissionsHttpAllowedUrlsValueElem{
+ "*": {schema.PluginManifestPermissionsHttpAllowedUrlsValueElemWildcard},
+ },
+ AllowLocalNetwork: false,
+ },
+ }
+
+ runtimeFunc := mgr.createRuntime("http-only-plugin", permissions)
+ runtime, err := runtimeFunc(ctx)
+ Expect(err).NotTo(HaveOccurred())
+ defer runtime.Close(ctx)
+
+ // Runtime should be created successfully - host functions are loaded during runtime creation
+ Expect(runtime).NotTo(BeNil())
+ })
+
+ It("should successfully create runtime with multiple permitted services", func() {
+ // Create runtime with multiple permissions using typed structs
+ permissions := schema.PluginManifestPermissions{
+ Http: &schema.PluginManifestPermissionsHttp{
+ Reason: "To fetch data from external APIs",
+ AllowedUrls: map[string][]schema.PluginManifestPermissionsHttpAllowedUrlsValueElem{
+ "*": {schema.PluginManifestPermissionsHttpAllowedUrlsValueElemWildcard},
+ },
+ AllowLocalNetwork: false,
+ },
+ Config: &schema.PluginManifestPermissionsConfig{
+ Reason: "To read configuration settings",
+ },
+ Scheduler: &schema.PluginManifestPermissionsScheduler{
+ Reason: "To schedule periodic tasks",
+ },
+ }
+
+ runtimeFunc := mgr.createRuntime("multi-service-plugin", permissions)
+ runtime, err := runtimeFunc(ctx)
+ Expect(err).NotTo(HaveOccurred())
+ defer runtime.Close(ctx)
+
+ // Runtime should be created successfully
+ Expect(runtime).NotTo(BeNil())
+ })
+
+ It("should create runtime with no services when no permissions granted", func() {
+ // Create runtime with empty permissions using typed struct
+ emptyPermissions := schema.PluginManifestPermissions{}
+
+ runtimeFunc := mgr.createRuntime("no-service-plugin", emptyPermissions)
+ runtime, err := runtimeFunc(ctx)
+ Expect(err).NotTo(HaveOccurred())
+ defer runtime.Close(ctx)
+
+ // Runtime should still be created, but with no host services
+ Expect(runtime).NotTo(BeNil())
+ })
+
+ It("should demonstrate secure-by-default behavior", func() {
+ // Test that default (empty permissions) provides no services
+ defaultPermissions := schema.PluginManifestPermissions{}
+ runtimeFunc := mgr.createRuntime("default-plugin", defaultPermissions)
+ runtime, err := runtimeFunc(ctx)
+ Expect(err).NotTo(HaveOccurred())
+ defer runtime.Close(ctx)
+
+ // Runtime should be created but with no host services
+ Expect(runtime).NotTo(BeNil())
+ })
+
+ It("should test permission enforcement by simulating unauthorized service access", func() {
+ // This test demonstrates that plugins would fail at runtime when trying to call
+ // host functions they don't have permission for, since those functions are simply
+ // not loaded into the WASM runtime environment.
+
+ // Create two different runtimes with different permissions using typed structs
+ httpOnlyPermissions := schema.PluginManifestPermissions{
+ Http: &schema.PluginManifestPermissionsHttp{
+ Reason: "To fetch data from external APIs",
+ AllowedUrls: map[string][]schema.PluginManifestPermissionsHttpAllowedUrlsValueElem{
+ "*": {schema.PluginManifestPermissionsHttpAllowedUrlsValueElemWildcard},
+ },
+ AllowLocalNetwork: false,
+ },
+ }
+ configOnlyPermissions := schema.PluginManifestPermissions{
+ Config: &schema.PluginManifestPermissionsConfig{
+ Reason: "To read configuration settings",
+ },
+ }
+
+ httpRuntime, err := mgr.createRuntime("http-only", httpOnlyPermissions)(ctx)
+ Expect(err).NotTo(HaveOccurred())
+ defer httpRuntime.Close(ctx)
+
+ configRuntime, err := mgr.createRuntime("config-only", configOnlyPermissions)(ctx)
+ Expect(err).NotTo(HaveOccurred())
+ defer configRuntime.Close(ctx)
+
+ // Both runtimes should be created successfully, but they will have different
+ // sets of host functions available. A plugin trying to call unauthorized
+ // functions would get "function not found" errors during instantiation or execution.
+ Expect(httpRuntime).NotTo(BeNil())
+ Expect(configRuntime).NotTo(BeNil())
+ })
+ })
+})
diff --git a/plugins/manifest_test.go b/plugins/manifest_test.go
new file mode 100644
index 000000000..2ec3edd19
--- /dev/null
+++ b/plugins/manifest_test.go
@@ -0,0 +1,144 @@
+package plugins
+
+import (
+ "os"
+ "path/filepath"
+
+ "github.com/navidrome/navidrome/plugins/schema"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("Plugin Manifest", func() {
+ var tempDir string
+
+ BeforeEach(func() {
+ tempDir = GinkgoT().TempDir()
+ })
+
+ It("should load and parse a valid manifest", func() {
+ manifestPath := filepath.Join(tempDir, "manifest.json")
+ manifestContent := []byte(`{
+ "name": "test-plugin",
+ "author": "Test Author",
+ "version": "1.0.0",
+ "description": "A test plugin",
+ "website": "https://test.navidrome.org/test-plugin",
+ "capabilities": ["MetadataAgent", "Scrobbler"],
+ "permissions": {
+ "http": {
+ "reason": "To fetch metadata",
+ "allowedUrls": {
+ "https://api.example.com/*": ["GET"]
+ }
+ }
+ }
+ }`)
+
+ err := os.WriteFile(manifestPath, manifestContent, 0600)
+ Expect(err).NotTo(HaveOccurred())
+
+ manifest, err := LoadManifest(tempDir)
+ Expect(err).NotTo(HaveOccurred())
+ Expect(manifest).NotTo(BeNil())
+ Expect(manifest.Name).To(Equal("test-plugin"))
+ Expect(manifest.Author).To(Equal("Test Author"))
+ Expect(manifest.Version).To(Equal("1.0.0"))
+ Expect(manifest.Description).To(Equal("A test plugin"))
+ Expect(manifest.Capabilities).To(HaveLen(2))
+ Expect(manifest.Capabilities[0]).To(Equal(schema.PluginManifestCapabilitiesElemMetadataAgent))
+ Expect(manifest.Capabilities[1]).To(Equal(schema.PluginManifestCapabilitiesElemScrobbler))
+ Expect(manifest.Permissions.Http).NotTo(BeNil())
+ Expect(manifest.Permissions.Http.Reason).To(Equal("To fetch metadata"))
+ })
+
+ It("should fail with proper error for non-existent manifest", func() {
+ _, err := LoadManifest(filepath.Join(tempDir, "non-existent"))
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("failed to read manifest file"))
+ })
+
+ It("should fail with JSON parse error for invalid JSON", func() {
+ // Create invalid JSON
+ invalidJSON := `{
+ "name": "test-plugin",
+ "author": "Test Author"
+ "version": "1.0.0"
+ "description": "A test plugin",
+ "capabilities": ["MetadataAgent"],
+ "permissions": {}
+ }`
+
+ pluginDir := filepath.Join(tempDir, "invalid-json")
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+ Expect(os.WriteFile(filepath.Join(pluginDir, "manifest.json"), []byte(invalidJSON), 0600)).To(Succeed())
+
+ // Test validation fails
+ _, err := LoadManifest(pluginDir)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("invalid manifest"))
+ })
+
+ It("should validate manifest against schema with detailed error for missing required field", func() {
+ // Create manifest missing required name field
+ manifestContent := `{
+ "author": "Test Author",
+ "version": "1.0.0",
+ "description": "A test plugin",
+ "website": "https://test.navidrome.org/test-plugin",
+ "capabilities": ["MetadataAgent"],
+ "permissions": {}
+ }`
+
+ pluginDir := filepath.Join(tempDir, "test-plugin")
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+ Expect(os.WriteFile(filepath.Join(pluginDir, "manifest.json"), []byte(manifestContent), 0600)).To(Succeed())
+
+ _, err := LoadManifest(pluginDir)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("field name in PluginManifest: required"))
+ })
+
+ It("should validate manifest with wrong capability type", func() {
+ // Create manifest with invalid capability
+ manifestContent := `{
+ "name": "test-plugin",
+ "author": "Test Author",
+ "version": "1.0.0",
+ "description": "A test plugin",
+ "website": "https://test.navidrome.org/test-plugin",
+ "capabilities": ["UnsupportedService"],
+ "permissions": {}
+ }`
+
+ pluginDir := filepath.Join(tempDir, "test-plugin")
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+ Expect(os.WriteFile(filepath.Join(pluginDir, "manifest.json"), []byte(manifestContent), 0600)).To(Succeed())
+
+ _, err := LoadManifest(pluginDir)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("invalid value"))
+ Expect(err.Error()).To(ContainSubstring("UnsupportedService"))
+ })
+
+ It("should validate manifest with empty capabilities array", func() {
+ // Create manifest with empty capabilities array
+ manifestContent := `{
+ "name": "test-plugin",
+ "author": "Test Author",
+ "version": "1.0.0",
+ "description": "A test plugin",
+ "website": "https://test.navidrome.org/test-plugin",
+ "capabilities": [],
+ "permissions": {}
+ }`
+
+ pluginDir := filepath.Join(tempDir, "test-plugin")
+ Expect(os.MkdirAll(pluginDir, 0755)).To(Succeed())
+ Expect(os.WriteFile(filepath.Join(pluginDir, "manifest.json"), []byte(manifestContent), 0600)).To(Succeed())
+
+ _, err := LoadManifest(pluginDir)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("field capabilities length: must be >= 1"))
+ })
+})
diff --git a/plugins/package.go b/plugins/package.go
new file mode 100644
index 000000000..5273b0431
--- /dev/null
+++ b/plugins/package.go
@@ -0,0 +1,177 @@
+package plugins
+
+import (
+ "archive/zip"
+ "encoding/json"
+ "errors"
+ "fmt"
+ "io"
+ "os"
+ "path/filepath"
+ "strings"
+
+ "github.com/navidrome/navidrome/plugins/schema"
+)
+
+// PluginPackage represents a Navidrome Plugin Package (.ndp file)
+type PluginPackage struct {
+ ManifestJSON []byte
+ Manifest *schema.PluginManifest
+ WasmBytes []byte
+ Docs map[string][]byte
+}
+
+// ExtractPackage extracts a .ndp file to the target directory
+func ExtractPackage(ndpPath, targetDir string) error {
+ r, err := zip.OpenReader(ndpPath)
+ if err != nil {
+ return fmt.Errorf("error opening .ndp file: %w", err)
+ }
+ defer r.Close()
+
+ // Create target directory if it doesn't exist
+ if err := os.MkdirAll(targetDir, 0755); err != nil {
+ return fmt.Errorf("error creating plugin directory: %w", err)
+ }
+
+ // Define a reasonable size limit for plugin files to prevent decompression bombs
+ const maxFileSize = 10 * 1024 * 1024 // 10 MB limit
+
+ // Extract all files from the zip
+ for _, f := range r.File {
+ // Skip directories (they will be created as needed)
+ if f.FileInfo().IsDir() {
+ continue
+ }
+
+ // Create the file path for extraction
+ // Validate the file name to prevent directory traversal or absolute paths
+ if strings.Contains(f.Name, "..") || filepath.IsAbs(f.Name) {
+ return fmt.Errorf("illegal file path in plugin package: %s", f.Name)
+ }
+
+ // Create the file path for extraction
+ targetPath := filepath.Join(targetDir, f.Name) // #nosec G305
+
+ // Clean the path to prevent directory traversal.
+ cleanedPath := filepath.Clean(targetPath)
+ // Ensure the cleaned path is still within the target directory.
+ // We resolve both paths to absolute paths to be sure.
+ absTargetDir, err := filepath.Abs(targetDir)
+ if err != nil {
+ return fmt.Errorf("failed to resolve target directory path: %w", err)
+ }
+ absTargetPath, err := filepath.Abs(cleanedPath)
+ if err != nil {
+ return fmt.Errorf("failed to resolve extracted file path: %w", err)
+ }
+ if !strings.HasPrefix(absTargetPath, absTargetDir+string(os.PathSeparator)) && absTargetPath != absTargetDir {
+ return fmt.Errorf("illegal file path in plugin package: %s", f.Name)
+ }
+
+ // Open the file inside the zip
+ rc, err := f.Open()
+ if err != nil {
+ return fmt.Errorf("error opening file in plugin package: %w", err)
+ }
+
+ // Create parent directories if they don't exist
+ if err := os.MkdirAll(filepath.Dir(targetPath), 0755); err != nil {
+ rc.Close()
+ return fmt.Errorf("error creating directory structure: %w", err)
+ }
+
+ // Create the file
+ outFile, err := os.Create(targetPath)
+ if err != nil {
+ rc.Close()
+ return fmt.Errorf("error creating extracted file: %w", err)
+ }
+
+ // Copy the file contents with size limit
+ if _, err := io.CopyN(outFile, rc, maxFileSize); err != nil && !errors.Is(err, io.EOF) {
+ outFile.Close()
+ rc.Close()
+ if errors.Is(err, io.ErrUnexpectedEOF) { // File size exceeds limit
+ return fmt.Errorf("error extracting file: size exceeds limit (%d bytes) for %s", maxFileSize, f.Name)
+ }
+ return fmt.Errorf("error writing extracted file: %w", err)
+ }
+
+ outFile.Close()
+ rc.Close()
+
+ // Set appropriate file permissions (0600 - readable only by owner)
+ if err := os.Chmod(targetPath, 0600); err != nil {
+ return fmt.Errorf("error setting permissions on extracted file: %w", err)
+ }
+ }
+
+ return nil
+}
+
+// LoadPackage loads and validates an .ndp file without extracting it
+func LoadPackage(ndpPath string) (*PluginPackage, error) {
+ r, err := zip.OpenReader(ndpPath)
+ if err != nil {
+ return nil, fmt.Errorf("error opening .ndp file: %w", err)
+ }
+ defer r.Close()
+
+ pkg := &PluginPackage{
+ Docs: make(map[string][]byte),
+ }
+
+ // Required files
+ var hasManifest, hasWasm bool
+
+ // Read all files in the zip
+ for _, f := range r.File {
+ // Skip directories
+ if f.FileInfo().IsDir() {
+ continue
+ }
+
+ // Get file content
+ rc, err := f.Open()
+ if err != nil {
+ return nil, fmt.Errorf("error opening file in plugin package: %w", err)
+ }
+
+ content, err := io.ReadAll(rc)
+ rc.Close()
+ if err != nil {
+ return nil, fmt.Errorf("error reading file in plugin package: %w", err)
+ }
+
+ // Process based on file name
+ switch strings.ToLower(f.Name) {
+ case "manifest.json":
+ pkg.ManifestJSON = content
+ hasManifest = true
+ case "plugin.wasm":
+ pkg.WasmBytes = content
+ hasWasm = true
+ default:
+ // Store other files as documentation
+ pkg.Docs[f.Name] = content
+ }
+ }
+
+ // Ensure required files exist
+ if !hasManifest {
+ return nil, fmt.Errorf("plugin package missing required manifest.json")
+ }
+ if !hasWasm {
+ return nil, fmt.Errorf("plugin package missing required plugin.wasm")
+ }
+
+ // Parse and validate the manifest
+ var manifest schema.PluginManifest
+ if err := json.Unmarshal(pkg.ManifestJSON, &manifest); err != nil {
+ return nil, fmt.Errorf("invalid manifest: %w", err)
+ }
+
+ pkg.Manifest = &manifest
+ return pkg, nil
+}
diff --git a/plugins/package_test.go b/plugins/package_test.go
new file mode 100644
index 000000000..8ff4b354a
--- /dev/null
+++ b/plugins/package_test.go
@@ -0,0 +1,116 @@
+package plugins
+
+import (
+ "archive/zip"
+ "os"
+ "path/filepath"
+
+ "github.com/navidrome/navidrome/plugins/schema"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("Plugin Package", func() {
+ var tempDir string
+ var ndpPath string
+
+ BeforeEach(func() {
+ tempDir = GinkgoT().TempDir()
+
+ // Create a test .ndp file
+ ndpPath = filepath.Join(tempDir, "test-plugin.ndp")
+
+ // Create the required plugin files
+ manifestContent := []byte(`{
+ "name": "test-plugin",
+ "author": "Test Author",
+ "version": "1.0.0",
+ "description": "A test plugin",
+ "website": "https://test.navidrome.org/test-plugin",
+ "capabilities": ["MetadataAgent"],
+ "permissions": {}
+ }`)
+
+ wasmContent := []byte("dummy wasm content")
+ readmeContent := []byte("# Test Plugin\nThis is a test plugin")
+
+ // Create the zip file
+ zipFile, err := os.Create(ndpPath)
+ Expect(err).NotTo(HaveOccurred())
+ defer zipFile.Close()
+
+ zipWriter := zip.NewWriter(zipFile)
+ defer zipWriter.Close()
+
+ // Add manifest.json
+ manifestWriter, err := zipWriter.Create("manifest.json")
+ Expect(err).NotTo(HaveOccurred())
+ _, err = manifestWriter.Write(manifestContent)
+ Expect(err).NotTo(HaveOccurred())
+
+ // Add plugin.wasm
+ wasmWriter, err := zipWriter.Create("plugin.wasm")
+ Expect(err).NotTo(HaveOccurred())
+ _, err = wasmWriter.Write(wasmContent)
+ Expect(err).NotTo(HaveOccurred())
+
+ // Add README.md
+ readmeWriter, err := zipWriter.Create("README.md")
+ Expect(err).NotTo(HaveOccurred())
+ _, err = readmeWriter.Write(readmeContent)
+ Expect(err).NotTo(HaveOccurred())
+ })
+
+ It("should load and validate a plugin package", func() {
+ pkg, err := LoadPackage(ndpPath)
+ Expect(err).NotTo(HaveOccurred())
+ Expect(pkg).NotTo(BeNil())
+
+ // Check manifest was parsed
+ Expect(pkg.Manifest).NotTo(BeNil())
+ Expect(pkg.Manifest.Name).To(Equal("test-plugin"))
+ Expect(pkg.Manifest.Author).To(Equal("Test Author"))
+ Expect(pkg.Manifest.Version).To(Equal("1.0.0"))
+ Expect(pkg.Manifest.Description).To(Equal("A test plugin"))
+ Expect(pkg.Manifest.Capabilities).To(HaveLen(1))
+ Expect(pkg.Manifest.Capabilities[0]).To(Equal(schema.PluginManifestCapabilitiesElemMetadataAgent))
+
+ // Check WASM file was loaded
+ Expect(pkg.WasmBytes).NotTo(BeEmpty())
+
+ // Check docs were loaded
+ Expect(pkg.Docs).To(HaveKey("README.md"))
+ })
+
+ It("should extract a plugin package to a directory", func() {
+ targetDir := filepath.Join(tempDir, "extracted")
+
+ err := ExtractPackage(ndpPath, targetDir)
+ Expect(err).NotTo(HaveOccurred())
+
+ // Check files were extracted
+ Expect(filepath.Join(targetDir, "manifest.json")).To(BeARegularFile())
+ Expect(filepath.Join(targetDir, "plugin.wasm")).To(BeARegularFile())
+ Expect(filepath.Join(targetDir, "README.md")).To(BeARegularFile())
+ })
+
+ It("should fail to load an invalid package", func() {
+ // Create an invalid package (missing required files)
+ invalidPath := filepath.Join(tempDir, "invalid.ndp")
+ zipFile, err := os.Create(invalidPath)
+ Expect(err).NotTo(HaveOccurred())
+
+ zipWriter := zip.NewWriter(zipFile)
+ // Only add a README, missing manifest and wasm
+ readmeWriter, err := zipWriter.Create("README.md")
+ Expect(err).NotTo(HaveOccurred())
+ _, err = readmeWriter.Write([]byte("Invalid package"))
+ Expect(err).NotTo(HaveOccurred())
+ zipWriter.Close()
+ zipFile.Close()
+
+ // Test loading fails
+ _, err = LoadPackage(invalidPath)
+ Expect(err).To(HaveOccurred())
+ })
+})
diff --git a/plugins/plugin_lifecycle_manager.go b/plugins/plugin_lifecycle_manager.go
new file mode 100644
index 000000000..e00e7e5f3
--- /dev/null
+++ b/plugins/plugin_lifecycle_manager.go
@@ -0,0 +1,95 @@
+package plugins
+
+import (
+ "context"
+ "maps"
+ "sync"
+ "time"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/consts"
+ "github.com/navidrome/navidrome/core/metrics"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/plugins/api"
+)
+
+// pluginLifecycleManager tracks which plugins have been initialized and manages their lifecycle
+type pluginLifecycleManager struct {
+ plugins sync.Map // string -> bool
+ config map[string]map[string]string
+ metrics metrics.Metrics
+}
+
+// newPluginLifecycleManager creates a new plugin lifecycle manager
+func newPluginLifecycleManager(metrics metrics.Metrics) *pluginLifecycleManager {
+ config := maps.Clone(conf.Server.PluginConfig)
+ return &pluginLifecycleManager{
+ config: config,
+ metrics: metrics,
+ }
+}
+
+// isInitialized checks if a plugin has been initialized
+func (m *pluginLifecycleManager) isInitialized(plugin *plugin) bool {
+ key := plugin.ID + consts.Zwsp + plugin.Manifest.Version
+ value, exists := m.plugins.Load(key)
+ return exists && value.(bool)
+}
+
+// markInitialized marks a plugin as initialized
+func (m *pluginLifecycleManager) markInitialized(plugin *plugin) {
+ key := plugin.ID + consts.Zwsp + plugin.Manifest.Version
+ m.plugins.Store(key, true)
+}
+
+// clearInitialized removes the initialization state of a plugin
+func (m *pluginLifecycleManager) clearInitialized(plugin *plugin) {
+ key := plugin.ID + consts.Zwsp + plugin.Manifest.Version
+ m.plugins.Delete(key)
+}
+
+// callOnInit calls the OnInit method on a plugin that implements LifecycleManagement
+func (m *pluginLifecycleManager) callOnInit(plugin *plugin) error {
+ ctx := context.Background()
+ log.Debug("Initializing plugin", "name", plugin.ID)
+ start := time.Now()
+
+ // Create LifecycleManagement plugin instance
+ loader, err := api.NewLifecycleManagementPlugin(ctx, api.WazeroRuntime(plugin.Runtime), api.WazeroModuleConfig(plugin.ModConfig))
+ if loader == nil || err != nil {
+ log.Error("Error creating LifecycleManagement plugin", "plugin", plugin.ID, err)
+ return err
+ }
+
+ initPlugin, err := loader.Load(ctx, plugin.WasmPath)
+ if err != nil {
+ log.Error("Error loading LifecycleManagement plugin", "plugin", plugin.ID, "path", plugin.WasmPath, err)
+ return err
+ }
+ defer initPlugin.Close(ctx)
+
+ // Prepare the request with plugin-specific configuration
+ req := &api.InitRequest{}
+
+ // Add plugin configuration if available
+ if m.config != nil {
+ if pluginConfig, ok := m.config[plugin.ID]; ok && len(pluginConfig) > 0 {
+ req.Config = maps.Clone(pluginConfig)
+ log.Debug("Passing configuration to plugin", "plugin", plugin.ID, "configKeys", len(pluginConfig))
+ }
+ }
+
+ // Call OnInit
+ callStart := time.Now()
+ _, err = checkErr(initPlugin.OnInit(ctx, req))
+ m.metrics.RecordPluginRequest(ctx, plugin.ID, "OnInit", err == nil, time.Since(callStart).Milliseconds())
+ if err != nil {
+ log.Error("Error initializing plugin", "plugin", plugin.ID, "elapsed", time.Since(start), err)
+ return err
+ }
+
+ // Mark the plugin as initialized
+ m.markInitialized(plugin)
+ log.Debug("Plugin initialized successfully", "plugin", plugin.ID, "elapsed", time.Since(start))
+ return nil
+}
diff --git a/plugins/plugin_lifecycle_manager_test.go b/plugins/plugin_lifecycle_manager_test.go
new file mode 100644
index 000000000..800630ce9
--- /dev/null
+++ b/plugins/plugin_lifecycle_manager_test.go
@@ -0,0 +1,166 @@
+package plugins
+
+import (
+ "github.com/navidrome/navidrome/consts"
+ "github.com/navidrome/navidrome/core/metrics"
+ "github.com/navidrome/navidrome/plugins/schema"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+// Helper function to check if a plugin implements LifecycleManagement
+func hasInitService(info *plugin) bool {
+ for _, c := range info.Capabilities {
+ if c == CapabilityLifecycleManagement {
+ return true
+ }
+ }
+ return false
+}
+
+var _ = Describe("LifecycleManagement", func() {
+ Describe("Plugin Lifecycle Manager", func() {
+ var lifecycleManager *pluginLifecycleManager
+
+ BeforeEach(func() {
+ lifecycleManager = newPluginLifecycleManager(metrics.NewNoopInstance())
+ })
+
+ It("should track initialization state of plugins", func() {
+ // Create test plugins
+ plugin1 := &plugin{
+ ID: "test-plugin",
+ Capabilities: []string{CapabilityLifecycleManagement},
+ Manifest: &schema.PluginManifest{
+ Version: "1.0.0",
+ },
+ }
+
+ plugin2 := &plugin{
+ ID: "another-plugin",
+ Capabilities: []string{CapabilityLifecycleManagement},
+ Manifest: &schema.PluginManifest{
+ Version: "0.5.0",
+ },
+ }
+
+ // Initially, no plugins should be initialized
+ Expect(lifecycleManager.isInitialized(plugin1)).To(BeFalse())
+ Expect(lifecycleManager.isInitialized(plugin2)).To(BeFalse())
+
+ // Mark first plugin as initialized
+ lifecycleManager.markInitialized(plugin1)
+
+ // Check state
+ Expect(lifecycleManager.isInitialized(plugin1)).To(BeTrue())
+ Expect(lifecycleManager.isInitialized(plugin2)).To(BeFalse())
+
+ // Mark second plugin as initialized
+ lifecycleManager.markInitialized(plugin2)
+
+ // Both should be initialized now
+ Expect(lifecycleManager.isInitialized(plugin1)).To(BeTrue())
+ Expect(lifecycleManager.isInitialized(plugin2)).To(BeTrue())
+ })
+
+ It("should handle plugins with same name but different versions", func() {
+ plugin1 := &plugin{
+ ID: "test-plugin",
+ Capabilities: []string{CapabilityLifecycleManagement},
+ Manifest: &schema.PluginManifest{
+ Version: "1.0.0",
+ },
+ }
+
+ plugin2 := &plugin{
+ ID: "test-plugin", // Same name
+ Capabilities: []string{CapabilityLifecycleManagement},
+ Manifest: &schema.PluginManifest{
+ Version: "2.0.0", // Different version
+ },
+ }
+
+ // Mark v1 as initialized
+ lifecycleManager.markInitialized(plugin1)
+
+ // v1 should be initialized but not v2
+ Expect(lifecycleManager.isInitialized(plugin1)).To(BeTrue())
+ Expect(lifecycleManager.isInitialized(plugin2)).To(BeFalse())
+
+ // Mark v2 as initialized
+ lifecycleManager.markInitialized(plugin2)
+
+ // Both versions should be initialized now
+ Expect(lifecycleManager.isInitialized(plugin1)).To(BeTrue())
+ Expect(lifecycleManager.isInitialized(plugin2)).To(BeTrue())
+
+ // Verify the keys used for tracking
+ key1 := plugin1.ID + consts.Zwsp + plugin1.Manifest.Version
+ key2 := plugin1.ID + consts.Zwsp + plugin2.Manifest.Version
+ _, exists1 := lifecycleManager.plugins.Load(key1)
+ _, exists2 := lifecycleManager.plugins.Load(key2)
+ Expect(exists1).To(BeTrue())
+ Expect(exists2).To(BeTrue())
+ Expect(key1).NotTo(Equal(key2))
+ })
+
+ It("should only consider plugins that implement LifecycleManagement", func() {
+ // Plugin that implements LifecycleManagement
+ initPlugin := &plugin{
+ ID: "init-plugin",
+ Capabilities: []string{CapabilityLifecycleManagement},
+ Manifest: &schema.PluginManifest{
+ Version: "1.0.0",
+ },
+ }
+
+ // Plugin that doesn't implement LifecycleManagement
+ regularPlugin := &plugin{
+ ID: "regular-plugin",
+ Capabilities: []string{"MetadataAgent"},
+ Manifest: &schema.PluginManifest{
+ Version: "1.0.0",
+ },
+ }
+
+ // Check if plugins can be initialized
+ Expect(hasInitService(initPlugin)).To(BeTrue())
+ Expect(hasInitService(regularPlugin)).To(BeFalse())
+ })
+
+ It("should properly construct the plugin key", func() {
+ plugin := &plugin{
+ ID: "test-plugin",
+ Manifest: &schema.PluginManifest{
+ Version: "1.0.0",
+ },
+ }
+
+ expectedKey := "test-plugin" + consts.Zwsp + "1.0.0"
+ actualKey := plugin.ID + consts.Zwsp + plugin.Manifest.Version
+
+ Expect(actualKey).To(Equal(expectedKey))
+ })
+
+ It("should clear initialization state when requested", func() {
+ plugin := &plugin{
+ ID: "test-plugin",
+ Capabilities: []string{CapabilityLifecycleManagement},
+ Manifest: &schema.PluginManifest{
+ Version: "1.0.0",
+ },
+ }
+
+ // Initially not initialized
+ Expect(lifecycleManager.isInitialized(plugin)).To(BeFalse())
+
+ // Mark as initialized
+ lifecycleManager.markInitialized(plugin)
+ Expect(lifecycleManager.isInitialized(plugin)).To(BeTrue())
+
+ // Clear initialization state
+ lifecycleManager.clearInitialized(plugin)
+ Expect(lifecycleManager.isInitialized(plugin)).To(BeFalse())
+ })
+ })
+})
diff --git a/plugins/plugins_suite_test.go b/plugins/plugins_suite_test.go
new file mode 100644
index 000000000..153426317
--- /dev/null
+++ b/plugins/plugins_suite_test.go
@@ -0,0 +1,32 @@
+package plugins
+
+import (
+ "os/exec"
+ "testing"
+
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+const testDataDir = "plugins/testdata"
+
+func TestPlugins(t *testing.T) {
+ tests.Init(t, false)
+ buildTestPlugins(t, testDataDir)
+ log.SetLevel(log.LevelFatal)
+ RegisterFailHandler(Fail)
+ RunSpecs(t, "Plugins Suite")
+}
+
+func buildTestPlugins(t *testing.T, path string) {
+ t.Helper()
+ t.Logf("[BeforeSuite] Current working directory: %s", path)
+ cmd := exec.Command("make", "-C", path)
+ out, err := cmd.CombinedOutput()
+ t.Logf("[BeforeSuite] Make output: %s", string(out))
+ if err != nil {
+ t.Fatalf("Failed to build test plugins: %v", err)
+ }
+}
diff --git a/plugins/runtime.go b/plugins/runtime.go
new file mode 100644
index 000000000..ee298e63d
--- /dev/null
+++ b/plugins/runtime.go
@@ -0,0 +1,626 @@
+package plugins
+
+import (
+ "context"
+ "crypto/md5"
+ "fmt"
+ "io/fs"
+ "maps"
+ "os"
+ "path/filepath"
+ "sort"
+ "sync"
+ "sync/atomic"
+ "time"
+
+ "github.com/dustin/go-humanize"
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/navidrome/navidrome/plugins/host/artwork"
+ "github.com/navidrome/navidrome/plugins/host/cache"
+ "github.com/navidrome/navidrome/plugins/host/config"
+ "github.com/navidrome/navidrome/plugins/host/http"
+ "github.com/navidrome/navidrome/plugins/host/scheduler"
+ "github.com/navidrome/navidrome/plugins/host/subsonicapi"
+ "github.com/navidrome/navidrome/plugins/host/websocket"
+ "github.com/navidrome/navidrome/plugins/schema"
+ "github.com/tetratelabs/wazero"
+ wazeroapi "github.com/tetratelabs/wazero/api"
+ "github.com/tetratelabs/wazero/imports/wasi_snapshot_preview1"
+)
+
+const maxParallelCompilations = 2 // Limit to 2 concurrent compilations
+
+var (
+ compileSemaphore = make(chan struct{}, maxParallelCompilations)
+ compilationCache wazero.CompilationCache
+ cacheOnce sync.Once
+ runtimePool sync.Map // map[string]*cachingRuntime
+)
+
+// createRuntime returns a function that creates a new wazero runtime and instantiates the required host functions
+// based on the given plugin permissions
+func (m *managerImpl) createRuntime(pluginID string, permissions schema.PluginManifestPermissions) api.WazeroNewRuntime {
+ return func(ctx context.Context) (wazero.Runtime, error) {
+ // Check if runtime already exists
+ if rt, ok := runtimePool.Load(pluginID); ok {
+ log.Trace(ctx, "Using existing runtime", "plugin", pluginID, "runtime", fmt.Sprintf("%p", rt))
+ // Return a new wrapper for each call, so each instance gets its own module capture
+ return newScopedRuntime(rt.(wazero.Runtime)), nil
+ }
+
+ // Create new runtime with all the setup
+ cachingRT, err := m.createCachingRuntime(ctx, pluginID, permissions)
+ if err != nil {
+ return nil, err
+ }
+
+ // Use LoadOrStore to atomically check and store, preventing race conditions
+ if existing, loaded := runtimePool.LoadOrStore(pluginID, cachingRT); loaded {
+ // Another goroutine created the runtime first, close ours and return the existing one
+ log.Trace(ctx, "Race condition detected, using existing runtime", "plugin", pluginID, "runtime", fmt.Sprintf("%p", existing))
+ _ = cachingRT.Close(ctx)
+ return newScopedRuntime(existing.(wazero.Runtime)), nil
+ }
+
+ log.Trace(ctx, "Created new runtime", "plugin", pluginID, "runtime", fmt.Sprintf("%p", cachingRT))
+ return newScopedRuntime(cachingRT), nil
+ }
+}
+
+// createCachingRuntime handles the complex logic of setting up a new cachingRuntime
+func (m *managerImpl) createCachingRuntime(ctx context.Context, pluginID string, permissions schema.PluginManifestPermissions) (*cachingRuntime, error) {
+ // Get compilation cache
+ compCache, err := getCompilationCache()
+ if err != nil {
+ return nil, fmt.Errorf("failed to get compilation cache: %w", err)
+ }
+
+ // Create the runtime
+ runtimeConfig := wazero.NewRuntimeConfig().WithCompilationCache(compCache)
+ r := wazero.NewRuntimeWithConfig(ctx, runtimeConfig)
+ if _, err := wasi_snapshot_preview1.Instantiate(ctx, r); err != nil {
+ return nil, err
+ }
+
+ // Setup host services
+ if err := m.setupHostServices(ctx, r, pluginID, permissions); err != nil {
+ _ = r.Close(ctx)
+ return nil, err
+ }
+
+ return newCachingRuntime(r, pluginID), nil
+}
+
+// setupHostServices configures all the permitted host services for a plugin
+func (m *managerImpl) setupHostServices(ctx context.Context, r wazero.Runtime, pluginID string, permissions schema.PluginManifestPermissions) error {
+ // Define all available host services
+ type hostService struct {
+ name string
+ isPermitted bool
+ loadFunc func() (map[string]wazeroapi.FunctionDefinition, error)
+ }
+
+ // List of all available host services with their permissions and loading functions
+ availableServices := []hostService{
+ {"config", permissions.Config != nil, func() (map[string]wazeroapi.FunctionDefinition, error) {
+ return loadHostLibrary[config.ConfigService](ctx, config.Instantiate, &configServiceImpl{pluginID: pluginID})
+ }},
+ {"scheduler", permissions.Scheduler != nil, func() (map[string]wazeroapi.FunctionDefinition, error) {
+ return loadHostLibrary[scheduler.SchedulerService](ctx, scheduler.Instantiate, m.schedulerService.HostFunctions(pluginID))
+ }},
+ {"cache", permissions.Cache != nil, func() (map[string]wazeroapi.FunctionDefinition, error) {
+ return loadHostLibrary[cache.CacheService](ctx, cache.Instantiate, newCacheService(pluginID))
+ }},
+ {"artwork", permissions.Artwork != nil, func() (map[string]wazeroapi.FunctionDefinition, error) {
+ return loadHostLibrary[artwork.ArtworkService](ctx, artwork.Instantiate, &artworkServiceImpl{})
+ }},
+ {"http", permissions.Http != nil, func() (map[string]wazeroapi.FunctionDefinition, error) {
+ httpPerms, err := parseHTTPPermissions(permissions.Http)
+ if err != nil {
+ return nil, fmt.Errorf("invalid http permissions for plugin %s: %w", pluginID, err)
+ }
+ return loadHostLibrary[http.HttpService](ctx, http.Instantiate, &httpServiceImpl{
+ pluginID: pluginID,
+ permissions: httpPerms,
+ })
+ }},
+ {"websocket", permissions.Websocket != nil, func() (map[string]wazeroapi.FunctionDefinition, error) {
+ wsPerms, err := parseWebSocketPermissions(permissions.Websocket)
+ if err != nil {
+ return nil, fmt.Errorf("invalid websocket permissions for plugin %s: %w", pluginID, err)
+ }
+ return loadHostLibrary[websocket.WebSocketService](ctx, websocket.Instantiate, m.websocketService.HostFunctions(pluginID, wsPerms))
+ }},
+ {"subsonicapi", permissions.Subsonicapi != nil, func() (map[string]wazeroapi.FunctionDefinition, error) {
+ if router := m.subsonicRouter.Load(); router != nil {
+ service := newSubsonicAPIService(pluginID, m.subsonicRouter.Load(), m.ds, permissions.Subsonicapi)
+ return loadHostLibrary[subsonicapi.SubsonicAPIService](ctx, subsonicapi.Instantiate, service)
+ }
+ log.Error(ctx, "SubsonicAPI service requested but router not available", "plugin", pluginID)
+ return nil, fmt.Errorf("SubsonicAPI router not available for plugin %s", pluginID)
+ }},
+ }
+
+ // Load only permitted services
+ var grantedPermissions []string
+ var libraries []map[string]wazeroapi.FunctionDefinition
+ for _, service := range availableServices {
+ if service.isPermitted {
+ lib, err := service.loadFunc()
+ if err != nil {
+ return fmt.Errorf("error loading %s lib: %w", service.name, err)
+ }
+ libraries = append(libraries, lib)
+ grantedPermissions = append(grantedPermissions, service.name)
+ }
+ }
+ log.Trace(ctx, "Granting permissions for plugin", "plugin", pluginID, "permissions", grantedPermissions)
+
+ // Combine the permitted libraries
+ return combineLibraries(ctx, r, libraries...)
+}
+
+// purgeCacheBySize removes the oldest files in dir until its total size is
+// lower than or equal to maxSize. maxSize should be a human-readable string
+// like "10MB" or "200K". If parsing fails or maxSize is "0", the function is
+// a no-op.
+func purgeCacheBySize(dir, maxSize string) {
+ sizeLimit, err := humanize.ParseBytes(maxSize)
+ if err != nil || sizeLimit == 0 {
+ return
+ }
+
+ type fileInfo struct {
+ path string
+ size uint64
+ mod int64
+ }
+
+ var files []fileInfo
+ var total uint64
+
+ walk := func(path string, d fs.DirEntry, err error) error {
+ if err != nil {
+ log.Trace("Failed to access plugin cache entry", "path", path, err)
+ return nil //nolint:nilerr
+ }
+ if d.IsDir() {
+ return nil
+ }
+ info, err := d.Info()
+ if err != nil {
+ log.Trace("Failed to get file info for plugin cache entry", "path", path, err)
+ return nil //nolint:nilerr
+ }
+ files = append(files, fileInfo{
+ path: path,
+ size: uint64(info.Size()),
+ mod: info.ModTime().UnixMilli(),
+ })
+ total += uint64(info.Size())
+ return nil
+ }
+
+ if err := filepath.WalkDir(dir, walk); err != nil {
+ if !os.IsNotExist(err) {
+ log.Warn("Failed to traverse plugin cache directory", "path", dir, err)
+ }
+ return
+ }
+
+ log.Trace("Current plugin cache size", "path", dir, "size", humanize.Bytes(total), "sizeLimit", humanize.Bytes(sizeLimit))
+ if total <= sizeLimit {
+ return
+ }
+
+ log.Debug("Purging plugin cache", "path", dir, "sizeLimit", humanize.Bytes(sizeLimit), "currentSize", humanize.Bytes(total))
+ sort.Slice(files, func(i, j int) bool { return files[i].mod < files[j].mod })
+ for _, f := range files {
+ if total <= sizeLimit {
+ break
+ }
+ if err := os.Remove(f.path); err != nil {
+ log.Warn("Failed to remove plugin cache entry", "path", f.path, "size", humanize.Bytes(f.size), err)
+ continue
+ }
+ total -= f.size
+ log.Debug("Removed plugin cache entry", "path", f.path, "size", humanize.Bytes(f.size), "time", time.UnixMilli(f.mod), "remainingSize", humanize.Bytes(total))
+
+ // Remove empty parent directories
+ dirPath := filepath.Dir(f.path)
+ for dirPath != dir {
+ if err := os.Remove(dirPath); err != nil {
+ break
+ }
+ dirPath = filepath.Dir(dirPath)
+ }
+ }
+}
+
+// getCompilationCache returns the global compilation cache, creating it if necessary
+func getCompilationCache() (wazero.CompilationCache, error) {
+ var err error
+ cacheOnce.Do(func() {
+ cacheDir := filepath.Join(conf.Server.CacheFolder, "plugins")
+ purgeCacheBySize(cacheDir, conf.Server.Plugins.CacheSize)
+ compilationCache, err = wazero.NewCompilationCacheWithDir(cacheDir)
+ })
+ return compilationCache, err
+}
+
+// newWazeroModuleConfig creates the correct ModuleConfig for plugins
+func newWazeroModuleConfig() wazero.ModuleConfig {
+ return wazero.NewModuleConfig().WithStartFunctions("_initialize").WithStderr(log.Writer())
+}
+
+// pluginCompilationTimeout returns the timeout for plugin compilation
+func pluginCompilationTimeout() time.Duration {
+ if conf.Server.DevPluginCompilationTimeout > 0 {
+ return conf.Server.DevPluginCompilationTimeout
+ }
+ return time.Minute
+}
+
+// precompilePlugin compiles the WASM module in the background and updates the pluginState.
+func precompilePlugin(p *plugin) {
+ compileSemaphore <- struct{}{}
+ defer func() { <-compileSemaphore }()
+ ctx := context.Background()
+ r, err := p.Runtime(ctx)
+ if err != nil {
+ p.compilationErr = fmt.Errorf("failed to create runtime for plugin %s: %w", p.ID, err)
+ close(p.compilationReady)
+ return
+ }
+
+ b, err := os.ReadFile(p.WasmPath)
+ if err != nil {
+ p.compilationErr = fmt.Errorf("failed to read wasm file: %w", err)
+ close(p.compilationReady)
+ return
+ }
+
+ // We know r is always a *scopedRuntime from createRuntime
+ scopedRT := r.(*scopedRuntime)
+ cachingRT := scopedRT.GetCachingRuntime()
+ if cachingRT == nil {
+ p.compilationErr = fmt.Errorf("failed to get cachingRuntime for plugin %s", p.ID)
+ close(p.compilationReady)
+ return
+ }
+
+ _, err = cachingRT.CompileModule(ctx, b)
+ if err != nil {
+ p.compilationErr = fmt.Errorf("failed to compile WASM for plugin %s: %w", p.ID, err)
+ log.Warn("Plugin compilation failed", "name", p.ID, "path", p.WasmPath, "err", err)
+ } else {
+ p.compilationErr = nil
+ log.Debug("Plugin compilation completed", "name", p.ID, "path", p.WasmPath)
+ }
+ close(p.compilationReady)
+}
+
+// loadHostLibrary loads the given host library and returns its exported functions
+func loadHostLibrary[S any](
+ ctx context.Context,
+ instantiateFn func(context.Context, wazero.Runtime, S) error,
+ service S,
+) (map[string]wazeroapi.FunctionDefinition, error) {
+ r := wazero.NewRuntime(ctx)
+ if err := instantiateFn(ctx, r, service); err != nil {
+ return nil, err
+ }
+ m := r.Module("env")
+ return m.ExportedFunctionDefinitions(), nil
+}
+
+// combineLibraries combines the given host libraries into a single "env" module
+func combineLibraries(ctx context.Context, r wazero.Runtime, libs ...map[string]wazeroapi.FunctionDefinition) error {
+ // Merge the libraries
+ hostLib := map[string]wazeroapi.FunctionDefinition{}
+ for _, lib := range libs {
+ maps.Copy(hostLib, lib)
+ }
+
+ // Create the combined host module
+ envBuilder := r.NewHostModuleBuilder("env")
+ for name, fd := range hostLib {
+ fn, ok := fd.GoFunction().(wazeroapi.GoModuleFunction)
+ if !ok {
+ return fmt.Errorf("invalid function definition: %s", fd.DebugName())
+ }
+ envBuilder.NewFunctionBuilder().
+ WithGoModuleFunction(fn, fd.ParamTypes(), fd.ResultTypes()).
+ WithParameterNames(fd.ParamNames()...).Export(name)
+ }
+
+ // Instantiate the combined host module
+ if _, err := envBuilder.Instantiate(ctx); err != nil {
+ return err
+ }
+ return nil
+}
+
+const (
+ // WASM Instance pool configuration
+ // defaultPoolSize is the maximum number of instances per plugin that are kept in the pool for reuse
+ defaultPoolSize = 8
+ // defaultInstanceTTL is the time after which an instance is considered stale and can be evicted
+ defaultInstanceTTL = time.Minute
+ // defaultMaxConcurrentInstances is the hard limit on total instances that can exist simultaneously
+ defaultMaxConcurrentInstances = 10
+ // defaultGetTimeout is the maximum time to wait when getting an instance if at the concurrent limit
+ defaultGetTimeout = 5 * time.Second
+
+ // Compiled module cache configuration
+ // defaultCompiledModuleTTL is the time after which a compiled module is evicted from the cache
+ defaultCompiledModuleTTL = 5 * time.Minute
+)
+
+// cachedCompiledModule encapsulates a compiled WebAssembly module with TTL management
+type cachedCompiledModule struct {
+ module wazero.CompiledModule
+ hash [16]byte
+ lastAccess time.Time
+ timer *time.Timer
+ mu sync.Mutex
+ pluginID string // for logging purposes
+}
+
+// newCachedCompiledModule creates a new cached compiled module with TTL management
+func newCachedCompiledModule(module wazero.CompiledModule, wasmBytes []byte, pluginID string) *cachedCompiledModule {
+ c := &cachedCompiledModule{
+ module: module,
+ hash: md5.Sum(wasmBytes),
+ lastAccess: time.Now(),
+ pluginID: pluginID,
+ }
+
+ // Set up the TTL timer
+ c.timer = time.AfterFunc(defaultCompiledModuleTTL, c.evict)
+
+ return c
+}
+
+// get returns the cached module if the hash matches, nil otherwise
+// Also resets the TTL timer on successful access
+func (c *cachedCompiledModule) get(wasmHash [16]byte) wazero.CompiledModule {
+ c.mu.Lock() // Use write lock because we modify state in resetTimer
+ defer c.mu.Unlock()
+
+ if c.module != nil && c.hash == wasmHash {
+ // Reset TTL timer on access
+ c.resetTimer()
+ return c.module
+ }
+
+ return nil
+}
+
+// resetTimer resets the TTL timer (must be called with lock held)
+func (c *cachedCompiledModule) resetTimer() {
+ c.lastAccess = time.Now()
+
+ if c.timer != nil {
+ c.timer.Stop()
+ c.timer = time.AfterFunc(defaultCompiledModuleTTL, c.evict)
+ }
+}
+
+// evict removes the cached module and cleans up resources
+func (c *cachedCompiledModule) evict() {
+ c.mu.Lock()
+ defer c.mu.Unlock()
+
+ if c.module != nil {
+ log.Trace("cachedCompiledModule: evicting due to TTL expiry", "plugin", c.pluginID, "ttl", defaultCompiledModuleTTL)
+ c.module.Close(context.Background())
+ c.module = nil
+ c.hash = [16]byte{}
+ c.lastAccess = time.Time{}
+ }
+
+ if c.timer != nil {
+ c.timer.Stop()
+ c.timer = nil
+ }
+}
+
+// close cleans up the cached module and stops the timer
+func (c *cachedCompiledModule) close(ctx context.Context) {
+ c.mu.Lock()
+ defer c.mu.Unlock()
+
+ if c.timer != nil {
+ c.timer.Stop()
+ c.timer = nil
+ }
+
+ if c.module != nil {
+ c.module.Close(ctx)
+ c.module = nil
+ }
+}
+
+// pooledModule wraps a wazero Module and returns it to the pool when closed.
+type pooledModule struct {
+ wazeroapi.Module
+ pool *wasmInstancePool[wazeroapi.Module]
+ closed bool
+}
+
+func (m *pooledModule) Close(ctx context.Context) error {
+ if !m.closed {
+ m.closed = true
+ m.pool.Put(ctx, m.Module)
+ }
+ return nil
+}
+
+func (m *pooledModule) CloseWithExitCode(ctx context.Context, exitCode uint32) error {
+ return m.Close(ctx)
+}
+
+func (m *pooledModule) IsClosed() bool {
+ return m.closed
+}
+
+// newScopedRuntime creates a new scopedRuntime that wraps the given runtime
+func newScopedRuntime(runtime wazero.Runtime) *scopedRuntime {
+ return &scopedRuntime{Runtime: runtime}
+}
+
+// scopedRuntime wraps a cachingRuntime and captures a specific module
+// so that Close() only affects that module, not the entire shared runtime
+type scopedRuntime struct {
+ wazero.Runtime
+ capturedModule wazeroapi.Module
+}
+
+func (w *scopedRuntime) InstantiateModule(ctx context.Context, code wazero.CompiledModule, config wazero.ModuleConfig) (wazeroapi.Module, error) {
+ module, err := w.Runtime.InstantiateModule(ctx, code, config)
+ if err != nil {
+ return nil, err
+ }
+ // Capture the module for later cleanup
+ w.capturedModule = module
+ log.Trace(ctx, "scopedRuntime: captured module", "moduleID", getInstanceID(module))
+ return module, nil
+}
+
+func (w *scopedRuntime) Close(ctx context.Context) error {
+ // Close only the captured module, not the entire runtime
+ if w.capturedModule != nil {
+ log.Trace(ctx, "scopedRuntime: closing captured module", "moduleID", getInstanceID(w.capturedModule))
+ return w.capturedModule.Close(ctx)
+ }
+ log.Trace(ctx, "scopedRuntime: no captured module to close")
+ return nil
+}
+
+func (w *scopedRuntime) CloseWithExitCode(ctx context.Context, exitCode uint32) error {
+ return w.Close(ctx)
+}
+
+// GetCachingRuntime returns the underlying cachingRuntime for internal use
+func (w *scopedRuntime) GetCachingRuntime() *cachingRuntime {
+ if cr, ok := w.Runtime.(*cachingRuntime); ok {
+ return cr
+ }
+ return nil
+}
+
+// cachingRuntime wraps wazero.Runtime and pools module instances per plugin,
+// while also caching the compiled module in memory.
+type cachingRuntime struct {
+ wazero.Runtime
+
+ // pluginID is required to differentiate between different plugins that use the same file to initialize their
+ // runtime. The runtime will serve as a singleton for all instances of a given plugin.
+ pluginID string
+
+ // cachedModule manages the compiled module cache with TTL
+ cachedModule atomic.Pointer[cachedCompiledModule]
+
+ // pool manages reusable module instances
+ pool *wasmInstancePool[wazeroapi.Module]
+
+ // poolInitOnce ensures the pool is initialized only once
+ poolInitOnce sync.Once
+
+ // compilationMu ensures only one compilation happens at a time per runtime
+ compilationMu sync.Mutex
+}
+
+func newCachingRuntime(runtime wazero.Runtime, pluginID string) *cachingRuntime {
+ return &cachingRuntime{
+ Runtime: runtime,
+ pluginID: pluginID,
+ }
+}
+
+func (r *cachingRuntime) initPool(code wazero.CompiledModule, config wazero.ModuleConfig) {
+ r.poolInitOnce.Do(func() {
+ r.pool = newWasmInstancePool[wazeroapi.Module](r.pluginID, defaultPoolSize, defaultMaxConcurrentInstances, defaultGetTimeout, defaultInstanceTTL, func(ctx context.Context) (wazeroapi.Module, error) {
+ log.Trace(ctx, "cachingRuntime: creating new module instance", "plugin", r.pluginID)
+ return r.Runtime.InstantiateModule(ctx, code, config)
+ })
+ })
+}
+
+func (r *cachingRuntime) InstantiateModule(ctx context.Context, code wazero.CompiledModule, config wazero.ModuleConfig) (wazeroapi.Module, error) {
+ r.initPool(code, config)
+ mod, err := r.pool.Get(ctx)
+ if err != nil {
+ return nil, err
+ }
+ wrapped := &pooledModule{Module: mod, pool: r.pool}
+ log.Trace(ctx, "cachingRuntime: created wrapper for module", "plugin", r.pluginID, "underlyingModuleID", fmt.Sprintf("%p", mod), "wrapperID", fmt.Sprintf("%p", wrapped))
+ return wrapped, nil
+}
+
+func (r *cachingRuntime) Close(ctx context.Context) error {
+ log.Trace(ctx, "cachingRuntime: closing runtime", "plugin", r.pluginID)
+
+ // Clean up compiled module cache
+ if cached := r.cachedModule.Swap(nil); cached != nil {
+ cached.close(ctx)
+ }
+
+ // Close the instance pool
+ if r.pool != nil {
+ r.pool.Close(ctx)
+ }
+ // Close the underlying runtime
+ return r.Runtime.Close(ctx)
+}
+
+// setCachedModule stores a newly compiled module in the cache with TTL management
+func (r *cachingRuntime) setCachedModule(module wazero.CompiledModule, wasmBytes []byte) {
+ newCached := newCachedCompiledModule(module, wasmBytes, r.pluginID)
+
+ // Replace old cached module and clean it up
+ if old := r.cachedModule.Swap(newCached); old != nil {
+ old.close(context.Background())
+ }
+}
+
+// CompileModule checks if the provided bytes match our cached hash and returns
+// the cached compiled module if so, avoiding both file read and compilation.
+func (r *cachingRuntime) CompileModule(ctx context.Context, wasmBytes []byte) (wazero.CompiledModule, error) {
+ incomingHash := md5.Sum(wasmBytes)
+
+ // Try to get from cache first (without lock for performance)
+ if cached := r.cachedModule.Load(); cached != nil {
+ if module := cached.get(incomingHash); module != nil {
+ log.Trace(ctx, "cachingRuntime: using cached compiled module", "plugin", r.pluginID)
+ return module, nil
+ }
+ }
+
+ // Synchronize compilation to prevent concurrent compilation issues
+ r.compilationMu.Lock()
+ defer r.compilationMu.Unlock()
+
+ // Double-check cache after acquiring lock (another goroutine might have compiled it)
+ if cached := r.cachedModule.Load(); cached != nil {
+ if module := cached.get(incomingHash); module != nil {
+ log.Trace(ctx, "cachingRuntime: using cached compiled module (after lock)", "plugin", r.pluginID)
+ return module, nil
+ }
+ }
+
+ // Fall back to normal compilation for different bytes
+ log.Trace(ctx, "cachingRuntime: hash doesn't match cache, compiling normally", "plugin", r.pluginID)
+ module, err := r.Runtime.CompileModule(ctx, wasmBytes)
+ if err != nil {
+ return nil, err
+ }
+
+ // Cache the newly compiled module
+ r.setCachedModule(module, wasmBytes)
+
+ return module, nil
+}
diff --git a/plugins/runtime_test.go b/plugins/runtime_test.go
new file mode 100644
index 000000000..05efe1d1d
--- /dev/null
+++ b/plugins/runtime_test.go
@@ -0,0 +1,173 @@
+package plugins
+
+import (
+ "context"
+ "fmt"
+ "os"
+ "path/filepath"
+ "time"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/core/metrics"
+ "github.com/navidrome/navidrome/plugins/schema"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+ "github.com/tetratelabs/wazero"
+)
+
+var _ = Describe("Runtime", func() {
+ Describe("pluginCompilationTimeout", func() {
+ It("should use DevPluginCompilationTimeout config for plugin compilation timeout", func() {
+ originalTimeout := conf.Server.DevPluginCompilationTimeout
+ DeferCleanup(func() {
+ conf.Server.DevPluginCompilationTimeout = originalTimeout
+ })
+
+ conf.Server.DevPluginCompilationTimeout = 123 * time.Second
+ Expect(pluginCompilationTimeout()).To(Equal(123 * time.Second))
+
+ conf.Server.DevPluginCompilationTimeout = 0
+ Expect(pluginCompilationTimeout()).To(Equal(time.Minute))
+ })
+ })
+})
+
+var _ = Describe("CachingRuntime", func() {
+ var (
+ ctx context.Context
+ mgr *managerImpl
+ plugin *wasmScrobblerPlugin
+ )
+
+ BeforeEach(func() {
+ ctx = GinkgoT().Context()
+ mgr = createManager(nil, metrics.NewNoopInstance())
+ // Add permissions for the test plugin using typed struct
+ permissions := schema.PluginManifestPermissions{
+ Http: &schema.PluginManifestPermissionsHttp{
+ Reason: "For testing HTTP functionality",
+ AllowedUrls: map[string][]schema.PluginManifestPermissionsHttpAllowedUrlsValueElem{
+ "*": {schema.PluginManifestPermissionsHttpAllowedUrlsValueElemWildcard},
+ },
+ AllowLocalNetwork: false,
+ },
+ Config: &schema.PluginManifestPermissionsConfig{
+ Reason: "For testing config functionality",
+ },
+ }
+ rtFunc := mgr.createRuntime("fake_scrobbler", permissions)
+ plugin = newWasmScrobblerPlugin(
+ filepath.Join(testDataDir, "fake_scrobbler", "plugin.wasm"),
+ "fake_scrobbler",
+ mgr,
+ rtFunc,
+ wazero.NewModuleConfig().WithStartFunctions("_initialize"),
+ ).(*wasmScrobblerPlugin)
+ // runtime will be created on first plugin load
+ })
+
+ It("reuses module instances across calls", func() {
+ // First call to create the runtime and pool
+ _, done, err := plugin.getInstance(ctx, "first")
+ Expect(err).ToNot(HaveOccurred())
+ done()
+
+ val, ok := runtimePool.Load("fake_scrobbler")
+ Expect(ok).To(BeTrue())
+ cachingRT := val.(*cachingRuntime)
+
+ // Verify the pool exists and is initialized
+ Expect(cachingRT.pool).ToNot(BeNil())
+
+ // Test that multiple calls work without error (indicating pool reuse)
+ for i := 0; i < 5; i++ {
+ inst, done, err := plugin.getInstance(ctx, fmt.Sprintf("call_%d", i))
+ Expect(err).ToNot(HaveOccurred())
+ Expect(inst).ToNot(BeNil())
+ done()
+ }
+
+ // Test concurrent access to verify pool handles concurrency
+ const numGoroutines = 3
+ errChan := make(chan error, numGoroutines)
+
+ for i := 0; i < numGoroutines; i++ {
+ go func(id int) {
+ inst, done, err := plugin.getInstance(ctx, fmt.Sprintf("concurrent_%d", id))
+ if err != nil {
+ errChan <- err
+ return
+ }
+ defer done()
+
+ // Verify we got a valid instance
+ if inst == nil {
+ errChan <- fmt.Errorf("got nil instance")
+ return
+ }
+ errChan <- nil
+ }(i)
+ }
+
+ // Check all goroutines succeeded
+ for i := 0; i < numGoroutines; i++ {
+ err := <-errChan
+ Expect(err).To(BeNil())
+ }
+ })
+})
+
+var _ = Describe("purgeCacheBySize", func() {
+ var tmpDir string
+
+ BeforeEach(func() {
+ var err error
+ tmpDir, err = os.MkdirTemp("", "cache_test")
+ Expect(err).ToNot(HaveOccurred())
+ DeferCleanup(os.RemoveAll, tmpDir)
+ })
+
+ It("removes oldest entries when above the size limit", func() {
+ oldDir := filepath.Join(tmpDir, "d1")
+ newDir := filepath.Join(tmpDir, "d2")
+ Expect(os.Mkdir(oldDir, 0700)).To(Succeed())
+ Expect(os.Mkdir(newDir, 0700)).To(Succeed())
+
+ oldFile := filepath.Join(oldDir, "old")
+ newFile := filepath.Join(newDir, "new")
+ Expect(os.WriteFile(oldFile, []byte("xx"), 0600)).To(Succeed())
+ Expect(os.WriteFile(newFile, []byte("xx"), 0600)).To(Succeed())
+
+ oldTime := time.Now().Add(-2 * time.Hour)
+ Expect(os.Chtimes(oldFile, oldTime, oldTime)).To(Succeed())
+
+ purgeCacheBySize(tmpDir, "3")
+
+ _, err := os.Stat(oldFile)
+ Expect(os.IsNotExist(err)).To(BeTrue())
+ _, err = os.Stat(oldDir)
+ Expect(os.IsNotExist(err)).To(BeTrue())
+
+ _, err = os.Stat(newFile)
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ It("does nothing when below the size limit", func() {
+ dir1 := filepath.Join(tmpDir, "a")
+ dir2 := filepath.Join(tmpDir, "b")
+ Expect(os.Mkdir(dir1, 0700)).To(Succeed())
+ Expect(os.Mkdir(dir2, 0700)).To(Succeed())
+
+ file1 := filepath.Join(dir1, "f1")
+ file2 := filepath.Join(dir2, "f2")
+ Expect(os.WriteFile(file1, []byte("x"), 0600)).To(Succeed())
+ Expect(os.WriteFile(file2, []byte("x"), 0600)).To(Succeed())
+
+ purgeCacheBySize(tmpDir, "10MB")
+
+ _, err := os.Stat(file1)
+ Expect(err).ToNot(HaveOccurred())
+ _, err = os.Stat(file2)
+ Expect(err).ToNot(HaveOccurred())
+ })
+})
diff --git a/plugins/schema/manifest.schema.json b/plugins/schema/manifest.schema.json
new file mode 100644
index 000000000..0c323126b
--- /dev/null
+++ b/plugins/schema/manifest.schema.json
@@ -0,0 +1,199 @@
+{
+ "$id": "navidrome://plugins/manifest",
+ "$schema": "http://json-schema.org/draft-07/schema#",
+ "title": "Navidrome Plugin Manifest",
+ "description": "Schema for Navidrome Plugin manifest.json files",
+ "type": "object",
+ "required": [
+ "name",
+ "author",
+ "version",
+ "description",
+ "website",
+ "capabilities",
+ "permissions"
+ ],
+ "properties": {
+ "name": {
+ "type": "string",
+ "description": "Name of the plugin"
+ },
+ "author": {
+ "type": "string",
+ "description": "Author or organization that created the plugin"
+ },
+ "version": {
+ "type": "string",
+ "description": "Plugin version using semantic versioning format"
+ },
+ "description": {
+ "type": "string",
+ "description": "A brief description of the plugin's functionality"
+ },
+ "website": {
+ "type": "string",
+ "format": "uri",
+ "description": "Website URL for the plugin or its documentation"
+ },
+ "capabilities": {
+ "type": "array",
+ "description": "List of capabilities implemented by this plugin",
+ "minItems": 1,
+ "items": {
+ "type": "string",
+ "enum": [
+ "MetadataAgent",
+ "Scrobbler",
+ "SchedulerCallback",
+ "LifecycleManagement",
+ "WebSocketCallback"
+ ]
+ }
+ },
+ "permissions": {
+ "type": "object",
+ "description": "Host services the plugin is allowed to access",
+ "additionalProperties": true,
+ "properties": {
+ "http": {
+ "allOf": [
+ { "$ref": "#/$defs/basePermission" },
+ {
+ "type": "object",
+ "description": "HTTP service permissions",
+ "required": ["allowedUrls"],
+ "properties": {
+ "allowedUrls": {
+ "type": "object",
+ "description": "Map of URL patterns (e.g., 'https://api.example.com/*') to allowed HTTP methods. Redirect destinations must also be included.",
+ "additionalProperties": {
+ "type": "array",
+ "items": {
+ "type": "string",
+ "enum": [
+ "GET",
+ "POST",
+ "PUT",
+ "DELETE",
+ "PATCH",
+ "HEAD",
+ "OPTIONS",
+ "*"
+ ]
+ },
+ "minItems": 1,
+ "uniqueItems": true
+ },
+ "minProperties": 1
+ },
+ "allowLocalNetwork": {
+ "type": "boolean",
+ "description": "Whether to allow requests to local/private network addresses",
+ "default": false
+ }
+ }
+ }
+ ]
+ },
+ "config": {
+ "allOf": [
+ { "$ref": "#/$defs/basePermission" },
+ {
+ "type": "object",
+ "description": "Configuration service permissions"
+ }
+ ]
+ },
+ "scheduler": {
+ "allOf": [
+ { "$ref": "#/$defs/basePermission" },
+ {
+ "type": "object",
+ "description": "Scheduler service permissions"
+ }
+ ]
+ },
+ "websocket": {
+ "allOf": [
+ { "$ref": "#/$defs/basePermission" },
+ {
+ "type": "object",
+ "description": "WebSocket service permissions",
+ "required": ["allowedUrls"],
+ "properties": {
+ "allowedUrls": {
+ "type": "array",
+ "description": "List of WebSocket URL patterns that the plugin is allowed to connect to",
+ "items": {
+ "type": "string",
+ "pattern": "^wss?://.*$"
+ },
+ "minItems": 1,
+ "uniqueItems": true
+ },
+ "allowLocalNetwork": {
+ "type": "boolean",
+ "description": "Whether to allow connections to local/private network addresses",
+ "default": false
+ }
+ }
+ }
+ ]
+ },
+ "cache": {
+ "allOf": [
+ { "$ref": "#/$defs/basePermission" },
+ {
+ "type": "object",
+ "description": "Cache service permissions"
+ }
+ ]
+ },
+ "artwork": {
+ "allOf": [
+ { "$ref": "#/$defs/basePermission" },
+ {
+ "type": "object",
+ "description": "Artwork service permissions"
+ }
+ ]
+ },
+ "subsonicapi": {
+ "allOf": [
+ { "$ref": "#/$defs/basePermission" },
+ {
+ "type": "object",
+ "description": "SubsonicAPI service permissions",
+ "properties": {
+ "allowedUsernames": {
+ "type": "array",
+ "description": "List of usernames the plugin can pass as u. Any user if empty",
+ "items": { "type": "string" }
+ },
+ "allowAdmins": {
+ "type": "boolean",
+ "description": "If false, reject calls where the u is an admin",
+ "default": false
+ }
+ }
+ }
+ ]
+ }
+ }
+ }
+ },
+ "$defs": {
+ "basePermission": {
+ "type": "object",
+ "required": ["reason"],
+ "properties": {
+ "reason": {
+ "type": "string",
+ "minLength": 1,
+ "description": "Explanation of why this permission is needed"
+ }
+ },
+ "additionalProperties": false
+ }
+ }
+}
diff --git a/plugins/schema/manifest_gen.go b/plugins/schema/manifest_gen.go
new file mode 100644
index 000000000..97e07a077
--- /dev/null
+++ b/plugins/schema/manifest_gen.go
@@ -0,0 +1,426 @@
+// Code generated by github.com/atombender/go-jsonschema, DO NOT EDIT.
+
+package schema
+
+import "encoding/json"
+import "fmt"
+import "reflect"
+
+type BasePermission struct {
+ // Explanation of why this permission is needed
+ Reason string `json:"reason" yaml:"reason" mapstructure:"reason"`
+}
+
+// UnmarshalJSON implements json.Unmarshaler.
+func (j *BasePermission) UnmarshalJSON(value []byte) error {
+ var raw map[string]interface{}
+ if err := json.Unmarshal(value, &raw); err != nil {
+ return err
+ }
+ if _, ok := raw["reason"]; raw != nil && !ok {
+ return fmt.Errorf("field reason in BasePermission: required")
+ }
+ type Plain BasePermission
+ var plain Plain
+ if err := json.Unmarshal(value, &plain); err != nil {
+ return err
+ }
+ if len(plain.Reason) < 1 {
+ return fmt.Errorf("field %s length: must be >= %d", "reason", 1)
+ }
+ *j = BasePermission(plain)
+ return nil
+}
+
+// Schema for Navidrome Plugin manifest.json files
+type PluginManifest struct {
+ // Author or organization that created the plugin
+ Author string `json:"author" yaml:"author" mapstructure:"author"`
+
+ // List of capabilities implemented by this plugin
+ Capabilities []PluginManifestCapabilitiesElem `json:"capabilities" yaml:"capabilities" mapstructure:"capabilities"`
+
+ // A brief description of the plugin's functionality
+ Description string `json:"description" yaml:"description" mapstructure:"description"`
+
+ // Name of the plugin
+ Name string `json:"name" yaml:"name" mapstructure:"name"`
+
+ // Host services the plugin is allowed to access
+ Permissions PluginManifestPermissions `json:"permissions" yaml:"permissions" mapstructure:"permissions"`
+
+ // Plugin version using semantic versioning format
+ Version string `json:"version" yaml:"version" mapstructure:"version"`
+
+ // Website URL for the plugin or its documentation
+ Website string `json:"website" yaml:"website" mapstructure:"website"`
+}
+
+type PluginManifestCapabilitiesElem string
+
+const PluginManifestCapabilitiesElemLifecycleManagement PluginManifestCapabilitiesElem = "LifecycleManagement"
+const PluginManifestCapabilitiesElemMetadataAgent PluginManifestCapabilitiesElem = "MetadataAgent"
+const PluginManifestCapabilitiesElemSchedulerCallback PluginManifestCapabilitiesElem = "SchedulerCallback"
+const PluginManifestCapabilitiesElemScrobbler PluginManifestCapabilitiesElem = "Scrobbler"
+const PluginManifestCapabilitiesElemWebSocketCallback PluginManifestCapabilitiesElem = "WebSocketCallback"
+
+var enumValues_PluginManifestCapabilitiesElem = []interface{}{
+ "MetadataAgent",
+ "Scrobbler",
+ "SchedulerCallback",
+ "LifecycleManagement",
+ "WebSocketCallback",
+}
+
+// UnmarshalJSON implements json.Unmarshaler.
+func (j *PluginManifestCapabilitiesElem) UnmarshalJSON(value []byte) error {
+ var v string
+ if err := json.Unmarshal(value, &v); err != nil {
+ return err
+ }
+ var ok bool
+ for _, expected := range enumValues_PluginManifestCapabilitiesElem {
+ if reflect.DeepEqual(v, expected) {
+ ok = true
+ break
+ }
+ }
+ if !ok {
+ return fmt.Errorf("invalid value (expected one of %#v): %#v", enumValues_PluginManifestCapabilitiesElem, v)
+ }
+ *j = PluginManifestCapabilitiesElem(v)
+ return nil
+}
+
+// Host services the plugin is allowed to access
+type PluginManifestPermissions struct {
+ // Artwork corresponds to the JSON schema field "artwork".
+ Artwork *PluginManifestPermissionsArtwork `json:"artwork,omitempty" yaml:"artwork,omitempty" mapstructure:"artwork,omitempty"`
+
+ // Cache corresponds to the JSON schema field "cache".
+ Cache *PluginManifestPermissionsCache `json:"cache,omitempty" yaml:"cache,omitempty" mapstructure:"cache,omitempty"`
+
+ // Config corresponds to the JSON schema field "config".
+ Config *PluginManifestPermissionsConfig `json:"config,omitempty" yaml:"config,omitempty" mapstructure:"config,omitempty"`
+
+ // Http corresponds to the JSON schema field "http".
+ Http *PluginManifestPermissionsHttp `json:"http,omitempty" yaml:"http,omitempty" mapstructure:"http,omitempty"`
+
+ // Scheduler corresponds to the JSON schema field "scheduler".
+ Scheduler *PluginManifestPermissionsScheduler `json:"scheduler,omitempty" yaml:"scheduler,omitempty" mapstructure:"scheduler,omitempty"`
+
+ // Subsonicapi corresponds to the JSON schema field "subsonicapi".
+ Subsonicapi *PluginManifestPermissionsSubsonicapi `json:"subsonicapi,omitempty" yaml:"subsonicapi,omitempty" mapstructure:"subsonicapi,omitempty"`
+
+ // Websocket corresponds to the JSON schema field "websocket".
+ Websocket *PluginManifestPermissionsWebsocket `json:"websocket,omitempty" yaml:"websocket,omitempty" mapstructure:"websocket,omitempty"`
+
+ AdditionalProperties interface{} `mapstructure:",remain"`
+}
+
+// Artwork service permissions
+type PluginManifestPermissionsArtwork struct {
+ // Explanation of why this permission is needed
+ Reason string `json:"reason" yaml:"reason" mapstructure:"reason"`
+}
+
+// UnmarshalJSON implements json.Unmarshaler.
+func (j *PluginManifestPermissionsArtwork) UnmarshalJSON(value []byte) error {
+ var raw map[string]interface{}
+ if err := json.Unmarshal(value, &raw); err != nil {
+ return err
+ }
+ if _, ok := raw["reason"]; raw != nil && !ok {
+ return fmt.Errorf("field reason in PluginManifestPermissionsArtwork: required")
+ }
+ type Plain PluginManifestPermissionsArtwork
+ var plain Plain
+ if err := json.Unmarshal(value, &plain); err != nil {
+ return err
+ }
+ if len(plain.Reason) < 1 {
+ return fmt.Errorf("field %s length: must be >= %d", "reason", 1)
+ }
+ *j = PluginManifestPermissionsArtwork(plain)
+ return nil
+}
+
+// Cache service permissions
+type PluginManifestPermissionsCache struct {
+ // Explanation of why this permission is needed
+ Reason string `json:"reason" yaml:"reason" mapstructure:"reason"`
+}
+
+// UnmarshalJSON implements json.Unmarshaler.
+func (j *PluginManifestPermissionsCache) UnmarshalJSON(value []byte) error {
+ var raw map[string]interface{}
+ if err := json.Unmarshal(value, &raw); err != nil {
+ return err
+ }
+ if _, ok := raw["reason"]; raw != nil && !ok {
+ return fmt.Errorf("field reason in PluginManifestPermissionsCache: required")
+ }
+ type Plain PluginManifestPermissionsCache
+ var plain Plain
+ if err := json.Unmarshal(value, &plain); err != nil {
+ return err
+ }
+ if len(plain.Reason) < 1 {
+ return fmt.Errorf("field %s length: must be >= %d", "reason", 1)
+ }
+ *j = PluginManifestPermissionsCache(plain)
+ return nil
+}
+
+// Configuration service permissions
+type PluginManifestPermissionsConfig struct {
+ // Explanation of why this permission is needed
+ Reason string `json:"reason" yaml:"reason" mapstructure:"reason"`
+}
+
+// UnmarshalJSON implements json.Unmarshaler.
+func (j *PluginManifestPermissionsConfig) UnmarshalJSON(value []byte) error {
+ var raw map[string]interface{}
+ if err := json.Unmarshal(value, &raw); err != nil {
+ return err
+ }
+ if _, ok := raw["reason"]; raw != nil && !ok {
+ return fmt.Errorf("field reason in PluginManifestPermissionsConfig: required")
+ }
+ type Plain PluginManifestPermissionsConfig
+ var plain Plain
+ if err := json.Unmarshal(value, &plain); err != nil {
+ return err
+ }
+ if len(plain.Reason) < 1 {
+ return fmt.Errorf("field %s length: must be >= %d", "reason", 1)
+ }
+ *j = PluginManifestPermissionsConfig(plain)
+ return nil
+}
+
+// HTTP service permissions
+type PluginManifestPermissionsHttp struct {
+ // Whether to allow requests to local/private network addresses
+ AllowLocalNetwork bool `json:"allowLocalNetwork,omitempty" yaml:"allowLocalNetwork,omitempty" mapstructure:"allowLocalNetwork,omitempty"`
+
+ // Map of URL patterns (e.g., 'https://api.example.com/*') to allowed HTTP
+ // methods. Redirect destinations must also be included.
+ AllowedUrls map[string][]PluginManifestPermissionsHttpAllowedUrlsValueElem `json:"allowedUrls" yaml:"allowedUrls" mapstructure:"allowedUrls"`
+
+ // Explanation of why this permission is needed
+ Reason string `json:"reason" yaml:"reason" mapstructure:"reason"`
+}
+
+type PluginManifestPermissionsHttpAllowedUrlsValueElem string
+
+const PluginManifestPermissionsHttpAllowedUrlsValueElemDELETE PluginManifestPermissionsHttpAllowedUrlsValueElem = "DELETE"
+const PluginManifestPermissionsHttpAllowedUrlsValueElemGET PluginManifestPermissionsHttpAllowedUrlsValueElem = "GET"
+const PluginManifestPermissionsHttpAllowedUrlsValueElemHEAD PluginManifestPermissionsHttpAllowedUrlsValueElem = "HEAD"
+const PluginManifestPermissionsHttpAllowedUrlsValueElemOPTIONS PluginManifestPermissionsHttpAllowedUrlsValueElem = "OPTIONS"
+const PluginManifestPermissionsHttpAllowedUrlsValueElemPATCH PluginManifestPermissionsHttpAllowedUrlsValueElem = "PATCH"
+const PluginManifestPermissionsHttpAllowedUrlsValueElemPOST PluginManifestPermissionsHttpAllowedUrlsValueElem = "POST"
+const PluginManifestPermissionsHttpAllowedUrlsValueElemPUT PluginManifestPermissionsHttpAllowedUrlsValueElem = "PUT"
+const PluginManifestPermissionsHttpAllowedUrlsValueElemWildcard PluginManifestPermissionsHttpAllowedUrlsValueElem = "*"
+
+var enumValues_PluginManifestPermissionsHttpAllowedUrlsValueElem = []interface{}{
+ "GET",
+ "POST",
+ "PUT",
+ "DELETE",
+ "PATCH",
+ "HEAD",
+ "OPTIONS",
+ "*",
+}
+
+// UnmarshalJSON implements json.Unmarshaler.
+func (j *PluginManifestPermissionsHttpAllowedUrlsValueElem) UnmarshalJSON(value []byte) error {
+ var v string
+ if err := json.Unmarshal(value, &v); err != nil {
+ return err
+ }
+ var ok bool
+ for _, expected := range enumValues_PluginManifestPermissionsHttpAllowedUrlsValueElem {
+ if reflect.DeepEqual(v, expected) {
+ ok = true
+ break
+ }
+ }
+ if !ok {
+ return fmt.Errorf("invalid value (expected one of %#v): %#v", enumValues_PluginManifestPermissionsHttpAllowedUrlsValueElem, v)
+ }
+ *j = PluginManifestPermissionsHttpAllowedUrlsValueElem(v)
+ return nil
+}
+
+// UnmarshalJSON implements json.Unmarshaler.
+func (j *PluginManifestPermissionsHttp) UnmarshalJSON(value []byte) error {
+ var raw map[string]interface{}
+ if err := json.Unmarshal(value, &raw); err != nil {
+ return err
+ }
+ if _, ok := raw["allowedUrls"]; raw != nil && !ok {
+ return fmt.Errorf("field allowedUrls in PluginManifestPermissionsHttp: required")
+ }
+ if _, ok := raw["reason"]; raw != nil && !ok {
+ return fmt.Errorf("field reason in PluginManifestPermissionsHttp: required")
+ }
+ type Plain PluginManifestPermissionsHttp
+ var plain Plain
+ if err := json.Unmarshal(value, &plain); err != nil {
+ return err
+ }
+ if v, ok := raw["allowLocalNetwork"]; !ok || v == nil {
+ plain.AllowLocalNetwork = false
+ }
+ if len(plain.Reason) < 1 {
+ return fmt.Errorf("field %s length: must be >= %d", "reason", 1)
+ }
+ *j = PluginManifestPermissionsHttp(plain)
+ return nil
+}
+
+// Scheduler service permissions
+type PluginManifestPermissionsScheduler struct {
+ // Explanation of why this permission is needed
+ Reason string `json:"reason" yaml:"reason" mapstructure:"reason"`
+}
+
+// UnmarshalJSON implements json.Unmarshaler.
+func (j *PluginManifestPermissionsScheduler) UnmarshalJSON(value []byte) error {
+ var raw map[string]interface{}
+ if err := json.Unmarshal(value, &raw); err != nil {
+ return err
+ }
+ if _, ok := raw["reason"]; raw != nil && !ok {
+ return fmt.Errorf("field reason in PluginManifestPermissionsScheduler: required")
+ }
+ type Plain PluginManifestPermissionsScheduler
+ var plain Plain
+ if err := json.Unmarshal(value, &plain); err != nil {
+ return err
+ }
+ if len(plain.Reason) < 1 {
+ return fmt.Errorf("field %s length: must be >= %d", "reason", 1)
+ }
+ *j = PluginManifestPermissionsScheduler(plain)
+ return nil
+}
+
+// SubsonicAPI service permissions
+type PluginManifestPermissionsSubsonicapi struct {
+ // If false, reject calls where the u is an admin
+ AllowAdmins bool `json:"allowAdmins,omitempty" yaml:"allowAdmins,omitempty" mapstructure:"allowAdmins,omitempty"`
+
+ // List of usernames the plugin can pass as u. Any user if empty
+ AllowedUsernames []string `json:"allowedUsernames,omitempty" yaml:"allowedUsernames,omitempty" mapstructure:"allowedUsernames,omitempty"`
+
+ // Explanation of why this permission is needed
+ Reason string `json:"reason" yaml:"reason" mapstructure:"reason"`
+}
+
+// UnmarshalJSON implements json.Unmarshaler.
+func (j *PluginManifestPermissionsSubsonicapi) UnmarshalJSON(value []byte) error {
+ var raw map[string]interface{}
+ if err := json.Unmarshal(value, &raw); err != nil {
+ return err
+ }
+ if _, ok := raw["reason"]; raw != nil && !ok {
+ return fmt.Errorf("field reason in PluginManifestPermissionsSubsonicapi: required")
+ }
+ type Plain PluginManifestPermissionsSubsonicapi
+ var plain Plain
+ if err := json.Unmarshal(value, &plain); err != nil {
+ return err
+ }
+ if v, ok := raw["allowAdmins"]; !ok || v == nil {
+ plain.AllowAdmins = false
+ }
+ if len(plain.Reason) < 1 {
+ return fmt.Errorf("field %s length: must be >= %d", "reason", 1)
+ }
+ *j = PluginManifestPermissionsSubsonicapi(plain)
+ return nil
+}
+
+// WebSocket service permissions
+type PluginManifestPermissionsWebsocket struct {
+ // Whether to allow connections to local/private network addresses
+ AllowLocalNetwork bool `json:"allowLocalNetwork,omitempty" yaml:"allowLocalNetwork,omitempty" mapstructure:"allowLocalNetwork,omitempty"`
+
+ // List of WebSocket URL patterns that the plugin is allowed to connect to
+ AllowedUrls []string `json:"allowedUrls" yaml:"allowedUrls" mapstructure:"allowedUrls"`
+
+ // Explanation of why this permission is needed
+ Reason string `json:"reason" yaml:"reason" mapstructure:"reason"`
+}
+
+// UnmarshalJSON implements json.Unmarshaler.
+func (j *PluginManifestPermissionsWebsocket) UnmarshalJSON(value []byte) error {
+ var raw map[string]interface{}
+ if err := json.Unmarshal(value, &raw); err != nil {
+ return err
+ }
+ if _, ok := raw["allowedUrls"]; raw != nil && !ok {
+ return fmt.Errorf("field allowedUrls in PluginManifestPermissionsWebsocket: required")
+ }
+ if _, ok := raw["reason"]; raw != nil && !ok {
+ return fmt.Errorf("field reason in PluginManifestPermissionsWebsocket: required")
+ }
+ type Plain PluginManifestPermissionsWebsocket
+ var plain Plain
+ if err := json.Unmarshal(value, &plain); err != nil {
+ return err
+ }
+ if v, ok := raw["allowLocalNetwork"]; !ok || v == nil {
+ plain.AllowLocalNetwork = false
+ }
+ if plain.AllowedUrls != nil && len(plain.AllowedUrls) < 1 {
+ return fmt.Errorf("field %s length: must be >= %d", "allowedUrls", 1)
+ }
+ if len(plain.Reason) < 1 {
+ return fmt.Errorf("field %s length: must be >= %d", "reason", 1)
+ }
+ *j = PluginManifestPermissionsWebsocket(plain)
+ return nil
+}
+
+// UnmarshalJSON implements json.Unmarshaler.
+func (j *PluginManifest) UnmarshalJSON(value []byte) error {
+ var raw map[string]interface{}
+ if err := json.Unmarshal(value, &raw); err != nil {
+ return err
+ }
+ if _, ok := raw["author"]; raw != nil && !ok {
+ return fmt.Errorf("field author in PluginManifest: required")
+ }
+ if _, ok := raw["capabilities"]; raw != nil && !ok {
+ return fmt.Errorf("field capabilities in PluginManifest: required")
+ }
+ if _, ok := raw["description"]; raw != nil && !ok {
+ return fmt.Errorf("field description in PluginManifest: required")
+ }
+ if _, ok := raw["name"]; raw != nil && !ok {
+ return fmt.Errorf("field name in PluginManifest: required")
+ }
+ if _, ok := raw["permissions"]; raw != nil && !ok {
+ return fmt.Errorf("field permissions in PluginManifest: required")
+ }
+ if _, ok := raw["version"]; raw != nil && !ok {
+ return fmt.Errorf("field version in PluginManifest: required")
+ }
+ if _, ok := raw["website"]; raw != nil && !ok {
+ return fmt.Errorf("field website in PluginManifest: required")
+ }
+ type Plain PluginManifest
+ var plain Plain
+ if err := json.Unmarshal(value, &plain); err != nil {
+ return err
+ }
+ if plain.Capabilities != nil && len(plain.Capabilities) < 1 {
+ return fmt.Errorf("field %s length: must be >= %d", "capabilities", 1)
+ }
+ *j = PluginManifest(plain)
+ return nil
+}
diff --git a/plugins/testdata/.gitignore b/plugins/testdata/.gitignore
new file mode 100644
index 000000000..917660a34
--- /dev/null
+++ b/plugins/testdata/.gitignore
@@ -0,0 +1 @@
+*.wasm
\ No newline at end of file
diff --git a/plugins/testdata/Makefile b/plugins/testdata/Makefile
new file mode 100644
index 000000000..f569cfce5
--- /dev/null
+++ b/plugins/testdata/Makefile
@@ -0,0 +1,10 @@
+# Fake sample plugins used for testing
+PLUGINS := fake_album_agent fake_artist_agent fake_scrobbler multi_plugin fake_init_service unauthorized_plugin
+
+all: $(PLUGINS:%=%/plugin.wasm)
+
+clean:
+ rm -f $(PLUGINS:%=%/plugin.wasm)
+
+%/plugin.wasm: %/plugin.go
+ GOOS=wasip1 GOARCH=wasm go build -buildmode=c-shared -o $@ ./$*
\ No newline at end of file
diff --git a/plugins/testdata/README.md b/plugins/testdata/README.md
new file mode 100644
index 000000000..abe840ff8
--- /dev/null
+++ b/plugins/testdata/README.md
@@ -0,0 +1,17 @@
+# Plugin Test Data
+
+This directory contains test data and mock implementations used for testing the Navidrome plugin system.
+
+## Contents
+
+Each of these directories contains the source code for a simple Go plugin that implements a specific agent interface
+(or multiple interfaces in the case of `multi_plugin`). These are compiled into WASM modules using the
+`Makefile` and used in integration tests for the plugin adapters (e.g., `adapter_media_agent_test.go`).
+
+Running `make` within this directory will build all test plugins.
+
+## Usage
+
+The primary use of this directory is during the development and testing phase. The `Makefile` is used to build the
+necessary WASM plugin binaries. The tests within the `plugins` package (and potentially other packages that interact
+with plugins) then utilize these compiled plugins and other test fixtures found here.
diff --git a/plugins/testdata/fake_album_agent/manifest.json b/plugins/testdata/fake_album_agent/manifest.json
new file mode 100644
index 000000000..e8dfb1fb3
--- /dev/null
+++ b/plugins/testdata/fake_album_agent/manifest.json
@@ -0,0 +1,9 @@
+{
+ "name": "fake_album_agent",
+ "author": "Navidrome Test",
+ "version": "1.0.0",
+ "description": "Test data for album agent",
+ "website": "https://test.navidrome.org/fake-album-agent",
+ "capabilities": ["MetadataAgent"],
+ "permissions": {}
+}
diff --git a/plugins/testdata/fake_album_agent/plugin.go b/plugins/testdata/fake_album_agent/plugin.go
new file mode 100644
index 000000000..c35e90397
--- /dev/null
+++ b/plugins/testdata/fake_album_agent/plugin.go
@@ -0,0 +1,70 @@
+//go:build wasip1
+
+package main
+
+import (
+ "context"
+
+ "github.com/navidrome/navidrome/plugins/api"
+)
+
+type FakeAlbumAgent struct{}
+
+var ErrNotFound = api.ErrNotFound
+
+func (FakeAlbumAgent) GetAlbumInfo(ctx context.Context, req *api.AlbumInfoRequest) (*api.AlbumInfoResponse, error) {
+ if req.Name != "" && req.Artist != "" {
+ return &api.AlbumInfoResponse{
+ Info: &api.AlbumInfo{
+ Name: req.Name,
+ Mbid: "album-mbid-123",
+ Description: "This is a test album description",
+ Url: "https://example.com/album",
+ },
+ }, nil
+ }
+ return nil, ErrNotFound
+}
+
+func (FakeAlbumAgent) GetAlbumImages(ctx context.Context, req *api.AlbumImagesRequest) (*api.AlbumImagesResponse, error) {
+ if req.Name != "" && req.Artist != "" {
+ return &api.AlbumImagesResponse{
+ Images: []*api.ExternalImage{
+ {Url: "https://example.com/album1.jpg", Size: 300},
+ {Url: "https://example.com/album2.jpg", Size: 400},
+ },
+ }, nil
+ }
+ return nil, ErrNotFound
+}
+
+func (FakeAlbumAgent) GetArtistMBID(ctx context.Context, req *api.ArtistMBIDRequest) (*api.ArtistMBIDResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (FakeAlbumAgent) GetArtistURL(ctx context.Context, req *api.ArtistURLRequest) (*api.ArtistURLResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (FakeAlbumAgent) GetArtistBiography(ctx context.Context, req *api.ArtistBiographyRequest) (*api.ArtistBiographyResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (FakeAlbumAgent) GetSimilarArtists(ctx context.Context, req *api.ArtistSimilarRequest) (*api.ArtistSimilarResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (FakeAlbumAgent) GetArtistImages(ctx context.Context, req *api.ArtistImageRequest) (*api.ArtistImageResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (FakeAlbumAgent) GetArtistTopSongs(ctx context.Context, req *api.ArtistTopSongsRequest) (*api.ArtistTopSongsResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func main() {}
+
+// Register the plugin implementation
+func init() {
+ api.RegisterMetadataAgent(FakeAlbumAgent{})
+}
diff --git a/plugins/testdata/fake_artist_agent/manifest.json b/plugins/testdata/fake_artist_agent/manifest.json
new file mode 100644
index 000000000..c5db72565
--- /dev/null
+++ b/plugins/testdata/fake_artist_agent/manifest.json
@@ -0,0 +1,9 @@
+{
+ "name": "fake_artist_agent",
+ "author": "Navidrome Test",
+ "version": "1.0.0",
+ "description": "Test data for artist agent",
+ "website": "https://test.navidrome.org/fake-artist-agent",
+ "capabilities": ["MetadataAgent"],
+ "permissions": {}
+}
diff --git a/plugins/testdata/fake_artist_agent/plugin.go b/plugins/testdata/fake_artist_agent/plugin.go
new file mode 100644
index 000000000..bd6b0f771
--- /dev/null
+++ b/plugins/testdata/fake_artist_agent/plugin.go
@@ -0,0 +1,82 @@
+//go:build wasip1
+
+package main
+
+import (
+ "context"
+
+ "github.com/navidrome/navidrome/plugins/api"
+)
+
+type FakeArtistAgent struct{}
+
+var ErrNotFound = api.ErrNotFound
+
+func (FakeArtistAgent) GetArtistMBID(ctx context.Context, req *api.ArtistMBIDRequest) (*api.ArtistMBIDResponse, error) {
+ if req.Name != "" {
+ return &api.ArtistMBIDResponse{Mbid: "1234567890"}, nil
+ }
+ return nil, ErrNotFound
+}
+func (FakeArtistAgent) GetArtistURL(ctx context.Context, req *api.ArtistURLRequest) (*api.ArtistURLResponse, error) {
+ if req.Name != "" {
+ return &api.ArtistURLResponse{Url: "https://example.com"}, nil
+ }
+ return nil, ErrNotFound
+}
+func (FakeArtistAgent) GetArtistBiography(ctx context.Context, req *api.ArtistBiographyRequest) (*api.ArtistBiographyResponse, error) {
+ if req.Name != "" {
+ return &api.ArtistBiographyResponse{Biography: "This is a test biography"}, nil
+ }
+ return nil, ErrNotFound
+}
+func (FakeArtistAgent) GetSimilarArtists(ctx context.Context, req *api.ArtistSimilarRequest) (*api.ArtistSimilarResponse, error) {
+ if req.Name != "" {
+ return &api.ArtistSimilarResponse{
+ Artists: []*api.Artist{
+ {Name: "Similar Artist 1", Mbid: "mbid1"},
+ {Name: "Similar Artist 2", Mbid: "mbid2"},
+ },
+ }, nil
+ }
+ return nil, ErrNotFound
+}
+func (FakeArtistAgent) GetArtistImages(ctx context.Context, req *api.ArtistImageRequest) (*api.ArtistImageResponse, error) {
+ if req.Name != "" {
+ return &api.ArtistImageResponse{
+ Images: []*api.ExternalImage{
+ {Url: "https://example.com/image1.jpg", Size: 100},
+ {Url: "https://example.com/image2.jpg", Size: 200},
+ },
+ }, nil
+ }
+ return nil, ErrNotFound
+}
+func (FakeArtistAgent) GetArtistTopSongs(ctx context.Context, req *api.ArtistTopSongsRequest) (*api.ArtistTopSongsResponse, error) {
+ if req.ArtistName != "" {
+ return &api.ArtistTopSongsResponse{
+ Songs: []*api.Song{
+ {Name: "Song 1", Mbid: "mbid1"},
+ {Name: "Song 2", Mbid: "mbid2"},
+ },
+ }, nil
+ }
+ return nil, ErrNotFound
+}
+
+// Add empty implementations for the album methods to satisfy the MetadataAgent interface
+func (FakeArtistAgent) GetAlbumInfo(ctx context.Context, req *api.AlbumInfoRequest) (*api.AlbumInfoResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (FakeArtistAgent) GetAlbumImages(ctx context.Context, req *api.AlbumImagesRequest) (*api.AlbumImagesResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+// main is required by Go WASI build
+func main() {}
+
+// init is used by go-plugin to register the implementation
+func init() {
+ api.RegisterMetadataAgent(FakeArtistAgent{})
+}
diff --git a/plugins/testdata/fake_init_service/manifest.json b/plugins/testdata/fake_init_service/manifest.json
new file mode 100644
index 000000000..ea8c45f58
--- /dev/null
+++ b/plugins/testdata/fake_init_service/manifest.json
@@ -0,0 +1,9 @@
+{
+ "name": "fake_init_service",
+ "version": "1.0.0",
+ "capabilities": ["LifecycleManagement"],
+ "author": "Test Author",
+ "description": "Test LifecycleManagement Callback",
+ "website": "https://test.navidrome.org/fake-init-service",
+ "permissions": {}
+}
diff --git a/plugins/testdata/fake_init_service/plugin.go b/plugins/testdata/fake_init_service/plugin.go
new file mode 100644
index 000000000..9e6171623
--- /dev/null
+++ b/plugins/testdata/fake_init_service/plugin.go
@@ -0,0 +1,42 @@
+//go:build wasip1
+
+package main
+
+import (
+ "context"
+ "errors"
+ "log"
+
+ "github.com/navidrome/navidrome/plugins/api"
+)
+
+type initServicePlugin struct{}
+
+func (p *initServicePlugin) OnInit(ctx context.Context, req *api.InitRequest) (*api.InitResponse, error) {
+ log.Printf("OnInit called with %v", req)
+
+ // Check for specific error conditions in the config
+ if req.Config != nil {
+ if errorType, exists := req.Config["returnError"]; exists {
+ switch errorType {
+ case "go_error":
+ return nil, errors.New("initialization failed with Go error")
+ case "response_error":
+ return &api.InitResponse{
+ Error: "initialization failed with response error",
+ }, nil
+ }
+ }
+ }
+
+ // Default: successful initialization
+ return &api.InitResponse{}, nil
+}
+
+// Required by Go WASI build
+func main() {}
+
+// Register the LifecycleManagement implementation
+func init() {
+ api.RegisterLifecycleManagement(&initServicePlugin{})
+}
diff --git a/plugins/testdata/fake_scrobbler/manifest.json b/plugins/testdata/fake_scrobbler/manifest.json
new file mode 100644
index 000000000..6fa41aa31
--- /dev/null
+++ b/plugins/testdata/fake_scrobbler/manifest.json
@@ -0,0 +1,9 @@
+{
+ "name": "fake_scrobbler",
+ "author": "Navidrome Test",
+ "version": "1.0.0",
+ "description": "Test data for scrobbler",
+ "website": "https://test.navidrome.org/fake-scrobbler",
+ "capabilities": ["Scrobbler"],
+ "permissions": {}
+}
diff --git a/plugins/testdata/fake_scrobbler/plugin.go b/plugins/testdata/fake_scrobbler/plugin.go
new file mode 100644
index 000000000..5a5c76699
--- /dev/null
+++ b/plugins/testdata/fake_scrobbler/plugin.go
@@ -0,0 +1,33 @@
+//go:build wasip1
+
+package main
+
+import (
+ "context"
+ "log"
+
+ "github.com/navidrome/navidrome/plugins/api"
+)
+
+type FakeScrobbler struct{}
+
+func (FakeScrobbler) IsAuthorized(ctx context.Context, req *api.ScrobblerIsAuthorizedRequest) (*api.ScrobblerIsAuthorizedResponse, error) {
+ log.Printf("[FakeScrobbler] IsAuthorized called for user: %s (%s)", req.Username, req.UserId)
+ return &api.ScrobblerIsAuthorizedResponse{Authorized: true}, nil
+}
+
+func (FakeScrobbler) NowPlaying(ctx context.Context, req *api.ScrobblerNowPlayingRequest) (*api.ScrobblerNowPlayingResponse, error) {
+ log.Printf("[FakeScrobbler] NowPlaying called for user: %s (%s), track: %s", req.Username, req.UserId, req.Track.Name)
+ return &api.ScrobblerNowPlayingResponse{}, nil
+}
+
+func (FakeScrobbler) Scrobble(ctx context.Context, req *api.ScrobblerScrobbleRequest) (*api.ScrobblerScrobbleResponse, error) {
+ log.Printf("[FakeScrobbler] Scrobble called for user: %s (%s), track: %s, timestamp: %d", req.Username, req.UserId, req.Track.Name, req.Timestamp)
+ return &api.ScrobblerScrobbleResponse{}, nil
+}
+
+func main() {}
+
+func init() {
+ api.RegisterScrobbler(FakeScrobbler{})
+}
diff --git a/plugins/testdata/multi_plugin/manifest.json b/plugins/testdata/multi_plugin/manifest.json
new file mode 100644
index 000000000..dc9e0a9a8
--- /dev/null
+++ b/plugins/testdata/multi_plugin/manifest.json
@@ -0,0 +1,13 @@
+{
+ "name": "multi_plugin",
+ "author": "Navidrome Test",
+ "version": "1.0.0",
+ "description": "Test data for multiple services",
+ "website": "https://test.navidrome.org/multi-plugin",
+ "capabilities": ["MetadataAgent", "SchedulerCallback", "LifecycleManagement"],
+ "permissions": {
+ "scheduler": {
+ "reason": "For testing scheduled callback functionality"
+ }
+ }
+}
diff --git a/plugins/testdata/multi_plugin/plugin.go b/plugins/testdata/multi_plugin/plugin.go
new file mode 100644
index 000000000..3c28bd214
--- /dev/null
+++ b/plugins/testdata/multi_plugin/plugin.go
@@ -0,0 +1,124 @@
+//go:build wasip1
+
+package main
+
+import (
+ "context"
+ "log"
+ "strings"
+
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/navidrome/navidrome/plugins/host/scheduler"
+)
+
+// MultiPlugin implements the MetadataAgent interface for testing
+type MultiPlugin struct{}
+
+var ErrNotFound = api.ErrNotFound
+
+var sched = scheduler.NewSchedulerService()
+
+// Artist-related methods
+func (MultiPlugin) GetArtistMBID(ctx context.Context, req *api.ArtistMBIDRequest) (*api.ArtistMBIDResponse, error) {
+ if req.Name != "" {
+ return &api.ArtistMBIDResponse{Mbid: "multi-artist-mbid"}, nil
+ }
+ return nil, ErrNotFound
+}
+
+func (MultiPlugin) GetArtistURL(ctx context.Context, req *api.ArtistURLRequest) (*api.ArtistURLResponse, error) {
+ log.Printf("GetArtistURL received: %v", req)
+
+ // Use an ID that could potentially clash with other plugins
+ // The host will ensure this doesn't conflict by prefixing with plugin name
+ customId := "artist:" + req.Name
+ log.Printf("Registering scheduler with custom ID: %s", customId)
+
+ // Use the scheduler service for one-time scheduling
+ resp, err := sched.ScheduleOneTime(ctx, &scheduler.ScheduleOneTimeRequest{
+ ScheduleId: customId,
+ DelaySeconds: 6,
+ Payload: []byte("test-payload"),
+ })
+ if err != nil {
+ log.Printf("Error scheduling one-time job: %v", err)
+ } else {
+ log.Printf("One-time schedule registered with ID: %s", resp.ScheduleId)
+ }
+
+ return &api.ArtistURLResponse{Url: "https://multi.example.com/artist"}, nil
+}
+
+func (MultiPlugin) GetArtistBiography(ctx context.Context, req *api.ArtistBiographyRequest) (*api.ArtistBiographyResponse, error) {
+ return &api.ArtistBiographyResponse{Biography: "Multi agent artist bio"}, nil
+}
+
+func (MultiPlugin) GetSimilarArtists(ctx context.Context, req *api.ArtistSimilarRequest) (*api.ArtistSimilarResponse, error) {
+ return &api.ArtistSimilarResponse{}, nil
+}
+
+func (MultiPlugin) GetArtistImages(ctx context.Context, req *api.ArtistImageRequest) (*api.ArtistImageResponse, error) {
+ return &api.ArtistImageResponse{}, nil
+}
+
+func (MultiPlugin) GetArtistTopSongs(ctx context.Context, req *api.ArtistTopSongsRequest) (*api.ArtistTopSongsResponse, error) {
+ return &api.ArtistTopSongsResponse{}, nil
+}
+
+// Album-related methods
+func (MultiPlugin) GetAlbumInfo(ctx context.Context, req *api.AlbumInfoRequest) (*api.AlbumInfoResponse, error) {
+ if req.Name != "" && req.Artist != "" {
+ return &api.AlbumInfoResponse{
+ Info: &api.AlbumInfo{
+ Name: req.Name,
+ Mbid: "multi-album-mbid",
+ Description: "Multi agent album description",
+ Url: "https://multi.example.com/album",
+ },
+ }, nil
+ }
+ return nil, ErrNotFound
+}
+
+func (MultiPlugin) GetAlbumImages(ctx context.Context, req *api.AlbumImagesRequest) (*api.AlbumImagesResponse, error) {
+ return &api.AlbumImagesResponse{}, nil
+}
+
+// Scheduler callback
+func (MultiPlugin) OnSchedulerCallback(ctx context.Context, req *api.SchedulerCallbackRequest) (*api.SchedulerCallbackResponse, error) {
+ log.Printf("Scheduler callback received with ID: %s, payload: '%s', isRecurring: %v",
+ req.ScheduleId, string(req.Payload), req.IsRecurring)
+
+ // Demonstrate how to parse the custom ID format
+ if strings.HasPrefix(req.ScheduleId, "artist:") {
+ parts := strings.Split(req.ScheduleId, ":")
+ if len(parts) == 2 {
+ artistName := parts[1]
+ log.Printf("This schedule was for artist: %s", artistName)
+ }
+ }
+
+ return &api.SchedulerCallbackResponse{}, nil
+}
+
+func (MultiPlugin) OnInit(ctx context.Context, req *api.InitRequest) (*api.InitResponse, error) {
+ log.Printf("OnInit called with %v", req)
+
+ // Schedule a recurring every 5 seconds
+ _, _ = sched.ScheduleRecurring(ctx, &scheduler.ScheduleRecurringRequest{
+ CronExpression: "@every 5s",
+ Payload: []byte("every 5 seconds"),
+ })
+
+ return &api.InitResponse{}, nil
+}
+
+// Required by Go WASI build
+func main() {}
+
+// Register the service implementations
+func init() {
+ api.RegisterLifecycleManagement(MultiPlugin{})
+ api.RegisterMetadataAgent(MultiPlugin{})
+ api.RegisterSchedulerCallback(MultiPlugin{})
+}
diff --git a/plugins/testdata/unauthorized_plugin/manifest.json b/plugins/testdata/unauthorized_plugin/manifest.json
new file mode 100644
index 000000000..38a00e0ea
--- /dev/null
+++ b/plugins/testdata/unauthorized_plugin/manifest.json
@@ -0,0 +1,9 @@
+{
+ "name": "unauthorized_plugin",
+ "author": "Navidrome Test",
+ "version": "1.0.0",
+ "description": "Test plugin that tries to access unauthorized services",
+ "website": "https://test.navidrome.org/unauthorized-plugin",
+ "capabilities": ["MetadataAgent"],
+ "permissions": {}
+}
diff --git a/plugins/testdata/unauthorized_plugin/plugin.go b/plugins/testdata/unauthorized_plugin/plugin.go
new file mode 100644
index 000000000..07c3e0f6b
--- /dev/null
+++ b/plugins/testdata/unauthorized_plugin/plugin.go
@@ -0,0 +1,78 @@
+//go:build wasip1
+
+package main
+
+import (
+ "context"
+
+ "github.com/navidrome/navidrome/plugins/api"
+ "github.com/navidrome/navidrome/plugins/host/http"
+)
+
+type UnauthorizedPlugin struct{}
+
+var ErrNotFound = api.ErrNotFound
+
+func (UnauthorizedPlugin) GetAlbumInfo(ctx context.Context, req *api.AlbumInfoRequest) (*api.AlbumInfoResponse, error) {
+ // This plugin attempts to make an HTTP call without having HTTP permission
+ // This should fail since the plugin has no permissions in its manifest
+ httpClient := http.NewHttpService()
+
+ request := &http.HttpRequest{
+ Url: "https://example.com/test",
+ Headers: map[string]string{
+ "Accept": "application/json",
+ },
+ TimeoutMs: 5000,
+ }
+
+ _, err := httpClient.Get(ctx, request)
+ if err != nil {
+ // Expected to fail due to missing permission
+ return nil, err
+ }
+
+ return &api.AlbumInfoResponse{
+ Info: &api.AlbumInfo{
+ Name: req.Name,
+ Mbid: "unauthorized-test",
+ Description: "This should not work",
+ Url: "https://example.com/unauthorized",
+ },
+ }, nil
+}
+
+func (UnauthorizedPlugin) GetAlbumImages(ctx context.Context, req *api.AlbumImagesRequest) (*api.AlbumImagesResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (UnauthorizedPlugin) GetArtistMBID(ctx context.Context, req *api.ArtistMBIDRequest) (*api.ArtistMBIDResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (UnauthorizedPlugin) GetArtistURL(ctx context.Context, req *api.ArtistURLRequest) (*api.ArtistURLResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (UnauthorizedPlugin) GetArtistBiography(ctx context.Context, req *api.ArtistBiographyRequest) (*api.ArtistBiographyResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (UnauthorizedPlugin) GetSimilarArtists(ctx context.Context, req *api.ArtistSimilarRequest) (*api.ArtistSimilarResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (UnauthorizedPlugin) GetArtistImages(ctx context.Context, req *api.ArtistImageRequest) (*api.ArtistImageResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func (UnauthorizedPlugin) GetArtistTopSongs(ctx context.Context, req *api.ArtistTopSongsRequest) (*api.ArtistTopSongsResponse, error) {
+ return nil, api.ErrNotImplemented
+}
+
+func main() {}
+
+// Register the plugin implementation
+func init() {
+ api.RegisterMetadataAgent(UnauthorizedPlugin{})
+}
diff --git a/plugins/wasm_instance_pool.go b/plugins/wasm_instance_pool.go
new file mode 100644
index 000000000..5ea1a82a6
--- /dev/null
+++ b/plugins/wasm_instance_pool.go
@@ -0,0 +1,223 @@
+package plugins
+
+import (
+ "context"
+ "fmt"
+ "sync"
+ "time"
+
+ "github.com/navidrome/navidrome/log"
+)
+
+// wasmInstancePool is a generic pool using channels for simplicity and Go idioms
+type wasmInstancePool[T any] struct {
+ name string
+ new func(ctx context.Context) (T, error)
+ poolSize int
+ getTimeout time.Duration
+ ttl time.Duration
+
+ mu sync.RWMutex
+ instances chan poolItem[T]
+ semaphore chan struct{}
+ closing chan struct{}
+ closed bool
+}
+
+type poolItem[T any] struct {
+ value T
+ created time.Time
+}
+
+func newWasmInstancePool[T any](name string, poolSize int, maxConcurrentInstances int, getTimeout time.Duration, ttl time.Duration, newFn func(ctx context.Context) (T, error)) *wasmInstancePool[T] {
+ p := &wasmInstancePool[T]{
+ name: name,
+ new: newFn,
+ poolSize: poolSize,
+ getTimeout: getTimeout,
+ ttl: ttl,
+ instances: make(chan poolItem[T], poolSize),
+ semaphore: make(chan struct{}, maxConcurrentInstances),
+ closing: make(chan struct{}),
+ }
+
+ // Fill semaphore to allow maxConcurrentInstances
+ for i := 0; i < maxConcurrentInstances; i++ {
+ p.semaphore <- struct{}{}
+ }
+
+ log.Debug(context.Background(), "wasmInstancePool: created new pool", "pool", p.name, "poolSize", p.poolSize, "maxConcurrentInstances", maxConcurrentInstances, "getTimeout", p.getTimeout, "ttl", p.ttl)
+ go p.cleanupLoop()
+ return p
+}
+
+func getInstanceID(inst any) string {
+ return fmt.Sprintf("%p", inst) //nolint:govet
+}
+
+func (p *wasmInstancePool[T]) Get(ctx context.Context) (T, error) {
+ // First acquire a semaphore slot (concurrent limit)
+ select {
+ case <-p.semaphore:
+ // Got slot, continue
+ case <-ctx.Done():
+ var zero T
+ return zero, ctx.Err()
+ case <-time.After(p.getTimeout):
+ var zero T
+ return zero, fmt.Errorf("timeout waiting for available instance after %v", p.getTimeout)
+ case <-p.closing:
+ var zero T
+ return zero, fmt.Errorf("pool is closing")
+ }
+
+ // Try to get from pool first
+ p.mu.RLock()
+ instances := p.instances
+ p.mu.RUnlock()
+
+ select {
+ case item := <-instances:
+ log.Trace(ctx, "wasmInstancePool: got instance from pool", "pool", p.name, "instanceID", getInstanceID(item.value))
+ return item.value, nil
+ default:
+ // Pool empty, create new instance
+ instance, err := p.new(ctx)
+ if err != nil {
+ // Failed to create, return semaphore slot
+ log.Trace(ctx, "wasmInstancePool: failed to create new instance", "pool", p.name, err)
+ p.semaphore <- struct{}{}
+ var zero T
+ return zero, err
+ }
+ log.Trace(ctx, "wasmInstancePool: new instance created", "pool", p.name, "instanceID", getInstanceID(instance))
+ return instance, nil
+ }
+}
+
+func (p *wasmInstancePool[T]) Put(ctx context.Context, v T) {
+ p.mu.RLock()
+ instances := p.instances
+ closed := p.closed
+ p.mu.RUnlock()
+
+ if closed {
+ log.Trace(ctx, "wasmInstancePool: pool closed, closing instance", "pool", p.name, "instanceID", getInstanceID(v))
+ p.closeItem(ctx, v)
+ // Return semaphore slot only if this instance came from Get()
+ select {
+ case p.semaphore <- struct{}{}:
+ case <-p.closing:
+ default:
+ // Semaphore full, this instance didn't come from Get()
+ }
+ return
+ }
+
+ // Try to return to pool
+ item := poolItem[T]{value: v, created: time.Now()}
+ select {
+ case instances <- item:
+ log.Trace(ctx, "wasmInstancePool: returned instance to pool", "pool", p.name, "instanceID", getInstanceID(v))
+ default:
+ // Pool full, close instance
+ log.Trace(ctx, "wasmInstancePool: pool full, closing instance", "pool", p.name, "instanceID", getInstanceID(v))
+ p.closeItem(ctx, v)
+ }
+
+ // Return semaphore slot only if this instance came from Get()
+ // If semaphore is full, this instance didn't come from Get(), so don't block
+ select {
+ case p.semaphore <- struct{}{}:
+ // Successfully returned token
+ case <-p.closing:
+ // Pool closing, don't block
+ default:
+ // Semaphore full, this instance didn't come from Get()
+ }
+}
+
+func (p *wasmInstancePool[T]) Close(ctx context.Context) {
+ p.mu.Lock()
+ if p.closed {
+ p.mu.Unlock()
+ return
+ }
+ p.closed = true
+ close(p.closing)
+ instances := p.instances
+ p.mu.Unlock()
+
+ log.Trace(ctx, "wasmInstancePool: closing pool and all instances", "pool", p.name)
+
+ // Drain and close all instances
+ for {
+ select {
+ case item := <-instances:
+ p.closeItem(ctx, item.value)
+ default:
+ return
+ }
+ }
+}
+
+func (p *wasmInstancePool[T]) cleanupLoop() {
+ ticker := time.NewTicker(p.ttl / 3)
+ defer ticker.Stop()
+ for {
+ select {
+ case <-ticker.C:
+ p.cleanupExpired()
+ case <-p.closing:
+ return
+ }
+ }
+}
+
+func (p *wasmInstancePool[T]) cleanupExpired() {
+ ctx := context.Background()
+ now := time.Now()
+
+ // Create new channel with same capacity
+ newInstances := make(chan poolItem[T], p.poolSize)
+
+ // Atomically swap channels
+ p.mu.Lock()
+ oldInstances := p.instances
+ p.instances = newInstances
+ p.mu.Unlock()
+
+ // Drain old channel, keeping fresh items
+ var expiredCount int
+ for {
+ select {
+ case item := <-oldInstances:
+ if now.Sub(item.created) <= p.ttl {
+ // Item is still fresh, move to new channel
+ select {
+ case newInstances <- item:
+ // Successfully moved
+ default:
+ // New channel full, close excess item
+ p.closeItem(ctx, item.value)
+ }
+ } else {
+ // Item expired, close it
+ expiredCount++
+ p.closeItem(ctx, item.value)
+ }
+ default:
+ // Old channel drained
+ if expiredCount > 0 {
+ log.Trace(ctx, "wasmInstancePool: cleaned up expired instances", "pool", p.name, "expiredCount", expiredCount)
+ }
+ return
+ }
+ }
+}
+
+func (p *wasmInstancePool[T]) closeItem(ctx context.Context, v T) {
+ if closer, ok := any(v).(interface{ Close(context.Context) error }); ok {
+ _ = closer.Close(ctx)
+ }
+}
diff --git a/plugins/wasm_instance_pool_test.go b/plugins/wasm_instance_pool_test.go
new file mode 100644
index 000000000..141210473
--- /dev/null
+++ b/plugins/wasm_instance_pool_test.go
@@ -0,0 +1,193 @@
+package plugins
+
+import (
+ "context"
+ "sync/atomic"
+ "time"
+
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+type testInstance struct {
+ closed atomic.Bool
+}
+
+func (t *testInstance) Close(ctx context.Context) error {
+ t.closed.Store(true)
+ return nil
+}
+
+var _ = Describe("wasmInstancePool", func() {
+ var (
+ ctx = context.Background()
+ )
+
+ It("should Get and Put instances", func() {
+ pool := newWasmInstancePool[*testInstance]("test", 2, 10, 5*time.Second, time.Second, func(ctx context.Context) (*testInstance, error) {
+ return &testInstance{}, nil
+ })
+ inst, err := pool.Get(ctx)
+ Expect(err).To(BeNil())
+ Expect(inst).ToNot(BeNil())
+ pool.Put(ctx, inst)
+ inst2, err := pool.Get(ctx)
+ Expect(err).To(BeNil())
+ Expect(inst2).To(Equal(inst))
+ pool.Close(ctx)
+ })
+
+ It("should not exceed max instances", func() {
+ pool := newWasmInstancePool[*testInstance]("test", 1, 10, 5*time.Second, time.Second, func(ctx context.Context) (*testInstance, error) {
+ return &testInstance{}, nil
+ })
+ inst1, err := pool.Get(ctx)
+ Expect(err).To(BeNil())
+ inst2 := &testInstance{}
+ pool.Put(ctx, inst1)
+ pool.Put(ctx, inst2) // should close inst2
+ Expect(inst2.closed.Load()).To(BeTrue())
+ pool.Close(ctx)
+ })
+
+ It("should expire and close instances after TTL", func() {
+ pool := newWasmInstancePool[*testInstance]("test", 2, 10, 5*time.Second, 100*time.Millisecond, func(ctx context.Context) (*testInstance, error) {
+ return &testInstance{}, nil
+ })
+ inst, err := pool.Get(ctx)
+ Expect(err).To(BeNil())
+ pool.Put(ctx, inst)
+ // Wait for TTL cleanup
+ time.Sleep(300 * time.Millisecond)
+ Expect(inst.closed.Load()).To(BeTrue())
+ pool.Close(ctx)
+ })
+
+ It("should close all on pool Close", func() {
+ pool := newWasmInstancePool[*testInstance]("test", 2, 10, 5*time.Second, time.Second, func(ctx context.Context) (*testInstance, error) {
+ return &testInstance{}, nil
+ })
+ inst1, err := pool.Get(ctx)
+ Expect(err).To(BeNil())
+ inst2, err := pool.Get(ctx)
+ Expect(err).To(BeNil())
+ pool.Put(ctx, inst1)
+ pool.Put(ctx, inst2)
+ pool.Close(ctx)
+ Expect(inst1.closed.Load()).To(BeTrue())
+ Expect(inst2.closed.Load()).To(BeTrue())
+ })
+
+ It("should be safe for concurrent Get/Put", func() {
+ pool := newWasmInstancePool[*testInstance]("test", 4, 10, 5*time.Second, time.Second, func(ctx context.Context) (*testInstance, error) {
+ return &testInstance{}, nil
+ })
+ done := make(chan struct{})
+ for i := 0; i < 8; i++ {
+ go func() {
+ inst, err := pool.Get(ctx)
+ Expect(err).To(BeNil())
+ pool.Put(ctx, inst)
+ done <- struct{}{}
+ }()
+ }
+ for i := 0; i < 8; i++ {
+ <-done
+ }
+ pool.Close(ctx)
+ })
+
+ It("should enforce max concurrent instances limit", func() {
+ callCount := atomic.Int32{}
+ pool := newWasmInstancePool[*testInstance]("test", 2, 3, 100*time.Millisecond, time.Second, func(ctx context.Context) (*testInstance, error) {
+ callCount.Add(1)
+ return &testInstance{}, nil
+ })
+
+ // Get 3 instances (should hit the limit)
+ inst1, err := pool.Get(ctx)
+ Expect(err).To(BeNil())
+ inst2, err := pool.Get(ctx)
+ Expect(err).To(BeNil())
+ inst3, err := pool.Get(ctx)
+ Expect(err).To(BeNil())
+
+ // Should have created exactly 3 instances at this point
+ Expect(callCount.Load()).To(Equal(int32(3)))
+
+ // Fourth call should timeout without creating a new instance
+ start := time.Now()
+ _, err = pool.Get(ctx)
+ duration := time.Since(start)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("timeout waiting for available instance"))
+ Expect(duration).To(BeNumerically(">=", 100*time.Millisecond))
+ Expect(duration).To(BeNumerically("<", 200*time.Millisecond))
+
+ // Still should have only 3 instances (timeout didn't create new one)
+ Expect(callCount.Load()).To(Equal(int32(3)))
+
+ // Return one instance and try again - should succeed by reusing returned instance
+ pool.Put(ctx, inst1)
+ inst4, err := pool.Get(ctx)
+ Expect(err).To(BeNil())
+ Expect(inst4).To(Equal(inst1)) // Should be the same instance we returned
+
+ // Still should have only 3 instances total (reused inst1)
+ Expect(callCount.Load()).To(Equal(int32(3)))
+
+ pool.Put(ctx, inst2)
+ pool.Put(ctx, inst3)
+ pool.Put(ctx, inst4)
+ pool.Close(ctx)
+ })
+
+ It("should handle concurrent waiters properly", func() {
+ pool := newWasmInstancePool[*testInstance]("test", 1, 2, time.Second, time.Second, func(ctx context.Context) (*testInstance, error) {
+ return &testInstance{}, nil
+ })
+
+ // Fill up the concurrent slots
+ inst1, err := pool.Get(ctx)
+ Expect(err).To(BeNil())
+ inst2, err := pool.Get(ctx)
+ Expect(err).To(BeNil())
+
+ // Start multiple waiters
+ waiterResults := make(chan error, 3)
+ for i := 0; i < 3; i++ {
+ go func() {
+ _, err := pool.Get(ctx)
+ waiterResults <- err
+ }()
+ }
+
+ // Wait a bit to ensure waiters are queued
+ time.Sleep(50 * time.Millisecond)
+
+ // Return instances one by one
+ pool.Put(ctx, inst1)
+ pool.Put(ctx, inst2)
+
+ // Two waiters should succeed, one should timeout
+ successCount := 0
+ timeoutCount := 0
+ for i := 0; i < 3; i++ {
+ select {
+ case err := <-waiterResults:
+ if err == nil {
+ successCount++
+ } else {
+ timeoutCount++
+ }
+ case <-time.After(2 * time.Second):
+ Fail("Test timed out waiting for waiter results")
+ }
+ }
+
+ Expect(successCount).To(Equal(2))
+ Expect(timeoutCount).To(Equal(1))
+
+ pool.Close(ctx)
+ })
+})
diff --git a/reflex.conf b/reflex.conf
index 2eb4d131c..4cd64baf9 100644
--- a/reflex.conf
+++ b/reflex.conf
@@ -1 +1 @@
--s -r "(\.go$$|\.cpp$$|\.h$$|navidrome.toml|resources|token_received.html)" -R "(^ui|^data|^db/migrations)" -- go run -race -tags netgo .
+-s -r "(\.go$$|\.cpp$$|\.h$$|\.wasm$$|navidrome.toml|resources|token_received.html)" -R "(^ui|^data|^db/migrations)" -- go run -race -tags netgo .
diff --git a/release/goreleaser.yml b/release/goreleaser.yml
index 1a420c927..30c0d6f3b 100644
--- a/release/goreleaser.yml
+++ b/release/goreleaser.yml
@@ -25,7 +25,8 @@ builds:
archives:
- format_overrides:
- goos: windows
- format: zip
+ formats:
+ - zip
name_template: '{{ .ProjectName }}_{{ .Version }}_{{ .Os }}_{{ .Arch }}{{ with .Arm }}{{ . }}{{ end }}{{ with .Mips }}_{{ . }}{{ end }}{{ if not (eq .Amd64 "v1") }}{{ .Amd64 }}{{ end }}'
checksum:
@@ -82,6 +83,15 @@ nfpms:
owner: navidrome
group: navidrome
+ - src: release/linux/.package.rpm # contents: "rpm"
+ dst: /var/lib/navidrome/.package
+ type: "config|noreplace"
+ packager: rpm
+ - src: release/linux/.package.deb # contents: "deb"
+ dst: /var/lib/navidrome/.package
+ type: "config|noreplace"
+ packager: deb
+
scripts:
preinstall: "release/linux/preinstall.sh"
postinstall: "release/linux/postinstall.sh"
diff --git a/release/linux/.package.deb b/release/linux/.package.deb
new file mode 100644
index 000000000..811c85f42
--- /dev/null
+++ b/release/linux/.package.deb
@@ -0,0 +1 @@
+deb
\ No newline at end of file
diff --git a/release/linux/.package.rpm b/release/linux/.package.rpm
new file mode 100644
index 000000000..7c88ef3c0
--- /dev/null
+++ b/release/linux/.package.rpm
@@ -0,0 +1 @@
+rpm
\ No newline at end of file
diff --git a/release/linux/postinstall.sh b/release/linux/postinstall.sh
index 65f1d208d..f3d9c9277 100644
--- a/release/linux/postinstall.sh
+++ b/release/linux/postinstall.sh
@@ -4,7 +4,7 @@
# the package manager (in particular, deb) thinks that the file exists, while it is
# no longer on disk. Specifically, doing a `rm /etc/navidrome/navidrome.toml`
# without something like `apt purge navidrome` will result in the system believing that
-# the file still exists. In this case, during isntall it will NOT extract the configuration
+# the file still exists. In this case, during install it will NOT extract the configuration
# file (as to not override it). Since `navidrome service install` depends on this file existing,
# we will create it with the defaults anyway.
if [ ! -f /etc/navidrome/navidrome.toml ]; then
diff --git a/release/wix/build_msi.sh b/release/wix/build_msi.sh
index 9fc008446..7e595311e 100755
--- a/release/wix/build_msi.sh
+++ b/release/wix/build_msi.sh
@@ -49,6 +49,9 @@ cp "${DOWNLOAD_FOLDER}"/extracted_ffmpeg/${FFMPEG_FILE}/bin/ffmpeg.exe "$MSI_OUT
cp "$WORKSPACE"/LICENSE "$WORKSPACE"/README.md "$MSI_OUTPUT_DIR"
cp "$BINARY" "$MSI_OUTPUT_DIR"
+# package type indicator file
+echo "msi" > "$MSI_OUTPUT_DIR/.package"
+
# workaround for wixl WixVariable not working to override bmp locations
cp "$WORKSPACE"/release/wix/bmp/banner.bmp /usr/share/wixl-*/ext/ui/bitmaps/bannrbmp.bmp
cp "$WORKSPACE"/release/wix/bmp/dialogue.bmp /usr/share/wixl-*/ext/ui/bitmaps/dlgbmp.bmp
diff --git a/release/wix/navidrome.wxs b/release/wix/navidrome.wxs
index ec8b164e8..8ebba4632 100644
--- a/release/wix/navidrome.wxs
+++ b/release/wix/navidrome.wxs
@@ -69,6 +69,12 @@
+
+
+
+
+
+
@@ -81,6 +87,7 @@
+
diff --git a/resources/i18n/bs.json b/resources/i18n/bs.json
new file mode 100644
index 000000000..9d5c552e7
--- /dev/null
+++ b/resources/i18n/bs.json
@@ -0,0 +1,628 @@
+{
+ "languageName": "Bosanski",
+ "resources": {
+ "song": {
+ "name": "Pjesma |||| Pjesme",
+ "fields": {
+ "albumArtist": "Izvođač albuma",
+ "duration": "Trajanje",
+ "trackNumber": "Pjesma #",
+ "playCount": "Reprodukcija",
+ "title": "Naslov",
+ "artist": "Izvođač",
+ "album": "Album",
+ "path": "Putanja datoteke",
+ "genre": "Žanr",
+ "compilation": "Kompilacija",
+ "year": "Godina",
+ "size": "Veličina datoteke",
+ "updatedAt": "Dodano",
+ "bitRate": "Brzina prijenosa",
+ "discSubtitle": "Podnaslov CD-a",
+ "starred": "Favorit",
+ "comment": "Komentar",
+ "rating": "Ocjena",
+ "quality": "Kvaliteta",
+ "bpm": "BPM",
+ "playDate": "Posljednja reprodukcija",
+ "channels": "Kanali",
+ "createdAt": "Dodano",
+ "grouping": "Grupisanje",
+ "mood": "Raspoloženje",
+ "participants": "Dodatni učesnici",
+ "tags": "Dodatne oznake",
+ "mappedTags": "Mapirane oznake",
+ "rawTags": "Sirovi podaci oznaka",
+ "bitDepth": "Dubina bita",
+ "sampleRate": "Uzorkovanje",
+ "missing": "Nedostaje",
+ "libraryName": "Biblioteka"
+ },
+ "actions": {
+ "addToQueue": "Reprodukcija kasnije",
+ "playNow": "Reprodukcija sada",
+ "addToPlaylist": "Dodaj u playlistu",
+ "shuffleAll": "Nasumična reprodukcija",
+ "download": "Preuzmi",
+ "playNext": "Reprodukcija sljedeće",
+ "info": "Više informacija",
+ "showInPlaylist": "Prikaži u playlisti"
+ }
+ },
+ "album": {
+ "name": "Album |||| Albumi",
+ "fields": {
+ "albumArtist": "Izvođač albuma",
+ "artist": "Izvođač",
+ "duration": "Trajanje",
+ "songCount": "Broj pjesama",
+ "playCount": "Reprodukcija",
+ "name": "Naziv",
+ "genre": "Žanr",
+ "compilation": "Kompilacija",
+ "year": "Godina",
+ "updatedAt": "Ažurirano",
+ "comment": "Komentar",
+ "rating": "Ocjena",
+ "createdAt": "Dodano",
+ "size": "Veličina",
+ "originalDate": "Originalni datum",
+ "releaseDate": "Datum izdanja",
+ "releases": "Izdanje |||| Izdanja",
+ "released": "Objavljeno",
+ "recordLabel": "Izdavač",
+ "catalogNum": "Kataloški broj",
+ "releaseType": "Tip",
+ "grouping": "Grupisanje",
+ "media": "Medij",
+ "mood": "Raspoloženje",
+ "date": "Datum snimanja",
+ "missing": "Nedostaje",
+ "libraryName": "Biblioteka"
+ },
+ "actions": {
+ "playAll": "Reprodukcija",
+ "playNext": "Reprodukcija sljedeće",
+ "addToQueue": "Dodaj u red",
+ "shuffle": "Nasumična reprodukcija",
+ "addToPlaylist": "Dodaj u playlistu",
+ "download": "Preuzmi",
+ "info": "Više informacija",
+ "share": "Podijeli"
+ },
+ "lists": {
+ "all": "Sve",
+ "random": "Nasumično",
+ "recentlyAdded": "Nedavno dodano",
+ "recentlyPlayed": "Nedavno reproducirano",
+ "mostPlayed": "Najviše reproducirano",
+ "starred": "Favoriti",
+ "topRated": "Najbolje ocijenjeno"
+ }
+ },
+ "artist": {
+ "name": "Izvođač |||| Izvođači",
+ "fields": {
+ "name": "Naziv",
+ "albumCount": "Broj albuma",
+ "songCount": "Broj pjesama",
+ "playCount": "Reprodukcija",
+ "rating": "Ocjena",
+ "genre": "Žanr",
+ "size": "Veličina",
+ "role": "Uloga",
+ "missing": "Nedostaje"
+ },
+ "roles": {
+ "albumartist": "Izvođač albuma |||| Izvođači albuma",
+ "artist": "Izvođač |||| Izvođači",
+ "composer": "Kompozitor |||| Kompozitori",
+ "conductor": "Dirigent |||| Dirigenti",
+ "lyricist": "Tekstopisac |||| Tekstopisci",
+ "arranger": "Aranžer |||| Aranžeri",
+ "producer": "Producent |||| Producenti",
+ "director": "Direktor |||| Direktori",
+ "engineer": "Inženjer |||| Inženjeri",
+ "mixer": "Mikser |||| Mikseri",
+ "remixer": "Remikser |||| Remikseri",
+ "djmixer": "DJ Mikser |||| DJ Mikseri",
+ "performer": "Izvođač |||| Izvođači",
+ "maincredit": "Izvođač albuma ili izvođač |||| Izvođači albuma ili izvođači"
+ },
+ "actions": {
+ "shuffle": "Nasumična reprodukcija",
+ "radio": "Radio",
+ "topSongs": "Najpopularnije pjesme"
+ }
+ },
+ "user": {
+ "name": "Korisnik |||| Korisnici",
+ "fields": {
+ "userName": "Korisničko ime",
+ "isAdmin": "Je admin",
+ "lastLoginAt": "Posljednja prijava",
+ "updatedAt": "Ažurirano",
+ "name": "Ime",
+ "password": "Lozinka",
+ "createdAt": "Kreirano",
+ "changePassword": "Promijeni lozinku?",
+ "currentPassword": "Trenutna lozinka",
+ "newPassword": "Nova lozinka",
+ "token": "Token",
+ "lastAccessAt": "Posljednji pristup",
+ "libraries": "Biblioteke"
+ },
+ "helperTexts": {
+ "name": "Promjena će biti aktivna nakon sljedeće prijave",
+ "libraries": "Odaberi specifične biblioteke za ovog korisnika ili ostavi prazno za standardne biblioteke"
+ },
+ "notifications": {
+ "created": "Korisnik kreiran",
+ "updated": "Korisnik ažuriran",
+ "deleted": "Korisnik obrisan"
+ },
+ "message": {
+ "listenBrainzToken": "Unesite svoj ListenBrainz korisnički token",
+ "clickHereForToken": "Kliknite ovdje za dobijanje tokena",
+ "selectAllLibraries": "Odaberi sve biblioteke",
+ "adminAutoLibraries": "Administratori automatski imaju pristup svim bibliotekama"
+ },
+ "validation": {
+ "librariesRequired": "Ne-administratori moraju imati barem jednu odabranu biblioteku"
+ }
+ },
+ "player": {
+ "name": "Player |||| Playeri",
+ "fields": {
+ "name": "Naziv",
+ "transcodingId": "ID transkodiranja",
+ "maxBitRate": "Maks. brzina prijenosa",
+ "client": "Klijent",
+ "userName": "Korisničko ime",
+ "lastSeen": "Posljednji put viđen",
+ "reportRealPath": "Prikaži stvarnu putanju",
+ "scrobbleEnabled": "Slanje podataka o reprodukciji (scrobbling)"
+ }
+ },
+ "transcoding": {
+ "name": "Transkodiranje |||| Transkodiranja",
+ "fields": {
+ "name": "Naziv",
+ "targetFormat": "Ciljani format",
+ "defaultBitRate": "Zadana brzina prijenosa",
+ "command": "Komanda"
+ }
+ },
+ "playlist": {
+ "name": "Playlista |||| Playliste",
+ "fields": {
+ "name": "Naziv",
+ "duration": "Trajanje",
+ "ownerName": "Vlasnik",
+ "public": "Javna",
+ "updatedAt": "Ažurirano",
+ "createdAt": "Kreirano",
+ "songCount": "Broj pjesama",
+ "comment": "Komentar",
+ "sync": "Auto-uvoz",
+ "path": "Uvezi iz"
+ },
+ "actions": {
+ "selectPlaylist": "Odaberi playlistu:",
+ "addNewPlaylist": "Kreiraj \"%{name}\"",
+ "export": "Izvezi",
+ "makePublic": "Učini javnom",
+ "makePrivate": "Učini privatnom",
+ "saveQueue": "Sačuvaj red čekanja u playlistu",
+ "searchOrCreate": "Pretraži playlistu ili kreiraj novu...",
+ "pressEnterToCreate": "Pritisni Enter za kreiranje nove playliste",
+ "removeFromSelection": "Ukloni iz odabira"
+ },
+ "message": {
+ "duplicate_song": "Dodaj duplikate",
+ "song_exist": "Neke pjesme su već u playlisti. Želiš li ih ipak dodati ili preskočiti?",
+ "noPlaylistsFound": "Nije pronađena nijedna playlista",
+ "noPlaylists": "Nema playlisti"
+ }
+ },
+ "radio": {
+ "name": "Radio |||| Radiji",
+ "fields": {
+ "name": "Naziv",
+ "streamUrl": "Stream URL",
+ "homePageUrl": "URL početne stranice",
+ "updatedAt": "Ažurirano",
+ "createdAt": "Dodano"
+ },
+ "actions": {
+ "playNow": "Reprodukcija sada"
+ }
+ },
+ "share": {
+ "name": "Dijeljenje |||| Dijeljenja",
+ "fields": {
+ "username": "Podijeljeno od strane",
+ "url": "URL",
+ "description": "Opis",
+ "contents": "Sadržaj",
+ "expiresAt": "Vrijedi do",
+ "lastVisitedAt": "Posljednja posjeta",
+ "visitCount": "Posjete",
+ "format": "Format",
+ "maxBitRate": "Maks. brzina prijenosa",
+ "updatedAt": "Ažurirano",
+ "createdAt": "Kreirano",
+ "downloadable": "Dozvoli preuzimanje?"
+ }
+ },
+ "missing": {
+ "name": "Nedostajuća datoteka |||| Nedostajuće datoteke",
+ "fields": {
+ "path": "Putanja",
+ "size": "Veličina",
+ "updatedAt": "Nedostaje od",
+ "libraryName": "Biblioteka"
+ },
+ "actions": {
+ "remove": "Ukloni",
+ "remove_all": "ukloni sve"
+ },
+ "notifications": {
+ "removed": "Nedostajuća(e) datoteka(e) uklonjena(e)"
+ },
+ "empty": "nema nedostajućih datoteka"
+ },
+ "library": {
+ "name": "Biblioteka |||| Biblioteke",
+ "fields": {
+ "name": "Naziv",
+ "path": "Putanja",
+ "remotePath": "Udaljena putanja",
+ "lastScanAt": "Posljednje skeniranje",
+ "songCount": "Pjesme",
+ "albumCount": "Albumi",
+ "artistCount": "Izvođači",
+ "totalSongs": "Pjesme",
+ "totalAlbums": "Albumi",
+ "totalArtists": "Izvođači",
+ "totalFolders": "Folderi",
+ "totalFiles": "Datoteke",
+ "totalMissingFiles": "Nedostajuće datoteke",
+ "totalSize": "Veličina",
+ "totalDuration": "Trajanje",
+ "defaultNewUsers": "Standardno za nove korisnike",
+ "createdAt": "Kreirano",
+ "updatedAt": "Ažurirano"
+ },
+ "sections": {
+ "basic": "Osnovne informacije",
+ "statistics": "Statistika"
+ },
+ "actions": {
+ "scan": "Skeniraj biblioteku",
+ "manageUsers": "Upravljaj pristupima",
+ "viewDetails": "Pogledaj detalje"
+ },
+ "notifications": {
+ "created": "Biblioteka uspješno kreirana",
+ "updated": "Biblioteka uspješno ažurirana",
+ "deleted": "Biblioteka uspješno obrisana",
+ "scanStarted": "Skeniranje biblioteke započeto",
+ "scanCompleted": "Skeniranje biblioteke završeno"
+ },
+ "validation": {
+ "nameRequired": "Naziv biblioteke je obavezan",
+ "pathRequired": "Putanja biblioteke je obavezna",
+ "pathNotDirectory": "Putanja biblioteke mora biti folder",
+ "pathNotFound": "Putanja biblioteke nije pronađena",
+ "pathNotAccessible": "Putanja biblioteke nije dostupna",
+ "pathInvalid": "Putanja biblioteke nije validna"
+ },
+ "messages": {
+ "deleteConfirm": "Da li zaista želiš obrisati ovu biblioteku? Pristup i podaci će biti uklonjeni.",
+ "scanInProgress": "Skeniranje biblioteke u toku...",
+ "noLibrariesAssigned": "Nema dodijeljenih biblioteka"
+ }
+ }
+ },
+ "ra": {
+ "auth": {
+ "welcome1": "Hvala što ste instalirali Navidrome!",
+ "welcome2": "Prvo kreirajte admin korisnika",
+ "confirmPassword": "Potvrdi lozinku",
+ "buttonCreateAdmin": "Kreiraj admina",
+ "auth_check_error": "Prijavite se da biste nastavili",
+ "user_menu": "Profil",
+ "username": "Korisničko ime",
+ "password": "Lozinka",
+ "sign_in": "Prijava",
+ "sign_in_error": "Greška pri prijavi",
+ "logout": "Odjava",
+ "insightsCollectionNote": "Navidrome prikuplja anonimne statistike \nda podrži razvoj projekta. \nKliknite [ovdje] za više informacija ili da isključite \"Insights\""
+ },
+ "validation": {
+ "invalidChars": "Koristite samo slova i brojeve",
+ "passwordDoesNotMatch": "Lozinke se ne podudaraju",
+ "required": "Obavezno",
+ "minLength": "Mora imati najmanje %{min} znakova",
+ "maxLength": "Mora imati najviše %{max} znakova",
+ "minValue": "Mora biti najmanje %{min}",
+ "maxValue": "Mora biti %{max} ili manje",
+ "number": "Mora biti broj",
+ "email": "Mora biti validna e-mail adresa",
+ "oneOf": "Mora biti jedan od: %{options}",
+ "regex": "Mora odgovarati regularnom izrazu: %{pattern}",
+ "unique": "Mora biti jedinstveno",
+ "url": "Mora biti validan URL"
+ },
+ "action": {
+ "add_filter": "Dodaj filter",
+ "add": "Dodaj",
+ "back": "Nazad",
+ "bulk_actions": "1 odabrana stavka |||| %{smart_count} odabrane stavke",
+ "cancel": "Otkaži",
+ "clear_input_value": "Obriši unos",
+ "clone": "Kloniraj",
+ "confirm": "Potvrdi",
+ "create": "Kreiraj",
+ "delete": "Obriši",
+ "edit": "Uredi",
+ "export": "Izvezi",
+ "list": "Lista",
+ "refresh": "Osvježi",
+ "remove_filter": "Ukloni filter",
+ "remove": "Ukloni",
+ "save": "Sačuvaj",
+ "search": "Pretraži",
+ "show": "Prikaži",
+ "sort": "Sortiraj",
+ "undo": "Poništi",
+ "expand": "Proširi",
+ "close": "Zatvori",
+ "open_menu": "Otvori meni",
+ "close_menu": "Zatvori meni",
+ "unselect": "Poništi odabir",
+ "skip": "Preskoči",
+ "bulk_actions_mobile": "1 |||| %{smart_count}",
+ "share": "Podijeli",
+ "download": "Preuzmi"
+ },
+ "boolean": {
+ "true": "Da",
+ "false": "Ne"
+ },
+ "page": {
+ "create": "Kreiraj %{name}",
+ "dashboard": "Kontrolna tabla",
+ "edit": "%{name} #%{id}",
+ "error": "Nešto je pošlo po zlu",
+ "list": "%{name}",
+ "loading": "Učitavanje",
+ "not_found": "Nije pronađeno",
+ "show": "%{name} #%{id}",
+ "empty": "Još nema %{name}.",
+ "invite": "Želiš li dodati jednu?"
+ },
+ "input": {
+ "file": {
+ "upload_several": "Povuci datoteke ovdje za prijenos ili klikni za odabir.",
+ "upload_single": "Povuci datoteku ovdje za prijenos ili klikni za odabir."
+ },
+ "image": {
+ "upload_several": "Povuci slike ovdje za prijenos ili klikni za odabir.",
+ "upload_single": "Povuci sliku ovdje za prijenos ili klikni za odabir."
+ },
+ "references": {
+ "all_missing": "Povezane reference nisu pronađene.",
+ "many_missing": "Neke povezane reference više nisu dostupne.",
+ "single_missing": "Povezana referenca više nije dostupna."
+ },
+ "password": {
+ "toggle_visible": "Sakrij lozinku",
+ "toggle_hidden": "Prikaži lozinku"
+ }
+ },
+ "message": {
+ "about": "O aplikaciji",
+ "are_you_sure": "Jesi li siguran?",
+ "bulk_delete_content": "Da li zaista želiš obrisati \"%{name}\"? |||| Da li zaista želiš obrisati %{smart_count} stavki?",
+ "bulk_delete_title": "Obriši %{name} |||| Obriši %{smart_count} %{name} stavki",
+ "delete_content": "Da li zaista želiš obrisati ovaj sadržaj?",
+ "delete_title": "Obriši %{name} #%{id}",
+ "details": "Detalji",
+ "error": "Došlo je do greške i zahtjev nije mogao biti završen.",
+ "invalid_form": "Formular nije validan. Provjeri unose.",
+ "loading": "Stranica se učitava",
+ "no": "Ne",
+ "not_found": "Stranica nije pronađena.",
+ "yes": "Da",
+ "unsaved_changes": "Neke promjene nisu sačuvane. Želiš li ih ignorisati?"
+ },
+ "navigation": {
+ "no_results": "Nema rezultata",
+ "no_more_results": "Stranica %{page} nema sadržaja.",
+ "page_out_of_boundaries": "Stranica %{page} je izvan opsega",
+ "page_out_from_end": "Posljednja stranica",
+ "page_out_from_begin": "Prva stranica",
+ "page_range_info": "%{offsetBegin}-%{offsetEnd} od %{total}",
+ "page_rows_per_page": "Redova po stranici:",
+ "next": "Sljedeća",
+ "prev": "Prethodna",
+ "skip_nav": "Preskoči na sadržaj"
+ },
+ "notification": {
+ "updated": "Stavka ažurirana |||| %{smart_count} stavki ažurirano",
+ "created": "Stavka kreirana",
+ "deleted": "Stavka obrisana |||| %{smart_count} stavki obrisano",
+ "bad_item": "Neispravna stavka",
+ "item_doesnt_exist": "Stavka ne postoji",
+ "http_error": "Greška u komunikaciji sa serverom",
+ "data_provider_error": "Greška u dataProvider-u. Provjeri konzolu za detalje.",
+ "i18n_error": "Prijevod za odabrani jezik nije dostupan",
+ "canceled": "Akcija otkazana",
+ "logged_out": "Sesija je istekla. Ponovo se prijavi.",
+ "new_version": "Nova verzija dostupna! Osveži stranicu."
+ },
+ "toggleFieldsMenu": {
+ "columnsToDisplay": "Odaberi kolone",
+ "layout": "Izgled",
+ "grid": "Mreža",
+ "table": "Tabela"
+ }
+ },
+ "message": {
+ "note": "NAPOMENA",
+ "transcodingDisabled": "Izmjena postavki transkodiranja preko web sučelja je onemogućena iz sigurnosnih razloga. Ako želiš promijeniti opcije transkodiranja (urediti ili dodati), ponovo pokreni server sa konfiguracijskom opcijom %{config}.",
+ "transcodingEnabled": "Navidrome trenutno radi sa %{config}, što omogućava izvršavanje sistemskih komandi kroz postavke transkodiranja preko web sučelja. Preporučujemo da ovo onemogućiš iz sigurnosnih razloga i koristiš samo prilikom konfiguracije transkodiranja.",
+ "songsAddedToPlaylist": "1 pjesma dodana u playlistu |||| %{smart_count} pjesme dodane u playlistu",
+ "noPlaylistsAvailable": "Nema dostupnih playlisti",
+ "delete_user_title": "Obriši korisnika '%{name}'",
+ "delete_user_content": "Da li zaista želiš obrisati ovog korisnika i sve njegove podatke (uključujući playliste i postavke)?",
+ "notifications_blocked": "Blokirali ste obavijesti za ovu stranicu u postavkama preglednika",
+ "notifications_not_available": "Ovaj preglednik ne podržava desktop obavijesti",
+ "lastfmLinkSuccess": "Last.fm veza uspostavljena i scrobbling omogućen",
+ "lastfmLinkFailure": "Last.fm veza nije uspjela",
+ "lastfmUnlinkSuccess": "Last.fm veza uklonjena i scrobbling onemogućen",
+ "lastfmUnlinkFailure": "Last.fm veza nije uklonjena",
+ "openIn": {
+ "lastfm": "Prikaži na Last.fm",
+ "musicbrainz": "Prikaži na MusicBrainz"
+ },
+ "lastfmLink": "Pročitaj više",
+ "listenBrainzLinkSuccess": "ListenBrainz veza uspostavljena i scrobbling omogućen kao korisnik: %{user}",
+ "listenBrainzLinkFailure": "ListenBrainz veza nije uspjela: %{error}",
+ "listenBrainzUnlinkSuccess": "ListenBrainz veza uklonjena i scrobbling onemogućen",
+ "listenBrainzUnlinkFailure": "ListenBrainz veza nije uklonjena",
+ "downloadOriginalFormat": "Preuzmi u originalnom formatu",
+ "shareOriginalFormat": "Podijeli u originalnom formatu",
+ "shareDialogTitle": "Podijeli %{resource} '%{name}'",
+ "shareBatchDialogTitle": "Podijeli 1 %{resource} |||| Podijeli %{smart_count} %{resource}",
+ "shareSuccess": "URL kopiran u međuspremnik: %{url}",
+ "shareFailure": "Greška pri kopiranju URL-a %{url} u međuspremnik",
+ "downloadDialogTitle": "Preuzmi %{resource} '%{name}' (%{size})",
+ "shareCopyToClipboard": "Kopiraj u međuspremnik: Ctrl+C, Enter",
+ "remove_missing_title": "Ukloni nedostajuće datoteke",
+ "remove_missing_content": "Da li zaista želiš ukloniti odabrane nedostajuće datoteke iz baze podataka? Sve reference na datoteke (broj reprodukcija, ocjene) bit će trajno obrisane.",
+ "remove_all_missing_title": "Ukloni sve nedostajuće datoteke",
+ "remove_all_missing_content": "Da li zaista želiš ukloniti sve nedostajuće datoteke iz baze podataka? Sve reference na datoteke (broj reprodukcija, ocjene) bit će trajno obrisane.",
+ "noSimilarSongsFound": "Nema sličnih pjesama",
+ "noTopSongsFound": "Nema popularnih pjesama"
+ },
+ "menu": {
+ "library": "Biblioteka",
+ "settings": "Postavke",
+ "version": "Verzija",
+ "theme": "Tema",
+ "personal": {
+ "name": "Lično",
+ "options": {
+ "theme": "Tema",
+ "language": "Jezik",
+ "defaultView": "Zadani pregled",
+ "desktop_notifications": "Desktop obavijesti",
+ "lastfmScrobbling": "Last.fm scrobbling",
+ "listenBrainzScrobbling": "ListenBrainz scrobbling",
+ "replaygain": "ReplayGain mod",
+ "preAmp": "ReplayGain pojačanje (dB)",
+ "gain": {
+ "none": "Isključeno",
+ "album": "Koristi album gain",
+ "track": "Koristi pjesmu gain"
+ },
+ "lastfmNotConfigured": "Last.fm API ključ nije konfiguriran"
+ }
+ },
+ "albumList": "Albumi",
+ "about": "O aplikaciji",
+ "playlists": "Playliste",
+ "sharedPlaylists": "Dijeljene playliste",
+ "librarySelector": {
+ "allLibraries": "Sve biblioteke (%{count})",
+ "multipleLibraries": "%{selected} od %{total} biblioteka",
+ "selectLibraries": "Odaberi biblioteke",
+ "none": "Nijedna"
+ }
+ },
+ "player": {
+ "playListsText": "Reprodukcija reda čekanja",
+ "openText": "Otvori",
+ "closeText": "Zatvori",
+ "notContentText": "Nema muzike",
+ "clickToPlayText": "Klikni za reprodukciju",
+ "clickToPauseText": "Klikni za pauzu",
+ "nextTrackText": "Sljedeća pjesma",
+ "previousTrackText": "Prethodna pjesma",
+ "reloadText": "Ponovo učitaj",
+ "volumeText": "Glasnoća",
+ "toggleLyricText": "Prikaži/sakrij tekst",
+ "toggleMiniModeText": "Minimiziraj",
+ "destroyText": "Uništi",
+ "downloadText": "Preuzmi",
+ "removeAudioListsText": "Ukloni audio liste",
+ "clickToDeleteText": "Klikni za brisanje %{name}",
+ "emptyLyricText": "Nema teksta",
+ "playModeText": {
+ "order": "Redom",
+ "orderLoop": "Ponavljaj",
+ "singleLoop": "Ponavljaj jednu",
+ "shufflePlay": "Nasumična reprodukcija"
+ }
+ },
+ "about": {
+ "links": {
+ "homepage": "Početna stranica",
+ "source": "Izvorni kod",
+ "featureRequests": "Zahtjevi za funkcijama",
+ "lastInsightsCollection": "Posljednje prikupljanje \"Insights\"",
+ "insights": {
+ "disabled": "Isključeno",
+ "waiting": "Čekanje"
+ }
+ },
+ "tabs": {
+ "about": "O aplikaciji",
+ "config": "Konfiguracija"
+ },
+ "config": {
+ "configName": "Postavka",
+ "environmentVariable": "Varijabla okruženja",
+ "currentValue": "Vrijednost",
+ "configurationFile": "Konfiguracijska datoteka",
+ "exportToml": "Izvezi konfiguraciju (TOML)",
+ "exportSuccess": "Konfiguracija kopirana u međuspremnik u TOML formatu",
+ "exportFailed": "Greška pri kopiranju konfiguracije",
+ "devFlagsHeader": "Dev postavke (mogu se promijeniti)",
+ "devFlagsComment": "Eksperimentalne postavke koje mogu biti uklonjene ili promijenjene u budućnosti"
+ }
+ },
+ "activity": {
+ "title": "Aktivnost",
+ "totalScanned": "Ukupno skeniranih foldera",
+ "quickScan": "Brzo skeniranje",
+ "fullScan": "Potpuno skeniranje",
+ "serverUptime": "Vrijeme rada servera",
+ "serverDown": "ISKLJUČEN",
+ "scanType": "Tip",
+ "status": "Greška pri skeniranju",
+ "elapsedTime": "Proteklo vrijeme"
+ },
+ "help": {
+ "title": "Navidrome prečice",
+ "hotkeys": {
+ "show_help": "Prikaži ovu pomoć",
+ "toggle_menu": "Uključi/isključi bočnu traku",
+ "toggle_play": "Reprodukcija / Pauza",
+ "prev_song": "Prethodna pjesma",
+ "next_song": "Sljedeća pjesma",
+ "vol_up": "Glasnije",
+ "vol_down": "Tiše",
+ "toggle_love": "Dodaj u favorite",
+ "current_song": "Prikaži trenutnu pjesmu"
+ }
+ },
+ "nowPlaying": {
+ "title": "Trenutna reprodukcija",
+ "empty": "Nema reprodukcije",
+ "minutesAgo": "Prije %{smart_count} minute |||| Prije %{smart_count} minuta"
+ }
+}
diff --git a/resources/i18n/da.json b/resources/i18n/da.json
index 7d4258d2a..105a20732 100644
--- a/resources/i18n/da.json
+++ b/resources/i18n/da.json
@@ -1,460 +1,628 @@
{
- "languageName": "Dansk",
- "resources": {
- "song": {
- "name": "Sang |||| Sange",
- "fields": {
- "albumArtist": "Album kunstner",
- "duration": "Varighed",
- "trackNumber": "#",
- "playCount": "Afspilninger",
- "title": "Titel",
- "artist": "Kunstner",
- "album": "Album navn",
- "path": "Fil placering",
- "genre": "Genre",
- "compilation": "Samling",
- "year": "År",
- "size": "Fil størrelse",
- "updatedAt": "Opdateret den",
- "bitRate": "Bitrate",
- "discSubtitle": "Plade undernavn",
- "starred": "Stjernemarkeret",
- "comment": "Kommentar",
- "rating": "",
- "quality": "",
- "bpm": "",
- "playDate": "",
- "channels": "",
- "createdAt": ""
- },
- "actions": {
- "addToQueue": "Afspil senere",
- "playNow": "Afspil nu",
- "addToPlaylist": "Tilføj til afspilningsliste",
- "shuffleAll": "Bland alle",
- "download": "Hent",
- "playNext": "Spil næste",
- "info": ""
- }
- },
- "album": {
- "name": "Album |||| Albums",
- "fields": {
- "albumArtist": "Album kunstner",
- "artist": "Kunstner",
- "duration": "Varighed",
- "songCount": "Sange",
- "playCount": "Afspilninger",
- "name": "Navn",
- "genre": "Genre",
- "compilation": "Samling",
- "year": "År",
- "updatedAt": "Opdateret den",
- "comment": "Kommentar",
- "rating": "",
- "createdAt": "",
- "size": "",
- "originalDate": "",
- "releaseDate": "",
- "releases": "",
- "released": ""
- },
- "actions": {
- "playAll": "Afspil",
- "playNext": "Afspil næste",
- "addToQueue": "Afspil senere",
- "shuffle": "Bland",
- "addToPlaylist": "Tilføj til afspilningsliste",
- "download": "Hent",
- "info": "",
- "share": ""
- },
- "lists": {
- "all": "Alle",
- "random": "Tilfældig",
- "recentlyAdded": "Nyligt tilføjet",
- "recentlyPlayed": "Nyligt Afspillet",
- "mostPlayed": "Mest Afspillet",
- "starred": "Stjernemarkeret",
- "topRated": ""
- }
- },
- "artist": {
- "name": "Kunstner |||| Kunstnere",
- "fields": {
- "name": "Navn",
- "albumCount": "Antal album",
- "songCount": "Antal sange",
- "playCount": "Afspilninger",
- "rating": "",
- "genre": "",
- "size": ""
- }
- },
- "user": {
- "name": "Bruger |||| Brugere",
- "fields": {
- "userName": "Brugernavn",
- "isAdmin": "Er administrator",
- "lastLoginAt": "Sidste login",
- "updatedAt": "Opdateret den",
- "name": "Navn",
- "password": "Kodeord",
- "createdAt": "Oprettet den",
- "changePassword": "",
- "currentPassword": "",
- "newPassword": "",
- "token": ""
- },
- "helperTexts": {
- "name": ""
- },
- "notifications": {
- "created": "",
- "updated": "",
- "deleted": ""
- },
- "message": {
- "listenBrainzToken": "",
- "clickHereForToken": ""
- }
- },
- "player": {
- "name": "Afspiller |||| Afspillere",
- "fields": {
- "name": "Navn",
- "transcodingId": "Omkodning",
- "maxBitRate": "Maks. bitrate",
- "client": "Klient",
- "userName": "Brugernavn",
- "lastSeen": "Sidst set",
- "reportRealPath": "",
- "scrobbleEnabled": ""
- }
- },
- "transcoding": {
- "name": "Omkodning |||| Omkodninger",
- "fields": {
- "name": "Navn",
- "targetFormat": "Målformat",
- "defaultBitRate": "Standard bitrate",
- "command": "Kommando"
- }
- },
- "playlist": {
- "name": "Afspilningsliste |||| Afspilningslister",
- "fields": {
- "name": "Navn",
- "duration": "Varighed",
- "ownerName": "Ejer",
- "public": "Offentlig",
- "updatedAt": "Opdateret den",
- "createdAt": "Oprettet den",
- "songCount": "Sange",
- "comment": "Kommentar",
- "sync": "Auto-importér",
- "path": "Importér fra"
- },
- "actions": {
- "selectPlaylist": "Vælg en afspilningsliste:",
- "addNewPlaylist": "Opret \"%{name}\"",
- "export": "Eksporter",
- "makePublic": "",
- "makePrivate": ""
- },
- "message": {
- "duplicate_song": "",
- "song_exist": ""
- }
- },
- "radio": {
- "name": "",
- "fields": {
- "name": "",
- "streamUrl": "",
- "homePageUrl": "",
- "updatedAt": "",
- "createdAt": ""
- },
- "actions": {
- "playNow": ""
- }
- },
- "share": {
- "name": "",
- "fields": {
- "username": "",
- "url": "",
- "description": "",
- "contents": "",
- "expiresAt": "",
- "lastVisitedAt": "",
- "visitCount": "",
- "format": "",
- "maxBitRate": "",
- "updatedAt": "",
- "createdAt": "",
- "downloadable": ""
- }
- }
+ "languageName": "Dansk",
+ "resources": {
+ "song": {
+ "name": "Sang |||| Sange",
+ "fields": {
+ "albumArtist": "Album kunstner",
+ "duration": "Varighed",
+ "trackNumber": "#",
+ "playCount": "Afspilninger",
+ "title": "Titel",
+ "artist": "Kunstner",
+ "album": "Album navn",
+ "path": "Filsti",
+ "genre": "Genre",
+ "compilation": "Opsamling",
+ "year": "År",
+ "size": "Fil størrelse",
+ "updatedAt": "Opdateret den",
+ "bitRate": "Bitrate",
+ "discSubtitle": "Plade undertitel",
+ "starred": "Stjernemarkeret",
+ "comment": "Kommentar",
+ "rating": "Bedømmelse",
+ "quality": "Kvalitet",
+ "bpm": "BPM",
+ "playDate": "Senest afspillet",
+ "channels": "Kanaler",
+ "createdAt": "Tilføjet d.",
+ "grouping": "Gruppering",
+ "mood": "Humør",
+ "participants": "Yderligere deltagere",
+ "tags": "Yderligere tags",
+ "mappedTags": "Mappede tags",
+ "rawTags": "Rå tags",
+ "bitDepth": "Bitdybde",
+ "sampleRate": "Samplingfrekvens",
+ "missing": "Manglende",
+ "libraryName": "Bibliotek"
+ },
+ "actions": {
+ "addToQueue": "Afspil senere",
+ "playNow": "Afspil nu",
+ "addToPlaylist": "Føj til afspilningsliste",
+ "shuffleAll": "Bland alle",
+ "download": "Download",
+ "playNext": "Afspil næste",
+ "info": "Hent info",
+ "showInPlaylist": "Vis i afspilningsliste"
+ }
},
- "ra": {
- "auth": {
- "welcome1": "Tak fordi du installerede Navidrome!",
- "welcome2": "Opret administrator for at begynde",
- "confirmPassword": "Bekræft kodeord",
- "buttonCreateAdmin": "Opret administrator",
- "auth_check_error": "Venligst login for at fortsætte",
- "user_menu": "Profil",
- "username": "Brugernavn",
- "password": "Password",
- "sign_in": "Log ind",
- "sign_in_error": "Dit log ind fejlede, prøv igen",
- "logout": "Log ud"
- },
- "validation": {
- "invalidChars": "Vær venlig kun at benytte bogstaver og tal",
- "passwordDoesNotMatch": "Kodeord er ikke ens",
- "required": "Obligatorisk",
- "minLength": "Skal være mindst %{min} tegn",
- "maxLength": "Skal være max %{max} tegn",
- "minValue": "Skal være mindst %{min}",
- "maxValue": "Skal være max %{max}",
- "number": "Skal være et nummer",
- "email": "Skal være en gyldig e-mail-adresse",
- "oneOf": "Skal være en af: %{options}",
- "regex": "Skal matche et bestemt format (regexp): %{pattern}",
- "unique": "",
- "url": ""
- },
- "action": {
- "add_filter": "Tilføj filter",
- "add": "Tilføj",
- "back": "Tilbage",
- "bulk_actions": "%{smart_count} valgt",
- "cancel": "Annuller",
- "clear_input_value": "Ryd",
- "clone": "Klon",
- "confirm": "Bekræft",
- "create": "Opret",
- "delete": "Slet",
- "edit": "Rediger",
- "export": "Eksporter",
- "list": "Liste",
- "refresh": "Opdater",
- "remove_filter": "Slet filter",
- "remove": "Fjern",
- "save": "Gem",
- "search": "Søg",
- "show": "Vis",
- "sort": "Sortér",
- "undo": "Fortryd",
- "expand": "Udvid",
- "close": "Luk",
- "open_menu": "Åben menu",
- "close_menu": "Luk menu",
- "unselect": "Fravælg",
- "skip": "",
- "bulk_actions_mobile": "",
- "share": "",
- "download": ""
- },
- "boolean": {
- "true": "Ja",
- "false": "Nej"
- },
- "page": {
- "create": "Opret %{name}",
- "dashboard": "Dashboard",
- "edit": "%{name} #%{id}",
- "error": "Noget gik galt",
- "list": "%{name} liste",
- "loading": "Henter",
- "not_found": "Ikke fundet",
- "show": "%{name} #%{id}",
- "empty": "Ingen %{name} endnu",
- "invite": "Vil du tilføje en?"
- },
- "input": {
- "file": {
- "upload_several": "Træk og slip filer for at uploade, eller klik for at vælge filer.",
- "upload_single": "Træk og slip en fil for at uploade, eller klik for at vælge en fil."
- },
- "image": {
- "upload_several": "Træk og slip filer for at uploade, eller klik for at vælge filer.",
- "upload_single": "Træk og slip et billede for at uploade, eller klik for at vælge en fil."
- },
- "references": {
- "all_missing": "Kan ikke finde nogle referencedata.",
- "many_missing": "Mindst en af de tilknyttede referencer synes ikke længere at være tilgængelig.",
- "single_missing": "Tilknyttede referencer synes ikke længere at være tilgængelige."
- },
- "password": {
- "toggle_visible": "Skjul kodeord",
- "toggle_hidden": "Vis kodeord"
- }
- },
- "message": {
- "about": "Om",
- "are_you_sure": "Er du sikker?",
- "bulk_delete_content": "Er du sikker på du vil slette %{name}? |||| Er du sikker på du ville slette %{smart_count} poster?",
- "bulk_delete_title": "Slet %{name} |||| Sletter %{smart_count} %{name} poster",
- "delete_content": "Er du sikker på du ville slette denne post?",
- "delete_title": "Slet %{name} #%{id}",
- "details": "Detaljer",
- "error": "Der opstod en klientfejl, og din forespørgsel kunne ikke udføres.",
- "invalid_form": "Formularen er ikke gyldig. Kontroller for fejl",
- "loading": "Siden indlæses, Vent et øjeblik",
- "no": "Nej",
- "not_found": "Enten har du skrevet en forkert URL eller du har fulgt et invalidt link.",
- "yes": "Ja",
- "unsaved_changes": "Du har lavet ændringer der ikke er gemt. Er du sikker på at du vil ignorere dem?"
- },
- "navigation": {
- "no_results": "Ingen resultater fundet",
- "no_more_results": "Sidenummeret %{page} eksistere ikke. Gå tilbage til forrige side.",
- "page_out_of_boundaries": "Sidenummeret %{page} eksistere ikke",
- "page_out_from_end": "Der findes ikke flere sider",
- "page_out_from_begin": "Der er ingen side før end side 1",
- "page_range_info": "%{offsetBegin}-%{offsetEnd} af %{total}",
- "page_rows_per_page": "Rækker pr. side:",
- "next": "Næste",
- "prev": "Forrige",
- "skip_nav": ""
- },
- "notification": {
- "updated": "Objekt opdateret |||| %{smart_count} objekter opdateret",
- "created": "Objekt oprettet",
- "deleted": "Objekt slettet |||| %{smart_count} objekter slettet",
- "bad_item": "Incorrect element",
- "item_doesnt_exist": "Objektet findes ikke",
- "http_error": "Kommunikationsfejl med serveren",
- "data_provider_error": "dataProvider fejl. Check din console for detaljer.",
- "i18n_error": "Kan ikke indlæse oversættelse af det ønskede sprog",
- "canceled": "Handling blev annulleret",
- "logged_out": "Din session er udløbet, venligst tilslut igen",
- "new_version": ""
- },
- "toggleFieldsMenu": {
- "columnsToDisplay": "",
- "layout": "",
- "grid": "",
- "table": ""
- }
+ "album": {
+ "name": "Album |||| Albums",
+ "fields": {
+ "albumArtist": "Album kunstner",
+ "artist": "Kunstner",
+ "duration": "Varighed",
+ "songCount": "Sange",
+ "playCount": "Afspilninger",
+ "name": "Navn",
+ "genre": "Genre",
+ "compilation": "Opsamling",
+ "year": "År",
+ "updatedAt": "Opdateret d.",
+ "comment": "Kommentar",
+ "rating": "Bedømmelse",
+ "createdAt": "Tilføjet d.",
+ "size": "Størrelse",
+ "originalDate": "Original",
+ "releaseDate": "Udgivet",
+ "releases": "Udgivelse |||| Udgivelser",
+ "released": "Udgivet",
+ "recordLabel": "Plademærke",
+ "catalogNum": "Katalognummer",
+ "releaseType": "Type",
+ "grouping": "Gruppering",
+ "media": "Medier",
+ "mood": "Humør",
+ "date": "Optagelsesdato",
+ "missing": "Manglende",
+ "libraryName": "Bibliotek"
+ },
+ "actions": {
+ "playAll": "Afspil",
+ "playNext": "Afspil næste",
+ "addToQueue": "Afspil senere",
+ "shuffle": "Bland",
+ "addToPlaylist": "Føj til afspilningsliste",
+ "download": "Download",
+ "info": "Hent info",
+ "share": "Del"
+ },
+ "lists": {
+ "all": "Alle",
+ "random": "Tilfældig",
+ "recentlyAdded": "Nyligt tilføjet",
+ "recentlyPlayed": "Nyligt Afspillet",
+ "mostPlayed": "Mest Afspillet",
+ "starred": "Stjernemarkerede",
+ "topRated": "Top bedømmelse"
+ }
},
- "message": {
- "note": "NOTE",
- "transcodingDisabled": "Skift af indstillinger for omkodning gennem web platformen er frakoblet af sikkerhedsgrunde. Genstart serveren med %{config} indstilling tilvalgt.",
- "transcodingEnabled": "Navidrome kører i øjeblikket med %{config}, hvilket gør det muligt at køre system kommandoer fra web platformen. Vi anbefaler at slå det fra af sikkerhedsgrunde og kun slå det til ved indstilling af omkodning.",
- "songsAddedToPlaylist": "Tilføjede 1 sang til afspilningsliste |||| Tilføjede %{smart_count} sange til afspilningsliste",
- "noPlaylistsAvailable": "Ingen tilgængelige",
- "delete_user_title": "Slet bruger '%{name}'",
- "delete_user_content": "Er du sikker på at du vil slette denne bruger og tilhørende data (inklusive afspilningslister og indstillinger)?",
- "notifications_blocked": "",
- "notifications_not_available": "",
- "lastfmLinkSuccess": "",
- "lastfmLinkFailure": "",
- "lastfmUnlinkSuccess": "",
- "lastfmUnlinkFailure": "",
- "openIn": {
- "lastfm": "",
- "musicbrainz": ""
- },
- "lastfmLink": "",
- "listenBrainzLinkSuccess": "",
- "listenBrainzLinkFailure": "",
- "listenBrainzUnlinkSuccess": "",
- "listenBrainzUnlinkFailure": "",
- "downloadOriginalFormat": "",
- "shareOriginalFormat": "",
- "shareDialogTitle": "",
- "shareBatchDialogTitle": "",
- "shareSuccess": "",
- "shareFailure": "",
- "downloadDialogTitle": "",
- "shareCopyToClipboard": ""
+ "artist": {
+ "name": "Kunstner |||| Kunstnere",
+ "fields": {
+ "name": "Navn",
+ "albumCount": "Antal albums",
+ "songCount": "Antal sange",
+ "playCount": "Afspilninger",
+ "rating": "Bedømmelse",
+ "genre": "Genre",
+ "size": "Størrelse",
+ "role": "Rolle",
+ "missing": "Manglende"
+ },
+ "roles": {
+ "albumartist": "Albumkunstner |||| Albumkunstnere",
+ "artist": "Kunstner |||| Kunstnere",
+ "composer": "Komponist |||| Komponister",
+ "conductor": "Dirigent |||| Dirigenter",
+ "lyricist": "Tekstforfatter |||| Tekstforfattere",
+ "arranger": "Arrangør |||| Arrangører",
+ "producer": "Producent |||| Producenter",
+ "director": "Instruktør |||| Instruktører",
+ "engineer": "Tekniker||||Teknikere",
+ "mixer": "Mixer |||| Mixere",
+ "remixer": "Remixer |||| Remixere",
+ "djmixer": "DJ-mixer |||| DJ-mixere",
+ "performer": "Udførende kunstner |||| Udførende kunstnere",
+ "maincredit": "Albumkunstner eller kunstner |||| Albumkunstnere eller kunstnere"
+ },
+ "actions": {
+ "shuffle": "Bland",
+ "radio": "Radio",
+ "topSongs": "Topsange"
+ }
},
- "menu": {
- "library": "Bibliotek",
- "settings": "Indstillinger",
- "version": "Version",
- "theme": "Tema",
- "personal": {
- "name": "Personligt",
- "options": {
- "theme": "Tema",
- "language": "Sprog",
- "defaultView": "Standardopsætning",
- "desktop_notifications": "",
- "lastfmScrobbling": "",
- "listenBrainzScrobbling": "",
- "replaygain": "",
- "preAmp": "",
- "gain": {
- "none": "",
- "album": "",
- "track": ""
- }
- }
- },
- "albumList": "Albums",
- "about": "Om",
- "playlists": "",
- "sharedPlaylists": ""
+ "user": {
+ "name": "Bruger |||| Brugere",
+ "fields": {
+ "userName": "Brugernavn",
+ "isAdmin": "Er administrator",
+ "lastLoginAt": "Seneste login",
+ "updatedAt": "Opdateret d.",
+ "name": "Navn",
+ "password": "Kodeord",
+ "createdAt": "Oprettet d.",
+ "changePassword": "Skifte kodeord?",
+ "currentPassword": "Nuværende kodeord",
+ "newPassword": "Nyt kodeord",
+ "token": "Token",
+ "lastAccessAt": "Senest tilgået",
+ "libraries": "Biblioteker"
+ },
+ "helperTexts": {
+ "name": "Ændringer i dit navn vises først ved næste login",
+ "libraries": "Vælg specifikke biblioteker til denne bruger, eller lad det stå tomt for at bruge standardbiblioteker"
+ },
+ "notifications": {
+ "created": "Bruger oprettet",
+ "updated": "Bruger opdateret",
+ "deleted": "Bruger slettet"
+ },
+ "message": {
+ "listenBrainzToken": "Skriv dit ListenBrainz token",
+ "clickHereForToken": "Tryk her for at få dit token",
+ "selectAllLibraries": "Vælg alle biblioteker",
+ "adminAutoLibraries": "Administratorbrugere har automatisk adgang til alle biblioteker"
+ },
+ "validation": {
+ "librariesRequired": "Der skal være valgt mindst ét bibliotek til ikke-administrative brugere"
+ }
},
"player": {
- "playListsText": "Afspilnings kø",
- "openText": "Åben",
- "closeText": "Luk",
- "notContentText": "Ingen musik",
- "clickToPlayText": "Tryk for at afspille",
- "clickToPauseText": "Tryk for at pause",
- "nextTrackText": "Næste nummer",
- "previousTrackText": "Forrige nummer",
- "reloadText": "Genindlæs",
- "volumeText": "Lydstyrke",
- "toggleLyricText": "Skift sangtekst",
- "toggleMiniModeText": "Minimer",
- "destroyText": "Fjern",
- "downloadText": "Hent",
- "removeAudioListsText": "Slet afspillingsliste",
- "clickToDeleteText": "Tryk for at slette %{name}",
- "emptyLyricText": "Ingen sangtekst",
- "playModeText": {
- "order": "I rækkefølge",
- "orderLoop": "Gentag",
- "singleLoop": "Gentag enkelt",
- "shufflePlay": "Bland"
- }
+ "name": "Afspiller |||| Afspillere",
+ "fields": {
+ "name": "Navn",
+ "transcodingId": "Transkodning",
+ "maxBitRate": "Maks. bitrate",
+ "client": "Klient",
+ "userName": "Brugernavn",
+ "lastSeen": "Sidst set",
+ "reportRealPath": "Vis den virkelige sti",
+ "scrobbleEnabled": "Send scrobbles til eksterne tjenester"
+ }
},
- "about": {
- "links": {
- "homepage": "Hjme",
- "source": "Kildekode",
- "featureRequests": "Ønskede funktioner"
- }
+ "transcoding": {
+ "name": "Transkodning |||| Transkodninger",
+ "fields": {
+ "name": "Navn",
+ "targetFormat": "Målformat",
+ "defaultBitRate": "Standard bitrate",
+ "command": "Kommando"
+ }
},
- "activity": {
- "title": "Aktivitet",
- "totalScanned": "Antal sange fundet",
- "quickScan": "Hurtig søgning",
- "fullScan": "Fuld søgning\n",
- "serverUptime": "Server uptime",
- "serverDown": "OFFLINE"
+ "playlist": {
+ "name": "Afspilningsliste |||| Afspilningslister",
+ "fields": {
+ "name": "Navn",
+ "duration": "Varighed",
+ "ownerName": "Ejer",
+ "public": "Offentlig",
+ "updatedAt": "Opdateret d.",
+ "createdAt": "Oprettet d.",
+ "songCount": "Sange",
+ "comment": "Kommentar",
+ "sync": "Auto-importér",
+ "path": "Importér fra"
+ },
+ "actions": {
+ "selectPlaylist": "Vælg en afspilningsliste:",
+ "addNewPlaylist": "Opret \"%{name}\"",
+ "export": "Eksportér",
+ "makePublic": "Offentliggør",
+ "makePrivate": "Gør privat",
+ "saveQueue": "Gem kø på afspilningsliste",
+ "searchOrCreate": "Søg i afspilningslister eller skriv for at oprette nye...",
+ "pressEnterToCreate": "Tryk Enter for at oprette en ny afspilningsliste",
+ "removeFromSelection": "Fjern fra valg"
+ },
+ "message": {
+ "duplicate_song": "Tilføj dubletter af sange",
+ "song_exist": "Der føjes dubletter til playlisten",
+ "noPlaylistsFound": "Ingen playlister fundet",
+ "noPlaylists": "Ingen tilgængelige playlister"
+ }
},
- "help": {
- "title": "",
- "hotkeys": {
- "show_help": "",
- "toggle_menu": "",
- "toggle_play": "",
- "prev_song": "",
- "next_song": "",
- "vol_up": "",
- "vol_down": "",
- "toggle_love": "",
- "current_song": ""
- }
+ "radio": {
+ "name": "Radio |||| Radioer",
+ "fields": {
+ "name": "Navn",
+ "streamUrl": "Stream-URL",
+ "homePageUrl": "Hjemmeside-URL",
+ "updatedAt": "Opdateret d.",
+ "createdAt": "Oprettet d."
+ },
+ "actions": {
+ "playNow": "Afspil nu"
+ }
+ },
+ "share": {
+ "name": "Del |||| Delinger",
+ "fields": {
+ "username": "Delt af",
+ "url": "URL",
+ "description": "Beskrivelse",
+ "contents": "Indhold",
+ "expiresAt": "Udløber",
+ "lastVisitedAt": "Senest besøgt",
+ "visitCount": "Besøg",
+ "format": "Format",
+ "maxBitRate": "Maks. bitrate",
+ "updatedAt": "Opdateret d.",
+ "createdAt": "Oprettet d.",
+ "downloadable": "Tillad downloads?"
+ }
+ },
+ "missing": {
+ "name": "Manglende fil |||| Manglende filer",
+ "fields": {
+ "path": "Sti",
+ "size": "Størrelse",
+ "updatedAt": "Forsvandt d.",
+ "libraryName": "Bibliotek"
+ },
+ "actions": {
+ "remove": "Fjern",
+ "remove_all": "Fjern alle"
+ },
+ "notifications": {
+ "removed": "Manglende fil(er) fjernet"
+ },
+ "empty": "Ingen manglende filer"
+ },
+ "library": {
+ "name": "Bibliotek |||| Biblioteker",
+ "fields": {
+ "name": "Navn",
+ "path": "Sti",
+ "remotePath": "Fjernsti",
+ "lastScanAt": "Sidste scanning",
+ "songCount": "Sange",
+ "albumCount": "Albummer",
+ "artistCount": "Kunstnere",
+ "totalSongs": "Sange",
+ "totalAlbums": "Albummer",
+ "totalArtists": "Kunstnere",
+ "totalFolders": "Mapper",
+ "totalFiles": "Filer",
+ "totalMissingFiles": "Manglende filer",
+ "totalSize": "Samlet størrelse",
+ "totalDuration": "Varighed",
+ "defaultNewUsers": "Standard for nye brugere",
+ "createdAt": "Oprettet d.",
+ "updatedAt": "Opdateret d."
+ },
+ "sections": {
+ "basic": "Grundlæggende oplysninger",
+ "statistics": "Statistik"
+ },
+ "actions": {
+ "scan": "Scanningsbibliotek",
+ "manageUsers": "Administrer brugeradgang",
+ "viewDetails": "Se detaljer"
+ },
+ "notifications": {
+ "created": "Bibliotek oprettet",
+ "updated": "Biblioteket er blevet opdateret",
+ "deleted": "Biblioteket er blevet slettet",
+ "scanStarted": "Biblioteksscanning startet",
+ "scanCompleted": "Biblioteksscanning fuldført"
+ },
+ "validation": {
+ "nameRequired": "Biblioteksnavn er påkrævet",
+ "pathRequired": "Bibliotekssti er påkrævet",
+ "pathNotDirectory": "Biblioteksstien skal være en mappe",
+ "pathNotFound": "Biblioteksstien blev ikke fundet",
+ "pathNotAccessible": "Biblioteksstien er ikke tilgængelig",
+ "pathInvalid": "Ugyldig bibliotekssti"
+ },
+ "messages": {
+ "deleteConfirm": "Er du sikker på, at du vil slette dette bibliotek? Dét vil fjerne alle tilknyttede data og brugeradgange",
+ "scanInProgress": "Scanning i gang...",
+ "noLibrariesAssigned": "Ingen biblioteker tildelt denne bruger"
+ }
}
+ },
+ "ra": {
+ "auth": {
+ "welcome1": "Tak fordi du installerede Navidrome!",
+ "welcome2": "Først, opret en administrator",
+ "confirmPassword": "Bekræft kodeord",
+ "buttonCreateAdmin": "Opret administrator",
+ "auth_check_error": "Venligst login for at fortsætte",
+ "user_menu": "Profil",
+ "username": "Brugernavn",
+ "password": "Kodeord",
+ "sign_in": "Log ind",
+ "sign_in_error": "Dit log ind slog fejl, prøv igen",
+ "logout": "Log ud",
+ "insightsCollectionNote": "Navidrome indsamler anonyme brugsdata for at forbedre projektet. Klik [her] for at få mere at vide og fravælge, hvis du ønsker det."
+ },
+ "validation": {
+ "invalidChars": "Venligst, benyt kun bogstaver og tal",
+ "passwordDoesNotMatch": "Kodeord er ikke ens",
+ "required": "Nødvendig",
+ "minLength": "Skal være mindst %{min} tegn",
+ "maxLength": "Skal være op til %{max} tegn",
+ "minValue": "Skal være mindst %{min}",
+ "maxValue": "Skal være op til %{max}",
+ "number": "Skal være et tal",
+ "email": "Skal være en gyldig e-mail-adresse",
+ "oneOf": "Skal være én af: %{options}",
+ "regex": "Skal matche et specifikt format (regexp): %{pattern}",
+ "unique": "Skal være unik",
+ "url": "Skal være en gyldig URL"
+ },
+ "action": {
+ "add_filter": "Tilføj filter",
+ "add": "Tilføj",
+ "back": "Tilbage",
+ "bulk_actions": "1 emne valgt |||| %{smart_count} emner valgt",
+ "cancel": "Annuller",
+ "clear_input_value": "Ryd",
+ "clone": "Klon",
+ "confirm": "Bekræft",
+ "create": "Opret",
+ "delete": "Slet",
+ "edit": "Rediger",
+ "export": "Eksportér",
+ "list": "Liste",
+ "refresh": "Opdater",
+ "remove_filter": "Slet filter",
+ "remove": "Fjern",
+ "save": "Gem",
+ "search": "Søg",
+ "show": "Vis",
+ "sort": "Sortér",
+ "undo": "Fortryd",
+ "expand": "Udvid",
+ "close": "Luk",
+ "open_menu": "Åbn menu",
+ "close_menu": "Luk menu",
+ "unselect": "Fravælg",
+ "skip": "Spring over",
+ "bulk_actions_mobile": "1 |||| %{smart_count}",
+ "share": "Del",
+ "download": "Download"
+ },
+ "boolean": {
+ "true": "Ja",
+ "false": "Nej"
+ },
+ "page": {
+ "create": "Opret %{name}",
+ "dashboard": "Instrumentbræt",
+ "edit": "%{name} #%{id}",
+ "error": "Noget gik galt",
+ "list": "%{name} liste",
+ "loading": "Henter",
+ "not_found": "Ikke fundet",
+ "show": "%{name} #%{id}",
+ "empty": "Ingen %{name} endnu.",
+ "invite": "Vil du tilføje en?"
+ },
+ "input": {
+ "file": {
+ "upload_several": "Træk nogle filer herind for at uploade, eller klik for at vælge en.",
+ "upload_single": "Træk en fil herind for at uploade, eller klik for at vælge den."
+ },
+ "image": {
+ "upload_several": "Træk billedfiler herind for at uploade, eller klik for at vælge en.",
+ "upload_single": "Træk en billedfil herind for at uploade, eller klik for at vælge den."
+ },
+ "references": {
+ "all_missing": "Kan ikke finde nogen referencedata.",
+ "many_missing": "Mindst en af de tilknyttede referencer synes ikke længere at være tilgængelig.",
+ "single_missing": "Tilknyttede referencer synes ikke længere at være tilgængelige."
+ },
+ "password": {
+ "toggle_visible": "Skjul kodeord",
+ "toggle_hidden": "Vis kodeord"
+ }
+ },
+ "message": {
+ "about": "Om",
+ "are_you_sure": "Er du sikker?",
+ "bulk_delete_content": "Er du sikker på, at du vil slette %{name}? |||| Er du sikker på, at du vil slette disse %{smart_count} poster?",
+ "bulk_delete_title": "Slet %{name} |||| Sletter %{smart_count} %{name} poster",
+ "delete_content": "Er du sikker på, at du vil slette denne post?",
+ "delete_title": "Slet %{name} #%{id}",
+ "details": "Detaljer",
+ "error": "Der opstod en klientfejl, og din forespørgsel kunne ikke udføres.",
+ "invalid_form": "Formularen er ikke gyldig. Tjek for fejl",
+ "loading": "Siden indlæses, vent et øjeblik",
+ "no": "Nej",
+ "not_found": "Enten har du skrevet en forkert URL eller du har fulgt et ugyldigt link.",
+ "yes": "Ja",
+ "unsaved_changes": "Du har lavet ændringer der ikke er gemt. Er du sikker på at du vil ignorere dem?"
+ },
+ "navigation": {
+ "no_results": "Ingen resultater fundet",
+ "no_more_results": "Sidenummeret %{page} eksisterer ikke. Gå tilbage til forrige side.",
+ "page_out_of_boundaries": "Sidenummeret %{page} ligger uden for grænserne",
+ "page_out_from_end": "Dette er sidste side",
+ "page_out_from_begin": "Dette er side 1",
+ "page_range_info": "%{offsetBegin}-%{offsetEnd} af %{total}",
+ "page_rows_per_page": "Rækker pr. side:",
+ "next": "Næste",
+ "prev": "Forrige",
+ "skip_nav": "Hop til indhold"
+ },
+ "notification": {
+ "updated": "Element opdateret |||| %{smart_count} elementer opdateret",
+ "created": "Element oprettet",
+ "deleted": "Element slettet |||| %{smart_count} elementer slettet",
+ "bad_item": "Forkert element",
+ "item_doesnt_exist": "Elementet findes ikke",
+ "http_error": "Kommunikationsfejl med serveren",
+ "data_provider_error": "dataProvider fejl. Tjek konsollen for detaljer.",
+ "i18n_error": "Kan ikke indlæse oversættelsen af det ønskede sprog",
+ "canceled": "Handling blev annulleret",
+ "logged_out": "Din session er udløbet, venligst tilslut igen",
+ "new_version": "Ny version tilgængelig! – genopfrisk venligst vinduet"
+ },
+ "toggleFieldsMenu": {
+ "columnsToDisplay": "Antal synlige kolonner",
+ "layout": "Layout",
+ "grid": "Gitter",
+ "table": "Tabel"
+ }
+ },
+ "message": {
+ "note": "NOTE",
+ "transcodingDisabled": "Ændring af indstillinger til transkodning via webgrænsefladen er deaktiveret af sikkerhedshensyn.\nFor at ændre eller tilføje indstillinger skal du genstarte serveren med %{config} konfigurations option.",
+ "transcodingEnabled": "Navidrome kører i øjeblikket med %{config}. Dét gør det muligt at køre systemkommandoer fra transkodningsindstillingerne, via webgrænsefladen.\nVi anbefaler at deaktivere dette af sikkerhedshensyn og kun have det aktiveret, når du konfigurerer indstillinger til transkodning.",
+ "songsAddedToPlaylist": "Føjede 1 sang til afspilningsliste |||| Føjede %{smart_count} sange til afspilningsliste",
+ "noPlaylistsAvailable": "Ingen tilgængelige",
+ "delete_user_title": "Slet bruger '%{name}'",
+ "delete_user_content": "Er du sikker på at du vil slette denne bruger og tilhørende data (inklusive afspilningslister og valgte indstillinger)?",
+ "notifications_blocked": "Du blokerer for notifikationer fra dette site i dine browserindstillinger",
+ "notifications_not_available": "Denne browser understøtter ikke skrivebordsnotifikationer, eller: Du tilgår ikke Navidrome over https",
+ "lastfmLinkSuccess": "Du er koblet til Last.fm, og scrobbling er slået til",
+ "lastfmLinkFailure": "Du kan ikke kobles til Last.fm",
+ "lastfmUnlinkSuccess": "Last.fm frakoblet, og scrobbling deaktiveret",
+ "lastfmUnlinkFailure": "Last.fm kunne ikke frakobles",
+ "openIn": {
+ "lastfm": "Åbn i Last.fm",
+ "musicbrainz": "Åbn i MusicBrainz"
+ },
+ "lastfmLink": "Læs mere...",
+ "listenBrainzLinkSuccess": "Du er koblet til ListenBrainz og scrobbling er aktiveret som bruger: %{user}",
+ "listenBrainzLinkFailure": "Du kunne ikke kobles til ListenBrainz: %{error}",
+ "listenBrainzUnlinkSuccess": "ListenBrainz er frakoblet, og scrobbling deaktiveret",
+ "listenBrainzUnlinkFailure": "ListenBrainz kunne ikke frakobles",
+ "downloadOriginalFormat": "Download i originalformat",
+ "shareOriginalFormat": "Del i originalformat",
+ "shareDialogTitle": "Del %{resource} '%{name}'",
+ "shareBatchDialogTitle": "Del 1 %{resource} |||| Del %{smart_count} %{resource}",
+ "shareSuccess": "URL kopieret til udklipsholder: %{url}",
+ "shareFailure": "Fejl ved kopiering af URL %{url} til udklipsholder",
+ "downloadDialogTitle": "Download %{resource} '%{name}' (%{size})",
+ "shareCopyToClipboard": "Kopiér til udklipsholder: Ctrl+C, Enter",
+ "remove_missing_title": "Fjern manglende filer",
+ "remove_missing_content": "Er du sikker på, at du vil fjerne de valgte manglende filer fra databasen? Dét vil permanent fjerne alle referencer til dem, inklusive deres afspilningstællere og vurderinger.",
+ "remove_all_missing_title": "Fjern alle manglende filer",
+ "remove_all_missing_content": "Er du sikker på, at du vil fjerne alle manglende filer fra databasen? Dét vil permanent fjerne alle referencer til dem, inklusive deres afspilningstællere og vurderinger.",
+ "noSimilarSongsFound": "Ingen lignende sange fundet",
+ "noTopSongsFound": "Ingen topsange fundet"
+ },
+ "menu": {
+ "library": "Bibliotek",
+ "settings": "Indstillinger",
+ "version": "Version",
+ "theme": "Tema",
+ "personal": {
+ "name": "Personligt",
+ "options": {
+ "theme": "Tema",
+ "language": "Sprog",
+ "defaultView": "Standardopsætning",
+ "desktop_notifications": "Skrivebordsnotifikationer",
+ "lastfmScrobbling": "Scrobble til Last.fm",
+ "listenBrainzScrobbling": "Scrobble til ListenBrainz",
+ "replaygain": "ReplayGain-tilstand",
+ "preAmp": "ReplayGain PreAmp (dB)",
+ "gain": {
+ "none": "Slået fra",
+ "album": "Brug Album Gain",
+ "track": "Brug Gain for spor"
+ },
+ "lastfmNotConfigured": "Last.fm API-nøglen er ikke konfigureret"
+ }
+ },
+ "albumList": "Albums",
+ "about": "Om",
+ "playlists": "Afspilningslister",
+ "sharedPlaylists": "Delte afspilningslister",
+ "librarySelector": {
+ "allLibraries": "Alle biblioteker (%{count})",
+ "multipleLibraries": "%{selected} af %{total} biblioteker",
+ "selectLibraries": "Vælg biblioteker",
+ "none": "Ingen"
+ }
+ },
+ "player": {
+ "playListsText": "Afspilningskø",
+ "openText": "Åbn",
+ "closeText": "Luk",
+ "notContentText": "Ingen musik",
+ "clickToPlayText": "Tryk for at afspille",
+ "clickToPauseText": "Tryk for at pause",
+ "nextTrackText": "Næste nummer",
+ "previousTrackText": "Forrige nummer",
+ "reloadText": "Genindlæs",
+ "volumeText": "Lydstyrke",
+ "toggleLyricText": "Skift sangtekst til/fra",
+ "toggleMiniModeText": "Minimer",
+ "destroyText": "Fjern",
+ "downloadText": "Hent",
+ "removeAudioListsText": "Slet afspilningslister",
+ "clickToDeleteText": "Tryk for at slette %{name}",
+ "emptyLyricText": "Ingen sangtekst",
+ "playModeText": {
+ "order": "I rækkefølge",
+ "orderLoop": "Gentag",
+ "singleLoop": "Gentag enkelt",
+ "shufflePlay": "Bland"
+ }
+ },
+ "about": {
+ "links": {
+ "homepage": "Hjemmeside",
+ "source": "Kildekode",
+ "featureRequests": "Funktionsønsker",
+ "lastInsightsCollection": "Seneste indsamling af indsigter",
+ "insights": {
+ "disabled": "Slået fra",
+ "waiting": "Venter"
+ }
+ },
+ "tabs": {
+ "about": "Om",
+ "config": "Konfiguration"
+ },
+ "config": {
+ "configName": "Navn på konfiguration",
+ "environmentVariable": "Miljøvariabel",
+ "currentValue": "Nuværende værdi",
+ "configurationFile": "Konfigurationsfil",
+ "exportToml": "Eksportér konfigurationen (TOML)",
+ "exportSuccess": "Konfigurationen eksporteret til udklipsholder i TOML-format",
+ "exportFailed": "Kunne ikke kopiere konfigurationen",
+ "devFlagsHeader": "Udviklingsflagget (med forbehold for ændring/fjernelse)",
+ "devFlagsComment": "Disse er eksperimental-indstillinger og kan blive fjernet i fremtidige udgaver"
+ }
+ },
+ "activity": {
+ "title": "Aktivitet",
+ "totalScanned": "Antal mapper gennemsøgt",
+ "quickScan": "Hurtig søgning",
+ "fullScan": "Fuld søgning",
+ "serverUptime": "Server oppetid",
+ "serverDown": "OFFLINE",
+ "scanType": "Type",
+ "status": "Scanningsfejl",
+ "elapsedTime": "Medgået tid"
+ },
+ "help": {
+ "title": "Navidrome genvejstaster",
+ "hotkeys": {
+ "show_help": "Vis denne hjælp",
+ "toggle_menu": "Skift menu sidepanel",
+ "toggle_play": "Play / Pause",
+ "prev_song": "Forrige sang",
+ "next_song": "Næste sang",
+ "vol_up": "Volumen op",
+ "vol_down": "Volumen ned",
+ "toggle_love": "Føj dette nummer til dine favoritter",
+ "current_song": "Gå til den aktuelle sang"
+ }
+ },
+ "nowPlaying": {
+ "title": "Afspilles nu",
+ "empty": "Intet afspilles nu",
+ "minutesAgo": "for %{smart_count} minut siden |||| for %{smart_count} minutter siden"
+ }
}
\ No newline at end of file
diff --git a/resources/i18n/de.json b/resources/i18n/de.json
index 6ffb31165..c9c7fa7f5 100644
--- a/resources/i18n/de.json
+++ b/resources/i18n/de.json
@@ -2,7 +2,7 @@
"languageName": "Deutsch",
"resources": {
"song": {
- "name": "Song |||| Songs",
+ "name": "Titel |||| Titel",
"fields": {
"albumArtist": "Albuminterpret",
"duration": "Dauer",
@@ -32,16 +32,21 @@
"participants": "Weitere Beteiligte",
"tags": "Weitere Tags",
"mappedTags": "Gemappte Tags",
- "rawTags": "Tag Rohdaten"
+ "rawTags": "Tag Rohdaten",
+ "bitDepth": "Bittiefe",
+ "sampleRate": "Samplerate",
+ "missing": "Fehlend",
+ "libraryName": "Bibliothek"
},
"actions": {
"addToQueue": "Später abspielen",
"playNow": "Jetzt abspielen",
- "addToPlaylist": "Zur Playlist hinzufügen",
+ "addToPlaylist": "Zu einer Wiedergabeliste hinzufügen",
"shuffleAll": "Zufallswiedergabe",
"download": "Herunterladen",
"playNext": "Als nächstes abspielen",
- "info": "Mehr Informationen"
+ "info": "Mehr Informationen",
+ "showInPlaylist": "In Wiedergabeliste anzeigen"
}
},
"album": {
@@ -70,14 +75,17 @@
"releaseType": "Typ",
"grouping": "Gruppierung",
"media": "Medium",
- "mood": "Stimmung"
+ "mood": "Stimmung",
+ "date": "Aufnahmedatum",
+ "missing": "Fehlend",
+ "libraryName": "Bibliothek"
},
"actions": {
"playAll": "Abspielen",
"playNext": "Als nächstes abspielen",
"addToQueue": "Später abspielen",
"shuffle": "Zufallswiedergabe",
- "addToPlaylist": "Zur Playlist hinzufügen",
+ "addToPlaylist": "Zu einer Wiedergabeliste hinzufügen",
"download": "Herunterladen",
"info": "Mehr Informationen",
"share": "Freigabe erstellen"
@@ -102,7 +110,8 @@
"rating": "Bewertung",
"genre": "Genre",
"size": "Größe",
- "role": "Rolle"
+ "role": "Rolle",
+ "missing": "Fehlend"
},
"roles": {
"albumartist": "Albuminterpret |||| Albuminterpreten",
@@ -117,7 +126,13 @@
"mixer": "Mixer |||| Mixer",
"remixer": "Remixer |||| Remixer",
"djmixer": "DJ Mixer |||| DJ Mixer",
- "performer": "ausübender Künstler |||| ausübende Künstler"
+ "performer": "ausübender Künstler |||| ausübende Künstler",
+ "maincredit": "Albuminterpret oder Interpret |||| Albuminterpreten oder Interpreten"
+ },
+ "actions": {
+ "shuffle": "Zufallswiedergabe",
+ "radio": "Radio",
+ "topSongs": "Beliebteste Titel"
}
},
"user": {
@@ -134,10 +149,12 @@
"currentPassword": "Aktuelles Passwort",
"newPassword": "Neues Passwort",
"token": "Token",
- "lastAccessAt": "Letzter Zugriff am"
+ "lastAccessAt": "Letzter Zugriff am",
+ "libraries": "Bibliotheken"
},
"helperTexts": {
- "name": "Die Änderung wird erst nach dem nächsten Login gültig"
+ "name": "Die Änderung wird erst nach dem nächsten Login gültig",
+ "libraries": "Wähle spezifische Bibliotheken für diesen Benutzer, oder leer lassen für Standard Bibliotheken"
},
"notifications": {
"created": "Benutzer erstellt",
@@ -146,7 +163,12 @@
},
"message": {
"listenBrainzToken": "Gib deinen ListenBrainz Benutzer Token ein",
- "clickHereForToken": "Hier klicken um deinen Token abzurufen"
+ "clickHereForToken": "Hier klicken um deinen Token abzurufen",
+ "selectAllLibraries": "Wähle alle Bibliotheken",
+ "adminAutoLibraries": "Administrator-Benutzer haben automatisch Zugriff auf alle Bibliotheken"
+ },
+ "validation": {
+ "librariesRequired": "Mindestens eine Bibliothek muss für nicht-administrator Benutzer ausgewählt sein"
}
},
"player": {
@@ -172,7 +194,7 @@
}
},
"playlist": {
- "name": "Playlist |||| Playlists",
+ "name": "Wiedergabeliste |||| Wiedergabelisten",
"fields": {
"name": "Name",
"duration": "Dauer",
@@ -186,15 +208,21 @@
"path": "Importieren aus"
},
"actions": {
- "selectPlaylist": "Titel zur Playlist hinzufügen",
+ "selectPlaylist": "Wiedergabeliste auswählen:",
"addNewPlaylist": "\"%{name}\" erstellen",
"export": "Exportieren",
"makePublic": "Öffentlich machen",
- "makePrivate": "Privat stellen"
+ "makePrivate": "Privat stellen",
+ "saveQueue": "Warteschlange in Wiedergabeliste speichern",
+ "searchOrCreate": "Wiedergabeliste suchen oder neue erstellen...",
+ "pressEnterToCreate": "Enter drücken um neue Wiedergabeliste zu erstellen",
+ "removeFromSelection": "Von Auswahl entfernen"
},
"message": {
"duplicate_song": "Duplikate hinzufügen",
- "song_exist": "Manche Titel sind bereits in der Playlist. Möchtest du sie trotzdem hinzufügen oder überspringen?"
+ "song_exist": "Manche Titel sind bereits in der Playlist. Möchtest du sie trotzdem hinzufügen oder überspringen?",
+ "noPlaylistsFound": "Keine Wiedergabeliste gefunden",
+ "noPlaylists": "Keine Wiedergabelisten vorhanden"
}
},
"radio": {
@@ -232,13 +260,68 @@
"fields": {
"path": "Pfad",
"size": "Größe",
- "updatedAt": "Fehlt seit"
+ "updatedAt": "Fehlt seit",
+ "libraryName": "Bibliothek"
},
"actions": {
- "remove": "Entfernen"
+ "remove": "Entfernen",
+ "remove_all": "alle entfernen"
},
"notifications": {
"removed": "Fehlende Datei(en) entfernt"
+ },
+ "empty": "keine fehlenden Dateien"
+ },
+ "library": {
+ "name": "Bibliothek |||| Bibliotheken",
+ "fields": {
+ "name": "Name",
+ "path": "Pfad",
+ "remotePath": "Remote Pfad",
+ "lastScanAt": "Letzter Scan",
+ "songCount": "Lieder",
+ "albumCount": "Alben",
+ "artistCount": "Interpreten",
+ "totalSongs": "Lieder",
+ "totalAlbums": "Alben",
+ "totalArtists": "Interpreten",
+ "totalFolders": "Ordner",
+ "totalFiles": "Dateien",
+ "totalMissingFiles": "Fehlende Dateien",
+ "totalSize": "Größe",
+ "totalDuration": "Dauer",
+ "defaultNewUsers": "Standard für neue Benutzer",
+ "createdAt": "Erstellt",
+ "updatedAt": "Geändert"
+ },
+ "sections": {
+ "basic": "Basis Informationen",
+ "statistics": "Statistik"
+ },
+ "actions": {
+ "scan": "Bibliothek scannen",
+ "manageUsers": "Zugriff verwalten",
+ "viewDetails": "Details ansehen"
+ },
+ "notifications": {
+ "created": "Bibliothek erfolgreich erstellt",
+ "updated": "Bibliothek erfolgreich geändert",
+ "deleted": "Bibliothek erfolgreich gelöscht",
+ "scanStarted": "Bibliothek Scan gestartet",
+ "scanCompleted": "Bibliothek Scan vollständig"
+ },
+ "validation": {
+ "nameRequired": "Bibliotheksname ist Pflichtfeld",
+ "pathRequired": "Bibliothekspfad ist Pflichtfeld",
+ "pathNotDirectory": "Bibliothekspfad muss ein Ordner sein",
+ "pathNotFound": "Bibliothekspfad nicht gefunden",
+ "pathNotAccessible": "Bibliothekspfad nicht zugänglich",
+ "pathInvalid": "Bibliothekspfad ungültig"
+ },
+ "messages": {
+ "deleteConfirm": "Möchtest du diese Bibliothek wirklich löschen? Zugriffsrechte und Daten werden entfernt. ",
+ "scanInProgress": "Bibliothek Scan läuft...",
+ "noLibrariesAssigned": "Keine Bibliotheken zugeordnet"
}
}
},
@@ -391,10 +474,10 @@
"note": "HINWEIS",
"transcodingDisabled": "Die Änderung der Transcodierungskonfiguration über die Web-UI ist aus Sicherheitsgründen deaktiviert. Wenn du die Transcodierungsoptionen ändern (bearbeiten oder hinzufügen) möchtest, starte den Server mit der Konfigurationsoption %{config} neu.",
"transcodingEnabled": "Navidrome läuft derzeit mit %{config}, wodurch es möglich ist, Systembefehle aus den Transkodierungseinstellungen über die Webschnittstelle auszuführen. Wir empfehlen, es aus Sicherheitsgründen zu deaktivieren und nur bei der Konfiguration von Transkodierungsoptionen zu aktivieren.",
- "songsAddedToPlaylist": "Einen Titel zur Playlist hinzugefügt |||| %{smart_count} Titel zur Playlist hinzugefügt",
- "noPlaylistsAvailable": "Keine Playlist verfügbar",
+ "songsAddedToPlaylist": "Einen Titel zur Wiedergabeliste hinzugefügt |||| %{smart_count} Titel zur Wiedergabeliste hinzugefügt",
+ "noPlaylistsAvailable": "Keine Wiedergabeliste verfügbar",
"delete_user_title": "Benutzer '%{name}' löschen",
- "delete_user_content": "Möchtest du diesen Benutzer und alle seine Daten (einschließlich Playlisten und Einstellungen) wirklich löschen?",
+ "delete_user_content": "Möchtest du diesen Benutzer und alle seine Daten (einschließlich Wiedergabelisten und Einstellungen) wirklich löschen?",
"notifications_blocked": "Sie haben Benachrichtigungen für diese Seite in den Einstellungen Ihres Browsers blockiert",
"notifications_not_available": "Dieser Browser unterstützt keine Desktop-Benachrichtigungen",
"lastfmLinkSuccess": "Last.fm Verbindung hergestellt und scrobbling aktiviert",
@@ -419,7 +502,11 @@
"downloadDialogTitle": "Download %{resource} '%{name}' (%{size})",
"shareCopyToClipboard": "In Zwischenablage kopieren: Ctrl+C, Enter",
"remove_missing_title": "Fehlende Dateien entfernen",
- "remove_missing_content": "Möchtest du die ausgewählten Fehlenden Dateien wirklich aus der Datenbank entfernen? Alle Referenzen zu den Dateien wie Anzahl Wiedergaben und Bewertungen werden permanent gelöscht."
+ "remove_missing_content": "Möchtest du die ausgewählten Fehlenden Dateien wirklich aus der Datenbank entfernen? Alle Referenzen zu den Dateien wie Anzahl Wiedergaben und Bewertungen werden permanent gelöscht.",
+ "remove_all_missing_title": "Alle fehlenden Dateien entfernen",
+ "remove_all_missing_content": "Möchtest du wirklich alle Fehlenden Dateien aus der Datenbank entfernen? Alle Referenzen zu den Dateien wie Anzahl Wiedergaben und Bewertungen werden permanent gelöscht.",
+ "noSimilarSongsFound": "Keine ähnlichen Titel gefunden",
+ "noTopSongsFound": "Keine beliebten Titel gefunden"
},
"menu": {
"library": "Bibliothek",
@@ -447,11 +534,17 @@
},
"albumList": "Alben",
"about": "Über",
- "playlists": "Playlisten",
- "sharedPlaylists": "Geteilte Playlisten"
+ "playlists": "Wiedergabelisten",
+ "sharedPlaylists": "Geteilte Wiedergabelisten",
+ "librarySelector": {
+ "allLibraries": "Alle Bibliotheken (%{count})",
+ "multipleLibraries": "%{selected} von %{total} Bibliotheken",
+ "selectLibraries": "Bibliotheken auswählen",
+ "none": "Keine"
+ }
},
"player": {
- "playListsText": "Wiedergabeliste abspielen",
+ "playListsText": "Warteschlange abspielen",
"openText": "Öffnen",
"closeText": "Schließen",
"notContentText": "Keine Musik",
@@ -485,6 +578,21 @@
"disabled": "Deaktiviert",
"waiting": "Warten"
}
+ },
+ "tabs": {
+ "about": "Über",
+ "config": "Konfiguration"
+ },
+ "config": {
+ "configName": "Einstellung",
+ "environmentVariable": "Umbegungsvariable",
+ "currentValue": "Wert",
+ "configurationFile": "Konfigurationsdatei",
+ "exportToml": "Konfiguration exportieren (TOML)",
+ "exportSuccess": "Konfiguration im TOML Format in die Zwischenablage kopiert",
+ "exportFailed": "Fehler beim Kopieren der Konfiguration",
+ "devFlagsHeader": "Entwicklungseinstellungen (können sich ändern)",
+ "devFlagsComment": "Experimentelle Einstellungen, die eventuell in Zukunft entfernt oder geändert werden"
}
},
"activity": {
@@ -493,7 +601,10 @@
"quickScan": "Schneller Scan",
"fullScan": "Kompletter Scan",
"serverUptime": "Server-Betriebszeit",
- "serverDown": "OFFLINE"
+ "serverDown": "OFFLINE",
+ "scanType": "Typ",
+ "status": "Scan Fehler",
+ "elapsedTime": "Laufzeit"
},
"help": {
"title": "Navidrome Hotkeys",
@@ -508,5 +619,10 @@
"toggle_love": "Titel zu Favoriten hinzufügen",
"current_song": "Aktuellen Titel Anzeigen"
}
+ },
+ "nowPlaying": {
+ "title": "Aktuelle Wiedergabe",
+ "empty": "Keine Wiedergabe",
+ "minutesAgo": "Vor %{smart_count} Minute |||| Vor %{smart_count} Minuten"
}
}
\ No newline at end of file
diff --git a/resources/i18n/el.json b/resources/i18n/el.json
index d574821d4..0d9ee05c5 100644
--- a/resources/i18n/el.json
+++ b/resources/i18n/el.json
@@ -33,7 +33,10 @@
"tags": "Πρόσθετες Ετικέτες",
"mappedTags": "Χαρτογραφημένες ετικέτες",
"rawTags": "Ακατέργαστες ετικέτες",
- "bitDepth": "Λίγο βάθος"
+ "bitDepth": "Λίγο βάθος",
+ "sampleRate": "Ποσοστό δειγματοληψίας",
+ "missing": "Απών",
+ "libraryName": "Βιβλιοθήκη"
},
"actions": {
"addToQueue": "Αναπαραγωγη Μετα",
@@ -42,7 +45,8 @@
"shuffleAll": "Ανακατεμα ολων",
"download": "Ληψη",
"playNext": "Επόμενη Αναπαραγωγή",
- "info": "Εμφάνιση Πληροφοριών"
+ "info": "Εμφάνιση Πληροφοριών",
+ "showInPlaylist": "Εμφάνιση στη λίστα αναπαραγωγής"
}
},
"album": {
@@ -72,7 +76,9 @@
"grouping": "Ομαδοποίηση",
"media": "Μέσα",
"mood": "Διάθεση",
- "date": "Ημερομηνία Ηχογράφησης"
+ "date": "Ημερομηνία Ηχογράφησης",
+ "missing": "Απών",
+ "libraryName": "Βιβλιοθήκη"
},
"actions": {
"playAll": "Αναπαραγωγή",
@@ -104,7 +110,8 @@
"rating": "Βαθμολογια",
"genre": "Είδος",
"size": "Μέγεθος",
- "role": "Ρόλος"
+ "role": "Ρόλος",
+ "missing": "Απών"
},
"roles": {
"albumartist": "Καλλιτέχνης Άλμπουμ |||| Καλλιτέχνες άλμπουμ",
@@ -119,7 +126,13 @@
"mixer": "Μίξερ |||| Μίξερ",
"remixer": "Ρεμίξερ |||| Ρεμίξερ",
"djmixer": "Dj Μίξερ |||| Dj Μίξερ",
- "performer": "Εκτελεστής |||| Ερμηνευτές"
+ "performer": "Εκτελεστής |||| Ερμηνευτές",
+ "maincredit": "Καλλιτέχνης Άλμπουμ ή Καλλιτέχνης |||| Καλλιτέχνες Άλμπουμ ή Καλλιτέχνες"
+ },
+ "actions": {
+ "shuffle": "Ανάμιξη",
+ "radio": "Ραδιόφωνο",
+ "topSongs": "Κορυφαία τραγούδια"
}
},
"user": {
@@ -132,14 +145,16 @@
"name": "Όνομα",
"password": "Κωδικός Πρόσβασης",
"createdAt": "Δημιουργήθηκε στις",
- "changePassword": "Αλλαγή Κωδικού Πρόσβασης;",
+ "changePassword": "Αλλαγή Κωδικού Πρόσβασης?",
"currentPassword": "Υπάρχων Κωδικός Πρόσβασης",
"newPassword": "Νέος Κωδικός Πρόσβασης",
"token": "Token",
- "lastAccessAt": "Τελευταία Πρόσβαση"
+ "lastAccessAt": "Τελευταία Πρόσβαση",
+ "libraries": "Βιβλιοθήκες"
},
"helperTexts": {
- "name": "Αλλαγές στο όνομα σας θα εφαρμοστούν στην επόμενη σύνδεση"
+ "name": "Αλλαγές στο όνομα σας θα εφαρμοστούν στην επόμενη σύνδεση",
+ "libraries": "Επιλέξτε συγκεκριμένες βιβλιοθήκες για αυτόν τον χρήστη, ή αφήστε την κενή για να χρησιμοποιήσετε την προεπιλεγμένη βιβλιοθήκη"
},
"notifications": {
"created": "Ο χρήστης δημιουργήθηκε",
@@ -148,7 +163,12 @@
},
"message": {
"listenBrainzToken": "Εισάγετε το token του χρήστη σας στο ListenBrainz.",
- "clickHereForToken": "Κάντε κλικ εδώ για να αποκτήσετε το token σας"
+ "clickHereForToken": "Κάντε κλικ εδώ για να αποκτήσετε το token σας",
+ "selectAllLibraries": "Επιλογή όλων των βιβλιοθηκών",
+ "adminAutoLibraries": "Οι χρήστες διαχειριστές έχουν αυτόματα πρόσβαση σε όλες τις βιβλιοθήκες"
+ },
+ "validation": {
+ "librariesRequired": "Πρέπει να επιλεγεί τουλάχιστον μία βιβλιοθήκη για χρήστες που δεν είναι διαχειριστές"
}
},
"player": {
@@ -192,11 +212,17 @@
"addNewPlaylist": "Δημιουργία \"%{name}\"",
"export": "Εξαγωγη",
"makePublic": "Να γίνει δημόσιο",
- "makePrivate": "Να γίνει ιδιωτικό"
+ "makePrivate": "Να γίνει ιδιωτικό",
+ "saveQueue": "Αποθήκευση ουράς στη λίστα αναπαραγωγής",
+ "searchOrCreate": "Αναζητήστε λίστες αναπαραγωγής ή πληκτρολογήστε για να δημιουργήσετε νέες...",
+ "pressEnterToCreate": "Πατήστε Enter για να δημιουργήσετε νέα λίστα αναπαραγωγής",
+ "removeFromSelection": "Αφαίρεση από την επιλογή"
},
"message": {
"duplicate_song": "Προσθήκη διπλοεγγραφών τραγουδιών",
- "song_exist": "Υπάρχουν διπλοεγγραφές στην λίστα αναπαραγωγής. Θέλετε να προστεθούν οι διπλοεγγραφές ή να τις παραβλέψετε;"
+ "song_exist": "Υπάρχουν διπλοεγγραφές στην λίστα αναπαραγωγής. Θέλετε να προστεθούν οι διπλοεγγραφές ή να τις παραβλέψετε?",
+ "noPlaylistsFound": "Δεν βρέθηκαν λίστες αναπαραγωγής",
+ "noPlaylists": "Δεν υπάρχουν διαθέσιμες λίστες αναπαραγωγής"
}
},
"radio": {
@@ -234,15 +260,69 @@
"fields": {
"path": "Διαδρομή",
"size": "Μέγεθος",
- "updatedAt": "Εξαφανίστηκε"
+ "updatedAt": "Εξαφανίστηκε",
+ "libraryName": "Βιβλιοθήκη"
},
"actions": {
- "remove": "Αφαίρεση"
+ "remove": "Αφαίρεση",
+ "remove_all": "Αφαίρεση όλων"
},
"notifications": {
"removed": "Λείπει αρχείο(α) αφαιρέθηκε"
},
"empty": "Δεν λείπουν αρχεία"
+ },
+ "library": {
+ "name": "Βιβλιοθήκη |||| Βιβλιοθήκες",
+ "fields": {
+ "name": "Ονομα",
+ "path": "διαδρομή",
+ "remotePath": "Απομακρυσμένη διαδρομή",
+ "lastScanAt": "Τελευταία σάρωση",
+ "songCount": "Τραγούδια",
+ "albumCount": "Άλμπουμ",
+ "artistCount": "Καλλιτέχνες",
+ "totalSongs": "Τραγούδια",
+ "totalAlbums": "Άλμπουμ",
+ "totalArtists": "Καλλιτέχνες",
+ "totalFolders": "Φάκελοι",
+ "totalFiles": "Αρχεία",
+ "totalMissingFiles": "Λείπει αρχείο",
+ "totalSize": "Συνολικό μέγεθος",
+ "totalDuration": "Διάρκεια",
+ "defaultNewUsers": "Προεπιλογή για νέους χρήστες",
+ "createdAt": "Δημιουργήθηκε",
+ "updatedAt": "Ενημερώθηκε"
+ },
+ "sections": {
+ "basic": "Βασικές πληροφορίες",
+ "statistics": "Στατιστική"
+ },
+ "actions": {
+ "scan": "Σάρωση βιβλιοθήκης",
+ "manageUsers": "Διαχείριση πρόσβασης χρήστη",
+ "viewDetails": "Προβολή λεπτομερειών"
+ },
+ "notifications": {
+ "created": "Η βιβλιοθήκη δημιουργήθηκε με επιτυχία",
+ "updated": "Η βιβλιοθήκη ενημερώθηκε με επιτυχία",
+ "deleted": "Η βιβλιοθήκη διαγράφηκε με επιτυχία",
+ "scanStarted": "Ξεκίνησε η σάρωση της βιβλιοθήκης",
+ "scanCompleted": "Η σάρωση της βιβλιοθήκης ολοκληρώθηκε"
+ },
+ "validation": {
+ "nameRequired": "Απαιτείται όνομα βιβλιοθήκης",
+ "pathRequired": "Απαιτείται διαδρομή βιβλιοθήκης",
+ "pathNotDirectory": "Η διαδρομή της βιβλιοθήκης πρέπει να είναι ένας κατάλογος",
+ "pathNotFound": "Η διαδρομή της βιβλιοθήκης δεν βρέθηκε",
+ "pathNotAccessible": "Η διαδρομή της βιβλιοθήκης δεν είναι προσβάσιμη",
+ "pathInvalid": "Μη έγκυρη διαδρομή βιβλιοθήκης"
+ },
+ "messages": {
+ "deleteConfirm": "Είστε βέβαιοι ότι θέλετε να διαγράψετε αυτήν τη βιβλιοθήκη? Αυτή η ενέργεια θα καταργήσει όλα τα σχετικά δεδομένα και την πρόσβαση των χρηστών.",
+ "scanInProgress": "Σάρωση σε εξέλιξη...",
+ "noLibrariesAssigned": "Δεν έχουν αντιστοιχιστεί βιβλιοθήκες σε αυτόν τον χρήστη"
+ }
}
},
"ra": {
@@ -305,7 +385,7 @@
"skip": "Παράβλεψη",
"bulk_actions_mobile": "1 |||| %{smart_count}",
"share": "Κοινοποίηση",
- "download": "Λήψη "
+ "download": "Λήψη"
},
"boolean": {
"true": "Ναι",
@@ -344,10 +424,10 @@
},
"message": {
"about": "Σχετικά",
- "are_you_sure": "Είστε σίγουροι;",
- "bulk_delete_content": "Είστε σίγουροι πως θέλετε να διαγράψετε το %{name}; |||| Είστε σίγουροι πως θέλετε να διαγράψετε τα %{smart_count};",
+ "are_you_sure": "Είστε σίγουροι?",
+ "bulk_delete_content": "Είστε σίγουροι πως θέλετε να διαγράψετε το %{name}? |||| Είστε σίγουροι πως θέλετε να διαγράψετε τα %{smart_count}?",
"bulk_delete_title": "Διαγραφή του %{name} |||| Διαγραφή του %{smart_count} %{name}",
- "delete_content": "Είστε σίγουροι πως θέλετε να διαγράψετε αυτό το αντικείμενο;",
+ "delete_content": "Είστε σίγουροι πως θέλετε να διαγράψετε αυτό το αντικείμενο?",
"delete_title": "Διαγραφή του %{name} #%{id}",
"details": "Λεπτομέρειες",
"error": "Παρουσιάστηκε ένα πρόβλημα από τη μεριά του πελάτη και το αίτημα σας δεν μπορεί να ολοκληρωθεί.",
@@ -356,12 +436,12 @@
"no": "Όχι",
"not_found": "Είτε έχετε εισάγει λανθασμένο URL, είτε ακολουθήσατε έναν υπερσύνδεσμο που δεν ισχύει.",
"yes": "Ναι",
- "unsaved_changes": "Μερικές από τις αλλαγές σας δεν έχουν αποθηκευτεί. Είστε σίγουροι πως θέλετε να τις αγνοήσετε;"
+ "unsaved_changes": "Μερικές από τις αλλαγές σας δεν έχουν αποθηκευτεί. Είστε σίγουροι πως θέλετε να τις αγνοήσετε?"
},
"navigation": {
"no_results": "Δεν βρέθηκαν αποτελέσματα",
"no_more_results": "Η σελίδα %{page} είναι εκτός ορίων. Δοκιμάστε την προηγούμενη σελίδα.",
- "page_out_of_boundaries": "Η σελίδα {page} είναι εκτός ορίων",
+ "page_out_of_boundaries": "Η σελίδα %{page} είναι εκτός ορίων",
"page_out_from_end": "Δεν είναι δυνατή η πλοήγηση πέραν της τελευταίας σελίδας",
"page_out_from_begin": "Δεν είναι δυνατή η πλοήγηση πριν τη σελίδα 1",
"page_range_info": "%{offsetBegin}-%{offsetEnd} από %{total}",
@@ -397,7 +477,7 @@
"songsAddedToPlaylist": "Προστέθηκε 1 τραγούδι στη λίστα αναπαραγωγής |||| Προστέθηκαν %{smart_count} τραγούδια στη λίστα αναπαραγωγής",
"noPlaylistsAvailable": "Κανένα διαθέσιμο",
"delete_user_title": "Διαγραφή του χρήστη '%{name}'",
- "delete_user_content": "Είστε σίγουροι πως θέλετε να διαγράψετε αυτό το χρήστη και όλα τα δεδομένα του (συμπεριλαμβανομένων των λιστών αναπαραγωγής και προτιμήσεων);",
+ "delete_user_content": "Είστε σίγουροι πως θέλετε να διαγράψετε αυτό το χρήστη και όλα τα δεδομένα του (συμπεριλαμβανομένων των λιστών αναπαραγωγής και προτιμήσεων)?",
"notifications_blocked": "Έχετε μπλοκάρει τις Ειδοποιήσεις από τη σελίδα, μέσω των ρυθμίσεων του περιηγητή ιστού σας",
"notifications_not_available": "Αυτός ο περιηγητής ιστού δεν υποστηρίζει ειδοποιήσεις στην επιφάνεια εργασίας ή δεν έχετε πρόσβαση στο Navidrome μέσω https",
"lastfmLinkSuccess": "Το Last.fm έχει διασυνδεθεί επιτυχώς και η λειτουργία scrobbling ενεργοποιήθηκε",
@@ -422,7 +502,11 @@
"downloadDialogTitle": "Λήψη %{resource} '%{name}'(%{size})",
"shareCopyToClipboard": "Αντιγραφή στο πρόχειρο: Ctrl+C, Enter",
"remove_missing_title": "Αφαιρέστε τα αρχεία που λείπουν",
- "remove_missing_content": "Είστε βέβαιοι ότι θέλετε να αφαιρέσετε τα επιλεγμένα αρχεία που λείπουν από τη βάση δεδομένων; Αυτό θα καταργήσει οριστικά τυχόν αναφορές σε αυτά, συμπεριλαμβανομένων των αριθμών παιχνιδιών και των αξιολογήσεών τους."
+ "remove_missing_content": "Είστε βέβαιοι ότι θέλετε να αφαιρέσετε τα επιλεγμένα αρχεία που λείπουν από τη βάση δεδομένων? Αυτό θα καταργήσει οριστικά τυχόν αναφορές σε αυτά, συμπεριλαμβανομένων των αριθμών παιχνιδιών και των αξιολογήσεών τους.",
+ "remove_all_missing_title": "Αφαίρεση όλων των αρχείων που λείπουν",
+ "remove_all_missing_content": "Είστε βέβαιοι ότι θέλετε να καταργήσετε όλα τα αρχεία που λείπουν από τη βάση δεδομένων? Αυτό θα καταργήσει οριστικά τυχόν αναφορές σε αυτά, συμπεριλαμβανομένου του αριθμού αναπαραγωγών και των αξιολογήσεών τους.",
+ "noSimilarSongsFound": "Δεν βρέθηκαν παρόμοια τραγούδια",
+ "noTopSongsFound": "Δεν βρέθηκαν κορυφαία τραγούδια"
},
"menu": {
"library": "Βιβλιοθήκη",
@@ -451,7 +535,13 @@
"albumList": "Άλμπουμ",
"about": "Σχετικά",
"playlists": "Λίστες Αναπαραγωγής",
- "sharedPlaylists": "Κοινοποιημένες Λίστες Αναπαραγωγής"
+ "sharedPlaylists": "Κοινοποιημένες Λίστες Αναπαραγωγής",
+ "librarySelector": {
+ "allLibraries": "Όλες οι βιβλιοθήκες (%{count})",
+ "multipleLibraries": "%{selected} από %{total} Βιβλιοθήκες",
+ "selectLibraries": "Επιλέξτε βιβλιοθήκες",
+ "none": "Κανένα"
+ }
},
"player": {
"playListsText": "Ουρά Αναπαραγωγής",
@@ -488,6 +578,21 @@
"disabled": "Απενεργοποιημένο",
"waiting": "Αναμονή"
}
+ },
+ "tabs": {
+ "about": "Σχετικά",
+ "config": "Διαμόρφωση"
+ },
+ "config": {
+ "configName": "Όνομα διαμόρφωσης",
+ "environmentVariable": "Μεταβλητή περιβάλλοντος",
+ "currentValue": "Τρέχουσα Αξία",
+ "configurationFile": "Αρχείο διαμόρφωσης",
+ "exportToml": "Ρύθμιση παραμέτρων εξαγωγής (TOML)",
+ "exportSuccess": "Η διαμόρφωση εξήχθη στο πρόχειρο σε μορφή TOML",
+ "exportFailed": "Η αντιγραφή της διαμόρφωσης απέτυχε",
+ "devFlagsHeader": "Σημαίες Ανάπτυξης (υπόκειται σε αλλαγές / αφαίρεση)",
+ "devFlagsComment": "Αυτές είναι πειραματικές ρυθμίσεις και ενδέχεται να καταργηθούν σε μελλοντικές εκδόσεις"
}
},
"activity": {
@@ -496,7 +601,10 @@
"quickScan": "Γρήγορη Σάρωση",
"fullScan": "Πλήρης Σάρωση",
"serverUptime": "Λειτουργία Διακομιστή",
- "serverDown": "ΕΚΤΟΣ ΣΥΝΔΕΣΗΣ"
+ "serverDown": "ΕΚΤΟΣ ΣΥΝΔΕΣΗΣ",
+ "scanType": "Τύπος",
+ "status": "Σφάλμα σάρωσης",
+ "elapsedTime": "Χρόνος που πέρασε"
},
"help": {
"title": "Συντομεύσεις του Navidrome",
@@ -511,5 +619,10 @@
"toggle_love": "Προσθήκη αυτού του κομματιού στα αγαπημένα",
"current_song": "Μεταβείτε στο Τρέχον τραγούδι"
}
+ },
+ "nowPlaying": {
+ "title": "Αναπαραγωγή τώρα",
+ "empty": "Δεν παίζει τίποτα",
+ "minutesAgo": "%{smart_count} λεπτό πριν |||| %{smart_count} λεπτά πριν"
}
}
\ No newline at end of file
diff --git a/resources/i18n/eo.json b/resources/i18n/eo.json
index 570943a1d..bdf143969 100644
--- a/resources/i18n/eo.json
+++ b/resources/i18n/eo.json
@@ -24,16 +24,18 @@
"rating": "Takso",
"quality": "Kvalito",
"bpm": "Pulsrapideco",
- "playDate": "",
- "channels": "",
- "createdAt": "",
+ "playDate": "Laste Ludita",
+ "channels": "Kanaloj",
+ "createdAt": "Dato de aligo",
"grouping": "",
- "mood": "",
+ "mood": "Humoro",
"participants": "",
- "tags": "",
- "mappedTags": "",
- "rawTags": "",
- "bitDepth": ""
+ "tags": "Aldonaj Etikedoj",
+ "mappedTags": "Mapigitaj etikedoj",
+ "rawTags": "Krudaj etikedoj",
+ "bitDepth": "",
+ "sampleRate": "",
+ "missing": ""
},
"actions": {
"addToQueue": "Ludi Poste",
@@ -42,7 +44,7 @@
"shuffleAll": "Miksu Ĉiujn",
"download": "Elŝuti",
"playNext": "Ludu Poste",
- "info": ""
+ "info": "Akiri Informon"
}
},
"album": {
@@ -60,19 +62,20 @@
"updatedAt": "Ĝisdatigita je :",
"comment": "Komento",
"rating": "Takso",
- "createdAt": "",
- "size": "",
- "originalDate": "",
- "releaseDate": "",
- "releases": "",
- "released": "",
+ "createdAt": "Dato aldonita",
+ "size": "Grando",
+ "originalDate": "Originala",
+ "releaseDate": "Publikiĝis",
+ "releases": "Publikiĝo |||| Publikiĝoj",
+ "released": "Publikiĝis",
"recordLabel": "",
"catalogNum": "",
- "releaseType": "",
+ "releaseType": "Tipo",
"grouping": "",
"media": "",
- "mood": "",
- "date": ""
+ "mood": "Humoro",
+ "date": "",
+ "missing": ""
},
"actions": {
"playAll": "Ludi",
@@ -81,43 +84,44 @@
"shuffle": "Miksi",
"addToPlaylist": "Aldoni al la Ludlisto",
"download": "Elŝuti",
- "info": "",
- "share": ""
+ "info": "Akiri Informon",
+ "share": "Diskonigi"
},
"lists": {
"all": "Ĉiuj",
- "random": "Hazarda",
- "recentlyAdded": "Lastatempe Aldonita",
- "recentlyPlayed": "Lastatempe Ludita",
+ "random": "Hazardaj",
+ "recentlyAdded": "Lastatempe Aldonitaj",
+ "recentlyPlayed": "Lastatempe Luditaj",
"mostPlayed": "Plej Luditaj",
- "starred": "Stelplena",
- "topRated": "Plej Alte Taksite"
+ "starred": "Stelplenaj",
+ "topRated": "Plej Alte Taksitaj"
}
},
"artist": {
"name": "Artisto |||| Artistoj",
"fields": {
"name": "Nomo",
- "albumCount": "Nombro da albumoj",
- "songCount": "Kanto kalkula",
- "playCount": "Teatraĵoj",
+ "albumCount": "Kvanto da Albumoj",
+ "songCount": "Kanta Kalkulo",
+ "playCount": "Ludoj",
"rating": "Takso",
- "genre": "",
- "size": "",
- "role": ""
+ "genre": "Ĝenro",
+ "size": "Grando",
+ "role": "",
+ "missing": ""
},
"roles": {
- "albumartist": "",
- "artist": "",
- "composer": "",
- "conductor": "",
- "lyricist": "",
- "arranger": "",
+ "albumartist": "Albuma Artisto |||| Albumaj Artistoj",
+ "artist": "Artisto |||| Artistoj",
+ "composer": "Komponisto |||| Komponistoj",
+ "conductor": "Dirigento |||| Dirigentoj",
+ "lyricist": "Kantoteksisto |||| Kantotekstistoj",
+ "arranger": "Aranĝisto |||| Aranĝistoj",
"producer": "",
"director": "",
"engineer": "",
- "mixer": "",
- "remixer": "",
+ "mixer": "Miksisto |||| Miksistoj",
+ "remixer": "Remiksisto |||| Remiksistoj",
"djmixer": "",
"performer": ""
}
@@ -135,8 +139,8 @@
"changePassword": "Ĉu Ŝanĝi Pasvorton?",
"currentPassword": "Nuna Pasvorto",
"newPassword": "Nova Pasvorto",
- "token": "",
- "lastAccessAt": ""
+ "token": "Ĵetono",
+ "lastAccessAt": "Lasta Atingo"
},
"helperTexts": {
"name": "Ŝanĝoj de via nomo nur ĝisdatiĝs je via sekvanta ensaluto"
@@ -147,8 +151,8 @@
"deleted": "Uzanto forigita"
},
"message": {
- "listenBrainzToken": "",
- "clickHereForToken": ""
+ "listenBrainzToken": "Enigi vian uzantan ĵetonon de ListenBrainz.",
+ "clickHereForToken": "Alkakli ĉi tie por akiri vian ĵetonon"
}
},
"player": {
@@ -161,7 +165,7 @@
"userName": "Uzantnomo",
"lastSeen": "Laste Vidita Je",
"reportRealPath": "Raporti vera pado",
- "scrobbleEnabled": ""
+ "scrobbleEnabled": "Sendi Scrobbles al eksteraj servoj"
}
},
"transcoding": {
@@ -191,8 +195,9 @@
"selectPlaylist": "Elektu ludliston :",
"addNewPlaylist": "Krei \"%{name}\"",
"export": "Eksporti",
- "makePublic": "",
- "makePrivate": ""
+ "makePublic": "Publikigi",
+ "makePrivate": "Malpublikigi",
+ "saveQueue": ""
},
"message": {
"duplicate_song": "Aldoni duobligitajn kantojn",
@@ -200,33 +205,33 @@
}
},
"radio": {
- "name": "",
+ "name": "Radio |||| Radioj",
"fields": {
- "name": "",
- "streamUrl": "",
- "homePageUrl": "",
- "updatedAt": "",
- "createdAt": ""
+ "name": "Nomo",
+ "streamUrl": "Flua Ligilo",
+ "homePageUrl": "Hejmpaĝa Ligilo",
+ "updatedAt": "Ĝisdatiĝis je",
+ "createdAt": "Kreiĝis je"
},
"actions": {
- "playNow": ""
+ "playNow": "Ludi Nun"
}
},
"share": {
- "name": "",
+ "name": "Diskonigo |||| Diskonigoj",
"fields": {
- "username": "",
- "url": "",
- "description": "",
- "contents": "",
- "expiresAt": "",
- "lastVisitedAt": "",
- "visitCount": "",
- "format": "",
- "maxBitRate": "",
- "updatedAt": "",
- "createdAt": "",
- "downloadable": ""
+ "username": "Diskonigite De",
+ "url": "Ligilo",
+ "description": "Priskribo",
+ "contents": "Enhavo",
+ "expiresAt": "Senvalidiĝas",
+ "lastVisitedAt": "Laste Vizitita",
+ "visitCount": "Vizitoj",
+ "format": "Formato",
+ "maxBitRate": "Maks. Bitrapido",
+ "updatedAt": "Ĝisdatiĝis je",
+ "createdAt": "Fariĝis je",
+ "downloadable": "Ĉu Ebligi Elŝutojn?"
}
},
"missing": {
@@ -237,7 +242,8 @@
"updatedAt": ""
},
"actions": {
- "remove": ""
+ "remove": "",
+ "remove_all": ""
},
"notifications": {
"removed": ""
@@ -258,7 +264,7 @@
"sign_in": "Ensaluti",
"sign_in_error": "Aŭtentikigo malsukcesis, bonvolu reprovi",
"logout": "Elsaluti",
- "insightsCollectionNote": ""
+ "insightsCollectionNote": "Navidrome kolektas anoniman uzdatumon por helpi\nplibonigi la projekton. Alklaku [ĉi tie] por lerni pli kaj\nsupozi permeson se vi volas"
},
"validation": {
"invalidChars": "Bonvolu uzi nur literojn kaj ciferojn",
@@ -273,7 +279,7 @@
"oneOf": "Devas esti unu el: %{options}",
"regex": "Devas kongrui kun specifa formato (regexp): %{pattern}",
"unique": "Devas esti unika",
- "url": ""
+ "url": "Devas esti valida ligilo"
},
"action": {
"add_filter": "Aldoni filtrilon",
@@ -303,9 +309,9 @@
"close_menu": "Fermu menuon",
"unselect": "Malelekti",
"skip": "Pasigi",
- "bulk_actions_mobile": "",
- "share": "",
- "download": ""
+ "bulk_actions_mobile": "1 |||| %{smart_count}",
+ "share": "Diskonigi",
+ "download": "Elŝuti"
},
"boolean": {
"true": "Jes",
@@ -381,13 +387,13 @@
"i18n_error": "Ne eblas ŝargi la tradukojn por la specifa lingvo",
"canceled": "Ago nuligita",
"logged_out": "Via seanco finiĝis, bonvolu rekonekti.",
- "new_version": ""
+ "new_version": "Nova versio haveblas! Bonvolu reŝargi la fenestron."
},
"toggleFieldsMenu": {
- "columnsToDisplay": "",
+ "columnsToDisplay": "Kolumnoj Por Montri",
"layout": "Aranĝo",
"grid": "Krado",
- "table": ""
+ "table": "Tabelo"
}
},
"message": {
@@ -400,29 +406,31 @@
"delete_user_content": "Ĉu vi certas, ke vi volas forigi ĉi tiun uzanton kaj ĉiujn iliajn datumojn (inkluzive ludlistojn kaj preferojn) ?",
"notifications_blocked": "Vi blokis sciigojn por ĉi tiu retejo en la agordoj de via retumilo",
"notifications_not_available": "Ĉi tiu retumilo ne subtenas labortablajn sciigojn aŭ vi ne aliras Navidrome per https",
- "lastfmLinkSuccess": "",
- "lastfmLinkFailure": "",
- "lastfmUnlinkSuccess": "",
- "lastfmUnlinkFailure": "",
+ "lastfmLinkSuccess": "Last.fm sukcese ligiĝis kaj scrobbling ebliĝis",
+ "lastfmLinkFailure": "Last.fm ne povis ligiĝi",
+ "lastfmUnlinkSuccess": "Last.fm malligiĝis kaj scrobbling malebliĝis",
+ "lastfmUnlinkFailure": "Last.fm ne povis malligiĝi",
"openIn": {
- "lastfm": "",
- "musicbrainz": ""
+ "lastfm": "Malfermi en Last.fm",
+ "musicbrainz": "Malfermi en MusicBrainz"
},
- "lastfmLink": "",
- "listenBrainzLinkSuccess": "",
- "listenBrainzLinkFailure": "",
- "listenBrainzUnlinkSuccess": "",
- "listenBrainzUnlinkFailure": "",
- "downloadOriginalFormat": "",
- "shareOriginalFormat": "",
- "shareDialogTitle": "",
- "shareBatchDialogTitle": "",
- "shareSuccess": "",
- "shareFailure": "",
- "downloadDialogTitle": "",
- "shareCopyToClipboard": "",
+ "lastfmLink": "Legi Pli...",
+ "listenBrainzLinkSuccess": "ListenBrainz sukcese ligiĝis kaj scrobbling ebliĝis kiel uzanto: %{user}",
+ "listenBrainzLinkFailure": "ListenBrainz ne povis ligiĝi: %{error}",
+ "listenBrainzUnlinkSuccess": "ListenBrainz malligiĝis kaj scrobbling malebliĝis",
+ "listenBrainzUnlinkFailure": "ListenBrainz ne povis malligiĝi",
+ "downloadOriginalFormat": "Elŝuti en originala formato",
+ "shareOriginalFormat": "Diskonigi en originala formato",
+ "shareDialogTitle": "Diskonigi %{resource} '%{name}'",
+ "shareBatchDialogTitle": "Diskonigi 1 %{resource} |||| Diskonigi %{smart_count} %{resource}",
+ "shareSuccess": "Ligilo kopiiĝis al la tondujo: %{url}",
+ "shareFailure": "Eraro de kopio de ligilo %{url} al la tondujo",
+ "downloadDialogTitle": "Elŝuti %{resource} '%{name}' (%{size})",
+ "shareCopyToClipboard": "Kopii al la tondujo: Ctrl+C, Enter",
"remove_missing_title": "",
- "remove_missing_content": ""
+ "remove_missing_content": "Ĉu vi certas, ke vi volas forigi la elektitajn mankajn dosierojn de la datumbazo? Ĉi tio forigos eterne ĉiujn referencojn de ili, inkluzive iliajn ludkvantojn kaj taksojn.",
+ "remove_all_missing_title": "",
+ "remove_all_missing_content": ""
},
"menu": {
"library": "Biblioteko",
@@ -436,22 +444,22 @@
"language": "Lingvo",
"defaultView": "Defaŭlta Vido",
"desktop_notifications": "Labortablaj sciigoj",
- "lastfmScrobbling": "",
- "listenBrainzScrobbling": "",
- "replaygain": "",
- "preAmp": "",
+ "lastfmScrobbling": "Scrobble al Last.fm",
+ "listenBrainzScrobbling": "Scrobble al ListenBrainz",
+ "replaygain": "ReplayGain-Reĝimo",
+ "preAmp": "ReplayGain PreAmp (dB)",
"gain": {
- "none": "",
- "album": "",
- "track": ""
+ "none": "Malebligita",
+ "album": "Uzi Albuman Songajnon",
+ "track": "Uzi Kantan Songajnon"
},
"lastfmNotConfigured": ""
}
},
"albumList": "Albumoj",
"about": "Pri",
- "playlists": "",
- "sharedPlaylists": ""
+ "playlists": "Ludlistoj",
+ "sharedPlaylists": "Diskonigitaj Ludistoj"
},
"player": {
"playListsText": "Atendovico",
@@ -485,7 +493,7 @@
"featureRequests": "Trajta peto",
"lastInsightsCollection": "",
"insights": {
- "disabled": "",
+ "disabled": "Malebligita",
"waiting": ""
}
}
@@ -496,7 +504,10 @@
"quickScan": "Rapida Skanado",
"fullScan": "Plena Skanado",
"serverUptime": "Servila daŭro de funkciado",
- "serverDown": "SENKONEKTA"
+ "serverDown": "SENKONEKTA",
+ "scanType": "",
+ "status": "",
+ "elapsedTime": ""
},
"help": {
"title": "Navidrome klavkomando",
@@ -509,7 +520,7 @@
"vol_up": "Pli volumo",
"vol_down": "Malpli volumo",
"toggle_love": "Baskuli la stelon de nuna kanto",
- "current_song": ""
+ "current_song": "Iri al Nuna Kanto"
}
}
}
\ No newline at end of file
diff --git a/resources/i18n/es.json b/resources/i18n/es.json
index 4c811b447..4c53b8986 100644
--- a/resources/i18n/es.json
+++ b/resources/i18n/es.json
@@ -28,20 +28,25 @@
"channels": "Canales",
"createdAt": "Creado el",
"grouping": "Agrupación",
- "mood": "",
+ "mood": "Estado de ánimo",
"participants": "Participantes",
"tags": "Etiquetas",
"mappedTags": "Etiquetas asignadas",
- "rawTags": "Etiquetas sin procesar"
+ "rawTags": "Etiquetas sin procesar",
+ "bitDepth": "Profundidad de bits",
+ "sampleRate": "Frecuencia de muestreo",
+ "missing": "Faltante",
+ "libraryName": ""
},
"actions": {
"addToQueue": "Reproducir después",
"playNow": "Reproducir ahora",
- "addToPlaylist": "Agregar a la lista de reproducción",
+ "addToPlaylist": "Agregar a la playlist",
"shuffleAll": "Todas aleatorias",
"download": "Descarga",
"playNext": "Siguiente",
- "info": "Obtener información"
+ "info": "Obtener información",
+ "showInPlaylist": "Mostrar en la lista de reproducción"
}
},
"album": {
@@ -69,8 +74,11 @@
"catalogNum": "Número de catálogo",
"releaseType": "Tipo de lanzamiento",
"grouping": "Agrupación",
- "media": "",
- "mood": ""
+ "media": "Medios",
+ "mood": "Estado de ánimo",
+ "date": "Fecha de grabación",
+ "missing": "Faltante",
+ "libraryName": ""
},
"actions": {
"playAll": "Reproducir",
@@ -89,7 +97,7 @@
"recentlyPlayed": "Recientes",
"mostPlayed": "Más reproducidos",
"starred": "Favoritos",
- "topRated": "Los mejores calificados"
+ "topRated": "Mejor calificados"
}
},
"artist": {
@@ -102,7 +110,8 @@
"rating": "Calificación",
"genre": "Género",
"size": "Tamaño",
- "role": "Rol"
+ "role": "Rol",
+ "missing": "Faltante"
},
"roles": {
"albumartist": "Artista del álbum",
@@ -117,7 +126,13 @@
"mixer": "Mezclador",
"remixer": "Remixer",
"djmixer": "DJ Mixer",
- "performer": "Intérprete"
+ "performer": "Intérprete",
+ "maincredit": ""
+ },
+ "actions": {
+ "shuffle": "Aleatorio",
+ "radio": "Radio",
+ "topSongs": ""
}
},
"user": {
@@ -134,10 +149,12 @@
"currentPassword": "Contraseña actual",
"newPassword": "Nueva contraseña",
"token": "Token",
- "lastAccessAt": "Último acceso"
+ "lastAccessAt": "Último acceso",
+ "libraries": ""
},
"helperTexts": {
- "name": "Los cambios a tu nombre se verán en el próximo inicio de sesión"
+ "name": "Los cambios a tu nombre se verán en el próximo inicio de sesión",
+ "libraries": ""
},
"notifications": {
"created": "Usuario creado",
@@ -146,7 +163,12 @@
},
"message": {
"listenBrainzToken": "Escribe tu token de usuario de ListenBrainz",
- "clickHereForToken": "Click aquí para obtener tu token"
+ "clickHereForToken": "Click aquí para obtener tu token",
+ "selectAllLibraries": "",
+ "adminAutoLibraries": ""
+ },
+ "validation": {
+ "librariesRequired": ""
}
},
"player": {
@@ -190,11 +212,17 @@
"addNewPlaylist": "Creada \"%{name}\"",
"export": "Exportar",
"makePublic": "Hazla pública",
- "makePrivate": "Hazla privada"
+ "makePrivate": "Hazla privada",
+ "saveQueue": "Guardar la fila de reproducción en una playlist",
+ "searchOrCreate": "Buscar listas de reproducción o escribe para crear una nueva…",
+ "pressEnterToCreate": "Pulsa Enter para crear una nueva lista de reproducción",
+ "removeFromSelection": "Quitar de la selección"
},
"message": {
- "duplicate_song": "Algunas de las canciones seleccionadas están presentes en la lista de reproducción",
- "song_exist": "Se están agregando duplicados a la lista de reproducción. ¿Quieres agregar los duplicados o omitirlos?"
+ "duplicate_song": "Algunas de las canciones seleccionadas están presentes en la playlist",
+ "song_exist": "Se están agregando duplicados a la playlist. ¿Quieres agregar los duplicados o omitirlos?",
+ "noPlaylistsFound": "No se encontraron listas de reproducción",
+ "noPlaylists": "No hay listas de reproducción disponibles"
}
},
"radio": {
@@ -232,13 +260,68 @@
"fields": {
"path": "Ruta",
"size": "Tamaño",
- "updatedAt": "Actualizado el"
+ "updatedAt": "Actualizado el",
+ "libraryName": ""
},
"actions": {
- "remove": "Eliminar"
+ "remove": "Eliminar",
+ "remove_all": "Eliminar todo"
},
"notifications": {
"removed": "Eliminado"
+ },
+ "empty": "No hay archivos perdidos"
+ },
+ "library": {
+ "name": "",
+ "fields": {
+ "name": "",
+ "path": "",
+ "remotePath": "",
+ "lastScanAt": "",
+ "songCount": "",
+ "albumCount": "",
+ "artistCount": "",
+ "totalSongs": "",
+ "totalAlbums": "",
+ "totalArtists": "",
+ "totalFolders": "",
+ "totalFiles": "",
+ "totalMissingFiles": "",
+ "totalSize": "",
+ "totalDuration": "",
+ "defaultNewUsers": "",
+ "createdAt": "",
+ "updatedAt": ""
+ },
+ "sections": {
+ "basic": "",
+ "statistics": ""
+ },
+ "actions": {
+ "scan": "",
+ "manageUsers": "",
+ "viewDetails": ""
+ },
+ "notifications": {
+ "created": "",
+ "updated": "",
+ "deleted": "",
+ "scanStarted": "",
+ "scanCompleted": ""
+ },
+ "validation": {
+ "nameRequired": "",
+ "pathRequired": "",
+ "pathNotDirectory": "",
+ "pathNotFound": "",
+ "pathNotAccessible": "",
+ "pathInvalid": ""
+ },
+ "messages": {
+ "deleteConfirm": "",
+ "scanInProgress": "",
+ "noLibrariesAssigned": ""
}
}
},
@@ -419,7 +502,11 @@
"downloadDialogTitle": "Descargar %{resource} '%{name}' (%{size})",
"shareCopyToClipboard": "Copiar al portapapeles: Ctrl+C, Intro",
"remove_missing_title": "Eliminar elemento faltante",
- "remove_missing_content": ""
+ "remove_missing_content": "¿Realmente desea eliminar los archivos faltantes seleccionados de la base de datos? Esto eliminará permanentemente cualquier referencia a ellos, incluidas sus reproducciones y valoraciones.",
+ "remove_all_missing_title": "Eliminar todos los archivos perdidos",
+ "remove_all_missing_content": "¿Realmente desea eliminar todos los archivos faltantes de la base de datos? Esto eliminará permanentemente cualquier referencia a ellos, incluidas sus reproducciones y valoraciones.",
+ "noSimilarSongsFound": "No se encontraron canciones similares",
+ "noTopSongsFound": ""
},
"menu": {
"library": "Biblioteca",
@@ -448,10 +535,16 @@
"albumList": "Álbumes",
"about": "Acerca de",
"playlists": "Playlists",
- "sharedPlaylists": "Playlists Compartidas"
+ "sharedPlaylists": "Playlists Compartidas",
+ "librarySelector": {
+ "allLibraries": "",
+ "multipleLibraries": "",
+ "selectLibraries": "",
+ "none": ""
+ }
},
"player": {
- "playListsText": "Lista de reproducción",
+ "playListsText": "Fila de reproducción",
"openText": "Abrir",
"closeText": "Cerrar",
"notContentText": "Sin música",
@@ -485,6 +578,21 @@
"disabled": "Deshabilitado",
"waiting": "Esperando"
}
+ },
+ "tabs": {
+ "about": "Acerca de",
+ "config": "Configuración"
+ },
+ "config": {
+ "configName": "Nombre de la configuración",
+ "environmentVariable": "Variables de entorno",
+ "currentValue": "Valor actual",
+ "configurationFile": "Archivo de configuración",
+ "exportToml": "Exportar configuración (TOML)",
+ "exportSuccess": "Configuración exportada al portapapeles en formato TOML",
+ "exportFailed": "Error al copiar la configuración",
+ "devFlagsHeader": "Indicadores de desarrollo (sujetos a cambios o eliminación)",
+ "devFlagsComment": "Estas son configuraciones experimentales y pueden eliminarse en versiones futuras"
}
},
"activity": {
@@ -493,7 +601,10 @@
"quickScan": "Escaneo rápido",
"fullScan": "Escaneo completo",
"serverUptime": "Uptime del servidor",
- "serverDown": "OFFLINE"
+ "serverDown": "OFFLINE",
+ "scanType": "Tipo",
+ "status": "Error de escaneo",
+ "elapsedTime": "Tiempo transcurrido"
},
"help": {
"title": "Atajos de teclado de Navidrome",
@@ -508,5 +619,10 @@
"toggle_love": "Marca esta canción como favorita",
"current_song": "Canción actual"
}
+ },
+ "nowPlaying": {
+ "title": "",
+ "empty": "",
+ "minutesAgo": ""
}
-}
+}
\ No newline at end of file
diff --git a/resources/i18n/eu.json b/resources/i18n/eu.json
index 5470ab38b..cb5927a74 100644
--- a/resources/i18n/eu.json
+++ b/resources/i18n/eu.json
@@ -27,23 +27,25 @@
"rating": "Balorazioa",
"quality": "Kalitatea",
"bpm": "BPM",
- "playDate": "Azkenekoz erreproduzitua:",
+ "playDate": "Azken erreprodukzioa:",
"createdAt": "Gehitu zen data:",
"grouping": "Multzokatzea",
"mood": "Aldartea",
"participants": "Partaide gehiago",
"tags": "Traola gehiago",
"mappedTags": "Esleitutako traolak",
- "rawTags": "Traola gordinak"
+ "rawTags": "Traola gordinak",
+ "missing": "Ez da aurkitu"
},
"actions": {
"addToQueue": "Erreproduzitu ondoren",
"playNow": "Erreproduzitu orain",
"addToPlaylist": "Gehitu erreprodukzio-zerrendara",
+ "showInPlaylist": "Erakutsi erreprodukzio-zerrendan",
"shuffleAll": "Erreprodukzio aleatorioa",
"download": "Deskargatu",
"playNext": "Hurrengoa",
- "info": "Lortu informazioa"
+ "info": "Erakutsi informazioa"
}
},
"album": {
@@ -61,7 +63,7 @@
"year": "Urtea",
"date": "Recording Date",
"originalDate": "Jatorrizkoa",
- "releaseDate": "Argitaratze-data:",
+ "releaseDate": "Argitaratze-data",
"releases": "Argitaratzea |||| Argitaratzeak",
"released": "Argitaratua",
"updatedAt": "Aktualizatze-data:",
@@ -73,21 +75,22 @@
"releaseType": "Mota",
"grouping": "Multzokatzea",
"media": "Multimedia",
- "mood": "Aldartea"
+ "mood": "Aldartea",
+ "missing": "Ez da aurkitu"
},
"actions": {
"playAll": "Erreproduzitu",
- "playNext": "Erreproduzitu segidan",
+ "playNext": "Erreproduzitu orain",
"addToQueue": "Erreproduzitu amaieran",
"shuffle": "Aletorioa",
"addToPlaylist": "Gehitu zerrendara",
"download": "Deskargatu",
- "info": "Lortu informazioa",
+ "info": "Erakutsi informazioa",
"share": "Partekatu"
},
"lists": {
"all": "Guztiak",
- "random": "Aleatorioki",
+ "random": "Aleatorioa",
"recentlyAdded": "Berriki gehitutakoak",
"recentlyPlayed": "Berriki entzundakoak",
"mostPlayed": "Gehien entzundakoak",
@@ -105,7 +108,8 @@
"playCount": "Erreprodukzio kopurua",
"rating": "Balorazioa",
"genre": "Generoa",
- "role": "Rola"
+ "role": "Rola",
+ "missing": "Ez da aurkitu"
},
"roles": {
"albumartist": "Albumeko egilea |||| Albumeko artistak",
@@ -120,7 +124,13 @@
"mixer": "Nahaslea |||| Nahasleak",
"remixer": "Remixerra |||| Remixerrak",
"djmixer": "DJ nahaslea |||| DJ nahasleak",
- "performer": "Interpretatzailea |||| Interpretatzaileak"
+ "performer": "Interpretatzailea |||| Interpretatzaileak",
+ "maincredit": "Albumeko egilea edo egilea |||| Albumeko egileak edo egileak"
+ },
+ "actions": {
+ "topSongs": "Abesti apartak",
+ "shuffle": "Aleatorioki",
+ "radio": "Irratia"
}
},
"user": {
@@ -192,12 +202,18 @@
"selectPlaylist": "Hautatu zerrenda:",
"addNewPlaylist": "Sortu \"%{name}\"",
"export": "Esportatu",
+ "saveQueue": "Gorde ilaran daudek erreprodukzio-zerrendan",
"makePublic": "Egin publikoa",
- "makePrivate": "Egin pribatua"
+ "makePrivate": "Egin pribatua",
+ "searchOrCreate": "Bilatu erreprodukzio-zerrenda edo idatzi berria sortzeko…",
+ "pressEnterToCreate": "Sakatu Enter erreprodukzio-zerrenda berria sortzeko",
+ "removeFromSelection": "Kendu hautaketatik"
},
"message": {
"duplicate_song": "Hautatutako abesti batzuk lehendik ere daude zerrendan",
- "song_exist": "Bikoiztutakoak gehitzen ari dira erreprodukzio-zerrendara. Ziur gehitu nahi dituzula?"
+ "song_exist": "Bikoiztutakoak gehitzen ari dira erreprodukzio-zerrendara. Ziur gehitu nahi dituzula?",
+ "noPlaylistsFound": "Ez da erreprodukzio-zerrenda aurkitu",
+ "noPlaylists": "Ez dago erreprodukzio-zerrendarik eskuragarri"
}
},
"radio": {
@@ -233,7 +249,7 @@
"actions": {}
},
"missing": {
- "name": "Fitxategia falta da|||| Fitxategiak falta dira",
+ "name": "Aurkitu ez den fitxategia |||| Aurkitu ez diren fitxategiak",
"empty": "Ez da fitxategirik falta",
"fields": {
"path": "Bidea",
@@ -242,10 +258,10 @@
},
"actions": {
"remove": "Kendu",
- "remove_all": "Kendu guztia"
+ "remove_all": "Kendu guztiak"
},
"notifications": {
- "removed": "Faltan zeuden fitxategiak kendu dira"
+ "removed": "Aurkitzen ez ziren fitxategiak kendu dira"
}
}
},
@@ -399,6 +415,8 @@
"transcodingDisabled": "Segurtasun arrazoiak direla-eta, transkodeketaren ezarpenak web-interfazearen bidez aldatzea ezgaituta dago. Transkodeketa-aukerak aldatu (editatu edo gehitu) nahi badituzu, berrabiarazi zerbitzaria konfigurazio-aukeraren %{config}-arekin.",
"transcodingEnabled": "Navidrome %{config}-ekin martxan dago eta, beraz, web-interfazeko transkodeketa-ataletik sistema-komandoak exekuta daitezke. Segurtasun arrazoiak tarteko, ezgaitzea gomendatzen dugu, eta transkodeketa-aukerak konfiguratzen ari zarenean bakarrik gaitzea.",
"songsAddedToPlaylist": "Abesti bat zerrendara gehitu da |||| %{smart_count} abesti zerrendara gehitu dira",
+ "noSimilarSongsFound": "Ez da antzeko abestirik aurkitu",
+ "noTopSongsFound": "Ez da aparteko abestirik aurkitu",
"noPlaylistsAvailable": "Ez dago zerrendarik erabilgarri",
"delete_user_title": "Ezabatu '%{name}' erabiltzailea",
"delete_user_content": "Ziur zaide erabiltzaile hau eta bere datu guztiak (zerrendak eta hobespenak barne) ezabatu nahi dituzula?",
@@ -480,8 +498,8 @@
"playModeText": {
"order": "Ordenean",
"orderLoop": "Errepikatu",
- "singleLoop": "Errepikatu bakarra",
- "shufflePlay": "Aleatorioa"
+ "singleLoop": "Errepikatu abesti hau",
+ "shufflePlay": "Aleatorioki"
}
},
"about": {
@@ -494,6 +512,21 @@
"disabled": "Ezgaituta",
"waiting": "Zain"
}
+ },
+ "tabs": {
+ "about": "Honi buruz",
+ "config": "Konfigurazioa"
+ },
+ "config": {
+ "configName": "Konfigurazioaren izena",
+ "environmentVariable": "Ingurune-aldagaia",
+ "currentValue": "Uneko balioa",
+ "configurationFile": "Konfigurazio-fitxategia",
+ "exportToml": "Esportatu konfigurazioa (TOML)",
+ "exportSuccess": "Konfigurazioa arbelera esportatu da TOML formatuan",
+ "exportFailed": "Konfigurazioa kopiatzeak huts egin du",
+ "devFlagsHeader": "Garapen-adierazleak (aldatu/kendu litezke)",
+ "devFlagsComment": "Ezarpen esperimentalak dira eta litekeena da etorkizunean desagertzea"
}
},
"activity": {
@@ -507,6 +540,11 @@
"status": "Errorea arakatzean",
"elapsedTime": "Igarotako denbora"
},
+ "nowPlaying": {
+ "title": "Une honetan erreproduzitzen",
+ "empty": "Ez dago erreproduzitzeko ezer",
+ "minutesAgo": "Duela minutu %{smart_count} |||| Duela %{smart_count} minutu"
+ },
"help": {
"title": "Navidromeren laster-teklak",
"hotkeys": {
diff --git a/resources/i18n/fi.json b/resources/i18n/fi.json
index 6c084a196..e5ecea2ce 100644
--- a/resources/i18n/fi.json
+++ b/resources/i18n/fi.json
@@ -32,7 +32,11 @@
"participants": "Lisäosallistujat",
"tags": "Lisätunnisteet",
"mappedTags": "Mäpättyt tunnisteet",
- "rawTags": "Raakatunnisteet"
+ "rawTags": "Raakatunnisteet",
+ "bitDepth": "Bittisyvyys",
+ "sampleRate": "Näytteenottotaajuus",
+ "missing": "Puuttuva",
+ "libraryName": "Kirjasto"
},
"actions": {
"addToQueue": "Lisää jonoon",
@@ -41,7 +45,8 @@
"shuffleAll": "Sekoita kaikki",
"download": "Lataa",
"playNext": "Soita seuraavaksi",
- "info": "Info"
+ "info": "Info",
+ "showInPlaylist": "Näytä soittolistassa"
}
},
"album": {
@@ -70,7 +75,10 @@
"releaseType": "Tyyppi",
"grouping": "Ryhmittely",
"media": "Media",
- "mood": "Tunnelma"
+ "mood": "Tunnelma",
+ "date": "Tallennuspäivä",
+ "missing": "Puuttuva",
+ "libraryName": "Kirjasto"
},
"actions": {
"playAll": "Soita",
@@ -102,7 +110,8 @@
"rating": "Arvostelu",
"genre": "Tyylilaji",
"size": "Koko",
- "role": "Rooli"
+ "role": "Rooli",
+ "missing": "Puuttuva"
},
"roles": {
"albumartist": "Albumitaiteilija |||| Albumitaiteilijat",
@@ -117,7 +126,13 @@
"mixer": "Miksaaja |||| Miksaajat",
"remixer": "Remiksaaja |||| Remiksaajat",
"djmixer": "DJ-miksaaja |||| DJ-miksaajat",
- "performer": "Esiintyjä |||| Esiintyjät"
+ "performer": "Esiintyjä |||| Esiintyjät",
+ "maincredit": "Albumin artisti tai artisti |||| Albumin artistit tai artistit"
+ },
+ "actions": {
+ "shuffle": "Sekoita",
+ "radio": "Radio",
+ "topSongs": "Suosituimmat kappaleet"
}
},
"user": {
@@ -134,10 +149,12 @@
"currentPassword": "Nykyinen salasana",
"newPassword": "Uusi salasana",
"token": "Avain",
- "lastAccessAt": "Viimeisin käyttö"
+ "lastAccessAt": "Viimeisin käyttö",
+ "libraries": "Kirjastot"
},
"helperTexts": {
- "name": "Nimen muutos tulee voimaan kun seuraavan kerran kirjaudut sisään"
+ "name": "Nimen muutos tulee voimaan kun seuraavan kerran kirjaudut sisään",
+ "libraries": "Valitse tietyt kirjastot tälle käyttäjälle tai jätä tyhjäksi käyttääksesi oletuskirjastoja"
},
"notifications": {
"created": "Käyttäjä luotu",
@@ -146,7 +163,12 @@
},
"message": {
"listenBrainzToken": "Syötä ListenBrainz avain.",
- "clickHereForToken": "Paina tästä saadaksesi avaimen"
+ "clickHereForToken": "Paina tästä saadaksesi avaimen",
+ "selectAllLibraries": "Valitse kaikki kirjastot",
+ "adminAutoLibraries": "Admin-käyttäjillä on automaattisesti pääsy kaikkiin kirjastoihin"
+ },
+ "validation": {
+ "librariesRequired": "Vähintään yksi kirjasto on valittava ei-admin käyttäjille"
}
},
"player": {
@@ -190,11 +212,17 @@
"addNewPlaylist": "Luo \"%{name}\"",
"export": "Vie",
"makePublic": "Tee julkinen",
- "makePrivate": "Tee yksityinen"
+ "makePrivate": "Tee yksityinen",
+ "saveQueue": "Tallenna jono soittolistaan",
+ "searchOrCreate": "Etsi soittolistoja tai kirjoita luodaksesi uuden...",
+ "pressEnterToCreate": "Paina Enter luodaksesi uuden soittolistan",
+ "removeFromSelection": "Poista valinnasta"
},
"message": {
"duplicate_song": "Lisää olemassa oleva kappale",
- "song_exist": "Olet lisäämässä soittolistalla jo olevaa kappaletta. Haluatko lisätä saman kappaleen vai ohittaa sen?"
+ "song_exist": "Olet lisäämässä soittolistalla jo olevaa kappaletta. Haluatko lisätä saman kappaleen vai ohittaa sen?",
+ "noPlaylistsFound": "Soittolistoja ei löytynyt",
+ "noPlaylists": "Soittolistoja ei ole saatavilla"
}
},
"radio": {
@@ -232,13 +260,68 @@
"fields": {
"path": "Polku",
"size": "Koko",
- "updatedAt": "Katosi"
+ "updatedAt": "Katosi",
+ "libraryName": "Kirjasto"
},
"actions": {
- "remove": "Poista"
+ "remove": "Poista",
+ "remove_all": "Poista kaikki"
},
"notifications": {
"removed": "Puuttuvat tiedostot poistettu"
+ },
+ "empty": "Ei puuttuvia tiedostoja"
+ },
+ "library": {
+ "name": "Kirjasto |||| Kirjastot",
+ "fields": {
+ "name": "Nimi",
+ "path": "Polku",
+ "remotePath": "Etäpolku",
+ "lastScanAt": "Viimeisin skannaus",
+ "songCount": "Kappaleet",
+ "albumCount": "Albumit",
+ "artistCount": "Artistit",
+ "totalSongs": "Kappaleet",
+ "totalAlbums": "Albumit",
+ "totalArtists": "Artistit",
+ "totalFolders": "Kansiot",
+ "totalFiles": "Tiedostot",
+ "totalMissingFiles": "Puuttuvat tiedostot",
+ "totalSize": "Kokonaiskoko",
+ "totalDuration": "Kesto",
+ "defaultNewUsers": "Oletus uusille käyttäjille",
+ "createdAt": "Luotu",
+ "updatedAt": "Päivitetty"
+ },
+ "sections": {
+ "basic": "Perustiedot",
+ "statistics": "Tilastot"
+ },
+ "actions": {
+ "scan": "Skannaa kirjasto",
+ "manageUsers": "Hallitse käyttäjien pääsyä",
+ "viewDetails": "Näytä tiedot"
+ },
+ "notifications": {
+ "created": "Kirjasto luotu onnistuneesti",
+ "updated": "Kirjasto päivitetty onnistuneesti",
+ "deleted": "Kirjasto poistettu onnistuneesti",
+ "scanStarted": "Kirjaston skannaus aloitettu",
+ "scanCompleted": "Kirjaston skannaus valmistunut"
+ },
+ "validation": {
+ "nameRequired": "Kirjaston nimi vaaditaan",
+ "pathRequired": "Kirjaston polku vaaditaan",
+ "pathNotDirectory": "Kirjaston polun tulee olla hakemisto",
+ "pathNotFound": "Kirjaston polkua ei löytynyt",
+ "pathNotAccessible": "Kirjaston polku ei ole käytettävissä",
+ "pathInvalid": "Virheellinen kirjaston polku"
+ },
+ "messages": {
+ "deleteConfirm": "Oletko varma, että haluat poistaa tämän kirjaston? Tämä poistaa kaikki liittyvät tiedot ja käyttäjien pääsyn.",
+ "scanInProgress": "Skannaus käynnissä...",
+ "noLibrariesAssigned": "Tälle käyttäjälle ei ole määritetty kirjastoja"
}
}
},
@@ -419,7 +502,11 @@
"downloadDialogTitle": "Lataa %{resource} '%{name}' (%{size})",
"shareCopyToClipboard": "Kopio leikepöydälle: Ctrl+C, Enter",
"remove_missing_title": "Poista puuttuvat tiedostot",
- "remove_missing_content": "Oletko varma, että haluat poistaa valitut puuttuvat tiedostot tietokannasta? Tämä poistaa pysyvästi kaikki viittaukset niihin, mukaan lukien niiden soittojen määrät ja arvostelut."
+ "remove_missing_content": "Oletko varma, että haluat poistaa valitut puuttuvat tiedostot tietokannasta? Tämä poistaa pysyvästi kaikki viittaukset niihin, mukaan lukien niiden soittojen määrät ja arvostelut.",
+ "remove_all_missing_title": "Poista kaikki puuttuvat tiedostot",
+ "remove_all_missing_content": "Haluatko varmasti poistaa kaikki puuttuvat tiedostot tietokannasta? Tämä poistaa pysyvästi kaikki viittaukset niihin, mukaan lukien toistomäärät ja arvostelut.",
+ "noSimilarSongsFound": "Samankaltaisia kappaleita ei löytynyt",
+ "noTopSongsFound": "Suosituimpia kappaleita ei löytynyt"
},
"menu": {
"library": "Kirjasto",
@@ -448,7 +535,13 @@
"albumList": "Albumit",
"about": "Tietoa",
"playlists": "Soittolista",
- "sharedPlaylists": "Jaettu soittolista"
+ "sharedPlaylists": "Jaettu soittolista",
+ "librarySelector": {
+ "allLibraries": "Kaikki kirjastot (%{count})",
+ "multipleLibraries": "%{selected} / %{total} kirjastoa",
+ "selectLibraries": "Valitse kirjastot",
+ "none": "Ei mitään"
+ }
},
"player": {
"playListsText": "Jono",
@@ -485,6 +578,21 @@
"disabled": "Ei käytössä",
"waiting": "Odottaa"
}
+ },
+ "tabs": {
+ "about": "Tietoja",
+ "config": "Kokoonpano"
+ },
+ "config": {
+ "configName": "Konfiguraation nimi",
+ "environmentVariable": "Ympäristömuuttuja",
+ "currentValue": "Nykyinen arvo",
+ "configurationFile": "Konfiguraatiotiedosto",
+ "exportToml": "Vie konfiguraatio (TOML)",
+ "exportSuccess": "Konfiguraatio viety leikepöydälle TOML-muodossa",
+ "exportFailed": "Konfiguraation kopiointi epäonnistui",
+ "devFlagsHeader": "Kehitysliput (voivat muuttua/poistua)",
+ "devFlagsComment": "Nämä ovat kokeellisia asetuksia ja ne voidaan poistaa tulevissa versioissa"
}
},
"activity": {
@@ -493,7 +601,10 @@
"quickScan": "Nopea tarkistus",
"fullScan": "Täysi tarkistus",
"serverUptime": "Palvelun käyttöaika",
- "serverDown": "SAMMUTETTU"
+ "serverDown": "SAMMUTETTU",
+ "scanType": "Tyyppi",
+ "status": "Skannausvirhe",
+ "elapsedTime": "Kulunut aika"
},
"help": {
"title": "Navidrome pikapainikkeet",
@@ -508,5 +619,10 @@
"toggle_love": "Lisää kappale suosikkeihin",
"current_song": "Siirry nykyiseen kappaleeseen"
}
+ },
+ "nowPlaying": {
+ "title": "Nyt soi",
+ "empty": "Ei soita mitään",
+ "minutesAgo": "%{smart_count} minuutti sitten |||| %{smart_count} minuuttia sitten"
}
}
\ No newline at end of file
diff --git a/resources/i18n/fr.json b/resources/i18n/fr.json
index 4137f5b26..af3a8dd31 100644
--- a/resources/i18n/fr.json
+++ b/resources/i18n/fr.json
@@ -2,7 +2,7 @@
"languageName": "Français",
"resources": {
"song": {
- "name": "Piste |||| Pistes",
+ "name": "Titre |||| Titres",
"fields": {
"albumArtist": "Artiste",
"duration": "Durée",
@@ -11,14 +11,13 @@
"title": "Titre",
"artist": "Artiste",
"album": "Album",
- "path": "Chemin",
+ "path": "Chemin d'accès",
"genre": "Genre",
"compilation": "Compilation",
"year": "Année",
"size": "Taille",
"updatedAt": "Mise à jour",
"bitRate": "Bitrate",
- "sampleRate": "Fréquence d'échantillonnage",
"discSubtitle": "Sous-titre du disque",
"starred": "Favoris",
"comment": "Commentaire",
@@ -34,7 +33,10 @@
"tags": "Étiquettes supplémentaires",
"mappedTags": "Étiquettes correspondantes",
"rawTags": "Étiquettes brutes",
- "bitDepth": "Profondeur de bit"
+ "bitDepth": "Profondeur de bits",
+ "sampleRate": "Fréquence d'échantillonnage",
+ "missing": "Manquant",
+ "libraryName": "Bibliothèque"
},
"actions": {
"addToQueue": "Ajouter à la file",
@@ -43,7 +45,8 @@
"shuffleAll": "Tout mélanger",
"download": "Télécharger",
"playNext": "Jouer ensuite",
- "info": "Plus d'informations"
+ "info": "Plus d'informations",
+ "showInPlaylist": "Montrer dans la playlist"
}
},
"album": {
@@ -52,13 +55,12 @@
"albumArtist": "Artiste",
"artist": "Artiste",
"duration": "Durée",
- "songCount": "Nombre de pistes",
+ "songCount": "Titres",
"playCount": "Nombre d'écoutes",
"name": "Nom",
"genre": "Genre",
"compilation": "Compilation",
"year": "Année",
- "date": "Date d'enregistrement",
"updatedAt": "Mis à jour le",
"comment": "Commentaire",
"rating": "Classement",
@@ -73,7 +75,10 @@
"releaseType": "Type",
"grouping": "Regroupement",
"media": "Média",
- "mood": "Humeur"
+ "mood": "Humeur",
+ "date": "Date d'enregistrement",
+ "missing": "Manquant",
+ "libraryName": "Bibliothèque"
},
"actions": {
"playAll": "Lire",
@@ -100,12 +105,13 @@
"fields": {
"name": "Nom",
"albumCount": "Nombre d'albums",
- "songCount": "Nombre de pistes",
+ "songCount": "Nombre de titres",
"playCount": "Lectures",
"rating": "Classement",
"genre": "Genre",
"size": "Taille",
- "role": "Rôle"
+ "role": "Rôle",
+ "missing": "Manquant"
},
"roles": {
"albumartist": "Artiste de l'album |||| Artistes de l'album",
@@ -120,7 +126,13 @@
"mixer": "Mixeur |||| Mixeurs",
"remixer": "Remixeur |||| Remixeurs",
"djmixer": "Mixeur DJ |||| Mixeurs DJ",
- "performer": "Interprète |||| Interprètes"
+ "performer": "Interprète |||| Interprètes",
+ "maincredit": "Artiste de l'album ou Artiste |||| Artistes de l'album ou Artistes"
+ },
+ "actions": {
+ "shuffle": "Lecture aléatoire",
+ "radio": "Radio",
+ "topSongs": "Meilleurs titres"
}
},
"user": {
@@ -137,10 +149,12 @@
"currentPassword": "Mot de passe actuel",
"newPassword": "Nouveau mot de passe",
"token": "Token",
- "lastAccessAt": "Dernier accès"
+ "lastAccessAt": "Dernier accès",
+ "libraries": "Bibliothèques"
},
"helperTexts": {
- "name": "Les changements liés à votre nom ne seront reflétés qu'à la prochaine connexion"
+ "name": "Les changements liés à votre nom ne seront reflétés qu'à la prochaine connexion",
+ "libraries": "Sélectionner une bibliothèque pour cet utilisateur ou laisser vide pour utiliser la bibliothèque par défaut"
},
"notifications": {
"created": "Utilisateur créé",
@@ -149,7 +163,12 @@
},
"message": {
"listenBrainzToken": "Entrez votre token ListenBrainz.",
- "clickHereForToken": "Cliquez ici pour recevoir votre token"
+ "clickHereForToken": "Cliquez ici pour recevoir votre token",
+ "selectAllLibraries": "Sélectionner toutes les bibliothèques",
+ "adminAutoLibraries": "Les utilisateurs admin ont automatiquement accès à l'ensemble des bibliothèques"
+ },
+ "validation": {
+ "librariesRequired": "Au moins une bibliothèque doit être sélectionnée pour les utilisateurs non administrateurs"
}
},
"player": {
@@ -161,7 +180,7 @@
"client": "Client",
"userName": "Nom d'utilisateur",
"lastSeen": "Vu pour la dernière fois",
- "reportRealPath": "Rapporter le chemin absolu",
+ "reportRealPath": "Rapporter le chemin d'accès absolu",
"scrobbleEnabled": "Scrobbler vers des services externes"
}
},
@@ -189,15 +208,21 @@
"path": "Importer depuis"
},
"actions": {
- "selectPlaylist": "Ajouter les pistes à la playlist",
+ "selectPlaylist": "Sélectionner une playlist :",
"addNewPlaylist": "Créer \"%{name}\"",
"export": "Exporter",
"makePublic": "Rendre publique",
- "makePrivate": "Rendre privée"
+ "makePrivate": "Rendre privée",
+ "saveQueue": "Sauvegarder la file de lecture dans la playlist",
+ "searchOrCreate": "Chercher ou créer une nouvelle playlist...",
+ "pressEnterToCreate": "Appuyer sur entrée pour créer une nouvelle playlist",
+ "removeFromSelection": "Supprimer de la sélection"
},
"message": {
- "duplicate_song": "Pistes déjà présentes dans la playlist",
- "song_exist": "Certaines des pistes sélectionnées font déjà partie de la playlist. Voulez-vous les ajouter ou les ignorer ?"
+ "duplicate_song": "Ajouter les titres déjà présents dans la playlist",
+ "song_exist": "Certains des titres sélectionnés font déjà partie de la playlist. Voulez-vous les ajouter ou les ignorer ?",
+ "noPlaylistsFound": "Aucune playlist trouvée",
+ "noPlaylists": "Aucune playlist disponible"
}
},
"radio": {
@@ -233,17 +258,71 @@
"missing": {
"name": "Fichier manquant|||| Fichiers manquants",
"fields": {
- "path": "Chemin",
+ "path": "Chemin d'accès",
"size": "Taille",
- "updatedAt": "A disparu le"
+ "updatedAt": "A disparu le",
+ "libraryName": "Bibliothèque"
},
"actions": {
- "remove": "Supprimer"
+ "remove": "Supprimer",
+ "remove_all": "Tout supprimer"
},
"notifications": {
"removed": "Fichier(s) manquant(s) supprimé(s)"
},
"empty": "Aucun fichier manquant"
+ },
+ "library": {
+ "name": "Bibliothèque |||| Bibliothèques",
+ "fields": {
+ "name": "Nom",
+ "path": "Chemin d'accès",
+ "remotePath": "Chemin d'accès distant",
+ "lastScanAt": "Dernier scan",
+ "songCount": "Titres",
+ "albumCount": "Albums",
+ "artistCount": "Artistes",
+ "totalSongs": "Titres",
+ "totalAlbums": "Albums",
+ "totalArtists": "Artistes",
+ "totalFolders": "Dossiers",
+ "totalFiles": "Fichiers",
+ "totalMissingFiles": "Fichiers manquants",
+ "totalSize": "Taille totale",
+ "totalDuration": "Durée",
+ "defaultNewUsers": "Défaut pour les nouveaux utilisateurs",
+ "createdAt": "Crée",
+ "updatedAt": "Mise à jour"
+ },
+ "sections": {
+ "basic": "Informations",
+ "statistics": "Statistiques"
+ },
+ "actions": {
+ "scan": "Scanner la bibliothèque",
+ "manageUsers": "Gérer les accès utilisateurs",
+ "viewDetails": "Voir les détails"
+ },
+ "notifications": {
+ "created": "Bibliothèque créée avec succès",
+ "updated": "Bibliothèque mise à jour avec succès",
+ "deleted": "Bibliothèque supprimée avec succès",
+ "scanStarted": "Le scan de la bibliothèque a commencé",
+ "scanCompleted": "Le scan de la bibliothèque est terminé"
+ },
+ "validation": {
+ "nameRequired": "La bibliothèque doit obligatoirement avoir un nom",
+ "pathRequired": "La bibliothèque doit obligatoirement avoir un chemin d'accès",
+ "pathNotDirectory": "Le chemin d'accès de la bibliothèque doit pointer sur un dossier",
+ "pathNotFound": "Impossible de trouver ce chemin d'accès",
+ "pathNotAccessible": "Impossible d'accéder à ce chemin d'accès",
+ "pathInvalid": "Ce chemin d'accès n'est pas valide"
+ },
+ "messages": {
+ "deleteConfirm": "Êtes-vous sûr(e) de vouloir supprimer cette bibliothèque ? Cela supprimera toutes les données associées ainsi que les accès utilisateurs.",
+ "scanInProgress": "Scan en cours...",
+ "noLibrariesAssigned": "Aucune bibliothèque pour cet utilisateur"
+ }
}
},
"ra": {
@@ -395,7 +474,7 @@
"note": "NOTE",
"transcodingDisabled": "Le changement de paramètres depuis l'interface web est désactivé pour des raisons de sécurité. Pour changer (éditer ou supprimer) les options de transcodage, relancer le serveur avec l'option %{config} activée.",
"transcodingEnabled": "Navidrome fonctionne actuellement avec %{config}, rendant possible l’exécution de commandes arbitraires depuis l'interface web. Il est recommandé d'activer cette fonctionnalité uniquement lors de la configuration du transcodage.",
- "songsAddedToPlaylist": "Une piste a été ajoutée à la playlist |||| %{smart_count} pistes ont été ajoutées à la playlist",
+ "songsAddedToPlaylist": "1 titre a été ajouté à la playlist |||| %{smart_count} titres ont été ajoutés à la playlist",
"noPlaylistsAvailable": "Aucune playlist",
"delete_user_title": "Supprimer l'utilisateur '%{name}'",
"delete_user_content": "Êtes-vous sûr(e) de vouloir supprimer cet utilisateur et ses données associées (y compris ses playlists et préférences) ?",
@@ -423,7 +502,11 @@
"downloadDialogTitle": "Télécharger %{resource} '%{name}' (%{size})",
"shareCopyToClipboard": "Copier vers le presse-papier : Ctrl+C, Enter",
"remove_missing_title": "Supprimer les fichiers manquants",
- "remove_missing_content": "Êtes-vous sûr(e) de vouloir supprimer les fichiers manquants sélectionnés de la base de données ? Ceci supprimera définitivement toute référence à ceux-ci, y compris leurs nombres d'écoutes et leurs notations"
+ "remove_missing_content": "Êtes-vous sûr(e) de vouloir supprimer les fichiers manquants sélectionnés de la base de données ? Ceci supprimera définitivement toute référence à ceux-ci, y compris leurs nombres d'écoutes et leurs notations",
+ "remove_all_missing_title": "Supprimer tous les fichiers manquants",
+ "remove_all_missing_content": "Êtes-vous sûr(e) de vouloir supprimer tous les fichiers manquants de la base de données ? Cette action est permanente et supprimera leurs nombres d'écoutes, leur notations et tout ce qui y fait référence.",
+ "noSimilarSongsFound": "Aucun titre similaire n'a été trouvé",
+ "noTopSongsFound": "Aucun meilleur titre n'a été trouvé"
},
"menu": {
"library": "Bibliothèque",
@@ -444,7 +527,7 @@
"gain": {
"none": "Désactivé",
"album": "Utiliser le gain de l'album",
- "track": "Utiliser le gain des pistes"
+ "track": "Utiliser le gain des titres"
},
"lastfmNotConfigured": "La clef API de Last.fm n'est pas configurée"
}
@@ -452,7 +535,13 @@
"albumList": "Albums",
"about": "À propos",
"playlists": "Playlists",
- "sharedPlaylists": "Playlists partagées"
+ "sharedPlaylists": "Playlists partagées",
+ "librarySelector": {
+ "allLibraries": "Toutes les bibliothèques (%{count})",
+ "multipleLibraries": "%{selected} bibliothèque(s) sélectionnée(s) sur %{total}",
+ "selectLibraries": "Sélectionner les bibliothèques",
+ "none": "Aucune"
+ }
},
"player": {
"playListsText": "File de lecture",
@@ -489,6 +578,21 @@
"disabled": "Désactivée",
"waiting": "En attente"
}
+ },
+ "tabs": {
+ "about": "À propos",
+ "config": "Paramètres"
+ },
+ "config": {
+ "configName": "Nom de la configuration",
+ "environmentVariable": "Variable d'environnement",
+ "currentValue": "Valeur actuelle",
+ "configurationFile": "Fichier de configuration",
+ "exportToml": "Exporter la configuration (TOML)",
+ "exportSuccess": "La configuration a été copiée vers le presse-papier au format TOML",
+ "exportFailed": "Une erreur est survenue en copiant la configuration",
+ "devFlagsHeader": "Options de développement (peuvent être amenés à changer / être supprimés)",
+ "devFlagsComment": "Ces paramètres sont expérimentaux et peuvent être amenés à changer dans le futur"
}
},
"activity": {
@@ -497,7 +601,10 @@
"quickScan": "Scan rapide",
"fullScan": "Scan complet",
"serverUptime": "Disponibilité du serveur",
- "serverDown": "HORS LIGNE"
+ "serverDown": "HORS LIGNE",
+ "scanType": "Type",
+ "status": "Erreur de scan",
+ "elapsedTime": "Temps écoulé"
},
"help": {
"title": "Raccourcis Navidrome",
@@ -510,7 +617,12 @@
"vol_up": "Augmenter le volume",
"vol_down": "Baisser le volume",
"toggle_love": "Ajouter/Enlever le morceau des favoris",
- "current_song": "Aller à la chanson en cours"
+ "current_song": "Aller au titre en cours"
}
+ },
+ "nowPlaying": {
+ "title": "En cours de lecture",
+ "empty": "Aucun titre en cours de lecture",
+ "minutesAgo": "Il y a %{smart_count} minute |||| Il y a %{smart_count} minutes"
}
}
\ No newline at end of file
diff --git a/resources/i18n/gl.json b/resources/i18n/gl.json
index 4d9a1a9a0..a6c3beb05 100644
--- a/resources/i18n/gl.json
+++ b/resources/i18n/gl.json
@@ -31,8 +31,12 @@
"mood": "Estado",
"participants": "Participantes adicionais",
"tags": "Etiquetas adicionais",
- "mappedTags": "",
- "rawTags": "Etiquetas en cru"
+ "mappedTags": "Etiquetas mapeadas",
+ "rawTags": "Etiquetas en cru",
+ "bitDepth": "Calidade de Bit",
+ "sampleRate": "Taxa de mostra",
+ "missing": "Falta",
+ "libraryName": "Biblioteca"
},
"actions": {
"addToQueue": "Ao final da cola",
@@ -41,7 +45,8 @@
"shuffleAll": "Remexer todo",
"download": "Descargar",
"playNext": "A continuación",
- "info": "Obter info"
+ "info": "Obter info",
+ "showInPlaylist": "Mostrar en Lista de reprodución"
}
},
"album": {
@@ -63,14 +68,17 @@
"size": "Tamaño",
"originalDate": "Orixinal",
"releaseDate": "Publicado",
- "releases": "Publicación ||| Publicacións",
+ "releases": "Publicación |||| Publicacións",
"released": "Publicado",
"recordLabel": "Editorial",
"catalogNum": "Número de catálogo",
"releaseType": "Tipo",
"grouping": "Grupos",
"media": "Multimedia",
- "mood": "Estado"
+ "mood": "Estado",
+ "date": "Data de gravación",
+ "missing": "Falta",
+ "libraryName": "Biblioteca"
},
"actions": {
"playAll": "Reproducir",
@@ -102,7 +110,8 @@
"rating": "Valoración",
"genre": "Xénero",
"size": "Tamaño",
- "role": "Rol"
+ "role": "Rol",
+ "missing": "Falta"
},
"roles": {
"albumartist": "Artista do álbum |||| Artistas do álbum",
@@ -117,7 +126,13 @@
"mixer": "Mistura |||| Mistura",
"remixer": "Remezcla |||| Remezcla",
"djmixer": "Mezcla DJs |||| Mezcla DJs",
- "performer": "Intérprete |||| Intérpretes"
+ "performer": "Intérprete |||| Intérpretes",
+ "maincredit": "Artista do álbum ou Artista |||| Artistas do álbum ou Artistas"
+ },
+ "actions": {
+ "shuffle": "Barallar",
+ "radio": "Radio",
+ "topSongs": "Cancións destacadas"
}
},
"user": {
@@ -134,10 +149,12 @@
"currentPassword": "Contrasinal actual",
"newPassword": "Novo contrasinal",
"token": "Token",
- "lastAccessAt": "Último acceso"
+ "lastAccessAt": "Último acceso",
+ "libraries": "Bibliotecas"
},
"helperTexts": {
- "name": "Os cambios no nome aplicaranse a próxima vez que accedas"
+ "name": "Os cambios no nome aplicaranse a próxima vez que accedas",
+ "libraries": "Selecciona bibliotecas específicas para esta usuaria, ou deixa baleiro para usar as bibliotecas por defecto"
},
"notifications": {
"created": "Creouse a usuaria",
@@ -146,7 +163,12 @@
},
"message": {
"listenBrainzToken": "Escribe o token de usuaria de ListenBrainz",
- "clickHereForToken": "Preme aquí para obter o token"
+ "clickHereForToken": "Preme aquí para obter o token",
+ "selectAllLibraries": "Seleccionar todas as bibliotecas",
+ "adminAutoLibraries": "As usuarias Admin teñen acceso por defecto a todas as bibliotecas"
+ },
+ "validation": {
+ "librariesRequired": "Debes seleccionar polo menos unha biblioteca para usuarias non admins"
}
},
"player": {
@@ -190,11 +212,17 @@
"addNewPlaylist": "Crear \"%{name}\"",
"export": "Exportar",
"makePublic": "Facela Pública",
- "makePrivate": "Facela Privada"
+ "makePrivate": "Facela Privada",
+ "saveQueue": "Salvar a Cola como Lista de reprodución",
+ "searchOrCreate": "Buscar listas ou escribe para crear nova…",
+ "pressEnterToCreate": "Preme Enter para crear nova lista",
+ "removeFromSelection": "Retirar da selección"
},
"message": {
"duplicate_song": "Engadir cancións duplicadas",
- "song_exist": "Hai duplicadas que serán engadidas á lista de reprodución. Desexas engadir as duplicadas ou omitilas?"
+ "song_exist": "Hai duplicadas que serán engadidas á lista de reprodución. Desexas engadir as duplicadas ou omitilas?",
+ "noPlaylistsFound": "Sen listas de reprodución",
+ "noPlaylists": "Sen listas dispoñibles"
}
},
"radio": {
@@ -232,13 +260,68 @@
"fields": {
"path": "Ruta",
"size": "Tamaño",
- "updatedAt": "Desapareceu o"
+ "updatedAt": "Desapareceu o",
+ "libraryName": "Biblioteca"
},
"actions": {
- "remove": "Retirar"
+ "remove": "Retirar",
+ "remove_all": "Retirar todo"
},
"notifications": {
"removed": "Ficheiro(s) faltantes retirados"
+ },
+ "empty": "Sen ficheiros faltantes"
+ },
+ "library": {
+ "name": "Biblioteca |||| Bibliotecas",
+ "fields": {
+ "name": "Nome",
+ "path": "Ruta",
+ "remotePath": "Ruta remota",
+ "lastScanAt": "Último escaneado",
+ "songCount": "Cancións",
+ "albumCount": "Álbums",
+ "artistCount": "Artistas",
+ "totalSongs": "Cancións",
+ "totalAlbums": "Álbums",
+ "totalArtists": "Artistas",
+ "totalFolders": "Cartafoles",
+ "totalFiles": "Ficheiros",
+ "totalMissingFiles": "Ficheiros que faltan",
+ "totalSize": "Tamaño total",
+ "totalDuration": "Duración",
+ "defaultNewUsers": "Por defecto para novas usuarias",
+ "createdAt": "Creada",
+ "updatedAt": "Actualizada"
+ },
+ "sections": {
+ "basic": "Información básica",
+ "statistics": "Estatísticas"
+ },
+ "actions": {
+ "scan": "Escanear Biblioteca",
+ "manageUsers": "Xestionar acceso das usuarias",
+ "viewDetails": "Ver detalles"
+ },
+ "notifications": {
+ "created": "Biblioteca creada correctamente",
+ "updated": "Biblioteca actualizada correctamente",
+ "deleted": "Biblioteca eliminada correctamente",
+ "scanStarted": "Comezou o escaneo da biblioteca",
+ "scanCompleted": "Completouse o escaneado da biblioteca"
+ },
+ "validation": {
+ "nameRequired": "Requírese un nome para a biblioteca",
+ "pathRequired": "Requírese unha ruta para a biblioteca",
+ "pathNotDirectory": "A ruta á biblioteca ten que ser un directorio",
+ "pathNotFound": "Non se atopa a ruta á biblioteca",
+ "pathNotAccessible": "A ruta á biblioteca non é accesible",
+ "pathInvalid": "Ruta non válida á biblioteca"
+ },
+ "messages": {
+ "deleteConfirm": "Tes certeza de querer eliminar esta biblioteca? Isto eliminará todos os datos asociados e accesos de usuarias.",
+ "scanInProgress": "Escaneo en progreso…",
+ "noLibrariesAssigned": "Sen bibliotecas asignadas a esta usuaria"
}
}
},
@@ -419,7 +502,11 @@
"downloadDialogTitle": "Descargar %{resource} '%{name}' (%{size})",
"shareCopyToClipboard": "Copiar ao portapapeis: Ctrl+C, Enter",
"remove_missing_title": "Retirar ficheiros que faltan",
- "remove_missing_content": "Tes certeza de querer retirar da base de datos os ficheiros que faltan? Isto retirará de xeito permanente todas a referencias a eles, incluíndo a conta de reproducións e valoracións."
+ "remove_missing_content": "Tes certeza de querer retirar da base de datos os ficheiros que faltan? Isto retirará de xeito permanente todas a referencias a eles, incluíndo a conta de reproducións e valoracións.",
+ "remove_all_missing_title": "Retirar todos os ficheiros que faltan",
+ "remove_all_missing_content": "Tes certeza de querer retirar da base de datos todos os ficheiros que faltan? Isto eliminará todas as referencias a eles, incluíndo o número de reproducións e valoracións.",
+ "noSimilarSongsFound": "Sen cancións parecidas",
+ "noTopSongsFound": "Sen cancións destacadas"
},
"menu": {
"library": "Biblioteca",
@@ -448,7 +535,13 @@
"albumList": "Álbums",
"about": "Acerca de",
"playlists": "Listas de reprodución",
- "sharedPlaylists": "Listas compartidas"
+ "sharedPlaylists": "Listas compartidas",
+ "librarySelector": {
+ "allLibraries": "Todas as bibliotecas (%{count})",
+ "multipleLibraries": "%{selected} de %{total} Bibliotecas",
+ "selectLibraries": "Seleccionar Bibliotecas",
+ "none": "Ningunha"
+ }
},
"player": {
"playListsText": "Reproducir cola",
@@ -485,6 +578,21 @@
"disabled": "Desactivado",
"waiting": "Agardando"
}
+ },
+ "tabs": {
+ "about": "Sobre",
+ "config": "Configuración"
+ },
+ "config": {
+ "configName": "Nome",
+ "environmentVariable": "Variable de entorno",
+ "currentValue": "Valor actual",
+ "configurationFile": "Ficheiro de configuración",
+ "exportToml": "Exportar configuración (TOML)",
+ "exportSuccess": "Configuración exportada ao portapapeis no formato TOML",
+ "exportFailed": "Fallou a copia da configuración",
+ "devFlagsHeader": "Configuracións de Desenvolvemento (suxeitas a cambio/retirada)",
+ "devFlagsComment": "Son axustes experimentais e poden retirarse en futuras versións"
}
},
"activity": {
@@ -493,7 +601,10 @@
"quickScan": "Escaneo rápido",
"fullScan": "Escaneo completo",
"serverUptime": "Servidor a funcionar",
- "serverDown": "SEN CONEXIÓN"
+ "serverDown": "SEN CONEXIÓN",
+ "scanType": "Tipo",
+ "status": "Erro de escaneado",
+ "elapsedTime": "Tempo transcurrido"
},
"help": {
"title": "Atallos de Navidrome",
@@ -508,5 +619,10 @@
"toggle_love": "Engadir canción a favoritas",
"current_song": "Ir á Canción actual "
}
+ },
+ "nowPlaying": {
+ "title": "En reprodución",
+ "empty": "Sen reprodución",
+ "minutesAgo": "hai %{smart_count} minuto |||| hai %{smart_count} minutos"
}
}
\ No newline at end of file
diff --git a/resources/i18n/hi.json b/resources/i18n/hi.json
new file mode 100644
index 000000000..5b9ece530
--- /dev/null
+++ b/resources/i18n/hi.json
@@ -0,0 +1,630 @@
+{
+ "languageName": "हिंदी",
+ "resources": {
+ "song": {
+ "name": "गाना |||| गाने",
+ "fields": {
+ "albumArtist": "एल्बम कलाकार",
+ "duration": "समय",
+ "trackNumber": "#",
+ "playCount": "प्ले संख्या",
+ "title": "शीर्षक",
+ "artist": "कलाकार",
+ "album": "एल्बम",
+ "path": "फ़ाइल पथ",
+ "libraryName": "लाइब्रेरी",
+ "genre": "शैली",
+ "compilation": "संकलन",
+ "year": "वर्ष",
+ "size": "फ़ाइल का आकार",
+ "updatedAt": "अपडेट किया गया",
+ "bitRate": "बिट रेट",
+ "bitDepth": "बिट गहराई",
+ "sampleRate": "सैंपल रेट",
+ "channels": "चैनल",
+ "discSubtitle": "डिस्क उपशीर्षक",
+ "starred": "पसंदीदा",
+ "comment": "टिप्पणी",
+ "rating": "रेटिंग",
+ "quality": "गुणवत्ता",
+ "bpm": "BPM",
+ "playDate": "अंतिम बार चलाया गया",
+ "createdAt": "जोड़ने की तारीख",
+ "grouping": "समूहीकरण",
+ "mood": "मूड",
+ "participants": "अतिरिक्त प्रतिभागी",
+ "tags": "अतिरिक्त टैग",
+ "mappedTags": "मैप किए गए टैग",
+ "rawTags": "रॉ टैग",
+ "missing": "गुम"
+ },
+ "actions": {
+ "addToQueue": "बाद में चलाएं",
+ "playNow": "अभी चलाएं",
+ "addToPlaylist": "प्लेलिस्ट में जोड़ें",
+ "showInPlaylist": "प्लेलिस्ट में दिखाएं",
+ "shuffleAll": "सभी को शफल करें",
+ "download": "डाउनलोड",
+ "playNext": "अगला चलाएं",
+ "info": "जानकारी प्राप्त करें"
+ }
+ },
+ "album": {
+ "name": "एल्बम |||| एल्बम",
+ "fields": {
+ "albumArtist": "एल्बम कलाकार",
+ "artist": "कलाकार",
+ "duration": "समय",
+ "songCount": "गाने",
+ "playCount": "प्ले संख्या",
+ "size": "आकार",
+ "name": "नाम",
+ "libraryName": "लाइब्रेरी",
+ "genre": "शैली",
+ "compilation": "संकलन",
+ "year": "वर्ष",
+ "date": "रिकॉर्डिंग की तारीख",
+ "originalDate": "मूल",
+ "releaseDate": "रिलीज़",
+ "releases": "रिलीज़ |||| रिलीज़",
+ "released": "रिलीज़ किया गया",
+ "updatedAt": "अपडेट किया गया",
+ "comment": "टिप्पणी",
+ "rating": "रेटिंग",
+ "createdAt": "जोड़ने की तारीख",
+ "recordLabel": "लेबल",
+ "catalogNum": "कैटलॉग नंबर",
+ "releaseType": "प्रकार",
+ "grouping": "समूहीकरण",
+ "media": "मीडिया",
+ "mood": "मूड",
+ "missing": "गुम"
+ },
+ "actions": {
+ "playAll": "चलाएं",
+ "playNext": "अगला चलाएं",
+ "addToQueue": "बाद में चलाएं",
+ "share": "साझा करें",
+ "shuffle": "शफल",
+ "addToPlaylist": "प्लेलिस्ट में जोड़ें",
+ "download": "डाउनलोड",
+ "info": "जानकारी प्राप्त करें"
+ },
+ "lists": {
+ "all": "सभी",
+ "random": "रैंडम",
+ "recentlyAdded": "हाल ही में जोड़े गए",
+ "recentlyPlayed": "हाल ही में चलाए गए",
+ "mostPlayed": "सबसे ज्यादा चलाए गए",
+ "starred": "पसंदीदा",
+ "topRated": "टॉप रेटेड"
+ }
+ },
+ "artist": {
+ "name": "कलाकार |||| कलाकार",
+ "fields": {
+ "name": "नाम",
+ "albumCount": "एल्बम की संख्या",
+ "songCount": "गानों की संख्या",
+ "size": "आकार",
+ "playCount": "प्ले संख्या",
+ "rating": "रेटिंग",
+ "genre": "शैली",
+ "role": "भूमिका",
+ "missing": "गुम"
+ },
+ "roles": {
+ "albumartist": "एल्बम कलाकार |||| एल्बम कलाकार",
+ "artist": "कलाकार |||| कलाकार",
+ "composer": "संगीतकार |||| संगीतकार",
+ "conductor": "संचालक |||| संचालक",
+ "lyricist": "गीतकार |||| गीतकार",
+ "arranger": "संयोजक |||| संयोजक",
+ "producer": "निर्माता |||| निर्माता",
+ "director": "निदेशक |||| निदेशक",
+ "engineer": "इंजीनियर |||| इंजीनियर",
+ "mixer": "मिक्सर |||| मिक्सर",
+ "remixer": "रीमिक्सर |||| रीमिक्सर",
+ "djmixer": "डीजे मिक्सर |||| डीजे मिक्सर",
+ "performer": "कलाकार |||| कलाकार",
+ "maincredit": "एल्बम कलाकार या कलाकार |||| एल्बम कलाकार या कलाकार"
+ },
+ "actions": {
+ "topSongs": "टॉप गाने",
+ "shuffle": "शफल",
+ "radio": "रेडियो"
+ }
+ },
+ "user": {
+ "name": "उपयोगकर्ता |||| उपयोगकर्ता",
+ "fields": {
+ "userName": "उपयोगकर्ता नाम",
+ "isAdmin": "एडमिन है",
+ "lastLoginAt": "अंतिम लॉगिन",
+ "lastAccessAt": "अंतिम पहुंच",
+ "updatedAt": "अपडेट किया गया",
+ "name": "नाम",
+ "password": "पासवर्ड",
+ "createdAt": "बनाया गया",
+ "changePassword": "पासवर्ड बदलें?",
+ "currentPassword": "वर्तमान पासवर्ड",
+ "newPassword": "नया पासवर्ड",
+ "token": "टोकन",
+ "libraries": "लाइब्रेरी"
+ },
+ "helperTexts": {
+ "name": "आपके नाम में परिवर्तन केवल अगली लॉगिन पर प्रभावी होगा",
+ "libraries": "इस उपयोगकर्ता के लिए विशिष्ट लाइब्रेरी चुनें, या डिफ़ॉल्ट लाइब्रेरी का उपयोग करने के लिए खाली छोड़ें"
+ },
+ "notifications": {
+ "created": "उपयोगकर्ता बनाया गया",
+ "updated": "उपयोगकर्ता अपडेट किया गया",
+ "deleted": "उपयोगकर्ता हटाया गया"
+ },
+ "validation": {
+ "librariesRequired": "गैर-एडमिन उपयोगकर्ताओं के लिए कम से कम एक लाइब्रेरी चुननी होगी"
+ },
+ "message": {
+ "listenBrainzToken": "अपना ListenBrainz उपयोगकर्ता टोकन दर्ज करें।",
+ "clickHereForToken": "अपना टोकन प्राप्त करने के लिए यहां क्लिक करें",
+ "selectAllLibraries": "सभी लाइब्रेरी चुनें",
+ "adminAutoLibraries": "एडमिन उपयोगकर्ताओं की सभी लाइब्रेरी तक स्वचालित पहुंच है"
+ }
+ },
+ "player": {
+ "name": "प्लेयर |||| प्लेयर",
+ "fields": {
+ "name": "नाम",
+ "transcodingId": "ट्रांसकोडिंग",
+ "maxBitRate": "अधिकतम बिट रेट",
+ "client": "क्लाइंट",
+ "userName": "उपयोगकर्ता नाम",
+ "lastSeen": "अंतिम बार देखा गया",
+ "reportRealPath": "वास्तविक पथ रिपोर्ट करें",
+ "scrobbleEnabled": "बाहरी सेवाओं को स्क्रॉबल भेजें"
+ }
+ },
+ "transcoding": {
+ "name": "ट्रांसकोडिंग |||| ट्रांसकोडिंग",
+ "fields": {
+ "name": "नाम",
+ "targetFormat": "लक्ष्य प्रारूप",
+ "defaultBitRate": "डिफ़ॉल्ट बिट रेट",
+ "command": "कमांड"
+ }
+ },
+ "playlist": {
+ "name": "प्लेलिस्ट |||| प्लेलिस्ट",
+ "fields": {
+ "name": "नाम",
+ "duration": "अवधि",
+ "ownerName": "मालिक",
+ "public": "सार्वजनिक",
+ "updatedAt": "अपडेट किया गया",
+ "createdAt": "बनाया गया",
+ "songCount": "गाने",
+ "comment": "टिप्पणी",
+ "sync": "ऑटो-इंपोर्ट",
+ "path": "से इंपोर्ट करें"
+ },
+ "actions": {
+ "selectPlaylist": "एक प्लेलिस्ट चुनें:",
+ "addNewPlaylist": "\"%{name}\" बनाएं",
+ "export": "निर्यात",
+ "saveQueue": "क्यू को प्लेलिस्ट में सेव करें",
+ "makePublic": "सार्वजनिक बनाएं",
+ "makePrivate": "निजी बनाएं",
+ "searchOrCreate": "प्लेलिस्ट खोजें या नई बनाने के लिए टाइप करें...",
+ "pressEnterToCreate": "नई प्लेलिस्ट बनाने के लिए Enter दबाएं",
+ "removeFromSelection": "चयन से हटाएं"
+ },
+ "message": {
+ "duplicate_song": "डुप्लिकेट गाने जोड़ें",
+ "song_exist": "प्लेलिस्ट में डुप्लिकेट जोड़े जा रहे हैं। क्या आप डुप्लिकेट जोड़ना चाहते हैं या उन्हें छोड़ना चाहते हैं?",
+ "noPlaylistsFound": "कोई प्लेलिस्ट नहीं मिली",
+ "noPlaylists": "कोई प्लेलिस्ट उपलब्ध नहीं"
+ }
+ },
+ "radio": {
+ "name": "रेडियो |||| रेडियो",
+ "fields": {
+ "name": "नाम",
+ "streamUrl": "स्ट्रीम URL",
+ "homePageUrl": "होम पेज URL",
+ "updatedAt": "अपडेट किया गया",
+ "createdAt": "बनाया गया"
+ },
+ "actions": {
+ "playNow": "अभी चलाएं"
+ }
+ },
+ "share": {
+ "name": "साझा |||| साझा",
+ "fields": {
+ "username": "द्वारा साझा किया गया",
+ "url": "URL",
+ "description": "विवरण",
+ "downloadable": "डाउनलोड की अनुमति दें?",
+ "contents": "सामग्री",
+ "expiresAt": "समाप्त होता है",
+ "lastVisitedAt": "अंतिम बार देखा गया",
+ "visitCount": "विज़िट",
+ "format": "प्रारूप",
+ "maxBitRate": "अधिकतम बिट रेट",
+ "updatedAt": "अपडेट किया गया",
+ "createdAt": "बनाया गया"
+ },
+ "notifications": {},
+ "actions": {}
+ },
+ "missing": {
+ "name": "गुम फ़ाइल |||| गुम फ़ाइलें",
+ "empty": "कोई गुम फ़ाइल नहीं",
+ "fields": {
+ "path": "पथ",
+ "size": "आकार",
+ "libraryName": "लाइब्रेरी",
+ "updatedAt": "गायब हुई"
+ },
+ "actions": {
+ "remove": "हटाएं",
+ "remove_all": "सभी हटाएं"
+ },
+ "notifications": {
+ "removed": "गुम फ़ाइल(एं) हटा दी गईं"
+ }
+ },
+ "library": {
+ "name": "लाइब्रेरी |||| लाइब्रेरी",
+ "fields": {
+ "name": "नाम",
+ "path": "पथ",
+ "remotePath": "रिमोट पथ",
+ "lastScanAt": "अंतिम स्कैन",
+ "songCount": "गाने",
+ "albumCount": "एल्बम",
+ "artistCount": "कलाकार",
+ "totalSongs": "गाने",
+ "totalAlbums": "एल्बम",
+ "totalArtists": "कलाकार",
+ "totalFolders": "फ़ोल्डर",
+ "totalFiles": "फ़ाइलें",
+ "totalMissingFiles": "गुम फ़ाइलें",
+ "totalSize": "कुल आकार",
+ "totalDuration": "अवधि",
+ "defaultNewUsers": "नए उपयोगकर्ताओं के लिए डिफ़ॉल्ट",
+ "createdAt": "बनाया गया",
+ "updatedAt": "अपडेट किया गया"
+ },
+ "sections": {
+ "basic": "बुनियादी जानकारी",
+ "statistics": "आंकड़े"
+ },
+ "actions": {
+ "scan": "लाइब्रेरी स्कैन करें",
+ "manageUsers": "उपयोगकर्ता पहुंच प्रबंधित करें",
+ "viewDetails": "विवरण देखें"
+ },
+ "notifications": {
+ "created": "लाइब्रेरी सफलतापूर्वक बनाई गई",
+ "updated": "लाइब्रेरी सफलतापूर्वक अपडेट की गई",
+ "deleted": "लाइब्रेरी सफलतापूर्वक हटाई गई",
+ "scanStarted": "लाइब्रेरी स्कैन शुरू किया गया",
+ "scanCompleted": "लाइब्रेरी स्कैन पूरा हुआ"
+ },
+ "validation": {
+ "nameRequired": "लाइब्रेरी का नाम आवश्यक है",
+ "pathRequired": "लाइब्रेरी पथ आवश्यक है",
+ "pathNotDirectory": "लाइब्रेरी पथ एक डायरेक्टरी होना चाहिए",
+ "pathNotFound": "लाइब्रेरी पथ नहीं मिला",
+ "pathNotAccessible": "लाइब्रेरी पथ पहुंच योग्य नहीं है",
+ "pathInvalid": "अमान्य लाइब्रेरी पथ"
+ },
+ "messages": {
+ "deleteConfirm": "क्या आप वाकई इस लाइब्रेरी को हटाना चाहते हैं? इससे सभी संबंधित डेटा और उपयोगकर्ता पहुंच हट जाएगी।",
+ "scanInProgress": "स्कैन चल रहा है...",
+ "noLibrariesAssigned": "इस उपयोगकर्ता को कोई लाइब्रेरी असाइन नहीं की गई"
+ }
+ }
+ },
+ "ra": {
+ "auth": {
+ "welcome1": "Navidrome इंस्टॉल करने के लिए धन्यवाद!",
+ "welcome2": "शुरू करने के लिए, एक एडमिन उपयोगकर्ता बनाएं",
+ "confirmPassword": "पासवर्ड की पुष्टि करें",
+ "buttonCreateAdmin": "एडमिन बनाएं",
+ "auth_check_error": "जारी रखने के लिए कृपया लॉगिन करें",
+ "user_menu": "प्रोफ़ाइल",
+ "username": "उपयोगकर्ता नाम",
+ "password": "पासवर्ड",
+ "sign_in": "साइन इन",
+ "sign_in_error": "प्रमाणीकरण विफल, कृपया पुनः प्रयास करें",
+ "logout": "लॉगआउट",
+ "insightsCollectionNote": "Navidrome परियोजना को बेहतर बनाने में मदद के लिए\nअज्ञात उपयोग डेटा एकत्र करता है। अधिक जानने\nऔर चाहें तो ऑप्ट-आउट करने के लिए [यहां] क्लिक करें"
+ },
+ "validation": {
+ "invalidChars": "कृपया केवल अक्षर और संख्याओं का उपयोग करें।",
+ "passwordDoesNotMatch": "पासवर्ड मेल नहीं खाता।",
+ "required": "आवश्यक",
+ "minLength": "कम से कम %{min} अक्षर होने चाहिए",
+ "maxLength": "%{max} अक्षर या उससे कम होने चाहिए",
+ "minValue": "कम से कम %{min} होना चाहिए",
+ "maxValue": "%{max} या उससे कम होना चाहिए",
+ "number": "एक संख्या होनी चाहिए",
+ "email": "एक वैध ईमेल होना चाहिए",
+ "oneOf": "इनमें से एक होना चाहिए: %{options}",
+ "regex": "एक विशिष्ट प्रारूप से मेल खाना चाहिए (regexp): %{pattern}",
+ "unique": "अद्वितीय होना चाहिए",
+ "url": "एक वैध URL होना चाहिए"
+ },
+ "action": {
+ "add_filter": "फ़िल्टर जोड़ें",
+ "add": "जोड़ें",
+ "back": "वापस जाएं",
+ "bulk_actions": "1 आइटम चुना गया |||| %{smart_count} आइटम चुने गए",
+ "bulk_actions_mobile": "1 |||| %{smart_count}",
+ "cancel": "रद्द करें",
+ "clear_input_value": "मान साफ़ करें",
+ "clone": "क्लोन",
+ "confirm": "पुष्टि करें",
+ "create": "बनाएं",
+ "delete": "हटाएं",
+ "edit": "संपादित करें",
+ "export": "निर्यात",
+ "list": "सूची",
+ "refresh": "रीफ्रेश",
+ "remove_filter": "इस फ़िल्टर को हटाएं",
+ "remove": "हटाएं",
+ "save": "सेव करें",
+ "search": "खोजें",
+ "show": "दिखाएं",
+ "sort": "क्रमबद्ध करें",
+ "undo": "पूर्ववत करें",
+ "expand": "विस्तार करें",
+ "close": "बंद करें",
+ "open_menu": "मेनू खोलें",
+ "close_menu": "मेनू बंद करें",
+ "unselect": "चयन हटाएं",
+ "skip": "छोड़ें",
+ "share": "साझा करें",
+ "download": "डाउनलोड"
+ },
+ "boolean": {
+ "true": "हां",
+ "false": "नहीं"
+ },
+ "page": {
+ "create": "%{name} बनाएं",
+ "dashboard": "डैशबोर्ड",
+ "edit": "%{name} #%{id}",
+ "error": "कुछ गलत हुआ",
+ "list": "%{name}",
+ "loading": "लोड हो रहा है",
+ "not_found": "नहीं मिला",
+ "show": "%{name} #%{id}",
+ "empty": "अभी तक कोई %{name} नहीं।",
+ "invite": "क्या आप एक जोड़ना चाहते हैं?"
+ },
+ "input": {
+ "file": {
+ "upload_several": "अपलोड करने के लिए कुछ फ़ाइलें छोड़ें, या चुनने के लिए क्लिक करें।",
+ "upload_single": "अपलोड करने के लिए एक फ़ाइल छोड़ें, या इसे चुनने के लिए क्लिक करें।"
+ },
+ "image": {
+ "upload_several": "अपलोड करने के लिए कुछ तस्वीरें छोड़ें, या चुनने के लिए क्लिक करें।",
+ "upload_single": "अपलोड करने के लिए एक तस्वीर छोड़ें, या इसे चुनने के लिए क्लिक करें।"
+ },
+ "references": {
+ "all_missing": "संदर्भ डेटा खोजने में असमर्थ।",
+ "many_missing": "संबंधित संदर्भों में से कम से कम एक अब उपलब्ध नहीं लगता।",
+ "single_missing": "संबंधित संदर्भ अब उपलब्ध नहीं लगता।"
+ },
+ "password": {
+ "toggle_visible": "पासवर्ड छुपाएं",
+ "toggle_hidden": "पासवर्ड दिखाएं"
+ }
+ },
+ "message": {
+ "about": "के बारे में",
+ "are_you_sure": "क्या आप सुनिश्चित हैं?",
+ "bulk_delete_content": "क्या आप वाकई इस %{name} को हटाना चाहते हैं? |||| क्या आप वाकई इन %{smart_count} आइटमों को हटाना चाहते हैं?",
+ "bulk_delete_title": "%{name} हटाएं |||| %{smart_count} %{name} हटाएं",
+ "delete_content": "क्या आप वाकई इस आइटम को हटाना चाहते हैं?",
+ "delete_title": "%{name} #%{id} हटाएं",
+ "details": "विवरण",
+ "error": "एक क्लाइंट त्रुटि हुई और आपका अनुरोध पूरा नहीं हो सका।",
+ "invalid_form": "फॉर्म मान्य नहीं है। कृपया त्रुटियों की जांच करें।",
+ "loading": "पेज लोड हो रहा है, कृपया एक क्षण प्रतीक्षा करें",
+ "no": "नहीं",
+ "not_found": "या तो आपने गलत URL टाइप किया है, या आपने गलत लिंक फॉलो किया है।",
+ "yes": "हां",
+ "unsaved_changes": "आपके कुछ बदलाव सेव नहीं हुए। क्या आप वाकई उन्हें नज़रअंदाज़ करना चाहते हैं?"
+ },
+ "navigation": {
+ "no_results": "कोई परिणाम नहीं मिला",
+ "no_more_results": "पेज नंबर %{page} सीमा से बाहर है। पिछले पेज को आज़माएं।",
+ "page_out_of_boundaries": "पेज नंबर %{page} सीमा से बाहर",
+ "page_out_from_end": "अंतिम पेज के बाद नहीं जा सकते",
+ "page_out_from_begin": "पेज 1 से पहले नहीं जा सकते",
+ "page_range_info": "%{total} में से %{offsetBegin}-%{offsetEnd}",
+ "page_rows_per_page": "प्रति पेज आइटम:",
+ "next": "अगला",
+ "prev": "पिछला",
+ "skip_nav": "सामग्री पर जाएं"
+ },
+ "notification": {
+ "updated": "एलिमेंट अपडेट किया गया |||| %{smart_count} एलिमेंट अपडेट किए गए",
+ "created": "एलिमेंट बनाया गया",
+ "deleted": "एलिमेंट हटाया गया |||| %{smart_count} एलिमेंट हटाए गए",
+ "bad_item": "गलत एलिमेंट",
+ "item_doesnt_exist": "एलिमेंट मौजूद नहीं है",
+ "http_error": "सर्वर संचार त्रुटि",
+ "data_provider_error": "dataProvider त्रुटि। विवरण के लिए कंसोल जांचें।",
+ "i18n_error": "निर्दिष्ट भाषा के लिए अनुवाद लोड नहीं हो सकते",
+ "canceled": "कार्रवाई रद्द की गई",
+ "logged_out": "आपका सत्र समाप्त हो गया है, कृपया फिर से कनेक्ट करें।",
+ "new_version": "नया संस्करण उपलब्ध! कृपया इस विंडो को रीफ्रेश करें।"
+ },
+ "toggleFieldsMenu": {
+ "columnsToDisplay": "प्रदर्शित करने वाले कॉलम",
+ "layout": "लेआउट",
+ "grid": "ग्रिड",
+ "table": "टेबल"
+ }
+ },
+ "message": {
+ "note": "नोट",
+ "transcodingDisabled": "सुरक्षा कारणों से वेब इंटरफेस के माध्यम से ट्रांसकोडिंग कॉन्फ़िगरेशन बदलना अक्षम है। यदि आप ट्रांसकोडिंग विकल्प बदलना (संपादित या जोड़ना) चाहते हैं, तो %{config} कॉन्फ़िगरेशन विकल्प के साथ सर्वर को पुनः आरंभ करें।",
+ "transcodingEnabled": "Navidrome वर्तमान में %{config} के साथ चल रहा है, जो वेब इंटरफेस का उपयोग करके ट्रांसकोडिंग सेटिंग्स से सिस्टम कमांड चलाना संभव बनाता है। हम सुरक्षा कारणों से इसे अक्षम करने और केवल ट्रांसकोडिंग विकल्प कॉन्फ़िगर करते समय इसे सक्षम करने की सलाह देते हैं।",
+ "songsAddedToPlaylist": "प्लेलिस्ट में 1 गाना जोड़ा गया |||| प्लेलिस्ट में %{smart_count} गाने जोड़े गए",
+ "noSimilarSongsFound": "कोई समान गाने नहीं मिले",
+ "noTopSongsFound": "कोई टॉप गाने नहीं मिले",
+ "noPlaylistsAvailable": "कोई उपलब्ध नहीं",
+ "delete_user_title": "उपयोगकर्ता '%{name}' को हटाएं",
+ "delete_user_content": "क्या आप वाकई इस उपयोगकर्ता और उनके सभी डेटा (प्लेलिस्ट और प्राथमिकताओं सहित) को हटाना चाहते हैं?",
+ "remove_missing_title": "गुम फ़ाइलें हटाएं",
+ "remove_missing_content": "क्या आप वाकई चयनित गुम फ़ाइलों को डेटाबेस से हटाना चाहते हैं? इससे उनके सभी संदर्भ स्थायी रूप से हट जाएंगे, जिसमें उनकी प्ले काउंट और रेटिंग शामिल है।",
+ "remove_all_missing_title": "सभी गुम फ़ाइलें हटाएं",
+ "remove_all_missing_content": "क्या आप वाकई सभी गुम फ़ाइलों को डेटाबेस से हटाना चाहते हैं? इससे उनके सभी संदर्भ स्थायी रूप से हट जाएंगे, जिसमें उनकी प्ले काउंट और रेटिंग शामिल है।",
+ "notifications_blocked": "आपने अपने ब्राउज़र की सेटिंग्स में इस साइट के लिए सूचनाएं ब्लॉक की हैं।",
+ "notifications_not_available": "यह ब्राउज़र डेस्कटॉप सूचनाओं का समर्थन नहीं करता या आप https पर Navidrome का उपयोग नहीं कर रहे।",
+ "lastfmLinkSuccess": "Last.fm सफलतापूर्वक लिंक किया गया और स्क्रॉबलिंग सक्षम की गई",
+ "lastfmLinkFailure": "Last.fm लिंक नहीं हो सका",
+ "lastfmUnlinkSuccess": "Last.fm अनलिंक किया गया और स्क्रॉबलिंग अक्षम की गई",
+ "lastfmUnlinkFailure": "Last.fm अनलिंक नहीं हो सका",
+ "listenBrainzLinkSuccess": "ListenBrainz सफलतापूर्वक लिंक किया गया और उपयोगकर्ता के रूप में स्क्रॉबलिंग सक्षम की गई: %{user}",
+ "listenBrainzLinkFailure": "ListenBrainz लिंक नहीं हो सका: %{error}",
+ "listenBrainzUnlinkSuccess": "ListenBrainz अनलिंक किया गया और स्क्रॉबलिंग अक्षम की गई",
+ "listenBrainzUnlinkFailure": "ListenBrainz अनलिंक नहीं हो सका",
+ "openIn": {
+ "lastfm": "Last.fm में खोलें",
+ "musicbrainz": "MusicBrainz में खोलें"
+ },
+ "lastfmLink": "और पढ़ें...",
+ "shareOriginalFormat": "मूल प्रारूप में साझा करें",
+ "shareDialogTitle": "%{resource} '%{name}' साझा करें",
+ "shareBatchDialogTitle": "1 %{resource} साझा करें |||| %{smart_count} %{resource} साझा करें",
+ "shareCopyToClipboard": "क्लिपबोर्ड में कॉपी करें: Ctrl+C, Enter",
+ "shareSuccess": "URL क्लिपबोर्ड में कॉपी किया गया: %{url}",
+ "shareFailure": "URL %{url} को क्लिपबोर्ड में कॉपी करने में त्रुटि",
+ "downloadDialogTitle": "%{resource} '%{name}' (%{size}) डाउनलोड करें",
+ "downloadOriginalFormat": "मूल प्रारूप में डाउनलोड करें"
+ },
+ "menu": {
+ "library": "लाइब्रेरी",
+ "librarySelector": {
+ "allLibraries": "सभी लाइब्रेरी (%{count})",
+ "multipleLibraries": "%{total} में से %{selected} लाइब्रेरी",
+ "selectLibraries": "लाइब्रेरी चुनें",
+ "none": "कोई नहीं"
+ },
+ "settings": "सेटिंग्स",
+ "version": "संस्करण",
+ "theme": "थीम",
+ "personal": {
+ "name": "व्यक्तिगत",
+ "options": {
+ "theme": "थीम",
+ "language": "भाषा",
+ "defaultView": "डिफ़ॉल्ट दृश्य",
+ "desktop_notifications": "डेस्कटॉप सूचनाएं",
+ "lastfmNotConfigured": "Last.fm API-Key कॉन्फ़िगर नहीं है",
+ "lastfmScrobbling": "Last.fm में स्क्रॉबल करें",
+ "listenBrainzScrobbling": "ListenBrainz में स्क्रॉबल करें",
+ "replaygain": "ReplayGain मोड",
+ "preAmp": "ReplayGain PreAmp (dB)",
+ "gain": {
+ "none": "अक्षम",
+ "album": "एल्बम गेन का उपयोग करें",
+ "track": "ट्रैक गेन का उपयोग करें"
+ }
+ }
+ },
+ "albumList": "एल्बम",
+ "playlists": "प्लेलिस्ट",
+ "sharedPlaylists": "साझा की गई प्लेलिस्ट",
+ "about": "के बारे में"
+ },
+ "player": {
+ "playListsText": "प्ले क्यू",
+ "openText": "खोलें",
+ "closeText": "बंद करें",
+ "notContentText": "कोई संगीत नहीं",
+ "clickToPlayText": "चलाने के लिए क्लिक करें",
+ "clickToPauseText": "रोकने के लिए क्लिक करें",
+ "nextTrackText": "अगला ट्रैक",
+ "previousTrackText": "पिछला ट्रैक",
+ "reloadText": "रीलोड",
+ "volumeText": "वॉल्यूम",
+ "toggleLyricText": "गीत टॉगल करें",
+ "toggleMiniModeText": "मिनिमाइज़ करें",
+ "destroyText": "नष्ट करें",
+ "downloadText": "डाउनलोड",
+ "removeAudioListsText": "ऑडियो सूची हटाएं",
+ "clickToDeleteText": "%{name} को हटाने के लिए क्लिक करें",
+ "emptyLyricText": "कोई गीत नहीं",
+ "playModeText": {
+ "order": "क्रम में",
+ "orderLoop": "दोहराएं",
+ "singleLoop": "एक दोहराएं",
+ "shufflePlay": "शफल"
+ }
+ },
+ "about": {
+ "links": {
+ "homepage": "होम पेज",
+ "source": "सोर्स कोड",
+ "featureRequests": "फीचर अनुरोध",
+ "lastInsightsCollection": "अंतिम अंतर्दृष्टि संग्रह",
+ "insights": {
+ "disabled": "अक्षम",
+ "waiting": "प्रतीक्षा में"
+ }
+ },
+ "tabs": {
+ "about": "के बारे में",
+ "config": "कॉन्फ़िगरेशन"
+ },
+ "config": {
+ "configName": "कॉन्फ़िग नाम",
+ "environmentVariable": "पर्यावरण चर",
+ "currentValue": "वर्तमान मान",
+ "configurationFile": "कॉन्फ़िगरेशन फ़ाइल",
+ "exportToml": "कॉन्फ़िगरेशन निर्यात करें (TOML)",
+ "exportSuccess": "कॉन्फ़िगरेशन TOML प्रारूप में क्लिपबोर्ड में निर्यात किया गया",
+ "exportFailed": "कॉन्फ़िगरेशन कॉपी करने में विफल",
+ "devFlagsHeader": "विकास फ्लैग (परिवर्तन/हटाने के अधीन)",
+ "devFlagsComment": "ये प्रयोगात्मक सेटिंग्स हैं और भविष्य के संस्करणों में हटाई जा सकती हैं"
+ }
+ },
+ "activity": {
+ "title": "गतिविधि",
+ "totalScanned": "कुल स्कैन किए गए फ़ोल्डर",
+ "quickScan": "त्वरित स्कैन",
+ "fullScan": "पूर्ण स्कैन",
+ "serverUptime": "सर्वर अपटाइम",
+ "serverDown": "ऑफलाइन",
+ "scanType": "प्रकार",
+ "status": "स्कैन त्रुटि",
+ "elapsedTime": "बीता समय"
+ },
+ "nowPlaying": {
+ "title": "अभी चल रहा है",
+ "empty": "कुछ नहीं चल रहा",
+ "minutesAgo": "%{smart_count} मिनट पहले |||| %{smart_count} मिनट पहले"
+ },
+ "help": {
+ "title": "Navidrome हॉटकीज़",
+ "hotkeys": {
+ "show_help": "यह सहायता दिखाएं",
+ "toggle_menu": "मेनू साइड बार टॉगल करें",
+ "toggle_play": "चलाएं / रोकें",
+ "prev_song": "पिछला गाना",
+ "next_song": "अगला गाना",
+ "current_song": "वर्तमान गाने पर जाएं",
+ "vol_up": "वॉल्यूम बढ़ाएं",
+ "vol_down": "वॉल्यूम कम करें",
+ "toggle_love": "इस ट्रैक को पसंदीदा में जोड़ें"
+ }
+ }
+}
diff --git a/resources/i18n/hu.json b/resources/i18n/hu.json
index edee39e38..a2037eb54 100644
--- a/resources/i18n/hu.json
+++ b/resources/i18n/hu.json
@@ -12,6 +12,7 @@
"artist": "Előadó",
"album": "Album",
"path": "Elérési út",
+ "libraryName": "Könyvtár",
"genre": "Műfaj",
"compilation": "Válogatásalbum",
"year": "Év",
@@ -34,12 +35,14 @@
"participants": "További résztvevők",
"tags": "További címkék",
"mappedTags": "Feldolgozott címkék",
- "rawTags": "Nyers címkék"
+ "rawTags": "Nyers címkék",
+ "missing": "Hiányzó"
},
"actions": {
"addToQueue": "Lejátszás útolsóként",
"playNow": "Lejátszás",
"addToPlaylist": "Lejátszási listához adás",
+ "showInPlaylist": "Megjelenítés a lejátszási listában",
"shuffleAll": "Keverés",
"download": "Letöltés",
"playNext": "Lejátszás következőként",
@@ -55,6 +58,7 @@
"songCount": "Számok",
"playCount": "Lejátszások",
"name": "Név",
+ "libraryName": "Könyvtár",
"genre": "Stílus",
"compilation": "Válogatásalbum",
"year": "Év",
@@ -73,7 +77,8 @@
"releaseType": "Típus",
"grouping": "Csoportosítás",
"media": "Média",
- "mood": "Hangulat"
+ "mood": "Hangulat",
+ "missing": "Hiányzó"
},
"actions": {
"playAll": "Lejátszás",
@@ -105,7 +110,8 @@
"rating": "Értékelés",
"genre": "Stílus",
"size": "Méret",
- "role": "Szerep"
+ "role": "Szerep",
+ "missing": "Hiányzó"
},
"roles": {
"albumartist": "Album előadó |||| Album előadók",
@@ -120,7 +126,13 @@
"mixer": "Keverő |||| Keverők",
"remixer": "Átdolgozó |||| Átdolgozók",
"djmixer": "DJ keverő |||| DJ keverők",
- "performer": "Előadóművész |||| Előadóművészek"
+ "performer": "Előadóművész |||| Előadóművészek",
+ "maincredit": "Album előadó vagy előadó |||| Album előadók vagy előadók"
+ },
+ "actions": {
+ "topSongs": "Top számok",
+ "shuffle": "Keverés",
+ "radio": "Rádió"
}
},
"user": {
@@ -137,19 +149,26 @@
"currentPassword": "Jelenlegi jelszó",
"newPassword": "Új jelszó",
"token": "Token",
- "lastAccessAt": "Utolsó elérés"
+ "lastAccessAt": "Utolsó elérés",
+ "libraries": "Könyvtárak"
},
"helperTexts": {
- "name": "A névváltoztatások csak a következő bejelentkezéskor jelennek meg"
+ "name": "A névváltoztatások csak a következő bejelentkezéskor jelennek meg",
+ "libraries": "Válassz könyvtárakat ehhez a felhasználóhoz vagy ne jelölj be egyet sem, az alapértelmezett könyvtárak használatához"
},
"notifications": {
"created": "Felhasználó létrehozva",
"updated": "Felhasználó frissítve",
"deleted": "Felhasználó törölve"
},
+ "validation": {
+ "librariesRequired": "Legalább egy könyvtárat ki kell választani nem admin felhasználókhoz"
+ },
"message": {
"listenBrainzToken": "Add meg a ListenBrainz felhasználó tokened.",
- "clickHereForToken": "Kattints ide, hogy megszerezd a tokened"
+ "clickHereForToken": "Kattints ide, hogy megszerezd a tokened",
+ "selectAllLibraries": "Minden könyvtár kiválasztása",
+ "adminAutoLibraries": "Minden admin felhasználó hozzáfér bármely könyvtárhoz"
}
},
"player": {
@@ -194,11 +213,16 @@
"export": "Exportálás",
"saveQueue": "Műsorlista elmentése lejátszási listaként",
"makePublic": "Publikussá tétel",
- "makePrivate": "Priváttá tétel"
+ "makePrivate": "Priváttá tétel",
+ "searchOrCreate": "Keress lejátszási listák között vagy hozz létre egyet...",
+ "pressEnterToCreate": "Nyomj Entert, hogy létrehozz egy lejátszási listát",
+ "removeFromSelection": "Eltávolítás a kiválasztásból"
},
"message": {
"duplicate_song": "Duplikált számok hozzáadása",
- "song_exist": "Egyes számok már hozzá vannak adva a listához. Még egyszer hozzá akarod adni?"
+ "song_exist": "Egyes számok már hozzá vannak adva a listához. Még egyszer hozzá akarod adni?",
+ "noPlaylistsFound": "Nem található lejátszási lista",
+ "noPlaylists": "Nincsenek lejátszási listák"
}
},
"radio": {
@@ -237,6 +261,7 @@
"fields": {
"path": "Útvonal",
"size": "Méret",
+ "libraryName": "Könyvtár",
"updatedAt": "Eltűnt ekkor:"
},
"actions": {
@@ -246,6 +271,58 @@
"notifications": {
"removed": "Hiányzó fájl(ok) eltávolítva"
}
+ },
+ "library": {
+ "name": "Könyvtár |||| Könyvtárak",
+ "fields": {
+ "name": "Név",
+ "path": "Elérési út",
+ "remotePath": "Távoli elérési út",
+ "lastScanAt": "Legutóbbi szkennelés",
+ "songCount": "Számok",
+ "albumCount": "Albumok",
+ "artistCount": "Előadók",
+ "totalSongs": "Számok",
+ "totalAlbums": "Albumok",
+ "totalArtists": "Előadók",
+ "totalFolders": "Mappák",
+ "totalFiles": "Fájlok",
+ "totalMissingFiles": "Hiányzó fájlok",
+ "totalSize": "Teljes méret",
+ "totalDuration": "Hossz",
+ "defaultNewUsers": "Alapértelmezett könyvtár új felhasználóknak",
+ "createdAt": "Létrehozva",
+ "updatedAt": "Frissítve"
+ },
+ "sections": {
+ "basic": "Alapinformációk",
+ "statistics": "Statisztikák"
+ },
+ "actions": {
+ "scan": "Könyvtár szkennelése",
+ "manageUsers": "Elérés kezelése",
+ "viewDetails": "Részletek"
+ },
+ "notifications": {
+ "created": "Könyvtár létrehozva",
+ "updated": "Könyvtár frissítve",
+ "deleted": "Könyvtár törölve",
+ "scanStarted": "Szkennelés folyamatban",
+ "scanCompleted": "Könyvtár szkennelés befelyezve"
+ },
+ "validation": {
+ "nameRequired": "Adj meg egy könyvtárnevet",
+ "pathRequired": "Adj meg egy útvonalat",
+ "pathNotDirectory": "A könyvtárútvonalnak egy mappának kell lennie",
+ "pathNotFound": "A könyvtár útvonala nem található",
+ "pathNotAccessible": "A könyvtár útvonala nem elérhető",
+ "pathInvalid": "Helytelen könyvtár útvonal"
+ },
+ "messages": {
+ "deleteConfirm": "Biztosan törlöd ezt a könyvtárt? Minden adata törlődni fog és elérhetetlenné válik.",
+ "scanInProgress": "Szkennelés folyamatban...",
+ "noLibrariesAssigned": "Ehhez a felhasználóhoz nincsenek könyvtárak adva"
+ }
}
},
"ra": {
@@ -398,6 +475,8 @@
"transcodingDisabled": "Az átkódolási konfiguráció módosítása a webes felületen keresztül biztonsági okokból nem lehetséges. Ha módosítani szeretnéd az átkódolási beállításokat, indítsd újra a kiszolgálót a %{config} konfigurációs opcióval.",
"transcodingEnabled": "A Navidrome jelenleg a következőkkel fut %{config}, ez lehetővé teszi a rendszerparancsok futtatását az átkódolási beállításokból a webes felület segítségével. Javasoljuk, hogy biztonsági okokból tiltsd ezt le, és csak az átkódolási beállítások konfigurálásának idejére kapcsold be.",
"songsAddedToPlaylist": "1 szám hozzáadva a lejátszási listához |||| %{smart_count} szám hozzáadva a lejátszási listához",
+ "noSimilarSongsFound": "Nem találhatóak hasonló számok",
+ "noTopSongsFound": "Nincsenek top számok",
"noPlaylistsAvailable": "Nem áll rendelkezésre",
"delete_user_title": "Felhasználó törlése '%{name}'",
"delete_user_content": "Biztos, hogy törölni akarod ezt a felhasználót az adataival (beállítások és lejátszási listák) együtt?",
@@ -431,6 +510,12 @@
},
"menu": {
"library": "Könyvtár",
+ "librarySelector": {
+ "allLibraries": "Minden %{count} könyvtár",
+ "multipleLibraries": "%{selected} kiválasztva %{total} könyvtárból",
+ "selectLibraries": "Kiválasztott kőnyvtárak",
+ "none": "Semmi"
+ },
"settings": "Beállítások",
"version": "Verzió",
"theme": "Téma",
@@ -493,19 +578,39 @@
"disabled": "Kikapcsolva",
"waiting": "Várakozás"
}
+ },
+ "tabs": {
+ "about": "Rólunk",
+ "config": "Konfiguráció"
+ },
+ "config": {
+ "configName": "Beállítás neve",
+ "environmentVariable": "Környezeti változó",
+ "currentValue": "Jelenlegi érték",
+ "configurationFile": "Konfigurációs fájl",
+ "exportToml": "Konfiguráció exportálása (TOML)",
+ "exportSuccess": "Konfiguráció kiexportálva a vágólapra, TOML formában",
+ "exportFailed": "Nem sikerült kimásolni a konfigurációt",
+ "devFlagsHeader": "Fejlesztői beállítások (változások/eltávolítás jogát fenntartjuk)",
+ "devFlagsComment": "Ezek kísérleti beállítások, és a jövőbeli verziókban eltávolíthatók"
}
},
"activity": {
"title": "Aktivitás",
"totalScanned": "Összes beolvasott mappa:",
- "quickScan": "Gyors beolvasás",
- "fullScan": "Teljes beolvasás",
+ "quickScan": "Gyors szkennelés",
+ "fullScan": "Teljes szkennelés",
"serverUptime": "Szerver üzemidő",
"serverDown": "OFFLINE",
"scanType": "Típus",
"status": "Szkennelési hiba",
"elapsedTime": "Eltelt idő"
},
+ "nowPlaying": {
+ "title": "Most megy",
+ "empty": "Nem hallgatsz semmit",
+ "minutesAgo": "%{smart_count} perce |||| %{smart_count} perce"
+ },
"help": {
"title": "Navidrome Gyorsbillentyűk",
"hotkeys": {
diff --git a/resources/i18n/id.json b/resources/i18n/id.json
index 3269b37b1..38ee2fff9 100644
--- a/resources/i18n/id.json
+++ b/resources/i18n/id.json
@@ -32,7 +32,11 @@
"participants": "Partisipan tambahan",
"tags": "Tag tambahan",
"mappedTags": "Tag yang dipetakan",
- "rawTags": "Tag raw"
+ "rawTags": "Tag raw",
+ "bitDepth": "Bit depth",
+ "sampleRate": "Sample rate",
+ "missing": "Hilang",
+ "libraryName": "Pustaka"
},
"actions": {
"addToQueue": "Tambah ke antrean",
@@ -41,7 +45,8 @@
"shuffleAll": "Acak Semua",
"download": "Unduh",
"playNext": "Putar Berikutnya",
- "info": "Lihat Info"
+ "info": "Lihat Info",
+ "showInPlaylist": "Tampilkan di Playlist"
}
},
"album": {
@@ -70,7 +75,10 @@
"releaseType": "Tipe",
"grouping": "Pengelompokkan",
"media": "Media",
- "mood": "Mood"
+ "mood": "Mood",
+ "date": "Tanggal Perekaman",
+ "missing": "Hilang",
+ "libraryName": "Pustaka"
},
"actions": {
"playAll": "Putar",
@@ -102,7 +110,8 @@
"rating": "Peringkat",
"genre": "Genre",
"size": "Ukuran",
- "role": "Peran"
+ "role": "Peran",
+ "missing": "Hilang"
},
"roles": {
"albumartist": "Artis Album |||| Artis Album",
@@ -117,7 +126,13 @@
"mixer": "Mixer |||| Mixer",
"remixer": "Remixer |||| Remixer",
"djmixer": "DJ Mixer |||| Dj Mixer",
- "performer": "Performer |||| Performer"
+ "performer": "Performer |||| Performer",
+ "maincredit": "Artis Album atau Artis |||| Artis Album or Artis"
+ },
+ "actions": {
+ "shuffle": "Acak",
+ "radio": "Radio",
+ "topSongs": "Lagu Teratas"
}
},
"user": {
@@ -134,10 +149,12 @@
"currentPassword": "Kata Sandi Sebelumnya",
"newPassword": "Kata Sandi Baru",
"token": "Token",
- "lastAccessAt": "Terakhir Diakses"
+ "lastAccessAt": "Terakhir Diakses",
+ "libraries": "Perpustakaan"
},
"helperTexts": {
- "name": "Perubahan pada nama Kamu akan terlihat pada login berikutnya"
+ "name": "Perubahan pada nama Kamu akan terlihat pada login berikutnya",
+ "libraries": "Pilih pustaka yang ditentukan untuk pengguna ini, atau biarkan kosong untuk menggunakan pustaka default"
},
"notifications": {
"created": "Pengguna dibuat",
@@ -146,7 +163,12 @@
},
"message": {
"listenBrainzToken": "Masukkan token pengguna ListenBrainz Kamu.",
- "clickHereForToken": "Klik di sini untuk mendapatkan token baru anda"
+ "clickHereForToken": "Klik di sini untuk mendapatkan token baru anda",
+ "selectAllLibraries": "Pilih semua pustaka",
+ "adminAutoLibraries": "Pengguna admin otomatis langsung memiliki akses ke semua perpustakaan"
+ },
+ "validation": {
+ "librariesRequired": "Setidaknya satu pustaka harus dipilih untuk pengguna non-admin"
}
},
"player": {
@@ -163,7 +185,7 @@
}
},
"transcoding": {
- "name": "Transkode |||| Transkode",
+ "name": "Transkoding |||| Transkoding",
"fields": {
"name": "Nama",
"targetFormat": "Target Format",
@@ -190,11 +212,17 @@
"addNewPlaylist": "Buat \"%{name}\"",
"export": "Ekspor",
"makePublic": "Jadikan Publik",
- "makePrivate": "Jadikan Pribadi"
+ "makePrivate": "Jadikan Pribadi",
+ "saveQueue": "Simpan Antrean ke Playlist",
+ "searchOrCreate": "Cari playlist atau ketik untuk buat baru..",
+ "pressEnterToCreate": "Tekan Enter untuk membuat playlist baru",
+ "removeFromSelection": "Hapus yang dipilih"
},
"message": {
"duplicate_song": "Tambahkan lagu duplikat",
- "song_exist": "Ada lagu duplikat yang ditambahkan ke daftar putar. Apakah Kamu ingin menambahkan lagu duplikat atau melewatkannya?"
+ "song_exist": "Ada lagu duplikat yang ditambahkan ke daftar putar. Apakah Kamu ingin menambahkan lagu duplikat atau melewatkannya?",
+ "noPlaylistsFound": "Playlist tidak ditemukan",
+ "noPlaylists": "Playlist tidak tersedia"
}
},
"radio": {
@@ -232,13 +260,68 @@
"fields": {
"path": "Jalur",
"size": "Ukuran",
- "updatedAt": "Tidak muncul di"
+ "updatedAt": "Tidak muncul di",
+ "libraryName": "Pustaka"
},
"actions": {
- "remove": "Hapus"
+ "remove": "Hapus",
+ "remove_all": "Hapus Semua"
},
"notifications": {
"removed": "File yang hilang dihapus"
+ },
+ "empty": "Tidak ada File yang Hilang"
+ },
+ "library": {
+ "name": "Pustaka |||| Perpustakaan",
+ "fields": {
+ "name": "Nama",
+ "path": "Jalur",
+ "remotePath": "Jalur Remote",
+ "lastScanAt": "Terakhir Dipindai",
+ "songCount": "Lagu",
+ "albumCount": "Album",
+ "artistCount": "Artis",
+ "totalSongs": "Lagu",
+ "totalAlbums": "Album",
+ "totalArtists": "Artis",
+ "totalFolders": "Folder",
+ "totalFiles": "File",
+ "totalMissingFiles": "File hilang",
+ "totalSize": "Ukuran Total",
+ "totalDuration": "Durasi",
+ "defaultNewUsers": "Default untuk Pengguna Baru",
+ "createdAt": "Dibuat",
+ "updatedAt": "Diperbarui"
+ },
+ "sections": {
+ "basic": "Informasi Dasar",
+ "statistics": "Statistik"
+ },
+ "actions": {
+ "scan": "Pindai Pustaka",
+ "manageUsers": "Kelola Akses Pengguna",
+ "viewDetails": "Lihat Detail"
+ },
+ "notifications": {
+ "created": "Pustaka berhasil dibuat",
+ "updated": "Pustaka berhasil dibuat",
+ "deleted": "Berhasil menghapus pustaka",
+ "scanStarted": "Memindai pustaka dimulai",
+ "scanCompleted": "Memindai pustaka selesai"
+ },
+ "validation": {
+ "nameRequired": "Nama pustaka diperlukan",
+ "pathRequired": "Lokasi pustaka diperlukan",
+ "pathNotDirectory": "Lokasi pustaka harus ada di direktori",
+ "pathNotFound": "Lokasi pustaka tidak ditemukan",
+ "pathNotAccessible": "Lokasi pustaka tidak dapat diakses",
+ "pathInvalid": "Lokasi pustaka tidak valid"
+ },
+ "messages": {
+ "deleteConfirm": "Kamu yakin ingin menghapus pustaka ini? Ini akan menghapus semua data yang terkait dan akses pengguna.",
+ "scanInProgress": "Pemindaian sedang berlangsung...",
+ "noLibrariesAssigned": "Tidak ada pustaka yang ditugaskan ke pengguna ini"
}
}
},
@@ -277,7 +360,7 @@
"add": "Tambah",
"back": "Kembali",
"bulk_actions": "1 item dipilih |||| %{smart_count} item dipilih",
- "cancel": "Batalkan",
+ "cancel": "Batal",
"clear_input_value": "Hapus",
"clone": "Klon",
"confirm": "Konfirmasi",
@@ -292,7 +375,7 @@
"save": "Simpan",
"search": "Cari",
"show": "Tampilkan",
- "sort": "Sortir",
+ "sort": "Urutkan",
"undo": "Batalkan",
"expand": "Luaskan",
"close": "Tutup",
@@ -312,7 +395,7 @@
"create": "Buat %{name}",
"dashboard": "Dasbor",
"edit": "%{name} #%{id}",
- "error": "Ada yang tidak beres",
+ "error": "Terjadi kesalahan",
"list": "%{name}",
"loading": "Memuat",
"not_found": "Tidak ditemukan",
@@ -356,7 +439,7 @@
"unsaved_changes": "Beberapa perubahan tidak disimpan. Apakah Kamu yakin ingin mengabaikannya?"
},
"navigation": {
- "no_results": "Tidak ada hasil yang ditemukan",
+ "no_results": "Hasil tidak ditemukan",
"no_more_results": "Nomor halaman %{page} melampaui batas. Coba halaman sebelumnya.",
"page_out_of_boundaries": "Nomor halaman %{page} melampaui batas",
"page_out_from_end": "Tidak dapat menelusuri sebelum halaman terakhir",
@@ -371,8 +454,8 @@
"updated": "Elemen diperbarui |||| %{smart_count} elemen diperbarui",
"created": "Elemen dibuat",
"deleted": "Elemen dihapus |||| %{smart_count} elemen dihapus",
- "bad_item": "Elemen salah",
- "item_doesnt_exist": "Tidak ada elemen",
+ "bad_item": "Kesalahan elemen",
+ "item_doesnt_exist": "Elemen tidak ditemukan",
"http_error": "Kesalahan komunikasi peladen",
"data_provider_error": "dataProvider galat. Periksa konsol untuk detailnya.",
"i18n_error": "Tidak dapat memuat terjemahan untuk bahasa yang diatur",
@@ -419,7 +502,11 @@
"downloadDialogTitle": "Unduh %{resource} '%{name}' (%{size})",
"shareCopyToClipboard": "Salin ke papan klip: Ctrl+C, Enter",
"remove_missing_title": "Hapus file yang hilang",
- "remove_missing_content": "Apakah Anda yakin ingin menghapus file-file yang hilang dari basis data? Tindakan ini akan menghapus secara permanen semua referensi ke file-file tersebut, termasuk jumlah pemutaran dan peringkatnya."
+ "remove_missing_content": "Apakah Anda yakin ingin menghapus file-file yang hilang dari basis data? Tindakan ini akan menghapus secara permanen semua referensi ke file-file tersebut, termasuk jumlah pemutaran dan peringkatnya.",
+ "remove_all_missing_title": "Hapus semua file yang hilang",
+ "remove_all_missing_content": "Apa kamu yakin ingin menghapus semua file dari database? Ini akan menghapus permanen dan apapun referensi ke mereka, termasuk hitungan pemutaran dan rating mereka.",
+ "noSimilarSongsFound": "Tidak ada lagu yang serupa ditemukan",
+ "noTopSongsFound": "Tidak ada lagu teratas ditemukan"
},
"menu": {
"library": "Pustaka",
@@ -448,10 +535,16 @@
"albumList": "Album",
"about": "Tentang",
"playlists": "Playlist",
- "sharedPlaylists": "Playlist yang Dibagikan"
+ "sharedPlaylists": "Playlist yang Dibagikan",
+ "librarySelector": {
+ "allLibraries": "Semua Pustaka (%{count})",
+ "multipleLibraries": "Pustaka %{selected} dari %{total}",
+ "selectLibraries": "Pilih Perpustakaan",
+ "none": "Tidak ada"
+ }
},
"player": {
- "playListsText": "Mainkan Antrean",
+ "playListsText": "Putar Antrean",
"openText": "Buka",
"closeText": "Tutup",
"notContentText": "Tidak ada musik",
@@ -471,7 +564,7 @@
"playModeText": {
"order": "Berurutan",
"orderLoop": "Ulang",
- "singleLoop": "Ulangi Satu",
+ "singleLoop": "Ulangi Sekali",
"shufflePlay": "Acak"
}
},
@@ -485,6 +578,21 @@
"disabled": "Nonaktifkan",
"waiting": "Menunggu"
}
+ },
+ "tabs": {
+ "about": "Tentang",
+ "config": "Konfigurasi"
+ },
+ "config": {
+ "configName": "Nama Konfigurasi",
+ "environmentVariable": "Variabel Environment",
+ "currentValue": "Value Saat Ini",
+ "configurationFile": "File Konfigurasi",
+ "exportToml": "Ekspor Konfigurasi (TOML)",
+ "exportSuccess": "Konfigurasi sudah diekspor ke papan klip dalam bentuk format TOML",
+ "exportFailed": "Gagal menyalin konfigurasi",
+ "devFlagsHeader": "Flag Pengembangan (subyek untuk perubahan/pemindahan)",
+ "devFlagsComment": "Ini adalan pengaturan eksperimen dan mungkin akan dihapus di versi mendatang"
}
},
"activity": {
@@ -493,7 +601,10 @@
"quickScan": "Pemindaian Cepat",
"fullScan": "Pemindaian Penuh",
"serverUptime": "Waktu Aktif Peladen",
- "serverDown": "LURING"
+ "serverDown": "LURING",
+ "scanType": "Tipe",
+ "status": "Kesalahan Memindai",
+ "elapsedTime": "Waktu Berakhir"
},
"help": {
"title": "Tombol Pintasan Navidrome",
@@ -508,5 +619,10 @@
"toggle_love": "Tambahkan lagu ini ke favorit",
"current_song": "Buka Lagu Saat Ini"
}
+ },
+ "nowPlaying": {
+ "title": "Sedang Diputar",
+ "empty": "Tidak ada yang diputar",
+ "minutesAgo": "%{smart_count} menit yang lalu |||| %{smart_count} menit yang lalu"
}
}
\ No newline at end of file
diff --git a/resources/i18n/it.json b/resources/i18n/it.json
index aaaa2f8c2..11fadb46b 100644
--- a/resources/i18n/it.json
+++ b/resources/i18n/it.json
@@ -232,7 +232,7 @@
"add_filter": "Aggiungi un filtro",
"add": "Aggiungi",
"back": "Indietro",
- "bulk_actions": "Un elemento selezionato ||| %{smart_count} elementi selezionati",
+ "bulk_actions": "Un elemento selezionato |||| %{smart_count} elementi selezionati",
"cancel": "Annulla",
"clear_input_value": "Cancella",
"clone": "Duplica",
@@ -400,8 +400,8 @@
},
"albumList": "Album",
"about": "Info",
- "playlists": "Scalette",
- "sharedPlaylists": "Scalette Condivise"
+ "playlists": "Playlist",
+ "sharedPlaylists": "Playlist Condivise"
},
"player": {
"playListsText": "Coda",
@@ -457,4 +457,4 @@
"current_song": ""
}
}
-}
\ No newline at end of file
+}
diff --git a/resources/i18n/ko.json b/resources/i18n/ko.json
index a8b26df6d..6b81e02d8 100644
--- a/resources/i18n/ko.json
+++ b/resources/i18n/ko.json
@@ -12,6 +12,7 @@
"artist": "아티스트",
"album": "앨범",
"path": "파일 경로",
+ "libraryName": "라이브러리",
"genre": "장르",
"compilation": "컴필레이션",
"year": "년",
@@ -34,7 +35,8 @@
"participants": "추가 참가자",
"tags": "추가 태그",
"mappedTags": "매핑된 태그",
- "rawTags": "원시 태그"
+ "rawTags": "원시 태그",
+ "missing": "누락"
},
"actions": {
"addToQueue": "나중에 재생",
@@ -56,6 +58,7 @@
"playCount": "재생 횟수",
"size": "크기",
"name": "이름",
+ "libraryName": "라이브러리",
"genre": "장르",
"compilation": "컴필레이션",
"year": "년",
@@ -73,7 +76,8 @@
"releaseType": "유형",
"grouping": "그룹",
"media": "미디어",
- "mood": "분위기"
+ "mood": "분위기",
+ "missing": "누락"
},
"actions": {
"playAll": "재생",
@@ -105,7 +109,8 @@
"playCount": "재생 횟수",
"rating": "평가",
"genre": "장르",
- "role": "역할"
+ "role": "역할",
+ "missing": "누락"
},
"roles": {
"albumartist": "앨범 아티스트 |||| 앨범 아티스트들",
@@ -120,7 +125,13 @@
"mixer": "믹서 |||| 믹서들",
"remixer": "리믹서 |||| 리믹서들",
"djmixer": "DJ 믹서 |||| DJ 믹서들",
- "performer": "공연자 |||| 공연자들"
+ "performer": "공연자 |||| 공연자들",
+ "maincredit": "앨범 아티스트 또는 아티스트 |||| 앨범 아티스트들 또는 아티스트들"
+ },
+ "actions": {
+ "topSongs": "인기곡",
+ "shuffle": "셔플",
+ "radio": "라디오"
}
},
"user": {
@@ -137,19 +148,26 @@
"changePassword": "비밀번호를 변경할까요?",
"currentPassword": "현재 비밀번호",
"newPassword": "새 비밀번호",
- "token": "토큰"
+ "token": "토큰",
+ "libraries": "라이브러리"
},
"helperTexts": {
- "name": "이름 변경 사항은 다음 로그인 시에만 반영됨"
+ "name": "이름 변경 사항은 다음 로그인 시에만 반영됨",
+ "libraries": "이 사용자에 대한 특정 라이브러리를 선택하거나 기본 라이브러리를 사용하려면 비움"
},
"notifications": {
"created": "사용자 생성됨",
"updated": "사용자 업데이트됨",
"deleted": "사용자 삭제됨"
},
+ "validation": {
+ "librariesRequired": "관리자가 아닌 사용자의 경우 최소한 하나의 라이브러리를 선택해야 함"
+ },
"message": {
"listenBrainzToken": "ListenBrainz 사용자 토큰을 입력하세요.",
- "clickHereForToken": "여기를 클릭하여 토큰을 얻으세요"
+ "clickHereForToken": "여기를 클릭하여 토큰을 얻으세요",
+ "selectAllLibraries": "모든 라이브러리 선택",
+ "adminAutoLibraries": "관리자 사용자는 자동으로 모든 라이브러리에 접속할 수 있음"
}
},
"player": {
@@ -192,12 +210,18 @@
"selectPlaylist": "재생목록 선택:",
"addNewPlaylist": "\"%{name}\" 만들기",
"export": "내보내기",
+ "saveQueue": "재생목록에 대기열 저장",
"makePublic": "공개 만들기",
- "makePrivate": "비공개 만들기"
+ "makePrivate": "비공개 만들기",
+ "searchOrCreate": "재생목록을 검색하거나 입력하여 새 재생목록을 만드세요...",
+ "pressEnterToCreate": "새 재생목록을 만드려면 Enter 키를 누름",
+ "removeFromSelection": "선택에서 제거"
},
"message": {
"duplicate_song": "중복된 노래 추가",
- "song_exist": "이미 재생목록에 존재하는 노래입니다. 중복을 추가할까요 아니면 건너뛸까요?"
+ "song_exist": "이미 재생목록에 존재하는 노래입니다. 중복을 추가할까요 아니면 건너뛸까요?",
+ "noPlaylistsFound": "재생목록을 찾을 수 없음",
+ "noPlaylists": "사용 가능한 재생 목록이 없음"
}
},
"radio": {
@@ -238,14 +262,68 @@
"fields": {
"path": "경로",
"size": "크기",
+ "libraryName": "라이브러리",
"updatedAt": "사라짐"
},
"actions": {
- "remove": "제거"
+ "remove": "제거",
+ "remove_all": "모두 제거"
},
"notifications": {
"removed": "누락된 파일이 제거되었음"
}
+ },
+ "library": {
+ "name": "라이브러리 |||| 라이브러리들",
+ "fields": {
+ "name": "이름",
+ "path": "경로",
+ "remotePath": "원격 경로",
+ "lastScanAt": "최근 스캔",
+ "songCount": "노래",
+ "albumCount": "앨범",
+ "artistCount": "아티스트",
+ "totalSongs": "노래",
+ "totalAlbums": "앨범",
+ "totalArtists": "아티스트",
+ "totalFolders": "폴더",
+ "totalFiles": "파일",
+ "totalMissingFiles": "누락된 파일",
+ "totalSize": "총 크기",
+ "totalDuration": "기간",
+ "defaultNewUsers": "신규 사용자 기본값",
+ "createdAt": "생성됨",
+ "updatedAt": "업데이트됨"
+ },
+ "sections": {
+ "basic": "기본 정보",
+ "statistics": "통계"
+ },
+ "actions": {
+ "scan": "라이브러리 스캔",
+ "manageUsers": "자용자 접속 관리",
+ "viewDetails": "상세 보기"
+ },
+ "notifications": {
+ "created": "라이브러리가 성공적으로 생성됨",
+ "updated": "라이브러리가 성공적으로 업데이트됨",
+ "deleted": "라이브러리가 성공적으로 삭제됨",
+ "scanStarted": "라이브러리 스캔 스작됨",
+ "scanCompleted": "라이브러리 스캔 완료됨"
+ },
+ "validation": {
+ "nameRequired": "라이브러리 이름이 필요함",
+ "pathRequired": "라이브러리 경로가 필요함",
+ "pathNotDirectory": "라이브러리 경로는 디렉터리여야 함",
+ "pathNotFound": "라이브러리 경로를 찾을 수 없음",
+ "pathNotAccessible": "라이브러리 경로에 접근할 수 없음",
+ "pathInvalid": "잘못된 라이브러리 경로"
+ },
+ "messages": {
+ "deleteConfirm": "이 라이브러리를 삭제할까요? 삭제하면 연결된 모든 데이터와 사용자 접속 권한이 제거됩니다.",
+ "scanInProgress": "스캔 진행 중...",
+ "noLibrariesAssigned": "이 사용자에게 할당된 라이브러리가 없음"
+ }
}
},
"ra": {
@@ -398,11 +476,15 @@
"transcodingDisabled": "웹 인터페이스를 통한 트랜스코딩 구성 변경은 보안상의 이유로 비활성화되어 있습니다. 트랜스코딩 옵션을 변경(편집 또는 추가)하려면, %{config} 구성 옵션으로 서버를 다시 시작하세요.",
"transcodingEnabled": "Navidrome은 현재 %{config}로 실행 중이므로 웹 인터페이스를 사용하여 트랜스코딩 설정에서 시스템 명령을 실행할 수 있습니다. 보안상의 이유로 비활성화하고 트랜스코딩 옵션을 구성할 때만 활성화하는 것이 좋습니다.",
"songsAddedToPlaylist": "1 개의 노래를 재생목록에 추가하였음 |||| %{smart_count} 개의 노래를 재생 목록에 추가하였음",
+ "noSimilarSongsFound": "비슷한 노래를 찾을 수 없음",
+ "noTopSongsFound": "인기곡을 찾을 수 없음",
"noPlaylistsAvailable": "사용 가능한 노래 없음",
"delete_user_title": "사용자 '%{name}' 삭제",
"delete_user_content": "이 사용자와 해당 사용자의 모든 데이터(재생 목록 및 환경 설정 포함)를 삭제할까요?",
"remove_missing_title": "누락된 파일들 제거",
"remove_missing_content": "선택한 누락된 파일을 데이터베이스에서 삭제할까요? 삭제하면 재생 횟수 및 평점을 포함하여 해당 파일에 대한 모든 참조가 영구적으로 삭제됩니다.",
+ "remove_all_missing_title": "누락된 모든 파일 제거",
+ "remove_all_missing_content": "데이터베이스에서 누락된 모든 파일을 제거할까요? 이렇게 하면 해당 게임의 플레이 횟수와 평점을 포함한 모든 참조 내용이 영구적으로 삭제됩니다.",
"notifications_blocked": "브라우저 설정에서 이 사이트의 알림을 차단하였음",
"notifications_not_available": "이 브라우저는 데스크톱 알림을 지원하지 않거나 https를 통해 Navidrome에 접속하고 있지 않음",
"lastfmLinkSuccess": "Last.fm이 성공적으로 연결되었고 스크로블링이 활성화되었음",
@@ -429,6 +511,12 @@
},
"menu": {
"library": "라이브러리",
+ "librarySelector": {
+ "allLibraries": "모든 라이브러리 (%{count})",
+ "multipleLibraries": "%{selected} / %{total} 라이브러리",
+ "selectLibraries": "라이브러리 선택",
+ "none": "없음"
+ },
"settings": "설정",
"version": "버전",
"theme": "테마",
@@ -491,6 +579,21 @@
"disabled": "비활성화",
"waiting": "대기중"
}
+ },
+ "tabs": {
+ "about": "정보",
+ "config": "구성"
+ },
+ "config": {
+ "configName": "구성 이름",
+ "environmentVariable": "환경 변수",
+ "currentValue": "현재 값",
+ "configurationFile": "구성 파일",
+ "exportToml": "구성 내보내기 (TOML)",
+ "exportSuccess": "TOML 형식으로 클립보드로 내보낸 구성",
+ "exportFailed": "구성 복사 실패",
+ "devFlagsHeader": "개발 플래그 (변경/삭제 가능)",
+ "devFlagsComment": "이는 실험적 설정이므로 향후 버전에서 제거될 수 있음"
}
},
"activity": {
@@ -499,7 +602,15 @@
"quickScan": "빠른 스캔",
"fullScan": "전체 스캔",
"serverUptime": "서버 가동 시간",
- "serverDown": "오프라인"
+ "serverDown": "오프라인",
+ "scanType": "유형",
+ "status": "스캔 오류",
+ "elapsedTime": "경과 시간"
+ },
+ "nowPlaying": {
+ "title": "현재 재생 중",
+ "empty": "재생 중인 콘텐츠 없음",
+ "minutesAgo": "%{smart_count} 분 전"
},
"help": {
"title": "Navidrome 단축키",
diff --git a/resources/i18n/nl.json b/resources/i18n/nl.json
index 07e222e2a..b6da47380 100644
--- a/resources/i18n/nl.json
+++ b/resources/i18n/nl.json
@@ -5,7 +5,7 @@
"name": "Nummer |||| Nummers",
"fields": {
"albumArtist": "Album Artiest",
- "duration": "Lengte",
+ "duration": "Afspeelduur",
"trackNumber": "Nummer #",
"playCount": "Aantal keren afgespeeld",
"title": "Titel",
@@ -26,7 +26,17 @@
"bpm": "BPM",
"playDate": "Laatst afgespeeld",
"channels": "Kanalen",
- "createdAt": "Datum toegevoegd"
+ "createdAt": "Datum toegevoegd",
+ "grouping": "Groep",
+ "mood": "Sfeer",
+ "participants": "Extra deelnemers",
+ "tags": "Extra tags",
+ "mappedTags": "Gemapte tags",
+ "rawTags": "Onbewerkte tags",
+ "bitDepth": "Bit diepte",
+ "sampleRate": "Sample waarde",
+ "missing": "Ontbrekend",
+ "libraryName": "Bibliotheek"
},
"actions": {
"addToQueue": "Voeg toe aan wachtrij",
@@ -35,7 +45,8 @@
"shuffleAll": "Shuffle alles",
"download": "Downloaden",
"playNext": "Volgende",
- "info": "Meer info"
+ "info": "Meer info",
+ "showInPlaylist": "Toon in afspeellijst"
}
},
"album": {
@@ -46,7 +57,7 @@
"duration": "Afspeelduur",
"songCount": "Nummers",
"playCount": "Aantal keren afgespeeld",
- "name": "Naam",
+ "name": "Titel",
"genre": "Genre",
"compilation": "Compilatie",
"year": "Jaar",
@@ -56,9 +67,18 @@
"createdAt": "Datum toegevoegd",
"size": "Grootte",
"originalDate": "Origineel",
- "releaseDate": "Uitgegeven",
+ "releaseDate": "Uitgave",
"releases": "Uitgave |||| Uitgaven",
- "released": "Uitgegeven"
+ "released": "Uitgave",
+ "recordLabel": "Label",
+ "catalogNum": "Catalogus nummer",
+ "releaseType": "Type",
+ "grouping": "Groep",
+ "media": "Media",
+ "mood": "Sfeer",
+ "date": "Opnamedatum",
+ "missing": "Ontbrekend",
+ "libraryName": "Bibliotheek"
},
"actions": {
"playAll": "Afspelen",
@@ -89,7 +109,30 @@
"playCount": "Afgespeeld",
"rating": "Beoordeling",
"genre": "Genre",
- "size": "Grootte"
+ "size": "Grootte",
+ "role": "Rol",
+ "missing": "Ontbrekend"
+ },
+ "roles": {
+ "albumartist": "Album artiest |||| Album artiesten",
+ "artist": "Artiest |||| Artiesten",
+ "composer": "Componist |||| Componisten",
+ "conductor": "Dirigent |||| Dirigenten",
+ "lyricist": "Tekstschrijver |||| Tekstschrijvers",
+ "arranger": "Arrangeur |||| Arrangeurs",
+ "producer": "Producent |||| Producenten",
+ "director": "Regisseur |||| Regisseurs",
+ "engineer": "Opnametechnicus |||| Opnametechnici",
+ "mixer": "Mixer |||| Mixers",
+ "remixer": "Remixer |||| Remixers",
+ "djmixer": "DJ Mixer |||| DJ Mixers",
+ "performer": "Performer |||| Performers",
+ "maincredit": "Album Artiest of Artiest |||| Album Artiesten or Artiesten"
+ },
+ "actions": {
+ "shuffle": "Shuffle",
+ "radio": "Radio",
+ "topSongs": "Beste nummers"
}
},
"user": {
@@ -98,7 +141,7 @@
"userName": "Gebruikersnaam",
"isAdmin": "Is beheerder",
"lastLoginAt": "Laatst ingelogd op",
- "updatedAt": "Laatst gewijzigd op",
+ "updatedAt": "Laatst bijgewerkt op",
"name": "Naam",
"password": "Wachtwoord",
"createdAt": "Aangemaakt op",
@@ -106,19 +149,26 @@
"currentPassword": "Huidig wachtwoord",
"newPassword": "Nieuw wachtwoord",
"token": "Token",
- "lastAccessAt": "Meest recente toegang"
+ "lastAccessAt": "Meest recente toegang",
+ "libraries": "Bibliotheken"
},
"helperTexts": {
- "name": "Naamswijziging wordt pas zichtbaar bij de volgende login"
+ "name": "Naamswijziging wordt pas zichtbaar bij de volgende login",
+ "libraries": "Selecteer specifieke bibliotheken voor deze gebruiker, of laat leeg om de standaardbiblliotheken te gebruiken"
},
"notifications": {
"created": "Aangemaakt door gebruiker",
- "updated": "Gewijzigd door gebruiker",
- "deleted": "Gewist door gebruiker"
+ "updated": "Bijgewerkt door gebruiker",
+ "deleted": "Gebruiker verwijderd"
},
"message": {
"listenBrainzToken": "Vul je ListenBrainz gebruikers-token in.",
- "clickHereForToken": "Klik hier voor je token"
+ "clickHereForToken": "Klik hier voor je token",
+ "selectAllLibraries": "Selecteer alle bibliotheken",
+ "adminAutoLibraries": "Admin gebruikers hebben automatisch toegang tot alle bibliotheken"
+ },
+ "validation": {
+ "librariesRequired": "Minstens één bibliotheek moet geselecteerd worden voor niet-admin gebruikers"
}
},
"player": {
@@ -147,10 +197,10 @@
"name": "Afspeellijst |||| Afspeellijsten",
"fields": {
"name": "Titel",
- "duration": "Lengte",
+ "duration": "Afspeelduur",
"ownerName": "Eigenaar",
"public": "Publiek",
- "updatedAt": "Laatst gewijzigd op",
+ "updatedAt": "Laatst bijgewerkt op",
"createdAt": "Aangemaakt op",
"songCount": "Nummers",
"comment": "Commentaar",
@@ -162,11 +212,17 @@
"addNewPlaylist": "Creëer \"%{name}\"",
"export": "Exporteer",
"makePublic": "Openbaar maken",
- "makePrivate": "Privé maken"
+ "makePrivate": "Privé maken",
+ "saveQueue": "Bewaar wachtrij als playlist",
+ "searchOrCreate": "Zoek afspeellijsten of typ om een nieuwe te starten...",
+ "pressEnterToCreate": "Druk Enter om nieuwe afspeellijst te maken",
+ "removeFromSelection": "Verwijder van selectie"
},
"message": {
"duplicate_song": "Dubbele nummers toevoegen",
- "song_exist": "Er komen nummers dubbel in de afspeellijst. Wil je de dubbele nummers toevoegen of overslaan?"
+ "song_exist": "Er komen nummers dubbel in de afspeellijst. Wil je de dubbele nummers toevoegen of overslaan?",
+ "noPlaylistsFound": "Geen playlists gevonden",
+ "noPlaylists": "Geen playlists beschikbaar"
}
},
"radio": {
@@ -175,8 +231,8 @@
"name": "Naam",
"streamUrl": "Stream URL",
"homePageUrl": "Hoofdpagina URL",
- "updatedAt": "Geüpdate op",
- "createdAt": "Gecreëerd op"
+ "updatedAt": "Bijgewerkt op",
+ "createdAt": "Aangemaakt op"
},
"actions": {
"playNow": "Speel nu"
@@ -194,10 +250,79 @@
"visitCount": "Bezocht",
"format": "Formaat",
"maxBitRate": "Max. bitrate",
- "updatedAt": "Geüpdatet op",
- "createdAt": "Gecreëerd op",
+ "updatedAt": "Bijgewerkt op",
+ "createdAt": "Aangemaakt op",
"downloadable": "Downloads toestaan?"
}
+ },
+ "missing": {
+ "name": "Ontbrekend bestand |||| Ontbrekende bestanden",
+ "fields": {
+ "path": "Pad",
+ "size": "Grootte",
+ "updatedAt": "Verdwenen op",
+ "libraryName": "Bibliotheek"
+ },
+ "actions": {
+ "remove": "Verwijder",
+ "remove_all": "Alles verwijderen"
+ },
+ "notifications": {
+ "removed": "Ontbrekende bestanden verwijderd"
+ },
+ "empty": "Geen ontbrekende bestanden"
+ },
+ "library": {
+ "name": "Bibliotheek |||| Bibliotheken",
+ "fields": {
+ "name": "Naam",
+ "path": "Pad",
+ "remotePath": "Extern pad",
+ "lastScanAt": "Laatste scan",
+ "songCount": "Nummers",
+ "albumCount": "Albums",
+ "artistCount": "Artiesten",
+ "totalSongs": "Nummers",
+ "totalAlbums": "Albums",
+ "totalArtists": "Artiesten",
+ "totalFolders": "Mappen",
+ "totalFiles": "Bestanden",
+ "totalMissingFiles": "Ontbrekende bestanden",
+ "totalSize": "Totale bestandsgrootte",
+ "totalDuration": "Afspeelduur",
+ "defaultNewUsers": "Standaard voor nieuwe gebruikers",
+ "createdAt": "Aangemaakt",
+ "updatedAt": "Bijgewerkt"
+ },
+ "sections": {
+ "basic": "Basisinformatie",
+ "statistics": "Statistieken"
+ },
+ "actions": {
+ "scan": "Scan bibliotheek",
+ "manageUsers": "Beheer gebruikerstoegang",
+ "viewDetails": "Bekijk details"
+ },
+ "notifications": {
+ "created": "Bibliotheek succesvol aangemaakt",
+ "updated": "Bibliotheek succesvol bijgewerkt",
+ "deleted": "Bibliotheek succesvol verwijderd",
+ "scanStarted": "Bibliotheekscan is gestart",
+ "scanCompleted": "Bibliotheekscan is voltooid"
+ },
+ "validation": {
+ "nameRequired": "Bibliotheek naam is vereist",
+ "pathRequired": "Pad naar bibliotheek is vereist",
+ "pathNotDirectory": "Pad naar bibliotheek moet een map zijn",
+ "pathNotFound": "Pad naar bibliotheek niet gevonden",
+ "pathNotAccessible": "Pad naar bibliotheek is niet toegankelijk",
+ "pathInvalid": "Ongeldig pad naar bibliotheek"
+ },
+ "messages": {
+ "deleteConfirm": "Weet je zeker dat je deze bibliotheek wil verwijderen? Dit verwijdert ook alle gerelateerde data en gebruikerstoegang.",
+ "scanInProgress": "Scan is bezig...",
+ "noLibrariesAssigned": "Geen bibliotheken aan deze gebruiker toegewezen"
+ }
}
},
"ra": {
@@ -212,7 +337,8 @@
"password": "Wachtwoord",
"sign_in": "Inloggen",
"sign_in_error": "Authenticatie mislukt, probeer opnieuw a.u.b.",
- "logout": "Uitloggen"
+ "logout": "Uitloggen",
+ "insightsCollectionNote": "Navidrome verzamelt anonieme gebruiksdata om het project te verbeteren. Klik [hier] voor meer info en de mogelijkheid om te weigeren"
},
"validation": {
"invalidChars": "Gebruik alleen letters en cijfers",
@@ -374,7 +500,13 @@
"shareSuccess": "URL gekopieeerd naar klembord: %{url}",
"shareFailure": "Fout bij kopieren URL %{url} naar klembord",
"downloadDialogTitle": "Download %{resource} '%{name}' (%{size})",
- "shareCopyToClipboard": "Kopieeer naar klembord: Ctrl+C, Enter"
+ "shareCopyToClipboard": "Kopieeer naar klembord: Ctrl+C, Enter",
+ "remove_missing_title": "Verwijder ontbrekende bestanden",
+ "remove_missing_content": "Weet je zeker dat je alle ontbrekende bestanden van de database wil verwijderen? Dit wist permanent al hun referenties inclusief afspeel tellers en beoordelingen.",
+ "remove_all_missing_title": "Verwijder alle ontbrekende bestanden",
+ "remove_all_missing_content": "Weet je zeker dat je alle ontbrekende bestanden van de database wil verwijderen? Dit wist permanent al hun referenties inclusief afspeel tellers en beoordelingen.",
+ "noSimilarSongsFound": "Geen vergelijkbare nummers gevonden",
+ "noTopSongsFound": "Geen beste nummers gevonden"
},
"menu": {
"library": "Bibliotheek",
@@ -396,22 +528,29 @@
"none": "Uitgeschakeld",
"album": "Gebruik Album Gain",
"track": "Gebruik Track Gain"
- }
+ },
+ "lastfmNotConfigured": "Last.fm API-sleutel is niet geconfigureerd"
}
},
"albumList": "Albums",
"about": "Over",
- "playlists": "Playlists",
- "sharedPlaylists": "Gedeelde playlists"
+ "playlists": "Afspeellijsten",
+ "sharedPlaylists": "Gedeelde afspeellijsten",
+ "librarySelector": {
+ "allLibraries": "Alle bibliotheken (%{count})",
+ "multipleLibraries": "%{selected} van %{total} bibliotheken",
+ "selectLibraries": "Selecteer bibliotheken",
+ "none": "Geen"
+ }
},
"player": {
- "playListsText": "Afspeellijst afspelen",
+ "playListsText": "Wachtrij",
"openText": "Openen",
"closeText": "Sluiten",
"notContentText": "Geen muziek",
"clickToPlayText": "Klik om af te spelen",
"clickToPauseText": "Klik om te pauzeren",
- "nextTrackText": "Volgende",
+ "nextTrackText": "Volgend nummer",
"previousTrackText": "Vorige",
"reloadText": "Herladen",
"volumeText": "Volume",
@@ -433,16 +572,39 @@
"links": {
"homepage": "Thuispagina",
"source": "Broncode",
- "featureRequests": "Functie verzoeken"
+ "featureRequests": "Functie verzoeken",
+ "lastInsightsCollection": "Laatste inzichten",
+ "insights": {
+ "disabled": "Uitgeschakeld",
+ "waiting": "Wachten"
+ }
+ },
+ "tabs": {
+ "about": "Over",
+ "config": "Configuratie"
+ },
+ "config": {
+ "configName": "Config Naam",
+ "environmentVariable": "Omgevingsvariabele",
+ "currentValue": "Huidige waarde",
+ "configurationFile": "Configuratiebestand",
+ "exportToml": "Exporteer configuratie (TOML)",
+ "exportSuccess": "Configuratie geëxporteerd naar klembord in TOML formaat",
+ "exportFailed": "Kopiëren van configuratie mislukt",
+ "devFlagsHeader": "Ontwikkelaarsinstellingen (onder voorbehoud)",
+ "devFlagsComment": "Dit zijn experimentele instellingen en worden mogelijk in latere versies verwijderd"
}
},
"activity": {
"title": "Activiteit",
- "totalScanned": "Totaal gescande folders",
+ "totalScanned": "Totaal gescande mappen",
"quickScan": "Snelle scan",
"fullScan": "Volledige scan",
"serverUptime": "Server uptime",
- "serverDown": "Offline"
+ "serverDown": "Offline",
+ "scanType": "Type",
+ "status": "Scan fout",
+ "elapsedTime": "Verlopen tijd"
},
"help": {
"title": "Navidrome sneltoetsen",
@@ -457,5 +619,10 @@
"toggle_love": "Voeg toe aan favorieten",
"current_song": "Ga naar huidig nummer"
}
+ },
+ "nowPlaying": {
+ "title": "Speelt nu",
+ "empty": "Er wordt niets afgespeed",
+ "minutesAgo": "%{smart_count} minuut geleden |||| %{smart_count} minuten geleden"
}
}
\ No newline at end of file
diff --git a/resources/i18n/pl.json b/resources/i18n/pl.json
index e75a9f404..4d78c7599 100644
--- a/resources/i18n/pl.json
+++ b/resources/i18n/pl.json
@@ -33,7 +33,10 @@
"tags": "Dodatkowe Tagi",
"mappedTags": "Zmapowane tagi",
"rawTags": "Surowe tagi",
- "bitDepth": "Głębokość próbkowania"
+ "bitDepth": "Głębokość próbkowania",
+ "sampleRate": "Częstotliwość próbkowania",
+ "missing": "Brak",
+ "libraryName": "Biblioteka"
},
"actions": {
"addToQueue": "Odtwarzaj Później",
@@ -42,7 +45,8 @@
"shuffleAll": "Losuj Wszystkie",
"download": "Pobierz",
"playNext": "Odtwarzaj Następny",
- "info": "Zdobądź Informacje"
+ "info": "Zdobądź Informacje",
+ "showInPlaylist": "Pokaż w Liście Odtwarzania"
}
},
"album": {
@@ -72,7 +76,9 @@
"grouping": "Grupowanie",
"media": "Media",
"mood": "Nastrój",
- "date": ""
+ "date": "Data Nagrania",
+ "missing": "Brak",
+ "libraryName": "Biblioteka"
},
"actions": {
"playAll": "Odtwarzaj",
@@ -104,7 +110,8 @@
"rating": "Ocena",
"genre": "Gatunek",
"size": "Rozmiar",
- "role": "Rola"
+ "role": "Rola",
+ "missing": "Brak"
},
"roles": {
"albumartist": "Wykonawca Albumu |||| Wykonawcy Albumu",
@@ -119,7 +126,13 @@
"mixer": "Mikser |||| Mikserzy",
"remixer": "Remixer |||| Remixerzy",
"djmixer": "Didżej |||| Didżerzy",
- "performer": "Wykonawca |||| Wykonawcy"
+ "performer": "Wykonawca |||| Wykonawcy",
+ "maincredit": "Artysta albumu lub Artysta |||| Artyści albumu lub Artyści"
+ },
+ "actions": {
+ "shuffle": "Losuj",
+ "radio": "Radio",
+ "topSongs": "Najlepsze Utwory"
}
},
"user": {
@@ -136,10 +149,12 @@
"currentPassword": "Obecne hasło",
"newPassword": "Nowe hasło",
"token": "Token",
- "lastAccessAt": "Ostatnia Aktywność"
+ "lastAccessAt": "Ostatnia Aktywność",
+ "libraries": "Biblioteki"
},
"helperTexts": {
- "name": "Zmiana nazwy będzie widoczna przy następnym logowaniu"
+ "name": "Zmiana nazwy będzie widoczna przy następnym logowaniu",
+ "libraries": "Wybierz biblioteki dla użytkownika lub pozostaw pustę, aby użyć domyślnej biblioteki"
},
"notifications": {
"created": "Dodano użytkownika",
@@ -148,7 +163,12 @@
},
"message": {
"listenBrainzToken": "Wprowadź swój token ListenBrainz.",
- "clickHereForToken": "Kliknij tutaj, aby uzyskać token"
+ "clickHereForToken": "Kliknij tutaj, aby uzyskać token",
+ "selectAllLibraries": "Wybierz wszystkie biblioteki",
+ "adminAutoLibraries": "Administratorzy automatycznie mają dostęp do wszystkich bibliotek"
+ },
+ "validation": {
+ "librariesRequired": "Przynajmniej jedna biblioteka musi być wybrana dla zwykłego użytkownika"
}
},
"player": {
@@ -192,11 +212,17 @@
"addNewPlaylist": "Stwórz \"%{name}\"",
"export": "Wyeksportuj",
"makePublic": "Zmień na Publiczną",
- "makePrivate": "Zmień na Prywatną"
+ "makePrivate": "Zmień na Prywatną",
+ "saveQueue": "Zapisz Kolejkę do Playlisty",
+ "searchOrCreate": "Szukaj list odtwarzania lub zacznij pisać, aby stworzyć nową...",
+ "pressEnterToCreate": "Wciśnij Enter, aby stworzyć nową listę odtwarzania",
+ "removeFromSelection": "Usuń z zaznaczenia"
},
"message": {
"duplicate_song": "Dodaj zduplikowane utwory",
- "song_exist": "Do playlisty dodawane są duplikaty. Czy chcesz je dodać czy pominąć?"
+ "song_exist": "Do playlisty dodawane są duplikaty. Czy chcesz je dodać czy pominąć?",
+ "noPlaylistsFound": "Brak list odtwarzania",
+ "noPlaylists": "Brak dostępnych list odtwarzania"
}
},
"radio": {
@@ -234,15 +260,69 @@
"fields": {
"path": "Ścieżka",
"size": "Rozmiar",
- "updatedAt": "Zniknął na"
+ "updatedAt": "Zniknął na",
+ "libraryName": "Biblioteka"
},
"actions": {
- "remove": "Usuń"
+ "remove": "Usuń",
+ "remove_all": "Usuń Wszystko"
},
"notifications": {
"removed": "Usunięto brakujące pliki"
},
- "empty": "Bez Brakujących Plików"
+ "empty": "Brak Brakujących Plików"
+ },
+ "library": {
+ "name": "Biblioteka |||| Biblioteki",
+ "fields": {
+ "name": "Nazwa",
+ "path": "Ścieżka",
+ "remotePath": "Zdalna Ścieżka",
+ "lastScanAt": "Ostatni Skan",
+ "songCount": "Utwory",
+ "albumCount": "Albumy",
+ "artistCount": "Artyści",
+ "totalSongs": "Utwory",
+ "totalAlbums": "Albumy",
+ "totalArtists": "Artyści",
+ "totalFolders": "Foldery",
+ "totalFiles": "Pliki",
+ "totalMissingFiles": "Brakujące Pliki",
+ "totalSize": "Całkowity Rozmiar",
+ "totalDuration": "Czas Trwania",
+ "defaultNewUsers": "Domyślne dla Nowych Użytkowników",
+ "createdAt": "Stworzona",
+ "updatedAt": "Zaktualizowana"
+ },
+ "sections": {
+ "basic": "Podstawowe Informacje",
+ "statistics": "Statystyki"
+ },
+ "actions": {
+ "scan": "Skanuj Bibliotekę",
+ "manageUsers": "Zarządzaj Dostępami Użytkownika",
+ "viewDetails": "Zobacz Szczegóły"
+ },
+ "notifications": {
+ "created": "Biblioteka utworzona prawidłowo",
+ "updated": "Biblioteka zaktualizowana prawidłowo",
+ "deleted": "Biblioteka usunięta prawidłowo",
+ "scanStarted": "Rozpoczęto skan biblioteki",
+ "scanCompleted": "Zakończono skan biblioteki"
+ },
+ "validation": {
+ "nameRequired": "Nazwa biblioteki jest wymagana",
+ "pathRequired": "Ścieżka biblioteki jest wymagana",
+ "pathNotDirectory": "Ścieżka biblioteki musi być katalogiem",
+ "pathNotFound": "Brak ścieżki biblioteki",
+ "pathNotAccessible": "Ścieżka biblioteki niedostępna",
+ "pathInvalid": "Niepoprawna ścieżka biblioteki"
+ },
+ "messages": {
+ "deleteConfirm": "Czy chcesz usunąć tę bibliotekę? Spowoduje to usunięcie wszystkich powiązanych danych i dostępów użytkowników.",
+ "scanInProgress": "Skanowanie w trakcie...",
+ "noLibrariesAssigned": "Brak bibliotek przypisanych do tego użytkownika"
+ }
}
},
"ra": {
@@ -422,7 +502,11 @@
"downloadDialogTitle": "Pobierz %{resource} '%{name}' (%{size})",
"shareCopyToClipboard": "Skopiuj do schowka: Ctrl+C, Enter",
"remove_missing_title": "Usuń brakujące dane",
- "remove_missing_content": "Czy na pewno chcesz usunąć wybrane brakujące pliki z bazy danych? Spowoduje to trwałe usunięcie wszystkich powiązań, takich jak liczba odtworzeń i oceny."
+ "remove_missing_content": "Czy na pewno chcesz usunąć wybrane brakujące pliki z bazy danych? Spowoduje to trwałe usunięcie wszystkich powiązań, takich jak liczba odtworzeń i oceny.",
+ "remove_all_missing_title": "Usuń wszystkie brakujące pliki",
+ "remove_all_missing_content": "Czy chcesz usunąć wszystkie brakujące pliki z bazy danych? Spowoduje to trwałe usunięcie wszelkich odniesień do tych plików, takich jak liczba odtworzeń, czy oceny.",
+ "noSimilarSongsFound": "Brak podobnych utworów",
+ "noTopSongsFound": "Brak najlepszych utworów"
},
"menu": {
"library": "Biblioteka",
@@ -451,7 +535,13 @@
"albumList": "Albumy",
"about": "O aplikacji",
"playlists": "Playlisty",
- "sharedPlaylists": "Udostępnione Playlisty"
+ "sharedPlaylists": "Udostępnione Playlisty",
+ "librarySelector": {
+ "allLibraries": "Wszystkie Biblioteki (%{count})",
+ "multipleLibraries": "%{selected} z %{total} Bibliotek",
+ "selectLibraries": "Wybierz Biblioteki",
+ "none": "Żadna"
+ }
},
"player": {
"playListsText": "Kolejka Odtwarzania",
@@ -488,6 +578,21 @@
"disabled": "Wyłączone",
"waiting": "Oczekujące"
}
+ },
+ "tabs": {
+ "about": "O",
+ "config": "Konfiguracja"
+ },
+ "config": {
+ "configName": "Nazwa Konfiguracji",
+ "environmentVariable": "Zmienna Środowiskowa",
+ "currentValue": "Obecna Wartość",
+ "configurationFile": "Plik Konfiguracyjny",
+ "exportToml": "Eksportuj Konfigurację (TOML)",
+ "exportSuccess": "Konfiguracja wyeksportowana do schowka w formacie TOML",
+ "exportFailed": "Błąd kopiowania konfiguracji",
+ "devFlagsHeader": "Flagi Rozwojowe (mogą ulec zmianie/usunięciu)",
+ "devFlagsComment": "To są ustawienia eksperymentalne i mogą zostać usunięte w przyszłych wydaniach"
}
},
"activity": {
@@ -496,7 +601,10 @@
"quickScan": "Szybkie Skanowanie",
"fullScan": "Pełne Skanowanie",
"serverUptime": "Czas Działania Serwera",
- "serverDown": "NIEDOSTĘPNY"
+ "serverDown": "NIEDOSTĘPNY",
+ "scanType": "Typ",
+ "status": "Błąd Skanowania",
+ "elapsedTime": "Upłynięty Czas"
},
"help": {
"title": "Skróty Klawiszowe Navidrome",
@@ -511,5 +619,10 @@
"toggle_love": "Dodaj ten utwór do ulubionych",
"current_song": "Przejdź do Bieżącego Utworu"
}
+ },
+ "nowPlaying": {
+ "title": "Teraz Odtwarzane",
+ "empty": "Nic nie jest odtwarzane",
+ "minutesAgo": "%{smart_count} minutę temu |||| %{smart_count} minut temu"
}
}
\ No newline at end of file
diff --git a/resources/i18n/pt-br.json b/resources/i18n/pt-br.json
index 3faa1b149..3f095b025 100644
--- a/resources/i18n/pt-br.json
+++ b/resources/i18n/pt-br.json
@@ -12,13 +12,13 @@
"artist": "Artista",
"album": "Álbum",
"path": "Arquivo",
+ "libraryName": "Biblioteca",
"genre": "Gênero",
"compilation": "Coletânea",
"year": "Ano",
"size": "Tamanho",
"updatedAt": "Últ. Atualização",
"bitRate": "Bitrate",
- "bitDepth": "Profundidade de bits",
"discSubtitle": "Sub-título do disco",
"starred": "Favorita",
"comment": "Comentário",
@@ -34,6 +34,8 @@
"tags": "Outras Tags",
"mappedTags": "Tags mapeadas",
"rawTags": "Tags originais",
+ "bitDepth": "Profundidade de bits",
+ "sampleRate": "Taxa de amostragem",
"missing": "Ausente"
},
"actions": {
@@ -43,7 +45,8 @@
"shuffleAll": "Aleatório",
"download": "Baixar",
"playNext": "Toca a seguir",
- "info": "Detalhes"
+ "info": "Detalhes",
+ "showInPlaylist": "Ir para playlist"
}
},
"album": {
@@ -55,10 +58,10 @@
"songCount": "Músicas",
"playCount": "Execuções",
"name": "Nome",
+ "libraryName": "Biblioteca",
"genre": "Gênero",
"compilation": "Coletânea",
"year": "Ano",
- "date": "Data de Lançamento",
"updatedAt": "Últ. Atualização",
"comment": "Comentário",
"rating": "Classificação",
@@ -74,6 +77,7 @@
"grouping": "Agrupamento",
"media": "Mídia",
"mood": "Mood",
+ "date": "Data de Lançamento",
"missing": "Ausente"
},
"actions": {
@@ -122,7 +126,13 @@
"mixer": "Mixador |||| Mixadores",
"remixer": "Remixador |||| Remixadores",
"djmixer": "DJ Mixer |||| DJ Mixers",
- "performer": "Músico |||| Músicos"
+ "performer": "Músico |||| Músicos",
+ "maincredit": "Artista do Álbum ou Artista |||| Artistas do Álbum ou Artistas"
+ },
+ "actions": {
+ "topSongs": "Mais tocadas",
+ "shuffle": "Aleatório",
+ "radio": "Rádio"
}
},
"user": {
@@ -139,19 +149,26 @@
"currentPassword": "Senha Atual",
"newPassword": "Nova Senha",
"token": "Token",
- "lastAccessAt": "Últ. Acesso"
+ "lastAccessAt": "Últ. Acesso",
+ "libraries": "Bibliotecas"
},
"helperTexts": {
- "name": "Alterações no seu nome só serão refletidas no próximo login"
+ "name": "Alterações no seu nome só serão refletidas no próximo login",
+ "libraries": "Selecione bibliotecas específicas para este usuário, ou deixe vazio para usar bibliotecas padrão"
},
"notifications": {
"created": "Novo usuário criado",
"updated": "Usuário atualizado com sucesso",
"deleted": "Usuário deletado com sucesso"
},
+ "validation": {
+ "librariesRequired": "Pelo menos uma biblioteca deve ser selecionada para usuários não-administradores"
+ },
"message": {
"listenBrainzToken": "Entre seu token do ListenBrainz",
- "clickHereForToken": "Clique aqui para obter seu token"
+ "clickHereForToken": "Clique aqui para obter seu token",
+ "selectAllLibraries": "Selecionar todas as bibliotecas",
+ "adminAutoLibraries": "Usuários administradores têm acesso automático a todas as bibliotecas"
}
},
"player": {
@@ -194,13 +211,18 @@
"selectPlaylist": "Selecione a playlist:",
"addNewPlaylist": "Criar \"%{name}\"",
"export": "Exportar",
- "saveQueue": "Salvar fila em nova Playlist",
"makePublic": "Pública",
- "makePrivate": "Pessoal"
+ "makePrivate": "Pessoal",
+ "saveQueue": "Salvar fila em nova Playlist",
+ "searchOrCreate": "Buscar playlists ou criar nova...",
+ "pressEnterToCreate": "Pressione Enter para criar nova playlist",
+ "removeFromSelection": "Remover da seleção"
},
"message": {
"duplicate_song": "Adicionar músicas duplicadas",
- "song_exist": "Algumas destas músicas já existem na playlist. Você quer adicionar as duplicadas ou ignorá-las?"
+ "song_exist": "Algumas destas músicas já existem na playlist. Você quer adicionar as duplicadas ou ignorá-las?",
+ "noPlaylistsFound": "Nenhuma playlist encontrada",
+ "noPlaylists": "Nenhuma playlist disponível"
}
},
"radio": {
@@ -235,10 +257,10 @@
},
"missing": {
"name": "Arquivo ausente |||| Arquivos ausentes",
- "empty": "Nenhum arquivo ausente",
"fields": {
"path": "Caminho",
"size": "Tamanho",
+ "libraryName": "Biblioteca",
"updatedAt": "Desaparecido em"
},
"actions": {
@@ -247,6 +269,64 @@
},
"notifications": {
"removed": "Arquivo(s) ausente(s) removido(s)"
+ },
+ "empty": "Nenhum arquivo ausente"
+ },
+ "library": {
+ "name": "Biblioteca |||| Bibliotecas",
+ "fields": {
+ "name": "Nome",
+ "path": "Caminho",
+ "remotePath": "Caminho Remoto",
+ "lastScanAt": "Último Scan",
+ "songCount": "Músicas",
+ "albumCount": "Álbuns",
+ "artistCount": "Artistas",
+ "totalSongs": "Músicas",
+ "totalAlbums": "Álbuns",
+ "totalArtists": "Artistas",
+ "totalFolders": "Pastas",
+ "totalFiles": "Arquivos",
+ "totalMissingFiles": "Arquivos Ausentes",
+ "totalSize": "Tamanho Total",
+ "totalDuration": "Duração",
+ "defaultNewUsers": "Padrão para Novos Usuários",
+ "createdAt": "Data de Criação",
+ "updatedAt": "Últ. Atualização"
+ },
+ "sections": {
+ "basic": "Informações Básicas",
+ "statistics": "Estatísticas"
+ },
+ "actions": {
+ "scan": "Scanear Biblioteca",
+ "quickScan": "Scan Rápido",
+ "fullScan": "Scan Completo",
+ "manageUsers": "Gerenciar Acesso do Usuário",
+ "viewDetails": "Ver Detalhes"
+ },
+ "notifications": {
+ "created": "Biblioteca criada com sucesso",
+ "updated": "Biblioteca atualizada com sucesso",
+ "deleted": "Biblioteca excluída com sucesso",
+ "scanStarted": "Scan da biblioteca iniciada",
+ "quickScanStarted": "Scan rápido iniciado",
+ "fullScanStarted": "Scan completo iniciado",
+ "scanError": "Erro ao iniciar o scan. Verifique os logs",
+ "scanCompleted": "Scan da biblioteca concluída"
+ },
+ "validation": {
+ "nameRequired": "Nome da biblioteca é obrigatório",
+ "pathRequired": "Caminho da biblioteca é obrigatório",
+ "pathNotDirectory": "Caminho da biblioteca deve ser um diretório",
+ "pathNotFound": "Caminho da biblioteca não encontrado",
+ "pathNotAccessible": "Caminho da biblioteca não está acessível",
+ "pathInvalid": "Caminho da biblioteca inválido"
+ },
+ "messages": {
+ "deleteConfirm": "Tem certeza que deseja excluir esta biblioteca? Isso removerá todos os dados associados.",
+ "scanInProgress": "Scan em progresso...",
+ "noLibrariesAssigned": "Nenhuma biblioteca atribuída a este usuário"
}
}
},
@@ -400,6 +480,8 @@
"transcodingDisabled": "Por questão de segurança, esta tela de configuração está desabilitada. Se você quiser alterar estas configurações, reinicie o servidor com a opção %{config}",
"transcodingEnabled": "Navidrome está sendo executado com a opção %{config}. Isto permite que potencialmente se execute comandos do sistema pela interface Web. É recomendado que vc mantenha esta opção desabilitada, e só a habilite quando precisar configurar opções de Conversão",
"songsAddedToPlaylist": "Música adicionada à playlist |||| %{smart_count} músicas adicionadas à playlist",
+ "noSimilarSongsFound": "Nenhuma música semelhante encontrada",
+ "noTopSongsFound": "Nenhuma música mais tocada encontrada",
"noPlaylistsAvailable": "Nenhuma playlist",
"delete_user_title": "Excluir usuário '%{name}'",
"delete_user_content": "Você tem certeza que deseja excluir o usuário e todos os seus dados (incluindo suas playlists e preferências)?",
@@ -433,6 +515,12 @@
},
"menu": {
"library": "Biblioteca",
+ "librarySelector": {
+ "allLibraries": "Todas as Bibliotecas (%{count})",
+ "multipleLibraries": "%{selected} de %{total} Bibliotecas",
+ "selectLibraries": "Selecionar Bibliotecas",
+ "none": "Nenhuma"
+ },
"settings": "Configurações",
"version": "Versão",
"theme": "Tema",
@@ -495,19 +583,40 @@
"disabled": "Desligado",
"waiting": "Aguardando"
}
+ },
+ "tabs": {
+ "about": "Sobre",
+ "config": "Configuração"
+ },
+ "config": {
+ "configName": "Nome da Configuração",
+ "environmentVariable": "Variável de Ambiente",
+ "currentValue": "Valor Atual",
+ "configurationFile": "Arquivo de Configuração",
+ "exportToml": "Exportar Configuração (TOML)",
+ "exportSuccess": "Configuração exportada para o clipboard em formato TOML",
+ "exportFailed": "Falha ao copiar configuração",
+ "devFlagsHeader": "Flags de Desenvolvimento (sujeitas a mudança/remoção)",
+ "devFlagsComment": "Estas são configurações experimentais e podem ser removidas em versões futuras"
}
},
"activity": {
"title": "Atividade",
- "totalScanned": "Total de pastas analisadas",
- "quickScan": "Scan rápido",
- "fullScan": "Scan completo",
+ "totalScanned": "Total de pastas scaneadas",
+ "quickScan": "Rápido",
+ "fullScan": "Completo",
+ "selectiveScan": "Seletivo",
"serverUptime": "Uptime do servidor",
"serverDown": "DESCONECTADO",
- "scanType": "Tipo",
+ "scanType": "Último Scan",
"status": "Erro",
"elapsedTime": "Duração"
},
+ "nowPlaying": {
+ "title": "Tocando agora",
+ "empty": "Nada tocando",
+ "minutesAgo": "%{smart_count} minuto atrás |||| %{smart_count} minutos atrás"
+ },
"help": {
"title": "Teclas de atalho",
"hotkeys": {
diff --git a/resources/i18n/ru.json b/resources/i18n/ru.json
index a4de263e4..e29996275 100644
--- a/resources/i18n/ru.json
+++ b/resources/i18n/ru.json
@@ -8,7 +8,7 @@
"duration": "Длительность",
"trackNumber": "#",
"playCount": "Проигрывания",
- "title": "Название",
+ "title": "Название трека",
"artist": "Исполнитель",
"album": "Альбом",
"path": "Путь",
@@ -23,7 +23,7 @@
"comment": "Комментарий",
"rating": "Рейтинг",
"quality": "Качество",
- "bpm": "Кол-во ударов в минуту",
+ "bpm": "BPM",
"playDate": "Последнее воспроизведение",
"channels": "Каналы",
"createdAt": "Дата добавления",
@@ -33,8 +33,10 @@
"tags": "Дополнительные теги",
"mappedTags": "Сопоставленные теги",
"rawTags": "Исходные теги",
- "bitDepth": "Битовая глубина",
- "sampleRate": "Частота дискретизации (Гц)"
+ "bitDepth": "Битовая глубина (Bit)",
+ "sampleRate": "Частота дискретизации (Hz)",
+ "missing": "Поле отсутствует",
+ "libraryName": "Библиотека"
},
"actions": {
"addToQueue": "В очередь",
@@ -43,7 +45,8 @@
"shuffleAll": "Перемешать",
"download": "Скачать",
"playNext": "Следующий",
- "info": "Информация"
+ "info": "Информация",
+ "showInPlaylist": "Показать в плейлисте"
}
},
"album": {
@@ -54,7 +57,7 @@
"duration": "Длительность",
"songCount": "Треков",
"playCount": "Проигрывания",
- "name": "Название",
+ "name": "Название альбома",
"genre": "Жанр",
"compilation": "Сборник",
"year": "Год",
@@ -73,7 +76,9 @@
"grouping": "Группирование",
"media": "Медиа",
"mood": "Настроение",
- "date": "Дата записи"
+ "date": "Дата записи",
+ "missing": "Поле отсутствует",
+ "libraryName": "Библиотека"
},
"actions": {
"playAll": "Играть",
@@ -98,14 +103,15 @@
"artist": {
"name": "Исполнитель |||| Исполнители",
"fields": {
- "name": "Название",
+ "name": "Название исполнителя",
"albumCount": "Количество альбомов",
"songCount": "Количество треков",
"playCount": "Проигрывания",
"rating": "Рейтинг",
"genre": "Жанр",
"size": "Размер",
- "role": "Роль"
+ "role": "Роль",
+ "missing": "Поле отсутствует"
},
"roles": {
"albumartist": "Исполнитель альбома |||| Исполнители альбома",
@@ -120,7 +126,13 @@
"mixer": "Звукоинженер |||| Звукоинженеры",
"remixer": "Ремиксер |||| Ремиксеры",
"djmixer": "DJ-миксер |||| DJ-миксеры",
- "performer": "Исполнитель |||| Исполнители"
+ "performer": "Исполнитель |||| Исполнители",
+ "maincredit": "Исполнитель альбома или Исполнитель |||| Исполнители альбома или Исполнители"
+ },
+ "actions": {
+ "shuffle": "Смешать",
+ "radio": "Радио",
+ "topSongs": "Топовые треки"
}
},
"user": {
@@ -132,15 +144,17 @@
"updatedAt": "Обновлено",
"name": "Имя",
"password": "Пароль",
- "createdAt": "Создан",
+ "createdAt": "Аккаунт создан",
"changePassword": "Сменить пароль?",
"currentPassword": "Текущий пароль",
"newPassword": "Новый пароль",
"token": "Токен",
- "lastAccessAt": "Последний доступ"
+ "lastAccessAt": "Последний доступ",
+ "libraries": "Библиотеки"
},
"helperTexts": {
- "name": "Изменение вступит в силу после следующего входа в систему"
+ "name": "Изменение вступит в силу после следующего входа в систему",
+ "libraries": "Выберите конкретные библиотеки для этого пользователя или оставьте поле пустым, чтобы использовать библиотеки по умолчанию"
},
"notifications": {
"created": "Пользователь создан",
@@ -149,7 +163,12 @@
},
"message": {
"listenBrainzToken": "Введите свой токен пользователя ListenBrainz.",
- "clickHereForToken": "Нажмите здесь, чтобы получить токен"
+ "clickHereForToken": "Нажмите здесь, чтобы получить токен",
+ "selectAllLibraries": "Выбрать все библиотеки",
+ "adminAutoLibraries": "Пользователи-администраторы автоматически получают доступ ко всем библиотекам"
+ },
+ "validation": {
+ "librariesRequired": "Для пользователей, не являющихся администраторами, должна быть выбрана хотя бы одна библиотека"
}
},
"player": {
@@ -157,7 +176,7 @@
"fields": {
"name": "Имя",
"transcodingId": "Транскодирование",
- "maxBitRate": "Макс. Битрейт",
+ "maxBitRate": "Макс. битрейт",
"client": "Клиент",
"userName": "Пользователь",
"lastSeen": "Был на сайте",
@@ -175,9 +194,9 @@
}
},
"playlist": {
- "name": "Плейлистов |||| Плейлисты",
+ "name": "Плейлист |||| Плейлисты",
"fields": {
- "name": "Название",
+ "name": "Название трека",
"duration": "Длительность",
"ownerName": "Владелец",
"public": "Публичный",
@@ -193,11 +212,17 @@
"addNewPlaylist": "Создать \"%{name}\"",
"export": "Экспорт",
"makePublic": "Опубликовать",
- "makePrivate": "Сделать личным"
+ "makePrivate": "Сделать личным",
+ "saveQueue": "Сохранить очередь в плейлист",
+ "searchOrCreate": "Поиск плейлистов или введите текст для создания новых...",
+ "pressEnterToCreate": "Нажмите Enter, чтобы создать новый список воспроизведения",
+ "removeFromSelection": "Удалить из списка выделенных"
},
"message": {
"duplicate_song": "Повторяющиеся треки",
- "song_exist": "Некоторые треки уже есть в плейлисте. Вы хотите добавить их или пропустить?"
+ "song_exist": "Некоторые треки уже есть в плейлисте. Вы хотите добавить их или пропустить?",
+ "noPlaylistsFound": "Плейлисты не найдены",
+ "noPlaylists": "Нет доступных плейлистов"
}
},
"radio": {
@@ -222,9 +247,9 @@
"contents": "Содержание",
"expiresAt": "Ссылка истекает",
"lastVisitedAt": "Последнее посещение",
- "visitCount": "Посещения",
+ "visitCount": "Количество посещений",
"format": "Формат",
- "maxBitRate": "Макс. Битрейт",
+ "maxBitRate": "Макс. битрейт",
"updatedAt": "Обновлено в",
"createdAt": "Создано",
"downloadable": "Разрешить загрузку?"
@@ -235,15 +260,69 @@
"fields": {
"path": "Место расположения",
"size": "Размер",
- "updatedAt": "Исчез"
+ "updatedAt": "Исчез",
+ "libraryName": "Библиотека"
},
"actions": {
- "remove": "Удалить"
+ "remove": "Удалить",
+ "remove_all": "Убрать все"
},
"notifications": {
"removed": "Отсутствующие файлы удалены"
},
"empty": "Нет отсутствующих файлов"
+ },
+ "library": {
+ "name": "Библиотека |||| Библиотеки",
+ "fields": {
+ "name": "Имя",
+ "path": "Путь",
+ "remotePath": "Удаленный путь",
+ "lastScanAt": "Последнее сканирование",
+ "songCount": "Треки",
+ "albumCount": "Альбомы",
+ "artistCount": "Исполнители",
+ "totalSongs": "Треки",
+ "totalAlbums": "Альбомы",
+ "totalArtists": "Исполнители",
+ "totalFolders": "Папки",
+ "totalFiles": "Файлов",
+ "totalMissingFiles": "Пропавших файлов",
+ "totalSize": "Общий размер",
+ "totalDuration": "Длительность",
+ "defaultNewUsers": "По умолчанию для новых пользователей",
+ "createdAt": "Создано",
+ "updatedAt": "Обновлено"
+ },
+ "sections": {
+ "basic": "Основная информация",
+ "statistics": "Статистика"
+ },
+ "actions": {
+ "scan": "Сканировать библиотеку",
+ "manageUsers": "Управление доступом пользователей",
+ "viewDetails": "Просмотреть подробности"
+ },
+ "notifications": {
+ "created": "Библиотека успешно создана",
+ "updated": "Библиотека успешно обновлена",
+ "deleted": "Библиотека успешно удалена",
+ "scanStarted": "Сканирование библиотеки начато",
+ "scanCompleted": "Сканирование библиотеки закончено"
+ },
+ "validation": {
+ "nameRequired": "Имя библиотеки обязательно",
+ "pathRequired": "Путь к библиотеке обязателен",
+ "pathNotDirectory": "Путь к библиотеке должен быть директорией",
+ "pathNotFound": "Путь к библиотеке не найдено",
+ "pathNotAccessible": "Путь к библиотеке недоступен",
+ "pathInvalid": "Неверный путь к библиотеке"
+ },
+ "messages": {
+ "deleteConfirm": "Вы уверены, что хотите удалить эту библиотеку? Это приведет к удалению всех связанных с ней данных и доступа пользователей.",
+ "scanInProgress": "Сканирование продолжается...",
+ "noLibrariesAssigned": "Нет библиотек, назначенных этому пользователю"
+ }
}
},
"ra": {
@@ -274,7 +353,7 @@
"oneOf": "Должно быть одним из: %{options}",
"regex": "Должно быть в формате (regexp): %{pattern}",
"unique": "Должно быть уникальным",
- "url": "Должен быть действительным URL адрес"
+ "url": "Должен быть действительный URL"
},
"action": {
"add_filter": "Фильтр",
@@ -291,7 +370,7 @@
"export": "Экспорт",
"list": "Список",
"refresh": "Обновить",
- "remove_filter": "Убрать фильтр",
+ "remove_filter": "Убрать этот фильтр",
"remove": "Удалить",
"save": "Сохранить",
"search": "Поиск",
@@ -382,7 +461,7 @@
"i18n_error": "Не удалось загрузить перевод для указанного языка",
"canceled": "Операция отменена",
"logged_out": "Ваша сессия завершена, попробуйте переподключиться/войти снова",
- "new_version": "Доступна новая версия! Пожалуйста, обновите это окно"
+ "new_version": "Доступна новая версия! Пожалуйста, обновите это окно."
},
"toggleFieldsMenu": {
"columnsToDisplay": "Отображение столбцов",
@@ -422,8 +501,12 @@
"shareFailure": "Ошибка копирования URL-адреса %{url} в буфер обмена",
"downloadDialogTitle": "Скачать %{resource} '%{name}' (%{size})",
"shareCopyToClipboard": "Копировать в буфер обмена: Ctrl+C, Enter",
- "remove_missing_title": "Удалить отсутствующие файлы",
- "remove_missing_content": "Вы уверены, что хотите удалить выбранные отсутствующие файлы из базы данных? Это навсегда удалит все ссылки на них, включая данные о прослушиваниях и рейтингах."
+ "remove_missing_title": "Удалить отсутствующие файлы?",
+ "remove_missing_content": "Вы уверены, что хотите удалить выбранные отсутствующие файлы из базы данных? Это навсегда удалит все ссылки на них, включая данные о прослушиваниях и рейтингах.",
+ "remove_all_missing_title": "Удалите все отсутствующие файлы",
+ "remove_all_missing_content": "Вы уверены, что хотите удалить все отсутствующие файлы из базы данных? Это навсегда удалит все упоминания о них, включая количество игр и рейтинг.",
+ "noSimilarSongsFound": "Похожих треков не найдено",
+ "noTopSongsFound": "Лучших треков не найдено"
},
"menu": {
"library": "Библиотека",
@@ -452,7 +535,13 @@
"albumList": "Альбомы",
"about": "О нас",
"playlists": "Плейлисты",
- "sharedPlaylists": "Поделиться плейлистом"
+ "sharedPlaylists": "Поделиться плейлистом",
+ "librarySelector": {
+ "allLibraries": "Все библиотеки (%{count})",
+ "multipleLibraries": "%{selected} из %{total} Библиотеки",
+ "selectLibraries": "Выбор библиотек",
+ "none": "Отсутствует"
+ }
},
"player": {
"playListsText": "Очередь Воспроизведения",
@@ -482,13 +571,28 @@
"about": {
"links": {
"homepage": "Главная",
- "source": "Код",
+ "source": "Исходный код",
"featureRequests": "Предложения",
"lastInsightsCollection": "Последний сбор данных",
"insights": {
"disabled": "Выключено",
"waiting": "Ожидание"
}
+ },
+ "tabs": {
+ "about": "О нас",
+ "config": "Конфигурация"
+ },
+ "config": {
+ "configName": "Имя конфигурации",
+ "environmentVariable": "Переменная среды",
+ "currentValue": "Текущее значение",
+ "configurationFile": "Файл конфигурации",
+ "exportToml": "Экспорт конфигурации (TOML)",
+ "exportSuccess": "Конфигурация экспортирована в буфер обмена в формате TOML",
+ "exportFailed": "Не удалось скопировать конфигурацию",
+ "devFlagsHeader": "Флаги разработки (могут быть изменены/удалены)",
+ "devFlagsComment": "Это экспериментальные настройки, которые могут быть удалены в будущих версиях."
}
},
"activity": {
@@ -497,7 +601,10 @@
"quickScan": "Быстрое сканирование",
"fullScan": "Полное сканирование",
"serverUptime": "Время работы сервера",
- "serverDown": "Оффлайн"
+ "serverDown": "Оффлайн",
+ "scanType": "Тип",
+ "status": "Ошибка сканирования",
+ "elapsedTime": "Прошедшее время"
},
"help": {
"title": "Горячие клавиши Navidrome",
@@ -510,7 +617,12 @@
"vol_up": "Увеличить громкость",
"vol_down": "Уменьшить громкость",
"toggle_love": "Добавить / удалить песню из избранного",
- "current_song": "Перейти к текущей песне"
+ "current_song": "Перейти к текущему треку"
}
+ },
+ "nowPlaying": {
+ "title": "Сейчас играет",
+ "empty": "Ничего не играет",
+ "minutesAgo": "%{smart_count} минут назад |||| %{smart_count} минут назад"
}
}
\ No newline at end of file
diff --git a/resources/i18n/sl.json b/resources/i18n/sl.json
index f860245e8..80bd8e4a3 100644
--- a/resources/i18n/sl.json
+++ b/resources/i18n/sl.json
@@ -1,460 +1,628 @@
{
- "languageName": "Slovenščina",
- "resources": {
- "song": {
- "name": "Pesem |||| Pesmi",
- "fields": {
- "albumArtist": "Avtor albuma",
- "duration": "Dolžina",
- "trackNumber": "#",
- "playCount": "Predvajano",
- "title": "Naslov",
- "artist": "Avtor",
- "album": "Album",
- "path": "Pot datoteke",
- "genre": "Žanr",
- "compilation": "Kompilacija",
- "year": "Leto",
- "size": "Velikost datoteke",
- "updatedAt": "Posodobljeno",
- "bitRate": "Bitna hitrost",
- "discSubtitle": "Podnapisi",
- "starred": "Priljubljen",
- "comment": "Opomba",
- "rating": "Ocena",
- "quality": "Kakovost",
- "bpm": "BPM",
- "playDate": "Zadnja predvajana",
- "channels": "Kanali",
- "createdAt": "Datum dodano"
- },
- "actions": {
- "addToQueue": "Predvajaj kasneje",
- "playNow": "Predvajaj",
- "addToPlaylist": "Dodaj na seznam predvajanj",
- "shuffleAll": "Premešaj vse",
- "download": "Naloži",
- "playNext": "Naslednji",
- "info": "Več informacij"
- }
- },
- "album": {
- "name": "Album |||| Albumi",
- "fields": {
- "albumArtist": "Avtor albuma",
- "artist": "Izvajalec",
- "duration": "Dolžina",
- "songCount": "Pesmi",
- "playCount": "Predvajano",
- "name": "Naslov",
- "genre": "Žanr",
- "compilation": "Kompilacija",
- "year": "Leto",
- "updatedAt": "Posodobljeno",
- "comment": "Opomba",
- "rating": "Ocena",
- "createdAt": "Datum dodano",
- "size": "Velikost",
- "originalDate": "Original",
- "releaseDate": "Izdano",
- "releases": "Izdaja |||| Izdaje",
- "released": "Izdano"
- },
- "actions": {
- "playAll": "Predvajaj vse",
- "playNext": "Naslednji",
- "addToQueue": "Predvajaj kasneje",
- "shuffle": "Premešaj",
- "addToPlaylist": "Dodaj v seznam predvajanja",
- "download": "Naloži",
- "info": "Več informacij",
- "share": "Deli"
- },
- "lists": {
- "all": "Vse",
- "random": "Naključno",
- "recentlyAdded": "Dodan nedavno",
- "recentlyPlayed": "Predvajan nedavno",
- "mostPlayed": "Največ predvajano",
- "starred": "Priljubljeni",
- "topRated": "Najvišje ocenjeno"
- }
- },
- "artist": {
- "name": "Izvajalec |||| Izvajalci",
- "fields": {
- "name": "Ime",
- "albumCount": "# albumov",
- "songCount": "# pesmi",
- "playCount": "# predvajanj",
- "rating": "Ocena",
- "genre": "Žanr",
- "size": "Velikost"
- }
- },
- "user": {
- "name": "Uporabnik |||| Uporabniki",
- "fields": {
- "userName": "Uporabnik",
- "isAdmin": "Upravitelj",
- "lastLoginAt": "Zadnji vpis",
- "updatedAt": "Posodobljeno",
- "name": "Ime",
- "password": "Geslo",
- "createdAt": "Ustvarjeno",
- "changePassword": "Spremeni geslo?",
- "currentPassword": "Trenutno geslo",
- "newPassword": "Novo geslo",
- "token": "Žeton"
- },
- "helperTexts": {
- "name": "Sprememba imena bo vidna pri naslednjem vpisu"
- },
- "notifications": {
- "created": "Uporabnik ustvarjen",
- "updated": "Uporabnik posodobljen",
- "deleted": "Uporabnik izbrisan"
- },
- "message": {
- "listenBrainzToken": "Vnesi žeton uporabnika ListenBrainz.",
- "clickHereForToken": "Klikni za žeton"
- }
- },
- "player": {
- "name": "Predvajalnik |||| Predvajalniki",
- "fields": {
- "name": "Naziv",
- "transcodingId": "Transkodiranje",
- "maxBitRate": "Maks. bitrate",
- "client": "Klijent",
- "userName": "Uporabnik",
- "lastSeen": "Zadnjič viden",
- "reportRealPath": "Zabeleži pravo pot",
- "scrobbleEnabled": "Pošlji Scrobbles zunanjim storitvam"
- }
- },
- "transcoding": {
- "name": "Transkodiranje |||| Transkodiranje",
- "fields": {
- "name": "Ime",
- "targetFormat": "Ciljni format",
- "defaultBitRate": "Privzet bitrate",
- "command": "Ukaz"
- }
- },
- "playlist": {
- "name": "Seznam predvajanj |||| Seznami predvajanj",
- "fields": {
- "name": "Ime",
- "duration": "Dolžina",
- "ownerName": "Lastnik",
- "public": "Javno",
- "updatedAt": "Posodobljen",
- "createdAt": "Ustvarjen",
- "songCount": "# pesmi",
- "comment": "Opomba",
- "sync": "Avtomatski uvoz",
- "path": "Uvozi iz"
- },
- "actions": {
- "selectPlaylist": "Izberi seznam",
- "addNewPlaylist": "Ustvari \"%{name}\"",
- "export": "Izvozi",
- "makePublic": "Naredi javno",
- "makePrivate": "Naredi zasebno"
- },
- "message": {
- "duplicate_song": "Dodaj podvojene pesmi",
- "song_exist": "Seznamu predvajanja boste dodali duplikate. Jih želite dodati ali izpustiti?"
- }
- },
- "radio": {
- "name": "Radio |||| Radiji",
- "fields": {
- "name": "Ime",
- "streamUrl": "URL toka",
- "homePageUrl": "URL domače strani",
- "updatedAt": "Posodobljeno ob",
- "createdAt": "Ustvarjeno ob"
- },
- "actions": {
- "playNow": "Predvajaj"
- }
- },
- "share": {
- "name": "Deli |||| Delitev",
- "fields": {
- "username": "Delil z",
- "url": "URL",
- "description": "Opis",
- "contents": "Vsebine",
- "expiresAt": "Poteče",
- "lastVisitedAt": "Nazadnje obiskano",
- "visitCount": "Obiski",
- "format": "Oblika",
- "maxBitRate": "Maks. bitna hitrost",
- "updatedAt": "Posodobljeno ob",
- "createdAt": "Ustvarjeno ob",
- "downloadable": "Dovoli prenose?"
- }
- }
+ "languageName": "Slovenščina",
+ "resources": {
+ "song": {
+ "name": "Pesem |||| Pesmi",
+ "fields": {
+ "albumArtist": "Avtor albuma",
+ "duration": "Dolžina",
+ "trackNumber": "#",
+ "playCount": "Predvajano",
+ "title": "Naslov",
+ "artist": "Avtor",
+ "album": "Album",
+ "path": "Pot datoteke",
+ "genre": "Žanr",
+ "compilation": "Kompilacija",
+ "year": "Leto",
+ "size": "Velikost datoteke",
+ "updatedAt": "Posodobljeno",
+ "bitRate": "Bitna hitrost",
+ "discSubtitle": "Podnapisi",
+ "starred": "Priljubljen",
+ "comment": "Opomba",
+ "rating": "Ocena",
+ "quality": "Kakovost",
+ "bpm": "BPM",
+ "playDate": "Zadnja predvajana",
+ "channels": "Kanali",
+ "createdAt": "Datum dodano",
+ "grouping": "Grupiranje",
+ "mood": "Razpoloženje",
+ "participants": "Dodatni udeleženci",
+ "tags": "Dodatne oznake",
+ "mappedTags": "Preslikane oznake",
+ "rawTags": "Nespremenjene oznake",
+ "bitDepth": "Bitna globina",
+ "sampleRate": "Frekvenca vzorčenja",
+ "missing": "Manjka",
+ "libraryName": "Knjižnica"
+ },
+ "actions": {
+ "addToQueue": "Predvajaj kasneje",
+ "playNow": "Predvajaj",
+ "addToPlaylist": "Dodaj na seznam predvajanj",
+ "shuffleAll": "Premešaj vse",
+ "download": "Naloži",
+ "playNext": "Naslednji",
+ "info": "Več informacij",
+ "showInPlaylist": "Prikaži na seznamu predvajanja"
+ }
},
- "ra": {
- "auth": {
- "welcome1": "Hvala, da ste naložili Navidrome!",
- "welcome2": "Za začetek, ustvarite upraviteljski račun",
- "confirmPassword": "Potrdi Geslo",
- "buttonCreateAdmin": "Ustvari upravitelja",
- "auth_check_error": "Vpišite se za nadaljevanje",
- "user_menu": "Profil",
- "username": "Uporabnik",
- "password": "Geslo",
- "sign_in": "Vpis",
- "sign_in_error": "Avtentikacija neuspešna, poskusite ponovno",
- "logout": "Izpis"
- },
- "validation": {
- "invalidChars": "Uporabi samo alfanumerične znake",
- "passwordDoesNotMatch": "Geslo se ne ujema",
- "required": "Potreben",
- "minLength": "Potrebnih je vsaj %{min} znakov",
- "maxLength": "Potrebnih je največ %{max}",
- "minValue": "Potrebnih je vsaj %{min}",
- "maxValue": "Potrebnih je največ %{max}",
- "number": "Mora biti številka",
- "email": "Veljaven e-poštni naslov",
- "oneOf": "Mora biti ena izmed %{options}",
- "regex": "Mora se ujemati z določeno obliko (regexp): %{pattern}",
- "unique": "Mora biti edinstven",
- "url": "Biti mora veljaven URL"
- },
- "action": {
- "add_filter": "Dodaj filter",
- "add": "Dodaj",
- "back": "Nazaj",
- "bulk_actions": "Izbran 1 element |||| Izbranih %{smart_count} elementov",
- "cancel": "Prekliči",
- "clear_input_value": "Pobriši",
- "clone": "Podvoji",
- "confirm": "Potrdi",
- "create": "Ustvari",
- "delete": "Izbriši",
- "edit": "Uredi",
- "export": "Izvozi",
- "list": "Seznam",
- "refresh": "Osveži",
- "remove_filter": "Odstrani filter",
- "remove": "Odstrani",
- "save": "Shrani",
- "search": "Išči",
- "show": "Prikaži",
- "sort": "Razvrsti",
- "undo": "Razveljavi",
- "expand": "Razširi",
- "close": "Zapri",
- "open_menu": "Odpri meni",
- "close_menu": "Zapri meni",
- "unselect": "Prekliči izbiro",
- "skip": "Izpusti",
- "bulk_actions_mobile": "1 |||| %{smart_count}",
- "share": "Deli",
- "download": "Prenesi"
- },
- "boolean": {
- "true": "Da",
- "false": "Ne"
- },
- "page": {
- "create": "Ustvari %{name}",
- "dashboard": "Nadzorna plošča",
- "edit": "%{name} #%{id}",
- "error": "Nedoločena napaka",
- "list": "%{name}",
- "loading": "Nalagam",
- "not_found": "Ni zadetka",
- "show": "%{name} #%{id}",
- "empty": "Še brez %{name}.",
- "invite": "Ga želite dodati?"
- },
- "input": {
- "file": {
- "upload_several": "Povlecite datoteke ali pa kliknite in izberite.",
- "upload_single": "Povlecite datoteko ali pa kliknite in izberite."
- },
- "image": {
- "upload_several": "Povlecite slike, ali pa kliknite in izberite.",
- "upload_single": "Povlecite sliko, ali pa kliknite in izberite."
- },
- "references": {
- "all_missing": "Ne najdem referenciranih podatkov.",
- "many_missing": "Zdi se, da vsaj ena asociirana referenca ni več na voljo.",
- "single_missing": "Zdi se, da asociirana referenca ni več na voljo."
- },
- "password": {
- "toggle_visible": "Skrij geslo",
- "toggle_hidden": "Prikaži geslo"
- }
- },
- "message": {
- "about": "O programu",
- "are_you_sure": "Ste prepričani?",
- "bulk_delete_content": "Ste prepričani, da želite izbrisati %{name}? |||| Ste prepričani, da želite izbrisati %{smart_count} elementov?",
- "bulk_delete_title": "Izbriši %{name} |||| Izbriši %{smart_count} %{name}",
- "delete_content": "Ste prepričani, da želite izbrisati ta element?",
- "delete_title": "Izbriši %{name} #%{id}",
- "details": "Podrobnosti",
- "error": "Napak klijenta. Vaš zahtevek se je zaključil neuspešno.",
- "invalid_form": "Oblika ni veljavna. Prosim preverite napake",
- "loading": "Stran se nalaga, trenutek",
- "no": "Ne",
- "not_found": "Ali ste vtipkali napačen naslov (URL), ali pa sledili neobstoječi povezavi.",
- "yes": "Da",
- "unsaved_changes": "Nekate spremembe se niso shranile. Ste prepričani, da jih želite ignorirati?"
- },
- "navigation": {
- "no_results": "Ni zadetkov",
- "no_more_results": "Številka strani %{page} je zunaj meja. Preizkusite prejšnjo stran.",
- "page_out_of_boundaries": "Številka strani %{page} je zunaj meja",
- "page_out_from_end": "Ne gre dalje od zadnje strani",
- "page_out_from_begin": "Ne gre pred prvo stran",
- "page_range_info": "%{offsetBegin}-%{offsetEnd} od %{total}",
- "page_rows_per_page": "Elementov na stran:",
- "next": "Naslednji",
- "prev": "Prejšnji",
- "skip_nav": "Preskoči k vsebini"
- },
- "notification": {
- "updated": "Element posodobljen |||| Posodobljenih %{smart_count} elementov",
- "created": "Element dodan",
- "deleted": "Element izbrisan |||| %{smart_count} elementov izbrisanih",
- "bad_item": "Nepravilen element",
- "item_doesnt_exist": "Element ne obstaja",
- "http_error": "Strežnika napaka v komunikaciji",
- "data_provider_error": "Napaka dataProvider error. Preverite konzolo za podrobnosti.",
- "i18n_error": "Ne uspem naložiti prevode za izbran jezik",
- "canceled": "Akcija preklicana",
- "logged_out": "Seja je potekla, prosim povežite se ponovno.",
- "new_version": "Na voljo je nova verzija! Prosim osvežite okno."
- },
- "toggleFieldsMenu": {
- "columnsToDisplay": "Prikaži stolpce",
- "layout": "Razporeditev",
- "grid": "Mreža",
- "table": "Tabela"
- }
+ "album": {
+ "name": "Album |||| Albumi",
+ "fields": {
+ "albumArtist": "Avtor albuma",
+ "artist": "Izvajalec",
+ "duration": "Dolžina",
+ "songCount": "Pesmi",
+ "playCount": "Predvajano",
+ "name": "Naslov",
+ "genre": "Žanr",
+ "compilation": "Kompilacija",
+ "year": "Leto",
+ "updatedAt": "Posodobljeno",
+ "comment": "Opomba",
+ "rating": "Ocena",
+ "createdAt": "Datum dodano",
+ "size": "Velikost",
+ "originalDate": "Original",
+ "releaseDate": "Izdano",
+ "releases": "Izdaja |||| Izdaje",
+ "released": "Izdano",
+ "recordLabel": "Založba",
+ "catalogNum": "Kataloška številka",
+ "releaseType": "Tip",
+ "grouping": "Grupiranje",
+ "media": "Medij",
+ "mood": "Razpoloženje",
+ "date": "Datum snemanja",
+ "missing": "Manjka",
+ "libraryName": "Knjižnica"
+ },
+ "actions": {
+ "playAll": "Predvajaj vse",
+ "playNext": "Naslednji",
+ "addToQueue": "Predvajaj kasneje",
+ "shuffle": "Premešaj",
+ "addToPlaylist": "Dodaj v seznam predvajanja",
+ "download": "Naloži",
+ "info": "Več informacij",
+ "share": "Deli"
+ },
+ "lists": {
+ "all": "Vse",
+ "random": "Naključno",
+ "recentlyAdded": "Dodan nedavno",
+ "recentlyPlayed": "Predvajan nedavno",
+ "mostPlayed": "Največ predvajano",
+ "starred": "Priljubljeni",
+ "topRated": "Najvišje ocenjeno"
+ }
},
- "message": {
- "note": "OPOMBA",
- "transcodingDisabled": "Sprememba konfiguracije transkodiranja skozi spletni vmesnik je onemogočeno zaradi varnostnih razlogov. Če želite spremeniti (urediti ali izbrisati) možnosti transkodiranja, ponovno zaženite strežnik z %{config} nastavitvami.",
- "transcodingEnabled": "Navidrome trenutno uporablja nastavitve %{config}, kar pomeni da je možno pognati sistemske ukaze v nastavitvah transkodiranja preko spletnega vmesnika.\nZaradi varnostnih razlogov je možnost priporočeno onemogočiti , razen v primeru spreminjanja nastavitev.",
- "songsAddedToPlaylist": "Dodaj pesem na seznam predvajanj |||| Dodaj %{smart_count} pesmi na seznam predvajanj",
- "noPlaylistsAvailable": "Ni seznamov",
- "delete_user_title": "Odstrani uporabnika '%{name}'",
- "delete_user_content": "Ste prepričani o izbrisu uporabnika, vključno z njegovimi podatki (tudi seznami predvajanj in nastavitvami)?",
- "notifications_blocked": "V vašem brskljalniku Imate blokirana možnost obvestil za to spletno stran",
- "notifications_not_available": "Vaš brskljalnik ne omogoča obvestil na namizju ali pa do Navidrome ne dostopate po varni povezavi (https)",
- "lastfmLinkSuccess": "Last.fm uspešno povezan in 'scrobbling' omogočen",
- "lastfmLinkFailure": "Last.fm ni uspešno povezan",
- "lastfmUnlinkSuccess": "Last.fm povezava prekinjena in 'scrobbling' onemogočen",
- "lastfmUnlinkFailure": "Last.fm povezava neuspešno prekinjena",
- "openIn": {
- "lastfm": "Odpri v Last.fm",
- "musicbrainz": "Odpri v MusicBrainz"
- },
- "lastfmLink": "Preberi več...",
- "listenBrainzLinkSuccess": "ListenBrainz uspešno povezan in scrobbling vključen za uporabnika: %{user}",
- "listenBrainzLinkFailure": "ListBrainz neuspešno povezan: %{error}",
- "listenBrainzUnlinkSuccess": "ListenBrainz povezava prekinjena in scrobbling izključen",
- "listenBrainzUnlinkFailure": "ListenBrainz prekinitev povezave neuspešna",
- "downloadOriginalFormat": "Prenesi v izvirni obliki",
- "shareOriginalFormat": "Deli v izvirni obliki",
- "shareDialogTitle": "Deli %{resource} '%{name}'",
- "shareBatchDialogTitle": "Deli 1 %{resource} |||| Deli %{smart_count} %{resource}",
- "shareSuccess": "URL kopiran v odložišče: %{url}",
- "shareFailure": "Napaka pri kopiranju URL-ja %{url} v odložišče",
- "downloadDialogTitle": "Prenesi %{resource} '%{name}' (%{size})",
- "shareCopyToClipboard": "Kopiraj v odložišče: Ctrl+C, Enter"
+ "artist": {
+ "name": "Izvajalec |||| Izvajalci",
+ "fields": {
+ "name": "Ime",
+ "albumCount": "# albumov",
+ "songCount": "# pesmi",
+ "playCount": "# predvajanj",
+ "rating": "Ocena",
+ "genre": "Žanr",
+ "size": "Velikost",
+ "role": "Vloga",
+ "missing": "Manjka"
+ },
+ "roles": {
+ "albumartist": "Izvajalec albuma |||| Izvajalci albuma",
+ "artist": "Izvajalec |||| Izvajalci",
+ "composer": "Skladatelj |||| Skladatelji",
+ "conductor": "Dirigent |||| Dirigenti",
+ "lyricist": "Tekstopisec |||| Tekstopisci",
+ "arranger": "Aranžer |||| Aranžerji",
+ "producer": "Producent |||| Producenti",
+ "director": "Glasbeni vodja |||| Glasbene vodje",
+ "engineer": "Inženir |||| Inženirji",
+ "mixer": "Mešalec |||| Mešalci",
+ "remixer": "Remikser |||| Remikserji",
+ "djmixer": "DJ mešalec |||| DJ mešalci",
+ "performer": "Izvajalec |||| Izvajalci",
+ "maincredit": "Izvajalec albuma ali izvajalec |||| Izvajalci albuma ali izvajalci"
+ },
+ "actions": {
+ "shuffle": "Naključno predvajanje",
+ "radio": "Radio",
+ "topSongs": "Najboljše pesmi"
+ }
},
- "menu": {
- "library": "Knjižnica",
- "settings": "Nastavitve",
- "version": "Različica",
- "theme": "Tema",
- "personal": {
- "name": "Osebno",
- "options": {
- "theme": "Tema",
- "language": "Jezik",
- "defaultView": "Privzet pogled",
- "desktop_notifications": "Namizna obvestila",
- "lastfmScrobbling": "'Scrobble' do Last.fm",
- "listenBrainzScrobbling": "Scrobble k ListenBrainz",
- "replaygain": "ReplayGain način",
- "preAmp": "ReplayGain PreAmp (dB)",
- "gain": {
- "none": "Onemogočeno",
- "album": "Uporabi Album Gain",
- "track": "Uporabi Track Gain"
- }
- }
- },
- "albumList": "Albumi",
- "about": "O programu",
- "playlists": "Seznami predvajanj",
- "sharedPlaylists": "Deljeni seznami predvajanj"
+ "user": {
+ "name": "Uporabnik |||| Uporabniki",
+ "fields": {
+ "userName": "Uporabnik",
+ "isAdmin": "Upravitelj",
+ "lastLoginAt": "Zadnji vpis",
+ "updatedAt": "Posodobljeno",
+ "name": "Ime",
+ "password": "Geslo",
+ "createdAt": "Ustvarjeno",
+ "changePassword": "Spremeni geslo?",
+ "currentPassword": "Trenutno geslo",
+ "newPassword": "Novo geslo",
+ "token": "Žeton",
+ "lastAccessAt": "Zadnji dostop",
+ "libraries": "Knjižnice"
+ },
+ "helperTexts": {
+ "name": "Sprememba imena bo vidna pri naslednjem vpisu",
+ "libraries": "Izberite določene knjižnice za uporabnika ali pustite prazno, če želite uporabiti privzete knjižnice"
+ },
+ "notifications": {
+ "created": "Uporabnik ustvarjen",
+ "updated": "Uporabnik posodobljen",
+ "deleted": "Uporabnik izbrisan"
+ },
+ "message": {
+ "listenBrainzToken": "Vnesi žeton uporabnika ListenBrainz.",
+ "clickHereForToken": "Klikni za žeton",
+ "selectAllLibraries": "Izberi vse knjižnice",
+ "adminAutoLibraries": "Skrbniški uporabniki imajo samodejno dostop do vseh knjižnic"
+ },
+ "validation": {
+ "librariesRequired": "Za uporabnike brez skrbniških pravic mora biti izbrana vsaj ena knjižnica"
+ }
},
"player": {
- "playListsText": "Predvajaj vrsto",
- "openText": "Odpri",
- "closeText": "Zapri",
- "notContentText": "Ni glasbe",
- "clickToPlayText": "Predvajaj",
- "clickToPauseText": "Premor predvajanja",
- "nextTrackText": "Naslednje predvajanje",
- "previousTrackText": "Prejšnji",
- "reloadText": "Ponovno naloži",
- "volumeText": "Glasnost",
- "toggleLyricText": "Preklopi besedila",
- "toggleMiniModeText": "Pomanjšaj",
- "destroyText": "Uniči",
- "downloadText": "Naloži",
- "removeAudioListsText": "Izbriši avdio seznam",
- "clickToDeleteText": "Klikni za izbris %{name}",
- "emptyLyricText": "Ni besedila",
- "playModeText": {
- "order": "Po vrsti",
- "orderLoop": "Ponavljaj",
- "singleLoop": "Ponovi enkrat",
- "shufflePlay": "Premešaj"
- }
+ "name": "Predvajalnik |||| Predvajalniki",
+ "fields": {
+ "name": "Naziv",
+ "transcodingId": "Transkodiranje",
+ "maxBitRate": "Maks. bitrate",
+ "client": "Klijent",
+ "userName": "Uporabnik",
+ "lastSeen": "Zadnjič viden",
+ "reportRealPath": "Zabeleži pravo pot",
+ "scrobbleEnabled": "Pošlji Scrobbles zunanjim storitvam"
+ }
},
- "about": {
- "links": {
- "homepage": "Domača stran",
- "source": "Izvorna koda",
- "featureRequests": "Funkcionalni zahtevki"
- }
+ "transcoding": {
+ "name": "Transkodiranje |||| Transkodiranje",
+ "fields": {
+ "name": "Ime",
+ "targetFormat": "Ciljni format",
+ "defaultBitRate": "Privzet bitrate",
+ "command": "Ukaz"
+ }
},
- "activity": {
- "title": "Aktivnost",
- "totalScanned": "Skupaj preiskanih map",
- "quickScan": "Hitro preišči",
- "fullScan": "Polno preišči",
- "serverUptime": "Čas delovanja",
- "serverDown": "NEPOVEZAN"
+ "playlist": {
+ "name": "Seznam predvajanj |||| Seznami predvajanj",
+ "fields": {
+ "name": "Ime",
+ "duration": "Dolžina",
+ "ownerName": "Lastnik",
+ "public": "Javno",
+ "updatedAt": "Posodobljen",
+ "createdAt": "Ustvarjen",
+ "songCount": "# pesmi",
+ "comment": "Opomba",
+ "sync": "Avtomatski uvoz",
+ "path": "Uvozi iz"
+ },
+ "actions": {
+ "selectPlaylist": "Izberi seznam",
+ "addNewPlaylist": "Ustvari \"%{name}\"",
+ "export": "Izvozi",
+ "makePublic": "Naredi javno",
+ "makePrivate": "Naredi zasebno",
+ "saveQueue": "Shrani čakalno vrsto na seznam predvajanja",
+ "searchOrCreate": "Iščite po seznamih predvajanja ali vnesite besedilo, da ustvarite nove ...",
+ "pressEnterToCreate": "Pritisnite Enter za ustvarjanje novega seznama predvajanja",
+ "removeFromSelection": "Odstrani iz izbora"
+ },
+ "message": {
+ "duplicate_song": "Dodaj podvojene pesmi",
+ "song_exist": "Seznamu predvajanja boste dodali duplikate. Jih želite dodati ali izpustiti?",
+ "noPlaylistsFound": "Ni najdenih seznamov predvajanja",
+ "noPlaylists": "Ni na voljo seznamov predvajanja"
+ }
},
- "help": {
- "title": "Hitre tipke",
- "hotkeys": {
- "show_help": "Prikaži pomoč",
- "toggle_menu": "Preklopi stransko vrstico menija",
- "toggle_play": "Predvajaj / Pavza",
- "prev_song": "Prejšnja",
- "next_song": "Naslednja",
- "vol_up": "Zvišaj glasnost",
- "vol_down": "Znižaj glasnost",
- "toggle_love": "Dodaj med priljubljene",
- "current_song": "Skoči na predvajano"
- }
+ "radio": {
+ "name": "Radio |||| Radiji",
+ "fields": {
+ "name": "Ime",
+ "streamUrl": "URL toka",
+ "homePageUrl": "URL domače strani",
+ "updatedAt": "Posodobljeno ob",
+ "createdAt": "Ustvarjeno ob"
+ },
+ "actions": {
+ "playNow": "Predvajaj"
+ }
+ },
+ "share": {
+ "name": "Deli |||| Delitev",
+ "fields": {
+ "username": "Delil z",
+ "url": "URL",
+ "description": "Opis",
+ "contents": "Vsebine",
+ "expiresAt": "Poteče",
+ "lastVisitedAt": "Nazadnje obiskano",
+ "visitCount": "Obiski",
+ "format": "Oblika",
+ "maxBitRate": "Maks. bitna hitrost",
+ "updatedAt": "Posodobljeno ob",
+ "createdAt": "Ustvarjeno ob",
+ "downloadable": "Dovoli prenose?"
+ }
+ },
+ "missing": {
+ "name": "Manjkajoča datoteka |||| Manjkajoče datoteke",
+ "fields": {
+ "path": "Pot",
+ "size": "Velikost",
+ "updatedAt": "Izginil",
+ "libraryName": "Knjižnica"
+ },
+ "actions": {
+ "remove": "Odstrani",
+ "remove_all": "Odstrani vse"
+ },
+ "notifications": {
+ "removed": "Manjkajoče datoteke odstranjene"
+ },
+ "empty": "Brez manjkajočih datotek"
+ },
+ "library": {
+ "name": "Knjižnica |||| Knjižnice",
+ "fields": {
+ "name": "Ime",
+ "path": "Pot",
+ "remotePath": "Oddaljena pot",
+ "lastScanAt": "Zadnje skeniranje",
+ "songCount": "Pesmi",
+ "albumCount": "Albumi",
+ "artistCount": "Umetniki",
+ "totalSongs": "Pesmi",
+ "totalAlbums": "Albumi",
+ "totalArtists": "Umetniki",
+ "totalFolders": "Mape",
+ "totalFiles": "Datoteke",
+ "totalMissingFiles": "Manjkajoče datoteke",
+ "totalSize": "Skupna velikost",
+ "totalDuration": "Trajanje",
+ "defaultNewUsers": "Privzeto za nove uporabnike",
+ "createdAt": "Ustvarjeno",
+ "updatedAt": "Posodobljeno"
+ },
+ "sections": {
+ "basic": "Osnovne informacije",
+ "statistics": "Statistika"
+ },
+ "actions": {
+ "scan": "Skeniraj knjižnico",
+ "manageUsers": "Upravljanje dostopa uporabnikov",
+ "viewDetails": "Ogled podrobnosti"
+ },
+ "notifications": {
+ "created": "Knjižnica je uspešno ustvarjena",
+ "updated": "Knjižnica je bila uspešno posodobljena",
+ "deleted": "Knjižnica je uspešno izbrisana",
+ "scanStarted": "Skeniranje knjižnice se je začelo",
+ "scanCompleted": "Skeniranje knjižnice končano"
+ },
+ "validation": {
+ "nameRequired": "Ime knjižnice je obvezno",
+ "pathRequired": "Pot do knjižnice je obvezna",
+ "pathNotDirectory": "Pot do knjižnice mora biti imenik",
+ "pathNotFound": "Pot do knjižnice ni bila najdena",
+ "pathNotAccessible": "Pot do knjižnice ni dostopna",
+ "pathInvalid": "Neveljavna pot do knjižnice"
+ },
+ "messages": {
+ "deleteConfirm": "Ali ste prepričani, da želite izbrisati to knjižnico? S tem boste odstranili vse povezane podatke in dostop uporabnikov.",
+ "scanInProgress": "Skeniranje v teku...",
+ "noLibrariesAssigned": "Uporabnik nima dodeljenih knjižnic"
+ }
}
+ },
+ "ra": {
+ "auth": {
+ "welcome1": "Hvala, da ste naložili Navidrome!",
+ "welcome2": "Za začetek, ustvarite upraviteljski račun",
+ "confirmPassword": "Potrdi Geslo",
+ "buttonCreateAdmin": "Ustvari upravitelja",
+ "auth_check_error": "Vpišite se za nadaljevanje",
+ "user_menu": "Profil",
+ "username": "Uporabnik",
+ "password": "Geslo",
+ "sign_in": "Vpis",
+ "sign_in_error": "Avtentikacija neuspešna, poskusite ponovno",
+ "logout": "Izpis",
+ "insightsCollectionNote": "Navidrome zbira anonimne podatke o uporabi \nz namenom izboljšanja projekta. \nKliknite [tukaj], če želite izvedeti več ali se odjaviti"
+ },
+ "validation": {
+ "invalidChars": "Uporabi samo alfanumerične znake",
+ "passwordDoesNotMatch": "Geslo se ne ujema",
+ "required": "Potreben",
+ "minLength": "Potrebnih je vsaj %{min} znakov",
+ "maxLength": "Potrebnih je največ %{max}",
+ "minValue": "Potrebnih je vsaj %{min}",
+ "maxValue": "Potrebnih je največ %{max}",
+ "number": "Mora biti številka",
+ "email": "Veljaven e-poštni naslov",
+ "oneOf": "Mora biti ena izmed %{options}",
+ "regex": "Mora se ujemati z določeno obliko (regexp): %{pattern}",
+ "unique": "Mora biti edinstven",
+ "url": "Biti mora veljaven URL"
+ },
+ "action": {
+ "add_filter": "Dodaj filter",
+ "add": "Dodaj",
+ "back": "Nazaj",
+ "bulk_actions": "Izbran 1 element |||| Izbranih %{smart_count} elementov",
+ "cancel": "Prekliči",
+ "clear_input_value": "Pobriši",
+ "clone": "Podvoji",
+ "confirm": "Potrdi",
+ "create": "Ustvari",
+ "delete": "Izbriši",
+ "edit": "Uredi",
+ "export": "Izvozi",
+ "list": "Seznam",
+ "refresh": "Osveži",
+ "remove_filter": "Odstrani filter",
+ "remove": "Odstrani",
+ "save": "Shrani",
+ "search": "Išči",
+ "show": "Prikaži",
+ "sort": "Razvrsti",
+ "undo": "Razveljavi",
+ "expand": "Razširi",
+ "close": "Zapri",
+ "open_menu": "Odpri meni",
+ "close_menu": "Zapri meni",
+ "unselect": "Prekliči izbiro",
+ "skip": "Izpusti",
+ "bulk_actions_mobile": "1 |||| %{smart_count}",
+ "share": "Deli",
+ "download": "Prenesi"
+ },
+ "boolean": {
+ "true": "Da",
+ "false": "Ne"
+ },
+ "page": {
+ "create": "Ustvari %{name}",
+ "dashboard": "Nadzorna plošča",
+ "edit": "%{name} #%{id}",
+ "error": "Nedoločena napaka",
+ "list": "%{name}",
+ "loading": "Nalagam",
+ "not_found": "Ni zadetka",
+ "show": "%{name} #%{id}",
+ "empty": "Še brez %{name}.",
+ "invite": "Ga želite dodati?"
+ },
+ "input": {
+ "file": {
+ "upload_several": "Povlecite datoteke ali pa kliknite in izberite.",
+ "upload_single": "Povlecite datoteko ali pa kliknite in izberite."
+ },
+ "image": {
+ "upload_several": "Povlecite slike, ali pa kliknite in izberite.",
+ "upload_single": "Povlecite sliko, ali pa kliknite in izberite."
+ },
+ "references": {
+ "all_missing": "Ne najdem referenciranih podatkov.",
+ "many_missing": "Zdi se, da vsaj ena asociirana referenca ni več na voljo.",
+ "single_missing": "Zdi se, da asociirana referenca ni več na voljo."
+ },
+ "password": {
+ "toggle_visible": "Skrij geslo",
+ "toggle_hidden": "Prikaži geslo"
+ }
+ },
+ "message": {
+ "about": "O programu",
+ "are_you_sure": "Ste prepričani?",
+ "bulk_delete_content": "Ste prepričani, da želite izbrisati %{name}? |||| Ste prepričani, da želite izbrisati %{smart_count} elementov?",
+ "bulk_delete_title": "Izbriši %{name} |||| Izbriši %{smart_count} %{name}",
+ "delete_content": "Ste prepričani, da želite izbrisati ta element?",
+ "delete_title": "Izbriši %{name} #%{id}",
+ "details": "Podrobnosti",
+ "error": "Napak klijenta. Vaš zahtevek se je zaključil neuspešno.",
+ "invalid_form": "Oblika ni veljavna. Prosim preverite napake",
+ "loading": "Stran se nalaga, trenutek",
+ "no": "Ne",
+ "not_found": "Ali ste vtipkali napačen naslov (URL), ali pa sledili neobstoječi povezavi.",
+ "yes": "Da",
+ "unsaved_changes": "Nekate spremembe se niso shranile. Ste prepričani, da jih želite ignorirati?"
+ },
+ "navigation": {
+ "no_results": "Ni zadetkov",
+ "no_more_results": "Številka strani %{page} je zunaj meja. Preizkusite prejšnjo stran.",
+ "page_out_of_boundaries": "Številka strani %{page} je zunaj meja",
+ "page_out_from_end": "Ne gre dalje od zadnje strani",
+ "page_out_from_begin": "Ne gre pred prvo stran",
+ "page_range_info": "%{offsetBegin}-%{offsetEnd} od %{total}",
+ "page_rows_per_page": "Elementov na stran:",
+ "next": "Naslednji",
+ "prev": "Prejšnji",
+ "skip_nav": "Preskoči k vsebini"
+ },
+ "notification": {
+ "updated": "Element posodobljen |||| Posodobljenih %{smart_count} elementov",
+ "created": "Element dodan",
+ "deleted": "Element izbrisan |||| %{smart_count} elementov izbrisanih",
+ "bad_item": "Nepravilen element",
+ "item_doesnt_exist": "Element ne obstaja",
+ "http_error": "Strežnika napaka v komunikaciji",
+ "data_provider_error": "Napaka dataProvider error. Preverite konzolo za podrobnosti.",
+ "i18n_error": "Ne uspem naložiti prevode za izbran jezik",
+ "canceled": "Akcija preklicana",
+ "logged_out": "Seja je potekla, prosim povežite se ponovno.",
+ "new_version": "Na voljo je nova verzija! Prosim osvežite okno."
+ },
+ "toggleFieldsMenu": {
+ "columnsToDisplay": "Prikaži stolpce",
+ "layout": "Razporeditev",
+ "grid": "Mreža",
+ "table": "Tabela"
+ }
+ },
+ "message": {
+ "note": "OPOMBA",
+ "transcodingDisabled": "Sprememba konfiguracije transkodiranja skozi spletni vmesnik je onemogočeno zaradi varnostnih razlogov. Če želite spremeniti (urediti ali izbrisati) možnosti transkodiranja, ponovno zaženite strežnik z %{config} nastavitvami.",
+ "transcodingEnabled": "Navidrome trenutno uporablja nastavitve %{config}, kar pomeni da je možno pognati sistemske ukaze v nastavitvah transkodiranja preko spletnega vmesnika.\nZaradi varnostnih razlogov je možnost priporočeno onemogočiti , razen v primeru spreminjanja nastavitev.",
+ "songsAddedToPlaylist": "Dodaj pesem na seznam predvajanj |||| Dodaj %{smart_count} pesmi na seznam predvajanj",
+ "noPlaylistsAvailable": "Ni seznamov",
+ "delete_user_title": "Odstrani uporabnika '%{name}'",
+ "delete_user_content": "Ste prepričani o izbrisu uporabnika, vključno z njegovimi podatki (tudi seznami predvajanj in nastavitvami)?",
+ "notifications_blocked": "V vašem brskljalniku Imate blokirana možnost obvestil za to spletno stran",
+ "notifications_not_available": "Vaš brskljalnik ne omogoča obvestil na namizju ali pa do Navidrome ne dostopate po varni povezavi (https)",
+ "lastfmLinkSuccess": "Last.fm uspešno povezan in 'scrobbling' omogočen",
+ "lastfmLinkFailure": "Last.fm ni uspešno povezan",
+ "lastfmUnlinkSuccess": "Last.fm povezava prekinjena in 'scrobbling' onemogočen",
+ "lastfmUnlinkFailure": "Last.fm povezava neuspešno prekinjena",
+ "openIn": {
+ "lastfm": "Odpri v Last.fm",
+ "musicbrainz": "Odpri v MusicBrainz"
+ },
+ "lastfmLink": "Preberi več...",
+ "listenBrainzLinkSuccess": "ListenBrainz uspešno povezan in scrobbling vključen za uporabnika: %{user}",
+ "listenBrainzLinkFailure": "ListBrainz neuspešno povezan: %{error}",
+ "listenBrainzUnlinkSuccess": "ListenBrainz povezava prekinjena in scrobbling izključen",
+ "listenBrainzUnlinkFailure": "ListenBrainz prekinitev povezave neuspešna",
+ "downloadOriginalFormat": "Prenesi v izvirni obliki",
+ "shareOriginalFormat": "Deli v izvirni obliki",
+ "shareDialogTitle": "Deli %{resource} '%{name}'",
+ "shareBatchDialogTitle": "Deli 1 %{resource} |||| Deli %{smart_count} %{resource}",
+ "shareSuccess": "URL kopiran v odložišče: %{url}",
+ "shareFailure": "Napaka pri kopiranju URL-ja %{url} v odložišče",
+ "downloadDialogTitle": "Prenesi %{resource} '%{name}' (%{size})",
+ "shareCopyToClipboard": "Kopiraj v odložišče: Ctrl+C, Enter",
+ "remove_missing_title": "Odstrani manjkajoče datoteke",
+ "remove_missing_content": "Ste prepričani, da želite odstraniti izbrane manjkajoče datoteke iz baze? Trajno boste odstranili vse reference nanje, vključno s številom predvajanj in ocenami.",
+ "remove_all_missing_title": "Odstrani vse manjkajoče datoteke",
+ "remove_all_missing_content": "Ste prepričani, da želite odstraniti vse manjkajoče datoteke iz baze? Trajno boste odstranili vse reference nanje, vključno s številom predvajanj in ocenami.",
+ "noSimilarSongsFound": "Ni najdenih podobnih pesmi",
+ "noTopSongsFound": "Ni najdenih najboljših pesmi"
+ },
+ "menu": {
+ "library": "Knjižnica",
+ "settings": "Nastavitve",
+ "version": "Različica",
+ "theme": "Tema",
+ "personal": {
+ "name": "Osebno",
+ "options": {
+ "theme": "Tema",
+ "language": "Jezik",
+ "defaultView": "Privzet pogled",
+ "desktop_notifications": "Namizna obvestila",
+ "lastfmScrobbling": "'Scrobble' do Last.fm",
+ "listenBrainzScrobbling": "Scrobble k ListenBrainz",
+ "replaygain": "ReplayGain način",
+ "preAmp": "ReplayGain PreAmp (dB)",
+ "gain": {
+ "none": "Onemogočeno",
+ "album": "Uporabi Album Gain",
+ "track": "Uporabi Track Gain"
+ },
+ "lastfmNotConfigured": "Last.fm API ključ ni konfiguriran"
+ }
+ },
+ "albumList": "Albumi",
+ "about": "O programu",
+ "playlists": "Seznami predvajanj",
+ "sharedPlaylists": "Deljeni seznami predvajanj",
+ "librarySelector": {
+ "allLibraries": "Vse knjižnice (%{count})",
+ "multipleLibraries": "%{selected} od %{total} knjižnic",
+ "selectLibraries": "Izberite knjižnice",
+ "none": "Nobena"
+ }
+ },
+ "player": {
+ "playListsText": "Predvajaj vrsto",
+ "openText": "Odpri",
+ "closeText": "Zapri",
+ "notContentText": "Ni glasbe",
+ "clickToPlayText": "Predvajaj",
+ "clickToPauseText": "Premor predvajanja",
+ "nextTrackText": "Naslednje predvajanje",
+ "previousTrackText": "Prejšnji",
+ "reloadText": "Ponovno naloži",
+ "volumeText": "Glasnost",
+ "toggleLyricText": "Preklopi besedila",
+ "toggleMiniModeText": "Pomanjšaj",
+ "destroyText": "Uniči",
+ "downloadText": "Naloži",
+ "removeAudioListsText": "Izbriši avdio seznam",
+ "clickToDeleteText": "Klikni za izbris %{name}",
+ "emptyLyricText": "Ni besedila",
+ "playModeText": {
+ "order": "Po vrsti",
+ "orderLoop": "Ponavljaj",
+ "singleLoop": "Ponovi enkrat",
+ "shufflePlay": "Premešaj"
+ }
+ },
+ "about": {
+ "links": {
+ "homepage": "Domača stran",
+ "source": "Izvorna koda",
+ "featureRequests": "Funkcionalni zahtevki",
+ "lastInsightsCollection": "Zbirka zadnjih vpogledov",
+ "insights": {
+ "disabled": "Onemogočeno",
+ "waiting": "Čakanje"
+ }
+ },
+ "tabs": {
+ "about": "O nas",
+ "config": "Konfiguracija"
+ },
+ "config": {
+ "configName": "Ime konfiguracije",
+ "environmentVariable": "Spremenljivka okolja",
+ "currentValue": "Trenutna vrednost",
+ "configurationFile": "Konfiguracijska datoteka",
+ "exportToml": "Izvozi konfiguracijo (TOML)",
+ "exportSuccess": "Konfiguracija izvožena v odložišče v formatu TOML",
+ "exportFailed": "Kopiranje konfiguracije ni uspelo",
+ "devFlagsHeader": "Razvojne zastavice (lahko se spremenijo/odstranijo)",
+ "devFlagsComment": "To so eksperimentalne nastavitve in bodo morda odstranjene v prihodnjih različicah"
+ }
+ },
+ "activity": {
+ "title": "Aktivnost",
+ "totalScanned": "Skupaj preiskanih map",
+ "quickScan": "Hitro preišči",
+ "fullScan": "Polno preišči",
+ "serverUptime": "Čas delovanja",
+ "serverDown": "NEPOVEZAN",
+ "scanType": "Tip",
+ "status": "Napaka pri skeniranju",
+ "elapsedTime": "Pretečeni čas"
+ },
+ "help": {
+ "title": "Hitre tipke",
+ "hotkeys": {
+ "show_help": "Prikaži pomoč",
+ "toggle_menu": "Preklopi stransko vrstico menija",
+ "toggle_play": "Predvajaj / Pavza",
+ "prev_song": "Prejšnja",
+ "next_song": "Naslednja",
+ "vol_up": "Zvišaj glasnost",
+ "vol_down": "Znižaj glasnost",
+ "toggle_love": "Dodaj med priljubljene",
+ "current_song": "Skoči na predvajano"
+ }
+ },
+ "nowPlaying": {
+ "title": "Zdaj se predvaja",
+ "empty": "Nič se ne predvaja",
+ "minutesAgo": "Pred %{smart_count} minuto |||| Pred %{smart_count} minutami"
+ }
}
\ No newline at end of file
diff --git a/resources/i18n/sv.json b/resources/i18n/sv.json
index 9392706cb..521f997a8 100644
--- a/resources/i18n/sv.json
+++ b/resources/i18n/sv.json
@@ -26,7 +26,17 @@
"bpm": "BPM",
"playDate": "Senast spelad",
"channels": "Channels",
- "createdAt": "Skapad"
+ "createdAt": "Skapad",
+ "grouping": "Gruppering",
+ "mood": "Stämning",
+ "participants": "Ytterligare medverkande",
+ "tags": "Ytterligare taggar",
+ "mappedTags": "Mappade taggar",
+ "rawTags": "Omodifierade taggar",
+ "bitDepth": "Bitdjup",
+ "sampleRate": "Samplingsfrekvens",
+ "missing": "Saknade",
+ "libraryName": "Bibliotek"
},
"actions": {
"addToQueue": "Lägg till i kön",
@@ -35,7 +45,8 @@
"shuffleAll": "Shuffle",
"download": "Ladda ner",
"playNext": "Spela nästa",
- "info": "Mer information"
+ "info": "Mer information",
+ "showInPlaylist": "Visa i spellista"
}
},
"album": {
@@ -58,7 +69,16 @@
"originalDate": "Originaldatum",
"releaseDate": "Utgivningsdatum",
"releases": "Utgåva |||| Utgåvor",
- "released": "Utgiven"
+ "released": "Utgiven",
+ "recordLabel": "Skivbolag",
+ "catalogNum": "Katalognummer",
+ "releaseType": "Typ",
+ "grouping": "Gruppering",
+ "media": "Media",
+ "mood": "Stämning",
+ "date": "Inspelningsdatum",
+ "missing": "Saknade",
+ "libraryName": "Bibliotek"
},
"actions": {
"playAll": "Spela",
@@ -89,7 +109,30 @@
"playCount": "Spelningar",
"rating": "Betyg",
"genre": "Genre",
- "size": "Storlek"
+ "size": "Storlek",
+ "role": "Roll",
+ "missing": "Saknade"
+ },
+ "roles": {
+ "albumartist": "Albumartist |||| Albumartister",
+ "artist": "Artist |||| Artister",
+ "composer": "Kompositör |||| Kompositörer",
+ "conductor": "Dirigent |||| Dirigenter",
+ "lyricist": "Textförfattare |||| Textförfattare",
+ "arranger": "Arrangör |||| Arrangörer",
+ "producer": "Producent |||| Producenter",
+ "director": "Inspelningsledare |||| Inspelningsledare",
+ "engineer": "Ljudtekniker |||| Ljudtekniker",
+ "mixer": "Mixare |||| Mixare",
+ "remixer": "Remixare |||| Remixare",
+ "djmixer": "DJ-mixare |||| DJ-mixare",
+ "performer": "Utövande artist |||| Utövande artister",
+ "maincredit": "Albumartister eller Artist |||| Albumartister eller Artister"
+ },
+ "actions": {
+ "shuffle": "Shuffle",
+ "radio": "Radio",
+ "topSongs": "Topplåtar"
}
},
"user": {
@@ -106,10 +149,12 @@
"currentPassword": "Nuvarande lösenord",
"newPassword": "Nytt lösenord",
"token": "Token",
- "lastAccessAt": "Senaste åtkomst"
+ "lastAccessAt": "Senaste åtkomst",
+ "libraries": "Bibliotek"
},
"helperTexts": {
- "name": "Ändringar av ditt namn syns först vid nästa inloggning"
+ "name": "Ändringar av ditt namn syns först vid nästa inloggning",
+ "libraries": "Välj ett bibliotek för denna användare eller lämna blankt för standardbibliotek"
},
"notifications": {
"created": "Användare skapad",
@@ -118,7 +163,12 @@
},
"message": {
"listenBrainzToken": "Ange din ListenBrainz användar-token.",
- "clickHereForToken": "Klicka här för att hämta din token"
+ "clickHereForToken": "Klicka här för att hämta din token",
+ "selectAllLibraries": "Välj alla bibliotek",
+ "adminAutoLibraries": "Administratörer har automatiskt tillgång till alla bibliotek"
+ },
+ "validation": {
+ "librariesRequired": "Minst ett bibliotek måste väljas för icke-administratörer"
}
},
"player": {
@@ -162,11 +212,17 @@
"addNewPlaylist": "Skapa \"%{name}\"",
"export": "Exportera",
"makePublic": "Gör offentlig",
- "makePrivate": "Gör privat"
+ "makePrivate": "Gör privat",
+ "saveQueue": "Spara kö till spellista",
+ "searchOrCreate": "Sök spellista eller skapa ny...",
+ "pressEnterToCreate": "Tryck Enter för att skapa ny spellista",
+ "removeFromSelection": "Ta bort från urval"
},
"message": {
"duplicate_song": "Lägg till dubletter",
- "song_exist": "Vissa låtar finns redan i spellistan. Vill du lägga till dubbletterna eller hoppa över dem?"
+ "song_exist": "Vissa låtar finns redan i spellistan. Vill du lägga till dubbletterna eller hoppa över dem?",
+ "noPlaylistsFound": "Hittade inga spellistor",
+ "noPlaylists": "Inga spellistor tillgängliga"
}
},
"radio": {
@@ -198,6 +254,75 @@
"createdAt": "Skapad",
"downloadable": "Tillåt nedladdning?"
}
+ },
+ "missing": {
+ "name": "Saknad fil |||| Saknade filer",
+ "fields": {
+ "path": "Sökväg",
+ "size": "Storlek",
+ "updatedAt": "Försvann",
+ "libraryName": "Bibliotek"
+ },
+ "actions": {
+ "remove": "Radera",
+ "remove_all": "Radera alla"
+ },
+ "notifications": {
+ "removed": "Saknade fil(er) borttagna"
+ },
+ "empty": "Inga saknade filer"
+ },
+ "library": {
+ "name": "Bibliotek |||| Bibliotek",
+ "fields": {
+ "name": "Namn",
+ "path": "Sökväg",
+ "remotePath": "Ta bort sökväg",
+ "lastScanAt": "Senaste scan",
+ "songCount": "Låtar",
+ "albumCount": "Album",
+ "artistCount": "Artister",
+ "totalSongs": "Låtar",
+ "totalAlbums": "Album",
+ "totalArtists": "Artister",
+ "totalFolders": "Mappar",
+ "totalFiles": "Filer",
+ "totalMissingFiles": "Saknade filer",
+ "totalSize": "Sammanlagd storlek",
+ "totalDuration": "Längd",
+ "defaultNewUsers": "Standard för nya användare",
+ "createdAt": "Skapad",
+ "updatedAt": "Uppdaterad"
+ },
+ "sections": {
+ "basic": "Grundinformation",
+ "statistics": "Statistik"
+ },
+ "actions": {
+ "scan": "Scanna bibliotek",
+ "manageUsers": "Hantera användaråtkomst",
+ "viewDetails": "Se detaljer"
+ },
+ "notifications": {
+ "created": "Biblioteket har skapats",
+ "updated": "Biblioteket har uppdaterats",
+ "deleted": "Biblioteket har raderats",
+ "scanStarted": "Biblioteksscan startad",
+ "scanCompleted": "Biblioteksscan avslutad"
+ },
+ "validation": {
+ "nameRequired": "Biblioteksnamn krävs",
+ "pathRequired": "Bibliotekssökväg krävs",
+ "pathNotDirectory": "Bibliotekssökvägen måste vara en katalog",
+ "pathNotFound": "Bibliotekssökväg hittades inte",
+ "pathNotAccessible": "Bibliotekssökväg inte tillgänglig",
+ "pathInvalid": "Ogiltig bibliotekssökväg"
+ },
+ "messages": {
+ "deleteConfirm": "Är du säker på att du vill ta bort detta bibliotek? Detta raderar all förbunden data och användartillgång.",
+ "scanInProgress": "Scanning pågår...",
+ "noLibrariesAssigned": "Inga bibliotek har tilldelats den här användaren"
+ }
}
},
"ra": {
@@ -375,7 +500,13 @@
"shareSuccess": "URL kopierades till urklipp: %{url}",
"shareFailure": "Fel vid kopiering av URL %{url} till urklipp",
"downloadDialogTitle": "Ladda ner %{resource} '%{name}' (%{size})",
- "shareCopyToClipboard": "Kopiera till urklipp: Ctrl+C, Enter"
+ "shareCopyToClipboard": "Kopiera till urklipp: Ctrl+C, Enter",
+ "remove_missing_title": "Ta bort saknade filer",
+ "remove_missing_content": "Är du säker på att du vill ta bort de valda saknade filerna från databasen? Detta kommer permanent radera alla referenser till dem, inklusive antal spelningar och betyg.",
+ "remove_all_missing_title": "Ta bort alla saknade filer",
+ "remove_all_missing_content": "Är du säker på att du vill ta bort alla saknade filer från databasen? Detta kommer permanent radera alla referenser till dem, inklusive antal spelningar och betyg.",
+ "noSimilarSongsFound": "Hittade inga liknande låtar",
+ "noTopSongsFound": "Hittade inga topplåtar"
},
"menu": {
"library": "Bibliotek",
@@ -404,7 +535,13 @@
"albumList": "Album",
"about": "Om",
"playlists": "Spellistor",
- "sharedPlaylists": "Delade spellistor"
+ "sharedPlaylists": "Delade spellistor",
+ "librarySelector": {
+ "allLibraries": "Alla bibliotek (%{count})",
+ "multipleLibraries": "%{selected} av %{total} bibliotek",
+ "selectLibraries": "Valda bibliotek",
+ "none": "Inga"
+ }
},
"player": {
"playListsText": "Spela kön",
@@ -441,6 +578,21 @@
"disabled": "Inaktiverad",
"waiting": "Väntar"
}
+ },
+ "tabs": {
+ "about": "Om",
+ "config": "Inställningar"
+ },
+ "config": {
+ "configName": "Inställningsnamn",
+ "environmentVariable": "Miljövariabel",
+ "currentValue": "Nuvarande värde",
+ "configurationFile": "Inställningsfil",
+ "exportToml": "Exportera inställningar (TOML)",
+ "exportSuccess": "Inställningarna kopierade till urklippet i TOML-format",
+ "exportFailed": "Kopiering av inställningarna misslyckades",
+ "devFlagsHeader": "Utvecklingsflaggor (kan ändras eller tas bort)",
+ "devFlagsComment": "Dessa inställningar är experimentella och kan tas bort i framtida versioner"
}
},
"activity": {
@@ -449,7 +601,10 @@
"quickScan": "Snabbscan",
"fullScan": "Komplett scan",
"serverUptime": "Serverdrifttid",
- "serverDown": "OFFLINE"
+ "serverDown": "OFFLINE",
+ "scanType": "Typ",
+ "status": "Fel vid scanning",
+ "elapsedTime": "Spelad tid"
},
"help": {
"title": "Navidrome kortkommandon",
@@ -464,5 +619,10 @@
"toggle_love": "Lägg till låt i favoriter",
"current_song": "Hoppa till nuvarande låt"
}
+ },
+ "nowPlaying": {
+ "title": "Spelas nu",
+ "empty": "Inget spelas",
+ "minutesAgo": "%{smart_count} minut sedan |||| %{smart_count} minuter sedan"
}
}
\ No newline at end of file
diff --git a/resources/i18n/th.json b/resources/i18n/th.json
index 2f96f4958..65d51860f 100644
--- a/resources/i18n/th.json
+++ b/resources/i18n/th.json
@@ -26,7 +26,17 @@
"bpm": "BPM",
"playDate": "เล่นล่าสุด",
"channels": "ช่อง",
- "createdAt": "เพิ่มเมื่อ"
+ "createdAt": "เพิ่มเมื่อ",
+ "grouping": "จัดกลุ่ม",
+ "mood": "อารมณ์",
+ "participants": "ผู้มีส่วนร่วม",
+ "tags": "แทกเพิ่มเติม",
+ "mappedTags": "แมพแทก",
+ "rawTags": "แทกเริ่มต้น",
+ "bitDepth": "Bit depth",
+ "sampleRate": "แซมเปิ้ลเรต",
+ "missing": "หายไป",
+ "libraryName": "ห้องสมุด"
},
"actions": {
"addToQueue": "เพิ่มในคิว",
@@ -35,7 +45,8 @@
"shuffleAll": "สุ่มทั้งหมด",
"download": "ดาวน์โหลด",
"playNext": "เล่นถัดไป",
- "info": "ดูรายละเอียด"
+ "info": "ดูรายละเอียด",
+ "showInPlaylist": "แสดงในเพลย์ลิสต์"
}
},
"album": {
@@ -58,7 +69,16 @@
"originalDate": "วันที่เริ่ม",
"releaseDate": "เผยแพร่เมื่อ",
"releases": "เผยแพร่ |||| เผยแพร่",
- "released": "เผยแพร่เมื่อ"
+ "released": "เผยแพร่เมื่อ",
+ "recordLabel": "ป้าย",
+ "catalogNum": "หมายเลขแคตาล็อก",
+ "releaseType": "ประเภท",
+ "grouping": "จัดกลุ่ม",
+ "media": "มีเดีย",
+ "mood": "อารมณ์",
+ "date": "บันทึกเมื่อ",
+ "missing": "หายไป",
+ "libraryName": "ห้องสมุด"
},
"actions": {
"playAll": "เล่นทั้งหมด",
@@ -89,7 +109,30 @@
"playCount": "เล่นแล้ว",
"rating": "ความนิยม",
"genre": "ประเภท",
- "size": "ขนาด"
+ "size": "ขนาด",
+ "role": "Role",
+ "missing": "หายไป"
+ },
+ "roles": {
+ "albumartist": "ศิลปินอัลบั้ม |||| ศิลปินอัลบั้ม",
+ "artist": "ศิลปิน |||| ศิลปิน",
+ "composer": "ผู้แต่ง |||| ผู้แต่ง",
+ "conductor": "คอนดักเตอร์ |||| คอนดักเตอร์",
+ "lyricist": "เนื้อเพลง |||| เนื้อเพลง",
+ "arranger": "ผู้ดำเนินการ |||| ผู้ดำเนินการ",
+ "producer": "ผู้จัด |||| ผู้จัด",
+ "director": "ไดเรกเตอร์ |||| ไดเรกเตอร์",
+ "engineer": "วิศวกร |||| วิศวกร",
+ "mixer": "มิกเซอร์ |||| มิกเซอร์",
+ "remixer": "รีมิกเซอร์ |||| รีมิกเซอร์",
+ "djmixer": "ดีเจมิกเซอร์ |||| ดีเจมิกเซอร์",
+ "performer": "ผู้เล่น |||| ผู้เล่น",
+ "maincredit": "ศิลปิน |||| ศิลปิน"
+ },
+ "actions": {
+ "shuffle": "เล่นสุ่ม",
+ "radio": "วิทยุ",
+ "topSongs": "เพลงยอดนิยม"
}
},
"user": {
@@ -106,10 +149,12 @@
"currentPassword": "รหัสผ่านปัจจุบัน",
"newPassword": "รหัสผ่านใหม่",
"token": "โทเคน",
- "lastAccessAt": "เข้าใช้ล่าสุด"
+ "lastAccessAt": "เข้าใช้ล่าสุด",
+ "libraries": "ห้องสมุด"
},
"helperTexts": {
- "name": "การเปลี่ยนชื่อจะมีผลในการล็อกอินครั้งถัดไป"
+ "name": "การเปลี่ยนชื่อจะมีผลในการล็อกอินครั้งถัดไป",
+ "libraries": "เลือกห้องสมุดสำหรับผู้ใช้นี้หรือปล่อยว่างเพื่อใช้ห้องสมุดเริ่มต้น"
},
"notifications": {
"created": "สร้างชื่อผู้ใช้",
@@ -118,7 +163,12 @@
},
"message": {
"listenBrainzToken": "ใส่โทเคน ListenBrainz ของคุณ",
- "clickHereForToken": "กดที่นี่เพื่อรับโทเคนของคุณ"
+ "clickHereForToken": "กดที่นี่เพื่อรับโทเคนของคุณ",
+ "selectAllLibraries": "เลือกห้องสมุดทั้งหมด",
+ "adminAutoLibraries": "ผู้ดูแลเข้าถึงห้องสมุดทั้งหมดโดยอัตโนมัติ"
+ },
+ "validation": {
+ "librariesRequired": "ต้องเลือกห้องสมุด 1 ห้อง สำหรับผู้ใช้ที่ไม่ใช่ผู้ดูแล"
}
},
"player": {
@@ -162,11 +212,17 @@
"addNewPlaylist": "สร้าง \"%{name}\"",
"export": "ส่งออก",
"makePublic": "ทำเป็นสาธารณะ",
- "makePrivate": "ทำเป็นส่วนตัว"
+ "makePrivate": "ทำเป็นส่วนตัว",
+ "saveQueue": "บันทึกคิวลงเพลย์ลิสต์",
+ "searchOrCreate": "ค้นหาเพลย์ลิสต์หรือพิมพ์เพื่อสร้างใหม่",
+ "pressEnterToCreate": "กด Enter เพื่อสร้างเพลย์ลิสต์",
+ "removeFromSelection": "เอาออกจากที่เลือกไว้"
},
"message": {
"duplicate_song": "เพิ่มเพลงซ้ำ",
- "song_exist": "เพิ่มเพลงซ้ำกันในเพลย์ลิสต์ คุณจะเพิ่มเพลงต่อหรือข้าม"
+ "song_exist": "เพิ่มเพลงซ้ำกันในเพลย์ลิสต์ คุณจะเพิ่มเพลงต่อหรือข้าม",
+ "noPlaylistsFound": "ไม่พบเพลย์ลิสต์",
+ "noPlaylists": "ไม่มีเพลย์ลิสต์อยู่"
}
},
"radio": {
@@ -198,6 +254,75 @@
"createdAt": "สร้างเมื่อ",
"downloadable": "อนุญาตให้ดาวโหลด?"
}
+ },
+ "missing": {
+ "name": "ไฟล์ที่หายไป |||| ไฟล์ที่หายไป",
+ "fields": {
+ "path": "พาร์ท",
+ "size": "ขนาด",
+ "updatedAt": "หายไปจาก",
+ "libraryName": "ห้องสมุด"
+ },
+ "actions": {
+ "remove": "เอาออก",
+ "remove_all": "เอาออกทั้งหมด"
+ },
+ "notifications": {
+ "removed": "เอาไฟล์ที่หายไปออกแล้ว"
+ },
+ "empty": "ไม่มีไฟล์หาย"
+ },
+ "library": {
+ "name": "ห้องสมุด |||| ห้องสมุด",
+ "fields": {
+ "name": "ชื่อ",
+ "path": "พาร์ท",
+ "remotePath": "รีโมทพาร์ท",
+ "lastScanAt": "สแกนล่าสุด",
+ "songCount": "เพลง",
+ "albumCount": "อัลบัม",
+ "artistCount": "ศิลปิน",
+ "totalSongs": "เพลง",
+ "totalAlbums": "อัลบัม",
+ "totalArtists": "ศิลปิน",
+ "totalFolders": "แฟ้ม",
+ "totalFiles": "ไฟล์",
+ "totalMissingFiles": "ไฟล์ที่หายไป",
+ "totalSize": "ขนาดทั้งหมด",
+ "totalDuration": "ความยาว",
+ "defaultNewUsers": "ค่าเริ่มต้นผู้ใช้ใหม่",
+ "createdAt": "สร้าง",
+ "updatedAt": "อัพเดท"
+ },
+ "sections": {
+ "basic": "ข้อมูลเบื้องต้น",
+ "statistics": "สถิติ"
+ },
+ "actions": {
+ "scan": "สแกนห้องสมุด",
+ "manageUsers": "ตั้งค่าการเข้าถึง",
+ "viewDetails": "ดูรายละเอียด"
+ },
+ "notifications": {
+ "created": "สร้างห้องสมุดเรียบร้อย",
+ "updated": "อัพเดทห้องสมุดเรียบร้อย",
+ "deleted": "ลบห้องสมุดเพลงเรียบร้อยแล้ว",
+ "scanStarted": "เริ่มสแกนห้องสมุด",
+ "scanCompleted": "สแกนห้องสมุดเสร็จแล้ว"
+ },
+ "validation": {
+ "nameRequired": "ต้องใส่ชื่อห้องสมุดเพลง",
+ "pathRequired": "ต้องใส่พาร์ทของห้องสมุด",
+ "pathNotDirectory": "พาร์ทของห้องสมุดต้องเป็นแฟ้ม",
+ "pathNotFound": "ไม่เจอพาร์ทของห้องสมุด",
+ "pathNotAccessible": "ไม่สามารถเข้าพาร์ทของห้องสมุด",
+ "pathInvalid": "พาร์ทห้องสมุดไม่ถูก"
+ },
+ "messages": {
+ "deleteConfirm": "คุณแน่ใจว่าจะลบห้องสมุดนี้? นี่จะลบข้อมูลและการเข้าถึงของผู้ใช้ที่เกี่ยวข้องทั้งหมด",
+ "scanInProgress": "กำลังสแกน...",
+ "noLibrariesAssigned": "ไม่มีห้องสมุดสำหรับผู้ใช้นี้"
+ }
}
},
"ra": {
@@ -375,7 +500,13 @@
"shareSuccess": "คัดลอก URL ไปคลิปบอร์ด: %{url}",
"shareFailure": "คัดลอก URL %{url} ไปคลิปบอร์ดผิดพลาด",
"downloadDialogTitle": "ดาวโหลด %{resource} '%{name}' (%{size})",
- "shareCopyToClipboard": "คัดลอกไปคลิปบอร์ด: Ctrl+C, Enter"
+ "shareCopyToClipboard": "คัดลอกไปคลิปบอร์ด: Ctrl+C, Enter",
+ "remove_missing_title": "ลบรายการไฟล์ที่หายไป",
+ "remove_missing_content": "คุณแน่ใจว่าจะเอารายการไฟล์ที่หายไปออกจากดาต้าเบส นี่จะเป็นการลบข้อมูลอ้างอิงทั้งหมดของไฟล์ออกอย่างถาวร",
+ "remove_all_missing_title": "เอารายการไฟล์ที่หายไปออกทั้งหมด",
+ "remove_all_missing_content": "คุณแน่ใจว่าจะเอารายการไฟล์ที่หายไปออกจากดาต้าเบส นี่จะเป็นการลบข้อมูลอ้างอิงทั้งหมดของไฟล์ออกอย่างถาวร",
+ "noSimilarSongsFound": "ไม่มีเพลงคล้ายกัน",
+ "noTopSongsFound": "ไม่พบเพลงยอดนิยม"
},
"menu": {
"library": "ห้องสมุดเพลง",
@@ -404,7 +535,13 @@
"albumList": "อัลบั้ม",
"about": "เกี่ยวกับ",
"playlists": "เพลย์ลิสต์",
- "sharedPlaylists": "เพลย์ลิสต์ที่แบ่งปัน"
+ "sharedPlaylists": "เพลย์ลิสต์ที่แบ่งปัน",
+ "librarySelector": {
+ "allLibraries": "ห้องสมุด (%{count}) ห้อง",
+ "multipleLibraries": "%{selected} ของ %{total} ห้องสมุด",
+ "selectLibraries": "เลือกห้องสมุด",
+ "none": "ไม่มี"
+ }
},
"player": {
"playListsText": "คิวเล่น",
@@ -441,6 +578,21 @@
"disabled": "ปิดการทำงาน",
"waiting": "รอ"
}
+ },
+ "tabs": {
+ "about": "เกี่ยวกับ",
+ "config": "การตั้งค่า"
+ },
+ "config": {
+ "configName": "ชื่อการตั้งค่า",
+ "environmentVariable": "ค่าทั่วไป",
+ "currentValue": "ค่าปัจจุบัน",
+ "configurationFile": "ไฟล์การตั้งค่า",
+ "exportToml": "นำออกการตั้งค่า (TOML)",
+ "exportSuccess": "นำออกการตั้งค่าไปยังคลิปบอร์ดในรูปแบบ TOML แล้ว",
+ "exportFailed": "คัดลอกการตั้งค่าล้มเหลว",
+ "devFlagsHeader": "ปักธงการพัฒนา (อาจมีการเปลี่ยน/เอาออก)",
+ "devFlagsComment": "การตั้งค่านี้อยู่ในช่วงทดลองและอาจจะมีการเอาออกในเวอร์ชั่นหลัง"
}
},
"activity": {
@@ -449,7 +601,10 @@
"quickScan": "สแกนแบบเร็ว",
"fullScan": "สแกนทั้งหมด",
"serverUptime": "เซิร์ฟเวอร์ออนไลน์นาน",
- "serverDown": "ออฟไลน์"
+ "serverDown": "ออฟไลน์",
+ "scanType": "ประเภท",
+ "status": "สแกนผิดพลาด",
+ "elapsedTime": "เวลาที่ใช้"
},
"help": {
"title": "คีย์ลัด Navidrome",
@@ -464,5 +619,10 @@
"toggle_love": "เพิ่มเพลงนี้ไปยังรายการโปรด",
"current_song": "ไปยังเพลงปัจจุบัน"
}
+ },
+ "nowPlaying": {
+ "title": "กำลังเล่น",
+ "empty": "ไม่มีเพลงเล่น",
+ "minutesAgo": "%{smart_count} นาทีที่แล้ว |||| %{smart_count} นาทีที่แล้ว"
}
}
\ No newline at end of file
diff --git a/resources/i18n/tr.json b/resources/i18n/tr.json
index cb1a0262d..7c1a82c08 100644
--- a/resources/i18n/tr.json
+++ b/resources/i18n/tr.json
@@ -34,7 +34,9 @@
"mappedTags": "Eşlenen etiketler",
"rawTags": "Ham etiketler",
"bitDepth": "Bit derinliği",
- "sampleRate": "Örnekleme Oranı"
+ "sampleRate": "Örnekleme Oranı",
+ "missing": "Eksik",
+ "libraryName": "Kütüphane"
},
"actions": {
"addToQueue": "Oynatma Sırasına Ekle",
@@ -43,7 +45,8 @@
"shuffleAll": "Tümünü karıştır",
"download": "İndir",
"playNext": "Dinlenenden Sonra Oynat",
- "info": "Bilgiler"
+ "info": "Bilgiler",
+ "showInPlaylist": "Çalma Listesinde Göster"
}
},
"album": {
@@ -73,7 +76,9 @@
"grouping": "Gruplama",
"media": "Medya",
"mood": "Mod",
- "date": "Kayıt Tarihi"
+ "date": "Kayıt Tarihi",
+ "missing": "Eksik",
+ "libraryName": "Kütüphane"
},
"actions": {
"playAll": "Oynat",
@@ -105,7 +110,8 @@
"rating": "Derecelendirme",
"genre": "Tür",
"size": "Boyut",
- "role": "Rol"
+ "role": "Rol",
+ "missing": "Eksik"
},
"roles": {
"albumartist": "Albüm Sanatçısı |||| Albüm Sanatçısı",
@@ -120,7 +126,13 @@
"mixer": "Mikser |||| Mikser",
"remixer": "Remiks |||| Remiks",
"djmixer": "DJ Mikseri |||| DJ Mikseri",
- "performer": "Sanatçı |||| Sanatçı"
+ "performer": "Sanatçı |||| Sanatçı",
+ "maincredit": "Albüm Sanatçısı veya Sanatçı |||| Albüm Sanatçısı veya Sanatçılar"
+ },
+ "actions": {
+ "shuffle": "Karıştır",
+ "radio": "Radyo",
+ "topSongs": "En İyi Şarkılar"
}
},
"user": {
@@ -137,10 +149,12 @@
"currentPassword": "Mevcut Şifre",
"newPassword": "Yeni Şifre",
"token": "Token",
- "lastAccessAt": "Son Erişim Tarihi"
+ "lastAccessAt": "Son Erişim Tarihi",
+ "libraries": "Kütüphaneler"
},
"helperTexts": {
- "name": "Adınızda yaptığımız değişikliğin geçerli olması için tekrar giriş yapmanız gerekmektedir"
+ "name": "Adınızda yaptığımız değişikliğin geçerli olması için tekrar giriş yapmanız gerekmektedir",
+ "libraries": "Bu kullanıcı için belirli kütüphaneleri seçin veya varsayılan kütüphaneleri kullanmak için boş bırakın"
},
"notifications": {
"created": "Kullanıcı oluşturuldu",
@@ -149,7 +163,12 @@
},
"message": {
"listenBrainzToken": "ListenBrainz kullanıcı Token'ınızı girin.",
- "clickHereForToken": "Token almak için buraya tıklayın"
+ "clickHereForToken": "Token almak için buraya tıklayın",
+ "selectAllLibraries": "Tüm kütüphaneleri seç",
+ "adminAutoLibraries": "Yönetici yetkili kullanıcılar tüm kütüphanelere otomatik olarak erişebilir"
+ },
+ "validation": {
+ "librariesRequired": "Yönetici olmayan kullanıcılar için en az bir kütüphane seçilmelidir"
}
},
"player": {
@@ -193,11 +212,17 @@
"addNewPlaylist": "Oluştur \"%{name}\"",
"export": "Aktar",
"makePublic": "Herkese Açık Yap",
- "makePrivate": "Özel Yap"
+ "makePrivate": "Özel Yap",
+ "saveQueue": "Kuyruktakileri Çalma Listesine Kaydet",
+ "searchOrCreate": "Çalma listelerini arayın veya yenisini oluşturmak için yazın...",
+ "pressEnterToCreate": "Yeni çalma listesi oluşturmak için Enter'a basın",
+ "removeFromSelection": "Seçimden kaldır"
},
"message": {
"duplicate_song": "Yinelenen şarkıları ekle",
- "song_exist": "Seçili müziklerin bazıları eklemek istediğin çalma listesinde mevcut. Yine de eklemek ister misin ?"
+ "song_exist": "Seçili müziklerin bazıları eklemek istediğin çalma listesinde mevcut. Yine de eklemek ister misin ?",
+ "noPlaylistsFound": "Hiç çalma listesi bulunamadı",
+ "noPlaylists": "Çalma listesi mevcut değil"
}
},
"radio": {
@@ -235,15 +260,69 @@
"fields": {
"path": "Yol",
"size": "Boyut",
- "updatedAt": "Kaybolma"
+ "updatedAt": "Kaybolma",
+ "libraryName": "Kütüphane"
},
"actions": {
- "remove": "Kaldır"
+ "remove": "Kaldır",
+ "remove_all": "Tümünü Kaldır"
},
"notifications": {
"removed": "Eksik dosya(lar) kaldırıldı"
},
"empty": "Eksik Dosya Yok"
+ },
+ "library": {
+ "name": "Kütüphane |||| Kütüphaneler",
+ "fields": {
+ "name": "İsim",
+ "path": "Yol",
+ "remotePath": "Uzak Yol",
+ "lastScanAt": "Son Tarama",
+ "songCount": "Şarkılar",
+ "albumCount": "Albümler",
+ "artistCount": "Sanatçılar",
+ "totalSongs": "Şarkılar",
+ "totalAlbums": "Albümler",
+ "totalArtists": "Sanatçılar",
+ "totalFolders": "Klasörler",
+ "totalFiles": "Dosyalar",
+ "totalMissingFiles": "Eksik Dosyalar",
+ "totalSize": "Toplam Boyut",
+ "totalDuration": "Süre",
+ "defaultNewUsers": "Yeni Kullanıcılar için Varsayılan",
+ "createdAt": "Oluşturuldu",
+ "updatedAt": "Güncellendi"
+ },
+ "sections": {
+ "basic": "Temel Bilgiler",
+ "statistics": "İstatistikler"
+ },
+ "actions": {
+ "scan": "Kütüphaneyi Tara",
+ "manageUsers": "Kullanıcı Erişimini Yönet",
+ "viewDetails": "Ayrıntıları Görüntüle"
+ },
+ "notifications": {
+ "created": "Kütüphane başarıyla oluşturuldu",
+ "updated": "Kütüphane başarıyla güncellendi",
+ "deleted": "Kütüphane başarıyla silindi",
+ "scanStarted": "Kütüphane taraması başladı",
+ "scanCompleted": "Kütüphane taraması tamamlandı"
+ },
+ "validation": {
+ "nameRequired": "Kütüphane adı gereklidir",
+ "pathRequired": "Kütüphane yolu gereklidir",
+ "pathNotDirectory": "Kütüphane yolu bir dizin olmalıdır",
+ "pathNotFound": "Kütüphane yolu bulunamadı",
+ "pathNotAccessible": "Kütüphane yoluna erişim sağlanamıyor",
+ "pathInvalid": "Geçersiz kütüphane yolu"
+ },
+ "messages": {
+ "deleteConfirm": "Bu kütüphaneyi silmek istediğinizden emin misiniz? Bu işlem, ilgili tüm verileri ve kullanıcı erişimini kaldıracaktır.",
+ "scanInProgress": "Tarama devam ediyor...",
+ "noLibrariesAssigned": "Bu kullanıcıya hiçbir kütüphane atanmadı"
+ }
}
},
"ra": {
@@ -423,7 +502,11 @@
"downloadDialogTitle": "%{resource}: '%{name}' (%{size}) dosyasını indirin",
"shareCopyToClipboard": "Panoya kopyala: Ctrl+C, Enter",
"remove_missing_title": "Eksik dosyaları kaldır",
- "remove_missing_content": "Seçili eksik dosyaları veritabanından kaldırmak istediğinizden emin misiniz? Bu, oynatma sayıları ve derecelendirmeleri dahil olmak üzere bunlara ilişkin tüm referansları kalıcı olarak kaldıracaktır."
+ "remove_missing_content": "Seçili eksik dosyaları veritabanından kaldırmak istediğinizden emin misiniz? Bu, oynatma sayıları ve derecelendirmeleri dahil olmak üzere bunlara ilişkin tüm referansları kalıcı olarak kaldıracaktır.",
+ "remove_all_missing_title": "Tüm eksik dosyaları kaldırın",
+ "remove_all_missing_content": "Veritabanından tüm eksik dosyaları kaldırmak istediğinizden emin misiniz? Bu, oynatma sayısı ve derecelendirmelerde dahil olmak üzere bunlara ilişkili tüm değerleri kalıcı olarak kaldıracaktır.",
+ "noSimilarSongsFound": "Benzer şarkı bulunamadı",
+ "noTopSongsFound": "En iyi şarkı listesi boş"
},
"menu": {
"library": "Kütüphane",
@@ -452,7 +535,13 @@
"albumList": "Albümler",
"about": "Hakkında",
"playlists": "Çalma Listeleri",
- "sharedPlaylists": "Paylaşılan Çalma Listeleri"
+ "sharedPlaylists": "Paylaşılan Çalma Listeleri",
+ "librarySelector": {
+ "allLibraries": "Tüm Kitaplıklar (%{count})",
+ "multipleLibraries": "%{total} kütüphaneden %{selected} tanesi seçildi",
+ "selectLibraries": "Seçili Kütüphaneler",
+ "none": "Hiçbiri"
+ }
},
"player": {
"playListsText": "Oynatma Sırası",
@@ -489,6 +578,21 @@
"disabled": "Pasif",
"waiting": "Bekle"
}
+ },
+ "tabs": {
+ "about": "Hakkında",
+ "config": "Yapılandırma"
+ },
+ "config": {
+ "configName": "Yapılandırma Adı",
+ "environmentVariable": "Çevre Değişkeni",
+ "currentValue": "Güncel Değer",
+ "configurationFile": "Yapılandırma Dosyası",
+ "exportToml": "Yapılandırmayı Dışa Aktar (TOML)",
+ "exportSuccess": "Yapılandırma TOML formatında dışa aktarıldı",
+ "exportFailed": "Yapılandırma kopyalanamadı",
+ "devFlagsHeader": "Geliştirme Bayrakları (değişime/kaldırılmaya tabidir)",
+ "devFlagsComment": "Bunlar deneysel ayarlardır ve gelecekteki sürümlerde kaldırılabilir"
}
},
"activity": {
@@ -497,7 +601,10 @@
"quickScan": "Hızlı Tarama",
"fullScan": "Tam Tarama",
"serverUptime": "Sunucu Çalışma Süresi",
- "serverDown": "ÇEVRİMDIŞI"
+ "serverDown": "ÇEVRİMDIŞI",
+ "scanType": "Tür",
+ "status": "Tarama Hatası",
+ "elapsedTime": "Geçen Süre"
},
"help": {
"title": "Navidrome Kısayolları",
@@ -512,5 +619,10 @@
"toggle_love": "Bu şarkıyı favorilere ekle",
"current_song": "Mevcut Şarkıya Git"
}
+ },
+ "nowPlaying": {
+ "title": "Şu An Çalıyor",
+ "empty": "Çalan şarkı yok",
+ "minutesAgo": "%{smart_count} dakika önce"
}
}
\ No newline at end of file
diff --git a/resources/i18n/uk.json b/resources/i18n/uk.json
index d0c4713e3..c500a7457 100644
--- a/resources/i18n/uk.json
+++ b/resources/i18n/uk.json
@@ -32,7 +32,11 @@
"participants": "Додаткові вчасники",
"tags": "Додаткові теги",
"mappedTags": "Зіставлені теги",
- "rawTags": "Вихідні теги"
+ "rawTags": "Вихідні теги",
+ "bitDepth": "Глибина розрядності",
+ "sampleRate": "Частота дискретизації",
+ "missing": "Поле відсутнє",
+ "libraryName": "Бібліотека"
},
"actions": {
"addToQueue": "Прослухати пізніше",
@@ -41,7 +45,8 @@
"shuffleAll": "Перемішати",
"download": "Завантажити",
"playNext": "Наступна",
- "info": "Отримати інформацію"
+ "info": "Отримати інформацію",
+ "showInPlaylist": "Показати у плейлісті"
}
},
"album": {
@@ -70,7 +75,10 @@
"releaseType": "Тип",
"grouping": "Групування",
"media": "Медіа",
- "mood": "Настрій"
+ "mood": "Настрій",
+ "date": "Дата запису",
+ "missing": "Поле відсутнє",
+ "libraryName": "Бібліотека"
},
"actions": {
"playAll": "Прослухати",
@@ -102,7 +110,8 @@
"rating": "Рейтинг",
"genre": "Жанр",
"size": "Розмір",
- "role": "Роль"
+ "role": "Роль",
+ "missing": "Поле відсутнє"
},
"roles": {
"albumartist": "Виконавець альбому |||| Виконавці альбому",
@@ -117,7 +126,13 @@
"mixer": "Звукоінженер |||| Звукоінженери",
"remixer": "Реміксер |||| Реміксери",
"djmixer": "DJ-звукоінженер |||| DJ-звукоінженери",
- "performer": "Виконавець |||| Виконавці"
+ "performer": "Виконавець |||| Виконавці",
+ "maincredit": "Виконавець альбому або Виконавець |||| Виконавці альбому або Виконавці"
+ },
+ "actions": {
+ "shuffle": "Перетасовка",
+ "radio": "Радіо",
+ "topSongs": "ТОП-треки"
}
},
"user": {
@@ -134,10 +149,12 @@
"currentPassword": "Поточний пароль",
"newPassword": "Новий пароль",
"token": "Токен",
- "lastAccessAt": "Останній доступ"
+ "lastAccessAt": "Останній доступ",
+ "libraries": "Бібліотеки"
},
"helperTexts": {
- "name": "Змінене ім'я буде відображатися при наступній авторизації"
+ "name": "Змінене ім'я буде відображатися при наступній авторизації",
+ "libraries": "Виберіть конкретні бібліотеки для цього користувача, або залиште поле порожнім, щоб використовувати бібліотеки за замовчуванням"
},
"notifications": {
"created": "Користувача створено",
@@ -146,7 +163,12 @@
},
"message": {
"listenBrainzToken": "Введіть свій токен користувача ListenBrainz.",
- "clickHereForToken": "Натисніть тут для отримання токену"
+ "clickHereForToken": "Натисніть тут для отримання токену",
+ "selectAllLibraries": "Вибрати всі бібліотеки",
+ "adminAutoLibraries": "Користувачі-адміністратори автоматично отримують доступ до всіх бібліотек"
+ },
+ "validation": {
+ "librariesRequired": "Для користувачів, які не є адміністраторами, має бути обрана хоча б одна бібліотека"
}
},
"player": {
@@ -190,11 +212,17 @@
"addNewPlaylist": "Створити \"%{name}\"",
"export": "Експортувати",
"makePublic": "Зробити публічним",
- "makePrivate": "Зробити приватним"
+ "makePrivate": "Зробити приватним",
+ "saveQueue": "Зберегти чергу до плейлиста",
+ "searchOrCreate": "Знайти плейлист або введіть текст, щоб створити новий...",
+ "pressEnterToCreate": "Натисніть Enter щоб створити новий плейлист",
+ "removeFromSelection": "Вилучити з вибору"
},
"message": {
"duplicate_song": "Додати повторювані пісні",
- "song_exist": "У список відтворення додаються дублікати. Хочете додати дублікати або пропустити їх?"
+ "song_exist": "У список відтворення додаються дублікати. Хочете додати дублікати або пропустити їх?",
+ "noPlaylistsFound": "Не знайдено плейлистів",
+ "noPlaylists": "Немає доступних плейлистів"
}
},
"radio": {
@@ -232,13 +260,68 @@
"fields": {
"path": "Шлях файлу",
"size": "Розмір",
- "updatedAt": "Зник"
+ "updatedAt": "Зник",
+ "libraryName": "Бібліотека"
},
"actions": {
- "remove": "Видалити"
+ "remove": "Видалити",
+ "remove_all": "Вилучити всі"
},
"notifications": {
"removed": "Видалено зниклі файл(и)"
+ },
+ "empty": "Немає відсутніх файлів"
+ },
+ "library": {
+ "name": "Бібліотека |||| Бібліотеки",
+ "fields": {
+ "name": "Ім'я",
+ "path": "Шлях",
+ "remotePath": "Віддалений шлях",
+ "lastScanAt": "Останнє сканування",
+ "songCount": "Треки",
+ "albumCount": "Альбоми",
+ "artistCount": "Виконавці",
+ "totalSongs": "Треки",
+ "totalAlbums": "Альбоми",
+ "totalArtists": "Виконавці",
+ "totalFolders": "Папки",
+ "totalFiles": "Файлів",
+ "totalMissingFiles": "Зниклих файлів",
+ "totalSize": "Загальний розмір",
+ "totalDuration": "Тривалість",
+ "defaultNewUsers": "За замовчуванням для нових користувачів",
+ "createdAt": "Створено",
+ "updatedAt": "Оновлено"
+ },
+ "sections": {
+ "basic": "Основна інформація",
+ "statistics": "Статистика"
+ },
+ "actions": {
+ "scan": "Сканувати бібліотеку",
+ "manageUsers": "Керування доступом користувачів",
+ "viewDetails": "Переглянути подробиці"
+ },
+ "notifications": {
+ "created": "Бібліотеку успішно створено",
+ "updated": "Бібліотеку успішно оновлено",
+ "deleted": "Бібліотеку успішно видалено",
+ "scanStarted": "Сканування бібліотеки розпочато",
+ "scanCompleted": "Сканування бібліотеки закінчено"
+ },
+ "validation": {
+ "nameRequired": "Ім'я бібліотеки обов'язкове",
+ "pathRequired": "Шлях до бібліотеки обов'язковий",
+ "pathNotDirectory": "Шлях до бібліотеки має бути директорією",
+ "pathNotFound": "Шлях до бібліотеки не знайдено",
+ "pathNotAccessible": "Шлях до бібліотеки недоступний",
+ "pathInvalid": "Помилковий шлях до бібліотеки"
+ },
+ "messages": {
+ "deleteConfirm": "Ви впевнені, що хочете видалити цю бібліотеку? Це призведе до видалення всіх пов'язаних з нею даних і доступу користувачів.",
+ "scanInProgress": "Сканування триває...",
+ "noLibrariesAssigned": "Немає бібліотек, призначених цьому користувачеві"
}
}
},
@@ -419,7 +502,11 @@
"downloadDialogTitle": "Завантаження %{resource} '%{name}' (%{size})",
"shareCopyToClipboard": "Скопіювати в буфер: Ctrl+C, Enter",
"remove_missing_title": "Видалити зниклі файли",
- "remove_missing_content": "Ви впевнені, що хочете видалити вибрані відсутні файли з бази даних? Це назавжди видалить усі посилання на них, включаючи кількість прослуховувань та рейтинги."
+ "remove_missing_content": "Ви впевнені, що хочете видалити вибрані відсутні файли з бази даних? Це назавжди видалить усі посилання на них, включаючи кількість прослуховувань та рейтинги.",
+ "remove_all_missing_title": "Видалити всі відсутні файли",
+ "remove_all_missing_content": "Ви впевнені, що хочете видалити всі відсутні файли з бази даних? Це назавжди видалить будь-які посилання на них, включно з кількістю відтворень та рейтингами.",
+ "noSimilarSongsFound": "Не знайдено схожих треків",
+ "noTopSongsFound": "Не знайдено ТОП-треків"
},
"menu": {
"library": "Бібліотека",
@@ -448,7 +535,13 @@
"albumList": "Альбом",
"about": "Довідка",
"playlists": "Списки відтворення",
- "sharedPlaylists": "Загальнодоступний список відтворення"
+ "sharedPlaylists": "Загальнодоступний список відтворення",
+ "librarySelector": {
+ "allLibraries": "Усі бібліотеки (%{count})",
+ "multipleLibraries": "%{selected} з %{total} Бібліотеки",
+ "selectLibraries": "Вибір бібліотек",
+ "none": "Відсутня"
+ }
},
"player": {
"playListsText": "Грати по черзі",
@@ -485,6 +578,21 @@
"disabled": "Вимкнено",
"waiting": "Очікування"
}
+ },
+ "tabs": {
+ "about": "Про",
+ "config": "Конфігурація"
+ },
+ "config": {
+ "configName": "Назва конфігурації",
+ "environmentVariable": "Змінна середовища",
+ "currentValue": "Поточне значення",
+ "configurationFile": "Файл конфігурації",
+ "exportToml": "Експортувати Конфігурацію (у форматі TOML)",
+ "exportSuccess": "Конфігурацію експортовано в буфер обміну у форматі TOML",
+ "exportFailed": "Не вдалося скопіювати конфігурацію",
+ "devFlagsHeader": "Прапорці розробки (можуть бути змінені/видалені)",
+ "devFlagsComment": "Це експериментальні налаштування, які можуть бути видалені в майбутніх версіях."
}
},
"activity": {
@@ -493,7 +601,10 @@
"quickScan": "Швидке сканування",
"fullScan": "Повне сканування",
"serverUptime": "Час роботи",
- "serverDown": "Оффлайн"
+ "serverDown": "Оффлайн",
+ "scanType": "Тип",
+ "status": "Помилка сканування",
+ "elapsedTime": "Пройдений час"
},
"help": {
"title": "Гарячі клавіші Navidrome",
@@ -508,5 +619,10 @@
"toggle_love": "Відмітити поточні пісні",
"current_song": "Перейти до поточної пісні"
}
+ },
+ "nowPlaying": {
+ "title": "Зараз грає",
+ "empty": "Нічого не грає",
+ "minutesAgo": "%{smart_count} хвилин тому |||| %{smart_count} хвилин тому"
}
}
\ No newline at end of file
diff --git a/resources/i18n/zh-Hans.json b/resources/i18n/zh-Hans.json
index c447f7d72..cde28c4f3 100644
--- a/resources/i18n/zh-Hans.json
+++ b/resources/i18n/zh-Hans.json
@@ -13,12 +13,14 @@
"album": "专辑",
"path": "文件路径",
"genre": "流派",
+ "libraryName": "媒体库",
"compilation": "合辑",
"year": "发行年份",
"size": "文件大小",
"updatedAt": "更新于",
"bitRate": "比特率",
"bitDepth": "比特深度",
+ "sampleRate": "采样率",
"channels": "声道",
"discSubtitle": "字幕",
"starred": "收藏",
@@ -33,12 +35,14 @@
"participants": "其他参与人员",
"tags": "附加标签",
"mappedTags": "映射标签",
- "rawTags": "原始标签"
+ "rawTags": "原始标签",
+ "missing": "缺失"
},
"actions": {
"addToQueue": "加入播放列表",
"playNow": "立即播放",
"addToPlaylist": "加入歌单",
+ "showInPlaylist": "定位到播放列表",
"shuffleAll": "全部随机播放",
"download": "下载",
"playNext": "下一首播放",
@@ -56,6 +60,7 @@
"size": "文件大小",
"name": "名称",
"genre": "流派",
+ "libraryName": "媒体库",
"compilation": "合辑",
"year": "发行年份",
"date": "录制日期",
@@ -72,7 +77,8 @@
"releaseType": "发行类型",
"grouping": "分组",
"media": "媒体类型",
- "mood": "情绪"
+ "mood": "情绪",
+ "missing": "缺失"
},
"actions": {
"playAll": "立即播放",
@@ -104,7 +110,8 @@
"playCount": "播放次数",
"rating": "评分",
"genre": "流派",
- "role": "参与角色"
+ "role": "参与角色",
+ "missing": "缺失"
},
"roles": {
"albumartist": "专辑歌手",
@@ -119,7 +126,13 @@
"mixer": "混音师",
"remixer": "重混师",
"djmixer": "DJ混音师",
- "performer": "演奏家"
+ "performer": "演奏家",
+ "maincredit": "主要艺术家"
+ },
+ "actions": {
+ "topSongs": "热门歌曲",
+ "shuffle": "随机播放",
+ "radio": "电台"
}
},
"user": {
@@ -136,19 +149,26 @@
"changePassword": "修改密码?",
"currentPassword": "当前密码",
"newPassword": "新密码",
- "token": "令牌"
+ "token": "令牌",
+ "libraries": "媒体库"
},
"helperTexts": {
- "name": "名称的更改将在下次登录时生效"
+ "name": "名称的更改将在下次登录时生效",
+ "libraries": "为该用户选择指定媒体库,留空则使用默认媒体库"
},
"notifications": {
"created": "用户已创建",
"updated": "用户已更新",
"deleted": "用户已删除"
},
+ "validation": {
+ "librariesRequired": "普通用户必须至少选择一个媒体库"
+ },
"message": {
"listenBrainzToken": "输入您的 ListenBrainz 用户令牌",
- "clickHereForToken": "点击这里来获得你的 ListenBrainz 令牌"
+ "clickHereForToken": "点击这里来获得你的 ListenBrainz 令牌",
+ "selectAllLibraries": "选择全部媒体库",
+ "adminAutoLibraries": "管理员默认可访问所有媒体库"
}
},
"player": {
@@ -191,12 +211,18 @@
"selectPlaylist": "选择歌单",
"addNewPlaylist": "新建 %{name}",
"export": "导出",
+ "saveQueue": "保存为歌单",
"makePublic": "设为公开",
- "makePrivate": "设为私有"
+ "makePrivate": "设为私有",
+ "searchOrCreate": "搜索歌单,或输入名称新建…",
+ "pressEnterToCreate": "按 Enter 键新建歌单",
+ "removeFromSelection": "移除选中项"
},
"message": {
"duplicate_song": "添加重复的歌曲",
- "song_exist": "部分选定的歌曲已存在歌单中,继续添加或是跳过它们?"
+ "song_exist": "部分选定的歌曲已存在歌单中,继续添加或是跳过它们?",
+ "noPlaylistsFound": "未找到歌单",
+ "noPlaylists": "暂无可用歌单"
}
},
"radio": {
@@ -237,14 +263,68 @@
"fields": {
"path": "路径",
"size": "文件大小",
+ "libraryName": "媒体库",
"updatedAt": "丢失于"
},
"actions": {
- "remove": "移除"
+ "remove": "移除",
+ "remove_all": "移除所有"
},
"notifications": {
"removed": "丢失文件已移除"
}
+ },
+ "library": {
+ "name": "媒体库",
+ "fields": {
+ "name": "名称",
+ "path": "路径",
+ "remotePath": "远程路径",
+ "lastScanAt": "上次扫描",
+ "songCount": "歌曲",
+ "albumCount": "专辑",
+ "artistCount": "艺术家",
+ "totalSongs": "歌曲",
+ "totalAlbums": "专辑",
+ "totalArtists": "艺术家",
+ "totalFolders": "目录",
+ "totalFiles": "文件",
+ "totalMissingFiles": "缺失的文件",
+ "totalSize": "总大小",
+ "totalDuration": "时长",
+ "defaultNewUsers": "新用户默认",
+ "createdAt": "创建于",
+ "updatedAt": "更新于"
+ },
+ "sections": {
+ "basic": "基本信息",
+ "statistics": "统计数据"
+ },
+ "actions": {
+ "scan": "扫描媒体库",
+ "manageUsers": "管理用户权限",
+ "viewDetails": "查看详情"
+ },
+ "notifications": {
+ "created": "媒体库已创建",
+ "updated": "媒体库已更新",
+ "deleted": "媒体库已删除",
+ "scanStarted": "开始扫描媒体库",
+ "scanCompleted": "媒体库扫描已完成"
+ },
+ "validation": {
+ "nameRequired": "媒体库名称不能为空!",
+ "pathRequired": "媒体库路径不能为空!",
+ "pathNotDirectory": "媒体库路径必须为目录!",
+ "pathNotFound": "媒体库路径不存在!",
+ "pathNotAccessible": "媒体库路径无法访问!",
+ "pathInvalid": "媒体库路径无效!"
+ },
+ "messages": {
+ "deleteConfirm": "您确定要删除此媒体库吗?此操作将删除所有关联数据及用户访问权限!",
+ "scanInProgress": "正在扫描...",
+ "noLibrariesAssigned": "该用户未分配任何媒体库!"
+ }
}
},
"ra": {
@@ -397,11 +477,15 @@
"transcodingDisabled": "出于安全原因,从 Web 界面更改转码配置的功能已被禁用。要更改(编辑或新增)转码选项,请在启用 %{config} 选项的情况下重新启动服务器。",
"transcodingEnabled": "Navidrome 当前与 %{config} 一起使用,可以通过配置转码选项来执行任意命令,建议仅在配置转码选项时启用此功能。",
"songsAddedToPlaylist": "已添加 %{smart_count} 首歌到歌单",
+ "noSimilarSongsFound": "未找到相似歌曲",
+ "noTopSongsFound": "未找到热门歌曲",
"noPlaylistsAvailable": "没有有效的歌单",
"delete_user_title": "删除用户 %{name}",
"delete_user_content": "您确定要删除该用户及其相关数据(包括歌单和用户配置)吗?",
"remove_missing_title": "移除丢失文件",
"remove_missing_content": "您确定要将选中的丢失文件从数据库中永久移除吗?此操作将删除所有相关信息,包括播放次数和评分。",
+ "remove_all_missing_title": "删除所有丢失文件",
+ "remove_all_missing_content": "您确定要从数据库中删除所有丢失文件吗?这将永久删除对它们的所有引用,包括它们的播放次数和评分。",
"notifications_blocked": "您已在浏览器的设置中屏蔽了此网站的通知",
"notifications_not_available": "此浏览器不支持桌面通知",
"lastfmLinkSuccess": "Last.fm 已关联并启用喜好记录",
@@ -428,6 +512,12 @@
},
"menu": {
"library": "曲库",
+ "librarySelector": {
+ "allLibraries": "全部媒体库 (%{count})",
+ "multipleLibraries": "已选 %{selected} 共 %{total} 媒体库",
+ "selectLibraries": "选择媒体库",
+ "none": "无"
+ },
"settings": "设置",
"version": "版本",
"theme": "主题",
@@ -490,6 +580,21 @@
"disabled": "禁用",
"waiting": "等待"
}
+ },
+ "tabs": {
+ "about": "关于",
+ "config": "配置"
+ },
+ "config": {
+ "configName": "配置名称",
+ "environmentVariable": "环境变量",
+ "currentValue": "当前值",
+ "configurationFile": "配置文件",
+ "exportToml": "导出配置(TOML)",
+ "exportSuccess": "配置以 TOML 格式导出到剪贴板",
+ "exportFailed": "复制配置失败",
+ "devFlagsHeader": "开发标志(可能会更改/删除)",
+ "devFlagsComment": "这些是实验性设置,可能会在未来版本中删除"
}
},
"activity": {
@@ -498,7 +603,15 @@
"quickScan": "快速扫描",
"fullScan": "完全扫描",
"serverUptime": "服务器已运行",
- "serverDown": "服务器已离线"
+ "serverDown": "服务器已离线",
+ "scanType": "扫描类型",
+ "status": "扫描状态",
+ "elapsedTime": "用时"
+ },
+ "nowPlaying": {
+ "title": "正在播放",
+ "empty": "无播放内容",
+ "minutesAgo": "%{smart_count} 分钟前"
},
"help": {
"title": "Navidrome 快捷键",
@@ -514,4 +627,4 @@
"toggle_love": "添加/移除星标"
}
}
-}
\ No newline at end of file
+}
diff --git a/resources/i18n/zh-Hant.json b/resources/i18n/zh-Hant.json
index 3d6bbd268..7d8ce2872 100644
--- a/resources/i18n/zh-Hant.json
+++ b/resources/i18n/zh-Hant.json
@@ -1,463 +1,630 @@
{
- "languageName": "繁體中文",
- "resources": {
- "song": {
- "name": "歌曲 |||| 歌曲",
- "fields": {
- "albumArtist": "專輯藝人",
- "duration": "長度",
- "trackNumber": "#",
- "playCount": "播放次數",
- "title": "標題",
- "artist": "藝人",
- "album": "專輯",
- "path": "文件路徑",
- "genre": "類型",
- "compilation": "合輯",
- "year": "發行年份",
- "size": "檔案大小",
- "updatedAt": "更新於",
- "bitRate": "位元率",
- "discSubtitle": "字幕",
- "starred": "收藏",
- "comment": "註解",
- "rating": "評分",
- "quality": "品質",
- "bpm": "BPM",
- "playDate": "上次播放",
- "channels": "聲道",
- "createdAt": "創建於"
- },
- "actions": {
- "addToQueue": "加入至播放佇列",
- "playNow": "立即播放",
- "addToPlaylist": "加入至播放清單",
- "shuffleAll": "全部隨機播放",
- "download": "下載",
- "playNext": "下一首播放",
- "info": "取得資訊"
- }
- },
- "album": {
- "name": "專輯 |||| 專輯",
- "fields": {
- "albumArtist": "專輯藝人",
- "artist": "藝人",
- "duration": "長度",
- "songCount": "歌曲數量",
- "playCount": "播放次數",
- "name": "名稱",
- "genre": "類型",
- "compilation": "合輯",
- "year": "發行年份",
- "updatedAt": "更新於",
- "comment": "註解",
- "rating": "評分",
- "createdAt": "創建於",
- "size": "檔案大小",
- "originalDate": "原始日期",
- "releaseDate": "發行日期",
- "releases": "發行",
- "released": "已發行"
- },
- "actions": {
- "playAll": "立即播放",
- "playNext": "下首播放",
- "addToQueue": "加入至播放佇列",
- "shuffle": "隨機播放",
- "addToPlaylist": "加入播放清單",
- "download": "下載",
- "info": "取得資訊",
- "share": "分享"
- },
- "lists": {
- "all": "所有",
- "random": "隨機",
- "recentlyAdded": "最近加入",
- "recentlyPlayed": "最近播放",
- "mostPlayed": "最多播放的",
- "starred": "收藏",
- "topRated": "最高評分"
- }
- },
- "artist": {
- "name": "藝人 |||| 藝人",
- "fields": {
- "name": "名稱",
- "albumCount": "專輯數",
- "songCount": "歌曲數",
- "playCount": "播放次數",
- "rating": "評分",
- "genre": "類型",
- "size": "檔案大小"
- }
- },
- "user": {
- "name": "使用者 |||| 使用者",
- "fields": {
- "userName": "使用者名稱",
- "isAdmin": "是否管理員",
- "lastLoginAt": "上次登入",
- "lastAccessAt": "上此訪問",
- "updatedAt": "更新於",
- "name": "名稱",
- "password": "密碼",
- "createdAt": "創建於",
- "changePassword": "變更密碼?",
- "currentPassword": "現在的密碼",
- "newPassword": "新密碼",
- "token": "權杖"
- },
- "helperTexts": {
- "name": "你的名稱會在下次登入時生效"
- },
- "notifications": {
- "created": "使用者已創建",
- "updated": "使用者已更新",
- "deleted": "使用者已刪除"
- },
- "message": {
- "listenBrainzToken": "輸入您的 ListenBrainz 使用者權杖",
- "clickHereForToken": "點擊此處來獲得你的 ListenBrainz 權杖"
- }
- },
- "player": {
- "name": "用戶端 |||| 用戶端",
- "fields": {
- "name": "名稱",
- "transcodingId": "轉碼",
- "maxBitRate": "最大位元率",
- "client": "用戶端",
- "userName": "使用者名稱",
- "lastSeen": "上次瀏覽",
- "reportRealPath": "回報實際路徑",
- "scrobbleEnabled": "傳送音樂記錄至外部服務"
- }
- },
- "transcoding": {
- "name": "轉碼 |||| 轉碼",
- "fields": {
- "name": "名稱",
- "targetFormat": "目標格式",
- "defaultBitRate": "預設位元率",
- "command": "命令"
- }
- },
- "playlist": {
- "name": "播放清單 |||| 播放清單",
- "fields": {
- "name": "名稱",
- "duration": "長度",
- "ownerName": "擁有者",
- "public": "公開",
- "updatedAt": "更新於",
- "createdAt": "創建於",
- "songCount": "歌曲數",
- "comment": "註解",
- "sync": "自動導入",
- "path": "導入"
- },
- "actions": {
- "selectPlaylist": "選擇播放清單",
- "addNewPlaylist": "創建 %{name}",
- "export": "導出",
- "makePublic": "設為公開",
- "makePrivate": "設為私人"
- },
- "message": {
- "duplicate_song": "加入重複的歌曲",
- "song_exist": "有重複歌曲正在播放清單裡,您要加入或略過重複歌曲?"
- }
- },
- "radio": {
- "name": "電台",
- "fields": {
- "name": "名稱",
- "streamUrl": "串流網址",
- "homePageUrl": "首頁網址",
- "updatedAt": "更新於",
- "createdAt": "創建於"
- },
- "actions": {
- "playNow": "立即播放"
- }
- },
- "share": {
- "name": "分享",
- "fields": {
- "username": "使用者名稱",
- "url": "網址",
- "description": "描述",
- "contents": "內容",
- "expiresAt": "過期時間",
- "lastVisitedAt": "上次訪問時間",
- "visitCount": "訪問次數",
- "format": "格式",
- "maxBitRate": "最大位元率",
- "updatedAt": "更新於",
- "createdAt": "創建於",
- "downloadable": "可下載"
- },
- "notifications": {},
- "actions": {}
- }
+ "languageName": "繁體中文",
+ "resources": {
+ "song": {
+ "name": "歌曲 |||| 歌曲",
+ "fields": {
+ "albumArtist": "專輯藝人",
+ "duration": "長度",
+ "trackNumber": "#",
+ "playCount": "播放次數",
+ "title": "標題",
+ "artist": "藝人",
+ "album": "專輯",
+ "path": "檔案路徑",
+ "libraryName": "媒體庫",
+ "genre": "曲風",
+ "compilation": "合輯",
+ "year": "發行年份",
+ "size": "檔案大小",
+ "updatedAt": "更新於",
+ "bitRate": "位元率",
+ "bitDepth": "位元深度",
+ "sampleRate": "取樣率",
+ "channels": "聲道",
+ "discSubtitle": "光碟副標題",
+ "starred": "收藏",
+ "comment": "註解",
+ "rating": "評分",
+ "quality": "品質",
+ "bpm": "BPM",
+ "playDate": "上次播放",
+ "createdAt": "建立於",
+ "grouping": "分組",
+ "mood": "情緒",
+ "participants": "其他參與人員",
+ "tags": "額外標籤",
+ "mappedTags": "分類後標籤",
+ "rawTags": "原始標籤",
+ "missing": "遺失"
+ },
+ "actions": {
+ "addToQueue": "加入至播放佇列",
+ "playNow": "立即播放",
+ "addToPlaylist": "加入至播放清單",
+ "showInPlaylist": "在播放清單中顯示",
+ "shuffleAll": "全部隨機播放",
+ "download": "下載",
+ "playNext": "下一首播放",
+ "info": "取得資訊"
+ }
},
- "ra": {
- "auth": {
- "welcome1": "感謝您安裝 Navidrome!",
- "welcome2": "開始前,請創建一個管理員帳戶",
- "confirmPassword": "確認密碼",
- "buttonCreateAdmin": "創建管理員",
- "auth_check_error": "請登入以訪問更多內容",
- "user_menu": "配置",
- "username": "使用者名稱",
- "password": "密碼",
- "sign_in": "登入",
- "sign_in_error": "驗證失敗,請重試",
- "logout": "登出"
- },
- "validation": {
- "invalidChars": "請使用字母和數字",
- "passwordDoesNotMatch": "密碼不相符",
- "required": "必填",
- "minLength": "必須不少於 %{min} 個字元",
- "maxLength": "必須不多於 %{max} 個字元",
- "minValue": "必須不小於 %{min}",
- "maxValue": "必須不大於 %{max}",
- "number": "必須為數字",
- "email": "必須是有效的電子郵件",
- "oneOf": "必須為: %{options}其中一項",
- "regex": "必須符合指定的格式(正規表達式):%{pattern}",
- "unique": "必須是唯一的",
- "url": "網址"
- },
- "action": {
- "add_filter": "加入篩選",
- "add": "加入",
- "back": "返回",
- "bulk_actions": "選中 %{smart_count} 項",
- "cancel": "取消",
- "clear_input_value": "清除",
- "clone": "複製",
- "confirm": "確認",
- "create": "創建",
- "delete": "刪除",
- "edit": "編輯",
- "export": "匯出",
- "list": "列表",
- "refresh": "重新整理",
- "remove_filter": "清除此條件",
- "remove": "清除",
- "save": "保存",
- "search": "搜尋",
- "show": "顯示",
- "sort": "排序",
- "undo": "撤銷",
- "expand": "展開",
- "close": "關閉",
- "open_menu": "打開選單",
- "close_menu": "關閉選單",
- "unselect": "未選擇",
- "skip": "略過",
- "bulk_actions_mobile": "%{smart_count}",
- "share": "分享",
- "download": "下載"
- },
- "boolean": {
- "true": "是",
- "false": "否"
- },
- "page": {
- "create": "創建 %{name}",
- "dashboard": "儀表板",
- "edit": "%{name} #%{id}",
- "error": "發生錯誤",
- "list": "%{name}",
- "loading": "載入中",
- "not_found": "未發現",
- "show": "%{name} #%{id}",
- "empty": "還沒有 %{name}。",
- "invite": "你要創建一個嗎?"
- },
- "input": {
- "file": {
- "upload_several": "拖拽多個文件上傳或點擊選擇一個",
- "upload_single": "拖拽單個文件上傳或點擊選擇一個"
- },
- "image": {
- "upload_several": "拖拽多個圖片上傳或點擊選擇一個",
- "upload_single": "拖拽單個圖片上傳或點擊選擇一個"
- },
- "references": {
- "all_missing": "未找到參考數據",
- "many_missing": "至少有一條參考數據不再可用",
- "single_missing": "關聯的參考數據不再可用"
- },
- "password": {
- "toggle_visible": "隱藏密碼",
- "toggle_hidden": "顯示密碼"
- }
- },
- "message": {
- "about": "關於",
- "are_you_sure": "確定進行此操作?",
- "bulk_delete_content": "您確定要刪除 %{name}? |||| 您確定要刪除 %{smart_count} 項?",
- "bulk_delete_title": "刪除 %{name} |||| 刪除 %{smart_count} 項 %{name}",
- "delete_content": "您確定要刪除該項目?",
- "delete_title": "刪除 %{name} #%{id}",
- "details": "詳細資訊",
- "error": "發生一個用戶端錯誤,您的請求無法完成",
- "invalid_form": "提交內容無效,請檢查錯誤",
- "loading": "正在載入頁面,請稍候",
- "no": "否",
- "not_found": "您輸入的連結格式不對或連結遺失",
- "yes": "是",
- "unsaved_changes": "某些更改尚未保存,您確定要離開此頁面嗎?"
- },
- "navigation": {
- "no_results": "無內容",
- "no_more_results": "頁碼 %{page} 超出邊界,嘗試返回上一頁",
- "page_out_of_boundaries": "頁碼 %{page} 超出邊界",
- "page_out_from_end": "已經最後一頁",
- "page_out_from_begin": "已經是第一頁",
- "page_range_info": "%{offsetBegin}-%{offsetEnd} / %{total}",
- "page_rows_per_page": "每頁行數:",
- "next": "下一頁",
- "prev": "上一頁",
- "skip_nav": "跳過"
- },
- "notification": {
- "updated": "項已更新 |||| %{smart_count} 項已更新",
- "created": "項已創建",
- "deleted": "項已刪除 |||| %{smart_count} 項已刪除",
- "bad_item": "不確定的項",
- "item_doesnt_exist": "項不存在",
- "http_error": "伺服器通訊錯誤",
- "data_provider_error": "資料來源錯誤,請檢查控制台的詳細資訊",
- "i18n_error": "無法載入所選語言",
- "canceled": "操作已取消",
- "logged_out": "您的會話已結束,請重新登入",
- "new_version": "發現新版本!請重新整理視窗"
- },
- "toggleFieldsMenu": {
- "columnsToDisplay": "顯示欄目",
- "layout": "版面",
- "grid": "框格",
- "table": "表格"
- }
+ "album": {
+ "name": "專輯 |||| 專輯",
+ "fields": {
+ "albumArtist": "專輯藝人",
+ "artist": "藝人",
+ "duration": "長度",
+ "songCount": "歌曲數",
+ "playCount": "播放次數",
+ "size": "檔案大小",
+ "name": "名稱",
+ "libraryName": "媒體庫",
+ "genre": "曲風",
+ "compilation": "合輯",
+ "year": "發行年份",
+ "date": "錄製日期",
+ "originalDate": "原始日期",
+ "releaseDate": "發行日期",
+ "releases": "發行",
+ "released": "已發行",
+ "updatedAt": "更新於",
+ "comment": "註解",
+ "rating": "評分",
+ "createdAt": "建立於",
+ "recordLabel": "唱片公司",
+ "catalogNum": "目錄編號",
+ "releaseType": "發行類型",
+ "grouping": "分組",
+ "media": "媒體類型",
+ "mood": "情緒",
+ "missing": "遺失"
+ },
+ "actions": {
+ "playAll": "播放全部",
+ "playNext": "下一首播放",
+ "addToQueue": "加入至播放佇列",
+ "share": "分享",
+ "shuffle": "隨機播放",
+ "addToPlaylist": "加入至播放清單",
+ "download": "下載",
+ "info": "取得資訊"
+ },
+ "lists": {
+ "all": "所有",
+ "random": "隨機",
+ "recentlyAdded": "最近加入",
+ "recentlyPlayed": "最近播放",
+ "mostPlayed": "最常播放",
+ "starred": "收藏",
+ "topRated": "最高評分"
+ }
},
- "message": {
- "note": "註解",
- "transcodingDisabled": "出於安全原因,禁用了從 Web 介面更改參數。要更改(編輯或新增)轉檔選項,請在啟用 %{config} 選項的情況下重新啟動伺服器。",
- "transcodingEnabled": "Navidrome 當前與 %{config} 一起使用,可以通過配置轉檔參數執行任意命令,建議僅在配置轉檔選項時啟用此功能。",
- "songsAddedToPlaylist": "已加入一首歌到播放清單 |||| 已添加 %{smart_count} 首歌到播放清單",
- "noPlaylistsAvailable": "沒有可用的播放清單",
- "delete_user_title": "刪除使用者 %{name}",
- "delete_user_content": "您確定要刪除該使用者及其相關數據(包括播放清單和使用者配置)嗎?",
- "notifications_blocked": "您已在瀏覽器的設置中封鎖了此網站的通知",
- "notifications_not_available": "此瀏覽器不支援桌面通知",
- "lastfmLinkSuccess": "Last.fm 成功連接並開啟音樂記錄",
- "lastfmLinkFailure": "Last.fm 無法連接",
- "lastfmUnlinkSuccess": "Last.fm 已無連接並停用音樂記錄",
- "lastfmUnlinkFailure": "Last.fm 無法取消連接",
- "openIn": {
- "lastfm": "在 Last.fm 打開",
- "musicbrainz": "在 MusicBrainz 打開"
- },
- "lastfmLink": "繼續閱讀…",
- "listenBrainzLinkSuccess": "ListenBrainz 成功連接並開啟音樂記錄",
- "listenBrainzLinkFailure": "ListenBrainz 無法連接:%{error}",
- "listenBrainzUnlinkSuccess": "ListenBrainz 已無連接並停用音樂記錄",
- "listenBrainzUnlinkFailure": "ListenBrainz 無法取消連接",
- "downloadOriginalFormat": "下載原始格式",
- "shareOriginalFormat": "分享原始格式",
- "shareDialogTitle": "分享",
- "shareBatchDialogTitle": "批次分享",
- "shareSuccess": "分享成功",
- "shareFailure": "分享失敗",
- "downloadDialogTitle": "下載",
- "shareCopyToClipboard": "複製到剪貼簿"
+ "artist": {
+ "name": "藝人 |||| 藝人",
+ "fields": {
+ "name": "名稱",
+ "albumCount": "專輯數",
+ "songCount": "歌曲數",
+ "size": "檔案大小",
+ "playCount": "播放次數",
+ "rating": "評分",
+ "genre": "曲風",
+ "role": "參與角色",
+ "missing": "遺失"
+ },
+ "roles": {
+ "albumartist": "專輯藝人 |||| 專輯藝人",
+ "artist": "藝人 |||| 藝人",
+ "composer": "作曲 |||| 作曲",
+ "conductor": "指揮 |||| 指揮",
+ "lyricist": "作詞 |||| 作詞",
+ "arranger": "編曲 |||| 編曲",
+ "producer": "製作人 |||| 製作人",
+ "director": "導演 |||| 導演",
+ "engineer": "工程師 |||| 工程師",
+ "mixer": "混音師 |||| 混音師",
+ "remixer": "重混師 |||| 重混師",
+ "djmixer": "DJ 混音師 |||| DJ 混音師",
+ "performer": "表演者 |||| 表演者",
+ "maincredit": "專輯藝人或藝人 |||| 專輯藝人或藝人"
+ },
+ "actions": {
+ "topSongs": "熱門歌曲",
+ "shuffle": "隨機播放",
+ "radio": "電台"
+ }
},
- "menu": {
- "library": "音樂庫",
- "settings": "設定",
- "version": "版本",
- "theme": "主題",
- "personal": {
- "name": "個人化",
- "options": {
- "theme": "主題",
- "language": "語言",
- "defaultView": "預設畫面",
- "desktop_notifications": "桌面通知",
- "lastfmScrobbling": "啟用 Last.fm 音樂記錄",
- "listenBrainzScrobbling": "啟用 ListenBrainz 音樂記錄",
- "replaygain": "重播增益",
- "preAmp": "前置放大器 (dB)",
- "gain": {
- "none": "無",
- "album": "專輯增益",
- "track": "曲目增益"
- }
- }
- },
- "albumList": "專輯",
- "about": "關於",
- "playlists": "播放清單",
- "sharedPlaylists": "分享的播放清單"
+ "user": {
+ "name": "使用者 |||| 使用者",
+ "fields": {
+ "userName": "使用者名稱",
+ "isAdmin": "管理員",
+ "lastLoginAt": "上次登入",
+ "lastAccessAt": "上次存取",
+ "updatedAt": "更新於",
+ "name": "名稱",
+ "password": "密碼",
+ "createdAt": "建立於",
+ "changePassword": "變更密碼?",
+ "currentPassword": "目前密碼",
+ "newPassword": "新密碼",
+ "token": "權杖",
+ "libraries": "媒體庫"
+ },
+ "helperTexts": {
+ "name": "您的名稱會在下次登入時生效",
+ "libraries": "為該使用者選擇指定媒體庫,留空則使用預設媒體庫"
+ },
+ "notifications": {
+ "created": "使用者已建立",
+ "updated": "使用者已更新",
+ "deleted": "使用者已刪除"
+ },
+ "validation": {
+ "librariesRequired": "非管理員使用者必須至少選擇一個媒體庫"
+ },
+ "message": {
+ "listenBrainzToken": "輸入您的 ListenBrainz 使用者權杖",
+ "clickHereForToken": "點擊此處來獲得您的 ListenBrainz 權杖",
+ "selectAllLibraries": "選取全部媒體庫",
+ "adminAutoLibraries": "管理員預設可存取所有媒體庫"
+ }
},
"player": {
- "playListsText": "播放佇列",
- "openText": "打開",
- "closeText": "關閉",
- "notContentText": "沒有音樂",
- "clickToPlayText": "點擊播放",
- "clickToPauseText": "點擊暫停",
- "nextTrackText": "下一首",
- "previousTrackText": "上一首",
- "reloadText": "重新播放",
- "volumeText": "音量",
- "toggleLyricText": "切換歌詞",
- "toggleMiniModeText": "最小化",
- "destroyText": "關閉",
- "downloadText": "下載",
- "removeAudioListsText": "清空播放佇列",
- "clickToDeleteText": "點擊刪除 %{name}",
- "emptyLyricText": "無歌詞",
- "playModeText": {
- "order": "順序播放",
- "orderLoop": "列表循環",
- "singleLoop": "單曲循環",
- "shufflePlay": "隨機播放"
- }
+ "name": "播放器 |||| 播放器",
+ "fields": {
+ "name": "名稱",
+ "transcodingId": "轉碼",
+ "maxBitRate": "最大位元率",
+ "client": "客戶端",
+ "userName": "使用者名稱",
+ "lastSeen": "上次上線",
+ "reportRealPath": "回報實際路徑",
+ "scrobbleEnabled": "傳送音樂記錄至外部服務"
+ }
},
- "about": {
- "links": {
- "homepage": "主頁",
- "source": "原始碼",
- "featureRequests": "功能請求"
- }
+ "transcoding": {
+ "name": "轉碼 |||| 轉碼",
+ "fields": {
+ "name": "名稱",
+ "targetFormat": "目標格式",
+ "defaultBitRate": "預設位元率",
+ "command": "指令"
+ }
},
- "activity": {
- "title": "運作狀況",
- "totalScanned": "已完成掃描的目錄",
- "quickScan": "快速掃描",
- "fullScan": "完全掃描",
- "serverUptime": "伺服器已運作時間",
- "serverDown": "伺服器離線"
+ "playlist": {
+ "name": "播放清單 |||| 播放清單",
+ "fields": {
+ "name": "名稱",
+ "duration": "長度",
+ "ownerName": "擁有者",
+ "public": "公開",
+ "updatedAt": "更新於",
+ "createdAt": "建立於",
+ "songCount": "歌曲數",
+ "comment": "註解",
+ "sync": "自動匯入",
+ "path": "匯入來源"
+ },
+ "actions": {
+ "selectPlaylist": "選取播放清單:",
+ "addNewPlaylist": "建立「%{name}」",
+ "export": "匯出",
+ "saveQueue": "將播放佇列儲存到播放清單",
+ "makePublic": "設為公開",
+ "makePrivate": "設為私人",
+ "searchOrCreate": "搜尋播放清單,或輸入名稱來新建…",
+ "pressEnterToCreate": "按 Enter 鍵建立新的播放清單",
+ "removeFromSelection": "移除選取項目"
+ },
+ "message": {
+ "duplicate_song": "加入重複的歌曲",
+ "song_exist": "有重複歌曲正要加入播放清單,您要加入或略過重複歌曲?",
+ "noPlaylistsFound": "找不到播放清單",
+ "noPlaylists": "暫無播放清單"
+ }
},
- "help": {
- "title": "Navidrome 快捷鍵",
- "hotkeys": {
- "show_help": "顯示此幫助",
- "toggle_menu": "顯示/隱藏選單側欄",
- "toggle_play": "播放/暫停",
- "prev_song": "上一首歌",
- "next_song": "下一首歌",
- "vol_up": "提高音量",
- "vol_down": "降低音量",
- "toggle_love": "添加或移除星標",
- "current_song": "目前歌曲"
- }
+ "radio": {
+ "name": "電台 |||| 電台",
+ "fields": {
+ "name": "名稱",
+ "streamUrl": "串流網址",
+ "homePageUrl": "首頁網址",
+ "updatedAt": "更新於",
+ "createdAt": "建立於"
+ },
+ "actions": {
+ "playNow": "立即播放"
+ }
+ },
+ "share": {
+ "name": "分享 |||| 分享",
+ "fields": {
+ "username": "分享者",
+ "url": "網址",
+ "description": "描述",
+ "downloadable": "允許下載?",
+ "contents": "內容",
+ "expiresAt": "過期時間",
+ "lastVisitedAt": "上次造訪時間",
+ "visitCount": "造訪次數",
+ "format": "格式",
+ "maxBitRate": "最大位元率",
+ "updatedAt": "更新於",
+ "createdAt": "建立於"
+ },
+ "notifications": {},
+ "actions": {}
+ },
+ "missing": {
+ "name": "遺失檔案 |||| 遺失檔案",
+ "empty": "無遺失檔案",
+ "fields": {
+ "path": "路徑",
+ "size": "檔案大小",
+ "libraryName": "媒體庫",
+ "updatedAt": "遺失於"
+ },
+ "actions": {
+ "remove": "刪除",
+ "remove_all": "刪除所有"
+ },
+ "notifications": {
+ "removed": "遺失檔案已刪除"
+ }
+ },
+ "library": {
+ "name": "媒體庫 |||| 媒體庫",
+ "fields": {
+ "name": "名稱",
+ "path": "路徑",
+ "remotePath": "遠端路徑",
+ "lastScanAt": "上次掃描",
+ "songCount": "歌曲",
+ "albumCount": "專輯",
+ "artistCount": "藝人",
+ "totalSongs": "歌曲",
+ "totalAlbums": "專輯",
+ "totalArtists": "藝人",
+ "totalFolders": "資料夾",
+ "totalFiles": "檔案",
+ "totalMissingFiles": "遺失檔案",
+ "totalSize": "總大小",
+ "totalDuration": "時長",
+ "defaultNewUsers": "新使用者預設媒體庫",
+ "createdAt": "建立於",
+ "updatedAt": "更新於"
+ },
+ "sections": {
+ "basic": "基本資訊",
+ "statistics": "統計"
+ },
+ "actions": {
+ "scan": "掃描媒體庫",
+ "manageUsers": "管理使用者權限",
+ "viewDetails": "查看詳細資料"
+ },
+ "notifications": {
+ "created": "成功建立媒體庫",
+ "updated": "成功更新媒體庫",
+ "deleted": "成功刪除媒體庫",
+ "scanStarted": "開始掃描媒體庫",
+ "scanCompleted": "媒體庫掃描完成"
+ },
+ "validation": {
+ "nameRequired": "請輸入媒體庫名稱",
+ "pathRequired": "請提供媒體庫路徑",
+ "pathNotDirectory": "媒體庫路徑必須為目錄",
+ "pathNotFound": "媒體庫路徑不存在",
+ "pathNotAccessible": "無法存取媒體庫路徑",
+ "pathInvalid": "媒體庫路徑無效"
+ },
+ "messages": {
+ "deleteConfirm": "您確定要刪除此媒體庫嗎?這將刪除所有相關資料和使用者存取權限。",
+ "scanInProgress": "正在掃描...",
+ "noLibrariesAssigned": "沒有為該使用者指派任何媒體庫"
+ }
}
+ },
+ "ra": {
+ "auth": {
+ "welcome1": "感謝您安裝 Navidrome!",
+ "welcome2": "開始前,請先建立一個管理員帳號",
+ "confirmPassword": "確認密碼",
+ "buttonCreateAdmin": "建立管理員",
+ "auth_check_error": "請登入以繼續",
+ "user_menu": "個人檔案",
+ "username": "使用者名稱",
+ "password": "密碼",
+ "sign_in": "登入",
+ "sign_in_error": "驗證失敗,請重試",
+ "logout": "登出",
+ "insightsCollectionNote": "Navidrome 會收集匿名使用資料以協助改善項目。\n點擊[此處]了解更多資訊或選擇退出。"
+ },
+ "validation": {
+ "invalidChars": "請使用字母和數字",
+ "passwordDoesNotMatch": "密碼不相符",
+ "required": "必填",
+ "minLength": "必須不少於 %{min} 個字元",
+ "maxLength": "必須不多於 %{max} 個字元",
+ "minValue": "必須不小於 %{min}",
+ "maxValue": "必須不大於 %{max}",
+ "number": "必須為數字",
+ "email": "必須為有效的電子郵件",
+ "oneOf": "必須為以下其中一項:%{options}",
+ "regex": "必須符合指定的格式(正規表達式):%{pattern}",
+ "unique": "必須是唯一的",
+ "url": "必須為有效的網址"
+ },
+ "action": {
+ "add_filter": "加入篩選",
+ "add": "加入",
+ "back": "返回",
+ "bulk_actions": "選中 1 項 |||| 選中 %{smart_count} 項",
+ "bulk_actions_mobile": "1 |||| %{smart_count}",
+ "cancel": "取消",
+ "clear_input_value": "清除",
+ "clone": "複製",
+ "confirm": "確認",
+ "create": "建立",
+ "delete": "刪除",
+ "edit": "編輯",
+ "export": "匯出",
+ "list": "列表",
+ "refresh": "重新整理",
+ "remove_filter": "清除此條件",
+ "remove": "移除",
+ "save": "儲存",
+ "search": "搜尋",
+ "show": "顯示",
+ "sort": "排序",
+ "undo": "復原",
+ "expand": "展開",
+ "close": "關閉",
+ "open_menu": "開啟選單",
+ "close_menu": "關閉選單",
+ "unselect": "取消選取",
+ "skip": "略過",
+ "share": "分享",
+ "download": "下載"
+ },
+ "boolean": {
+ "true": "是",
+ "false": "否"
+ },
+ "page": {
+ "create": "建立 %{name}",
+ "dashboard": "儀表板",
+ "edit": "%{name} #%{id}",
+ "error": "發生錯誤",
+ "list": "%{name}",
+ "loading": "載入中",
+ "not_found": "找不到",
+ "show": "%{name} #%{id}",
+ "empty": "還沒有 %{name}。",
+ "invite": "您要建立一個嗎?"
+ },
+ "input": {
+ "file": {
+ "upload_several": "拖曳多個檔案上傳或點擊選擇一個",
+ "upload_single": "拖曳單個檔案上傳或點擊選擇一個"
+ },
+ "image": {
+ "upload_several": "拖曳多個圖片上傳或點擊選擇一個",
+ "upload_single": "拖曳單個圖片上傳或點擊選擇一個"
+ },
+ "references": {
+ "all_missing": "未找到參考數據",
+ "many_missing": "至少有一條參考數據不再可用",
+ "single_missing": "關聯的參考數據不再可用"
+ },
+ "password": {
+ "toggle_visible": "隱藏密碼",
+ "toggle_hidden": "顯示密碼"
+ }
+ },
+ "message": {
+ "about": "關於",
+ "are_you_sure": "您確定嗎?",
+ "bulk_delete_content": "您確定要刪除 %{name}? |||| 您確定要刪除這 %{smart_count} 個項目嗎?",
+ "bulk_delete_title": "刪除 %{name} |||| 刪除 %{smart_count} 項 %{name}",
+ "delete_content": "您確定要刪除該項目?",
+ "delete_title": "刪除 %{name} #%{id}",
+ "details": "詳細資訊",
+ "error": "發生客戶端錯誤,您的請求無法完成",
+ "invalid_form": "提交內容無效,請檢查錯誤",
+ "loading": "正在載入頁面,請稍候",
+ "no": "否",
+ "not_found": "您輸入了錯誤的連結或連結遺失",
+ "yes": "是",
+ "unsaved_changes": "某些更改尚未儲存,您確定要離開此頁面嗎?"
+ },
+ "navigation": {
+ "no_results": "沒有找到結果",
+ "no_more_results": "頁碼 %{page} 超出邊界,嘗試返回上一頁",
+ "page_out_of_boundaries": "頁碼 %{page} 超出邊界",
+ "page_out_from_end": "已經是最後一頁",
+ "page_out_from_begin": "已經是第一頁",
+ "page_range_info": "%{offsetBegin}-%{offsetEnd} / %{total}",
+ "page_rows_per_page": "每頁項目數:",
+ "next": "下一頁",
+ "prev": "上一頁",
+ "skip_nav": "跳至內容"
+ },
+ "notification": {
+ "updated": "項目已更新 |||| %{smart_count} 項已更新",
+ "created": "項目已建立",
+ "deleted": "項目已刪除 |||| %{smart_count} 項已刪除",
+ "bad_item": "項目不正確",
+ "item_doesnt_exist": "項目不存在",
+ "http_error": "伺服器通訊錯誤",
+ "data_provider_error": "資料來源錯誤,請檢查控制台的詳細資訊",
+ "i18n_error": "無法載入所選語言",
+ "canceled": "操作已取消",
+ "logged_out": "您的工作階段已結束,請重新登入",
+ "new_version": "發現新版本!請重新整理視窗"
+ },
+ "toggleFieldsMenu": {
+ "columnsToDisplay": "顯示欄位",
+ "layout": "版面",
+ "grid": "網格",
+ "table": "表格"
+ }
+ },
+ "message": {
+ "note": "注意",
+ "transcodingDisabled": "出於安全原因,已停用了從 Web 介面更改參數。要更改(編輯或新增)轉碼選項,請在啟用 %{config} 選項的情況下重新啟動伺服器。",
+ "transcodingEnabled": "Navidrome 目前與 %{config} 一起使用,因此可以透過 Web 介面從轉碼設定中執行系統命令。出於安全考慮,我們建議停用此功能,並僅在設定轉碼選項時啟用。",
+ "songsAddedToPlaylist": "已加入一首歌到播放清單 |||| 已新增 %{smart_count} 首歌到播放清單",
+ "noSimilarSongsFound": "找不到相似歌曲",
+ "noTopSongsFound": "找不到熱門歌曲",
+ "noPlaylistsAvailable": "沒有可用的播放清單",
+ "delete_user_title": "刪除使用者「%{name}」",
+ "delete_user_content": "您確定要刪除此使用者及其所有資料(包括播放清單和偏好設定)嗎?",
+ "remove_missing_title": "刪除遺失檔案",
+ "remove_missing_content": "您確定要從媒體庫中刪除所選的遺失的檔案嗎?這將永久刪除它們的所有相關資訊,包括其播放次數和評分。",
+ "remove_all_missing_title": "刪除所有遺失檔案",
+ "remove_all_missing_content": "您確定要從媒體庫中刪除所有遺失的檔案嗎?這將永久刪除它們的所有相關資訊,包括它們的播放次數和評分。",
+ "notifications_blocked": "您已在瀏覽器設定中封鎖了此網站的通知",
+ "notifications_not_available": "此瀏覽器不支援桌面通知,或您並非透過 HTTPS 存取 Navidrome",
+ "lastfmLinkSuccess": "已成功連接 Last.fm 並開啟音樂記錄",
+ "lastfmLinkFailure": "無法連接 Last.fm",
+ "lastfmUnlinkSuccess": "已取消 Last.fm 的連接並停用音樂記錄",
+ "lastfmUnlinkFailure": "無法取消 Last.fm 的連接",
+ "listenBrainzLinkSuccess": "已成功以 %{user} 身份連接 ListenBrainz 並開啟音樂記錄",
+ "listenBrainzLinkFailure": "無法連接 ListenBrainz:%{error}",
+ "listenBrainzUnlinkSuccess": "已取消 ListenBrainz 的連接並停用音樂記錄",
+ "listenBrainzUnlinkFailure": "無法取消 ListenBrainz 的連接",
+ "openIn": {
+ "lastfm": "在 Last.fm 中開啟",
+ "musicbrainz": "在 MusicBrainz 中開啟"
+ },
+ "lastfmLink": "查看更多…",
+ "shareOriginalFormat": "分享原始格式",
+ "shareDialogTitle": "分享 %{resource} '%{name}'",
+ "shareBatchDialogTitle": "分享 1 個%{resource} |||| 分享 %{smart_count} 個%{resource}",
+ "shareCopyToClipboard": "複製到剪貼簿:Ctrl+C, Enter",
+ "shareSuccess": "分享成功,連結已複製到剪貼簿:%{url}",
+ "shareFailure": "分享連結複製失敗:%{url}",
+ "downloadDialogTitle": "下載 %{resource} '%{name}' (%{size})",
+ "downloadOriginalFormat": "下載原始格式"
+ },
+ "menu": {
+ "library": "媒體庫",
+ "librarySelector": {
+ "allLibraries": "所有媒體庫 (%{count})",
+ "multipleLibraries": "已選 %{selected} 共 %{total} 媒體庫",
+ "selectLibraries": "選取媒體庫",
+ "none": "無"
+ },
+ "settings": "設定",
+ "version": "版本",
+ "theme": "主題",
+ "personal": {
+ "name": "個人化",
+ "options": {
+ "theme": "主題",
+ "language": "語言",
+ "defaultView": "預設畫面",
+ "desktop_notifications": "桌面通知",
+ "lastfmNotConfigured": "Last.fm API 金鑰未設定",
+ "lastfmScrobbling": "啟用 Last.fm 音樂記錄",
+ "listenBrainzScrobbling": "啟用 ListenBrainz 音樂記錄",
+ "replaygain": "重播增益模式",
+ "preAmp": "重播增益前置放大器 (dB)",
+ "gain": {
+ "none": "無",
+ "album": "專輯增益",
+ "track": "曲目增益"
+ }
+ }
+ },
+ "albumList": "專輯",
+ "playlists": "播放清單",
+ "sharedPlaylists": "分享的播放清單",
+ "about": "關於"
+ },
+ "player": {
+ "playListsText": "播放佇列",
+ "openText": "開啟",
+ "closeText": "關閉",
+ "notContentText": "沒有音樂",
+ "clickToPlayText": "點擊播放",
+ "clickToPauseText": "點擊暫停",
+ "nextTrackText": "下一首",
+ "previousTrackText": "上一首",
+ "reloadText": "重新載入",
+ "volumeText": "音量",
+ "toggleLyricText": "切換歌詞",
+ "toggleMiniModeText": "最小化",
+ "destroyText": "關閉",
+ "downloadText": "下載",
+ "removeAudioListsText": "清空播放佇列",
+ "clickToDeleteText": "點擊刪除 %{name}",
+ "emptyLyricText": "無歌詞",
+ "playModeText": {
+ "order": "順序播放",
+ "orderLoop": "循環播放",
+ "singleLoop": "單曲循環",
+ "shufflePlay": "隨機播放"
+ }
+ },
+ "about": {
+ "links": {
+ "homepage": "首頁",
+ "source": "原始碼",
+ "featureRequests": "功能請求",
+ "lastInsightsCollection": "最近一次洞察資料收集",
+ "insights": {
+ "disabled": "已停用",
+ "waiting": "等待中"
+ }
+ },
+ "tabs": {
+ "about": "關於",
+ "config": "設定"
+ },
+ "config": {
+ "configName": "設定名稱",
+ "environmentVariable": "環境變數",
+ "currentValue": "目前值",
+ "configurationFile": "設定檔案",
+ "exportToml": "匯出設定(TOML 格式)",
+ "exportSuccess": "設定已以 TOML 格式匯出至剪貼簿",
+ "exportFailed": "設定複製失敗",
+ "devFlagsHeader": "開發旗標(可能會更改/刪除)",
+ "devFlagsComment": "這些是實驗性設定,可能會在未來版本中刪除"
+ }
+ },
+ "activity": {
+ "title": "運作狀況",
+ "totalScanned": "已掃描的資料夾總數",
+ "quickScan": "快速掃描",
+ "fullScan": "完全掃描",
+ "serverUptime": "伺服器運作時間",
+ "serverDown": "伺服器已離線",
+ "scanType": "掃描類型",
+ "status": "掃描錯誤",
+ "elapsedTime": "經過時間"
+ },
+ "nowPlaying": {
+ "title": "正在播放",
+ "empty": "無播放內容",
+ "minutesAgo": "1 分鐘前 |||| %{smart_count} 分鐘前"
+ },
+ "help": {
+ "title": "Navidrome 快捷鍵",
+ "hotkeys": {
+ "show_help": "顯示此說明",
+ "toggle_menu": "顯示/隱藏選單側欄",
+ "toggle_play": "播放/暫停",
+ "prev_song": "上一首歌",
+ "next_song": "下一首歌",
+ "current_song": "前往目前歌曲",
+ "vol_up": "提高音量",
+ "vol_down": "降低音量",
+ "toggle_love": "新增此歌曲至收藏"
+ }
+ }
}
diff --git a/resources/mappings.yaml b/resources/mappings.yaml
index 650665c78..d1da5c620 100644
--- a/resources/mappings.yaml
+++ b/resources/mappings.yaml
@@ -69,7 +69,7 @@ main:
remixer:
aliases: [ tpe4, remixer, mixartist, ----:com.apple.itunes:remixer, wm/modifiedby ]
albumartist:
- aliases: [ tpe2, albumartist, album artist, aart, wm/albumartist ]
+ aliases: [ tpe2, albumartist, album artist, album_artist, aart, wm/albumartist ]
albumartistsort:
aliases: [ tso2, txxx:albumartistsort, albumartistsort, soaa, wm/albumartistsortorder ]
albumartists:
@@ -108,7 +108,8 @@ main:
bpm:
aliases: [ tbpm, bpm, tmpo, wm/beatsperminute ]
lyrics:
- aliases: [ uslt:description, lyrics, ©lyr, wm/lyrics, unsyncedlyrics ]
+ # Note, @lyr and wm/lyrics have been removed. Taglib somehow appears to always populate `lyrics:xxx`
+ aliases: [ uslt:description, lyrics, unsyncedlyrics ]
maxLength: 32768
type: pair # ex: lyrics:eng, lyrics:xxx
comment:
diff --git a/scanner/controller.go b/scanner/controller.go
index 0b3e5d122..b42246a50 100644
--- a/scanner/controller.go
+++ b/scanner/controller.go
@@ -7,7 +7,6 @@ import (
"sync/atomic"
"time"
- "github.com/Masterminds/squirrel"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/consts"
"github.com/navidrome/navidrome/core"
@@ -27,24 +26,8 @@ var (
ErrAlreadyScanning = errors.New("already scanning")
)
-type Scanner interface {
- // ScanAll starts a full scan of the music library. This is a blocking operation.
- ScanAll(ctx context.Context, fullScan bool) (warnings []string, err error)
- Status(context.Context) (*StatusInfo, error)
-}
-
-type StatusInfo struct {
- Scanning bool
- LastScan time.Time
- Count uint32
- FolderCount uint32
- LastError string
- ScanType string
- ElapsedTime time.Duration
-}
-
func New(rootCtx context.Context, ds model.DataStore, cw artwork.CacheWarmer, broker events.Broker,
- pls core.Playlists, m metrics.Metrics) Scanner {
+ pls core.Playlists, m metrics.Metrics) model.Scanner {
c := &controller{
rootCtx: rootCtx,
ds: ds,
@@ -63,13 +46,13 @@ func (s *controller) getScanner() scanner {
if conf.Server.DevExternalScanner {
return &scannerExternal{}
}
- return &scannerImpl{ds: s.ds, cw: s.cw, pls: s.pls, metrics: s.metrics}
+ return &scannerImpl{ds: s.ds, cw: s.cw, pls: s.pls}
}
-// CallScan starts an in-process scan of the music library.
+// CallScan starts an in-process scan of specific library/folder pairs.
+// If targets is empty, it scans all libraries.
// This is meant to be called from the command line (see cmd/scan.go).
-func CallScan(ctx context.Context, ds model.DataStore, cw artwork.CacheWarmer, pls core.Playlists,
- metrics metrics.Metrics, fullScan bool) (<-chan *ProgressInfo, error) {
+func CallScan(ctx context.Context, ds model.DataStore, pls core.Playlists, fullScan bool, targets []model.ScanTarget) (<-chan *ProgressInfo, error) {
release, err := lockScan(ctx)
if err != nil {
return nil, err
@@ -80,8 +63,8 @@ func CallScan(ctx context.Context, ds model.DataStore, cw artwork.CacheWarmer, p
progress := make(chan *ProgressInfo, 100)
go func() {
defer close(progress)
- scanner := &scannerImpl{ds: ds, cw: cw, pls: pls, metrics: metrics}
- scanner.scanAll(ctx, fullScan, progress)
+ scanner := &scannerImpl{ds: ds, cw: artwork.NoopCacheWarmer(), pls: pls}
+ scanner.scanFolders(ctx, fullScan, targets, progress)
}()
return progress, nil
}
@@ -101,8 +84,11 @@ type ProgressInfo struct {
ForceUpdate bool
}
+// scanner defines the interface for different scanner implementations.
+// This allows for swapping between in-process and external scanners.
type scanner interface {
- scanAll(ctx context.Context, fullScan bool, progress chan<- *ProgressInfo)
+ // scanFolders performs the actual scanning of folders. If targets is nil, it scans all libraries.
+ scanFolders(ctx context.Context, fullScan bool, targets []model.ScanTarget, progress chan<- *ProgressInfo)
}
type controller struct {
@@ -118,6 +104,24 @@ type controller struct {
changesDetected bool
}
+// getLastScanTime returns the most recent scan time across all libraries
+func (s *controller) getLastScanTime(ctx context.Context) (time.Time, error) {
+ libs, err := s.ds.Library(ctx).GetAll(model.QueryOptions{
+ Sort: "last_scan_at",
+ Order: "desc",
+ Max: 1,
+ })
+ if err != nil {
+ return time.Time{}, fmt.Errorf("getting libraries: %w", err)
+ }
+
+ if len(libs) == 0 {
+ return time.Time{}, nil
+ }
+
+ return libs[0].LastScanAt, nil
+}
+
// getScanInfo retrieves scan status from the database
func (s *controller) getScanInfo(ctx context.Context) (scanType string, elapsed time.Duration, lastErr string) {
lastErr, _ = s.ds.Property(ctx).DefaultGet(consts.LastScanErrorKey, "")
@@ -130,10 +134,10 @@ func (s *controller) getScanInfo(ctx context.Context) (scanType string, elapsed
if running.Load() {
elapsed = time.Since(startTime)
} else {
- // If scan is not running, try to get the last scan time for the library
- lib, err := s.ds.Library(ctx).Get(1) //TODO Multi-library
- if err == nil {
- elapsed = lib.LastScanAt.Sub(startTime)
+ // If scan is not running, calculate elapsed time using the most recent scan time
+ lastScanTime, err := s.getLastScanTime(ctx)
+ if err == nil && !lastScanTime.IsZero() {
+ elapsed = lastScanTime.Sub(startTime)
}
}
}
@@ -142,18 +146,18 @@ func (s *controller) getScanInfo(ctx context.Context) (scanType string, elapsed
return scanType, elapsed, lastErr
}
-func (s *controller) Status(ctx context.Context) (*StatusInfo, error) {
- lib, err := s.ds.Library(ctx).Get(1) //TODO Multi-library
+func (s *controller) Status(ctx context.Context) (*model.ScannerStatus, error) {
+ lastScanTime, err := s.getLastScanTime(ctx)
if err != nil {
- return nil, fmt.Errorf("getting library: %w", err)
+ return nil, fmt.Errorf("getting last scan time: %w", err)
}
scanType, elapsed, lastErr := s.getScanInfo(ctx)
if running.Load() {
- status := &StatusInfo{
+ status := &model.ScannerStatus{
Scanning: true,
- LastScan: lib.LastScanAt,
+ LastScan: lastScanTime,
Count: s.count.Load(),
FolderCount: s.folderCount.Load(),
LastError: lastErr,
@@ -167,9 +171,9 @@ func (s *controller) Status(ctx context.Context) (*StatusInfo, error) {
if err != nil {
return nil, fmt.Errorf("getting library stats: %w", err)
}
- return &StatusInfo{
+ return &model.ScannerStatus{
Scanning: false,
- LastScan: lib.LastScanAt,
+ LastScan: lastScanTime,
Count: uint32(count),
FolderCount: uint32(folderCount),
LastError: lastErr,
@@ -179,25 +183,23 @@ func (s *controller) Status(ctx context.Context) (*StatusInfo, error) {
}
func (s *controller) getCounters(ctx context.Context) (int64, int64, error) {
- count, err := s.ds.MediaFile(ctx).CountAll()
+ libs, err := s.ds.Library(ctx).GetAll()
if err != nil {
- return 0, 0, fmt.Errorf("media file count: %w", err)
+ return 0, 0, fmt.Errorf("library count: %w", err)
}
- folderCount, err := s.ds.Folder(ctx).CountAll(
- model.QueryOptions{
- Filters: squirrel.And{
- squirrel.Gt{"num_audio_files": 0},
- squirrel.Eq{"missing": false},
- },
- },
- )
- if err != nil {
- return 0, 0, fmt.Errorf("folder count: %w", err)
+ var count, folderCount int64
+ for _, l := range libs {
+ count += int64(l.TotalSongs)
+ folderCount += int64(l.TotalFolders)
}
return count, folderCount, nil
}
func (s *controller) ScanAll(requestCtx context.Context, fullScan bool) ([]string, error) {
+ return s.ScanFolders(requestCtx, fullScan, nil)
+}
+
+func (s *controller) ScanFolders(requestCtx context.Context, fullScan bool, targets []model.ScanTarget) ([]string, error) {
release, err := lockScan(requestCtx)
if err != nil {
return nil, err
@@ -206,7 +208,6 @@ func (s *controller) ScanAll(requestCtx context.Context, fullScan bool) ([]strin
// Prepare the context for the scan
ctx := request.AddValues(s.rootCtx, requestCtx)
- ctx = events.BroadcastToAll(ctx)
ctx = auth.WithAdminUser(ctx, s.ds)
// Send the initial scan status event
@@ -215,7 +216,7 @@ func (s *controller) ScanAll(requestCtx context.Context, fullScan bool) ([]strin
go func() {
defer close(progress)
scanner := s.getScanner()
- scanner.scanAll(ctx, fullScan, progress)
+ scanner.scanFolders(ctx, fullScan, targets, progress)
}()
// Wait for the scan to finish, sending progress events to all connected clients
@@ -226,13 +227,15 @@ func (s *controller) ScanAll(requestCtx context.Context, fullScan bool) ([]strin
// If changes were detected, send a refresh event to all clients
if s.changesDetected {
log.Debug(ctx, "Library changes imported. Sending refresh event")
- s.broker.SendMessage(ctx, &events.RefreshResource{})
+ s.broker.SendBroadcastMessage(ctx, &events.RefreshResource{})
}
// Send the final scan status event, with totals
if count, folderCount, err := s.getCounters(ctx); err != nil {
+ s.metrics.WriteAfterScanMetrics(ctx, false)
return scanWarnings, err
} else {
scanType, elapsed, lastErr := s.getScanInfo(ctx)
+ s.metrics.WriteAfterScanMetrics(ctx, true)
s.sendMessage(ctx, &events.ScanStatus{
Scanning: false,
Count: count,
@@ -303,5 +306,5 @@ func (s *controller) trackProgress(ctx context.Context, progress <-chan *Progres
}
func (s *controller) sendMessage(ctx context.Context, status *events.ScanStatus) {
- s.broker.SendMessage(ctx, status)
+ s.broker.SendBroadcastMessage(ctx, status)
}
diff --git a/scanner/controller_test.go b/scanner/controller_test.go
index 4f6576a39..f5ccabc86 100644
--- a/scanner/controller_test.go
+++ b/scanner/controller_test.go
@@ -21,7 +21,7 @@ import (
var _ = Describe("Controller", func() {
var ctx context.Context
var ds *tests.MockDataStore
- var ctrl scanner.Scanner
+ var ctrl model.Scanner
Describe("Status", func() {
BeforeEach(func() {
@@ -32,7 +32,6 @@ var _ = Describe("Controller", func() {
ds = &tests.MockDataStore{RealDS: persistence.New(db.Db())}
ds.MockedProperty = &tests.MockedPropertyRepo{}
ctrl = scanner.New(ctx, ds, artwork.NoopCacheWarmer(), events.NoopBroker(), core.NewPlaylists(ds), metrics.NewNoopInstance())
- Expect(ds.Library(ctx).Put(&model.Library{ID: 1, Name: "lib", Path: "/tmp"})).To(Succeed())
})
It("includes last scan error", func() {
diff --git a/scanner/external.go b/scanner/external.go
index c4a29efa3..b6d7639be 100644
--- a/scanner/external.go
+++ b/scanner/external.go
@@ -8,10 +8,12 @@ import (
"io"
"os"
"os/exec"
+ "strings"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/log"
- . "github.com/navidrome/navidrome/utils/gg"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/utils/slice"
)
// scannerExternal is a scanner that runs an external process to do the scanning. It is used to avoid
@@ -23,19 +25,41 @@ import (
// process will forward them to the caller.
type scannerExternal struct{}
-func (s *scannerExternal) scanAll(ctx context.Context, fullScan bool, progress chan<- *ProgressInfo) {
+func (s *scannerExternal) scanFolders(ctx context.Context, fullScan bool, targets []model.ScanTarget, progress chan<- *ProgressInfo) {
+ s.scan(ctx, fullScan, targets, progress)
+}
+
+func (s *scannerExternal) scan(ctx context.Context, fullScan bool, targets []model.ScanTarget, progress chan<- *ProgressInfo) {
exe, err := os.Executable()
if err != nil {
progress <- &ProgressInfo{Error: fmt.Sprintf("failed to get executable path: %s", err)}
return
}
- log.Debug(ctx, "Spawning external scanner process", "fullScan", fullScan, "path", exe)
- cmd := exec.CommandContext(ctx, exe, "scan",
+
+ // Build command arguments
+ args := []string{
+ "scan",
"--nobanner", "--subprocess",
"--configfile", conf.Server.ConfigFile,
"--datafolder", conf.Server.DataFolder,
"--cachefolder", conf.Server.CacheFolder,
- If(fullScan, "--full", ""))
+ }
+
+ // Add targets if provided
+ if len(targets) > 0 {
+ targetsStr := strings.Join(slice.Map(targets, func(t model.ScanTarget) string { return t.String() }), ",")
+ args = append(args, "--targets", targetsStr)
+ log.Debug(ctx, "Spawning external scanner process with targets", "fullScan", fullScan, "path", exe, "targets", targetsStr)
+ } else {
+ log.Debug(ctx, "Spawning external scanner process", "fullScan", fullScan, "path", exe)
+ }
+
+ // Add full scan flag if needed
+ if fullScan {
+ args = append(args, "--full")
+ }
+
+ cmd := exec.CommandContext(ctx, exe, args...)
in, out := io.Pipe()
defer in.Close()
diff --git a/scanner/folder_entry.go b/scanner/folder_entry.go
new file mode 100644
index 000000000..9d8d0c571
--- /dev/null
+++ b/scanner/folder_entry.go
@@ -0,0 +1,118 @@
+package scanner
+
+import (
+ "crypto/md5"
+ "encoding/hex"
+ "fmt"
+ "io"
+ "io/fs"
+ "maps"
+ "slices"
+ "time"
+
+ "github.com/navidrome/navidrome/core"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/utils/chrono"
+)
+
+func newFolderEntry(job *scanJob, id, path string, updTime time.Time, hash string) *folderEntry {
+ f := &folderEntry{
+ id: id,
+ job: job,
+ path: path,
+ audioFiles: make(map[string]fs.DirEntry),
+ imageFiles: make(map[string]fs.DirEntry),
+ albumIDMap: make(map[string]string),
+ updTime: updTime,
+ prevHash: hash,
+ }
+ return f
+}
+
+type folderEntry struct {
+ job *scanJob
+ elapsed chrono.Meter
+ path string // Full path
+ id string // DB ID
+ modTime time.Time // From FS
+ updTime time.Time // from DB
+ audioFiles map[string]fs.DirEntry
+ imageFiles map[string]fs.DirEntry
+ numPlaylists int
+ numSubFolders int
+ imagesUpdatedAt time.Time
+ prevHash string // Previous hash from DB
+ tracks model.MediaFiles
+ albums model.Albums
+ albumIDMap map[string]string
+ artists model.Artists
+ tags model.TagList
+ missingTracks []*model.MediaFile
+}
+
+func (f *folderEntry) hasNoFiles() bool {
+ return len(f.audioFiles) == 0 && len(f.imageFiles) == 0 && f.numPlaylists == 0
+}
+
+func (f *folderEntry) isEmpty() bool {
+ return f.hasNoFiles() && f.numSubFolders == 0
+}
+
+func (f *folderEntry) isNew() bool {
+ return f.updTime.IsZero()
+}
+
+func (f *folderEntry) isOutdated() bool {
+ if f.job.lib.FullScanInProgress && f.updTime.Before(f.job.lib.LastScanStartedAt) {
+ return true
+ }
+ return f.prevHash != f.hash()
+}
+
+func (f *folderEntry) toFolder() *model.Folder {
+ folder := model.NewFolder(f.job.lib, f.path)
+ folder.NumAudioFiles = len(f.audioFiles)
+ if core.InPlaylistsPath(*folder) {
+ folder.NumPlaylists = f.numPlaylists
+ }
+ folder.ImageFiles = slices.Collect(maps.Keys(f.imageFiles))
+ folder.ImagesUpdatedAt = f.imagesUpdatedAt
+ folder.Hash = f.hash()
+ return folder
+}
+
+func (f *folderEntry) hash() string {
+ h := md5.New()
+ _, _ = fmt.Fprintf(
+ h,
+ "%s:%d:%d:%s",
+ f.modTime.UTC(),
+ f.numPlaylists,
+ f.numSubFolders,
+ f.imagesUpdatedAt.UTC(),
+ )
+
+ // Sort the keys of audio and image files to ensure consistent hashing
+ audioKeys := slices.Collect(maps.Keys(f.audioFiles))
+ slices.Sort(audioKeys)
+ imageKeys := slices.Collect(maps.Keys(f.imageFiles))
+ slices.Sort(imageKeys)
+
+ // Include audio files with their size and modtime
+ for _, key := range audioKeys {
+ _, _ = io.WriteString(h, key)
+ if info, err := f.audioFiles[key].Info(); err == nil {
+ _, _ = fmt.Fprintf(h, ":%d:%s", info.Size(), info.ModTime().UTC().String())
+ }
+ }
+
+ // Include image files with their size and modtime
+ for _, key := range imageKeys {
+ _, _ = io.WriteString(h, key)
+ if info, err := f.imageFiles[key].Info(); err == nil {
+ _, _ = fmt.Fprintf(h, ":%d:%s", info.Size(), info.ModTime().UTC().String())
+ }
+ }
+
+ return hex.EncodeToString(h.Sum(nil))
+}
diff --git a/scanner/folder_entry_test.go b/scanner/folder_entry_test.go
new file mode 100644
index 000000000..0328c6653
--- /dev/null
+++ b/scanner/folder_entry_test.go
@@ -0,0 +1,543 @@
+package scanner
+
+import (
+ "io/fs"
+ "time"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/model"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("folder_entry", func() {
+ var (
+ lib model.Library
+ job *scanJob
+ path string
+ )
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ lib = model.Library{
+ ID: 500,
+ Path: "/music",
+ LastScanStartedAt: time.Now().Add(-1 * time.Hour),
+ FullScanInProgress: false,
+ }
+ job = &scanJob{
+ lib: lib,
+ lastUpdates: make(map[string]model.FolderUpdateInfo),
+ }
+ path = "test/folder"
+ })
+
+ Describe("newFolderEntry", func() {
+ It("creates a new folder entry with correct initialization", func() {
+ folderID := model.FolderID(lib, path)
+ updateInfo := model.FolderUpdateInfo{
+ UpdatedAt: time.Now().Add(-30 * time.Minute),
+ Hash: "previous-hash",
+ }
+
+ entry := newFolderEntry(job, folderID, path, updateInfo.UpdatedAt, updateInfo.Hash)
+
+ Expect(entry.id).To(Equal(folderID))
+ Expect(entry.job).To(Equal(job))
+ Expect(entry.path).To(Equal(path))
+ Expect(entry.audioFiles).To(BeEmpty())
+ Expect(entry.imageFiles).To(BeEmpty())
+ Expect(entry.albumIDMap).To(BeEmpty())
+ Expect(entry.updTime).To(Equal(updateInfo.UpdatedAt))
+ Expect(entry.prevHash).To(Equal(updateInfo.Hash))
+ })
+ })
+
+ Describe("createFolderEntry", func() {
+ It("removes the lastUpdate from the job after creation", func() {
+ folderID := model.FolderID(lib, path)
+ updateInfo := model.FolderUpdateInfo{
+ UpdatedAt: time.Now().Add(-30 * time.Minute),
+ Hash: "previous-hash",
+ }
+ job.lastUpdates[folderID] = updateInfo
+
+ entry := job.createFolderEntry(path)
+
+ Expect(entry.updTime).To(Equal(updateInfo.UpdatedAt))
+ Expect(entry.prevHash).To(Equal(updateInfo.Hash))
+ Expect(job.lastUpdates).ToNot(HaveKey(folderID))
+ })
+ })
+
+ Describe("folderEntry", func() {
+ var entry *folderEntry
+
+ BeforeEach(func() {
+ folderID := model.FolderID(lib, path)
+ entry = newFolderEntry(job, folderID, path, time.Time{}, "")
+ })
+
+ Describe("hasNoFiles", func() {
+ It("returns true when folder has no files or subfolders", func() {
+ Expect(entry.hasNoFiles()).To(BeTrue())
+ })
+
+ It("returns false when folder has audio files", func() {
+ entry.audioFiles["test.mp3"] = &fakeDirEntry{name: "test.mp3"}
+ Expect(entry.hasNoFiles()).To(BeFalse())
+ })
+
+ It("returns false when folder has image files", func() {
+ entry.imageFiles["cover.jpg"] = &fakeDirEntry{name: "cover.jpg"}
+ Expect(entry.hasNoFiles()).To(BeFalse())
+ })
+
+ It("returns false when folder has playlists", func() {
+ entry.numPlaylists = 1
+ Expect(entry.hasNoFiles()).To(BeFalse())
+ })
+
+ It("ignores subfolders when checking for no files", func() {
+ entry.numSubFolders = 1
+ Expect(entry.hasNoFiles()).To(BeTrue())
+ })
+
+ It("returns false when folder has multiple types of content", func() {
+ entry.audioFiles["test.mp3"] = &fakeDirEntry{name: "test.mp3"}
+ entry.imageFiles["cover.jpg"] = &fakeDirEntry{name: "cover.jpg"}
+ entry.numPlaylists = 2
+ entry.numSubFolders = 3
+ Expect(entry.hasNoFiles()).To(BeFalse())
+ })
+ })
+
+ Describe("isEmpty", func() {
+ It("returns true when folder has no files or subfolders", func() {
+ Expect(entry.isEmpty()).To(BeTrue())
+ })
+ It("returns false when folder has audio files", func() {
+ entry.audioFiles["test.mp3"] = &fakeDirEntry{name: "test.mp3"}
+ Expect(entry.isEmpty()).To(BeFalse())
+ })
+ It("returns false when folder has subfolders", func() {
+ entry.numSubFolders = 1
+ Expect(entry.isEmpty()).To(BeFalse())
+ })
+ })
+
+ Describe("isNew", func() {
+ It("returns true when updTime is zero", func() {
+ entry.updTime = time.Time{}
+ Expect(entry.isNew()).To(BeTrue())
+ })
+
+ It("returns false when updTime is not zero", func() {
+ entry.updTime = time.Now()
+ Expect(entry.isNew()).To(BeFalse())
+ })
+ })
+
+ Describe("toFolder", func() {
+ BeforeEach(func() {
+ entry.audioFiles = map[string]fs.DirEntry{
+ "song1.mp3": &fakeDirEntry{name: "song1.mp3"},
+ "song2.mp3": &fakeDirEntry{name: "song2.mp3"},
+ }
+ entry.imageFiles = map[string]fs.DirEntry{
+ "cover.jpg": &fakeDirEntry{name: "cover.jpg"},
+ "folder.png": &fakeDirEntry{name: "folder.png"},
+ }
+ entry.numPlaylists = 3
+ entry.imagesUpdatedAt = time.Now()
+ })
+
+ It("converts folder entry to model.Folder correctly", func() {
+ folder := entry.toFolder()
+
+ Expect(folder.LibraryID).To(Equal(lib.ID))
+ Expect(folder.ID).To(Equal(entry.id))
+ Expect(folder.NumAudioFiles).To(Equal(2))
+ Expect(folder.ImageFiles).To(ConsistOf("cover.jpg", "folder.png"))
+ Expect(folder.ImagesUpdatedAt).To(Equal(entry.imagesUpdatedAt))
+ Expect(folder.Hash).To(Equal(entry.hash()))
+ })
+
+ It("sets NumPlaylists when folder is in playlists path", func() {
+ // Mock InPlaylistsPath to return true by setting empty PlaylistsPath
+ originalPath := conf.Server.PlaylistsPath
+ conf.Server.PlaylistsPath = ""
+ DeferCleanup(func() { conf.Server.PlaylistsPath = originalPath })
+
+ folder := entry.toFolder()
+ Expect(folder.NumPlaylists).To(Equal(3))
+ })
+
+ It("does not set NumPlaylists when folder is not in playlists path", func() {
+ // Mock InPlaylistsPath to return false by setting a different path
+ originalPath := conf.Server.PlaylistsPath
+ conf.Server.PlaylistsPath = "different/path"
+ DeferCleanup(func() { conf.Server.PlaylistsPath = originalPath })
+
+ folder := entry.toFolder()
+ Expect(folder.NumPlaylists).To(BeZero())
+ })
+ })
+
+ Describe("hash", func() {
+ BeforeEach(func() {
+ entry.modTime = time.Date(2023, 1, 15, 12, 0, 0, 0, time.UTC)
+ entry.imagesUpdatedAt = time.Date(2023, 1, 16, 14, 30, 0, 0, time.UTC)
+ })
+
+ It("produces deterministic hash for same content", func() {
+ entry.audioFiles = map[string]fs.DirEntry{
+ "b.mp3": &fakeDirEntry{name: "b.mp3"},
+ "a.mp3": &fakeDirEntry{name: "a.mp3"},
+ }
+ entry.imageFiles = map[string]fs.DirEntry{
+ "z.jpg": &fakeDirEntry{name: "z.jpg"},
+ "x.png": &fakeDirEntry{name: "x.png"},
+ }
+ entry.numPlaylists = 2
+ entry.numSubFolders = 3
+
+ hash1 := entry.hash()
+
+ // Reverse order of maps
+ entry.audioFiles = map[string]fs.DirEntry{
+ "a.mp3": &fakeDirEntry{name: "a.mp3"},
+ "b.mp3": &fakeDirEntry{name: "b.mp3"},
+ }
+ entry.imageFiles = map[string]fs.DirEntry{
+ "x.png": &fakeDirEntry{name: "x.png"},
+ "z.jpg": &fakeDirEntry{name: "z.jpg"},
+ }
+
+ hash2 := entry.hash()
+ Expect(hash1).To(Equal(hash2))
+ })
+
+ It("produces different hash when audio files change", func() {
+ entry.audioFiles = map[string]fs.DirEntry{
+ "song1.mp3": &fakeDirEntry{name: "song1.mp3"},
+ }
+ hash1 := entry.hash()
+
+ entry.audioFiles["song2.mp3"] = &fakeDirEntry{name: "song2.mp3"}
+ hash2 := entry.hash()
+
+ Expect(hash1).ToNot(Equal(hash2))
+ })
+
+ It("produces different hash when image files change", func() {
+ entry.imageFiles = map[string]fs.DirEntry{
+ "cover.jpg": &fakeDirEntry{name: "cover.jpg"},
+ }
+ hash1 := entry.hash()
+
+ entry.imageFiles["folder.png"] = &fakeDirEntry{name: "folder.png"}
+ hash2 := entry.hash()
+
+ Expect(hash1).ToNot(Equal(hash2))
+ })
+
+ It("produces different hash when modification time changes", func() {
+ hash1 := entry.hash()
+
+ entry.modTime = entry.modTime.Add(1 * time.Hour)
+ hash2 := entry.hash()
+
+ Expect(hash1).ToNot(Equal(hash2))
+ })
+
+ It("produces different hash when playlist count changes", func() {
+ hash1 := entry.hash()
+
+ entry.numPlaylists = 5
+ hash2 := entry.hash()
+
+ Expect(hash1).ToNot(Equal(hash2))
+ })
+
+ It("produces different hash when subfolder count changes", func() {
+ hash1 := entry.hash()
+
+ entry.numSubFolders = 3
+ hash2 := entry.hash()
+
+ Expect(hash1).ToNot(Equal(hash2))
+ })
+
+ It("produces different hash when images updated time changes", func() {
+ hash1 := entry.hash()
+
+ entry.imagesUpdatedAt = entry.imagesUpdatedAt.Add(2 * time.Hour)
+ hash2 := entry.hash()
+
+ Expect(hash1).ToNot(Equal(hash2))
+ })
+
+ It("produces different hash when audio file size changes", func() {
+ entry.audioFiles["test.mp3"] = &fakeDirEntry{
+ name: "test.mp3",
+ fileInfo: &fakeFileInfo{
+ name: "test.mp3",
+ size: 1000,
+ modTime: time.Now(),
+ },
+ }
+ hash1 := entry.hash()
+
+ entry.audioFiles["test.mp3"] = &fakeDirEntry{
+ name: "test.mp3",
+ fileInfo: &fakeFileInfo{
+ name: "test.mp3",
+ size: 2000, // Different size
+ modTime: time.Now(),
+ },
+ }
+ hash2 := entry.hash()
+
+ Expect(hash1).ToNot(Equal(hash2))
+ })
+
+ It("produces different hash when audio file modification time changes", func() {
+ baseTime := time.Now()
+ entry.audioFiles["test.mp3"] = &fakeDirEntry{
+ name: "test.mp3",
+ fileInfo: &fakeFileInfo{
+ name: "test.mp3",
+ size: 1000,
+ modTime: baseTime,
+ },
+ }
+ hash1 := entry.hash()
+
+ entry.audioFiles["test.mp3"] = &fakeDirEntry{
+ name: "test.mp3",
+ fileInfo: &fakeFileInfo{
+ name: "test.mp3",
+ size: 1000,
+ modTime: baseTime.Add(1 * time.Hour), // Different modtime
+ },
+ }
+ hash2 := entry.hash()
+
+ Expect(hash1).ToNot(Equal(hash2))
+ })
+
+ It("produces different hash when image file size changes", func() {
+ entry.imageFiles["cover.jpg"] = &fakeDirEntry{
+ name: "cover.jpg",
+ fileInfo: &fakeFileInfo{
+ name: "cover.jpg",
+ size: 5000,
+ modTime: time.Now(),
+ },
+ }
+ hash1 := entry.hash()
+
+ entry.imageFiles["cover.jpg"] = &fakeDirEntry{
+ name: "cover.jpg",
+ fileInfo: &fakeFileInfo{
+ name: "cover.jpg",
+ size: 6000, // Different size
+ modTime: time.Now(),
+ },
+ }
+ hash2 := entry.hash()
+
+ Expect(hash1).ToNot(Equal(hash2))
+ })
+
+ It("produces different hash when image file modification time changes", func() {
+ baseTime := time.Now()
+ entry.imageFiles["cover.jpg"] = &fakeDirEntry{
+ name: "cover.jpg",
+ fileInfo: &fakeFileInfo{
+ name: "cover.jpg",
+ size: 5000,
+ modTime: baseTime,
+ },
+ }
+ hash1 := entry.hash()
+
+ entry.imageFiles["cover.jpg"] = &fakeDirEntry{
+ name: "cover.jpg",
+ fileInfo: &fakeFileInfo{
+ name: "cover.jpg",
+ size: 5000,
+ modTime: baseTime.Add(1 * time.Hour), // Different modtime
+ },
+ }
+ hash2 := entry.hash()
+
+ Expect(hash1).ToNot(Equal(hash2))
+ })
+
+ It("produces valid hex-encoded hash", func() {
+ hash := entry.hash()
+ Expect(hash).To(HaveLen(32)) // MD5 hash should be 32 hex characters
+ Expect(hash).To(MatchRegexp("^[a-f0-9]{32}$"))
+ })
+ })
+
+ Describe("isOutdated", func() {
+ BeforeEach(func() {
+ entry.prevHash = entry.hash()
+ })
+
+ Context("when full scan is in progress", func() {
+ BeforeEach(func() {
+ entry.job.lib.FullScanInProgress = true
+ entry.job.lib.LastScanStartedAt = time.Now()
+ })
+
+ It("returns true when updTime is before LastScanStartedAt", func() {
+ entry.updTime = entry.job.lib.LastScanStartedAt.Add(-1 * time.Hour)
+ Expect(entry.isOutdated()).To(BeTrue())
+ })
+
+ It("returns false when updTime is after LastScanStartedAt", func() {
+ entry.updTime = entry.job.lib.LastScanStartedAt.Add(1 * time.Hour)
+ Expect(entry.isOutdated()).To(BeFalse())
+ })
+
+ It("returns false when updTime equals LastScanStartedAt", func() {
+ entry.updTime = entry.job.lib.LastScanStartedAt
+ Expect(entry.isOutdated()).To(BeFalse())
+ })
+ })
+
+ Context("when full scan is not in progress", func() {
+ BeforeEach(func() {
+ entry.job.lib.FullScanInProgress = false
+ })
+
+ It("returns false when hash hasn't changed", func() {
+ Expect(entry.isOutdated()).To(BeFalse())
+ })
+
+ It("returns true when hash has changed", func() {
+ entry.numPlaylists = 10 // Change something to change the hash
+ Expect(entry.isOutdated()).To(BeTrue())
+ })
+
+ It("returns true when prevHash is empty", func() {
+ entry.prevHash = ""
+ Expect(entry.isOutdated()).To(BeTrue())
+ })
+ })
+
+ Context("priority between conditions", func() {
+ BeforeEach(func() {
+ entry.job.lib.FullScanInProgress = true
+ entry.job.lib.LastScanStartedAt = time.Now()
+ entry.updTime = entry.job.lib.LastScanStartedAt.Add(-1 * time.Hour)
+ })
+
+ It("returns true for full scan condition even when hash hasn't changed", func() {
+ // Hash is the same but full scan condition should take priority
+ Expect(entry.isOutdated()).To(BeTrue())
+ })
+
+ It("returns true when full scan condition is not met but hash changed", func() {
+ entry.updTime = entry.job.lib.LastScanStartedAt.Add(1 * time.Hour)
+ entry.numPlaylists = 10 // Change hash
+ Expect(entry.isOutdated()).To(BeTrue())
+ })
+ })
+ })
+ })
+
+ Describe("integration scenarios", func() {
+ It("handles complete folder lifecycle", func() {
+ // Create new folder entry
+ folderPath := "music/rock/album"
+ folderID := model.FolderID(lib, folderPath)
+ entry := newFolderEntry(job, folderID, folderPath, time.Time{}, "")
+
+ // Initially new and has no files
+ Expect(entry.isNew()).To(BeTrue())
+ Expect(entry.hasNoFiles()).To(BeTrue())
+
+ // Add some files
+ entry.audioFiles["track1.mp3"] = &fakeDirEntry{name: "track1.mp3"}
+ entry.audioFiles["track2.mp3"] = &fakeDirEntry{name: "track2.mp3"}
+ entry.imageFiles["cover.jpg"] = &fakeDirEntry{name: "cover.jpg"}
+ entry.numSubFolders = 1
+ entry.modTime = time.Now()
+ entry.imagesUpdatedAt = time.Now()
+
+ // No longer empty
+ Expect(entry.hasNoFiles()).To(BeFalse())
+
+ // Set previous hash to current hash (simulating it's been saved)
+ entry.prevHash = entry.hash()
+ entry.updTime = time.Now()
+
+ // Should not be new or outdated
+ Expect(entry.isNew()).To(BeFalse())
+ Expect(entry.isOutdated()).To(BeFalse())
+
+ // Convert to model folder
+ folder := entry.toFolder()
+ Expect(folder.NumAudioFiles).To(Equal(2))
+ Expect(folder.ImageFiles).To(HaveLen(1))
+ Expect(folder.Hash).To(Equal(entry.hash()))
+
+ // Modify folder and verify it becomes outdated
+ entry.audioFiles["track3.mp3"] = &fakeDirEntry{name: "track3.mp3"}
+ Expect(entry.isOutdated()).To(BeTrue())
+ })
+ })
+})
+
+// fakeDirEntry implements fs.DirEntry for testing
+type fakeDirEntry struct {
+ name string
+ isDir bool
+ typ fs.FileMode
+ fileInfo fs.FileInfo
+}
+
+func (f *fakeDirEntry) Name() string {
+ return f.name
+}
+
+func (f *fakeDirEntry) IsDir() bool {
+ return f.isDir
+}
+
+func (f *fakeDirEntry) Type() fs.FileMode {
+ return f.typ
+}
+
+func (f *fakeDirEntry) Info() (fs.FileInfo, error) {
+ if f.fileInfo != nil {
+ return f.fileInfo, nil
+ }
+ return &fakeFileInfo{
+ name: f.name,
+ isDir: f.isDir,
+ mode: f.typ,
+ }, nil
+}
+
+// fakeFileInfo implements fs.FileInfo for testing
+type fakeFileInfo struct {
+ name string
+ size int64
+ mode fs.FileMode
+ modTime time.Time
+ isDir bool
+}
+
+func (f *fakeFileInfo) Name() string { return f.name }
+func (f *fakeFileInfo) Size() int64 { return f.size }
+func (f *fakeFileInfo) Mode() fs.FileMode { return f.mode }
+func (f *fakeFileInfo) ModTime() time.Time { return f.modTime }
+func (f *fakeFileInfo) IsDir() bool { return f.isDir }
+func (f *fakeFileInfo) Sys() any { return nil }
diff --git a/scanner/ignore_checker.go b/scanner/ignore_checker.go
new file mode 100644
index 000000000..da74293fa
--- /dev/null
+++ b/scanner/ignore_checker.go
@@ -0,0 +1,163 @@
+package scanner
+
+import (
+ "bufio"
+ "context"
+ "io/fs"
+ "path"
+ "strings"
+
+ "github.com/navidrome/navidrome/consts"
+ "github.com/navidrome/navidrome/log"
+ ignore "github.com/sabhiram/go-gitignore"
+)
+
+// IgnoreChecker manages .ndignore patterns using a stack-based approach.
+// Use Push() to add patterns when entering a folder, Pop() when leaving,
+// and ShouldIgnore() to check if a path should be ignored.
+type IgnoreChecker struct {
+ fsys fs.FS
+ patternStack [][]string // Stack of patterns for each folder level
+ currentPatterns []string // Flattened current patterns
+ matcher *ignore.GitIgnore // Compiled matcher for current patterns
+}
+
+// newIgnoreChecker creates a new IgnoreChecker for the given filesystem.
+func newIgnoreChecker(fsys fs.FS) *IgnoreChecker {
+ return &IgnoreChecker{
+ fsys: fsys,
+ patternStack: make([][]string, 0),
+ }
+}
+
+// Push loads .ndignore patterns from the specified folder and adds them to the pattern stack.
+// Use this when entering a folder during directory tree traversal.
+func (ic *IgnoreChecker) Push(ctx context.Context, folder string) error {
+ patterns := ic.loadPatternsFromFolder(ctx, folder)
+ ic.patternStack = append(ic.patternStack, patterns)
+ ic.rebuildCurrentPatterns()
+ return nil
+}
+
+// Pop removes the most recent patterns from the stack.
+// Use this when leaving a folder during directory tree traversal.
+func (ic *IgnoreChecker) Pop() {
+ if len(ic.patternStack) > 0 {
+ ic.patternStack = ic.patternStack[:len(ic.patternStack)-1]
+ ic.rebuildCurrentPatterns()
+ }
+}
+
+// PushAllParents pushes patterns from root down to the target path.
+// This is a convenience method for when you need to check a specific path
+// without recursively walking the tree. It handles the common pattern of
+// pushing all parent directories from root to the target.
+// This method is optimized to compile patterns only once at the end.
+func (ic *IgnoreChecker) PushAllParents(ctx context.Context, targetPath string) error {
+ if targetPath == "." || targetPath == "" {
+ // Simple case: just push root
+ return ic.Push(ctx, ".")
+ }
+
+ // Load patterns for root
+ patterns := ic.loadPatternsFromFolder(ctx, ".")
+ ic.patternStack = append(ic.patternStack, patterns)
+
+ // Load patterns for each parent directory
+ currentPath := "."
+ parts := strings.Split(path.Clean(targetPath), "/")
+ for _, part := range parts {
+ if part == "." || part == "" {
+ continue
+ }
+ currentPath = path.Join(currentPath, part)
+ patterns = ic.loadPatternsFromFolder(ctx, currentPath)
+ ic.patternStack = append(ic.patternStack, patterns)
+ }
+
+ // Rebuild and compile patterns only once at the end
+ ic.rebuildCurrentPatterns()
+ return nil
+}
+
+// ShouldIgnore checks if the given path should be ignored based on the current patterns.
+// Returns true if the path matches any ignore pattern, false otherwise.
+func (ic *IgnoreChecker) ShouldIgnore(ctx context.Context, relPath string) bool {
+ // Handle root/empty path - never ignore
+ if relPath == "" || relPath == "." {
+ return false
+ }
+
+ // If no patterns loaded, nothing to ignore
+ if ic.matcher == nil {
+ return false
+ }
+
+ matches := ic.matcher.MatchesPath(relPath)
+ if matches {
+ log.Trace(ctx, "Scanner: Ignoring entry matching .ndignore", "path", relPath)
+ }
+ return matches
+}
+
+// loadPatternsFromFolder reads the .ndignore file in the specified folder and returns the patterns.
+// If the file doesn't exist, returns an empty slice.
+// If the file exists but is empty, returns a pattern to ignore everything ("**/*").
+func (ic *IgnoreChecker) loadPatternsFromFolder(ctx context.Context, folder string) []string {
+ ignoreFilePath := path.Join(folder, consts.ScanIgnoreFile)
+ var patterns []string
+
+ // Check if .ndignore file exists
+ if _, err := fs.Stat(ic.fsys, ignoreFilePath); err != nil {
+ // No .ndignore file in this folder
+ return patterns
+ }
+
+ // Read and parse the .ndignore file
+ ignoreFile, err := ic.fsys.Open(ignoreFilePath)
+ if err != nil {
+ log.Warn(ctx, "Scanner: Error opening .ndignore file", "path", ignoreFilePath, err)
+ return patterns
+ }
+ defer ignoreFile.Close()
+
+ lineScanner := bufio.NewScanner(ignoreFile)
+ for lineScanner.Scan() {
+ line := strings.TrimSpace(lineScanner.Text())
+ if line == "" || strings.HasPrefix(line, "#") {
+ continue // Skip empty lines, whitespace-only lines, and comments
+ }
+ patterns = append(patterns, line)
+ }
+
+ if err := lineScanner.Err(); err != nil {
+ log.Warn(ctx, "Scanner: Error reading .ndignore file", "path", ignoreFilePath, err)
+ return patterns
+ }
+
+ // If the .ndignore file is empty, ignore everything
+ if len(patterns) == 0 {
+ log.Trace(ctx, "Scanner: .ndignore file is empty, ignoring everything", "path", folder)
+ patterns = []string{"**/*"}
+ }
+
+ return patterns
+}
+
+// rebuildCurrentPatterns flattens the pattern stack into currentPatterns and recompiles the matcher.
+func (ic *IgnoreChecker) rebuildCurrentPatterns() {
+ ic.currentPatterns = make([]string, 0)
+ for _, patterns := range ic.patternStack {
+ ic.currentPatterns = append(ic.currentPatterns, patterns...)
+ }
+ ic.compilePatterns()
+}
+
+// compilePatterns compiles the current patterns into a GitIgnore matcher.
+func (ic *IgnoreChecker) compilePatterns() {
+ if len(ic.currentPatterns) == 0 {
+ ic.matcher = nil
+ return
+ }
+ ic.matcher = ignore.CompileIgnoreLines(ic.currentPatterns...)
+}
diff --git a/scanner/ignore_checker_test.go b/scanner/ignore_checker_test.go
new file mode 100644
index 000000000..5378ed4fa
--- /dev/null
+++ b/scanner/ignore_checker_test.go
@@ -0,0 +1,313 @@
+package scanner
+
+import (
+ "context"
+ "testing/fstest"
+
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("IgnoreChecker", func() {
+ Describe("loadPatternsFromFolder", func() {
+ var ic *IgnoreChecker
+ var ctx context.Context
+
+ BeforeEach(func() {
+ ctx = context.Background()
+ })
+
+ Context("when .ndignore file does not exist", func() {
+ It("should return empty patterns", func() {
+ fsys := fstest.MapFS{}
+ ic = newIgnoreChecker(fsys)
+ patterns := ic.loadPatternsFromFolder(ctx, ".")
+ Expect(patterns).To(BeEmpty())
+ })
+ })
+
+ Context("when .ndignore file is empty", func() {
+ It("should return wildcard to ignore everything", func() {
+ fsys := fstest.MapFS{
+ ".ndignore": &fstest.MapFile{Data: []byte("")},
+ }
+ ic = newIgnoreChecker(fsys)
+ patterns := ic.loadPatternsFromFolder(ctx, ".")
+ Expect(patterns).To(Equal([]string{"**/*"}))
+ })
+ })
+
+ DescribeTable("parsing .ndignore content",
+ func(content string, expectedPatterns []string) {
+ fsys := fstest.MapFS{
+ ".ndignore": &fstest.MapFile{Data: []byte(content)},
+ }
+ ic = newIgnoreChecker(fsys)
+ patterns := ic.loadPatternsFromFolder(ctx, ".")
+ Expect(patterns).To(Equal(expectedPatterns))
+ },
+ Entry("single pattern", "*.txt", []string{"*.txt"}),
+ Entry("multiple patterns", "*.txt\n*.log", []string{"*.txt", "*.log"}),
+ Entry("with comments", "# comment\n*.txt\n# another\n*.log", []string{"*.txt", "*.log"}),
+ Entry("with empty lines", "*.txt\n\n*.log\n\n", []string{"*.txt", "*.log"}),
+ Entry("mixed content", "# header\n\n*.txt\n# middle\n*.log\n\n", []string{"*.txt", "*.log"}),
+ Entry("only comments and empty lines", "# comment\n\n# another\n", []string{"**/*"}),
+ Entry("trailing newline", "*.txt\n*.log\n", []string{"*.txt", "*.log"}),
+ Entry("directory pattern", "temp/", []string{"temp/"}),
+ Entry("wildcard pattern", "**/*.mp3", []string{"**/*.mp3"}),
+ Entry("multiple wildcards", "**/*.mp3\n**/*.flac\n*.log", []string{"**/*.mp3", "**/*.flac", "*.log"}),
+ Entry("negation pattern", "!important.txt", []string{"!important.txt"}),
+ Entry("comment with hash not at start is pattern", "not#comment", []string{"not#comment"}),
+ Entry("whitespace-only lines skipped", "*.txt\n \n*.log\n\t\n", []string{"*.txt", "*.log"}),
+ Entry("patterns with whitespace trimmed", " *.txt \n\t*.log\t", []string{"*.txt", "*.log"}),
+ )
+ })
+
+ Describe("Push and Pop", func() {
+ var ic *IgnoreChecker
+ var fsys fstest.MapFS
+ var ctx context.Context
+
+ BeforeEach(func() {
+ ctx = context.Background()
+ fsys = fstest.MapFS{
+ ".ndignore": &fstest.MapFile{Data: []byte("*.txt")},
+ "folder1/.ndignore": &fstest.MapFile{Data: []byte("*.mp3")},
+ "folder2/.ndignore": &fstest.MapFile{Data: []byte("*.flac")},
+ }
+ ic = newIgnoreChecker(fsys)
+ })
+
+ Context("Push", func() {
+ It("should add patterns to stack", func() {
+ err := ic.Push(ctx, ".")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(len(ic.patternStack)).To(Equal(1))
+ Expect(ic.currentPatterns).To(ContainElement("*.txt"))
+ })
+
+ It("should compile matcher after push", func() {
+ err := ic.Push(ctx, ".")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ic.matcher).ToNot(BeNil())
+ })
+
+ It("should accumulate patterns from multiple levels", func() {
+ err := ic.Push(ctx, ".")
+ Expect(err).ToNot(HaveOccurred())
+ err = ic.Push(ctx, "folder1")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(len(ic.patternStack)).To(Equal(2))
+ Expect(ic.currentPatterns).To(ConsistOf("*.txt", "*.mp3"))
+ })
+
+ It("should handle push when no .ndignore exists", func() {
+ err := ic.Push(ctx, "nonexistent")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(len(ic.patternStack)).To(Equal(1))
+ Expect(ic.currentPatterns).To(BeEmpty())
+ })
+ })
+
+ Context("Pop", func() {
+ It("should remove most recent patterns", func() {
+ err := ic.Push(ctx, ".")
+ Expect(err).ToNot(HaveOccurred())
+ err = ic.Push(ctx, "folder1")
+ Expect(err).ToNot(HaveOccurred())
+ ic.Pop()
+ Expect(len(ic.patternStack)).To(Equal(1))
+ Expect(ic.currentPatterns).To(Equal([]string{"*.txt"}))
+ })
+
+ It("should handle Pop on empty stack gracefully", func() {
+ Expect(func() { ic.Pop() }).ToNot(Panic())
+ Expect(ic.patternStack).To(BeEmpty())
+ })
+
+ It("should set matcher to nil when all patterns popped", func() {
+ err := ic.Push(ctx, ".")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ic.matcher).ToNot(BeNil())
+ ic.Pop()
+ Expect(ic.matcher).To(BeNil())
+ })
+
+ It("should update matcher after pop", func() {
+ err := ic.Push(ctx, ".")
+ Expect(err).ToNot(HaveOccurred())
+ err = ic.Push(ctx, "folder1")
+ Expect(err).ToNot(HaveOccurred())
+ matcher1 := ic.matcher
+ ic.Pop()
+ matcher2 := ic.matcher
+ Expect(matcher1).ToNot(Equal(matcher2))
+ })
+ })
+
+ Context("multiple Push/Pop cycles", func() {
+ It("should maintain correct state through cycles", func() {
+ err := ic.Push(ctx, ".")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ic.currentPatterns).To(Equal([]string{"*.txt"}))
+
+ err = ic.Push(ctx, "folder1")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ic.currentPatterns).To(ConsistOf("*.txt", "*.mp3"))
+
+ ic.Pop()
+ Expect(ic.currentPatterns).To(Equal([]string{"*.txt"}))
+
+ err = ic.Push(ctx, "folder2")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ic.currentPatterns).To(ConsistOf("*.txt", "*.flac"))
+
+ ic.Pop()
+ Expect(ic.currentPatterns).To(Equal([]string{"*.txt"}))
+
+ ic.Pop()
+ Expect(ic.currentPatterns).To(BeEmpty())
+ })
+ })
+ })
+
+ Describe("PushAllParents", func() {
+ var ic *IgnoreChecker
+ var ctx context.Context
+
+ BeforeEach(func() {
+ ctx = context.Background()
+ fsys := fstest.MapFS{
+ ".ndignore": &fstest.MapFile{Data: []byte("root.txt")},
+ "folder1/.ndignore": &fstest.MapFile{Data: []byte("level1.txt")},
+ "folder1/folder2/.ndignore": &fstest.MapFile{Data: []byte("level2.txt")},
+ "folder1/folder2/folder3/.ndignore": &fstest.MapFile{Data: []byte("level3.txt")},
+ }
+ ic = newIgnoreChecker(fsys)
+ })
+
+ DescribeTable("loading parent patterns",
+ func(targetPath string, expectedStackDepth int, expectedPatterns []string) {
+ err := ic.PushAllParents(ctx, targetPath)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(len(ic.patternStack)).To(Equal(expectedStackDepth))
+ Expect(ic.currentPatterns).To(ConsistOf(expectedPatterns))
+ },
+ Entry("root path", ".", 1, []string{"root.txt"}),
+ Entry("empty path", "", 1, []string{"root.txt"}),
+ Entry("single level", "folder1", 2, []string{"root.txt", "level1.txt"}),
+ Entry("two levels", "folder1/folder2", 3, []string{"root.txt", "level1.txt", "level2.txt"}),
+ Entry("three levels", "folder1/folder2/folder3", 4, []string{"root.txt", "level1.txt", "level2.txt", "level3.txt"}),
+ )
+
+ It("should only compile patterns once at the end", func() {
+ // This is more of a behavioral test - we verify the matcher is not nil after PushAllParents
+ err := ic.PushAllParents(ctx, "folder1/folder2")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ic.matcher).ToNot(BeNil())
+ })
+
+ It("should handle paths with dot", func() {
+ err := ic.PushAllParents(ctx, "./folder1")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(len(ic.patternStack)).To(Equal(2))
+ })
+
+ Context("when some parent folders have no .ndignore", func() {
+ BeforeEach(func() {
+ fsys := fstest.MapFS{
+ ".ndignore": &fstest.MapFile{Data: []byte("root.txt")},
+ "folder1/folder2/.ndignore": &fstest.MapFile{Data: []byte("level2.txt")},
+ }
+ ic = newIgnoreChecker(fsys)
+ })
+
+ It("should still push all parent levels", func() {
+ err := ic.PushAllParents(ctx, "folder1/folder2")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(len(ic.patternStack)).To(Equal(3)) // root, folder1 (empty), folder2
+ Expect(ic.currentPatterns).To(ConsistOf("root.txt", "level2.txt"))
+ })
+ })
+ })
+
+ Describe("ShouldIgnore", func() {
+ var ic *IgnoreChecker
+ var ctx context.Context
+
+ BeforeEach(func() {
+ ctx = context.Background()
+ })
+
+ Context("with no patterns loaded", func() {
+ It("should not ignore any path", func() {
+ fsys := fstest.MapFS{}
+ ic = newIgnoreChecker(fsys)
+ Expect(ic.ShouldIgnore(ctx, "anything.txt")).To(BeFalse())
+ Expect(ic.ShouldIgnore(ctx, "folder/file.mp3")).To(BeFalse())
+ })
+ })
+
+ Context("special paths", func() {
+ BeforeEach(func() {
+ fsys := fstest.MapFS{
+ ".ndignore": &fstest.MapFile{Data: []byte("**/*")},
+ }
+ ic = newIgnoreChecker(fsys)
+ err := ic.Push(ctx, ".")
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ It("should never ignore root or empty paths", func() {
+ Expect(ic.ShouldIgnore(ctx, "")).To(BeFalse())
+ Expect(ic.ShouldIgnore(ctx, ".")).To(BeFalse())
+ })
+
+ It("should ignore all other paths with wildcard", func() {
+ Expect(ic.ShouldIgnore(ctx, "file.txt")).To(BeTrue())
+ Expect(ic.ShouldIgnore(ctx, "folder/file.mp3")).To(BeTrue())
+ })
+ })
+
+ DescribeTable("pattern matching",
+ func(pattern string, path string, shouldMatch bool) {
+ fsys := fstest.MapFS{
+ ".ndignore": &fstest.MapFile{Data: []byte(pattern)},
+ }
+ ic = newIgnoreChecker(fsys)
+ err := ic.Push(ctx, ".")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ic.ShouldIgnore(ctx, path)).To(Equal(shouldMatch))
+ },
+ Entry("glob match", "*.txt", "file.txt", true),
+ Entry("glob no match", "*.txt", "file.mp3", false),
+ Entry("directory pattern match", "tmp/", "tmp/file.txt", true),
+ Entry("directory pattern no match", "tmp/", "temporary/file.txt", false),
+ Entry("nested glob match", "**/*.log", "deep/nested/file.log", true),
+ Entry("nested glob no match", "**/*.log", "deep/nested/file.txt", false),
+ Entry("specific file match", "ignore.me", "ignore.me", true),
+ Entry("specific file no match", "ignore.me", "keep.me", false),
+ Entry("wildcard all", "**/*", "any/path/file.txt", true),
+ Entry("nested specific match", "temp/*", "temp/cache.db", true),
+ Entry("nested specific no match", "temp/*", "temporary/cache.db", false),
+ )
+
+ Context("with multiple patterns", func() {
+ BeforeEach(func() {
+ fsys := fstest.MapFS{
+ ".ndignore": &fstest.MapFile{Data: []byte("*.txt\n*.log\ntemp/")},
+ }
+ ic = newIgnoreChecker(fsys)
+ err := ic.Push(ctx, ".")
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ It("should match any of the patterns", func() {
+ Expect(ic.ShouldIgnore(ctx, "file.txt")).To(BeTrue())
+ Expect(ic.ShouldIgnore(ctx, "debug.log")).To(BeTrue())
+ Expect(ic.ShouldIgnore(ctx, "temp/cache")).To(BeTrue())
+ Expect(ic.ShouldIgnore(ctx, "music.mp3")).To(BeFalse())
+ })
+ })
+ })
+})
diff --git a/scanner/phase_1_folders.go b/scanner/phase_1_folders.go
index ae0d906de..329029951 100644
--- a/scanner/phase_1_folders.go
+++ b/scanner/phase_1_folders.go
@@ -26,28 +26,18 @@ import (
"github.com/navidrome/navidrome/utils/slice"
)
-func createPhaseFolders(ctx context.Context, state *scanState, ds model.DataStore, cw artwork.CacheWarmer, libs []model.Library) *phaseFolders {
+func createPhaseFolders(ctx context.Context, state *scanState, ds model.DataStore, cw artwork.CacheWarmer) *phaseFolders {
var jobs []*scanJob
- for _, lib := range libs {
- if lib.LastScanStartedAt.IsZero() {
- err := ds.Library(ctx).ScanBegin(lib.ID, state.fullScan)
- if err != nil {
- log.Error(ctx, "Scanner: Error updating last scan started at", "lib", lib.Name, err)
- state.sendWarning(err.Error())
- continue
- }
- // Reload library to get updated state
- l, err := ds.Library(ctx).Get(lib.ID)
- if err != nil {
- log.Error(ctx, "Scanner: Error reloading library", "lib", lib.Name, err)
- state.sendWarning(err.Error())
- continue
- }
- lib = *l
- } else {
- log.Debug(ctx, "Scanner: Resuming previous scan", "lib", lib.Name, "lastScanStartedAt", lib.LastScanStartedAt, "fullScan", lib.FullScanInProgress)
+
+ // Create scan jobs for all libraries
+ for _, lib := range state.libraries {
+ // Get target folders for this library if selective scan
+ var targetFolders []string
+ if state.isSelectiveScan() {
+ targetFolders = state.targets[lib.ID]
}
- job, err := newScanJob(ctx, ds, cw, lib, state.fullScan)
+
+ job, err := newScanJob(ctx, ds, cw, lib, state.fullScan, targetFolders)
if err != nil {
log.Error(ctx, "Scanner: Error creating scan context", "lib", lib.Name, err)
state.sendWarning(err.Error())
@@ -55,23 +45,27 @@ func createPhaseFolders(ctx context.Context, state *scanState, ds model.DataStor
}
jobs = append(jobs, job)
}
+
return &phaseFolders{jobs: jobs, ctx: ctx, ds: ds, state: state}
}
type scanJob struct {
- lib model.Library
- fs storage.MusicFS
- cw artwork.CacheWarmer
- lastUpdates map[string]time.Time
- lock sync.Mutex
- numFolders atomic.Int64
+ lib model.Library
+ fs storage.MusicFS
+ cw artwork.CacheWarmer
+ lastUpdates map[string]model.FolderUpdateInfo // Holds last update info for all (DB) folders in this library
+ targetFolders []string // Specific folders to scan (including all descendants)
+ lock sync.Mutex
+ numFolders atomic.Int64
}
-func newScanJob(ctx context.Context, ds model.DataStore, cw artwork.CacheWarmer, lib model.Library, fullScan bool) (*scanJob, error) {
- lastUpdates, err := ds.Folder(ctx).GetLastUpdates(lib)
+func newScanJob(ctx context.Context, ds model.DataStore, cw artwork.CacheWarmer, lib model.Library, fullScan bool, targetFolders []string) (*scanJob, error) {
+ // Get folder updates, optionally filtered to specific target folders
+ lastUpdates, err := ds.Folder(ctx).GetFolderUpdateInfo(lib, targetFolders...)
if err != nil {
return nil, fmt.Errorf("getting last updates: %w", err)
}
+
fileStore, err := storage.For(lib.Path)
if err != nil {
log.Error(ctx, "Error getting storage for library", "library", lib.Name, "path", lib.Path, err)
@@ -82,16 +76,18 @@ func newScanJob(ctx context.Context, ds model.DataStore, cw artwork.CacheWarmer,
log.Error(ctx, "Error getting fs for library", "library", lib.Name, "path", lib.Path, err)
return nil, fmt.Errorf("getting fs for library: %w", err)
}
- lib.FullScanInProgress = lib.FullScanInProgress || fullScan
return &scanJob{
- lib: lib,
- fs: fsys,
- cw: cw,
- lastUpdates: lastUpdates,
+ lib: lib,
+ fs: fsys,
+ cw: cw,
+ lastUpdates: lastUpdates,
+ targetFolders: targetFolders,
}, nil
}
-func (j *scanJob) popLastUpdate(folderID string) time.Time {
+// popLastUpdate retrieves and removes the last update info for the given folder ID
+// This is used to track which folders have been found during the walk_dir_tree
+func (j *scanJob) popLastUpdate(folderID string) model.FolderUpdateInfo {
j.lock.Lock()
defer j.lock.Unlock()
@@ -100,6 +96,15 @@ func (j *scanJob) popLastUpdate(folderID string) time.Time {
return lastUpdate
}
+// createFolderEntry creates a new folderEntry for the given path, using the last update info from the job
+// to populate the previous update time and hash. It also removes the folder from the job's lastUpdates map.
+// This is used to track which folders have been found during the walk_dir_tree.
+func (j *scanJob) createFolderEntry(path string) *folderEntry {
+ id := model.FolderID(j.lib, path)
+ info := j.popLastUpdate(id)
+ return newFolderEntry(j, id, path, info.UpdatedAt, info.Hash)
+}
+
// phaseFolders represents the first phase of the scanning process, which is responsible
// for scanning all libraries and importing new or updated files. This phase involves
// traversing the directory tree of each library, identifying new or modified media files,
@@ -138,7 +143,8 @@ func (p *phaseFolders) producer() ppl.Producer[*folderEntry] {
if utils.IsCtxDone(p.ctx) {
break
}
- outputChan, err := walkDirTree(p.ctx, job)
+
+ outputChan, err := walkDirTree(p.ctx, job, job.targetFolders...)
if err != nil {
log.Warn(p.ctx, "Scanner: Error scanning library", "lib", job.lib.Name, err)
}
@@ -164,7 +170,7 @@ func (p *phaseFolders) producer() ppl.Producer[*folderEntry] {
log.Trace(p.ctx, "Scanner: Skipping new folder with no files", "folder", folder.path, "lib", job.lib.Name)
continue
}
- log.Trace(p.ctx, "Scanner: Detected changes in folder", "folder", folder.path, "lastUpdate", folder.modTime, "lib", job.lib.Name)
+ log.Debug(p.ctx, "Scanner: Detected changes in folder", "folder", folder.path, "lastUpdate", folder.modTime, "lib", job.lib.Name)
}
totalChanged++
folder.elapsed.Stop()
@@ -318,6 +324,9 @@ func (p *phaseFolders) persistChanges(entry *folderEntry) (*folderEntry, error)
defer p.measure(entry)()
p.state.changesDetected.Store(true)
+ // Collect artwork IDs to pre-cache after the transaction commits
+ var artworkIDs []model.ArtworkID
+
err := p.ds.WithTx(func(tx model.DataStore) error {
// Instantiate all repositories just once per folder
folderRepo := tx.Folder(p.ctx)
@@ -336,7 +345,7 @@ func (p *phaseFolders) persistChanges(entry *folderEntry) (*folderEntry, error)
}
// Save all tags to DB
- err = tagRepo.Add(entry.tags...)
+ err = tagRepo.Add(entry.job.lib.ID, entry.tags...)
if err != nil {
log.Error(p.ctx, "Scanner: Error persisting tags to DB", "folder", entry.path, err)
return err
@@ -345,7 +354,7 @@ func (p *phaseFolders) persistChanges(entry *folderEntry) (*folderEntry, error)
// Save all new/modified artists to DB. Their information will be incomplete, but they will be refreshed later
for i := range entry.artists {
err = artistRepo.Put(&entry.artists[i], "name",
- "mbz_artist_id", "sort_artist_name", "order_artist_name", "full_text")
+ "mbz_artist_id", "sort_artist_name", "order_artist_name", "full_text", "updated_at")
if err != nil {
log.Error(p.ctx, "Scanner: Error persisting artist to DB", "folder", entry.path, "artist", entry.artists[i].Name, err)
return err
@@ -356,7 +365,7 @@ func (p *phaseFolders) persistChanges(entry *folderEntry) (*folderEntry, error)
return err
}
if entry.artists[i].Name != consts.UnknownArtist && entry.artists[i].Name != consts.VariousArtists {
- entry.job.cw.PreCache(entry.artists[i].CoverArtID())
+ artworkIDs = append(artworkIDs, entry.artists[i].CoverArtID())
}
}
@@ -368,7 +377,7 @@ func (p *phaseFolders) persistChanges(entry *folderEntry) (*folderEntry, error)
return err
}
if entry.albums[i].Name != consts.UnknownAlbum {
- entry.job.cw.PreCache(entry.albums[i].CoverArtID())
+ artworkIDs = append(artworkIDs, entry.albums[i].CoverArtID())
}
}
@@ -405,6 +414,14 @@ func (p *phaseFolders) persistChanges(entry *folderEntry) (*folderEntry, error)
if err != nil {
log.Error(p.ctx, "Scanner: Error persisting changes to DB", "folder", entry.path, err)
}
+
+ // Pre-cache artwork after the transaction commits successfully
+ if err == nil {
+ for _, artID := range artworkIDs {
+ entry.job.cw.PreCache(artID)
+ }
+ }
+
return entry, err
}
@@ -418,12 +435,14 @@ func (p *phaseFolders) persistAlbum(repo model.AlbumRepository, a *model.Album,
if prevID == "" {
return nil
}
+
// Reassign annotation from previous album to new album
log.Trace(p.ctx, "Reassigning album annotations", "from", prevID, "to", a.ID, "album", a.Name)
if err := repo.ReassignAnnotation(prevID, a.ID); err != nil {
log.Warn(p.ctx, "Scanner: Could not reassign annotations", "from", prevID, "to", a.ID, "album", a.Name, err)
p.state.sendWarning(fmt.Sprintf("Could not reassign annotations from %s to %s ('%s'): %v", prevID, a.ID, a.Name, err))
}
+
// Keep created_at field from previous instance of the album
if err := repo.CopyAttributes(prevID, a.ID, "created_at"); err != nil {
// Silently ignore when the previous album is not found
@@ -439,7 +458,7 @@ func (p *phaseFolders) persistAlbum(repo model.AlbumRepository, a *model.Album,
func (p *phaseFolders) logFolder(entry *folderEntry) (*folderEntry, error) {
logCall := log.Info
- if entry.hasNoFiles() {
+ if entry.isEmpty() {
logCall = log.Trace
}
logCall(p.ctx, "Scanner: Completed processing folder",
diff --git a/scanner/phase_2_missing_tracks.go b/scanner/phase_2_missing_tracks.go
index 6f56f6a52..de93ed6ee 100644
--- a/scanner/phase_2_missing_tracks.go
+++ b/scanner/phase_2_missing_tracks.go
@@ -3,6 +3,7 @@ package scanner
import (
"context"
"fmt"
+ "sync"
"sync/atomic"
ppl "github.com/google/go-pipeline/pkg/pipeline"
@@ -31,14 +32,21 @@ type missingTracks struct {
// 4. Updates the database with the new locations of the matched files and removes the old entries.
// 5. Logs the results and finalizes the phase by reporting the total number of matched files.
type phaseMissingTracks struct {
- ctx context.Context
- ds model.DataStore
- totalMatched atomic.Uint32
- state *scanState
+ ctx context.Context
+ ds model.DataStore
+ totalMatched atomic.Uint32
+ state *scanState
+ processedAlbumAnnotations map[string]bool // Track processed album annotation reassignments
+ annotationMutex sync.RWMutex // Protects processedAlbumAnnotations
}
func createPhaseMissingTracks(ctx context.Context, state *scanState, ds model.DataStore) *phaseMissingTracks {
- return &phaseMissingTracks{ctx: ctx, ds: ds, state: state}
+ return &phaseMissingTracks{
+ ctx: ctx,
+ ds: ds,
+ state: state,
+ processedAlbumAnnotations: make(map[string]bool),
+ }
}
func (p *phaseMissingTracks) description() string {
@@ -52,20 +60,15 @@ func (p *phaseMissingTracks) producer() ppl.Producer[*missingTracks] {
func (p *phaseMissingTracks) produce(put func(tracks *missingTracks)) error {
count := 0
var putIfMatched = func(mt missingTracks) {
- if mt.pid != "" && len(mt.matched) > 0 {
- log.Trace(p.ctx, "Scanner: Found missing and matching tracks", "pid", mt.pid, "missing", len(mt.missing), "matched", len(mt.matched), "lib", mt.lib.Name)
+ if mt.pid != "" && len(mt.missing) > 0 {
+ log.Trace(p.ctx, "Scanner: Found missing tracks", "pid", mt.pid, "missing", "title", mt.missing[0].Title,
+ len(mt.missing), "matched", len(mt.matched), "lib", mt.lib.Name,
+ )
count++
put(&mt)
}
}
- libs, err := p.ds.Library(p.ctx).GetAll()
- if err != nil {
- return fmt.Errorf("loading libraries: %w", err)
- }
- for _, lib := range libs {
- if lib.LastScanStartedAt.IsZero() {
- continue
- }
+ for _, lib := range p.state.libraries {
log.Debug(p.ctx, "Scanner: Checking missing tracks", "libraryId", lib.ID, "libraryName", lib.Name)
cursor, err := p.ds.MediaFile(p.ctx).GetMissingAndMatching(lib.ID)
if err != nil {
@@ -104,10 +107,13 @@ func (p *phaseMissingTracks) produce(put func(tracks *missingTracks)) error {
func (p *phaseMissingTracks) stages() []ppl.Stage[*missingTracks] {
return []ppl.Stage[*missingTracks]{
ppl.NewStage(p.processMissingTracks, ppl.Name("process missing tracks")),
+ ppl.NewStage(p.processCrossLibraryMoves, ppl.Name("process cross-library moves")),
}
}
func (p *phaseMissingTracks) processMissingTracks(in *missingTracks) (*missingTracks, error) {
+ hasMatches := false
+
for _, ms := range in.missing {
var exactMatch model.MediaFile
var equivalentMatch model.MediaFile
@@ -132,6 +138,7 @@ func (p *phaseMissingTracks) processMissingTracks(in *missingTracks) (*missingTr
return nil, err
}
p.totalMatched.Add(1)
+ hasMatches = true
continue
}
@@ -145,6 +152,7 @@ func (p *phaseMissingTracks) processMissingTracks(in *missingTracks) (*missingTr
return nil, err
}
p.totalMatched.Add(1)
+ hasMatches = true
continue
}
@@ -157,23 +165,141 @@ func (p *phaseMissingTracks) processMissingTracks(in *missingTracks) (*missingTr
return nil, err
}
p.totalMatched.Add(1)
+ hasMatches = true
}
}
+
+ // If any matches were found in this missingTracks group, return nil
+ // This signals the next stage to skip processing this group
+ if hasMatches {
+ return nil, nil
+ }
+
+ // If no matches found, pass through to next stage
return in, nil
}
-func (p *phaseMissingTracks) moveMatched(mt, ms model.MediaFile) error {
+// processCrossLibraryMoves processes files that weren't matched within their library
+// and attempts to find matches in other libraries
+func (p *phaseMissingTracks) processCrossLibraryMoves(in *missingTracks) (*missingTracks, error) {
+ // Skip if input is nil (meaning previous stage found matches)
+ if in == nil {
+ return nil, nil
+ }
+
+ log.Debug(p.ctx, "Scanner: Processing cross-library moves", "pid", in.pid, "missing", len(in.missing), "lib", in.lib.Name)
+
+ for _, missing := range in.missing {
+ found, err := p.findCrossLibraryMatch(missing)
+ if err != nil {
+ log.Error(p.ctx, "Scanner: Error searching for cross-library matches", "missing", missing.Path, "lib", in.lib.Name, err)
+ continue
+ }
+
+ if found.ID != "" {
+ log.Debug(p.ctx, "Scanner: Found cross-library moved track", "missing", missing.Path, "movedTo", found.Path, "fromLib", in.lib.Name, "toLib", found.LibraryName)
+ err := p.moveMatched(found, missing)
+ if err != nil {
+ log.Error(p.ctx, "Scanner: Error moving cross-library track", "missing", missing.Path, "movedTo", found.Path, err)
+ continue
+ }
+ p.totalMatched.Add(1)
+ }
+ }
+
+ return in, nil
+}
+
+// findCrossLibraryMatch searches for a missing file in other libraries using two-tier matching
+func (p *phaseMissingTracks) findCrossLibraryMatch(missing model.MediaFile) (model.MediaFile, error) {
+ // First tier: Search by MusicBrainz Track ID if available
+ if missing.MbzReleaseTrackID != "" {
+ matches, err := p.ds.MediaFile(p.ctx).FindRecentFilesByMBZTrackID(missing, missing.CreatedAt)
+ if err != nil {
+ log.Error(p.ctx, "Scanner: Error searching for recent files by MBZ Track ID", "mbzTrackID", missing.MbzReleaseTrackID, err)
+ } else {
+ // Apply the same matching logic as within-library matching
+ for _, match := range matches {
+ if missing.Equals(match) {
+ return match, nil // Exact match found
+ }
+ }
+
+ // If only one match and it's equivalent, use it
+ if len(matches) == 1 && missing.IsEquivalent(matches[0]) {
+ return matches[0], nil
+ }
+ }
+ }
+
+ // Second tier: Search by intrinsic properties (title, size, suffix, etc.)
+ matches, err := p.ds.MediaFile(p.ctx).FindRecentFilesByProperties(missing, missing.CreatedAt)
+ if err != nil {
+ log.Error(p.ctx, "Scanner: Error searching for recent files by properties", "missing", missing.Path, err)
+ return model.MediaFile{}, err
+ }
+
+ // Apply the same matching logic as within-library matching
+ for _, match := range matches {
+ if missing.Equals(match) {
+ return match, nil // Exact match found
+ }
+ }
+
+ // If only one match and it's equivalent, use it
+ if len(matches) == 1 && missing.IsEquivalent(matches[0]) {
+ return matches[0], nil
+ }
+
+ return model.MediaFile{}, nil
+}
+
+func (p *phaseMissingTracks) moveMatched(target, missing model.MediaFile) error {
return p.ds.WithTx(func(tx model.DataStore) error {
- discardedID := mt.ID
- mt.ID = ms.ID
- err := tx.MediaFile(p.ctx).Put(&mt)
+ discardedID := target.ID
+ oldAlbumID := missing.AlbumID
+ newAlbumID := target.AlbumID
+
+ // Update the target media file with the missing file's ID. This effectively "moves" the track
+ // to the new location while keeping its annotations and references intact.
+ target.ID = missing.ID
+ err := tx.MediaFile(p.ctx).Put(&target)
if err != nil {
return fmt.Errorf("update matched track: %w", err)
}
+
+ // Discard the new mediafile row (the one that was moved to)
err = tx.MediaFile(p.ctx).Delete(discardedID)
if err != nil {
return fmt.Errorf("delete discarded track: %w", err)
}
+
+ // Handle album annotation reassignment if AlbumID changed
+ if oldAlbumID != newAlbumID {
+ // Use newAlbumID as key since we only care about avoiding duplicate reassignments to the same target
+ p.annotationMutex.RLock()
+ alreadyProcessed := p.processedAlbumAnnotations[newAlbumID]
+ p.annotationMutex.RUnlock()
+
+ if !alreadyProcessed {
+ p.annotationMutex.Lock()
+ // Double-check pattern to avoid race conditions
+ if !p.processedAlbumAnnotations[newAlbumID] {
+ // Reassign direct album annotations (starred, rating)
+ log.Debug(p.ctx, "Scanner: Reassigning album annotations", "from", oldAlbumID, "to", newAlbumID)
+ if err := tx.Album(p.ctx).ReassignAnnotation(oldAlbumID, newAlbumID); err != nil {
+ log.Warn(p.ctx, "Scanner: Could not reassign album annotations", "from", oldAlbumID, "to", newAlbumID, err)
+ }
+
+ // Note: RefreshPlayCounts will be called in later phases, so we don't need to call it here
+ p.processedAlbumAnnotations[newAlbumID] = true
+ }
+ p.annotationMutex.Unlock()
+ } else {
+ log.Trace(p.ctx, "Scanner: Skipping album annotation reassignment", "from", oldAlbumID, "to", newAlbumID)
+ }
+ }
+
p.state.changesDetected.Store(true)
return nil
})
diff --git a/scanner/phase_2_missing_tracks_test.go b/scanner/phase_2_missing_tracks_test.go
index 5dd6cc679..e709004c9 100644
--- a/scanner/phase_2_missing_tracks_test.go
+++ b/scanner/phase_2_missing_tracks_test.go
@@ -28,7 +28,9 @@ var _ = Describe("phaseMissingTracks", func() {
lr = &tests.MockLibraryRepo{}
lr.SetData(model.Libraries{{ID: 1, LastScanStartedAt: time.Date(2021, 1, 1, 0, 0, 0, 0, time.UTC)}})
ds = &tests.MockDataStore{MockedMediaFile: mr, MockedLibrary: lr}
- state = &scanState{}
+ state = &scanState{
+ libraries: model.Libraries{{ID: 1, LastScanStartedAt: time.Date(2021, 1, 1, 0, 0, 0, 0, time.UTC)}},
+ }
phase = createPhaseMissingTracks(ctx, state, ds)
})
@@ -68,12 +70,31 @@ var _ = Describe("phaseMissingTracks", func() {
err := phase.produce(put)
Expect(err).ToNot(HaveOccurred())
- Expect(produced).To(HaveLen(1))
- Expect(produced[0].pid).To(Equal("A"))
- Expect(produced[0].missing).To(HaveLen(1))
- Expect(produced[0].matched).To(HaveLen(1))
+ Expect(produced).To(HaveLen(2))
+ // PID A should have both missing and matched tracks
+ var pidA *missingTracks
+ for _, p := range produced {
+ if p.pid == "A" {
+ pidA = p
+ break
+ }
+ }
+ Expect(pidA).ToNot(BeNil())
+ Expect(pidA.missing).To(HaveLen(1))
+ Expect(pidA.matched).To(HaveLen(1))
+ // PID B should have only missing tracks
+ var pidB *missingTracks
+ for _, p := range produced {
+ if p.pid == "B" {
+ pidB = p
+ break
+ }
+ }
+ Expect(pidB).ToNot(BeNil())
+ Expect(pidB.missing).To(HaveLen(1))
+ Expect(pidB.matched).To(HaveLen(0))
})
- It("should not call put if there are no matches for any missing tracks", func() {
+ It("should call put for any missing tracks even without matches", func() {
mr.SetData(model.MediaFiles{
{ID: "1", PID: "A", Missing: true, LibraryID: 1},
{ID: "2", PID: "B", Missing: true, LibraryID: 1},
@@ -82,7 +103,22 @@ var _ = Describe("phaseMissingTracks", func() {
err := phase.produce(put)
Expect(err).ToNot(HaveOccurred())
- Expect(produced).To(BeZero())
+ Expect(produced).To(HaveLen(2))
+ // Both PID A and PID B should be produced even without matches
+ var pidA, pidB *missingTracks
+ for _, p := range produced {
+ if p.pid == "A" {
+ pidA = p
+ } else if p.pid == "B" {
+ pidB = p
+ }
+ }
+ Expect(pidA).ToNot(BeNil())
+ Expect(pidA.missing).To(HaveLen(1))
+ Expect(pidA.matched).To(HaveLen(0))
+ Expect(pidB).ToNot(BeNil())
+ Expect(pidB.missing).To(HaveLen(1))
+ Expect(pidB.matched).To(HaveLen(0))
})
})
})
@@ -286,4 +322,448 @@ var _ = Describe("phaseMissingTracks", func() {
})
})
})
+
+ Describe("processCrossLibraryMoves", func() {
+ It("should skip processing if input is nil", func() {
+ result, err := phase.processCrossLibraryMoves(nil)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(result).To(BeNil())
+ })
+
+ It("should process cross-library moves using MusicBrainz Track ID", func() {
+ scanStartTime := time.Now().Add(-1 * time.Hour)
+ missingTrack := model.MediaFile{
+ ID: "missing1",
+ LibraryID: 1,
+ MbzReleaseTrackID: "mbz-track-123",
+ Title: "Test Track",
+ Size: 1000,
+ Suffix: "mp3",
+ Path: "/lib1/track.mp3",
+ Missing: true,
+ CreatedAt: scanStartTime.Add(-30 * time.Minute),
+ }
+
+ movedTrack := model.MediaFile{
+ ID: "moved1",
+ LibraryID: 2,
+ MbzReleaseTrackID: "mbz-track-123",
+ Title: "Test Track",
+ Size: 1000,
+ Suffix: "mp3",
+ Path: "/lib2/track.mp3",
+ Missing: false,
+ CreatedAt: scanStartTime.Add(-10 * time.Minute),
+ }
+
+ _ = ds.MediaFile(ctx).Put(&missingTrack)
+ _ = ds.MediaFile(ctx).Put(&movedTrack)
+
+ in := &missingTracks{
+ lib: model.Library{ID: 1, Name: "Library 1"},
+ missing: []model.MediaFile{missingTrack},
+ }
+
+ result, err := phase.processCrossLibraryMoves(in)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(result).To(Equal(in))
+ Expect(phase.totalMatched.Load()).To(Equal(uint32(1)))
+ Expect(state.changesDetected.Load()).To(BeTrue())
+
+ // Verify the move was performed
+ updatedTrack, _ := ds.MediaFile(ctx).Get("missing1")
+ Expect(updatedTrack.Path).To(Equal("/lib2/track.mp3"))
+ Expect(updatedTrack.LibraryID).To(Equal(2))
+ })
+
+ It("should fall back to intrinsic properties when MBZ Track ID is empty", func() {
+ scanStartTime := time.Now().Add(-1 * time.Hour)
+ missingTrack := model.MediaFile{
+ ID: "missing2",
+ LibraryID: 1,
+ MbzReleaseTrackID: "",
+ Title: "Test Track 2",
+ Size: 2000,
+ Suffix: "flac",
+ DiscNumber: 1,
+ TrackNumber: 1,
+ Album: "Test Album",
+ Path: "/lib1/track2.flac",
+ Missing: true,
+ CreatedAt: scanStartTime.Add(-30 * time.Minute),
+ }
+
+ movedTrack := model.MediaFile{
+ ID: "moved2",
+ LibraryID: 2,
+ MbzReleaseTrackID: "",
+ Title: "Test Track 2",
+ Size: 2000,
+ Suffix: "flac",
+ DiscNumber: 1,
+ TrackNumber: 1,
+ Album: "Test Album",
+ Path: "/lib2/track2.flac",
+ Missing: false,
+ CreatedAt: scanStartTime.Add(-10 * time.Minute),
+ }
+
+ _ = ds.MediaFile(ctx).Put(&missingTrack)
+ _ = ds.MediaFile(ctx).Put(&movedTrack)
+
+ in := &missingTracks{
+ lib: model.Library{ID: 1, Name: "Library 1"},
+ missing: []model.MediaFile{missingTrack},
+ }
+
+ result, err := phase.processCrossLibraryMoves(in)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(result).To(Equal(in))
+ Expect(phase.totalMatched.Load()).To(Equal(uint32(1)))
+ Expect(state.changesDetected.Load()).To(BeTrue())
+
+ // Verify the move was performed
+ updatedTrack, _ := ds.MediaFile(ctx).Get("missing2")
+ Expect(updatedTrack.Path).To(Equal("/lib2/track2.flac"))
+ Expect(updatedTrack.LibraryID).To(Equal(2))
+ })
+
+ It("should not match files in the same library", func() {
+ scanStartTime := time.Now().Add(-1 * time.Hour)
+ missingTrack := model.MediaFile{
+ ID: "missing3",
+ LibraryID: 1,
+ MbzReleaseTrackID: "mbz-track-456",
+ Title: "Test Track 3",
+ Size: 3000,
+ Suffix: "mp3",
+ Path: "/lib1/track3.mp3",
+ Missing: true,
+ CreatedAt: scanStartTime.Add(-30 * time.Minute),
+ }
+
+ sameLibTrack := model.MediaFile{
+ ID: "same1",
+ LibraryID: 1, // Same library
+ MbzReleaseTrackID: "mbz-track-456",
+ Title: "Test Track 3",
+ Size: 3000,
+ Suffix: "mp3",
+ Path: "/lib1/other/track3.mp3",
+ Missing: false,
+ CreatedAt: scanStartTime.Add(-10 * time.Minute),
+ }
+
+ _ = ds.MediaFile(ctx).Put(&missingTrack)
+ _ = ds.MediaFile(ctx).Put(&sameLibTrack)
+
+ in := &missingTracks{
+ lib: model.Library{ID: 1, Name: "Library 1"},
+ missing: []model.MediaFile{missingTrack},
+ }
+
+ result, err := phase.processCrossLibraryMoves(in)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(result).To(Equal(in))
+ Expect(phase.totalMatched.Load()).To(Equal(uint32(0)))
+ Expect(state.changesDetected.Load()).To(BeFalse())
+ })
+
+ It("should prioritize MBZ Track ID over intrinsic properties", func() {
+ scanStartTime := time.Now().Add(-1 * time.Hour)
+ missingTrack := model.MediaFile{
+ ID: "missing4",
+ LibraryID: 1,
+ MbzReleaseTrackID: "mbz-track-789",
+ Title: "Test Track 4",
+ Size: 4000,
+ Suffix: "mp3",
+ Path: "/lib1/track4.mp3",
+ Missing: true,
+ CreatedAt: scanStartTime.Add(-30 * time.Minute),
+ }
+
+ // Track with same MBZ ID
+ mbzTrack := model.MediaFile{
+ ID: "mbz1",
+ LibraryID: 2,
+ MbzReleaseTrackID: "mbz-track-789",
+ Title: "Test Track 4",
+ Size: 4000,
+ Suffix: "mp3",
+ Path: "/lib2/track4.mp3",
+ Missing: false,
+ CreatedAt: scanStartTime.Add(-10 * time.Minute),
+ }
+
+ // Track with same intrinsic properties but no MBZ ID
+ intrinsicTrack := model.MediaFile{
+ ID: "intrinsic1",
+ LibraryID: 3,
+ MbzReleaseTrackID: "",
+ Title: "Test Track 4",
+ Size: 4000,
+ Suffix: "mp3",
+ DiscNumber: 1,
+ TrackNumber: 1,
+ Album: "Test Album",
+ Path: "/lib3/track4.mp3",
+ Missing: false,
+ CreatedAt: scanStartTime.Add(-5 * time.Minute),
+ }
+
+ _ = ds.MediaFile(ctx).Put(&missingTrack)
+ _ = ds.MediaFile(ctx).Put(&mbzTrack)
+ _ = ds.MediaFile(ctx).Put(&intrinsicTrack)
+
+ in := &missingTracks{
+ lib: model.Library{ID: 1, Name: "Library 1"},
+ missing: []model.MediaFile{missingTrack},
+ }
+
+ result, err := phase.processCrossLibraryMoves(in)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(result).To(Equal(in))
+ Expect(phase.totalMatched.Load()).To(Equal(uint32(1)))
+ Expect(state.changesDetected.Load()).To(BeTrue())
+
+ // Verify the MBZ track was chosen (not the intrinsic one)
+ updatedTrack, _ := ds.MediaFile(ctx).Get("missing4")
+ Expect(updatedTrack.Path).To(Equal("/lib2/track4.mp3"))
+ Expect(updatedTrack.LibraryID).To(Equal(2))
+ })
+
+ It("should handle equivalent matches correctly", func() {
+ scanStartTime := time.Now().Add(-1 * time.Hour)
+ missingTrack := model.MediaFile{
+ ID: "missing5",
+ LibraryID: 1,
+ MbzReleaseTrackID: "",
+ Title: "Test Track 5",
+ Size: 5000,
+ Suffix: "mp3",
+ Path: "/lib1/path/track5.mp3",
+ Missing: true,
+ CreatedAt: scanStartTime.Add(-30 * time.Minute),
+ }
+
+ // Equivalent match (same filename, different directory)
+ equivalentTrack := model.MediaFile{
+ ID: "equiv1",
+ LibraryID: 2,
+ MbzReleaseTrackID: "",
+ Title: "Test Track 5",
+ Size: 5000,
+ Suffix: "mp3",
+ Path: "/lib2/different/track5.mp3",
+ Missing: false,
+ CreatedAt: scanStartTime.Add(-10 * time.Minute),
+ }
+
+ _ = ds.MediaFile(ctx).Put(&missingTrack)
+ _ = ds.MediaFile(ctx).Put(&equivalentTrack)
+
+ in := &missingTracks{
+ lib: model.Library{ID: 1, Name: "Library 1"},
+ missing: []model.MediaFile{missingTrack},
+ }
+
+ result, err := phase.processCrossLibraryMoves(in)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(result).To(Equal(in))
+ Expect(phase.totalMatched.Load()).To(Equal(uint32(1)))
+ Expect(state.changesDetected.Load()).To(BeTrue())
+
+ // Verify the equivalent match was accepted
+ updatedTrack, _ := ds.MediaFile(ctx).Get("missing5")
+ Expect(updatedTrack.Path).To(Equal("/lib2/different/track5.mp3"))
+ Expect(updatedTrack.LibraryID).To(Equal(2))
+ })
+
+ It("should skip matching when multiple matches are found but none are exact", func() {
+ scanStartTime := time.Now().Add(-1 * time.Hour)
+ missingTrack := model.MediaFile{
+ ID: "missing6",
+ LibraryID: 1,
+ MbzReleaseTrackID: "",
+ Title: "Test Track 6",
+ Size: 6000,
+ Suffix: "mp3",
+ DiscNumber: 1,
+ TrackNumber: 1,
+ Album: "Test Album",
+ Path: "/lib1/track6.mp3",
+ Missing: true,
+ CreatedAt: scanStartTime.Add(-30 * time.Minute),
+ }
+
+ // Multiple matches with different metadata (not exact matches)
+ match1 := model.MediaFile{
+ ID: "match1",
+ LibraryID: 2,
+ MbzReleaseTrackID: "",
+ Title: "Test Track 6",
+ Size: 6000,
+ Suffix: "mp3",
+ DiscNumber: 1,
+ TrackNumber: 1,
+ Album: "Test Album",
+ Path: "/lib2/different_track.mp3",
+ Artist: "Different Artist", // This makes it non-exact
+ Missing: false,
+ CreatedAt: scanStartTime.Add(-10 * time.Minute),
+ }
+
+ match2 := model.MediaFile{
+ ID: "match2",
+ LibraryID: 3,
+ MbzReleaseTrackID: "",
+ Title: "Test Track 6",
+ Size: 6000,
+ Suffix: "mp3",
+ DiscNumber: 1,
+ TrackNumber: 1,
+ Album: "Test Album",
+ Path: "/lib3/another_track.mp3",
+ Artist: "Another Artist", // This makes it non-exact
+ Missing: false,
+ CreatedAt: scanStartTime.Add(-5 * time.Minute),
+ }
+
+ _ = ds.MediaFile(ctx).Put(&missingTrack)
+ _ = ds.MediaFile(ctx).Put(&match1)
+ _ = ds.MediaFile(ctx).Put(&match2)
+
+ in := &missingTracks{
+ lib: model.Library{ID: 1, Name: "Library 1"},
+ missing: []model.MediaFile{missingTrack},
+ }
+
+ result, err := phase.processCrossLibraryMoves(in)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(result).To(Equal(in))
+ Expect(phase.totalMatched.Load()).To(Equal(uint32(0)))
+ Expect(state.changesDetected.Load()).To(BeFalse())
+
+ // Verify no move was performed
+ unchangedTrack, _ := ds.MediaFile(ctx).Get("missing6")
+ Expect(unchangedTrack.Path).To(Equal("/lib1/track6.mp3"))
+ Expect(unchangedTrack.LibraryID).To(Equal(1))
+ })
+
+ It("should handle errors gracefully", func() {
+ // Set up mock to return error
+ mr.Err = true
+
+ missingTrack := model.MediaFile{
+ ID: "missing7",
+ LibraryID: 1,
+ MbzReleaseTrackID: "mbz-track-error",
+ Title: "Test Track 7",
+ Size: 7000,
+ Suffix: "mp3",
+ Path: "/lib1/track7.mp3",
+ Missing: true,
+ CreatedAt: time.Now().Add(-30 * time.Minute),
+ }
+
+ in := &missingTracks{
+ lib: model.Library{ID: 1, Name: "Library 1"},
+ missing: []model.MediaFile{missingTrack},
+ }
+
+ // Should not fail completely, just skip the problematic file
+ result, err := phase.processCrossLibraryMoves(in)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(result).To(Equal(in))
+ Expect(phase.totalMatched.Load()).To(Equal(uint32(0)))
+ Expect(state.changesDetected.Load()).To(BeFalse())
+ })
+ })
+
+ Describe("Album Annotation Reassignment", func() {
+ var (
+ albumRepo *tests.MockAlbumRepo
+ missingTrack model.MediaFile
+ matchedTrack model.MediaFile
+ oldAlbumID string
+ newAlbumID string
+ )
+
+ BeforeEach(func() {
+ albumRepo = ds.Album(ctx).(*tests.MockAlbumRepo)
+ albumRepo.ReassignAnnotationCalls = make(map[string]string)
+
+ oldAlbumID = "old-album-id"
+ newAlbumID = "new-album-id"
+
+ missingTrack = model.MediaFile{
+ ID: "missing-track-id",
+ PID: "same-pid",
+ Path: "old/path.mp3",
+ AlbumID: oldAlbumID,
+ LibraryID: 1,
+ Missing: true,
+ Annotations: model.Annotations{
+ PlayCount: 5,
+ Rating: 4,
+ Starred: true,
+ },
+ }
+
+ matchedTrack = model.MediaFile{
+ ID: "matched-track-id",
+ PID: "same-pid",
+ Path: "new/path.mp3",
+ AlbumID: newAlbumID,
+ LibraryID: 2, // Different library
+ Missing: false,
+ Annotations: model.Annotations{
+ PlayCount: 2,
+ Rating: 3,
+ Starred: false,
+ },
+ }
+
+ // Store both tracks in the database
+ _ = ds.MediaFile(ctx).Put(&missingTrack)
+ _ = ds.MediaFile(ctx).Put(&matchedTrack)
+ })
+
+ When("album ID changes during cross-library move", func() {
+ It("should reassign album annotations when AlbumID changes", func() {
+ err := phase.moveMatched(matchedTrack, missingTrack)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify that ReassignAnnotation was called
+ Expect(albumRepo.ReassignAnnotationCalls).To(HaveKeyWithValue(oldAlbumID, newAlbumID))
+ })
+
+ It("should not reassign annotations when AlbumID is the same", func() {
+ missingTrack.AlbumID = newAlbumID // Same album
+
+ err := phase.moveMatched(matchedTrack, missingTrack)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify that ReassignAnnotation was NOT called
+ Expect(albumRepo.ReassignAnnotationCalls).To(BeEmpty())
+ })
+ })
+
+ When("error handling", func() {
+ It("should handle ReassignAnnotation errors gracefully", func() {
+ // Make the album repo return an error
+ albumRepo.SetError(true)
+
+ // The move should still succeed even if annotation reassignment fails
+ err := phase.moveMatched(matchedTrack, missingTrack)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify that the track was still moved (ID should be updated)
+ movedTrack, err := ds.MediaFile(ctx).Get(missingTrack.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(movedTrack.Path).To(Equal(matchedTrack.Path))
+ })
+ })
+ })
})
diff --git a/scanner/phase_3_refresh_albums.go b/scanner/phase_3_refresh_albums.go
index f51aa8f4b..33e0fed01 100644
--- a/scanner/phase_3_refresh_albums.go
+++ b/scanner/phase_3_refresh_albums.go
@@ -27,14 +27,13 @@ import (
type phaseRefreshAlbums struct {
ds model.DataStore
ctx context.Context
- libs model.Libraries
refreshed atomic.Uint32
skipped atomic.Uint32
state *scanState
}
-func createPhaseRefreshAlbums(ctx context.Context, state *scanState, ds model.DataStore, libs model.Libraries) *phaseRefreshAlbums {
- return &phaseRefreshAlbums{ctx: ctx, ds: ds, libs: libs, state: state}
+func createPhaseRefreshAlbums(ctx context.Context, state *scanState, ds model.DataStore) *phaseRefreshAlbums {
+ return &phaseRefreshAlbums{ctx: ctx, ds: ds, state: state}
}
func (p *phaseRefreshAlbums) description() string {
@@ -47,7 +46,7 @@ func (p *phaseRefreshAlbums) producer() ppl.Producer[*model.Album] {
func (p *phaseRefreshAlbums) produce(put func(album *model.Album)) error {
count := 0
- for _, lib := range p.libs {
+ for _, lib := range p.state.libraries {
cursor, err := p.ds.Album(p.ctx).GetTouchedAlbums(lib.ID)
if err != nil {
return fmt.Errorf("loading touched albums: %w", err)
diff --git a/scanner/phase_3_refresh_albums_test.go b/scanner/phase_3_refresh_albums_test.go
index dea2556f0..1f0baf428 100644
--- a/scanner/phase_3_refresh_albums_test.go
+++ b/scanner/phase_3_refresh_albums_test.go
@@ -32,8 +32,8 @@ var _ = Describe("phaseRefreshAlbums", func() {
{ID: 1, Name: "Library 1"},
{ID: 2, Name: "Library 2"},
}
- state = &scanState{}
- phase = createPhaseRefreshAlbums(ctx, state, ds, libs)
+ state = &scanState{libraries: libs}
+ phase = createPhaseRefreshAlbums(ctx, state, ds)
})
Describe("description", func() {
diff --git a/scanner/scanner.go b/scanner/scanner.go
index f08dec311..20f3f5da8 100644
--- a/scanner/scanner.go
+++ b/scanner/scanner.go
@@ -3,6 +3,8 @@ package scanner
import (
"context"
"fmt"
+ "maps"
+ "slices"
"sync/atomic"
"time"
@@ -11,18 +13,17 @@ import (
"github.com/navidrome/navidrome/consts"
"github.com/navidrome/navidrome/core"
"github.com/navidrome/navidrome/core/artwork"
- "github.com/navidrome/navidrome/core/metrics"
"github.com/navidrome/navidrome/db"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
- "github.com/navidrome/navidrome/utils/chain"
+ "github.com/navidrome/navidrome/utils/run"
+ "github.com/navidrome/navidrome/utils/slice"
)
type scannerImpl struct {
- ds model.DataStore
- cw artwork.CacheWarmer
- pls core.Playlists
- metrics metrics.Metrics
+ ds model.DataStore
+ cw artwork.CacheWarmer
+ pls core.Playlists
}
// scanState holds the state of an in-progress scan, to be passed to the various phases
@@ -30,6 +31,8 @@ type scanState struct {
progress chan<- *ProgressInfo
fullScan bool
changesDetected atomic.Bool
+ libraries model.Libraries // Store libraries list for consistency across phases
+ targets map[int][]string // Optional: map[libraryID][]folderPaths for selective scans
}
func (s *scanState) sendProgress(info *ProgressInfo) {
@@ -38,6 +41,10 @@ func (s *scanState) sendProgress(info *ProgressInfo) {
}
}
+func (s *scanState) isSelectiveScan() bool {
+ return len(s.targets) > 0
+}
+
func (s *scanState) sendWarning(msg string) {
s.sendProgress(&ProgressInfo{Warning: msg})
}
@@ -46,48 +53,97 @@ func (s *scanState) sendError(err error) {
s.sendProgress(&ProgressInfo{Error: err.Error()})
}
-func (s *scannerImpl) scanAll(ctx context.Context, fullScan bool, progress chan<- *ProgressInfo) {
- state := scanState{progress: progress, fullScan: fullScan}
- libs, err := s.ds.Library(ctx).GetAll()
+func (s *scannerImpl) scanFolders(ctx context.Context, fullScan bool, targets []model.ScanTarget, progress chan<- *ProgressInfo) {
+ startTime := time.Now()
+
+ state := scanState{
+ progress: progress,
+ fullScan: fullScan,
+ changesDetected: atomic.Bool{},
+ }
+
+ // Set changesDetected to true for full scans to ensure all maintenance operations run
+ if fullScan {
+ state.changesDetected.Store(true)
+ }
+
+ // Get libraries and optionally filter by targets
+ allLibs, err := s.ds.Library(ctx).GetAll()
if err != nil {
state.sendWarning(fmt.Sprintf("getting libraries: %s", err))
return
}
- startTime := time.Now()
- log.Info(ctx, "Scanner: Starting scan", "fullScan", state.fullScan, "numLibraries", len(libs))
+ if len(targets) > 0 {
+ // Selective scan: filter libraries and build targets map
+ state.targets = make(map[int][]string)
+
+ for _, target := range targets {
+ folderPath := target.FolderPath
+ if folderPath == "" {
+ folderPath = "."
+ }
+ state.targets[target.LibraryID] = append(state.targets[target.LibraryID], folderPath)
+ }
+
+ // Filter libraries to only those in targets
+ state.libraries = slice.Filter(allLibs, func(lib model.Library) bool {
+ return len(state.targets[lib.ID]) > 0
+ })
+
+ log.Info(ctx, "Scanner: Starting selective scan", "fullScan", state.fullScan, "numLibraries", len(state.libraries), "numTargets", len(targets))
+ } else {
+ // Full library scan
+ state.libraries = allLibs
+ log.Info(ctx, "Scanner: Starting scan", "fullScan", state.fullScan, "numLibraries", len(state.libraries))
+ }
// Store scan type and start time
scanType := "quick"
if state.fullScan {
scanType = "full"
}
+ if state.isSelectiveScan() {
+ scanType += "-selective"
+ }
_ = s.ds.Property(ctx).Put(consts.LastScanTypeKey, scanType)
_ = s.ds.Property(ctx).Put(consts.LastScanStartTimeKey, startTime.Format(time.RFC3339))
// if there was a full scan in progress, force a full scan
if !state.fullScan {
- for _, lib := range libs {
+ for _, lib := range state.libraries {
if lib.FullScanInProgress {
log.Info(ctx, "Scanner: Interrupted full scan detected", "lib", lib.Name)
state.fullScan = true
- _ = s.ds.Property(ctx).Put(consts.LastScanTypeKey, "full")
+ if state.isSelectiveScan() {
+ _ = s.ds.Property(ctx).Put(consts.LastScanTypeKey, "full-selective")
+ } else {
+ _ = s.ds.Property(ctx).Put(consts.LastScanTypeKey, "full")
+ }
break
}
}
}
- err = chain.RunSequentially(
+ // Prepare libraries for scanning (initialize LastScanStartedAt if needed)
+ err = s.prepareLibrariesForScan(ctx, &state)
+ if err != nil {
+ log.Error(ctx, "Scanner: Error preparing libraries for scan", err)
+ state.sendError(err)
+ return
+ }
+
+ err = run.Sequentially(
// Phase 1: Scan all libraries and import new/updated files
- runPhase[*folderEntry](ctx, 1, createPhaseFolders(ctx, &state, s.ds, s.cw, libs)),
+ runPhase[*folderEntry](ctx, 1, createPhaseFolders(ctx, &state, s.ds, s.cw)),
// Phase 2: Process missing files, checking for moves
runPhase[*missingTracks](ctx, 2, createPhaseMissingTracks(ctx, &state, s.ds)),
// Phases 3 and 4 can be run in parallel
- chain.RunParallel(
+ run.Parallel(
// Phase 3: Refresh all new/changed albums and update artists
- runPhase[*model.Album](ctx, 3, createPhaseRefreshAlbums(ctx, &state, s.ds, libs)),
+ runPhase[*model.Album](ctx, 3, createPhaseRefreshAlbums(ctx, &state, s.ds)),
// Phase 4: Import/update playlists
runPhase[*model.Folder](ctx, 4, createPhasePlaylists(ctx, &state, s.ds, s.pls, s.cw)),
@@ -102,7 +158,7 @@ func (s *scannerImpl) scanAll(ctx context.Context, fullScan bool, progress chan<
s.runRefreshStats(ctx, &state),
// Update last_scan_completed_at for all libraries
- s.runUpdateLibraries(ctx, libs),
+ s.runUpdateLibraries(ctx, &state),
// Optimize DB
s.runOptimize(ctx),
@@ -111,7 +167,6 @@ func (s *scannerImpl) scanAll(ctx context.Context, fullScan bool, progress chan<
log.Error(ctx, "Scanner: Finished with error", "duration", time.Since(startTime), err)
_ = s.ds.Property(ctx).Put(consts.LastScanErrorKey, err.Error())
state.sendError(err)
- s.metrics.WriteAfterScanMetrics(ctx, false)
return
}
@@ -121,8 +176,53 @@ func (s *scannerImpl) scanAll(ctx context.Context, fullScan bool, progress chan<
state.sendProgress(&ProgressInfo{ChangesDetected: true})
}
- s.metrics.WriteAfterScanMetrics(ctx, err == nil)
- log.Info(ctx, "Scanner: Finished scanning all libraries", "duration", time.Since(startTime))
+ if state.isSelectiveScan() {
+ log.Info(ctx, "Scanner: Finished scanning selected folders", "duration", time.Since(startTime), "numTargets", len(targets))
+ } else {
+ log.Info(ctx, "Scanner: Finished scanning all libraries", "duration", time.Since(startTime))
+ }
+}
+
+// prepareLibrariesForScan initializes the scan for all libraries in the state.
+// It calls ScanBegin for libraries that haven't started scanning yet (LastScanStartedAt is zero),
+// reloads them to get the updated state, and filters out any libraries that fail to initialize.
+func (s *scannerImpl) prepareLibrariesForScan(ctx context.Context, state *scanState) error {
+ var successfulLibs []model.Library
+
+ for _, lib := range state.libraries {
+ if lib.LastScanStartedAt.IsZero() {
+ // This is a new scan - mark it as started
+ err := s.ds.Library(ctx).ScanBegin(lib.ID, state.fullScan)
+ if err != nil {
+ log.Error(ctx, "Scanner: Error marking scan start", "lib", lib.Name, err)
+ state.sendWarning(err.Error())
+ continue
+ }
+
+ // Reload library to get updated state (timestamps, etc.)
+ reloadedLib, err := s.ds.Library(ctx).Get(lib.ID)
+ if err != nil {
+ log.Error(ctx, "Scanner: Error reloading library", "lib", lib.Name, err)
+ state.sendWarning(err.Error())
+ continue
+ }
+ lib = *reloadedLib
+ } else {
+ // This is a resumed scan
+ log.Debug(ctx, "Scanner: Resuming previous scan", "lib", lib.Name,
+ "lastScanStartedAt", lib.LastScanStartedAt, "fullScan", lib.FullScanInProgress)
+ }
+
+ successfulLibs = append(successfulLibs, lib)
+ }
+
+ if len(successfulLibs) == 0 {
+ return fmt.Errorf("no libraries available for scanning")
+ }
+
+ // Update state with only successfully initialized libraries
+ state.libraries = successfulLibs
+ return nil
}
func (s *scannerImpl) runGC(ctx context.Context, state *scanState) func() error {
@@ -131,7 +231,15 @@ func (s *scannerImpl) runGC(ctx context.Context, state *scanState) func() error
return s.ds.WithTx(func(tx model.DataStore) error {
if state.changesDetected.Load() {
start := time.Now()
- err := tx.GC(ctx)
+
+ // For selective scans, extract library IDs to scope GC operations
+ var libraryIDs []int
+ if state.isSelectiveScan() {
+ libraryIDs = slices.Collect(maps.Keys(state.targets))
+ log.Debug(ctx, "Scanner: Running selective GC", "libraryIDs", libraryIDs)
+ }
+
+ err := tx.GC(ctx, libraryIDs...)
if err != nil {
log.Error(ctx, "Scanner: Error running GC", err)
return fmt.Errorf("running GC: %w", err)
@@ -152,7 +260,7 @@ func (s *scannerImpl) runRefreshStats(ctx context.Context, state *scanState) fun
return nil
}
start := time.Now()
- stats, err := s.ds.Artist(ctx).RefreshStats()
+ stats, err := s.ds.Artist(ctx).RefreshStats(state.fullScan)
if err != nil {
log.Error(ctx, "Scanner: Error refreshing artists stats", err)
return fmt.Errorf("refreshing artists stats: %w", err)
@@ -179,10 +287,11 @@ func (s *scannerImpl) runOptimize(ctx context.Context) func() error {
}
}
-func (s *scannerImpl) runUpdateLibraries(ctx context.Context, libs model.Libraries) func() error {
+func (s *scannerImpl) runUpdateLibraries(ctx context.Context, state *scanState) func() error {
return func() error {
+ start := time.Now()
return s.ds.WithTx(func(tx model.DataStore) error {
- for _, lib := range libs {
+ for _, lib := range state.libraries {
err := tx.Library(ctx).ScanEnd(lib.ID)
if err != nil {
log.Error(ctx, "Scanner: Error updating last scan completed", "lib", lib.Name, err)
@@ -198,7 +307,17 @@ func (s *scannerImpl) runUpdateLibraries(ctx context.Context, libs model.Librari
log.Error(ctx, "Scanner: Error updating album PID conf", err)
return fmt.Errorf("updating album PID conf: %w", err)
}
+ if state.changesDetected.Load() {
+ log.Debug(ctx, "Scanner: Refreshing library stats", "lib", lib.Name)
+ if err := tx.Library(ctx).RefreshStats(lib.ID); err != nil {
+ log.Error(ctx, "Scanner: Error refreshing library stats", "lib", lib.Name, err)
+ return fmt.Errorf("refreshing library stats: %w", err)
+ }
+ } else {
+ log.Debug(ctx, "Scanner: No changes detected, skipping library stats refresh", "lib", lib.Name)
+ }
}
+ log.Debug(ctx, "Scanner: Updated libraries after scan", "elapsed", time.Since(start), "numLibraries", len(state.libraries))
return nil
}, "scanner: update libraries")
}
diff --git a/scanner/scanner_multilibrary_test.go b/scanner/scanner_multilibrary_test.go
new file mode 100644
index 000000000..66db62edf
--- /dev/null
+++ b/scanner/scanner_multilibrary_test.go
@@ -0,0 +1,831 @@
+package scanner_test
+
+import (
+ "context"
+ "errors"
+ "path/filepath"
+ "testing/fstest"
+ "time"
+
+ "github.com/Masterminds/squirrel"
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/consts"
+ "github.com/navidrome/navidrome/core"
+ "github.com/navidrome/navidrome/core/artwork"
+ "github.com/navidrome/navidrome/core/metrics"
+ "github.com/navidrome/navidrome/core/storage/storagetest"
+ "github.com/navidrome/navidrome/db"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/persistence"
+ "github.com/navidrome/navidrome/scanner"
+ "github.com/navidrome/navidrome/server/events"
+ "github.com/navidrome/navidrome/tests"
+ "github.com/navidrome/navidrome/utils/slice"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("Scanner - Multi-Library", Ordered, func() {
+ var ctx context.Context
+ var lib1, lib2 model.Library
+ var ds *tests.MockDataStore
+ var s model.Scanner
+
+ createFS := func(path string, files fstest.MapFS) storagetest.FakeFS {
+ fs := storagetest.FakeFS{}
+ fs.SetFiles(files)
+ storagetest.Register(path, &fs)
+ return fs
+ }
+
+ BeforeAll(func() {
+ ctx = request.WithUser(GinkgoT().Context(), model.User{ID: "123", IsAdmin: true})
+ tmpDir := GinkgoT().TempDir()
+ conf.Server.DbPath = filepath.Join(tmpDir, "test-scanner-multilibrary.db?_journal_mode=WAL")
+ log.Warn("Using DB at " + conf.Server.DbPath)
+ db.Db().SetMaxOpenConns(1)
+ })
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ conf.Server.DevExternalScanner = false
+
+ db.Init(ctx)
+ DeferCleanup(func() {
+ Expect(tests.ClearDB()).To(Succeed())
+ })
+
+ ds = &tests.MockDataStore{RealDS: persistence.New(db.Db())}
+
+ // Create the admin user in the database to match the context
+ adminUser := model.User{
+ ID: "123",
+ UserName: "admin",
+ Name: "Admin User",
+ IsAdmin: true,
+ NewPassword: "password",
+ }
+ Expect(ds.User(ctx).Put(&adminUser)).To(Succeed())
+
+ s = scanner.New(ctx, ds, artwork.NoopCacheWarmer(), events.NoopBroker(),
+ core.NewPlaylists(ds), metrics.NewNoopInstance())
+
+ // Create two test libraries (let DB auto-assign IDs)
+ lib1 = model.Library{Name: "Rock Collection", Path: "rock:///music"}
+ lib2 = model.Library{Name: "Jazz Collection", Path: "jazz:///music"}
+ Expect(ds.Library(ctx).Put(&lib1)).To(Succeed())
+ Expect(ds.Library(ctx).Put(&lib2)).To(Succeed())
+ })
+
+ runScanner := func(ctx context.Context, fullScan bool) error {
+ _, err := s.ScanAll(ctx, fullScan)
+ return err
+ }
+
+ Context("Two Libraries with Different Content", func() {
+ BeforeEach(func() {
+ // Rock library content
+ beatles := template(_t{"albumartist": "The Beatles", "album": "Abbey Road", "year": 1969, "genre": "Rock"})
+ zeppelin := template(_t{"albumartist": "Led Zeppelin", "album": "IV", "year": 1971, "genre": "Rock"})
+
+ _ = createFS("rock", fstest.MapFS{
+ "The Beatles/Abbey Road/01 - Come Together.mp3": beatles(track(1, "Come Together")),
+ "The Beatles/Abbey Road/02 - Something.mp3": beatles(track(2, "Something")),
+ "Led Zeppelin/IV/01 - Black Dog.mp3": zeppelin(track(1, "Black Dog")),
+ "Led Zeppelin/IV/02 - Rock and Roll.mp3": zeppelin(track(2, "Rock and Roll")),
+ })
+
+ // Jazz library content
+ miles := template(_t{"albumartist": "Miles Davis", "album": "Kind of Blue", "year": 1959, "genre": "Jazz"})
+ coltrane := template(_t{"albumartist": "John Coltrane", "album": "Giant Steps", "year": 1960, "genre": "Jazz"})
+
+ _ = createFS("jazz", fstest.MapFS{
+ "Miles Davis/Kind of Blue/01 - So What.mp3": miles(track(1, "So What")),
+ "Miles Davis/Kind of Blue/02 - Freddie Freeloader.mp3": miles(track(2, "Freddie Freeloader")),
+ "John Coltrane/Giant Steps/01 - Giant Steps.mp3": coltrane(track(1, "Giant Steps")),
+ "John Coltrane/Giant Steps/02 - Cousin Mary.mp3": coltrane(track(2, "Cousin Mary")),
+ })
+ })
+
+ When("scanning both libraries", func() {
+ It("should import files with correct library_id", func() {
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ // Check Rock library media files
+ rockFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib1.ID},
+ Sort: "title",
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(rockFiles).To(HaveLen(4))
+
+ rockTitles := slice.Map(rockFiles, func(f model.MediaFile) string { return f.Title })
+ Expect(rockTitles).To(ContainElements("Come Together", "Something", "Black Dog", "Rock and Roll"))
+
+ // Verify all rock files have correct library_id
+ for _, mf := range rockFiles {
+ Expect(mf.LibraryID).To(Equal(lib1.ID), "Rock file %s should have library_id %d", mf.Title, lib1.ID)
+ }
+
+ // Check Jazz library media files
+ jazzFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID},
+ Sort: "title",
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzFiles).To(HaveLen(4))
+
+ jazzTitles := slice.Map(jazzFiles, func(f model.MediaFile) string { return f.Title })
+ Expect(jazzTitles).To(ContainElements("So What", "Freddie Freeloader", "Giant Steps", "Cousin Mary"))
+
+ // Verify all jazz files have correct library_id
+ for _, mf := range jazzFiles {
+ Expect(mf.LibraryID).To(Equal(lib2.ID), "Jazz file %s should have library_id %d", mf.Title, lib2.ID)
+ }
+ })
+
+ It("should create albums with correct library_id", func() {
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ // Check Rock library albums
+ rockAlbums, err := ds.Album(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib1.ID},
+ Sort: "name",
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(rockAlbums).To(HaveLen(2))
+ Expect(rockAlbums[0].Name).To(Equal("Abbey Road"))
+ Expect(rockAlbums[0].LibraryID).To(Equal(lib1.ID))
+ Expect(rockAlbums[0].SongCount).To(Equal(2))
+ Expect(rockAlbums[1].Name).To(Equal("IV"))
+ Expect(rockAlbums[1].LibraryID).To(Equal(lib1.ID))
+ Expect(rockAlbums[1].SongCount).To(Equal(2))
+
+ // Check Jazz library albums
+ jazzAlbums, err := ds.Album(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID},
+ Sort: "name",
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzAlbums).To(HaveLen(2))
+ Expect(jazzAlbums[0].Name).To(Equal("Giant Steps"))
+ Expect(jazzAlbums[0].LibraryID).To(Equal(lib2.ID))
+ Expect(jazzAlbums[0].SongCount).To(Equal(2))
+ Expect(jazzAlbums[1].Name).To(Equal("Kind of Blue"))
+ Expect(jazzAlbums[1].LibraryID).To(Equal(lib2.ID))
+ Expect(jazzAlbums[1].SongCount).To(Equal(2))
+ })
+
+ It("should create folders with correct library_id", func() {
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ // Check Rock library folders
+ rockFolders, err := ds.Folder(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib1.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(rockFolders).To(HaveLen(5)) // ., The Beatles, Led Zeppelin, Abbey Road, IV
+
+ for _, folder := range rockFolders {
+ Expect(folder.LibraryID).To(Equal(lib1.ID), "Rock folder %s should have library_id %d", folder.Name, lib1.ID)
+ }
+
+ // Check Jazz library folders
+ jazzFolders, err := ds.Folder(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzFolders).To(HaveLen(5)) // ., Miles Davis, John Coltrane, Kind of Blue, Giant Steps
+
+ for _, folder := range jazzFolders {
+ Expect(folder.LibraryID).To(Equal(lib2.ID), "Jazz folder %s should have library_id %d", folder.Name, lib2.ID)
+ }
+ })
+
+ It("should create library-artist associations correctly", func() {
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ // Check library-artist associations
+
+ // Get all artists and check library associations
+ allArtists, err := ds.Artist(ctx).GetAll()
+ Expect(err).ToNot(HaveOccurred())
+
+ rockArtistNames := []string{}
+ jazzArtistNames := []string{}
+
+ for _, artist := range allArtists {
+ // Check if artist is associated with rock library
+ var count int64
+ err := db.Db().QueryRow(
+ "SELECT COUNT(*) FROM library_artist WHERE library_id = ? AND artist_id = ?",
+ lib1.ID, artist.ID,
+ ).Scan(&count)
+ Expect(err).ToNot(HaveOccurred())
+ if count > 0 {
+ rockArtistNames = append(rockArtistNames, artist.Name)
+ }
+
+ // Check if artist is associated with jazz library
+ err = db.Db().QueryRow(
+ "SELECT COUNT(*) FROM library_artist WHERE library_id = ? AND artist_id = ?",
+ lib2.ID, artist.ID,
+ ).Scan(&count)
+ Expect(err).ToNot(HaveOccurred())
+ if count > 0 {
+ jazzArtistNames = append(jazzArtistNames, artist.Name)
+ }
+ }
+
+ Expect(rockArtistNames).To(ContainElements("The Beatles", "Led Zeppelin"))
+ Expect(jazzArtistNames).To(ContainElements("Miles Davis", "John Coltrane"))
+
+ // Artists should not be shared between libraries (except [Unknown Artist])
+ for _, name := range rockArtistNames {
+ if name != "[Unknown Artist]" {
+ Expect(jazzArtistNames).ToNot(ContainElement(name))
+ }
+ }
+ })
+
+ It("should update library statistics correctly", func() {
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ // Check Rock library stats
+ rockLib, err := ds.Library(ctx).Get(lib1.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(rockLib.TotalSongs).To(Equal(4))
+ Expect(rockLib.TotalAlbums).To(Equal(2))
+
+ Expect(rockLib.TotalArtists).To(Equal(3)) // The Beatles, Led Zeppelin, [Unknown Artist]
+ Expect(rockLib.TotalFolders).To(Equal(2)) // Abbey Road, IV (only folders with audio files)
+
+ // Check Jazz library stats
+ jazzLib, err := ds.Library(ctx).Get(lib2.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzLib.TotalSongs).To(Equal(4))
+ Expect(jazzLib.TotalAlbums).To(Equal(2))
+ Expect(jazzLib.TotalArtists).To(Equal(3)) // Miles Davis, John Coltrane, [Unknown Artist]
+ Expect(jazzLib.TotalFolders).To(Equal(2)) // Kind of Blue, Giant Steps (only folders with audio files)
+ })
+ })
+
+ When("libraries have different content", func() {
+ It("should maintain separate statistics per library", func() {
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ // Verify rock library stats
+ rockLib, err := ds.Library(ctx).Get(lib1.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(rockLib.TotalSongs).To(Equal(4))
+ Expect(rockLib.TotalAlbums).To(Equal(2))
+
+ // Verify jazz library stats
+ jazzLib, err := ds.Library(ctx).Get(lib2.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzLib.TotalSongs).To(Equal(4))
+ Expect(jazzLib.TotalAlbums).To(Equal(2))
+
+ // Verify that libraries don't interfere with each other
+ rockFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib1.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(rockFiles).To(HaveLen(4))
+
+ jazzFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzFiles).To(HaveLen(4))
+ })
+ })
+
+ When("verifying library isolation", func() {
+ It("should keep library data completely separate", func() {
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ // Verify that rock library only contains rock content
+ rockAlbums, err := ds.Album(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib1.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ rockAlbumNames := slice.Map(rockAlbums, func(a model.Album) string { return a.Name })
+ Expect(rockAlbumNames).To(ContainElements("Abbey Road", "IV"))
+ Expect(rockAlbumNames).ToNot(ContainElements("Kind of Blue", "Giant Steps"))
+
+ // Verify that jazz library only contains jazz content
+ jazzAlbums, err := ds.Album(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ jazzAlbumNames := slice.Map(jazzAlbums, func(a model.Album) string { return a.Name })
+ Expect(jazzAlbumNames).To(ContainElements("Kind of Blue", "Giant Steps"))
+ Expect(jazzAlbumNames).ToNot(ContainElements("Abbey Road", "IV"))
+ })
+ })
+
+ When("same artist appears in different libraries", func() {
+ It("should associate artist with both libraries correctly", func() {
+ // Create libraries with Jeff Beck albums in both
+ jeffRock := template(_t{"albumartist": "Jeff Beck", "album": "Truth", "year": 1968, "genre": "Rock"})
+ jeffJazz := template(_t{"albumartist": "Jeff Beck", "album": "Blow by Blow", "year": 1975, "genre": "Jazz"})
+ beatles := template(_t{"albumartist": "The Beatles", "album": "Abbey Road", "year": 1969, "genre": "Rock"})
+ miles := template(_t{"albumartist": "Miles Davis", "album": "Kind of Blue", "year": 1959, "genre": "Jazz"})
+
+ // Create rock library with Jeff Beck's Truth album
+ _ = createFS("rock", fstest.MapFS{
+ "The Beatles/Abbey Road/01 - Come Together.mp3": beatles(track(1, "Come Together")),
+ "The Beatles/Abbey Road/02 - Something.mp3": beatles(track(2, "Something")),
+ "Jeff Beck/Truth/01 - Beck's Bolero.mp3": jeffRock(track(1, "Beck's Bolero")),
+ "Jeff Beck/Truth/02 - Ol' Man River.mp3": jeffRock(track(2, "Ol' Man River")),
+ })
+
+ // Create jazz library with Jeff Beck's Blow by Blow album
+ _ = createFS("jazz", fstest.MapFS{
+ "Miles Davis/Kind of Blue/01 - So What.mp3": miles(track(1, "So What")),
+ "Miles Davis/Kind of Blue/02 - Freddie Freeloader.mp3": miles(track(2, "Freddie Freeloader")),
+ "Jeff Beck/Blow by Blow/01 - You Know What I Mean.mp3": jeffJazz(track(1, "You Know What I Mean")),
+ "Jeff Beck/Blow by Blow/02 - She's a Woman.mp3": jeffJazz(track(2, "She's a Woman")),
+ })
+
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ // Jeff Beck should be associated with both libraries
+ var rockCount, jazzCount int64
+
+ // Get Jeff Beck artist ID
+ jeffArtists, err := ds.Artist(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"name": "Jeff Beck"},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jeffArtists).To(HaveLen(1))
+ jeffID := jeffArtists[0].ID
+
+ // Check rock library association
+ err = db.Db().QueryRow(
+ "SELECT COUNT(*) FROM library_artist WHERE library_id = ? AND artist_id = ?",
+ lib1.ID, jeffID,
+ ).Scan(&rockCount)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(rockCount).To(Equal(int64(1)))
+
+ // Check jazz library association
+ err = db.Db().QueryRow(
+ "SELECT COUNT(*) FROM library_artist WHERE library_id = ? AND artist_id = ?",
+ lib2.ID, jeffID,
+ ).Scan(&jazzCount)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzCount).To(Equal(int64(1)))
+
+ // Verify Jeff Beck albums are in correct libraries
+ rockAlbums, err := ds.Album(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib1.ID, "album_artist": "Jeff Beck"},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(rockAlbums).To(HaveLen(1))
+ Expect(rockAlbums[0].Name).To(Equal("Truth"))
+
+ jazzAlbums, err := ds.Album(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID, "album_artist": "Jeff Beck"},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzAlbums).To(HaveLen(1))
+ Expect(jazzAlbums[0].Name).To(Equal("Blow by Blow"))
+ })
+ })
+ })
+
+ Context("Incremental Scan Behavior", func() {
+ BeforeEach(func() {
+ // Start with minimal content in both libraries
+ rock := template(_t{"albumartist": "Queen", "album": "News of the World", "year": 1977, "genre": "Rock"})
+ jazz := template(_t{"albumartist": "Bill Evans", "album": "Waltz for Debby", "year": 1961, "genre": "Jazz"})
+
+ createFS("rock", fstest.MapFS{
+ "Queen/News of the World/01 - We Will Rock You.mp3": rock(track(1, "We Will Rock You")),
+ })
+
+ createFS("jazz", fstest.MapFS{
+ "Bill Evans/Waltz for Debby/01 - My Foolish Heart.mp3": jazz(track(1, "My Foolish Heart")),
+ })
+ })
+
+ It("should handle incremental scans per library correctly", func() {
+ // Initial full scan
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ // Verify initial state
+ rockFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib1.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(rockFiles).To(HaveLen(1))
+
+ jazzFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzFiles).To(HaveLen(1))
+
+ // Incremental scan should not duplicate existing files
+ Expect(runScanner(ctx, false)).To(Succeed())
+
+ // Verify counts remain the same
+ rockFiles, err = ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib1.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(rockFiles).To(HaveLen(1))
+
+ jazzFiles, err = ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzFiles).To(HaveLen(1))
+ })
+ })
+
+ Context("Missing Files Handling", func() {
+ var rockFS storagetest.FakeFS
+
+ BeforeEach(func() {
+ rock := template(_t{"albumartist": "AC/DC", "album": "Back in Black", "year": 1980, "genre": "Rock"})
+
+ rockFS = createFS("rock", fstest.MapFS{
+ "AC-DC/Back in Black/01 - Hells Bells.mp3": rock(track(1, "Hells Bells")),
+ "AC-DC/Back in Black/02 - Shoot to Thrill.mp3": rock(track(2, "Shoot to Thrill")),
+ })
+
+ createFS("jazz", fstest.MapFS{
+ "Herbie Hancock/Head Hunters/01 - Chameleon.mp3": template(_t{
+ "albumartist": "Herbie Hancock", "album": "Head Hunters", "year": 1973, "genre": "Jazz",
+ })(track(1, "Chameleon")),
+ })
+ })
+
+ It("should mark missing files correctly per library", func() {
+ // Initial scan
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ // Remove one file from rock library only
+ rockFS.Remove("AC-DC/Back in Black/02 - Shoot to Thrill.mp3")
+
+ // Rescan
+ Expect(runScanner(ctx, false)).To(Succeed())
+
+ // Check that only the rock library file is marked as missing
+ missingRockFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.And{
+ squirrel.Eq{"library_id": lib1.ID},
+ squirrel.Eq{"missing": true},
+ },
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(missingRockFiles).To(HaveLen(1))
+ Expect(missingRockFiles[0].Title).To(Equal("Shoot to Thrill"))
+
+ // Check that jazz library files are not affected
+ missingJazzFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.And{
+ squirrel.Eq{"library_id": lib2.ID},
+ squirrel.Eq{"missing": true},
+ },
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(missingJazzFiles).To(HaveLen(0))
+
+ // Verify non-missing files
+ presentRockFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.And{
+ squirrel.Eq{"library_id": lib1.ID},
+ squirrel.Eq{"missing": false},
+ },
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(presentRockFiles).To(HaveLen(1))
+ Expect(presentRockFiles[0].Title).To(Equal("Hells Bells"))
+ })
+ })
+
+ Context("Error Handling - Multi-Library", func() {
+ Context("Filesystem errors affecting one library", func() {
+ var rockFS storagetest.FakeFS
+
+ BeforeEach(func() {
+ // Set up content for both libraries
+ rock := template(_t{"albumartist": "AC/DC", "album": "Back in Black", "year": 1980, "genre": "Rock"})
+ jazz := template(_t{"albumartist": "Miles Davis", "album": "Kind of Blue", "year": 1959, "genre": "Jazz"})
+
+ rockFS = createFS("rock", fstest.MapFS{
+ "AC-DC/Back in Black/01 - Hells Bells.mp3": rock(track(1, "Hells Bells")),
+ "AC-DC/Back in Black/02 - Shoot to Thrill.mp3": rock(track(2, "Shoot to Thrill")),
+ })
+
+ createFS("jazz", fstest.MapFS{
+ "Miles Davis/Kind of Blue/01 - So What.mp3": jazz(track(1, "So What")),
+ "Miles Davis/Kind of Blue/02 - Freddie Freeloader.mp3": jazz(track(2, "Freddie Freeloader")),
+ })
+ })
+
+ It("should not affect scanning of other libraries", func() {
+ // Inject filesystem read error in rock library only
+ rockFS.SetError("AC-DC/Back in Black/01 - Hells Bells.mp3", errors.New("filesystem read error"))
+
+ // Scan should succeed overall and return warnings
+ warnings, err := s.ScanAll(ctx, true)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(warnings).ToNot(BeEmpty(), "Should have warnings for filesystem errors")
+
+ // Jazz library should have been scanned successfully
+ jazzFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzFiles).To(HaveLen(2))
+ Expect(jazzFiles[0].Title).To(BeElementOf("So What", "Freddie Freeloader"))
+ Expect(jazzFiles[1].Title).To(BeElementOf("So What", "Freddie Freeloader"))
+
+ // Rock library may have partial content (depending on scanner implementation)
+ rockFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib1.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ // No specific expectation - some files may have been imported despite errors
+ _ = rockFiles
+
+ // Verify jazz library stats are correct
+ jazzLib, err := ds.Library(ctx).Get(lib2.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzLib.TotalSongs).To(Equal(2))
+
+ // Error should be empty (warnings don't count as scan errors)
+ lastError, err := ds.Property(ctx).DefaultGet(consts.LastScanErrorKey, "unset")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(lastError).To(BeEmpty())
+ })
+
+ It("should continue with warnings for affected library", func() {
+ // Inject read errors on multiple files in rock library
+ rockFS.SetError("AC-DC/Back in Black/01 - Hells Bells.mp3", errors.New("read error 1"))
+ rockFS.SetError("AC-DC/Back in Black/02 - Shoot to Thrill.mp3", errors.New("read error 2"))
+
+ // Scan should complete with warnings for multiple filesystem errors
+ warnings, err := s.ScanAll(ctx, true)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(warnings).ToNot(BeEmpty(), "Should have warnings for multiple filesystem errors")
+
+ // Jazz library should be completely unaffected
+ jazzFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzFiles).To(HaveLen(2))
+
+ // Jazz library statistics should be accurate
+ jazzLib, err := ds.Library(ctx).Get(lib2.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzLib.TotalSongs).To(Equal(2))
+ Expect(jazzLib.TotalAlbums).To(Equal(1))
+
+ // Error should be empty (warnings don't count as scan errors)
+ lastError, err := ds.Property(ctx).DefaultGet(consts.LastScanErrorKey, "unset")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(lastError).To(BeEmpty())
+ })
+ })
+
+ Context("Database errors during multi-library scanning", func() {
+ BeforeEach(func() {
+ // Set up content for both libraries
+ rock := template(_t{"albumartist": "Queen", "album": "News of the World", "year": 1977, "genre": "Rock"})
+ jazz := template(_t{"albumartist": "Bill Evans", "album": "Waltz for Debby", "year": 1961, "genre": "Jazz"})
+
+ createFS("rock", fstest.MapFS{
+ "Queen/News of the World/01 - We Will Rock You.mp3": rock(track(1, "We Will Rock You")),
+ })
+
+ createFS("jazz", fstest.MapFS{
+ "Bill Evans/Waltz for Debby/01 - My Foolish Heart.mp3": jazz(track(1, "My Foolish Heart")),
+ })
+ })
+
+ It("should propagate database errors and stop scanning", func() {
+ // Install mock repo that injects DB error
+ mfRepo := &mockMediaFileRepo{
+ MediaFileRepository: ds.RealDS.MediaFile(ctx),
+ GetMissingAndMatchingError: errors.New("database connection failed"),
+ }
+ ds.MockedMediaFile = mfRepo
+
+ // Scan should return the database error
+ Expect(runScanner(ctx, false)).To(MatchError(ContainSubstring("database connection failed")))
+
+ // Error should be recorded in scanner properties
+ lastError, err := ds.Property(ctx).DefaultGet(consts.LastScanErrorKey, "")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(lastError).To(ContainSubstring("database connection failed"))
+ })
+
+ It("should preserve error information in scanner properties", func() {
+ // Install mock repo that injects DB error
+ mfRepo := &mockMediaFileRepo{
+ MediaFileRepository: ds.RealDS.MediaFile(ctx),
+ GetMissingAndMatchingError: errors.New("critical database error"),
+ }
+ ds.MockedMediaFile = mfRepo
+
+ // Attempt scan (should fail)
+ Expect(runScanner(ctx, false)).To(HaveOccurred())
+
+ // Check that error is recorded in scanner properties
+ lastError, err := ds.Property(ctx).DefaultGet(consts.LastScanErrorKey, "")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(lastError).To(ContainSubstring("critical database error"))
+
+ // Scan type should still be recorded
+ scanType, _ := ds.Property(ctx).DefaultGet(consts.LastScanTypeKey, "")
+ Expect(scanType).To(BeElementOf("incremental", "quick"))
+ })
+ })
+
+ Context("Mixed error scenarios", func() {
+ var rockFS storagetest.FakeFS
+
+ BeforeEach(func() {
+ // Set up rock library with filesystem that can error
+ rock := template(_t{"albumartist": "Metallica", "album": "Master of Puppets", "year": 1986, "genre": "Metal"})
+ rockFS = createFS("rock", fstest.MapFS{
+ "Metallica/Master of Puppets/01 - Battery.mp3": rock(track(1, "Battery")),
+ "Metallica/Master of Puppets/02 - Master of Puppets.mp3": rock(track(2, "Master of Puppets")),
+ })
+
+ // Set up jazz library normally
+ jazz := template(_t{"albumartist": "Herbie Hancock", "album": "Head Hunters", "year": 1973, "genre": "Jazz"})
+ createFS("jazz", fstest.MapFS{
+ "Herbie Hancock/Head Hunters/01 - Chameleon.mp3": jazz(track(1, "Chameleon")),
+ })
+ })
+
+ It("should handle filesystem errors in one library while other succeeds", func() {
+ // Inject filesystem error in rock library
+ rockFS.SetError("Metallica/Master of Puppets/01 - Battery.mp3", errors.New("disk read error"))
+
+ // Scan should complete with warnings (not hard error)
+ warnings, err := s.ScanAll(ctx, true)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(warnings).ToNot(BeEmpty(), "Should have warnings for filesystem error")
+
+ // Jazz library should scan completely successfully
+ jazzFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzFiles).To(HaveLen(1))
+ Expect(jazzFiles[0].Title).To(Equal("Chameleon"))
+
+ // Jazz library statistics should be accurate
+ jazzLib, err := ds.Library(ctx).Get(lib2.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzLib.TotalSongs).To(Equal(1))
+ Expect(jazzLib.TotalAlbums).To(Equal(1))
+
+ // Rock library may have partial content (depending on scanner implementation)
+ rockFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib1.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ // No specific expectation - some files may have been imported despite errors
+ _ = rockFiles
+
+ // Error should be empty (warnings don't count as scan errors)
+ lastError, err := ds.Property(ctx).DefaultGet(consts.LastScanErrorKey, "unset")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(lastError).To(BeEmpty())
+ })
+
+ It("should handle partial failures gracefully", func() {
+ // Create a scenario where rock has filesystem issues and jazz has normal content
+ rockFS.SetError("Metallica/Master of Puppets/01 - Battery.mp3", errors.New("file corruption"))
+
+ // Do an initial scan with filesystem error
+ warnings, err := s.ScanAll(ctx, true)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(warnings).ToNot(BeEmpty(), "Should have warnings for file corruption")
+
+ // Verify that the working parts completed successfully
+ jazzFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzFiles).To(HaveLen(1))
+
+ // Scanner properties should reflect successful completion despite warnings
+ scanType, _ := ds.Property(ctx).DefaultGet(consts.LastScanTypeKey, "")
+ Expect(scanType).To(Equal("full"))
+
+ // Start time should be recorded
+ startTimeStr, _ := ds.Property(ctx).DefaultGet(consts.LastScanStartTimeKey, "")
+ Expect(startTimeStr).ToNot(BeEmpty())
+
+ // Error should be empty (warnings don't count as scan errors)
+ lastError, err := ds.Property(ctx).DefaultGet(consts.LastScanErrorKey, "unset")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(lastError).To(BeEmpty())
+ })
+ })
+
+ Context("Error recovery in multi-library context", func() {
+ It("should recover from previous library-specific errors", func() {
+ // Set up initial content
+ rock := template(_t{"albumartist": "Iron Maiden", "album": "The Number of the Beast", "year": 1982, "genre": "Metal"})
+ jazz := template(_t{"albumartist": "John Coltrane", "album": "Giant Steps", "year": 1960, "genre": "Jazz"})
+
+ rockFS := createFS("rock", fstest.MapFS{
+ "Iron Maiden/The Number of the Beast/01 - Invaders.mp3": rock(track(1, "Invaders")),
+ })
+
+ createFS("jazz", fstest.MapFS{
+ "John Coltrane/Giant Steps/01 - Giant Steps.mp3": jazz(track(1, "Giant Steps")),
+ })
+
+ // First scan with filesystem error in rock
+ rockFS.SetError("Iron Maiden/The Number of the Beast/01 - Invaders.mp3", errors.New("temporary disk error"))
+ warnings, err := s.ScanAll(ctx, true)
+ Expect(err).ToNot(HaveOccurred()) // Should succeed with warnings
+ Expect(warnings).ToNot(BeEmpty(), "Should have warnings for temporary disk error")
+
+ // Clear the error and add more content - recreate the filesystem completely
+ rockFS.ClearError("Iron Maiden/The Number of the Beast/01 - Invaders.mp3")
+
+ // Create a new filesystem with both files
+ createFS("rock", fstest.MapFS{
+ "Iron Maiden/The Number of the Beast/01 - Invaders.mp3": rock(track(1, "Invaders")),
+ "Iron Maiden/The Number of the Beast/02 - Children of the Damned.mp3": rock(track(2, "Children of the Damned")),
+ })
+
+ // Second scan should recover and import all rock content
+ warnings, err = s.ScanAll(ctx, true)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(warnings).ToNot(BeEmpty(), "Should have warnings for temporary disk error")
+
+ // Verify both libraries now have content (at least jazz should work)
+ rockFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib1.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ // The scanner should recover and import both rock files
+ Expect(len(rockFiles)).To(Equal(2))
+
+ jazzFiles, err := ds.MediaFile(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"library_id": lib2.ID},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzFiles).To(HaveLen(1))
+
+ // Both libraries should have correct content counts
+ rockLib, err := ds.Library(ctx).Get(lib1.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(rockLib.TotalSongs).To(Equal(2))
+
+ jazzLib, err := ds.Library(ctx).Get(lib2.ID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(jazzLib.TotalSongs).To(Equal(1))
+
+ // Error should be empty (successful recovery)
+ lastError, err := ds.Property(ctx).DefaultGet(consts.LastScanErrorKey, "unset")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(lastError).To(BeEmpty())
+ })
+ })
+ })
+
+ Context("Scanner Properties", func() {
+ It("should persist last scan type, start time and error properties", func() {
+ // trivial FS setup
+ rock := template(_t{"albumartist": "AC/DC", "album": "Back in Black", "year": 1980, "genre": "Rock"})
+ _ = createFS("rock", fstest.MapFS{
+ "AC-DC/Back in Black/01 - Hells Bells.mp3": rock(track(1, "Hells Bells")),
+ })
+
+ // Run a full scan
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ // Validate properties
+ scanType, _ := ds.Property(ctx).DefaultGet(consts.LastScanTypeKey, "")
+ Expect(scanType).To(Equal("full"))
+
+ startTimeStr, _ := ds.Property(ctx).DefaultGet(consts.LastScanStartTimeKey, "")
+ Expect(startTimeStr).ToNot(BeEmpty())
+ _, err := time.Parse(time.RFC3339, startTimeStr)
+ Expect(err).ToNot(HaveOccurred())
+
+ lastError, err := ds.Property(ctx).DefaultGet(consts.LastScanErrorKey, "unset")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(lastError).To(BeEmpty())
+ })
+ })
+})
diff --git a/scanner/scanner_selective_test.go b/scanner/scanner_selective_test.go
new file mode 100644
index 000000000..629826db4
--- /dev/null
+++ b/scanner/scanner_selective_test.go
@@ -0,0 +1,293 @@
+package scanner_test
+
+import (
+ "context"
+ "path/filepath"
+ "testing/fstest"
+
+ "github.com/Masterminds/squirrel"
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/core"
+ "github.com/navidrome/navidrome/core/artwork"
+ "github.com/navidrome/navidrome/core/metrics"
+ "github.com/navidrome/navidrome/core/storage/storagetest"
+ "github.com/navidrome/navidrome/db"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/persistence"
+ "github.com/navidrome/navidrome/scanner"
+ "github.com/navidrome/navidrome/server/events"
+ "github.com/navidrome/navidrome/tests"
+ "github.com/navidrome/navidrome/utils/slice"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("ScanFolders", Ordered, func() {
+ var ctx context.Context
+ var lib model.Library
+ var ds model.DataStore
+ var s model.Scanner
+ var fsys storagetest.FakeFS
+
+ BeforeAll(func() {
+ ctx = request.WithUser(GinkgoT().Context(), model.User{ID: "123", IsAdmin: true})
+ tmpDir := GinkgoT().TempDir()
+ conf.Server.DbPath = filepath.Join(tmpDir, "test-selective-scan.db?_journal_mode=WAL")
+ log.Warn("Using DB at " + conf.Server.DbPath)
+ db.Db().SetMaxOpenConns(1)
+ })
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ conf.Server.MusicFolder = "fake:///music"
+ conf.Server.DevExternalScanner = false
+
+ db.Init(ctx)
+ DeferCleanup(func() {
+ Expect(tests.ClearDB()).To(Succeed())
+ })
+
+ ds = persistence.New(db.Db())
+
+ // Create the admin user in the database to match the context
+ adminUser := model.User{
+ ID: "123",
+ UserName: "admin",
+ Name: "Admin User",
+ IsAdmin: true,
+ NewPassword: "password",
+ }
+ Expect(ds.User(ctx).Put(&adminUser)).To(Succeed())
+
+ s = scanner.New(ctx, ds, artwork.NoopCacheWarmer(), events.NoopBroker(),
+ core.NewPlaylists(ds), metrics.NewNoopInstance())
+
+ lib = model.Library{ID: 1, Name: "Fake Library", Path: "fake:///music"}
+ Expect(ds.Library(ctx).Put(&lib)).To(Succeed())
+
+ // Initialize fake filesystem
+ fsys = storagetest.FakeFS{}
+ storagetest.Register("fake", &fsys)
+ })
+
+ Describe("Adding tracks to the library", func() {
+ It("scans specified folders recursively including all subdirectories", func() {
+ rock := template(_t{"albumartist": "Rock Artist", "album": "Rock Album"})
+ jazz := template(_t{"albumartist": "Jazz Artist", "album": "Jazz Album"})
+ pop := template(_t{"albumartist": "Pop Artist", "album": "Pop Album"})
+ createFS(fstest.MapFS{
+ "rock/track1.mp3": rock(track(1, "Rock Track 1")),
+ "rock/track2.mp3": rock(track(2, "Rock Track 2")),
+ "rock/subdir/track3.mp3": rock(track(3, "Rock Track 3")),
+ "jazz/track4.mp3": jazz(track(1, "Jazz Track 1")),
+ "jazz/subdir/track5.mp3": jazz(track(2, "Jazz Track 2")),
+ "pop/track6.mp3": pop(track(1, "Pop Track 1")),
+ })
+
+ // Scan only the "rock" and "jazz" folders (including their subdirectories)
+ targets := []model.ScanTarget{
+ {LibraryID: lib.ID, FolderPath: "rock"},
+ {LibraryID: lib.ID, FolderPath: "jazz"},
+ }
+
+ warnings, err := s.ScanFolders(ctx, false, targets)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(warnings).To(BeEmpty())
+
+ // Verify all tracks in rock and jazz folders (including subdirectories) were imported
+ allFiles, err := ds.MediaFile(ctx).GetAll()
+ Expect(err).ToNot(HaveOccurred())
+
+ // Should have 5 tracks (all rock and jazz tracks including subdirectories)
+ Expect(allFiles).To(HaveLen(5))
+
+ // Get the file paths
+ paths := slice.Map(allFiles, func(mf model.MediaFile) string {
+ return filepath.ToSlash(mf.Path)
+ })
+
+ // Verify the correct files were scanned (including subdirectories)
+ Expect(paths).To(ContainElements(
+ "rock/track1.mp3",
+ "rock/track2.mp3",
+ "rock/subdir/track3.mp3",
+ "jazz/track4.mp3",
+ "jazz/subdir/track5.mp3",
+ ))
+
+ // Verify files in the pop folder were NOT scanned
+ Expect(paths).ToNot(ContainElement("pop/track6.mp3"))
+ })
+ })
+
+ Describe("Deleting folders", func() {
+ Context("when a child folder is deleted", func() {
+ var (
+ revolver, help func(...map[string]any) *fstest.MapFile
+ artistFolderID string
+ album1FolderID string
+ album2FolderID string
+ album1TrackIDs []string
+ album2TrackIDs []string
+ )
+
+ BeforeEach(func() {
+ // Setup template functions for creating test files
+ revolver = storagetest.Template(_t{"albumartist": "The Beatles", "album": "Revolver", "year": 1966})
+ help = storagetest.Template(_t{"albumartist": "The Beatles", "album": "Help!", "year": 1965})
+
+ // Initial filesystem with nested folders
+ fsys.SetFiles(fstest.MapFS{
+ "The Beatles/Revolver/01 - Taxman.mp3": revolver(storagetest.Track(1, "Taxman")),
+ "The Beatles/Revolver/02 - Eleanor Rigby.mp3": revolver(storagetest.Track(2, "Eleanor Rigby")),
+ "The Beatles/Help!/01 - Help!.mp3": help(storagetest.Track(1, "Help!")),
+ "The Beatles/Help!/02 - The Night Before.mp3": help(storagetest.Track(2, "The Night Before")),
+ })
+
+ // First scan - import everything
+ _, err := s.ScanAll(ctx, true)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify initial state - all folders exist
+ folders, err := ds.Folder(ctx).GetAll(model.QueryOptions{Filters: squirrel.Eq{"library_id": lib.ID}})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(folders).To(HaveLen(4)) // root, Artist, Album1, Album2
+
+ // Store folder IDs for later verification
+ for _, f := range folders {
+ switch f.Name {
+ case "The Beatles":
+ artistFolderID = f.ID
+ case "Revolver":
+ album1FolderID = f.ID
+ case "Help!":
+ album2FolderID = f.ID
+ }
+ }
+
+ // Verify all tracks exist
+ allTracks, err := ds.MediaFile(ctx).GetAll()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(allTracks).To(HaveLen(4))
+
+ // Store track IDs for later verification
+ for _, t := range allTracks {
+ if t.Album == "Revolver" {
+ album1TrackIDs = append(album1TrackIDs, t.ID)
+ } else if t.Album == "Help!" {
+ album2TrackIDs = append(album2TrackIDs, t.ID)
+ }
+ }
+
+ // Verify no tracks are missing initially
+ for _, t := range allTracks {
+ Expect(t.Missing).To(BeFalse())
+ }
+ })
+
+ It("should mark child folder and its tracks as missing when parent is scanned", func() {
+ // Delete the child folder (Help!) from the filesystem
+ fsys.SetFiles(fstest.MapFS{
+ "The Beatles/Revolver/01 - Taxman.mp3": revolver(storagetest.Track(1, "Taxman")),
+ "The Beatles/Revolver/02 - Eleanor Rigby.mp3": revolver(storagetest.Track(2, "Eleanor Rigby")),
+ // "The Beatles/Help!" folder and its contents are DELETED
+ })
+
+ // Run selective scan on the parent folder (Artist)
+ // This simulates what the watcher does when a child folder is deleted
+ _, err := s.ScanFolders(ctx, false, []model.ScanTarget{
+ {LibraryID: lib.ID, FolderPath: "The Beatles"},
+ })
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify the deleted child folder is now marked as missing
+ deletedFolder, err := ds.Folder(ctx).Get(album2FolderID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(deletedFolder.Missing).To(BeTrue(), "Deleted child folder should be marked as missing")
+
+ // Verify the deleted folder's tracks are marked as missing
+ for _, trackID := range album2TrackIDs {
+ track, err := ds.MediaFile(ctx).Get(trackID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(track.Missing).To(BeTrue(), "Track in deleted folder should be marked as missing")
+ }
+
+ // Verify the parent folder is still present and not marked as missing
+ parentFolder, err := ds.Folder(ctx).Get(artistFolderID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(parentFolder.Missing).To(BeFalse(), "Parent folder should not be marked as missing")
+
+ // Verify the sibling folder and its tracks are still present and not missing
+ siblingFolder, err := ds.Folder(ctx).Get(album1FolderID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(siblingFolder.Missing).To(BeFalse(), "Sibling folder should not be marked as missing")
+
+ for _, trackID := range album1TrackIDs {
+ track, err := ds.MediaFile(ctx).Get(trackID)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(track.Missing).To(BeFalse(), "Track in sibling folder should not be marked as missing")
+ }
+ })
+
+ It("should mark deeply nested child folders as missing", func() {
+ // Add a deeply nested folder structure
+ fsys.SetFiles(fstest.MapFS{
+ "The Beatles/Revolver/01 - Taxman.mp3": revolver(storagetest.Track(1, "Taxman")),
+ "The Beatles/Revolver/02 - Eleanor Rigby.mp3": revolver(storagetest.Track(2, "Eleanor Rigby")),
+ "The Beatles/Help!/01 - Help!.mp3": help(storagetest.Track(1, "Help!")),
+ "The Beatles/Help!/02 - The Night Before.mp3": help(storagetest.Track(2, "The Night Before")),
+ "The Beatles/Help!/Bonus/01 - Bonus Track.mp3": help(storagetest.Track(99, "Bonus Track")),
+ "The Beatles/Help!/Bonus/Nested/01 - Deep Track.mp3": help(storagetest.Track(100, "Deep Track")),
+ })
+
+ // Rescan to import the new nested structure
+ _, err := s.ScanAll(ctx, true)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify nested folders were created
+ allFolders, err := ds.Folder(ctx).GetAll(model.QueryOptions{Filters: squirrel.Eq{"library_id": lib.ID}})
+ Expect(err).ToNot(HaveOccurred())
+ Expect(len(allFolders)).To(BeNumerically(">", 4), "Should have more folders with nested structure")
+
+ // Now delete the entire Help! folder including nested children
+ fsys.SetFiles(fstest.MapFS{
+ "The Beatles/Revolver/01 - Taxman.mp3": revolver(storagetest.Track(1, "Taxman")),
+ "The Beatles/Revolver/02 - Eleanor Rigby.mp3": revolver(storagetest.Track(2, "Eleanor Rigby")),
+ // All Help! subfolders are deleted
+ })
+
+ // Run selective scan on parent
+ _, err = s.ScanFolders(ctx, false, []model.ScanTarget{
+ {LibraryID: lib.ID, FolderPath: "The Beatles"},
+ })
+ Expect(err).ToNot(HaveOccurred())
+
+ // Verify all Help! folders (including nested ones) are marked as missing
+ missingFolders, err := ds.Folder(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.And{
+ squirrel.Eq{"library_id": lib.ID},
+ squirrel.Eq{"missing": true},
+ },
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(len(missingFolders)).To(BeNumerically(">", 0), "At least one folder should be marked as missing")
+
+ // Verify all tracks in deleted folders are marked as missing
+ allTracks, err := ds.MediaFile(ctx).GetAll()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(allTracks).To(HaveLen(6))
+
+ for _, track := range allTracks {
+ if track.Album == "Help!" {
+ Expect(track.Missing).To(BeTrue(), "All tracks in deleted Help! folder should be marked as missing")
+ } else if track.Album == "Revolver" {
+ Expect(track.Missing).To(BeFalse(), "Tracks in Revolver folder should not be marked as missing")
+ }
+ }
+ })
+ })
+ })
+})
diff --git a/scanner/scanner_test.go b/scanner/scanner_test.go
index 3ed5a4704..873065aa3 100644
--- a/scanner/scanner_test.go
+++ b/scanner/scanner_test.go
@@ -34,19 +34,19 @@ type _t = map[string]any
var template = storagetest.Template
var track = storagetest.Track
+func createFS(files fstest.MapFS) storagetest.FakeFS {
+ fs := storagetest.FakeFS{}
+ fs.SetFiles(files)
+ storagetest.Register("fake", &fs)
+ return fs
+}
+
var _ = Describe("Scanner", Ordered, func() {
var ctx context.Context
var lib model.Library
var ds *tests.MockDataStore
var mfRepo *mockMediaFileRepo
- var s scanner.Scanner
-
- createFS := func(files fstest.MapFS) storagetest.FakeFS {
- fs := storagetest.FakeFS{}
- fs.SetFiles(files)
- storagetest.Register("fake", &fs)
- return fs
- }
+ var s model.Scanner
BeforeAll(func() {
ctx = request.WithUser(GinkgoT().Context(), model.User{ID: "123", IsAdmin: true})
@@ -58,12 +58,14 @@ var _ = Describe("Scanner", Ordered, func() {
})
BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ conf.Server.MusicFolder = "fake:///music" // Set to match test library path
+ conf.Server.DevExternalScanner = false
+
db.Init(ctx)
DeferCleanup(func() {
Expect(tests.ClearDB()).To(Succeed())
})
- DeferCleanup(configtest.SetupConfig())
- conf.Server.DevExternalScanner = false
ds = &tests.MockDataStore{RealDS: persistence.New(db.Db())}
mfRepo = &mockMediaFileRepo{
@@ -71,6 +73,16 @@ var _ = Describe("Scanner", Ordered, func() {
}
ds.MockedMediaFile = mfRepo
+ // Create the admin user in the database to match the context
+ adminUser := model.User{
+ ID: "123",
+ UserName: "admin",
+ Name: "Admin User",
+ IsAdmin: true,
+ NewPassword: "password",
+ }
+ Expect(ds.User(ctx).Put(&adminUser)).To(Succeed())
+
s = scanner.New(ctx, ds, artwork.NoopCacheWarmer(), events.NoopBroker(),
core.NewPlaylists(ds), metrics.NewNoopInstance())
@@ -466,6 +478,56 @@ var _ = Describe("Scanner", Ordered, func() {
Expect(mf.Missing).To(BeFalse())
})
+ It("marks tracks as missing when scanning a deleted folder with ScanFolders", func() {
+ By("Adding a third track to Revolver to have more test data")
+ fsys.Add("The Beatles/Revolver/03 - I'm Only Sleeping.mp3", revolver(track(3, "I'm Only Sleeping")))
+ Expect(runScanner(ctx, false)).To(Succeed())
+
+ By("Verifying initial state has 5 tracks")
+ Expect(ds.MediaFile(ctx).CountAll(model.QueryOptions{
+ Filters: squirrel.Eq{"missing": false},
+ })).To(Equal(int64(5)))
+
+ By("Removing the entire Revolver folder from filesystem")
+ fsys.Remove("The Beatles/Revolver/01 - Taxman.mp3")
+ fsys.Remove("The Beatles/Revolver/02 - Eleanor Rigby.mp3")
+ fsys.Remove("The Beatles/Revolver/03 - I'm Only Sleeping.mp3")
+
+ By("Scanning the parent folder (simulating watcher behavior)")
+ targets := []model.ScanTarget{
+ {LibraryID: lib.ID, FolderPath: "The Beatles"},
+ }
+ _, err := s.ScanFolders(ctx, false, targets)
+ Expect(err).To(Succeed())
+
+ By("Checking all Revolver tracks are marked as missing")
+ mf, err := findByPath("The Beatles/Revolver/01 - Taxman.mp3")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(mf.Missing).To(BeTrue())
+
+ mf, err = findByPath("The Beatles/Revolver/02 - Eleanor Rigby.mp3")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(mf.Missing).To(BeTrue())
+
+ mf, err = findByPath("The Beatles/Revolver/03 - I'm Only Sleeping.mp3")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(mf.Missing).To(BeTrue())
+
+ By("Checking the Help! tracks are not affected")
+ mf, err = findByPath("The Beatles/Help!/01 - Help!.mp3")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(mf.Missing).To(BeFalse())
+
+ mf, err = findByPath("The Beatles/Help!/02 - The Night Before.mp3")
+ Expect(err).ToNot(HaveOccurred())
+ Expect(mf.Missing).To(BeFalse())
+
+ By("Verifying only 2 non-missing tracks remain (Help! tracks)")
+ Expect(ds.MediaFile(ctx).CountAll(model.QueryOptions{
+ Filters: squirrel.Eq{"missing": false},
+ })).To(Equal(int64(2)))
+ })
+
It("does not override artist fields when importing an undertagged file", func() {
By("Making sure artist in the DB contains MBID and sort name")
aa, err := ds.Artist(ctx).GetAll(model.QueryOptions{
@@ -612,6 +674,99 @@ var _ = Describe("Scanner", Ordered, func() {
})
})
})
+
+ Describe("RefreshStats", func() {
+ var refreshStatsCalls []bool
+ var fsys storagetest.FakeFS
+ var help func(...map[string]any) *fstest.MapFile
+
+ BeforeEach(func() {
+ refreshStatsCalls = nil
+
+ // Create a mock artist repository that tracks RefreshStats calls
+ originalArtistRepo := ds.RealDS.Artist(ctx)
+ ds.MockedArtist = &testArtistRepo{
+ ArtistRepository: originalArtistRepo,
+ callTracker: &refreshStatsCalls,
+ }
+
+ // Create a simple filesystem for testing
+ help = template(_t{"albumartist": "The Beatles", "album": "Help!", "year": 1965})
+ fsys = createFS(fstest.MapFS{
+ "The Beatles/Help!/01 - Help!.mp3": help(track(1, "Help!")),
+ })
+ })
+
+ It("should call RefreshStats with allArtists=true for full scans", func() {
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ Expect(refreshStatsCalls).To(HaveLen(1))
+ Expect(refreshStatsCalls[0]).To(BeTrue(), "RefreshStats should be called with allArtists=true for full scans")
+ })
+
+ It("should call RefreshStats with allArtists=false for incremental scans", func() {
+ // First do a full scan to set up the data
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ // Reset the tracker to only track the incremental scan
+ refreshStatsCalls = nil
+
+ // Add a new file to trigger changes detection
+ fsys.Add("The Beatles/Help!/02 - The Night Before.mp3", help(track(2, "The Night Before")))
+
+ // Do an incremental scan
+ Expect(runScanner(ctx, false)).To(Succeed())
+
+ Expect(refreshStatsCalls).To(HaveLen(1))
+ Expect(refreshStatsCalls[0]).To(BeFalse(), "RefreshStats should be called with allArtists=false for incremental scans")
+ })
+
+ It("should update artist stats during quick scans when new albums are added", func() {
+ // Don't use the mocked artist repo for this test - we need the real one
+ ds.MockedArtist = nil
+
+ By("Initial scan with one album")
+ Expect(runScanner(ctx, true)).To(Succeed())
+
+ // Verify initial artist stats - should have 1 album, 1 song
+ artists, err := ds.Artist(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"name": "The Beatles"},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(artists).To(HaveLen(1))
+ artist := artists[0]
+ Expect(artist.AlbumCount).To(Equal(1)) // 1 album
+ Expect(artist.SongCount).To(Equal(1)) // 1 song
+
+ By("Adding files to an existing directory during incremental scan")
+ // Add more files to the existing Help! album - this should trigger artist stats update during incremental scan
+ fsys.Add("The Beatles/Help!/02 - The Night Before.mp3", help(track(2, "The Night Before")))
+ fsys.Add("The Beatles/Help!/03 - You've Got to Hide Your Love Away.mp3", help(track(3, "You've Got to Hide Your Love Away")))
+
+ // Do a quick scan (incremental)
+ Expect(runScanner(ctx, false)).To(Succeed())
+
+ By("Verifying artist stats were updated correctly")
+ // Fetch the artist again to check updated stats
+ artists, err = ds.Artist(ctx).GetAll(model.QueryOptions{
+ Filters: squirrel.Eq{"name": "The Beatles"},
+ })
+ Expect(err).ToNot(HaveOccurred())
+ Expect(artists).To(HaveLen(1))
+ updatedArtist := artists[0]
+
+ // Should now have 1 album and 3 songs total
+ // This is the key test - that artist stats are updated during quick scans
+ Expect(updatedArtist.AlbumCount).To(Equal(1)) // 1 album
+ Expect(updatedArtist.SongCount).To(Equal(3)) // 3 songs
+
+ // Also verify that role-specific stats are updated (albumartist role)
+ Expect(updatedArtist.Stats).To(HaveKey(model.RoleAlbumArtist))
+ albumArtistStats := updatedArtist.Stats[model.RoleAlbumArtist]
+ Expect(albumArtistStats.AlbumCount).To(Equal(1)) // 1 album
+ Expect(albumArtistStats.SongCount).To(Equal(3)) // 3 songs
+ })
+ })
})
func createFindByPath(ctx context.Context, ds model.DataStore) func(string) (*model.MediaFile, error) {
@@ -638,3 +793,13 @@ func (m *mockMediaFileRepo) GetMissingAndMatching(libId int) (model.MediaFileCur
}
return m.MediaFileRepository.GetMissingAndMatching(libId)
}
+
+type testArtistRepo struct {
+ model.ArtistRepository
+ callTracker *[]bool
+}
+
+func (m *testArtistRepo) RefreshStats(allArtists bool) (int64, error) {
+ *m.callTracker = append(*m.callTracker, allArtists)
+ return m.ArtistRepository.RefreshStats(allArtists)
+}
diff --git a/scanner/walk_dir_tree.go b/scanner/walk_dir_tree.go
index 4f9f26b1b..e6a694f2b 100644
--- a/scanner/walk_dir_tree.go
+++ b/scanner/walk_dir_tree.go
@@ -1,7 +1,6 @@
package scanner
import (
- "bufio"
"context"
"io/fs"
"maps"
@@ -9,102 +8,71 @@ import (
"slices"
"sort"
"strings"
- "time"
"github.com/navidrome/navidrome/conf"
- "github.com/navidrome/navidrome/consts"
- "github.com/navidrome/navidrome/core"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/utils"
- "github.com/navidrome/navidrome/utils/chrono"
- ignore "github.com/sabhiram/go-gitignore"
)
-type folderEntry struct {
- job *scanJob
- elapsed chrono.Meter
- path string // Full path
- id string // DB ID
- modTime time.Time // From FS
- updTime time.Time // from DB
- audioFiles map[string]fs.DirEntry
- imageFiles map[string]fs.DirEntry
- numPlaylists int
- numSubFolders int
- imagesUpdatedAt time.Time
- tracks model.MediaFiles
- albums model.Albums
- albumIDMap map[string]string
- artists model.Artists
- tags model.TagList
- missingTracks []*model.MediaFile
-}
-
-func (f *folderEntry) hasNoFiles() bool {
- return len(f.audioFiles) == 0 && len(f.imageFiles) == 0 && f.numPlaylists == 0 && f.numSubFolders == 0
-}
-
-func (f *folderEntry) isNew() bool {
- return f.updTime.IsZero()
-}
-
-func (f *folderEntry) toFolder() *model.Folder {
- folder := model.NewFolder(f.job.lib, f.path)
- folder.NumAudioFiles = len(f.audioFiles)
- if core.InPlaylistsPath(*folder) {
- folder.NumPlaylists = f.numPlaylists
- }
- folder.ImageFiles = slices.Collect(maps.Keys(f.imageFiles))
- folder.ImagesUpdatedAt = f.imagesUpdatedAt
- return folder
-}
-
-func newFolderEntry(job *scanJob, path string) *folderEntry {
- id := model.FolderID(job.lib, path)
- f := &folderEntry{
- id: id,
- job: job,
- path: path,
- audioFiles: make(map[string]fs.DirEntry),
- imageFiles: make(map[string]fs.DirEntry),
- albumIDMap: make(map[string]string),
- updTime: job.popLastUpdate(id),
- }
- return f
-}
-
-func (f *folderEntry) isOutdated() bool {
- if f.job.lib.FullScanInProgress {
- return f.updTime.Before(f.job.lib.LastScanStartedAt)
- }
- return f.updTime.Before(f.modTime)
-}
-
-func walkDirTree(ctx context.Context, job *scanJob) (<-chan *folderEntry, error) {
+// walkDirTree recursively walks the directory tree starting from the given targetFolders.
+// If no targetFolders are provided, it starts from the root folder (".").
+// It returns a channel of folderEntry pointers representing each folder found.
+func walkDirTree(ctx context.Context, job *scanJob, targetFolders ...string) (<-chan *folderEntry, error) {
results := make(chan *folderEntry)
+ folders := targetFolders
+ if len(targetFolders) == 0 {
+ // No specific folders provided, scan the root folder
+ folders = []string{"."}
+ }
go func() {
defer close(results)
- err := walkFolder(ctx, job, ".", nil, results)
- if err != nil {
- log.Error(ctx, "Scanner: There were errors reading directories from filesystem", "path", job.lib.Path, err)
- return
+ for _, folderPath := range folders {
+ if utils.IsCtxDone(ctx) {
+ return
+ }
+
+ // Check if target folder exists before walking it
+ // If it doesn't exist (e.g., deleted between watcher detection and scan execution),
+ // skip it so it remains in job.lastUpdates and gets handled in following steps
+ _, err := fs.Stat(job.fs, folderPath)
+ if err != nil {
+ log.Warn(ctx, "Scanner: Target folder does not exist.", "path", folderPath, err)
+ continue
+ }
+
+ // Create checker and push patterns from root to this folder
+ checker := newIgnoreChecker(job.fs)
+ err = checker.PushAllParents(ctx, folderPath)
+ if err != nil {
+ log.Error(ctx, "Scanner: Error pushing ignore patterns for target folder", "path", folderPath, err)
+ continue
+ }
+
+ // Recursively walk this folder and all its children
+ err = walkFolder(ctx, job, folderPath, checker, results)
+ if err != nil {
+ log.Error(ctx, "Scanner: Error walking target folder", "path", folderPath, err)
+ continue
+ }
}
- log.Debug(ctx, "Scanner: Finished reading folders", "lib", job.lib.Name, "path", job.lib.Path, "numFolders", job.numFolders.Load())
+ log.Debug(ctx, "Scanner: Finished reading target folders", "lib", job.lib.Name, "path", job.lib.Path, "numFolders", job.numFolders.Load())
}()
return results, nil
}
-func walkFolder(ctx context.Context, job *scanJob, currentFolder string, ignorePatterns []string, results chan<- *folderEntry) error {
- ignorePatterns = loadIgnoredPatterns(ctx, job.fs, currentFolder, ignorePatterns)
+func walkFolder(ctx context.Context, job *scanJob, currentFolder string, checker *IgnoreChecker, results chan<- *folderEntry) error {
+ // Push patterns for this folder onto the stack
+ _ = checker.Push(ctx, currentFolder)
+ defer checker.Pop() // Pop patterns when leaving this folder
- folder, children, err := loadDir(ctx, job, currentFolder, ignorePatterns)
+ folder, children, err := loadDir(ctx, job, currentFolder, checker)
if err != nil {
log.Warn(ctx, "Scanner: Error loading dir. Skipping", "path", currentFolder, err)
return nil
}
for _, c := range children {
- err := walkFolder(ctx, job, c, ignorePatterns, results)
+ err := walkFolder(ctx, job, c, checker, results)
if err != nil {
return err
}
@@ -122,50 +90,17 @@ func walkFolder(ctx context.Context, job *scanJob, currentFolder string, ignoreP
return nil
}
-func loadIgnoredPatterns(ctx context.Context, fsys fs.FS, currentFolder string, currentPatterns []string) []string {
- ignoreFilePath := path.Join(currentFolder, consts.ScanIgnoreFile)
- var newPatterns []string
- if _, err := fs.Stat(fsys, ignoreFilePath); err == nil {
- // Read and parse the .ndignore file
- ignoreFile, err := fsys.Open(ignoreFilePath)
- if err != nil {
- log.Warn(ctx, "Scanner: Error opening .ndignore file", "path", ignoreFilePath, err)
- // Continue with previous patterns
- } else {
- defer ignoreFile.Close()
- scanner := bufio.NewScanner(ignoreFile)
- for scanner.Scan() {
- line := scanner.Text()
- if line == "" || strings.HasPrefix(line, "#") {
- continue // Skip empty lines and comments
- }
- newPatterns = append(newPatterns, line)
- }
- if err := scanner.Err(); err != nil {
- log.Warn(ctx, "Scanner: Error reading .ignore file", "path", ignoreFilePath, err)
- }
- }
- // If the .ndignore file is empty, mimic the current behavior and ignore everything
- if len(newPatterns) == 0 {
- log.Trace(ctx, "Scanner: .ndignore file is empty, ignoring everything", "path", currentFolder)
- newPatterns = []string{"**/*"}
- } else {
- log.Trace(ctx, "Scanner: .ndignore file found ", "path", ignoreFilePath, "patterns", newPatterns)
- }
- }
- // Combine the patterns from the .ndignore file with the ones passed as argument
- combinedPatterns := append([]string{}, currentPatterns...)
- return append(combinedPatterns, newPatterns...)
-}
-
-func loadDir(ctx context.Context, job *scanJob, dirPath string, ignorePatterns []string) (folder *folderEntry, children []string, err error) {
- folder = newFolderEntry(job, dirPath)
-
+func loadDir(ctx context.Context, job *scanJob, dirPath string, checker *IgnoreChecker) (folder *folderEntry, children []string, err error) {
+ // Check if directory exists before creating the folder entry
+ // This is important to avoid removing the folder from lastUpdates if it doesn't exist
dirInfo, err := fs.Stat(job.fs, dirPath)
if err != nil {
log.Warn(ctx, "Scanner: Error stating dir", "path", dirPath, err)
return nil, nil, err
}
+
+ // Now that we know the folder exists, create the entry (which removes it from lastUpdates)
+ folder = job.createFolderEntry(dirPath)
folder.modTime = dirInfo.ModTime()
dir, err := job.fs.Open(dirPath)
@@ -180,12 +115,11 @@ func loadDir(ctx context.Context, job *scanJob, dirPath string, ignorePatterns [
return folder, children, err
}
- ignoreMatcher := ignore.CompileIgnoreLines(ignorePatterns...)
entries := fullReadDir(ctx, dirFile)
children = make([]string, 0, len(entries))
for _, entry := range entries {
entryPath := path.Join(dirPath, entry.Name())
- if len(ignorePatterns) > 0 && isScanIgnored(ctx, ignoreMatcher, entryPath) {
+ if checker.ShouldIgnore(ctx, entryPath) {
log.Trace(ctx, "Scanner: Ignoring entry", "path", entryPath)
continue
}
@@ -297,6 +231,7 @@ func isDirReadable(ctx context.Context, fsys fs.FS, dirPath string) bool {
var ignoredDirs = []string{
"$RECYCLE.BIN",
"#snapshot",
+ "@Recycle",
"@Recently-Snapshot",
".streams",
"lost+found",
@@ -317,11 +252,3 @@ func isDirIgnored(name string) bool {
func isEntryIgnored(name string) bool {
return strings.HasPrefix(name, ".") && !strings.HasPrefix(name, "..")
}
-
-func isScanIgnored(ctx context.Context, matcher *ignore.GitIgnore, entryPath string) bool {
- matches := matcher.MatchesPath(entryPath)
- if matches {
- log.Trace(ctx, "Scanner: Ignoring entry matching .ndignore: ", "path", entryPath)
- }
- return matches
-}
diff --git a/scanner/walk_dir_tree_test.go b/scanner/walk_dir_tree_test.go
index c4278ef82..c9add0bd1 100644
--- a/scanner/walk_dir_tree_test.go
+++ b/scanner/walk_dir_tree_test.go
@@ -25,82 +25,196 @@ var _ = Describe("walk_dir_tree", func() {
ctx context.Context
)
- BeforeEach(func() {
- DeferCleanup(configtest.SetupConfig())
- ctx = GinkgoT().Context()
- fsys = &mockMusicFS{
- FS: fstest.MapFS{
- "root/a/.ndignore": {Data: []byte("ignored/*")},
- "root/a/f1.mp3": {},
- "root/a/f2.mp3": {},
- "root/a/ignored/bad.mp3": {},
- "root/b/cover.jpg": {},
- "root/c/f3": {},
- "root/d": {},
- "root/d/.ndignore": {},
- "root/d/f1.mp3": {},
- "root/d/f2.mp3": {},
- "root/d/f3.mp3": {},
- "root/e/original/f1.mp3": {},
- "root/e/symlink": {Mode: fs.ModeSymlink, Data: []byte("root/e/original")},
+ Context("full library", func() {
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ ctx = GinkgoT().Context()
+ fsys = &mockMusicFS{
+ FS: fstest.MapFS{
+ "root/a/.ndignore": {Data: []byte("ignored/*")},
+ "root/a/f1.mp3": {},
+ "root/a/f2.mp3": {},
+ "root/a/ignored/bad.mp3": {},
+ "root/b/cover.jpg": {},
+ "root/c/f3": {},
+ "root/d": {},
+ "root/d/.ndignore": {},
+ "root/d/f1.mp3": {},
+ "root/d/f2.mp3": {},
+ "root/d/f3.mp3": {},
+ "root/e/original/f1.mp3": {},
+ "root/e/symlink": {Mode: fs.ModeSymlink, Data: []byte("original")},
+ },
+ }
+ job = &scanJob{
+ fs: fsys,
+ lib: model.Library{Path: "/music"},
+ }
+ })
+
+ // Helper function to call walkDirTree and collect folders from the results channel
+ getFolders := func() map[string]*folderEntry {
+ results, err := walkDirTree(ctx, job)
+ Expect(err).ToNot(HaveOccurred())
+
+ folders := map[string]*folderEntry{}
+ g := errgroup.Group{}
+ g.Go(func() error {
+ for folder := range results {
+ folders[folder.path] = folder
+ }
+ return nil
+ })
+ _ = g.Wait()
+ return folders
+ }
+
+ DescribeTable("symlink handling",
+ func(followSymlinks bool, expectedFolderCount int) {
+ conf.Server.Scanner.FollowSymlinks = followSymlinks
+ folders := getFolders()
+
+ Expect(folders).To(HaveLen(expectedFolderCount + 2)) // +2 for `.` and `root`
+
+ // Basic folder structure checks
+ Expect(folders["root/a"].audioFiles).To(SatisfyAll(
+ HaveLen(2),
+ HaveKey("f1.mp3"),
+ HaveKey("f2.mp3"),
+ ))
+ Expect(folders["root/a"].imageFiles).To(BeEmpty())
+ Expect(folders["root/b"].audioFiles).To(BeEmpty())
+ Expect(folders["root/b"].imageFiles).To(SatisfyAll(
+ HaveLen(1),
+ HaveKey("cover.jpg"),
+ ))
+ Expect(folders["root/c"].audioFiles).To(BeEmpty())
+ Expect(folders["root/c"].imageFiles).To(BeEmpty())
+ Expect(folders).ToNot(HaveKey("root/d"))
+
+ // Symlink specific checks
+ if followSymlinks {
+ Expect(folders["root/e/symlink"].audioFiles).To(HaveLen(1))
+ } else {
+ Expect(folders).ToNot(HaveKey("root/e/symlink"))
+ }
},
- }
- job = &scanJob{
- fs: fsys,
- lib: model.Library{Path: "/music"},
- }
+ Entry("with symlinks enabled", true, 7),
+ Entry("with symlinks disabled", false, 6),
+ )
})
- // Helper function to call walkDirTree and collect folders from the results channel
- getFolders := func() map[string]*folderEntry {
- results, err := walkDirTree(ctx, job)
- Expect(err).ToNot(HaveOccurred())
-
- folders := map[string]*folderEntry{}
- g := errgroup.Group{}
- g.Go(func() error {
- for folder := range results {
- folders[folder.path] = folder
+ Context("with target folders", func() {
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ ctx = GinkgoT().Context()
+ fsys = &mockMusicFS{
+ FS: fstest.MapFS{
+ "Artist/Album1/track1.mp3": {},
+ "Artist/Album1/track2.mp3": {},
+ "Artist/Album2/track1.mp3": {},
+ "Artist/Album2/track2.mp3": {},
+ "Artist/Album2/Sub/track3.mp3": {},
+ "OtherArtist/Album3/track1.mp3": {},
+ },
+ }
+ job = &scanJob{
+ fs: fsys,
+ lib: model.Library{Path: "/music"},
}
- return nil
})
- _ = g.Wait()
- return folders
- }
- DescribeTable("symlink handling",
- func(followSymlinks bool, expectedFolderCount int) {
- conf.Server.Scanner.FollowSymlinks = followSymlinks
- folders := getFolders()
+ It("should recursively walk all subdirectories of target folders", func() {
+ results, err := walkDirTree(ctx, job, "Artist")
+ Expect(err).ToNot(HaveOccurred())
- Expect(folders).To(HaveLen(expectedFolderCount + 2)) // +2 for `.` and `root`
+ folders := map[string]*folderEntry{}
+ g := errgroup.Group{}
+ g.Go(func() error {
+ for folder := range results {
+ folders[folder.path] = folder
+ }
+ return nil
+ })
+ _ = g.Wait()
- // Basic folder structure checks
- Expect(folders["root/a"].audioFiles).To(SatisfyAll(
- HaveLen(2),
- HaveKey("f1.mp3"),
- HaveKey("f2.mp3"),
+ // Should include the target folder and all its descendants
+ Expect(folders).To(SatisfyAll(
+ HaveKey("Artist"),
+ HaveKey("Artist/Album1"),
+ HaveKey("Artist/Album2"),
+ HaveKey("Artist/Album2/Sub"),
))
- Expect(folders["root/a"].imageFiles).To(BeEmpty())
- Expect(folders["root/b"].audioFiles).To(BeEmpty())
- Expect(folders["root/b"].imageFiles).To(SatisfyAll(
- HaveLen(1),
- HaveKey("cover.jpg"),
- ))
- Expect(folders["root/c"].audioFiles).To(BeEmpty())
- Expect(folders["root/c"].imageFiles).To(BeEmpty())
- Expect(folders).ToNot(HaveKey("root/d"))
- // Symlink specific checks
- if followSymlinks {
- Expect(folders["root/e/symlink"].audioFiles).To(HaveLen(1))
- } else {
- Expect(folders).ToNot(HaveKey("root/e/symlink"))
+ // Should not include folders outside the target
+ Expect(folders).ToNot(HaveKey("OtherArtist"))
+ Expect(folders).ToNot(HaveKey("OtherArtist/Album3"))
+
+ // Verify audio files are present
+ Expect(folders["Artist/Album1"].audioFiles).To(HaveLen(2))
+ Expect(folders["Artist/Album2"].audioFiles).To(HaveLen(2))
+ Expect(folders["Artist/Album2/Sub"].audioFiles).To(HaveLen(1))
+ })
+
+ It("should handle multiple target folders", func() {
+ results, err := walkDirTree(ctx, job, "Artist/Album1", "OtherArtist")
+ Expect(err).ToNot(HaveOccurred())
+
+ folders := map[string]*folderEntry{}
+ g := errgroup.Group{}
+ g.Go(func() error {
+ for folder := range results {
+ folders[folder.path] = folder
+ }
+ return nil
+ })
+ _ = g.Wait()
+
+ // Should include both target folders and their descendants
+ Expect(folders).To(SatisfyAll(
+ HaveKey("Artist/Album1"),
+ HaveKey("OtherArtist"),
+ HaveKey("OtherArtist/Album3"),
+ ))
+
+ // Should not include other folders
+ Expect(folders).ToNot(HaveKey("Artist"))
+ Expect(folders).ToNot(HaveKey("Artist/Album2"))
+ Expect(folders).ToNot(HaveKey("Artist/Album2/Sub"))
+ })
+
+ It("should skip non-existent target folders and preserve them in lastUpdates", func() {
+ // Setup job with lastUpdates for both existing and non-existing folders
+ job.lastUpdates = map[string]model.FolderUpdateInfo{
+ model.FolderID(job.lib, "Artist/Album1"): {},
+ model.FolderID(job.lib, "NonExistent/DeletedFolder"): {},
+ model.FolderID(job.lib, "OtherArtist/Album3"): {},
}
- },
- Entry("with symlinks enabled", true, 7),
- Entry("with symlinks disabled", false, 6),
- )
+
+ // Try to scan existing folder and non-existing folder
+ results, err := walkDirTree(ctx, job, "Artist/Album1", "NonExistent/DeletedFolder")
+ Expect(err).ToNot(HaveOccurred())
+
+ // Collect results
+ folders := map[string]struct{}{}
+ for folder := range results {
+ folders[folder.path] = struct{}{}
+ }
+
+ // Should only include the existing folder
+ Expect(folders).To(HaveKey("Artist/Album1"))
+ Expect(folders).ToNot(HaveKey("NonExistent/DeletedFolder"))
+
+ // The non-existent folder should still be in lastUpdates (not removed by popLastUpdate)
+ Expect(job.lastUpdates).To(HaveKey(model.FolderID(job.lib, "NonExistent/DeletedFolder")))
+
+ // The existing folder should have been removed from lastUpdates
+ Expect(job.lastUpdates).ToNot(HaveKey(model.FolderID(job.lib, "Artist/Album1")))
+
+ // Folders not in targets should remain in lastUpdates
+ Expect(job.lastUpdates).To(HaveKey(model.FolderID(job.lib, "OtherArtist/Album3")))
+ })
+ })
})
Describe("helper functions", func() {
diff --git a/scanner/watcher.go b/scanner/watcher.go
index bf4f7f9d0..3efebaacc 100644
--- a/scanner/watcher.go
+++ b/scanner/watcher.go
@@ -5,49 +5,79 @@ import (
"fmt"
"io/fs"
"path/filepath"
+ "sync"
"time"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/core/storage"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/utils/singleton"
)
type Watcher interface {
Run(ctx context.Context) error
+ Watch(ctx context.Context, lib *model.Library) error
+ StopWatching(ctx context.Context, libraryID int) error
}
type watcher struct {
- ds model.DataStore
- scanner Scanner
- triggerWait time.Duration
+ mainCtx context.Context
+ ds model.DataStore
+ scanner model.Scanner
+ triggerWait time.Duration
+ watcherNotify chan scanNotification
+ libraryWatchers map[int]*libraryWatcherInstance
+ mu sync.RWMutex
}
-func NewWatcher(ds model.DataStore, s Scanner) Watcher {
- return &watcher{ds: ds, scanner: s, triggerWait: conf.Server.Scanner.WatcherWait}
+type libraryWatcherInstance struct {
+ library *model.Library
+ cancel context.CancelFunc
+}
+
+type scanNotification struct {
+ Library *model.Library
+ FolderPath string
+}
+
+// GetWatcher returns the watcher singleton
+func GetWatcher(ds model.DataStore, s model.Scanner) Watcher {
+ return singleton.GetInstance(func() *watcher {
+ return &watcher{
+ ds: ds,
+ scanner: s,
+ triggerWait: conf.Server.Scanner.WatcherWait,
+ watcherNotify: make(chan scanNotification, 1),
+ libraryWatchers: make(map[int]*libraryWatcherInstance),
+ }
+ })
}
func (w *watcher) Run(ctx context.Context) error {
+ // Keep the main context to be used in all watchers added later
+ w.mainCtx = ctx
+
+ // Start watchers for all existing libraries
libs, err := w.ds.Library(ctx).GetAll()
if err != nil {
return fmt.Errorf("getting libraries: %w", err)
}
- watcherChan := make(chan struct{})
- defer close(watcherChan)
-
- // Start a watcher for each library
for _, lib := range libs {
- go watchLib(ctx, lib, watcherChan)
+ if err := w.Watch(ctx, &lib); err != nil {
+ log.Warn(ctx, "Failed to start watcher for existing library", "libraryID", lib.ID, "name", lib.Name, "path", lib.Path, err)
+ }
}
+ // Main scan triggering loop
trigger := time.NewTimer(w.triggerWait)
trigger.Stop()
- waiting := false
+ targets := make(map[model.ScanTarget]struct{})
for {
select {
case <-trigger.C:
- log.Info("Watcher: Triggering scan")
+ log.Info("Watcher: Triggering scan for changed folders", "numTargets", len(targets))
status, err := w.scanner.Status(ctx)
if err != nil {
log.Error(ctx, "Watcher: Error retrieving Scanner status", err)
@@ -58,9 +88,23 @@ func (w *watcher) Run(ctx context.Context) error {
trigger.Reset(w.triggerWait * 3)
continue
}
- waiting = false
+
+ // Convert targets map to slice
+ targetSlice := make([]model.ScanTarget, 0, len(targets))
+ for target := range targets {
+ targetSlice = append(targetSlice, target)
+ }
+
+ // Clear targets for next batch
+ targets = make(map[model.ScanTarget]struct{})
+
go func() {
- _, err := w.scanner.ScanAll(ctx, false)
+ var err error
+ if conf.Server.DevSelectiveWatcher {
+ _, err = w.scanner.ScanFolders(ctx, false, targetSlice)
+ } else {
+ _, err = w.scanner.ScanAll(ctx, false)
+ }
if err != nil {
log.Error(ctx, "Watcher: Error scanning", err)
} else {
@@ -68,65 +112,211 @@ func (w *watcher) Run(ctx context.Context) error {
}
}()
case <-ctx.Done():
- return nil
- case <-watcherChan:
- if !waiting {
- log.Debug(ctx, "Watcher: Detected changes. Waiting for more changes before triggering scan")
- waiting = true
+ // Stop all library watchers
+ w.mu.Lock()
+ for libraryID, instance := range w.libraryWatchers {
+ log.Debug(ctx, "Stopping library watcher due to context cancellation", "libraryID", libraryID)
+ instance.cancel()
}
-
+ w.libraryWatchers = make(map[int]*libraryWatcherInstance)
+ w.mu.Unlock()
+ return nil
+ case notification := <-w.watcherNotify:
+ // Reset the trigger timer for debounce
trigger.Reset(w.triggerWait)
+
+ lib := notification.Library
+ folderPath := notification.FolderPath
+
+ // If already scheduled for scan, skip
+ target := model.ScanTarget{LibraryID: lib.ID, FolderPath: folderPath}
+ if _, exists := targets[target]; exists {
+ continue
+ }
+ targets[target] = struct{}{}
+
+ log.Debug(ctx, "Watcher: Detected changes. Waiting for more changes before triggering scan",
+ "libraryID", lib.ID, "name", lib.Name, "path", lib.Path, "folderPath", folderPath)
}
}
}
-func watchLib(ctx context.Context, lib model.Library, watchChan chan struct{}) {
+func (w *watcher) Watch(ctx context.Context, lib *model.Library) error {
+ w.mu.Lock()
+ defer w.mu.Unlock()
+
+ // Stop existing watcher if any
+ if existingInstance, exists := w.libraryWatchers[lib.ID]; exists {
+ log.Debug(ctx, "Stopping existing watcher before starting new one", "libraryID", lib.ID, "name", lib.Name)
+ existingInstance.cancel()
+ }
+
+ // Start new watcher
+ watcherCtx, cancel := context.WithCancel(w.mainCtx)
+ instance := &libraryWatcherInstance{
+ library: lib,
+ cancel: cancel,
+ }
+
+ w.libraryWatchers[lib.ID] = instance
+
+ // Start watching in a goroutine
+ go func() {
+ defer func() {
+ w.mu.Lock()
+ if currentInstance, exists := w.libraryWatchers[lib.ID]; exists && currentInstance == instance {
+ delete(w.libraryWatchers, lib.ID)
+ }
+ w.mu.Unlock()
+ }()
+
+ err := w.watchLibrary(watcherCtx, lib)
+ if err != nil && watcherCtx.Err() == nil { // Only log error if not due to cancellation
+ log.Error(ctx, "Watcher error", "libraryID", lib.ID, "name", lib.Name, "path", lib.Path, err)
+ }
+ }()
+
+ log.Info(ctx, "Started watcher for library", "libraryID", lib.ID, "name", lib.Name, "path", lib.Path)
+ return nil
+}
+
+func (w *watcher) StopWatching(ctx context.Context, libraryID int) error {
+ w.mu.Lock()
+ defer w.mu.Unlock()
+
+ instance, exists := w.libraryWatchers[libraryID]
+ if !exists {
+ log.Debug(ctx, "No watcher found to stop", "libraryID", libraryID)
+ return nil
+ }
+
+ instance.cancel()
+ delete(w.libraryWatchers, libraryID)
+
+ log.Info(ctx, "Stopped watcher for library", "libraryID", libraryID, "name", instance.library.Name)
+ return nil
+}
+
+// watchLibrary implements the core watching logic for a single library (extracted from old watchLib function)
+func (w *watcher) watchLibrary(ctx context.Context, lib *model.Library) error {
s, err := storage.For(lib.Path)
if err != nil {
- log.Error(ctx, "Watcher: Error creating storage", "library", lib.ID, "path", lib.Path, err)
- return
+ return fmt.Errorf("creating storage: %w", err)
}
+
fsys, err := s.FS()
if err != nil {
- log.Error(ctx, "Watcher: Error getting FS", "library", lib.ID, "path", lib.Path, err)
- return
+ return fmt.Errorf("getting FS: %w", err)
}
+
watcher, ok := s.(storage.Watcher)
if !ok {
- log.Info(ctx, "Watcher not supported", "library", lib.ID, "path", lib.Path)
- return
+ log.Info(ctx, "Watcher not supported for storage type", "libraryID", lib.ID, "path", lib.Path)
+ return nil
}
+
c, err := watcher.Start(ctx)
if err != nil {
- log.Error(ctx, "Watcher: Error watching library", "library", lib.ID, "path", lib.Path, err)
- return
+ return fmt.Errorf("starting watcher: %w", err)
}
+
absLibPath, err := filepath.Abs(lib.Path)
if err != nil {
- log.Error(ctx, "Watcher: Error converting lib.Path to absolute", "library", lib.ID, "path", lib.Path, err)
- return
+ return fmt.Errorf("converting to absolute path: %w", err)
}
- log.Info(ctx, "Watcher started", "library", lib.ID, "libPath", lib.Path, "absoluteLibPath", absLibPath)
+
+ log.Info(ctx, "Watcher started for library", "libraryID", lib.ID, "name", lib.Name, "path", lib.Path, "absoluteLibPath", absLibPath)
+
+ return w.processLibraryEvents(ctx, lib, fsys, c, absLibPath)
+}
+
+// processLibraryEvents processes filesystem events for a library.
+func (w *watcher) processLibraryEvents(ctx context.Context, lib *model.Library, fsys storage.MusicFS, events <-chan string, absLibPath string) error {
for {
select {
case <-ctx.Done():
- return
- case path := <-c:
- path, err = filepath.Rel(absLibPath, path)
+ log.Debug(ctx, "Watcher stopped due to context cancellation", "libraryID", lib.ID, "name", lib.Name)
+ return nil
+ case path := <-events:
+ path, err := filepath.Rel(absLibPath, path)
if err != nil {
- log.Error(ctx, "Watcher: Error getting relative path", "library", lib.ID, "libPath", absLibPath, "path", path, err)
+ log.Error(ctx, "Error getting relative path", "libraryID", lib.ID, "absolutePath", absLibPath, "path", path, err)
continue
}
+
if isIgnoredPath(ctx, fsys, path) {
- log.Trace(ctx, "Watcher: Ignoring change", "library", lib.ID, "path", path)
+ log.Trace(ctx, "Ignoring change", "libraryID", lib.ID, "path", path)
continue
}
- log.Trace(ctx, "Watcher: Detected change", "library", lib.ID, "path", path, "libPath", absLibPath)
- watchChan <- struct{}{}
+ log.Trace(ctx, "Detected change", "libraryID", lib.ID, "path", path, "absoluteLibPath", absLibPath)
+
+ // Check if the original path (before resolution) matches .ndignore patterns
+ // This is crucial for deleted folders - if a deleted folder matches .ndignore,
+ // we should ignore it BEFORE resolveFolderPath walks up to the parent
+ if w.shouldIgnoreFolderPath(ctx, fsys, path) {
+ log.Debug(ctx, "Ignoring change matching .ndignore pattern", "libraryID", lib.ID, "path", path)
+ continue
+ }
+
+ // Find the folder to scan - validate path exists as directory, walk up if needed
+ folderPath := resolveFolderPath(fsys, path)
+ // Double-check after resolution in case the resolved path is different and also matches patterns
+ if folderPath != path && w.shouldIgnoreFolderPath(ctx, fsys, folderPath) {
+ log.Trace(ctx, "Ignoring change in folder matching .ndignore pattern", "libraryID", lib.ID, "folderPath", folderPath)
+ continue
+ }
+
+ // Notify the main watcher of changes
+ select {
+ case w.watcherNotify <- scanNotification{Library: lib, FolderPath: folderPath}:
+ default:
+ // Channel is full, notification already pending
+ }
}
}
}
+// resolveFolderPath takes a path (which may be a file or directory) and returns
+// the folder path to scan. If the path is a file, it walks up to find the parent
+// directory. Returns empty string if the path should scan the library root.
+func resolveFolderPath(fsys fs.FS, path string) string {
+ // Handle root paths immediately
+ if path == "." || path == "" {
+ return ""
+ }
+
+ folderPath := path
+ for {
+ info, err := fs.Stat(fsys, folderPath)
+ if err == nil && info.IsDir() {
+ // Found a valid directory
+ return folderPath
+ }
+ if folderPath == "." || folderPath == "" {
+ // Reached root, scan entire library
+ return ""
+ }
+ // Walk up the tree
+ dir, _ := filepath.Split(folderPath)
+ if dir == "" || dir == "." {
+ return ""
+ }
+ // Remove trailing slash
+ folderPath = filepath.Clean(dir)
+ }
+}
+
+// shouldIgnoreFolderPath checks if the given folderPath should be ignored based on .ndignore patterns
+// in the library. It pushes all parent folders onto the IgnoreChecker stack before checking.
+func (w *watcher) shouldIgnoreFolderPath(ctx context.Context, fsys storage.MusicFS, folderPath string) bool {
+ checker := newIgnoreChecker(fsys)
+ err := checker.PushAllParents(ctx, folderPath)
+ if err != nil {
+ log.Warn(ctx, "Watcher: Error pushing ignore patterns for folder", "path", folderPath, err)
+ }
+ return checker.ShouldIgnore(ctx, folderPath)
+}
+
func isIgnoredPath(_ context.Context, _ fs.FS, path string) bool {
baseDir, name := filepath.Split(path)
switch {
diff --git a/scanner/watcher_test.go b/scanner/watcher_test.go
new file mode 100644
index 000000000..01bfb2491
--- /dev/null
+++ b/scanner/watcher_test.go
@@ -0,0 +1,491 @@
+package scanner
+
+import (
+ "context"
+ "io/fs"
+ "path/filepath"
+ "testing/fstest"
+ "time"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("Watcher", func() {
+ var ctx context.Context
+ var cancel context.CancelFunc
+ var mockScanner *tests.MockScanner
+ var mockDS *tests.MockDataStore
+ var w *watcher
+ var lib *model.Library
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ conf.Server.Scanner.WatcherWait = 50 * time.Millisecond // Short wait for tests
+
+ ctx, cancel = context.WithCancel(context.Background())
+ DeferCleanup(cancel)
+
+ lib = &model.Library{
+ ID: 1,
+ Name: "Test Library",
+ Path: "/test/library",
+ }
+
+ // Set up mocks
+ mockScanner = tests.NewMockScanner()
+ mockDS = &tests.MockDataStore{}
+ mockLibRepo := &tests.MockLibraryRepo{}
+ mockLibRepo.SetData(model.Libraries{*lib})
+ mockDS.MockedLibrary = mockLibRepo
+
+ // Create a new watcher instance (not singleton) for testing
+ w = &watcher{
+ ds: mockDS,
+ scanner: mockScanner,
+ triggerWait: conf.Server.Scanner.WatcherWait,
+ watcherNotify: make(chan scanNotification, 10),
+ libraryWatchers: make(map[int]*libraryWatcherInstance),
+ mainCtx: ctx,
+ }
+ })
+
+ Describe("Target Collection and Deduplication", func() {
+ BeforeEach(func() {
+ // Start watcher in background
+ go func() {
+ _ = w.Run(ctx)
+ }()
+
+ // Give watcher time to initialize
+ time.Sleep(10 * time.Millisecond)
+ })
+
+ It("creates separate targets for different folders", func() {
+ // Send notifications for different folders
+ w.watcherNotify <- scanNotification{Library: lib, FolderPath: "artist1"}
+ time.Sleep(10 * time.Millisecond)
+ w.watcherNotify <- scanNotification{Library: lib, FolderPath: "artist2"}
+
+ // Wait for watcher to process and trigger scan
+ Eventually(func() int {
+ return mockScanner.GetScanFoldersCallCount()
+ }, 200*time.Millisecond, 10*time.Millisecond).Should(Equal(1))
+
+ // Verify two targets
+ calls := mockScanner.GetScanFoldersCalls()
+ Expect(calls).To(HaveLen(1))
+ Expect(calls[0].Targets).To(HaveLen(2))
+
+ // Extract folder paths
+ folderPaths := make(map[string]bool)
+ for _, target := range calls[0].Targets {
+ Expect(target.LibraryID).To(Equal(1))
+ folderPaths[target.FolderPath] = true
+ }
+ Expect(folderPaths).To(HaveKey("artist1"))
+ Expect(folderPaths).To(HaveKey("artist2"))
+ })
+
+ It("handles different folder paths correctly", func() {
+ // Send notification for nested folder
+ w.watcherNotify <- scanNotification{Library: lib, FolderPath: "artist1/album1"}
+
+ // Wait for watcher to process and trigger scan
+ Eventually(func() int {
+ return mockScanner.GetScanFoldersCallCount()
+ }, 200*time.Millisecond, 10*time.Millisecond).Should(Equal(1))
+
+ // Verify the target
+ calls := mockScanner.GetScanFoldersCalls()
+ Expect(calls).To(HaveLen(1))
+ Expect(calls[0].Targets).To(HaveLen(1))
+ Expect(calls[0].Targets[0].FolderPath).To(Equal("artist1/album1"))
+ })
+
+ It("deduplicates folder and file within same folder", func() {
+ // Send notification for a folder
+ w.watcherNotify <- scanNotification{Library: lib, FolderPath: "artist1/album1"}
+ time.Sleep(10 * time.Millisecond)
+ // Send notification for same folder (as if file change was detected there)
+ // In practice, watchLibrary() would walk up from file path to folder
+ w.watcherNotify <- scanNotification{Library: lib, FolderPath: "artist1/album1"}
+ time.Sleep(10 * time.Millisecond)
+ // Send another for same folder
+ w.watcherNotify <- scanNotification{Library: lib, FolderPath: "artist1/album1"}
+
+ // Wait for watcher to process and trigger scan
+ Eventually(func() int {
+ return mockScanner.GetScanFoldersCallCount()
+ }, 200*time.Millisecond, 10*time.Millisecond).Should(Equal(1))
+
+ // Verify only one target despite multiple file/folder changes
+ calls := mockScanner.GetScanFoldersCalls()
+ Expect(calls).To(HaveLen(1))
+ Expect(calls[0].Targets).To(HaveLen(1))
+ Expect(calls[0].Targets[0].FolderPath).To(Equal("artist1/album1"))
+ })
+ })
+
+ Describe("Timer Behavior", func() {
+ BeforeEach(func() {
+ // Start watcher in background
+ go func() {
+ _ = w.Run(ctx)
+ }()
+
+ // Give watcher time to initialize
+ time.Sleep(10 * time.Millisecond)
+ })
+
+ It("resets timer on each change (debouncing)", func() {
+ // Send first notification
+ w.watcherNotify <- scanNotification{Library: lib, FolderPath: "artist1"}
+
+ // Wait a bit less than half the watcher wait time to ensure timer doesn't fire
+ time.Sleep(20 * time.Millisecond)
+
+ // No scan should have been triggered yet
+ Expect(mockScanner.GetScanFoldersCallCount()).To(Equal(0))
+
+ // Send another notification (resets timer)
+ w.watcherNotify <- scanNotification{Library: lib, FolderPath: "artist1"}
+
+ // Wait a bit less than half the watcher wait time again
+ time.Sleep(20 * time.Millisecond)
+
+ // Still no scan
+ Expect(mockScanner.GetScanFoldersCallCount()).To(Equal(0))
+
+ // Wait for full timer to expire after last notification (plus margin)
+ time.Sleep(60 * time.Millisecond)
+
+ // Now scan should have been triggered
+ Eventually(func() int {
+ return mockScanner.GetScanFoldersCallCount()
+ }, 100*time.Millisecond, 10*time.Millisecond).Should(Equal(1))
+ })
+
+ It("triggers scan after quiet period", func() {
+ // Send notification
+ w.watcherNotify <- scanNotification{Library: lib, FolderPath: "artist1"}
+
+ // No scan immediately
+ Expect(mockScanner.GetScanFoldersCallCount()).To(Equal(0))
+
+ // Wait for quiet period
+ Eventually(func() int {
+ return mockScanner.GetScanFoldersCallCount()
+ }, 200*time.Millisecond, 10*time.Millisecond).Should(Equal(1))
+ })
+ })
+
+ Describe("Empty and Root Paths", func() {
+ BeforeEach(func() {
+ // Start watcher in background
+ go func() {
+ _ = w.Run(ctx)
+ }()
+
+ // Give watcher time to initialize
+ time.Sleep(10 * time.Millisecond)
+ })
+
+ It("handles empty folder path (library root)", func() {
+ // Send notification with empty folder path
+ w.watcherNotify <- scanNotification{Library: lib, FolderPath: ""}
+
+ // Wait for scan
+ Eventually(func() int {
+ return mockScanner.GetScanFoldersCallCount()
+ }, 200*time.Millisecond, 10*time.Millisecond).Should(Equal(1))
+
+ // Should scan the library root
+ calls := mockScanner.GetScanFoldersCalls()
+ Expect(calls).To(HaveLen(1))
+ Expect(calls[0].Targets).To(HaveLen(1))
+ Expect(calls[0].Targets[0].FolderPath).To(Equal(""))
+ })
+
+ It("deduplicates empty and dot paths", func() {
+ // Send notifications with empty and dot paths
+ w.watcherNotify <- scanNotification{Library: lib, FolderPath: ""}
+ time.Sleep(10 * time.Millisecond)
+ w.watcherNotify <- scanNotification{Library: lib, FolderPath: ""}
+
+ // Wait for scan
+ Eventually(func() int {
+ return mockScanner.GetScanFoldersCallCount()
+ }, 200*time.Millisecond, 10*time.Millisecond).Should(Equal(1))
+
+ // Should have only one target
+ calls := mockScanner.GetScanFoldersCalls()
+ Expect(calls).To(HaveLen(1))
+ Expect(calls[0].Targets).To(HaveLen(1))
+ })
+ })
+
+ Describe("Multiple Libraries", func() {
+ var lib2 *model.Library
+
+ BeforeEach(func() {
+ // Create second library
+ lib2 = &model.Library{
+ ID: 2,
+ Name: "Test Library 2",
+ Path: "/test/library2",
+ }
+
+ mockLibRepo := mockDS.MockedLibrary.(*tests.MockLibraryRepo)
+ mockLibRepo.SetData(model.Libraries{*lib, *lib2})
+
+ // Start watcher in background
+ go func() {
+ _ = w.Run(ctx)
+ }()
+
+ // Give watcher time to initialize
+ time.Sleep(10 * time.Millisecond)
+ })
+
+ It("creates separate targets for different libraries", func() {
+ // Send notifications for both libraries
+ w.watcherNotify <- scanNotification{Library: lib, FolderPath: "artist1"}
+ time.Sleep(10 * time.Millisecond)
+ w.watcherNotify <- scanNotification{Library: lib2, FolderPath: "artist2"}
+
+ // Wait for scan
+ Eventually(func() int {
+ return mockScanner.GetScanFoldersCallCount()
+ }, 200*time.Millisecond, 10*time.Millisecond).Should(Equal(1))
+
+ // Verify two targets for different libraries
+ calls := mockScanner.GetScanFoldersCalls()
+ Expect(calls).To(HaveLen(1))
+ Expect(calls[0].Targets).To(HaveLen(2))
+
+ // Verify library IDs are different
+ libraryIDs := make(map[int]bool)
+ for _, target := range calls[0].Targets {
+ libraryIDs[target.LibraryID] = true
+ }
+ Expect(libraryIDs).To(HaveKey(1))
+ Expect(libraryIDs).To(HaveKey(2))
+ })
+ })
+
+ Describe(".ndignore handling", func() {
+ var ctx context.Context
+ var cancel context.CancelFunc
+ var w *watcher
+ var mockFS *mockMusicFS
+ var lib *model.Library
+ var eventChan chan string
+ var absLibPath string
+
+ BeforeEach(func() {
+ ctx, cancel = context.WithCancel(GinkgoT().Context())
+ DeferCleanup(cancel)
+
+ // Set up library
+ var err error
+ absLibPath, err = filepath.Abs(".")
+ Expect(err).NotTo(HaveOccurred())
+
+ lib = &model.Library{
+ ID: 1,
+ Name: "Test Library",
+ Path: absLibPath,
+ }
+
+ // Create watcher with notification channel
+ w = &watcher{
+ watcherNotify: make(chan scanNotification, 10),
+ }
+
+ eventChan = make(chan string, 10)
+ })
+
+ // Helper to send an event - converts relative path to absolute
+ sendEvent := func(relativePath string) {
+ path := filepath.Join(absLibPath, relativePath)
+ eventChan <- path
+ }
+
+ // Helper to start the real event processing loop
+ startEventProcessing := func() {
+ go func() {
+ defer GinkgoRecover()
+ // Call the actual processLibraryEvents method - testing the real implementation!
+ _ = w.processLibraryEvents(ctx, lib, mockFS, eventChan, absLibPath)
+ }()
+ }
+
+ Context("when a folder matching .ndignore is deleted", func() {
+ BeforeEach(func() {
+ // Create filesystem with .ndignore containing _TEMP pattern
+ // The deleted folder (_TEMP) will NOT exist in the filesystem
+ mockFS = &mockMusicFS{
+ FS: fstest.MapFS{
+ "rock": &fstest.MapFile{Mode: fs.ModeDir},
+ "rock/.ndignore": &fstest.MapFile{Data: []byte("_TEMP\n")},
+ "rock/valid_album": &fstest.MapFile{Mode: fs.ModeDir},
+ "rock/valid_album/track.mp3": &fstest.MapFile{Data: []byte("audio")},
+ },
+ }
+ })
+
+ It("should NOT send scan notification when deleted folder matches .ndignore", func() {
+ startEventProcessing()
+
+ // Simulate deletion event for rock/_TEMP
+ sendEvent("rock/_TEMP")
+
+ // Wait a bit to ensure event is processed
+ time.Sleep(50 * time.Millisecond)
+
+ // No notification should have been sent
+ Consistently(eventChan, 100*time.Millisecond).Should(BeEmpty())
+ })
+
+ It("should send scan notification for valid folder deletion", func() {
+ startEventProcessing()
+
+ // Simulate deletion event for rock/other_folder (not in .ndignore and doesn't exist)
+ // Since it doesn't exist in mockFS, resolveFolderPath will walk up to "rock"
+ sendEvent("rock/other_folder")
+
+ // Should receive notification for parent folder
+ Eventually(w.watcherNotify, 200*time.Millisecond).Should(Receive(Equal(scanNotification{
+ Library: lib,
+ FolderPath: "rock",
+ })))
+ })
+ })
+
+ Context("with nested folder patterns", func() {
+ BeforeEach(func() {
+ mockFS = &mockMusicFS{
+ FS: fstest.MapFS{
+ "music": &fstest.MapFile{Mode: fs.ModeDir},
+ "music/.ndignore": &fstest.MapFile{Data: []byte("**/temp\n**/cache\n")},
+ "music/rock": &fstest.MapFile{Mode: fs.ModeDir},
+ "music/rock/artist": &fstest.MapFile{Mode: fs.ModeDir},
+ },
+ }
+ })
+
+ It("should NOT send notification when nested ignored folder is deleted", func() {
+ startEventProcessing()
+
+ // Simulate deletion of music/rock/artist/temp (matches **/temp)
+ sendEvent("music/rock/artist/temp")
+
+ // Wait to ensure event is processed
+ time.Sleep(50 * time.Millisecond)
+
+ // No notification should be sent
+ Expect(w.watcherNotify).To(BeEmpty(), "Expected no scan notification for nested ignored folder")
+ })
+
+ It("should send notification for non-ignored nested folder", func() {
+ startEventProcessing()
+
+ // Simulate change in music/rock/artist (doesn't match any pattern)
+ sendEvent("music/rock/artist")
+
+ // Should receive notification
+ Eventually(w.watcherNotify, 200*time.Millisecond).Should(Receive(Equal(scanNotification{
+ Library: lib,
+ FolderPath: "music/rock/artist",
+ })))
+ })
+ })
+
+ Context("with file events in ignored folders", func() {
+ BeforeEach(func() {
+ mockFS = &mockMusicFS{
+ FS: fstest.MapFS{
+ "rock": &fstest.MapFile{Mode: fs.ModeDir},
+ "rock/.ndignore": &fstest.MapFile{Data: []byte("_TEMP\n")},
+ },
+ }
+ })
+
+ It("should NOT send notification for file changes in ignored folders", func() {
+ startEventProcessing()
+
+ // Simulate file change in rock/_TEMP/file.mp3
+ sendEvent("rock/_TEMP/file.mp3")
+
+ // Wait to ensure event is processed
+ time.Sleep(50 * time.Millisecond)
+
+ // No notification should be sent
+ Expect(w.watcherNotify).To(BeEmpty(), "Expected no scan notification for file in ignored folder")
+ })
+ })
+ })
+})
+
+var _ = Describe("resolveFolderPath", func() {
+ var mockFS fs.FS
+
+ BeforeEach(func() {
+ // Create a mock filesystem with some directories and files
+ mockFS = fstest.MapFS{
+ "artist1": &fstest.MapFile{Mode: fs.ModeDir},
+ "artist1/album1": &fstest.MapFile{Mode: fs.ModeDir},
+ "artist1/album1/track1.mp3": &fstest.MapFile{Data: []byte("audio")},
+ "artist1/album1/track2.mp3": &fstest.MapFile{Data: []byte("audio")},
+ "artist1/album2": &fstest.MapFile{Mode: fs.ModeDir},
+ "artist1/album2/song.flac": &fstest.MapFile{Data: []byte("audio")},
+ "artist2": &fstest.MapFile{Mode: fs.ModeDir},
+ "artist2/cover.jpg": &fstest.MapFile{Data: []byte("image")},
+ }
+ })
+
+ It("returns directory path when given a directory", func() {
+ result := resolveFolderPath(mockFS, "artist1/album1")
+ Expect(result).To(Equal("artist1/album1"))
+ })
+
+ It("walks up to parent directory when given a file path", func() {
+ result := resolveFolderPath(mockFS, "artist1/album1/track1.mp3")
+ Expect(result).To(Equal("artist1/album1"))
+ })
+
+ It("walks up multiple levels if needed", func() {
+ result := resolveFolderPath(mockFS, "artist1/album1/nonexistent/file.mp3")
+ Expect(result).To(Equal("artist1/album1"))
+ })
+
+ It("returns empty string for non-existent paths at root", func() {
+ result := resolveFolderPath(mockFS, "nonexistent/path/file.mp3")
+ Expect(result).To(Equal(""))
+ })
+
+ It("returns empty string for dot path", func() {
+ result := resolveFolderPath(mockFS, ".")
+ Expect(result).To(Equal(""))
+ })
+
+ It("returns empty string for empty path", func() {
+ result := resolveFolderPath(mockFS, "")
+ Expect(result).To(Equal(""))
+ })
+
+ It("handles nested file paths correctly", func() {
+ result := resolveFolderPath(mockFS, "artist1/album2/song.flac")
+ Expect(result).To(Equal("artist1/album2"))
+ })
+
+ It("resolves to top-level directory", func() {
+ result := resolveFolderPath(mockFS, "artist2/cover.jpg")
+ Expect(result).To(Equal("artist2"))
+ })
+})
diff --git a/scheduler/scheduler.go b/scheduler/scheduler.go
index 062bf4344..b377e7947 100644
--- a/scheduler/scheduler.go
+++ b/scheduler/scheduler.go
@@ -9,7 +9,8 @@ import (
type Scheduler interface {
Run(ctx context.Context)
- Add(crontab string, cmd func()) error
+ Add(crontab string, cmd func()) (int, error)
+ Remove(id int)
}
func GetInstance() Scheduler {
@@ -31,7 +32,14 @@ func (s *scheduler) Run(ctx context.Context) {
s.c.Stop()
}
-func (s *scheduler) Add(crontab string, cmd func()) error {
- _, err := s.c.AddFunc(crontab, cmd)
- return err
+func (s *scheduler) Add(crontab string, cmd func()) (int, error) {
+ entryID, err := s.c.AddFunc(crontab, cmd)
+ if err != nil {
+ return 0, err
+ }
+ return int(entryID), nil
+}
+
+func (s *scheduler) Remove(id int) {
+ s.c.Remove(cron.EntryID(id))
}
diff --git a/scheduler/scheduler_test.go b/scheduler/scheduler_test.go
new file mode 100644
index 000000000..4737ae389
--- /dev/null
+++ b/scheduler/scheduler_test.go
@@ -0,0 +1,86 @@
+package scheduler
+
+import (
+ "sync"
+ "testing"
+ "time"
+
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+ "github.com/robfig/cron/v3"
+)
+
+func TestScheduler(t *testing.T) {
+ tests.Init(t, false)
+ log.SetLevel(log.LevelFatal)
+ RegisterFailHandler(Fail)
+ RunSpecs(t, "Scheduler Suite")
+}
+
+var _ = Describe("Scheduler", func() {
+ var s *scheduler
+
+ BeforeEach(func() {
+ c := cron.New(cron.WithLogger(&logger{}))
+ s = &scheduler{c: c}
+ s.c.Start() // Start the scheduler for tests
+ })
+
+ AfterEach(func() {
+ s.c.Stop() // Stop the scheduler after tests
+ })
+
+ It("adds and executes a job", func() {
+ wg := sync.WaitGroup{}
+ wg.Add(1)
+
+ executed := false
+ id, err := s.Add("@every 100ms", func() {
+ executed = true
+ wg.Done()
+ })
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(id).ToNot(BeZero())
+
+ wg.Wait()
+ Expect(executed).To(BeTrue())
+ })
+
+ It("removes a job", func() {
+ // Use a WaitGroup to ensure the job executes once
+ wg := sync.WaitGroup{}
+ wg.Add(1)
+
+ counter := 0
+ id, err := s.Add("@every 100ms", func() {
+ counter++
+ if counter == 1 {
+ wg.Done() // Signal that the job has executed once
+ }
+ })
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(id).ToNot(BeZero())
+
+ // Wait for the job to execute at least once
+ wg.Wait()
+
+ // Verify job executed
+ Expect(counter).To(Equal(1))
+
+ // Remove the job
+ s.Remove(id)
+
+ // Store the counter value
+ currentCount := counter
+
+ // Wait some time to ensure job doesn't execute again
+ time.Sleep(200 * time.Millisecond)
+
+ // Verify counter didn't increase
+ Expect(counter).To(Equal(currentCount))
+ })
+})
diff --git a/server/auth.go b/server/auth.go
index 5b35f72ed..ed43974dd 100644
--- a/server/auth.go
+++ b/server/auth.go
@@ -171,7 +171,7 @@ func validateLogin(userRepo model.UserRepository, userName, password string) (*m
return u, nil
}
-func jwtVerifier(next http.Handler) http.Handler {
+func JWTVerifier(next http.Handler) http.Handler {
return jwtauth.Verify(auth.TokenAuth, tokenFromHeader, jwtauth.TokenFromCookie, jwtauth.TokenFromQuery)(next)
}
@@ -214,6 +214,15 @@ func UsernameFromReverseProxyHeader(r *http.Request) string {
return username
}
+func InternalAuth(r *http.Request) string {
+ username, ok := request.InternalAuthFrom(r.Context())
+ if !ok {
+ return ""
+ }
+ log.Trace(r, "Found username in InternalAuth", "username", username)
+ return username
+}
+
func UsernameFromConfig(*http.Request) string {
return conf.Server.DevAutoLoginUsername
}
diff --git a/server/events/events.go b/server/events/events.go
index 73ff8eb5e..ff0a8a40a 100644
--- a/server/events/events.go
+++ b/server/events/events.go
@@ -13,8 +13,8 @@ type eventCtxKey string
const broadcastToAllKey eventCtxKey = "broadcastToAll"
-// BroadcastToAll is a context key that can be used to broadcast an event to all clients
-func BroadcastToAll(ctx context.Context) context.Context {
+// broadcastToAll is a context key that can be used to broadcast an event to all clients
+func broadcastToAll(ctx context.Context) context.Context {
return context.WithValue(ctx, broadcastToAllKey, true)
}
@@ -63,6 +63,11 @@ type RefreshResource struct {
resources map[string][]string
}
+type NowPlayingCount struct {
+ baseEvent
+ Count int `json:"count"`
+}
+
func (rr *RefreshResource) With(resource string, ids ...string) *RefreshResource {
if rr.resources == nil {
rr.resources = make(map[string][]string)
diff --git a/server/events/sse.go b/server/events/sse.go
index 690c79937..54a602985 100644
--- a/server/events/sse.go
+++ b/server/events/sse.go
@@ -19,6 +19,7 @@ import (
type Broker interface {
http.Handler
SendMessage(ctx context.Context, event Event)
+ SendBroadcastMessage(ctx context.Context, event Event)
}
const (
@@ -77,6 +78,11 @@ func GetBroker() Broker {
})
}
+func (b *broker) SendBroadcastMessage(ctx context.Context, evt Event) {
+ ctx = broadcastToAll(ctx)
+ b.SendMessage(ctx, evt)
+}
+
func (b *broker) SendMessage(ctx context.Context, evt Event) {
msg := b.prepareMessage(ctx, evt)
log.Trace("Broker received new event", "type", msg.event, "data", msg.data)
@@ -280,4 +286,6 @@ type noopBroker struct {
http.Handler
}
+func (b noopBroker) SendBroadcastMessage(context.Context, Event) {}
+
func (noopBroker) SendMessage(context.Context, Event) {}
diff --git a/server/nativeapi/config.go b/server/nativeapi/config.go
new file mode 100644
index 000000000..9a86a9add
--- /dev/null
+++ b/server/nativeapi/config.go
@@ -0,0 +1,132 @@
+package nativeapi
+
+import (
+ "context"
+ "encoding/json"
+ "fmt"
+ "net/http"
+ "strings"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/log"
+)
+
+// sensitiveFieldsPartialMask contains configuration field names that should be redacted
+// using partial masking (first and last character visible, middle replaced with *).
+// For values with 7+ characters: "secretvalue123" becomes "s***********3"
+// For values with <7 characters: "short" becomes "****"
+// Add field paths using dot notation (e.g., "LastFM.ApiKey", "Spotify.Secret")
+var sensitiveFieldsPartialMask = []string{
+ "LastFM.ApiKey",
+ "LastFM.Secret",
+ "Prometheus.MetricsPath",
+ "Spotify.ID",
+ "Spotify.Secret",
+ "DevAutoLoginUsername",
+}
+
+// sensitiveFieldsFullMask contains configuration field names that should always be
+// completely masked with "****" regardless of their length.
+// Add field paths using dot notation for any fields that should never show any content.
+var sensitiveFieldsFullMask = []string{
+ "DevAutoCreateAdminPassword",
+ "PasswordEncryptionKey",
+ "Prometheus.Password",
+}
+
+type configResponse struct {
+ ID string `json:"id"`
+ ConfigFile string `json:"configFile"`
+ Config map[string]interface{} `json:"config"`
+}
+
+func redactValue(key string, value string) string {
+ // Return empty values as-is
+ if len(value) == 0 {
+ return value
+ }
+
+ // Check if this field should be fully masked
+ for _, field := range sensitiveFieldsFullMask {
+ if field == key {
+ return "****"
+ }
+ }
+
+ // Check if this field should be partially masked
+ for _, field := range sensitiveFieldsPartialMask {
+ if field == key {
+ if len(value) < 7 {
+ return "****"
+ }
+ // Show first and last character with * in between
+ return string(value[0]) + strings.Repeat("*", len(value)-2) + string(value[len(value)-1])
+ }
+ }
+
+ // Return original value if not sensitive
+ return value
+}
+
+// applySensitiveFieldMasking recursively applies masking to sensitive fields in the configuration map
+func applySensitiveFieldMasking(ctx context.Context, config map[string]interface{}, prefix string) {
+ for key, value := range config {
+ fullKey := key
+ if prefix != "" {
+ fullKey = prefix + "." + key
+ }
+
+ switch v := value.(type) {
+ case map[string]interface{}:
+ // Recursively process nested maps
+ applySensitiveFieldMasking(ctx, v, fullKey)
+ case string:
+ // Apply masking to string values
+ config[key] = redactValue(fullKey, v)
+ default:
+ // For other types (numbers, booleans, etc.), convert to string and check for masking
+ if str := fmt.Sprint(v); str != "" {
+ masked := redactValue(fullKey, str)
+ if masked != str {
+ // Only replace if masking was applied
+ config[key] = masked
+ }
+ }
+ }
+ }
+}
+
+func getConfig(w http.ResponseWriter, r *http.Request) {
+ ctx := r.Context()
+
+ // Marshal the actual configuration struct to preserve original field names
+ configBytes, err := json.Marshal(*conf.Server)
+ if err != nil {
+ log.Error(ctx, "Error marshaling config", err)
+ http.Error(w, "Internal server error", http.StatusInternalServerError)
+ return
+ }
+
+ // Unmarshal back to map to get the structure with proper field names
+ var configMap map[string]interface{}
+ err = json.Unmarshal(configBytes, &configMap)
+ if err != nil {
+ log.Error(ctx, "Error unmarshaling config to map", err)
+ http.Error(w, "Internal server error", http.StatusInternalServerError)
+ return
+ }
+
+ // Apply sensitive field masking
+ applySensitiveFieldMasking(ctx, configMap, "")
+
+ resp := configResponse{
+ ID: "config",
+ ConfigFile: conf.Server.ConfigFile,
+ Config: configMap,
+ }
+
+ w.Header().Set("Content-Type", "application/json")
+ if err := json.NewEncoder(w).Encode(resp); err != nil {
+ log.Error(ctx, "Error encoding config response", err)
+ }
+}
diff --git a/server/nativeapi/config_test.go b/server/nativeapi/config_test.go
new file mode 100644
index 000000000..d9c722955
--- /dev/null
+++ b/server/nativeapi/config_test.go
@@ -0,0 +1,227 @@
+package nativeapi
+
+import (
+ "bytes"
+ "context"
+ "encoding/json"
+ "net/http"
+ "net/http/httptest"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/consts"
+ "github.com/navidrome/navidrome/core"
+ "github.com/navidrome/navidrome/core/auth"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/server"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("Config API", func() {
+ var ds model.DataStore
+ var router http.Handler
+ var adminUser, regularUser model.User
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ conf.Server.DevUIShowConfig = true // Enable config endpoint for tests
+ ds = &tests.MockDataStore{}
+ auth.Init(ds)
+ nativeRouter := New(ds, nil, nil, nil, core.NewMockLibraryService(), nil)
+ router = server.JWTVerifier(nativeRouter)
+
+ // Create test users
+ adminUser = model.User{
+ ID: "admin-1",
+ UserName: "admin",
+ Name: "Admin User",
+ IsAdmin: true,
+ NewPassword: "adminpass",
+ }
+ regularUser = model.User{
+ ID: "user-1",
+ UserName: "regular",
+ Name: "Regular User",
+ IsAdmin: false,
+ NewPassword: "userpass",
+ }
+
+ // Store in mock datastore
+ Expect(ds.User(context.TODO()).Put(&adminUser)).To(Succeed())
+ Expect(ds.User(context.TODO()).Put(®ularUser)).To(Succeed())
+ })
+
+ Describe("GET /api/config", func() {
+ Context("as admin user", func() {
+ var adminToken string
+
+ BeforeEach(func() {
+ var err error
+ adminToken, err = auth.CreateToken(&adminUser)
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ It("returns config successfully", func() {
+ req := createAuthenticatedConfigRequest(adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+ var resp configResponse
+ Expect(json.Unmarshal(w.Body.Bytes(), &resp)).To(Succeed())
+ Expect(resp.ID).To(Equal("config"))
+ Expect(resp.ConfigFile).To(Equal(conf.Server.ConfigFile))
+ Expect(resp.Config).ToNot(BeEmpty())
+ })
+
+ It("redacts sensitive fields", func() {
+ conf.Server.LastFM.ApiKey = "secretapikey123"
+ conf.Server.Spotify.Secret = "spotifysecret456"
+ conf.Server.PasswordEncryptionKey = "encryptionkey789"
+ conf.Server.DevAutoCreateAdminPassword = "adminpassword123"
+ conf.Server.Prometheus.Password = "prometheuspass"
+
+ req := createAuthenticatedConfigRequest(adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+ var resp configResponse
+ Expect(json.Unmarshal(w.Body.Bytes(), &resp)).To(Succeed())
+
+ // Check LastFM.ApiKey (partially masked)
+ lastfm, ok := resp.Config["LastFM"].(map[string]interface{})
+ Expect(ok).To(BeTrue())
+ Expect(lastfm["ApiKey"]).To(Equal("s*************3"))
+
+ // Check Spotify.Secret (partially masked)
+ spotify, ok := resp.Config["Spotify"].(map[string]interface{})
+ Expect(ok).To(BeTrue())
+ Expect(spotify["Secret"]).To(Equal("s**************6"))
+
+ // Check PasswordEncryptionKey (fully masked)
+ Expect(resp.Config["PasswordEncryptionKey"]).To(Equal("****"))
+
+ // Check DevAutoCreateAdminPassword (fully masked)
+ Expect(resp.Config["DevAutoCreateAdminPassword"]).To(Equal("****"))
+
+ // Check Prometheus.Password (fully masked)
+ prometheus, ok := resp.Config["Prometheus"].(map[string]interface{})
+ Expect(ok).To(BeTrue())
+ Expect(prometheus["Password"]).To(Equal("****"))
+ })
+
+ It("handles empty sensitive values", func() {
+ conf.Server.LastFM.ApiKey = ""
+ conf.Server.PasswordEncryptionKey = ""
+
+ req := createAuthenticatedConfigRequest(adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+ var resp configResponse
+ Expect(json.Unmarshal(w.Body.Bytes(), &resp)).To(Succeed())
+
+ // Check LastFM.ApiKey - should be preserved because it's sensitive
+ lastfm, ok := resp.Config["LastFM"].(map[string]interface{})
+ Expect(ok).To(BeTrue())
+ Expect(lastfm["ApiKey"]).To(Equal(""))
+
+ // Empty sensitive values should remain empty - should be preserved because it's sensitive
+ Expect(resp.Config["PasswordEncryptionKey"]).To(Equal(""))
+ })
+ })
+
+ Context("as regular user", func() {
+ var userToken string
+
+ BeforeEach(func() {
+ var err error
+ userToken, err = auth.CreateToken(®ularUser)
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ It("denies access with forbidden status", func() {
+ req := createAuthenticatedConfigRequest(userToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusForbidden))
+ })
+ })
+
+ Context("without authentication", func() {
+ It("denies access with unauthorized status", func() {
+ req := createUnauthenticatedConfigRequest("GET", "/config/", nil)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusUnauthorized))
+ })
+ })
+ })
+})
+
+var _ = Describe("redactValue function", func() {
+ It("partially masks long sensitive values", func() {
+ Expect(redactValue("LastFM.ApiKey", "ba46f0e84a")).To(Equal("b********a"))
+ Expect(redactValue("Spotify.Secret", "verylongsecret123")).To(Equal("v***************3"))
+ })
+
+ It("fully masks long sensitive values that should be completely hidden", func() {
+ Expect(redactValue("PasswordEncryptionKey", "1234567890")).To(Equal("****"))
+ Expect(redactValue("DevAutoCreateAdminPassword", "1234567890")).To(Equal("****"))
+ Expect(redactValue("Prometheus.Password", "1234567890")).To(Equal("****"))
+ })
+
+ It("fully masks short sensitive values", func() {
+ Expect(redactValue("LastFM.Secret", "short")).To(Equal("****"))
+ Expect(redactValue("Spotify.ID", "abc")).To(Equal("****"))
+ Expect(redactValue("PasswordEncryptionKey", "12345")).To(Equal("****"))
+ Expect(redactValue("DevAutoCreateAdminPassword", "short")).To(Equal("****"))
+ Expect(redactValue("Prometheus.Password", "short")).To(Equal("****"))
+ })
+
+ It("does not mask non-sensitive values", func() {
+ Expect(redactValue("MusicFolder", "/path/to/music")).To(Equal("/path/to/music"))
+ Expect(redactValue("Port", "4533")).To(Equal("4533"))
+ Expect(redactValue("SomeOtherField", "secretvalue")).To(Equal("secretvalue"))
+ })
+
+ It("handles empty values", func() {
+ Expect(redactValue("LastFM.ApiKey", "")).To(Equal(""))
+ Expect(redactValue("NonSensitive", "")).To(Equal(""))
+ })
+
+ It("handles edge case values", func() {
+ Expect(redactValue("LastFM.ApiKey", "a")).To(Equal("****"))
+ Expect(redactValue("LastFM.ApiKey", "ab")).To(Equal("****"))
+ Expect(redactValue("LastFM.ApiKey", "abcdefg")).To(Equal("a*****g"))
+ })
+})
+
+// Helper functions
+
+func createAuthenticatedConfigRequest(token string) *http.Request {
+ req := httptest.NewRequest(http.MethodGet, "/config/config", nil)
+ req.Header.Set(consts.UIAuthorizationHeader, "Bearer "+token)
+ req.Header.Set("Content-Type", "application/json")
+ return req
+}
+
+func createUnauthenticatedConfigRequest(method, path string, body *bytes.Buffer) *http.Request {
+ if body == nil {
+ body = &bytes.Buffer{}
+ }
+ req := httptest.NewRequest(method, path, body)
+ req.Header.Set("Content-Type", "application/json")
+ return req
+}
diff --git a/server/nativeapi/inspect.go b/server/nativeapi/inspect.go
index e74dc99c0..3178395ce 100644
--- a/server/nativeapi/inspect.go
+++ b/server/nativeapi/inspect.go
@@ -9,7 +9,6 @@ import (
"github.com/navidrome/navidrome/core"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
- "github.com/navidrome/navidrome/model/request"
"github.com/navidrome/navidrome/utils/req"
)
@@ -30,11 +29,6 @@ func inspect(ds model.DataStore) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
ctx := r.Context()
- user, _ := request.UserFrom(ctx)
- if !user.IsAdmin {
- http.Error(w, "Inspect is only available to admin users", http.StatusUnauthorized)
- }
-
p := req.Params(r)
id, err := p.String("id")
diff --git a/server/nativeapi/library.go b/server/nativeapi/library.go
new file mode 100644
index 000000000..1636e1dbb
--- /dev/null
+++ b/server/nativeapi/library.go
@@ -0,0 +1,101 @@
+package nativeapi
+
+import (
+ "context"
+ "encoding/json"
+ "errors"
+ "net/http"
+
+ "github.com/go-chi/chi/v5"
+ "github.com/navidrome/navidrome/core"
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+)
+
+// User-library association endpoints (admin only)
+func (api *Router) addUserLibraryRoute(r chi.Router) {
+ r.Route("/user/{id}/library", func(r chi.Router) {
+ r.Use(parseUserIDMiddleware)
+ r.Get("/", getUserLibraries(api.libs))
+ r.Put("/", setUserLibraries(api.libs))
+ })
+}
+
+// Middleware to parse user ID from URL
+func parseUserIDMiddleware(next http.Handler) http.Handler {
+ return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ userID := chi.URLParam(r, "id")
+ if userID == "" {
+ http.Error(w, "Invalid user ID", http.StatusBadRequest)
+ return
+ }
+ ctx := context.WithValue(r.Context(), "userID", userID)
+ next.ServeHTTP(w, r.WithContext(ctx))
+ })
+}
+
+// User-library association handlers
+
+func getUserLibraries(service core.Library) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ userID := r.Context().Value("userID").(string)
+
+ libraries, err := service.GetUserLibraries(r.Context(), userID)
+ if err != nil {
+ if errors.Is(err, model.ErrNotFound) {
+ http.Error(w, "User not found", http.StatusNotFound)
+ return
+ }
+ log.Error(r.Context(), "Error getting user libraries", "userID", userID, err)
+ http.Error(w, "Internal server error", http.StatusInternalServerError)
+ return
+ }
+
+ w.Header().Set("Content-Type", "application/json")
+ if err := json.NewEncoder(w).Encode(libraries); err != nil {
+ log.Error(r.Context(), "Error encoding user libraries response", err)
+ }
+ }
+}
+
+func setUserLibraries(service core.Library) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ userID := r.Context().Value("userID").(string)
+
+ var request struct {
+ LibraryIDs []int `json:"libraryIds"`
+ }
+ if err := json.NewDecoder(r.Body).Decode(&request); err != nil {
+ log.Error(r.Context(), "Error decoding request", err)
+ http.Error(w, "Invalid request body", http.StatusBadRequest)
+ return
+ }
+
+ if err := service.SetUserLibraries(r.Context(), userID, request.LibraryIDs); err != nil {
+ log.Error(r.Context(), "Error setting user libraries", "userID", userID, err)
+ if errors.Is(err, model.ErrNotFound) {
+ http.Error(w, "User not found", http.StatusNotFound)
+ return
+ }
+ if errors.Is(err, model.ErrValidation) {
+ http.Error(w, err.Error(), http.StatusBadRequest)
+ return
+ }
+ http.Error(w, "Failed to set user libraries", http.StatusInternalServerError)
+ return
+ }
+
+ // Return updated user libraries
+ libraries, err := service.GetUserLibraries(r.Context(), userID)
+ if err != nil {
+ log.Error(r.Context(), "Error getting updated user libraries", "userID", userID, err)
+ http.Error(w, "Internal server error", http.StatusInternalServerError)
+ return
+ }
+
+ w.Header().Set("Content-Type", "application/json")
+ if err := json.NewEncoder(w).Encode(libraries); err != nil {
+ log.Error(r.Context(), "Error encoding user libraries response", err)
+ }
+ }
+}
diff --git a/server/nativeapi/library_test.go b/server/nativeapi/library_test.go
new file mode 100644
index 000000000..950338492
--- /dev/null
+++ b/server/nativeapi/library_test.go
@@ -0,0 +1,424 @@
+package nativeapi
+
+import (
+ "bytes"
+ "context"
+ "encoding/json"
+ "fmt"
+ "net/http"
+ "net/http/httptest"
+ "strings"
+
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/consts"
+ "github.com/navidrome/navidrome/core"
+ "github.com/navidrome/navidrome/core/auth"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/server"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("Library API", func() {
+ var ds model.DataStore
+ var router http.Handler
+ var adminUser, regularUser model.User
+ var library1, library2 model.Library
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ ds = &tests.MockDataStore{}
+ auth.Init(ds)
+ nativeRouter := New(ds, nil, nil, nil, core.NewMockLibraryService(), nil)
+ router = server.JWTVerifier(nativeRouter)
+
+ // Create test users
+ adminUser = model.User{
+ ID: "admin-1",
+ UserName: "admin",
+ Name: "Admin User",
+ IsAdmin: true,
+ NewPassword: "adminpass",
+ }
+ regularUser = model.User{
+ ID: "user-1",
+ UserName: "regular",
+ Name: "Regular User",
+ IsAdmin: false,
+ NewPassword: "userpass",
+ }
+
+ // Create test libraries
+ library1 = model.Library{
+ ID: 1,
+ Name: "Test Library 1",
+ Path: "/music/library1",
+ }
+ library2 = model.Library{
+ ID: 2,
+ Name: "Test Library 2",
+ Path: "/music/library2",
+ }
+
+ // Store in mock datastore
+ Expect(ds.User(context.TODO()).Put(&adminUser)).To(Succeed())
+ Expect(ds.User(context.TODO()).Put(®ularUser)).To(Succeed())
+ Expect(ds.Library(context.TODO()).Put(&library1)).To(Succeed())
+ Expect(ds.Library(context.TODO()).Put(&library2)).To(Succeed())
+ })
+
+ Describe("Library CRUD Operations", func() {
+ Context("as admin user", func() {
+ var adminToken string
+
+ BeforeEach(func() {
+ var err error
+ adminToken, err = auth.CreateToken(&adminUser)
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ Describe("GET /api/library", func() {
+ It("returns all libraries", func() {
+ req := createAuthenticatedRequest("GET", "/library", nil, adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+
+ var libraries []model.Library
+ err := json.Unmarshal(w.Body.Bytes(), &libraries)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(libraries).To(HaveLen(2))
+ Expect(libraries[0].Name).To(Equal("Test Library 1"))
+ Expect(libraries[1].Name).To(Equal("Test Library 2"))
+ })
+ })
+
+ Describe("GET /api/library/{id}", func() {
+ It("returns a specific library", func() {
+ req := createAuthenticatedRequest("GET", "/library/1", nil, adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+
+ var library model.Library
+ err := json.Unmarshal(w.Body.Bytes(), &library)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(library.Name).To(Equal("Test Library 1"))
+ Expect(library.Path).To(Equal("/music/library1"))
+ })
+
+ It("returns 404 for non-existent library", func() {
+ req := createAuthenticatedRequest("GET", "/library/999", nil, adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusNotFound))
+ })
+
+ It("returns 400 for invalid library ID", func() {
+ req := createAuthenticatedRequest("GET", "/library/invalid", nil, adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusNotFound))
+ })
+ })
+
+ Describe("POST /api/library", func() {
+ It("creates a new library", func() {
+ newLibrary := model.Library{
+ Name: "New Library",
+ Path: "/music/new",
+ }
+ body, _ := json.Marshal(newLibrary)
+ req := createAuthenticatedRequest("POST", "/library", bytes.NewBuffer(body), adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+ })
+
+ It("validates required fields", func() {
+ invalidLibrary := model.Library{
+ Name: "", // Missing name
+ Path: "/music/invalid",
+ }
+ body, _ := json.Marshal(invalidLibrary)
+ req := createAuthenticatedRequest("POST", "/library", bytes.NewBuffer(body), adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusBadRequest))
+ Expect(w.Body.String()).To(ContainSubstring("library name is required"))
+ })
+
+ It("validates path field", func() {
+ invalidLibrary := model.Library{
+ Name: "Valid Name",
+ Path: "", // Missing path
+ }
+ body, _ := json.Marshal(invalidLibrary)
+ req := createAuthenticatedRequest("POST", "/library", bytes.NewBuffer(body), adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusBadRequest))
+ Expect(w.Body.String()).To(ContainSubstring("library path is required"))
+ })
+ })
+
+ Describe("PUT /api/library/{id}", func() {
+ It("updates an existing library", func() {
+ updatedLibrary := model.Library{
+ Name: "Updated Library 1",
+ Path: "/music/updated",
+ }
+ body, _ := json.Marshal(updatedLibrary)
+ req := createAuthenticatedRequest("PUT", "/library/1", bytes.NewBuffer(body), adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+
+ var updated model.Library
+ err := json.Unmarshal(w.Body.Bytes(), &updated)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(updated.ID).To(Equal(1))
+ Expect(updated.Name).To(Equal("Updated Library 1"))
+ Expect(updated.Path).To(Equal("/music/updated"))
+ })
+
+ It("validates required fields on update", func() {
+ invalidLibrary := model.Library{
+ Name: "",
+ Path: "/music/path",
+ }
+ body, _ := json.Marshal(invalidLibrary)
+ req := createAuthenticatedRequest("PUT", "/library/1", bytes.NewBuffer(body), adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusBadRequest))
+ })
+ })
+
+ Describe("DELETE /api/library/{id}", func() {
+ It("deletes an existing library", func() {
+ req := createAuthenticatedRequest("DELETE", "/library/1", nil, adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+ })
+
+ It("returns 404 for non-existent library", func() {
+ req := createAuthenticatedRequest("DELETE", "/library/999", nil, adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusNotFound))
+ })
+ })
+ })
+
+ Context("as regular user", func() {
+ var userToken string
+
+ BeforeEach(func() {
+ var err error
+ userToken, err = auth.CreateToken(®ularUser)
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ It("denies access to library management endpoints", func() {
+ endpoints := []string{
+ "GET /library",
+ "POST /library",
+ "GET /library/1",
+ "PUT /library/1",
+ "DELETE /library/1",
+ }
+
+ for _, endpoint := range endpoints {
+ parts := strings.Split(endpoint, " ")
+ method, path := parts[0], parts[1]
+
+ req := createAuthenticatedRequest(method, path, nil, userToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusForbidden))
+ }
+ })
+ })
+
+ Context("without authentication", func() {
+ It("denies access to library management endpoints", func() {
+ req := createUnauthenticatedRequest("GET", "/library", nil)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusUnauthorized))
+ })
+ })
+ })
+
+ Describe("User-Library Association Operations", func() {
+ Context("as admin user", func() {
+ var adminToken string
+
+ BeforeEach(func() {
+ var err error
+ adminToken, err = auth.CreateToken(&adminUser)
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ Describe("GET /api/user/{id}/library", func() {
+ It("returns user's libraries", func() {
+ // Set up user libraries
+ err := ds.User(context.TODO()).SetUserLibraries(regularUser.ID, []int{1, 2})
+ Expect(err).ToNot(HaveOccurred())
+
+ req := createAuthenticatedRequest("GET", fmt.Sprintf("/user/%s/library", regularUser.ID), nil, adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+
+ var libraries []model.Library
+ err = json.Unmarshal(w.Body.Bytes(), &libraries)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(libraries).To(HaveLen(2))
+ })
+
+ It("returns 404 for non-existent user", func() {
+ req := createAuthenticatedRequest("GET", "/user/non-existent/library", nil, adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusNotFound))
+ })
+ })
+
+ Describe("PUT /api/user/{id}/library", func() {
+ It("sets user's libraries", func() {
+ request := map[string][]int{
+ "libraryIds": {1, 2},
+ }
+ body, _ := json.Marshal(request)
+ req := createAuthenticatedRequest("PUT", fmt.Sprintf("/user/%s/library", regularUser.ID), bytes.NewBuffer(body), adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+
+ var libraries []model.Library
+ err := json.Unmarshal(w.Body.Bytes(), &libraries)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(libraries).To(HaveLen(2))
+ })
+
+ It("validates library IDs exist", func() {
+ request := map[string][]int{
+ "libraryIds": {999}, // Non-existent library
+ }
+ body, _ := json.Marshal(request)
+ req := createAuthenticatedRequest("PUT", fmt.Sprintf("/user/%s/library", regularUser.ID), bytes.NewBuffer(body), adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusBadRequest))
+ Expect(w.Body.String()).To(ContainSubstring("library ID 999 does not exist"))
+ })
+
+ It("requires at least one library for regular users", func() {
+ request := map[string][]int{
+ "libraryIds": {}, // Empty libraries
+ }
+ body, _ := json.Marshal(request)
+ req := createAuthenticatedRequest("PUT", fmt.Sprintf("/user/%s/library", regularUser.ID), bytes.NewBuffer(body), adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusBadRequest))
+ Expect(w.Body.String()).To(ContainSubstring("at least one library must be assigned"))
+ })
+
+ It("prevents manual assignment to admin users", func() {
+ request := map[string][]int{
+ "libraryIds": {1},
+ }
+ body, _ := json.Marshal(request)
+ req := createAuthenticatedRequest("PUT", fmt.Sprintf("/user/%s/library", adminUser.ID), bytes.NewBuffer(body), adminToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusBadRequest))
+ Expect(w.Body.String()).To(ContainSubstring("cannot manually assign libraries to admin users"))
+ })
+ })
+ })
+
+ Context("as regular user", func() {
+ var userToken string
+
+ BeforeEach(func() {
+ var err error
+ userToken, err = auth.CreateToken(®ularUser)
+ Expect(err).ToNot(HaveOccurred())
+ })
+
+ It("denies access to user-library association endpoints", func() {
+ req := createAuthenticatedRequest("GET", fmt.Sprintf("/user/%s/library", regularUser.ID), nil, userToken)
+ w := httptest.NewRecorder()
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusForbidden))
+ })
+ })
+ })
+})
+
+// Helper functions
+
+func createAuthenticatedRequest(method, path string, body *bytes.Buffer, token string) *http.Request {
+ if body == nil {
+ body = &bytes.Buffer{}
+ }
+ req := httptest.NewRequest(method, path, body)
+ req.Header.Set(consts.UIAuthorizationHeader, "Bearer "+token)
+ req.Header.Set("Content-Type", "application/json")
+ return req
+}
+
+func createUnauthenticatedRequest(method, path string, body *bytes.Buffer) *http.Request {
+ if body == nil {
+ body = &bytes.Buffer{}
+ }
+ req := httptest.NewRequest(method, path, body)
+ req.Header.Set("Content-Type", "application/json")
+ return req
+}
diff --git a/server/nativeapi/missing.go b/server/nativeapi/missing.go
index 5ccc15f55..2b455e622 100644
--- a/server/nativeapi/missing.go
+++ b/server/nativeapi/missing.go
@@ -8,6 +8,7 @@ import (
"github.com/Masterminds/squirrel"
"github.com/deluan/rest"
+ "github.com/navidrome/navidrome/core"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/utils/req"
@@ -62,34 +63,32 @@ func (r *missingRepository) EntityName() string {
return "missing_files"
}
-func deleteMissingFiles(ds model.DataStore, w http.ResponseWriter, r *http.Request) {
- ctx := r.Context()
- p := req.Params(r)
- ids, _ := p.Strings("id")
- err := ds.WithTx(func(tx model.DataStore) error {
+func deleteMissingFiles(maintenance core.Maintenance) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ ctx := r.Context()
+
+ p := req.Params(r)
+ ids, _ := p.Strings("id")
+
+ var err error
if len(ids) == 0 {
- _, err := tx.MediaFile(ctx).DeleteAllMissing()
- return err
+ err = maintenance.DeleteAllMissingFiles(ctx)
+ } else {
+ err = maintenance.DeleteMissingFiles(ctx, ids)
}
- return tx.MediaFile(ctx).DeleteMissing(ids)
- })
- if len(ids) == 1 && errors.Is(err, model.ErrNotFound) {
- log.Warn(ctx, "Missing file not found", "id", ids[0])
- http.Error(w, "not found", http.StatusNotFound)
- return
+
+ if len(ids) == 1 && errors.Is(err, model.ErrNotFound) {
+ log.Warn(ctx, "Missing file not found", "id", ids[0])
+ http.Error(w, "not found", http.StatusNotFound)
+ return
+ }
+ if err != nil {
+ http.Error(w, "failed to delete missing files", http.StatusInternalServerError)
+ return
+ }
+
+ writeDeleteManyResponse(w, r, ids)
}
- if err != nil {
- log.Error(ctx, "Error deleting missing tracks from DB", "ids", ids, err)
- http.Error(w, err.Error(), http.StatusInternalServerError)
- return
- }
- err = ds.GC(ctx)
- if err != nil {
- log.Error(ctx, "Error running GC after deleting missing tracks", err)
- http.Error(w, err.Error(), http.StatusInternalServerError)
- return
- }
- writeDeleteManyResponse(w, r, ids)
}
var _ model.ResourceRepository = &missingRepository{}
diff --git a/server/nativeapi/native_api.go b/server/nativeapi/native_api.go
index ddf5df1c3..969650e0a 100644
--- a/server/nativeapi/native_api.go
+++ b/server/nativeapi/native_api.go
@@ -16,79 +16,77 @@ import (
"github.com/navidrome/navidrome/core/metrics"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
"github.com/navidrome/navidrome/server"
)
type Router struct {
http.Handler
- ds model.DataStore
- share core.Share
- playlists core.Playlists
- insights metrics.Insights
+ ds model.DataStore
+ share core.Share
+ playlists core.Playlists
+ insights metrics.Insights
+ libs core.Library
+ maintenance core.Maintenance
}
-func New(ds model.DataStore, share core.Share, playlists core.Playlists, insights metrics.Insights) *Router {
- r := &Router{ds: ds, share: share, playlists: playlists, insights: insights}
+func New(ds model.DataStore, share core.Share, playlists core.Playlists, insights metrics.Insights, libraryService core.Library, maintenance core.Maintenance) *Router {
+ r := &Router{ds: ds, share: share, playlists: playlists, insights: insights, libs: libraryService, maintenance: maintenance}
r.Handler = r.routes()
return r
}
-func (n *Router) routes() http.Handler {
+func (api *Router) routes() http.Handler {
r := chi.NewRouter()
// Public
- n.RX(r, "/translation", newTranslationRepository, false)
+ api.RX(r, "/translation", newTranslationRepository, false)
// Protected
r.Group(func(r chi.Router) {
- r.Use(server.Authenticator(n.ds))
+ r.Use(server.Authenticator(api.ds))
r.Use(server.JWTRefresher)
- r.Use(server.UpdateLastAccessMiddleware(n.ds))
- n.R(r, "/user", model.User{}, true)
- n.R(r, "/song", model.MediaFile{}, false)
- n.R(r, "/album", model.Album{}, false)
- n.R(r, "/artist", model.Artist{}, false)
- n.R(r, "/genre", model.Genre{}, false)
- n.R(r, "/player", model.Player{}, true)
- n.R(r, "/transcoding", model.Transcoding{}, conf.Server.EnableTranscodingConfig)
- n.R(r, "/radio", model.Radio{}, true)
- n.R(r, "/tag", model.Tag{}, true)
+ r.Use(server.UpdateLastAccessMiddleware(api.ds))
+ api.R(r, "/user", model.User{}, true)
+ api.R(r, "/song", model.MediaFile{}, false)
+ api.R(r, "/album", model.Album{}, false)
+ api.R(r, "/artist", model.Artist{}, false)
+ api.R(r, "/genre", model.Genre{}, false)
+ api.R(r, "/player", model.Player{}, true)
+ api.R(r, "/transcoding", model.Transcoding{}, conf.Server.EnableTranscodingConfig)
+ api.R(r, "/radio", model.Radio{}, true)
+ api.R(r, "/tag", model.Tag{}, true)
if conf.Server.EnableSharing {
- n.RX(r, "/share", n.share.NewRepository, true)
+ api.RX(r, "/share", api.share.NewRepository, true)
}
- n.addPlaylistRoute(r)
- n.addPlaylistTrackRoute(r)
- n.addMissingFilesRoute(r)
- n.addInspectRoute(r)
+ api.addPlaylistRoute(r)
+ api.addPlaylistTrackRoute(r)
+ api.addSongPlaylistsRoute(r)
+ api.addQueueRoute(r)
+ api.addMissingFilesRoute(r)
+ api.addKeepAliveRoute(r)
+ api.addInsightsRoute(r)
- // Keepalive endpoint to be used to keep the session valid (ex: while playing songs)
- r.Get("/keepalive/*", func(w http.ResponseWriter, r *http.Request) {
- _, _ = w.Write([]byte(`{"response":"ok", "id":"keepalive"}`))
- })
-
- // Insights status endpoint
- r.Get("/insights/*", func(w http.ResponseWriter, r *http.Request) {
- last, success := n.insights.LastRun(r.Context())
- if conf.Server.EnableInsightsCollector {
- _, _ = w.Write([]byte(`{"id":"insights_status", "lastRun":"` + last.Format("2006-01-02 15:04:05") + `", "success":` + strconv.FormatBool(success) + `}`))
- } else {
- _, _ = w.Write([]byte(`{"id":"insights_status", "lastRun":"disabled", "success":false}`))
- }
+ r.With(adminOnlyMiddleware).Group(func(r chi.Router) {
+ api.addInspectRoute(r)
+ api.addConfigRoute(r)
+ api.addUserLibraryRoute(r)
+ api.RX(r, "/library", api.libs.NewRepository, true)
})
})
return r
}
-func (n *Router) R(r chi.Router, pathPrefix string, model interface{}, persistable bool) {
+func (api *Router) R(r chi.Router, pathPrefix string, model interface{}, persistable bool) {
constructor := func(ctx context.Context) rest.Repository {
- return n.ds.Resource(ctx, model)
+ return api.ds.Resource(ctx, model)
}
- n.RX(r, pathPrefix, constructor, persistable)
+ api.RX(r, pathPrefix, constructor, persistable)
}
-func (n *Router) RX(r chi.Router, pathPrefix string, constructor rest.RepositoryConstructor, persistable bool) {
+func (api *Router) RX(r chi.Router, pathPrefix string, constructor rest.RepositoryConstructor, persistable bool) {
r.Route(pathPrefix, func(r chi.Router) {
r.Get("/", rest.GetAll(constructor))
if persistable {
@@ -105,9 +103,9 @@ func (n *Router) RX(r chi.Router, pathPrefix string, constructor rest.Repository
})
}
-func (n *Router) addPlaylistRoute(r chi.Router) {
+func (api *Router) addPlaylistRoute(r chi.Router) {
constructor := func(ctx context.Context) rest.Repository {
- return n.ds.Resource(ctx, model.Playlist{})
+ return api.ds.Resource(ctx, model.Playlist{})
}
r.Route("/playlist", func(r chi.Router) {
@@ -117,7 +115,7 @@ func (n *Router) addPlaylistRoute(r chi.Router) {
rest.Post(constructor)(w, r)
return
}
- createPlaylistFromM3U(n.playlists)(w, r)
+ createPlaylistFromM3U(api.playlists)(w, r)
})
r.Route("/{id}", func(r chi.Router) {
@@ -129,37 +127,53 @@ func (n *Router) addPlaylistRoute(r chi.Router) {
})
}
-func (n *Router) addPlaylistTrackRoute(r chi.Router) {
+func (api *Router) addPlaylistTrackRoute(r chi.Router) {
r.Route("/playlist/{playlistId}/tracks", func(r chi.Router) {
r.Get("/", func(w http.ResponseWriter, r *http.Request) {
- getPlaylist(n.ds)(w, r)
+ getPlaylist(api.ds)(w, r)
})
r.With(server.URLParamsMiddleware).Route("/", func(r chi.Router) {
r.Delete("/", func(w http.ResponseWriter, r *http.Request) {
- deleteFromPlaylist(n.ds)(w, r)
+ deleteFromPlaylist(api.ds)(w, r)
})
r.Post("/", func(w http.ResponseWriter, r *http.Request) {
- addToPlaylist(n.ds)(w, r)
+ addToPlaylist(api.ds)(w, r)
})
})
r.Route("/{id}", func(r chi.Router) {
r.Use(server.URLParamsMiddleware)
+ r.Get("/", func(w http.ResponseWriter, r *http.Request) {
+ getPlaylistTrack(api.ds)(w, r)
+ })
r.Put("/", func(w http.ResponseWriter, r *http.Request) {
- reorderItem(n.ds)(w, r)
+ reorderItem(api.ds)(w, r)
})
r.Delete("/", func(w http.ResponseWriter, r *http.Request) {
- deleteFromPlaylist(n.ds)(w, r)
+ deleteFromPlaylist(api.ds)(w, r)
})
})
})
}
-func (n *Router) addMissingFilesRoute(r chi.Router) {
+func (api *Router) addSongPlaylistsRoute(r chi.Router) {
+ r.With(server.URLParamsMiddleware).Get("/song/{id}/playlists", func(w http.ResponseWriter, r *http.Request) {
+ getSongPlaylists(api.ds)(w, r)
+ })
+}
+
+func (api *Router) addQueueRoute(r chi.Router) {
+ r.Route("/queue", func(r chi.Router) {
+ r.Get("/", getQueue(api.ds))
+ r.Post("/", saveQueue(api.ds))
+ r.Put("/", updateQueue(api.ds))
+ r.Delete("/", clearQueue(api.ds))
+ })
+}
+
+func (api *Router) addMissingFilesRoute(r chi.Router) {
r.Route("/missing", func(r chi.Router) {
- n.RX(r, "/", newMissingRepository(n.ds), false)
- r.Delete("/", func(w http.ResponseWriter, r *http.Request) {
- deleteMissingFiles(n.ds, w, r)
- })
+ api.RX(r, "/", newMissingRepository(api.ds), false)
+ r.Delete("/", deleteMissingFiles(api.maintenance))
})
}
@@ -183,7 +197,7 @@ func writeDeleteManyResponse(w http.ResponseWriter, r *http.Request, ids []strin
}
}
-func (n *Router) addInspectRoute(r chi.Router) {
+func (api *Router) addInspectRoute(r chi.Router) {
if conf.Server.Inspect.Enabled {
r.Group(func(r chi.Router) {
if conf.Server.Inspect.MaxRequests > 0 {
@@ -192,7 +206,42 @@ func (n *Router) addInspectRoute(r chi.Router) {
conf.Server.Inspect.BacklogTimeout)
r.Use(middleware.ThrottleBacklog(conf.Server.Inspect.MaxRequests, conf.Server.Inspect.BacklogLimit, time.Duration(conf.Server.Inspect.BacklogTimeout)))
}
- r.Get("/inspect", inspect(n.ds))
+ r.Get("/inspect", inspect(api.ds))
})
}
}
+
+func (api *Router) addConfigRoute(r chi.Router) {
+ if conf.Server.DevUIShowConfig {
+ r.Get("/config/*", getConfig)
+ }
+}
+
+func (api *Router) addKeepAliveRoute(r chi.Router) {
+ r.Get("/keepalive/*", func(w http.ResponseWriter, r *http.Request) {
+ _, _ = w.Write([]byte(`{"response":"ok", "id":"keepalive"}`))
+ })
+}
+
+func (api *Router) addInsightsRoute(r chi.Router) {
+ r.Get("/insights/*", func(w http.ResponseWriter, r *http.Request) {
+ last, success := api.insights.LastRun(r.Context())
+ if conf.Server.EnableInsightsCollector {
+ _, _ = w.Write([]byte(`{"id":"insights_status", "lastRun":"` + last.Format("2006-01-02 15:04:05") + `", "success":` + strconv.FormatBool(success) + `}`))
+ } else {
+ _, _ = w.Write([]byte(`{"id":"insights_status", "lastRun":"disabled", "success":false}`))
+ }
+ })
+}
+
+// Middleware to ensure only admin users can access endpoints
+func adminOnlyMiddleware(next http.Handler) http.Handler {
+ return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ user, ok := request.UserFrom(r.Context())
+ if !ok || !user.IsAdmin {
+ http.Error(w, "Access denied: admin privileges required", http.StatusForbidden)
+ return
+ }
+ next.ServeHTTP(w, r)
+ })
+}
diff --git a/server/nativeapi/native_api_song_test.go b/server/nativeapi/native_api_song_test.go
new file mode 100644
index 000000000..b52042643
--- /dev/null
+++ b/server/nativeapi/native_api_song_test.go
@@ -0,0 +1,431 @@
+package nativeapi
+
+import (
+ "bytes"
+ "encoding/json"
+ "net/http"
+ "net/http/httptest"
+ "net/url"
+ "time"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/consts"
+ "github.com/navidrome/navidrome/core"
+ "github.com/navidrome/navidrome/core/auth"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/server"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("Song Endpoints", func() {
+ var (
+ router http.Handler
+ ds *tests.MockDataStore
+ mfRepo *tests.MockMediaFileRepo
+ userRepo *tests.MockedUserRepo
+ w *httptest.ResponseRecorder
+ testUser model.User
+ testSongs model.MediaFiles
+ )
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ conf.Server.SessionTimeout = time.Minute
+
+ // Setup mock repositories
+ mfRepo = tests.CreateMockMediaFileRepo()
+ userRepo = tests.CreateMockUserRepo()
+
+ ds = &tests.MockDataStore{
+ MockedMediaFile: mfRepo,
+ MockedUser: userRepo,
+ MockedProperty: &tests.MockedPropertyRepo{},
+ }
+
+ // Initialize auth system
+ auth.Init(ds)
+
+ // Create test user
+ testUser = model.User{
+ ID: "user-1",
+ UserName: "testuser",
+ Name: "Test User",
+ IsAdmin: false,
+ NewPassword: "testpass",
+ }
+ err := userRepo.Put(&testUser)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Create test songs
+ testSongs = model.MediaFiles{
+ {
+ ID: "song-1",
+ Title: "Test Song 1",
+ Artist: "Test Artist 1",
+ Album: "Test Album 1",
+ AlbumID: "album-1",
+ ArtistID: "artist-1",
+ Duration: 180.5,
+ BitRate: 320,
+ Path: "/music/song1.mp3",
+ Suffix: "mp3",
+ Size: 5242880,
+ CreatedAt: time.Now(),
+ UpdatedAt: time.Now(),
+ },
+ {
+ ID: "song-2",
+ Title: "Test Song 2",
+ Artist: "Test Artist 2",
+ Album: "Test Album 2",
+ AlbumID: "album-2",
+ ArtistID: "artist-2",
+ Duration: 240.0,
+ BitRate: 256,
+ Path: "/music/song2.mp3",
+ Suffix: "mp3",
+ Size: 7340032,
+ CreatedAt: time.Now(),
+ UpdatedAt: time.Now(),
+ },
+ }
+ mfRepo.SetData(testSongs)
+
+ // Create the native API router and wrap it with the JWTVerifier middleware
+ nativeRouter := New(ds, nil, nil, nil, core.NewMockLibraryService(), nil)
+ router = server.JWTVerifier(nativeRouter)
+ w = httptest.NewRecorder()
+ })
+
+ // Helper function to create unauthenticated request
+ createUnauthenticatedRequest := func(method, path string, body []byte) *http.Request {
+ var req *http.Request
+ if body != nil {
+ req = httptest.NewRequest(method, path, bytes.NewReader(body))
+ req.Header.Set("Content-Type", "application/json")
+ } else {
+ req = httptest.NewRequest(method, path, nil)
+ }
+ return req
+ }
+
+ // Helper function to create authenticated request with JWT token
+ createAuthenticatedRequest := func(method, path string, body []byte) *http.Request {
+ req := createUnauthenticatedRequest(method, path, body)
+
+ // Create JWT token for the test user
+ token, err := auth.CreateToken(&testUser)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Add JWT token to Authorization header
+ req.Header.Set(consts.UIAuthorizationHeader, "Bearer "+token)
+
+ return req
+ }
+
+ Describe("GET /song", func() {
+ Context("when user is authenticated", func() {
+ It("returns all songs", func() {
+ req := createAuthenticatedRequest("GET", "/song", nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+
+ var response []model.MediaFile
+ err := json.Unmarshal(w.Body.Bytes(), &response)
+ Expect(err).ToNot(HaveOccurred())
+
+ Expect(response).To(HaveLen(2))
+ Expect(response[0].ID).To(Equal("song-1"))
+ Expect(response[0].Title).To(Equal("Test Song 1"))
+ Expect(response[1].ID).To(Equal("song-2"))
+ Expect(response[1].Title).To(Equal("Test Song 2"))
+ })
+
+ It("handles repository errors gracefully", func() {
+ mfRepo.SetError(true)
+
+ req := createAuthenticatedRequest("GET", "/song", nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusInternalServerError))
+ })
+ })
+
+ Context("when user is not authenticated", func() {
+ It("returns unauthorized", func() {
+ req := createUnauthenticatedRequest("GET", "/song", nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusUnauthorized))
+ })
+ })
+ })
+
+ Describe("GET /song/{id}", func() {
+ Context("when user is authenticated", func() {
+ It("returns the specific song", func() {
+ req := createAuthenticatedRequest("GET", "/song/song-1", nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+
+ var response model.MediaFile
+ err := json.Unmarshal(w.Body.Bytes(), &response)
+ Expect(err).ToNot(HaveOccurred())
+
+ Expect(response.ID).To(Equal("song-1"))
+ Expect(response.Title).To(Equal("Test Song 1"))
+ Expect(response.Artist).To(Equal("Test Artist 1"))
+ })
+
+ It("returns 404 for non-existent song", func() {
+ req := createAuthenticatedRequest("GET", "/song/non-existent", nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusNotFound))
+ })
+
+ It("handles repository errors gracefully", func() {
+ mfRepo.SetError(true)
+
+ req := createAuthenticatedRequest("GET", "/song/song-1", nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusInternalServerError))
+ })
+ })
+
+ Context("when user is not authenticated", func() {
+ It("returns unauthorized", func() {
+ req := createUnauthenticatedRequest("GET", "/song/song-1", nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusUnauthorized))
+ })
+ })
+ })
+
+ Describe("Song endpoints are read-only", func() {
+ Context("POST /song", func() {
+ It("should not be available (songs are not persistable)", func() {
+ newSong := model.MediaFile{
+ Title: "New Song",
+ Artist: "New Artist",
+ Album: "New Album",
+ Duration: 200.0,
+ }
+
+ body, _ := json.Marshal(newSong)
+ req := createAuthenticatedRequest("POST", "/song", body)
+ router.ServeHTTP(w, req)
+
+ // Should return 405 Method Not Allowed or 404 Not Found
+ Expect(w.Code).To(Equal(http.StatusMethodNotAllowed))
+ })
+ })
+
+ Context("PUT /song/{id}", func() {
+ It("should not be available (songs are not persistable)", func() {
+ updatedSong := model.MediaFile{
+ ID: "song-1",
+ Title: "Updated Song",
+ Artist: "Updated Artist",
+ Album: "Updated Album",
+ Duration: 250.0,
+ }
+
+ body, _ := json.Marshal(updatedSong)
+ req := createAuthenticatedRequest("PUT", "/song/song-1", body)
+ router.ServeHTTP(w, req)
+
+ // Should return 405 Method Not Allowed or 404 Not Found
+ Expect(w.Code).To(Equal(http.StatusMethodNotAllowed))
+ })
+ })
+
+ Context("DELETE /song/{id}", func() {
+ It("should not be available (songs are not persistable)", func() {
+ req := createAuthenticatedRequest("DELETE", "/song/song-1", nil)
+ router.ServeHTTP(w, req)
+
+ // Should return 405 Method Not Allowed or 404 Not Found
+ Expect(w.Code).To(Equal(http.StatusMethodNotAllowed))
+ })
+ })
+ })
+
+ Describe("Query parameters and filtering", func() {
+ Context("when using query parameters", func() {
+ It("handles pagination parameters", func() {
+ req := createAuthenticatedRequest("GET", "/song?_start=0&_end=1", nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+
+ var response []model.MediaFile
+ err := json.Unmarshal(w.Body.Bytes(), &response)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Should still return all songs since our mock doesn't implement pagination
+ // but the request should be processed successfully
+ Expect(len(response)).To(BeNumerically(">=", 1))
+ })
+
+ It("handles sort parameters", func() {
+ req := createAuthenticatedRequest("GET", "/song?_sort=title&_order=ASC", nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+
+ var response []model.MediaFile
+ err := json.Unmarshal(w.Body.Bytes(), &response)
+ Expect(err).ToNot(HaveOccurred())
+
+ Expect(response).To(HaveLen(2))
+ })
+
+ It("handles filter parameters", func() {
+ // Properly encode the URL with query parameters
+ baseURL := "/song"
+ params := url.Values{}
+ params.Add("title", "Test Song 1")
+ fullURL := baseURL + "?" + params.Encode()
+
+ req := createAuthenticatedRequest("GET", fullURL, nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+
+ var response []model.MediaFile
+ err := json.Unmarshal(w.Body.Bytes(), &response)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Mock doesn't implement filtering, but request should be processed
+ Expect(len(response)).To(BeNumerically(">=", 1))
+ })
+ })
+ })
+
+ Describe("Response headers and content type", func() {
+ It("sets correct content type for JSON responses", func() {
+ req := createAuthenticatedRequest("GET", "/song", nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+ Expect(w.Header().Get("Content-Type")).To(ContainSubstring("application/json"))
+ })
+
+ It("includes total count header when available", func() {
+ req := createAuthenticatedRequest("GET", "/song", nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+ // The X-Total-Count header might be set by the REST framework
+ // We just verify the request is processed successfully
+ })
+ })
+
+ Describe("Edge cases and error handling", func() {
+ Context("when repository is unavailable", func() {
+ It("handles database connection errors", func() {
+ mfRepo.SetError(true)
+
+ req := createAuthenticatedRequest("GET", "/song", nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusInternalServerError))
+ })
+ })
+
+ Context("when no songs exist", func() {
+ It("returns empty array when no songs are found", func() {
+ mfRepo.SetData(model.MediaFiles{}) // Empty dataset
+
+ req := createAuthenticatedRequest("GET", "/song", nil)
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+
+ var response []model.MediaFile
+ err := json.Unmarshal(w.Body.Bytes(), &response)
+ Expect(err).ToNot(HaveOccurred())
+
+ Expect(response).To(HaveLen(0))
+ })
+ })
+ })
+
+ Describe("Authentication middleware integration", func() {
+ Context("with different user types", func() {
+ It("works with admin users", func() {
+ adminUser := model.User{
+ ID: "admin-1",
+ UserName: "admin",
+ Name: "Admin User",
+ IsAdmin: true,
+ NewPassword: "adminpass",
+ }
+ err := userRepo.Put(&adminUser)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Create JWT token for admin user
+ token, err := auth.CreateToken(&adminUser)
+ Expect(err).ToNot(HaveOccurred())
+
+ req := createUnauthenticatedRequest("GET", "/song", nil)
+ req.Header.Set(consts.UIAuthorizationHeader, "Bearer "+token)
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+ })
+
+ It("works with regular users", func() {
+ regularUser := model.User{
+ ID: "user-2",
+ UserName: "regular",
+ Name: "Regular User",
+ IsAdmin: false,
+ NewPassword: "userpass",
+ }
+ err := userRepo.Put(®ularUser)
+ Expect(err).ToNot(HaveOccurred())
+
+ // Create JWT token for regular user
+ token, err := auth.CreateToken(®ularUser)
+ Expect(err).ToNot(HaveOccurred())
+
+ req := createUnauthenticatedRequest("GET", "/song", nil)
+ req.Header.Set(consts.UIAuthorizationHeader, "Bearer "+token)
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusOK))
+ })
+ })
+
+ Context("with missing authentication context", func() {
+ It("rejects requests without user context", func() {
+ req := createUnauthenticatedRequest("GET", "/song", nil)
+ // No authentication header added
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusUnauthorized))
+ })
+
+ It("rejects requests with invalid JWT tokens", func() {
+ req := createUnauthenticatedRequest("GET", "/song", nil)
+ req.Header.Set(consts.UIAuthorizationHeader, "Bearer invalid.token.here")
+
+ router.ServeHTTP(w, req)
+
+ Expect(w.Code).To(Equal(http.StatusUnauthorized))
+ })
+ })
+ })
+})
diff --git a/server/nativeapi/playlists.go b/server/nativeapi/playlists.go
index 1e8e961ca..17af19475 100644
--- a/server/nativeapi/playlists.go
+++ b/server/nativeapi/playlists.go
@@ -45,6 +45,23 @@ func getPlaylist(ds model.DataStore) http.HandlerFunc {
}
}
+func getPlaylistTrack(ds model.DataStore) http.HandlerFunc {
+ // Add a middleware to capture the playlistId
+ wrapper := func(handler restHandler) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ constructor := func(ctx context.Context) rest.Repository {
+ plsRepo := ds.Playlist(ctx)
+ plsId := chi.URLParam(r, "playlistId")
+ return plsRepo.Tracks(plsId, true)
+ }
+
+ handler(constructor).ServeHTTP(w, r)
+ }
+ }
+
+ return wrapper(rest.Get)
+}
+
func createPlaylistFromM3U(playlists core.Playlists) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
ctx := r.Context()
@@ -207,3 +224,21 @@ func reorderItem(ds model.DataStore) http.HandlerFunc {
}
}
}
+
+func getSongPlaylists(ds model.DataStore) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ p := req.Params(r)
+ trackId, _ := p.String(":id")
+ playlists, err := ds.Playlist(r.Context()).GetPlaylists(trackId)
+ if err != nil {
+ http.Error(w, err.Error(), http.StatusInternalServerError)
+ return
+ }
+ data, err := json.Marshal(playlists)
+ if err != nil {
+ http.Error(w, err.Error(), http.StatusInternalServerError)
+ return
+ }
+ _, _ = w.Write(data)
+ }
+}
diff --git a/server/nativeapi/queue.go b/server/nativeapi/queue.go
new file mode 100644
index 000000000..0a3136660
--- /dev/null
+++ b/server/nativeapi/queue.go
@@ -0,0 +1,214 @@
+package nativeapi
+
+import (
+ "context"
+ "encoding/json"
+ "errors"
+ "net/http"
+
+ "github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ . "github.com/navidrome/navidrome/utils/gg"
+ "github.com/navidrome/navidrome/utils/slice"
+)
+
+type updateQueuePayload struct {
+ Ids *[]string `json:"ids,omitempty"`
+ Current *int `json:"current,omitempty"`
+ Position *int64 `json:"position,omitempty"`
+}
+
+// validateCurrentIndex validates that the current index is within bounds of the items array.
+// Returns false if validation fails (and sends error response), true if validation passes.
+func validateCurrentIndex(w http.ResponseWriter, current int, itemsLength int) bool {
+ if current < 0 || current >= itemsLength {
+ http.Error(w, "current index out of bounds", http.StatusBadRequest)
+ return false
+ }
+ return true
+}
+
+// retrieveExistingQueue retrieves an existing play queue for a user with proper error handling.
+// Returns the queue (nil if not found) and false if an error occurred and response was sent.
+func retrieveExistingQueue(ctx context.Context, w http.ResponseWriter, ds model.DataStore, userID string) (*model.PlayQueue, bool) {
+ existing, err := ds.PlayQueue(ctx).Retrieve(userID)
+ if err != nil && !errors.Is(err, model.ErrNotFound) {
+ log.Error(ctx, "Error retrieving queue", err)
+ http.Error(w, err.Error(), http.StatusInternalServerError)
+ return nil, false
+ }
+ return existing, true
+}
+
+// decodeUpdatePayload decodes the JSON payload from the request body.
+// Returns false if decoding fails (and sends error response), true if successful.
+func decodeUpdatePayload(w http.ResponseWriter, r *http.Request) (*updateQueuePayload, bool) {
+ var payload updateQueuePayload
+ if err := json.NewDecoder(r.Body).Decode(&payload); err != nil {
+ http.Error(w, err.Error(), http.StatusBadRequest)
+ return nil, false
+ }
+ return &payload, true
+}
+
+// createMediaFileItems converts a slice of IDs to MediaFile items.
+func createMediaFileItems(ids []string) []model.MediaFile {
+ return slice.Map(ids, func(id string) model.MediaFile {
+ return model.MediaFile{ID: id}
+ })
+}
+
+// extractUserAndClient extracts user and client from the request context.
+func extractUserAndClient(ctx context.Context) (model.User, string) {
+ user, _ := request.UserFrom(ctx)
+ client, _ := request.ClientFrom(ctx)
+ return user, client
+}
+
+func getQueue(ds model.DataStore) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ ctx := r.Context()
+ user, _ := request.UserFrom(ctx)
+ repo := ds.PlayQueue(ctx)
+ pq, err := repo.RetrieveWithMediaFiles(user.ID)
+ if err != nil && !errors.Is(err, model.ErrNotFound) {
+ log.Error(ctx, "Error retrieving queue", err)
+ http.Error(w, err.Error(), http.StatusInternalServerError)
+ return
+ }
+ if pq == nil {
+ pq = &model.PlayQueue{}
+ }
+ resp, err := json.Marshal(pq)
+ if err != nil {
+ log.Error(ctx, "Error marshalling queue", err)
+ http.Error(w, err.Error(), http.StatusInternalServerError)
+ return
+ }
+ w.Header().Set("Content-Type", "application/json")
+ _, _ = w.Write(resp)
+ }
+}
+
+func saveQueue(ds model.DataStore) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ ctx := r.Context()
+ payload, ok := decodeUpdatePayload(w, r)
+ if !ok {
+ return
+ }
+ user, client := extractUserAndClient(ctx)
+ ids := V(payload.Ids)
+ items := createMediaFileItems(ids)
+ current := V(payload.Current)
+ if len(ids) > 0 && !validateCurrentIndex(w, current, len(ids)) {
+ return
+ }
+ pq := &model.PlayQueue{
+ UserID: user.ID,
+ Current: current,
+ Position: max(V(payload.Position), 0),
+ ChangedBy: client,
+ Items: items,
+ }
+ if err := ds.PlayQueue(ctx).Store(pq); err != nil {
+ log.Error(ctx, "Error saving queue", err)
+ http.Error(w, err.Error(), http.StatusInternalServerError)
+ return
+ }
+ w.WriteHeader(http.StatusNoContent)
+ }
+}
+
+func updateQueue(ds model.DataStore) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ ctx := r.Context()
+
+ // Decode and validate the JSON payload
+ payload, ok := decodeUpdatePayload(w, r)
+ if !ok {
+ return
+ }
+
+ // Extract user and client information from request context
+ user, client := extractUserAndClient(ctx)
+
+ // Initialize play queue with user ID and client info
+ pq := &model.PlayQueue{UserID: user.ID, ChangedBy: client}
+ var cols []string // Track which columns to update in the database
+
+ // Handle queue items update
+ if payload.Ids != nil {
+ pq.Items = createMediaFileItems(*payload.Ids)
+ cols = append(cols, "items")
+
+ // If current index is not being updated, validate existing current index
+ // against the new items list to ensure it remains valid
+ if payload.Current == nil {
+ existing, ok := retrieveExistingQueue(ctx, w, ds, user.ID)
+ if !ok {
+ return
+ }
+ if existing != nil && !validateCurrentIndex(w, existing.Current, len(*payload.Ids)) {
+ return
+ }
+ }
+ }
+
+ // Handle current track index update
+ if payload.Current != nil {
+ pq.Current = *payload.Current
+ cols = append(cols, "current")
+
+ if payload.Ids != nil {
+ // If items are also being updated, validate current index against new items
+ if !validateCurrentIndex(w, *payload.Current, len(*payload.Ids)) {
+ return
+ }
+ } else {
+ // If only current index is being updated, validate against existing items
+ existing, ok := retrieveExistingQueue(ctx, w, ds, user.ID)
+ if !ok {
+ return
+ }
+ if existing != nil && !validateCurrentIndex(w, *payload.Current, len(existing.Items)) {
+ return
+ }
+ }
+ }
+
+ // Handle playback position update
+ if payload.Position != nil {
+ pq.Position = max(*payload.Position, 0) // Ensure position is non-negative
+ cols = append(cols, "position")
+ }
+
+ // If no fields were specified for update, return success without doing anything
+ if len(cols) == 0 {
+ w.WriteHeader(http.StatusNoContent)
+ return
+ }
+
+ // Perform partial update of the specified columns only
+ if err := ds.PlayQueue(ctx).Store(pq, cols...); err != nil {
+ log.Error(ctx, "Error updating queue", err)
+ http.Error(w, err.Error(), http.StatusInternalServerError)
+ return
+ }
+ w.WriteHeader(http.StatusNoContent)
+ }
+}
+
+func clearQueue(ds model.DataStore) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ ctx := r.Context()
+ user, _ := request.UserFrom(ctx)
+ if err := ds.PlayQueue(ctx).Clear(user.ID); err != nil {
+ log.Error(ctx, "Error clearing queue", err)
+ http.Error(w, err.Error(), http.StatusInternalServerError)
+ return
+ }
+ w.WriteHeader(http.StatusNoContent)
+ }
+}
diff --git a/server/nativeapi/queue_test.go b/server/nativeapi/queue_test.go
new file mode 100644
index 000000000..ef971ee68
--- /dev/null
+++ b/server/nativeapi/queue_test.go
@@ -0,0 +1,282 @@
+package nativeapi
+
+import (
+ "bytes"
+ "encoding/json"
+ "net/http"
+ "net/http/httptest"
+
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/tests"
+ "github.com/navidrome/navidrome/utils/gg"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("Queue Endpoints", func() {
+ var (
+ ds *tests.MockDataStore
+ repo *tests.MockPlayQueueRepo
+ user model.User
+ userRepo *tests.MockedUserRepo
+ )
+
+ BeforeEach(func() {
+ repo = &tests.MockPlayQueueRepo{}
+ user = model.User{ID: "u1", UserName: "user"}
+ userRepo = tests.CreateMockUserRepo()
+ _ = userRepo.Put(&user)
+ ds = &tests.MockDataStore{MockedPlayQueue: repo, MockedUser: userRepo, MockedProperty: &tests.MockedPropertyRepo{}}
+ })
+
+ Describe("POST /queue", func() {
+ It("saves the queue", func() {
+ payload := updateQueuePayload{Ids: gg.P([]string{"s1", "s2"}), Current: gg.P(1), Position: gg.P(int64(10))}
+ body, _ := json.Marshal(payload)
+ req := httptest.NewRequest("POST", "/queue", bytes.NewReader(body))
+ ctx := request.WithUser(req.Context(), user)
+ ctx = request.WithClient(ctx, "TestClient")
+ req = req.WithContext(ctx)
+ w := httptest.NewRecorder()
+
+ saveQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusNoContent))
+ Expect(repo.Queue).ToNot(BeNil())
+ Expect(repo.Queue.Current).To(Equal(1))
+ Expect(repo.Queue.Items).To(HaveLen(2))
+ Expect(repo.Queue.Items[1].ID).To(Equal("s2"))
+ Expect(repo.Queue.ChangedBy).To(Equal("TestClient"))
+ })
+
+ It("saves an empty queue", func() {
+ payload := updateQueuePayload{Ids: gg.P([]string{}), Current: gg.P(0), Position: gg.P(int64(0))}
+ body, _ := json.Marshal(payload)
+ req := httptest.NewRequest("POST", "/queue", bytes.NewReader(body))
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ saveQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusNoContent))
+ Expect(repo.Queue).ToNot(BeNil())
+ Expect(repo.Queue.Items).To(HaveLen(0))
+ })
+
+ It("returns bad request for invalid current index (negative)", func() {
+ payload := updateQueuePayload{Ids: gg.P([]string{"s1", "s2"}), Current: gg.P(-1), Position: gg.P(int64(10))}
+ body, _ := json.Marshal(payload)
+ req := httptest.NewRequest("POST", "/queue", bytes.NewReader(body))
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ saveQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusBadRequest))
+ Expect(w.Body.String()).To(ContainSubstring("current index out of bounds"))
+ })
+
+ It("returns bad request for invalid current index (too large)", func() {
+ payload := updateQueuePayload{Ids: gg.P([]string{"s1", "s2"}), Current: gg.P(2), Position: gg.P(int64(10))}
+ body, _ := json.Marshal(payload)
+ req := httptest.NewRequest("POST", "/queue", bytes.NewReader(body))
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ saveQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusBadRequest))
+ Expect(w.Body.String()).To(ContainSubstring("current index out of bounds"))
+ })
+
+ It("returns bad request for malformed JSON", func() {
+ req := httptest.NewRequest("POST", "/queue", bytes.NewReader([]byte("invalid json")))
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ saveQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusBadRequest))
+ })
+
+ It("returns internal server error when store fails", func() {
+ repo.Err = true
+ payload := updateQueuePayload{Ids: gg.P([]string{"s1"}), Current: gg.P(0), Position: gg.P(int64(10))}
+ body, _ := json.Marshal(payload)
+ req := httptest.NewRequest("POST", "/queue", bytes.NewReader(body))
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ saveQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusInternalServerError))
+ })
+ })
+
+ Describe("GET /queue", func() {
+ It("returns the queue", func() {
+ queue := &model.PlayQueue{
+ UserID: user.ID,
+ Current: 1,
+ Position: 55,
+ Items: model.MediaFiles{
+ {ID: "track1", Title: "Song 1"},
+ {ID: "track2", Title: "Song 2"},
+ {ID: "track3", Title: "Song 3"},
+ },
+ }
+ repo.Queue = queue
+ req := httptest.NewRequest("GET", "/queue", nil)
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ getQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusOK))
+ Expect(w.Header().Get("Content-Type")).To(Equal("application/json"))
+ var resp model.PlayQueue
+ Expect(json.Unmarshal(w.Body.Bytes(), &resp)).To(Succeed())
+ Expect(resp.Current).To(Equal(1))
+ Expect(resp.Position).To(Equal(int64(55)))
+ Expect(resp.Items).To(HaveLen(3))
+ Expect(resp.Items[0].ID).To(Equal("track1"))
+ Expect(resp.Items[1].ID).To(Equal("track2"))
+ Expect(resp.Items[2].ID).To(Equal("track3"))
+ })
+
+ It("returns empty queue when user has no queue", func() {
+ req := httptest.NewRequest("GET", "/queue", nil)
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ getQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusOK))
+ var resp model.PlayQueue
+ Expect(json.Unmarshal(w.Body.Bytes(), &resp)).To(Succeed())
+ Expect(resp.Items).To(BeEmpty())
+ Expect(resp.Current).To(Equal(0))
+ Expect(resp.Position).To(Equal(int64(0)))
+ })
+
+ It("returns internal server error when retrieve fails", func() {
+ repo.Err = true
+ req := httptest.NewRequest("GET", "/queue", nil)
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ getQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusInternalServerError))
+ })
+ })
+
+ Describe("PUT /queue", func() {
+ It("updates the queue fields", func() {
+ repo.Queue = &model.PlayQueue{UserID: user.ID, Items: model.MediaFiles{{ID: "s1"}, {ID: "s2"}, {ID: "s3"}}}
+ payload := updateQueuePayload{Current: gg.P(2), Position: gg.P(int64(20))}
+ body, _ := json.Marshal(payload)
+ req := httptest.NewRequest("PUT", "/queue", bytes.NewReader(body))
+ ctx := request.WithUser(req.Context(), user)
+ ctx = request.WithClient(ctx, "TestClient")
+ req = req.WithContext(ctx)
+ w := httptest.NewRecorder()
+
+ updateQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusNoContent))
+ Expect(repo.Queue).ToNot(BeNil())
+ Expect(repo.Queue.Current).To(Equal(2))
+ Expect(repo.Queue.Position).To(Equal(int64(20)))
+ Expect(repo.Queue.ChangedBy).To(Equal("TestClient"))
+ })
+
+ It("updates only ids", func() {
+ repo.Queue = &model.PlayQueue{UserID: user.ID, Current: 1}
+ payload := updateQueuePayload{Ids: gg.P([]string{"s1", "s2"})}
+ body, _ := json.Marshal(payload)
+ req := httptest.NewRequest("PUT", "/queue", bytes.NewReader(body))
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ updateQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusNoContent))
+ Expect(repo.Queue.Items).To(HaveLen(2))
+ Expect(repo.LastCols).To(ConsistOf("items"))
+ })
+
+ It("updates ids and current", func() {
+ repo.Queue = &model.PlayQueue{UserID: user.ID}
+ payload := updateQueuePayload{Ids: gg.P([]string{"s1", "s2"}), Current: gg.P(1)}
+ body, _ := json.Marshal(payload)
+ req := httptest.NewRequest("PUT", "/queue", bytes.NewReader(body))
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ updateQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusNoContent))
+ Expect(repo.Queue.Items).To(HaveLen(2))
+ Expect(repo.Queue.Current).To(Equal(1))
+ Expect(repo.LastCols).To(ConsistOf("items", "current"))
+ })
+
+ It("returns bad request when new ids invalidate current", func() {
+ repo.Queue = &model.PlayQueue{UserID: user.ID, Current: 2}
+ payload := updateQueuePayload{Ids: gg.P([]string{"s1", "s2"})}
+ body, _ := json.Marshal(payload)
+ req := httptest.NewRequest("PUT", "/queue", bytes.NewReader(body))
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ updateQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusBadRequest))
+ })
+
+ It("returns bad request when current out of bounds", func() {
+ repo.Queue = &model.PlayQueue{UserID: user.ID, Items: model.MediaFiles{{ID: "s1"}}}
+ payload := updateQueuePayload{Current: gg.P(3)}
+ body, _ := json.Marshal(payload)
+ req := httptest.NewRequest("PUT", "/queue", bytes.NewReader(body))
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ updateQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusBadRequest))
+ })
+
+ It("returns bad request for malformed JSON", func() {
+ req := httptest.NewRequest("PUT", "/queue", bytes.NewReader([]byte("{")))
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ updateQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusBadRequest))
+ })
+
+ It("returns internal server error when store fails", func() {
+ repo.Err = true
+ payload := updateQueuePayload{Position: gg.P(int64(10))}
+ body, _ := json.Marshal(payload)
+ req := httptest.NewRequest("PUT", "/queue", bytes.NewReader(body))
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ updateQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusInternalServerError))
+ })
+ })
+
+ Describe("DELETE /queue", func() {
+ It("clears the queue", func() {
+ repo.Queue = &model.PlayQueue{UserID: user.ID, Items: model.MediaFiles{{ID: "s1"}}}
+ req := httptest.NewRequest("DELETE", "/queue", nil)
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ clearQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusNoContent))
+ Expect(repo.Queue).To(BeNil())
+ })
+
+ It("returns internal server error when clear fails", func() {
+ repo.Err = true
+ req := httptest.NewRequest("DELETE", "/queue", nil)
+ req = req.WithContext(request.WithUser(req.Context(), user))
+ w := httptest.NewRecorder()
+
+ clearQueue(ds)(w, req)
+ Expect(w.Code).To(Equal(http.StatusInternalServerError))
+ })
+ })
+})
diff --git a/server/serve_index.go b/server/serve_index.go
index 9a457ac20..38e646982 100644
--- a/server/serve_index.go
+++ b/server/serve_index.go
@@ -55,6 +55,7 @@ func serveIndex(ds model.DataStore, fs fs.FS, shareInfo *model.Share) http.Handl
"defaultLanguage": conf.Server.DefaultLanguage,
"defaultUIVolume": conf.Server.DefaultUIVolume,
"enableCoverAnimation": conf.Server.EnableCoverAnimation,
+ "enableNowPlaying": conf.Server.EnableNowPlaying,
"gaTrackingId": conf.Server.GATrackingID,
"losslessFormats": strings.ToUpper(strings.Join(mime.LosslessFormats, ",")),
"devActivityPanel": conf.Server.DevActivityPanel,
@@ -65,6 +66,8 @@ func serveIndex(ds model.DataStore, fs fs.FS, shareInfo *model.Share) http.Handl
"devSidebarPlaylists": conf.Server.DevSidebarPlaylists,
"lastFMEnabled": conf.Server.LastFM.Enabled,
"devShowArtistPage": conf.Server.DevShowArtistPage,
+ "devUIShowConfig": conf.Server.DevUIShowConfig,
+ "devNewEventStream": conf.Server.DevNewEventStream,
"listenBrainzEnabled": conf.Server.ListenBrainz.Enabled,
"enableExternalServices": conf.Server.EnableExternalServices,
"enableReplayGain": conf.Server.EnableReplayGain,
diff --git a/server/serve_index_test.go b/server/serve_index_test.go
index 0f02153fd..4f179f22a 100644
--- a/server/serve_index_test.go
+++ b/server/serve_index_test.go
@@ -39,7 +39,7 @@ var _ = Describe("serveIndex", func() {
Expect(w.Code).To(Equal(200))
config := extractAppConfig(w.Body.String())
- Expect(config).To(BeAssignableToTypeOf(map[string]interface{}{}))
+ Expect(config).To(BeAssignableToTypeOf(map[string]any{}))
})
It("sets firstTime = true when User table is empty", func() {
@@ -53,17 +53,6 @@ var _ = Describe("serveIndex", func() {
Expect(config).To(HaveKeyWithValue("firstTime", true))
})
- It("includes the VariousArtistsID", func() {
- mockUser.empty = true
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("variousArtistsId", consts.VariousArtistsID))
- })
-
It("sets firstTime = false when User table is not empty", func() {
mockUser.empty = false
r := httptest.NewRequest("GET", "/index.html", nil)
@@ -75,267 +64,64 @@ var _ = Describe("serveIndex", func() {
Expect(config).To(HaveKeyWithValue("firstTime", false))
})
- It("sets baseURL", func() {
- conf.Server.BasePath = "base_url_test"
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("baseURL", "base_url_test"))
- })
-
- It("sets the welcomeMessage", func() {
- conf.Server.UIWelcomeMessage = "Hello"
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("welcomeMessage", "Hello"))
- })
-
- It("sets the maxSidebarPlaylists", func() {
- conf.Server.MaxSidebarPlaylists = 42
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("maxSidebarPlaylists", float64(42)))
- })
-
- It("sets the enableTranscodingConfig", func() {
- conf.Server.EnableTranscodingConfig = true
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("enableTranscodingConfig", true))
- })
-
- It("sets the enableDownloads", func() {
- conf.Server.EnableDownloads = true
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("enableDownloads", true))
- })
-
- It("sets the enableLoved", func() {
- conf.Server.EnableFavourites = true
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("enableFavourites", true))
- })
-
- It("sets the enableStarRating", func() {
- conf.Server.EnableStarRating = true
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("enableStarRating", true))
- })
-
- It("sets the defaultTheme", func() {
- conf.Server.DefaultTheme = "Light"
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("defaultTheme", "Light"))
- })
-
- It("sets the defaultLanguage", func() {
- conf.Server.DefaultLanguage = "pt"
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("defaultLanguage", "pt"))
- })
-
- It("sets the defaultUIVolume", func() {
- conf.Server.DefaultUIVolume = 45
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("defaultUIVolume", float64(45)))
- })
-
- It("sets the enableCoverAnimation", func() {
- conf.Server.EnableCoverAnimation = true
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("enableCoverAnimation", true))
- })
-
- It("sets the gaTrackingId", func() {
- conf.Server.GATrackingID = "UA-12345"
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("gaTrackingId", "UA-12345"))
- })
-
- It("sets the version", func() {
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("version", consts.Version))
- })
-
- It("sets the losslessFormats", func() {
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- expected := strings.ToUpper(strings.Join(mime.LosslessFormats, ","))
- Expect(config).To(HaveKeyWithValue("losslessFormats", expected))
- })
-
- It("sets the enableUserEditing", func() {
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("enableUserEditing", true))
- })
-
- It("sets the enableSharing", func() {
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("enableSharing", false))
- })
-
- It("sets the defaultDownloadableShare", func() {
- conf.Server.DefaultDownloadableShare = true
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("defaultDownloadableShare", true))
- })
-
- It("sets the defaultDownsamplingFormat", func() {
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("defaultDownsamplingFormat", conf.Server.DefaultDownsamplingFormat))
- })
-
- It("sets the devSidebarPlaylists", func() {
- conf.Server.DevSidebarPlaylists = true
-
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("devSidebarPlaylists", true))
- })
-
- It("sets the lastFMEnabled", func() {
- conf.Server.LastFM.Enabled = true
-
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("lastFMEnabled", true))
- })
-
- It("sets the devShowArtistPage", func() {
- conf.Server.DevShowArtistPage = true
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("devShowArtistPage", true))
- })
-
- It("sets the listenBrainzEnabled", func() {
- conf.Server.ListenBrainz.Enabled = true
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("listenBrainzEnabled", true))
- })
-
- It("sets the enableReplayGain", func() {
- conf.Server.EnableReplayGain = true
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("enableReplayGain", true))
- })
-
- It("sets the enableExternalServices", func() {
- conf.Server.EnableExternalServices = true
- r := httptest.NewRequest("GET", "/index.html", nil)
- w := httptest.NewRecorder()
-
- serveIndex(ds, fs, nil)(w, r)
-
- config := extractAppConfig(w.Body.String())
- Expect(config).To(HaveKeyWithValue("enableExternalServices", true))
- })
+ DescribeTable("sets configuration values",
+ func(configSetter func(), configKey string, expectedValue any) {
+ configSetter()
+ r := httptest.NewRequest("GET", "/index.html", nil)
+ w := httptest.NewRecorder()
+
+ serveIndex(ds, fs, nil)(w, r)
+
+ config := extractAppConfig(w.Body.String())
+ Expect(config).To(HaveKeyWithValue(configKey, expectedValue))
+ },
+ Entry("baseURL", func() { conf.Server.BasePath = "base_url_test" }, "baseURL", "base_url_test"),
+ Entry("welcomeMessage", func() { conf.Server.UIWelcomeMessage = "Hello" }, "welcomeMessage", "Hello"),
+ Entry("maxSidebarPlaylists", func() { conf.Server.MaxSidebarPlaylists = 42 }, "maxSidebarPlaylists", float64(42)),
+ Entry("enableTranscodingConfig", func() { conf.Server.EnableTranscodingConfig = true }, "enableTranscodingConfig", true),
+ Entry("enableDownloads", func() { conf.Server.EnableDownloads = true }, "enableDownloads", true),
+ Entry("enableFavourites", func() { conf.Server.EnableFavourites = true }, "enableFavourites", true),
+ Entry("enableStarRating", func() { conf.Server.EnableStarRating = true }, "enableStarRating", true),
+ Entry("defaultTheme", func() { conf.Server.DefaultTheme = "Light" }, "defaultTheme", "Light"),
+ Entry("defaultLanguage", func() { conf.Server.DefaultLanguage = "pt" }, "defaultLanguage", "pt"),
+ Entry("defaultUIVolume", func() { conf.Server.DefaultUIVolume = 45 }, "defaultUIVolume", float64(45)),
+ Entry("enableCoverAnimation", func() { conf.Server.EnableCoverAnimation = true }, "enableCoverAnimation", true),
+ Entry("enableNowPlaying", func() { conf.Server.EnableNowPlaying = true }, "enableNowPlaying", true),
+ Entry("gaTrackingId", func() { conf.Server.GATrackingID = "UA-12345" }, "gaTrackingId", "UA-12345"),
+ Entry("defaultDownloadableShare", func() { conf.Server.DefaultDownloadableShare = true }, "defaultDownloadableShare", true),
+ Entry("devSidebarPlaylists", func() { conf.Server.DevSidebarPlaylists = true }, "devSidebarPlaylists", true),
+ Entry("lastFMEnabled", func() { conf.Server.LastFM.Enabled = true }, "lastFMEnabled", true),
+ Entry("devShowArtistPage", func() { conf.Server.DevShowArtistPage = true }, "devShowArtistPage", true),
+ Entry("devUIShowConfig", func() { conf.Server.DevUIShowConfig = true }, "devUIShowConfig", true),
+ Entry("listenBrainzEnabled", func() { conf.Server.ListenBrainz.Enabled = true }, "listenBrainzEnabled", true),
+ Entry("enableReplayGain", func() { conf.Server.EnableReplayGain = true }, "enableReplayGain", true),
+ Entry("enableExternalServices", func() { conf.Server.EnableExternalServices = true }, "enableExternalServices", true),
+ Entry("devActivityPanel", func() { conf.Server.DevActivityPanel = true }, "devActivityPanel", true),
+ Entry("shareURL", func() { conf.Server.ShareURL = "https://share.example.com" }, "shareURL", "https://share.example.com"),
+ Entry("enableInspect", func() { conf.Server.Inspect.Enabled = true }, "enableInspect", true),
+ Entry("defaultDownsamplingFormat", func() { conf.Server.DefaultDownsamplingFormat = "mp3" }, "defaultDownsamplingFormat", "mp3"),
+ Entry("enableUserEditing", func() { conf.Server.EnableUserEditing = false }, "enableUserEditing", false),
+ Entry("enableSharing", func() { conf.Server.EnableSharing = true }, "enableSharing", true),
+ Entry("devNewEventStream", func() { conf.Server.DevNewEventStream = true }, "devNewEventStream", true),
+ )
+
+ DescribeTable("sets other UI configuration values",
+ func(configKey string, expectedValueFunc func() any) {
+ r := httptest.NewRequest("GET", "/index.html", nil)
+ w := httptest.NewRecorder()
+
+ serveIndex(ds, fs, nil)(w, r)
+
+ config := extractAppConfig(w.Body.String())
+ Expect(config).To(HaveKeyWithValue(configKey, expectedValueFunc()))
+ },
+ Entry("version", "version", func() any { return consts.Version }),
+ Entry("variousArtistsId", "variousArtistsId", func() any { return consts.VariousArtistsID }),
+ Entry("losslessFormats", "losslessFormats", func() any {
+ return strings.ToUpper(strings.Join(mime.LosslessFormats, ","))
+ }),
+ Entry("separator", "separator", func() any { return string(os.PathSeparator) }),
+ )
Describe("loginBackgroundURL", func() {
Context("empty BaseURL", func() {
@@ -426,12 +212,12 @@ var _ = Describe("serveIndex", func() {
var _ = Describe("addShareData", func() {
var (
r *http.Request
- data map[string]interface{}
+ data map[string]any
shareInfo *model.Share
)
BeforeEach(func() {
- data = make(map[string]interface{})
+ data = make(map[string]any)
r = httptest.NewRequest("GET", "/", nil)
})
@@ -516,8 +302,8 @@ var _ = Describe("addShareData", func() {
var appConfigRegex = regexp.MustCompile(`(?m)window.__APP_CONFIG__=(.*);`)
-func extractAppConfig(body string) map[string]interface{} {
- config := make(map[string]interface{})
+func extractAppConfig(body string) map[string]any {
+ config := make(map[string]any)
match := appConfigRegex.FindStringSubmatch(body)
if match == nil {
return config
diff --git a/server/server.go b/server/server.go
index 60350b6b4..49391e2b6 100644
--- a/server/server.go
+++ b/server/server.go
@@ -173,7 +173,7 @@ func (s *Server) initRoutes() {
clientUniqueIDMiddleware,
compressMiddleware(),
loggerInjector,
- jwtVerifier,
+ JWTVerifier,
}
// Mount the Native API /events endpoint with all default middlewares, adding the authentication middlewares
diff --git a/server/subsonic/album_lists.go b/server/subsonic/album_lists.go
index 39a164500..56cf469c5 100644
--- a/server/subsonic/album_lists.go
+++ b/server/subsonic/album_lists.go
@@ -12,6 +12,7 @@ import (
"github.com/navidrome/navidrome/server/subsonic/filter"
"github.com/navidrome/navidrome/server/subsonic/responses"
"github.com/navidrome/navidrome/utils/req"
+ "github.com/navidrome/navidrome/utils/run"
"github.com/navidrome/navidrome/utils/slice"
)
@@ -61,6 +62,13 @@ func (api *Router) getAlbumList(r *http.Request) (model.Albums, int64, error) {
return nil, 0, newError(responses.ErrorGeneric, "type '%s' not implemented", typ)
}
+ // Get optional library IDs from musicFolderId parameter
+ musicFolderIds, err := selectedMusicFolderIds(r, false)
+ if err != nil {
+ return nil, 0, err
+ }
+ opts = filter.ApplyLibraryFilter(opts, musicFolderIds)
+
opts.Offset = p.IntOr("offset", 0)
opts.Max = min(p.IntOr("size", 10), 500)
albums, err := api.ds.Album(r.Context()).GetAll(opts)
@@ -109,57 +117,87 @@ func (api *Router) GetAlbumList2(w http.ResponseWriter, r *http.Request) (*respo
return response, nil
}
-func (api *Router) GetStarred(r *http.Request) (*responses.Subsonic, error) {
+func (api *Router) getStarredItems(r *http.Request) (model.Artists, model.Albums, model.MediaFiles, error) {
ctx := r.Context()
- artists, err := api.ds.Artist(ctx).GetAll(filter.ArtistsByStarred())
+
+ // Get optional library IDs from musicFolderId parameter
+ musicFolderIds, err := selectedMusicFolderIds(r, false)
if err != nil {
- log.Error(r, "Error retrieving starred artists", err)
- return nil, err
+ return nil, nil, nil, err
}
- options := filter.ByStarred()
- albums, err := api.ds.Album(ctx).GetAll(options)
+
+ // Prepare variables to capture results from parallel execution
+ var artists model.Artists
+ var albums model.Albums
+ var mediaFiles model.MediaFiles
+
+ // Execute all three queries in parallel for better performance
+ err = run.Parallel(
+ // Query starred artists
+ func() error {
+ artistOpts := filter.ApplyArtistLibraryFilter(filter.ArtistsByStarred(), musicFolderIds)
+ var err error
+ artists, err = api.ds.Artist(ctx).GetAll(artistOpts)
+ if err != nil {
+ log.Error(r, "Error retrieving starred artists", err)
+ }
+ return err
+ },
+ // Query starred albums
+ func() error {
+ albumOpts := filter.ApplyLibraryFilter(filter.ByStarred(), musicFolderIds)
+ var err error
+ albums, err = api.ds.Album(ctx).GetAll(albumOpts)
+ if err != nil {
+ log.Error(r, "Error retrieving starred albums", err)
+ }
+ return err
+ },
+ // Query starred media files
+ func() error {
+ mediaFileOpts := filter.ApplyLibraryFilter(filter.ByStarred(), musicFolderIds)
+ var err error
+ mediaFiles, err = api.ds.MediaFile(ctx).GetAll(mediaFileOpts)
+ if err != nil {
+ log.Error(r, "Error retrieving starred mediaFiles", err)
+ }
+ return err
+ },
+ )()
+
+ // Return the first error if any occurred
if err != nil {
- log.Error(r, "Error retrieving starred albums", err)
- return nil, err
+ return nil, nil, nil, err
}
- mediaFiles, err := api.ds.MediaFile(ctx).GetAll(options)
+
+ return artists, albums, mediaFiles, nil
+}
+
+func (api *Router) GetStarred(r *http.Request) (*responses.Subsonic, error) {
+ artists, albums, mediaFiles, err := api.getStarredItems(r)
if err != nil {
- log.Error(r, "Error retrieving starred mediaFiles", err)
return nil, err
}
response := newResponse()
response.Starred = &responses.Starred{}
response.Starred.Artist = slice.MapWithArg(artists, r, toArtist)
- response.Starred.Album = slice.MapWithArg(albums, ctx, childFromAlbum)
- response.Starred.Song = slice.MapWithArg(mediaFiles, ctx, childFromMediaFile)
+ response.Starred.Album = slice.MapWithArg(albums, r.Context(), childFromAlbum)
+ response.Starred.Song = slice.MapWithArg(mediaFiles, r.Context(), childFromMediaFile)
return response, nil
}
func (api *Router) GetStarred2(r *http.Request) (*responses.Subsonic, error) {
- ctx := r.Context()
- artists, err := api.ds.Artist(ctx).GetAll(filter.ArtistsByStarred())
+ artists, albums, mediaFiles, err := api.getStarredItems(r)
if err != nil {
- log.Error(r, "Error retrieving starred artists", err)
- return nil, err
- }
- options := filter.ByStarred()
- albums, err := api.ds.Album(ctx).GetAll(options)
- if err != nil {
- log.Error(r, "Error retrieving starred albums", err)
- return nil, err
- }
- mediaFiles, err := api.ds.MediaFile(ctx).GetAll(options)
- if err != nil {
- log.Error(r, "Error retrieving starred mediaFiles", err)
return nil, err
}
response := newResponse()
response.Starred2 = &responses.Starred2{}
response.Starred2.Artist = slice.MapWithArg(artists, r, toArtistID3)
- response.Starred2.Album = slice.MapWithArg(albums, ctx, buildAlbumID3)
- response.Starred2.Song = slice.MapWithArg(mediaFiles, ctx, childFromMediaFile)
+ response.Starred2.Album = slice.MapWithArg(albums, r.Context(), buildAlbumID3)
+ response.Starred2.Song = slice.MapWithArg(mediaFiles, r.Context(), childFromMediaFile)
return response, nil
}
@@ -193,7 +231,15 @@ func (api *Router) GetRandomSongs(r *http.Request) (*responses.Subsonic, error)
fromYear := p.IntOr("fromYear", 0)
toYear := p.IntOr("toYear", 0)
- songs, err := api.getSongs(r.Context(), 0, size, filter.SongsByRandom(genre, fromYear, toYear))
+ // Get optional library IDs from musicFolderId parameter
+ musicFolderIds, err := selectedMusicFolderIds(r, false)
+ if err != nil {
+ return nil, err
+ }
+ opts := filter.SongsByRandom(genre, fromYear, toYear)
+ opts = filter.ApplyLibraryFilter(opts, musicFolderIds)
+
+ songs, err := api.getSongs(r.Context(), 0, size, opts)
if err != nil {
log.Error(r, "Error retrieving random songs", err)
return nil, err
@@ -211,8 +257,16 @@ func (api *Router) GetSongsByGenre(r *http.Request) (*responses.Subsonic, error)
offset := p.IntOr("offset", 0)
genre, _ := p.String("genre")
+ // Get optional library IDs from musicFolderId parameter
+ musicFolderIds, err := selectedMusicFolderIds(r, false)
+ if err != nil {
+ return nil, err
+ }
+ opts := filter.ByGenre(genre)
+ opts = filter.ApplyLibraryFilter(opts, musicFolderIds)
+
ctx := r.Context()
- songs, err := api.getSongs(ctx, offset, count, filter.ByGenre(genre))
+ songs, err := api.getSongs(ctx, offset, count, opts)
if err != nil {
log.Error(r, "Error retrieving random songs", err)
return nil, err
diff --git a/server/subsonic/album_lists_test.go b/server/subsonic/album_lists_test.go
index f187555e9..63c2614cd 100644
--- a/server/subsonic/album_lists_test.go
+++ b/server/subsonic/album_lists_test.go
@@ -5,12 +5,13 @@ import (
"errors"
"net/http/httptest"
- "github.com/navidrome/navidrome/server/subsonic/responses"
- "github.com/navidrome/navidrome/utils/req"
-
+ "github.com/navidrome/navidrome/core/auth"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/server/subsonic/responses"
"github.com/navidrome/navidrome/tests"
+ "github.com/navidrome/navidrome/utils/req"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
)
@@ -24,8 +25,9 @@ var _ = Describe("Album Lists", func() {
BeforeEach(func() {
ds = &tests.MockDataStore{}
+ auth.Init(ds)
mockRepo = ds.Album(ctx).(*tests.MockAlbumRepo)
- router = New(ds, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil)
+ router = New(ds, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil)
w = httptest.NewRecorder()
})
@@ -63,6 +65,74 @@ var _ = Describe("Album Lists", func() {
errors.As(err, &subErr)
Expect(subErr.code).To(Equal(responses.ErrorGeneric))
})
+
+ Context("with musicFolderId parameter", func() {
+ var user model.User
+ var ctx context.Context
+
+ BeforeEach(func() {
+ user = model.User{
+ ID: "test-user",
+ Libraries: []model.Library{
+ {ID: 1, Name: "Library 1"},
+ {ID: 2, Name: "Library 2"},
+ {ID: 3, Name: "Library 3"},
+ },
+ }
+ ctx = request.WithUser(context.Background(), user)
+ })
+
+ It("should filter albums by specific library when musicFolderId is provided", func() {
+ r := newGetRequest("type=newest", "musicFolderId=1")
+ r = r.WithContext(ctx)
+ mockRepo.SetData(model.Albums{
+ {ID: "1"}, {ID: "2"},
+ })
+
+ resp, err := router.GetAlbumList(w, r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.AlbumList.Album).To(HaveLen(2))
+ // Verify that library filter was applied
+ query, args, _ := mockRepo.Options.Filters.ToSql()
+ Expect(query).To(ContainSubstring("library_id IN (?)"))
+ Expect(args).To(ContainElement(1))
+ })
+
+ It("should filter albums by multiple libraries when multiple musicFolderId are provided", func() {
+ r := newGetRequest("type=newest", "musicFolderId=1", "musicFolderId=2")
+ r = r.WithContext(ctx)
+ mockRepo.SetData(model.Albums{
+ {ID: "1"}, {ID: "2"},
+ })
+
+ resp, err := router.GetAlbumList(w, r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.AlbumList.Album).To(HaveLen(2))
+ // Verify that library filter was applied
+ query, args, _ := mockRepo.Options.Filters.ToSql()
+ Expect(query).To(ContainSubstring("library_id IN (?,?)"))
+ Expect(args).To(ContainElements(1, 2))
+ })
+
+ It("should return all accessible albums when no musicFolderId is provided", func() {
+ r := newGetRequest("type=newest")
+ r = r.WithContext(ctx)
+ mockRepo.SetData(model.Albums{
+ {ID: "1"}, {ID: "2"},
+ })
+
+ resp, err := router.GetAlbumList(w, r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.AlbumList.Album).To(HaveLen(2))
+ // Verify that library filter was applied
+ query, args, _ := mockRepo.Options.Filters.ToSql()
+ Expect(query).To(ContainSubstring("library_id IN (?,?,?)"))
+ Expect(args).To(ContainElements(1, 2, 3))
+ })
+ })
})
Describe("GetAlbumList2", func() {
@@ -100,5 +170,373 @@ var _ = Describe("Album Lists", func() {
errors.As(err, &subErr)
Expect(subErr.code).To(Equal(responses.ErrorGeneric))
})
+
+ Context("with musicFolderId parameter", func() {
+ var user model.User
+ var ctx context.Context
+
+ BeforeEach(func() {
+ user = model.User{
+ ID: "test-user",
+ Libraries: []model.Library{
+ {ID: 1, Name: "Library 1"},
+ {ID: 2, Name: "Library 2"},
+ {ID: 3, Name: "Library 3"},
+ },
+ }
+ ctx = request.WithUser(context.Background(), user)
+ })
+
+ It("should filter albums by specific library when musicFolderId is provided", func() {
+ r := newGetRequest("type=newest", "musicFolderId=1")
+ r = r.WithContext(ctx)
+ mockRepo.SetData(model.Albums{
+ {ID: "1"}, {ID: "2"},
+ })
+
+ resp, err := router.GetAlbumList2(w, r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.AlbumList2.Album).To(HaveLen(2))
+ // Verify that library filter was applied
+ Expect(mockRepo.Options.Filters).ToNot(BeNil())
+ })
+
+ It("should filter albums by multiple libraries when multiple musicFolderId are provided", func() {
+ r := newGetRequest("type=newest", "musicFolderId=1", "musicFolderId=2")
+ r = r.WithContext(ctx)
+ mockRepo.SetData(model.Albums{
+ {ID: "1"}, {ID: "2"},
+ })
+
+ resp, err := router.GetAlbumList2(w, r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.AlbumList2.Album).To(HaveLen(2))
+ // Verify that library filter was applied
+ Expect(mockRepo.Options.Filters).ToNot(BeNil())
+ })
+
+ It("should return all accessible albums when no musicFolderId is provided", func() {
+ r := newGetRequest("type=newest")
+ r = r.WithContext(ctx)
+ mockRepo.SetData(model.Albums{
+ {ID: "1"}, {ID: "2"},
+ })
+
+ resp, err := router.GetAlbumList2(w, r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.AlbumList2.Album).To(HaveLen(2))
+ })
+ })
+ })
+
+ Describe("GetRandomSongs", func() {
+ var mockMediaFileRepo *tests.MockMediaFileRepo
+
+ BeforeEach(func() {
+ mockMediaFileRepo = ds.MediaFile(ctx).(*tests.MockMediaFileRepo)
+ })
+
+ It("should return random songs", func() {
+ mockMediaFileRepo.SetData(model.MediaFiles{
+ {ID: "1", Title: "Song 1"},
+ {ID: "2", Title: "Song 2"},
+ })
+ r := newGetRequest("size=2")
+
+ resp, err := router.GetRandomSongs(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.RandomSongs.Songs).To(HaveLen(2))
+ })
+
+ Context("with musicFolderId parameter", func() {
+ var user model.User
+ var ctx context.Context
+
+ BeforeEach(func() {
+ user = model.User{
+ ID: "test-user",
+ Libraries: []model.Library{
+ {ID: 1, Name: "Library 1"},
+ {ID: 2, Name: "Library 2"},
+ {ID: 3, Name: "Library 3"},
+ },
+ }
+ ctx = request.WithUser(context.Background(), user)
+ })
+
+ It("should filter songs by specific library when musicFolderId is provided", func() {
+ mockMediaFileRepo.SetData(model.MediaFiles{
+ {ID: "1", Title: "Song 1"},
+ {ID: "2", Title: "Song 2"},
+ })
+ r := newGetRequest("size=2", "musicFolderId=1")
+ r = r.WithContext(ctx)
+
+ resp, err := router.GetRandomSongs(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.RandomSongs.Songs).To(HaveLen(2))
+ // Verify that library filter was applied
+ query, args, _ := mockMediaFileRepo.Options.Filters.ToSql()
+ Expect(query).To(ContainSubstring("library_id IN (?)"))
+ Expect(args).To(ContainElement(1))
+ })
+
+ It("should filter songs by multiple libraries when multiple musicFolderId are provided", func() {
+ mockMediaFileRepo.SetData(model.MediaFiles{
+ {ID: "1", Title: "Song 1"},
+ {ID: "2", Title: "Song 2"},
+ })
+ r := newGetRequest("size=2", "musicFolderId=1", "musicFolderId=2")
+ r = r.WithContext(ctx)
+
+ resp, err := router.GetRandomSongs(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.RandomSongs.Songs).To(HaveLen(2))
+ // Verify that library filter was applied
+ query, args, _ := mockMediaFileRepo.Options.Filters.ToSql()
+ Expect(query).To(ContainSubstring("library_id IN (?,?)"))
+ Expect(args).To(ContainElements(1, 2))
+ })
+
+ It("should return all accessible songs when no musicFolderId is provided", func() {
+ mockMediaFileRepo.SetData(model.MediaFiles{
+ {ID: "1", Title: "Song 1"},
+ {ID: "2", Title: "Song 2"},
+ })
+ r := newGetRequest("size=2")
+ r = r.WithContext(ctx)
+
+ resp, err := router.GetRandomSongs(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.RandomSongs.Songs).To(HaveLen(2))
+ // Verify that library filter was applied
+ query, args, _ := mockMediaFileRepo.Options.Filters.ToSql()
+ Expect(query).To(ContainSubstring("library_id IN (?,?,?)"))
+ Expect(args).To(ContainElements(1, 2, 3))
+ })
+ })
+ })
+
+ Describe("GetSongsByGenre", func() {
+ var mockMediaFileRepo *tests.MockMediaFileRepo
+
+ BeforeEach(func() {
+ mockMediaFileRepo = ds.MediaFile(ctx).(*tests.MockMediaFileRepo)
+ })
+
+ It("should return songs by genre", func() {
+ mockMediaFileRepo.SetData(model.MediaFiles{
+ {ID: "1", Title: "Song 1"},
+ {ID: "2", Title: "Song 2"},
+ })
+ r := newGetRequest("count=2", "genre=rock")
+
+ resp, err := router.GetSongsByGenre(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.SongsByGenre.Songs).To(HaveLen(2))
+ })
+
+ Context("with musicFolderId parameter", func() {
+ var user model.User
+ var ctx context.Context
+
+ BeforeEach(func() {
+ user = model.User{
+ ID: "test-user",
+ Libraries: []model.Library{
+ {ID: 1, Name: "Library 1"},
+ {ID: 2, Name: "Library 2"},
+ {ID: 3, Name: "Library 3"},
+ },
+ }
+ ctx = request.WithUser(context.Background(), user)
+ })
+
+ It("should filter songs by specific library when musicFolderId is provided", func() {
+ mockMediaFileRepo.SetData(model.MediaFiles{
+ {ID: "1", Title: "Song 1"},
+ {ID: "2", Title: "Song 2"},
+ })
+ r := newGetRequest("count=2", "genre=rock", "musicFolderId=1")
+ r = r.WithContext(ctx)
+
+ resp, err := router.GetSongsByGenre(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.SongsByGenre.Songs).To(HaveLen(2))
+ // Verify that library filter was applied
+ query, args, _ := mockMediaFileRepo.Options.Filters.ToSql()
+ Expect(query).To(ContainSubstring("library_id IN (?)"))
+ Expect(args).To(ContainElement(1))
+ })
+
+ It("should filter songs by multiple libraries when multiple musicFolderId are provided", func() {
+ mockMediaFileRepo.SetData(model.MediaFiles{
+ {ID: "1", Title: "Song 1"},
+ {ID: "2", Title: "Song 2"},
+ })
+ r := newGetRequest("count=2", "genre=rock", "musicFolderId=1", "musicFolderId=2")
+ r = r.WithContext(ctx)
+
+ resp, err := router.GetSongsByGenre(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.SongsByGenre.Songs).To(HaveLen(2))
+ // Verify that library filter was applied
+ query, args, _ := mockMediaFileRepo.Options.Filters.ToSql()
+ Expect(query).To(ContainSubstring("library_id IN (?,?)"))
+ Expect(args).To(ContainElements(1, 2))
+ })
+
+ It("should return all accessible songs when no musicFolderId is provided", func() {
+ mockMediaFileRepo.SetData(model.MediaFiles{
+ {ID: "1", Title: "Song 1"},
+ {ID: "2", Title: "Song 2"},
+ })
+ r := newGetRequest("count=2", "genre=rock")
+ r = r.WithContext(ctx)
+
+ resp, err := router.GetSongsByGenre(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.SongsByGenre.Songs).To(HaveLen(2))
+ // Verify that library filter was applied
+ query, args, _ := mockMediaFileRepo.Options.Filters.ToSql()
+ Expect(query).To(ContainSubstring("library_id IN (?,?,?)"))
+ Expect(args).To(ContainElements(1, 2, 3))
+ })
+ })
+ })
+
+ Describe("GetStarred", func() {
+ var mockArtistRepo *tests.MockArtistRepo
+ var mockAlbumRepo *tests.MockAlbumRepo
+ var mockMediaFileRepo *tests.MockMediaFileRepo
+
+ BeforeEach(func() {
+ mockArtistRepo = ds.Artist(ctx).(*tests.MockArtistRepo)
+ mockAlbumRepo = ds.Album(ctx).(*tests.MockAlbumRepo)
+ mockMediaFileRepo = ds.MediaFile(ctx).(*tests.MockMediaFileRepo)
+ })
+
+ It("should return starred items", func() {
+ mockArtistRepo.SetData(model.Artists{{ID: "1", Name: "Artist 1"}})
+ mockAlbumRepo.SetData(model.Albums{{ID: "1", Name: "Album 1"}})
+ mockMediaFileRepo.SetData(model.MediaFiles{{ID: "1", Title: "Song 1"}})
+ r := newGetRequest()
+
+ resp, err := router.GetStarred(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.Starred.Artist).To(HaveLen(1))
+ Expect(resp.Starred.Album).To(HaveLen(1))
+ Expect(resp.Starred.Song).To(HaveLen(1))
+ })
+
+ Context("with musicFolderId parameter", func() {
+ var user model.User
+ var ctx context.Context
+
+ BeforeEach(func() {
+ user = model.User{
+ ID: "test-user",
+ Libraries: []model.Library{
+ {ID: 1, Name: "Library 1"},
+ {ID: 2, Name: "Library 2"},
+ {ID: 3, Name: "Library 3"},
+ },
+ }
+ ctx = request.WithUser(context.Background(), user)
+ })
+
+ It("should filter starred items by specific library when musicFolderId is provided", func() {
+ mockArtistRepo.SetData(model.Artists{{ID: "1", Name: "Artist 1"}})
+ mockAlbumRepo.SetData(model.Albums{{ID: "1", Name: "Album 1"}})
+ mockMediaFileRepo.SetData(model.MediaFiles{{ID: "1", Title: "Song 1"}})
+ r := newGetRequest("musicFolderId=1")
+ r = r.WithContext(ctx)
+
+ resp, err := router.GetStarred(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.Starred.Artist).To(HaveLen(1))
+ Expect(resp.Starred.Album).To(HaveLen(1))
+ Expect(resp.Starred.Song).To(HaveLen(1))
+ // Verify that library filter was applied to all types
+ artistQuery, artistArgs, _ := mockArtistRepo.Options.Filters.ToSql()
+ Expect(artistQuery).To(ContainSubstring("library_id IN (?)"))
+ Expect(artistArgs).To(ContainElement(1))
+ })
+ })
+ })
+
+ Describe("GetStarred2", func() {
+ var mockArtistRepo *tests.MockArtistRepo
+ var mockAlbumRepo *tests.MockAlbumRepo
+ var mockMediaFileRepo *tests.MockMediaFileRepo
+
+ BeforeEach(func() {
+ mockArtistRepo = ds.Artist(ctx).(*tests.MockArtistRepo)
+ mockAlbumRepo = ds.Album(ctx).(*tests.MockAlbumRepo)
+ mockMediaFileRepo = ds.MediaFile(ctx).(*tests.MockMediaFileRepo)
+ })
+
+ It("should return starred items in ID3 format", func() {
+ mockArtistRepo.SetData(model.Artists{{ID: "1", Name: "Artist 1"}})
+ mockAlbumRepo.SetData(model.Albums{{ID: "1", Name: "Album 1"}})
+ mockMediaFileRepo.SetData(model.MediaFiles{{ID: "1", Title: "Song 1"}})
+ r := newGetRequest()
+
+ resp, err := router.GetStarred2(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.Starred2.Artist).To(HaveLen(1))
+ Expect(resp.Starred2.Album).To(HaveLen(1))
+ Expect(resp.Starred2.Song).To(HaveLen(1))
+ })
+
+ Context("with musicFolderId parameter", func() {
+ var user model.User
+ var ctx context.Context
+
+ BeforeEach(func() {
+ user = model.User{
+ ID: "test-user",
+ Libraries: []model.Library{
+ {ID: 1, Name: "Library 1"},
+ {ID: 2, Name: "Library 2"},
+ {ID: 3, Name: "Library 3"},
+ },
+ }
+ ctx = request.WithUser(context.Background(), user)
+ })
+
+ It("should filter starred items by specific library when musicFolderId is provided", func() {
+ mockArtistRepo.SetData(model.Artists{{ID: "1", Name: "Artist 1"}})
+ mockAlbumRepo.SetData(model.Albums{{ID: "1", Name: "Album 1"}})
+ mockMediaFileRepo.SetData(model.MediaFiles{{ID: "1", Title: "Song 1"}})
+ r := newGetRequest("musicFolderId=1")
+ r = r.WithContext(ctx)
+
+ resp, err := router.GetStarred2(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp.Starred2.Artist).To(HaveLen(1))
+ Expect(resp.Starred2.Album).To(HaveLen(1))
+ Expect(resp.Starred2.Song).To(HaveLen(1))
+ // Verify that library filter was applied to all types
+ artistQuery, artistArgs, _ := mockArtistRepo.Options.Filters.ToSql()
+ Expect(artistQuery).To(ContainSubstring("library_id IN (?)"))
+ Expect(artistArgs).To(ContainElement(1))
+ })
+ })
})
})
diff --git a/server/subsonic/api.go b/server/subsonic/api.go
index fd8c3af28..f0e73c3d2 100644
--- a/server/subsonic/api.go
+++ b/server/subsonic/api.go
@@ -13,11 +13,11 @@ import (
"github.com/navidrome/navidrome/core"
"github.com/navidrome/navidrome/core/artwork"
"github.com/navidrome/navidrome/core/external"
+ "github.com/navidrome/navidrome/core/metrics"
"github.com/navidrome/navidrome/core/playback"
"github.com/navidrome/navidrome/core/scrobbler"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
- "github.com/navidrome/navidrome/scanner"
"github.com/navidrome/navidrome/server"
"github.com/navidrome/navidrome/server/events"
"github.com/navidrome/navidrome/server/subsonic/responses"
@@ -38,16 +38,18 @@ type Router struct {
players core.Players
provider external.Provider
playlists core.Playlists
- scanner scanner.Scanner
+ scanner model.Scanner
broker events.Broker
scrobbler scrobbler.PlayTracker
share core.Share
playback playback.PlaybackServer
+ metrics metrics.Metrics
}
func New(ds model.DataStore, artwork artwork.Artwork, streamer core.MediaStreamer, archiver core.Archiver,
- players core.Players, provider external.Provider, scanner scanner.Scanner, broker events.Broker,
+ players core.Players, provider external.Provider, scanner model.Scanner, broker events.Broker,
playlists core.Playlists, scrobbler scrobbler.PlayTracker, share core.Share, playback playback.PlaybackServer,
+ metrics metrics.Metrics,
) *Router {
r := &Router{
ds: ds,
@@ -62,6 +64,7 @@ func New(ds model.DataStore, artwork artwork.Artwork, streamer core.MediaStreame
scrobbler: scrobbler,
share: share,
playback: playback,
+ metrics: metrics,
}
r.Handler = r.routes()
return r
@@ -69,6 +72,11 @@ func New(ds model.DataStore, artwork artwork.Artwork, streamer core.MediaStreame
func (api *Router) routes() http.Handler {
r := chi.NewRouter()
+
+ if conf.Server.Prometheus.Enabled {
+ r.Use(recordStats(api.metrics))
+ }
+
r.Use(postFormToQueryParams)
// Public
@@ -110,7 +118,11 @@ func (api *Router) routes() http.Handler {
hr(r, "getAlbumList2", api.GetAlbumList2)
h(r, "getStarred", api.GetStarred)
h(r, "getStarred2", api.GetStarred2)
- h(r, "getNowPlaying", api.GetNowPlaying)
+ if conf.Server.EnableNowPlaying {
+ h(r, "getNowPlaying", api.GetNowPlaying)
+ } else {
+ h501(r, "getNowPlaying")
+ }
h(r, "getRandomSongs", api.GetRandomSongs)
h(r, "getSongsByGenre", api.GetSongsByGenre)
})
@@ -135,7 +147,9 @@ func (api *Router) routes() http.Handler {
h(r, "createBookmark", api.CreateBookmark)
h(r, "deleteBookmark", api.DeleteBookmark)
h(r, "getPlayQueue", api.GetPlayQueue)
+ h(r, "getPlayQueueByIndex", api.GetPlayQueueByIndex)
h(r, "savePlayQueue", api.SavePlayQueue)
+ h(r, "savePlayQueueByIndex", api.SavePlayQueueByIndex)
})
r.Group(func(r chi.Router) {
r.Use(getPlayer(api.players))
@@ -219,7 +233,7 @@ func h(r chi.Router, path string, f handler) {
})
}
-// Add a Subsonic handler that requires a http.ResponseWriter (ex: stream, getCoverArt...)
+// Add a Subsonic handler that requires an http.ResponseWriter (ex: stream, getCoverArt...)
func hr(r chi.Router, path string, f handlerRaw) {
handle := func(w http.ResponseWriter, r *http.Request) {
res, err := f(w, r)
@@ -320,6 +334,7 @@ func sendResponse(w http.ResponseWriter, r *http.Request, payload *responses.Sub
sendError(w, r, err)
return
}
+
if payload.Status == responses.StatusOK {
if log.IsGreaterOrEqualTo(log.LevelTrace) {
log.Debug(r.Context(), "API: Successful response", "endpoint", r.URL.Path, "status", "OK", "body", string(response))
@@ -329,6 +344,17 @@ func sendResponse(w http.ResponseWriter, r *http.Request, payload *responses.Sub
} else {
log.Warn(r.Context(), "API: Failed response", "endpoint", r.URL.Path, "error", payload.Error.Code, "message", payload.Error.Message)
}
+
+ statusPointer, ok := r.Context().Value(subsonicErrorPointer).(*int32)
+
+ if ok && statusPointer != nil {
+ if payload.Status == responses.StatusOK {
+ *statusPointer = 0
+ } else {
+ *statusPointer = payload.Error.Code
+ }
+ }
+
if _, err := w.Write(response); err != nil {
log.Error(r, "Error sending response to client", "endpoint", r.URL.Path, "payload", string(response), err)
}
diff --git a/server/subsonic/api_test.go b/server/subsonic/api_test.go
index 5d248c464..eaecd7c06 100644
--- a/server/subsonic/api_test.go
+++ b/server/subsonic/api_test.go
@@ -9,8 +9,10 @@ import (
"strings"
"github.com/navidrome/navidrome/server/subsonic/responses"
+ "github.com/navidrome/navidrome/utils/gg"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
+ "golang.org/x/net/context"
)
var _ = Describe("sendResponse", func() {
@@ -91,7 +93,7 @@ var _ = Describe("sendResponse", func() {
It("should return a fail response", func() {
payload.Song = &responses.Child{OpenSubsonicChild: &responses.OpenSubsonicChild{}}
// An +Inf value will cause an error when marshalling to JSON
- payload.Song.ReplayGain = responses.ReplayGain{TrackGain: math.Inf(1)}
+ payload.Song.ReplayGain = responses.ReplayGain{TrackGain: gg.P(math.Inf(1))}
q := r.URL.Query()
q.Add("f", "json")
r.URL.RawQuery = q.Encode()
@@ -108,4 +110,18 @@ var _ = Describe("sendResponse", func() {
})
})
+ It("updates status pointer when an error occurs", func() {
+ pointer := int32(0)
+
+ ctx := context.WithValue(r.Context(), subsonicErrorPointer, &pointer)
+ r = r.WithContext(ctx)
+
+ payload.Status = responses.StatusFailed
+ payload.Error = &responses.Error{Code: responses.ErrorDataNotFound}
+
+ sendResponse(w, r, payload)
+ Expect(w.Code).To(Equal(http.StatusOK))
+
+ Expect(pointer).To(Equal(responses.ErrorDataNotFound))
+ })
})
diff --git a/server/subsonic/bookmarks.go b/server/subsonic/bookmarks.go
index f6fd1a99e..b1e71b1c7 100644
--- a/server/subsonic/bookmarks.go
+++ b/server/subsonic/bookmarks.go
@@ -73,7 +73,7 @@ func (api *Router) GetPlayQueue(r *http.Request) (*responses.Subsonic, error) {
user, _ := request.UserFrom(r.Context())
repo := api.ds.PlayQueue(r.Context())
- pq, err := repo.Retrieve(user.ID)
+ pq, err := repo.RetrieveWithMediaFiles(user.ID)
if err != nil && !errors.Is(err, model.ErrNotFound) {
return nil, err
}
@@ -82,12 +82,16 @@ func (api *Router) GetPlayQueue(r *http.Request) (*responses.Subsonic, error) {
}
response := newResponse()
+ var currentID string
+ if pq.Current >= 0 && pq.Current < len(pq.Items) {
+ currentID = pq.Items[pq.Current].ID
+ }
response.PlayQueue = &responses.PlayQueue{
Entry: slice.MapWithArg(pq.Items, r.Context(), childFromMediaFile),
- Current: pq.Current,
+ Current: currentID,
Position: pq.Position,
Username: user.UserName,
- Changed: &pq.UpdatedAt,
+ Changed: pq.UpdatedAt,
ChangedBy: pq.ChangedBy,
}
return response, nil
@@ -96,20 +100,27 @@ func (api *Router) GetPlayQueue(r *http.Request) (*responses.Subsonic, error) {
func (api *Router) SavePlayQueue(r *http.Request) (*responses.Subsonic, error) {
p := req.Params(r)
ids, _ := p.Strings("id")
- current, _ := p.String("current")
+ currentID, _ := p.String("current")
position := p.Int64Or("position", 0)
user, _ := request.UserFrom(r.Context())
client, _ := request.ClientFrom(r.Context())
- var items model.MediaFiles
- for _, id := range ids {
- items = append(items, model.MediaFile{ID: id})
+ items := slice.Map(ids, func(id string) model.MediaFile {
+ return model.MediaFile{ID: id}
+ })
+
+ currentIndex := 0
+ for i, id := range ids {
+ if id == currentID {
+ currentIndex = i
+ break
+ }
}
pq := &model.PlayQueue{
UserID: user.ID,
- Current: current,
+ Current: currentIndex,
Position: position,
ChangedBy: client,
Items: items,
@@ -124,3 +135,74 @@ func (api *Router) SavePlayQueue(r *http.Request) (*responses.Subsonic, error) {
}
return newResponse(), nil
}
+
+func (api *Router) GetPlayQueueByIndex(r *http.Request) (*responses.Subsonic, error) {
+ user, _ := request.UserFrom(r.Context())
+
+ repo := api.ds.PlayQueue(r.Context())
+ pq, err := repo.RetrieveWithMediaFiles(user.ID)
+ if err != nil && !errors.Is(err, model.ErrNotFound) {
+ return nil, err
+ }
+ if pq == nil || len(pq.Items) == 0 {
+ return newResponse(), nil
+ }
+
+ response := newResponse()
+
+ var index *int
+ if len(pq.Items) > 0 {
+ index = &pq.Current
+ }
+
+ response.PlayQueueByIndex = &responses.PlayQueueByIndex{
+ Entry: slice.MapWithArg(pq.Items, r.Context(), childFromMediaFile),
+ CurrentIndex: index,
+ Position: pq.Position,
+ Username: user.UserName,
+ Changed: pq.UpdatedAt,
+ ChangedBy: pq.ChangedBy,
+ }
+ return response, nil
+}
+
+func (api *Router) SavePlayQueueByIndex(r *http.Request) (*responses.Subsonic, error) {
+ p := req.Params(r)
+ ids, _ := p.Strings("id")
+
+ position := p.Int64Or("position", 0)
+
+ var err error
+ var currentIndex int
+
+ if len(ids) > 0 {
+ currentIndex, err = p.Int("currentIndex")
+ if err != nil || currentIndex < 0 || currentIndex >= len(ids) {
+ return nil, newError(responses.ErrorMissingParameter, "missing parameter index, err: %s", err)
+ }
+ }
+
+ items := slice.Map(ids, func(id string) model.MediaFile {
+ return model.MediaFile{ID: id}
+ })
+
+ user, _ := request.UserFrom(r.Context())
+ client, _ := request.ClientFrom(r.Context())
+
+ pq := &model.PlayQueue{
+ UserID: user.ID,
+ Current: currentIndex,
+ Position: position,
+ ChangedBy: client,
+ Items: items,
+ CreatedAt: time.Time{},
+ UpdatedAt: time.Time{},
+ }
+
+ repo := api.ds.PlayQueue(r.Context())
+ err = repo.Store(pq)
+ if err != nil {
+ return nil, err
+ }
+ return newResponse(), nil
+}
diff --git a/server/subsonic/browsing.go b/server/subsonic/browsing.go
index 76023c862..c8584543d 100644
--- a/server/subsonic/browsing.go
+++ b/server/subsonic/browsing.go
@@ -7,6 +7,7 @@ import (
"time"
"github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/consts"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/server/public"
@@ -17,7 +18,8 @@ import (
)
func (api *Router) GetMusicFolders(r *http.Request) (*responses.Subsonic, error) {
- libraries, _ := api.ds.Library(r.Context()).GetAll()
+ libraries := getUserAccessibleLibraries(r.Context())
+
folders := make([]responses.MusicFolder, len(libraries))
for i, f := range libraries {
folders[i].Id = int32(f.ID)
@@ -28,28 +30,37 @@ func (api *Router) GetMusicFolders(r *http.Request) (*responses.Subsonic, error)
return response, nil
}
-func (api *Router) getArtist(r *http.Request, libId int, ifModifiedSince time.Time) (model.ArtistIndexes, int64, error) {
+func (api *Router) getArtist(r *http.Request, libIds []int, ifModifiedSince time.Time) (model.ArtistIndexes, int64, error) {
ctx := r.Context()
- lib, err := api.ds.Library(ctx).Get(libId)
+
+ lastScanStr, err := api.ds.Property(ctx).DefaultGet(consts.LastScanStartTimeKey, "")
if err != nil {
- log.Error(ctx, "Error retrieving Library", "id", libId, err)
+ log.Error(ctx, "Error retrieving last scan start time", err)
return nil, 0, err
}
+ lastScan := time.Now()
+ if lastScanStr != "" {
+ lastScan, err = time.Parse(time.RFC3339, lastScanStr)
+ }
var indexes model.ArtistIndexes
- if lib.LastScanAt.After(ifModifiedSince) {
- indexes, err = api.ds.Artist(ctx).GetIndex(false, model.RoleAlbumArtist)
+ if lastScan.After(ifModifiedSince) {
+ indexes, err = api.ds.Artist(ctx).GetIndex(false, libIds, model.RoleAlbumArtist)
if err != nil {
log.Error(ctx, "Error retrieving Indexes", err)
return nil, 0, err
}
+ if len(indexes) == 0 {
+ log.Debug(ctx, "No artists found in library", "libId", libIds)
+ return nil, 0, newError(responses.ErrorDataNotFound, "Library not found or empty")
+ }
}
- return indexes, lib.LastScanAt.UnixMilli(), err
+ return indexes, lastScan.UnixMilli(), err
}
-func (api *Router) getArtistIndex(r *http.Request, libId int, ifModifiedSince time.Time) (*responses.Indexes, error) {
- indexes, modified, err := api.getArtist(r, libId, ifModifiedSince)
+func (api *Router) getArtistIndex(r *http.Request, libIds []int, ifModifiedSince time.Time) (*responses.Indexes, error) {
+ indexes, modified, err := api.getArtist(r, libIds, ifModifiedSince)
if err != nil {
return nil, err
}
@@ -67,8 +78,8 @@ func (api *Router) getArtistIndex(r *http.Request, libId int, ifModifiedSince ti
return res, nil
}
-func (api *Router) getArtistIndexID3(r *http.Request, libId int, ifModifiedSince time.Time) (*responses.Artists, error) {
- indexes, modified, err := api.getArtist(r, libId, ifModifiedSince)
+func (api *Router) getArtistIndexID3(r *http.Request, libIds []int, ifModifiedSince time.Time) (*responses.Artists, error) {
+ indexes, modified, err := api.getArtist(r, libIds, ifModifiedSince)
if err != nil {
return nil, err
}
@@ -88,10 +99,10 @@ func (api *Router) getArtistIndexID3(r *http.Request, libId int, ifModifiedSince
func (api *Router) GetIndexes(r *http.Request) (*responses.Subsonic, error) {
p := req.Params(r)
- musicFolderId := p.IntOr("musicFolderId", 1)
+ musicFolderIds, _ := selectedMusicFolderIds(r, false)
ifModifiedSince := p.TimeOr("ifModifiedSince", time.Time{})
- res, err := api.getArtistIndex(r, musicFolderId, ifModifiedSince)
+ res, err := api.getArtistIndex(r, musicFolderIds, ifModifiedSince)
if err != nil {
return nil, err
}
@@ -102,9 +113,9 @@ func (api *Router) GetIndexes(r *http.Request) (*responses.Subsonic, error) {
}
func (api *Router) GetArtists(r *http.Request) (*responses.Subsonic, error) {
- p := req.Params(r)
- musicFolderId := p.IntOr("musicFolderId", 1)
- res, err := api.getArtistIndexID3(r, musicFolderId, time.Time{})
+ musicFolderIds, _ := selectedMusicFolderIds(r, false)
+
+ res, err := api.getArtistIndexID3(r, musicFolderIds, time.Time{})
if err != nil {
return nil, err
}
@@ -343,7 +354,7 @@ func (api *Router) GetSimilarSongs(r *http.Request) (*responses.Subsonic, error)
}
count := p.IntOr("count", 50)
- songs, err := api.provider.SimilarSongs(ctx, id, count)
+ songs, err := api.provider.ArtistRadio(ctx, id, count)
if err != nil {
return nil, err
}
@@ -397,7 +408,7 @@ func (api *Router) buildArtistDirectory(ctx context.Context, artist *model.Artis
if artist.PlayCount > 0 {
dir.Played = artist.PlayDate
}
- dir.AlbumCount = int32(artist.AlbumCount)
+ dir.AlbumCount = getArtistAlbumCount(artist)
dir.UserRating = int32(artist.Rating)
if artist.Starred {
dir.Starred = artist.StarredAt
diff --git a/server/subsonic/browsing_test.go b/server/subsonic/browsing_test.go
new file mode 100644
index 000000000..b8f510aed
--- /dev/null
+++ b/server/subsonic/browsing_test.go
@@ -0,0 +1,160 @@
+package subsonic
+
+import (
+ "context"
+ "fmt"
+ "net/http/httptest"
+
+ "github.com/navidrome/navidrome/core/auth"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+func contextWithUser(ctx context.Context, userID string, libraryIDs ...int) context.Context {
+ libraries := make([]model.Library, len(libraryIDs))
+ for i, id := range libraryIDs {
+ libraries[i] = model.Library{ID: id, Name: fmt.Sprintf("Test Library %d", id), Path: fmt.Sprintf("/music/library%d", id)}
+ }
+ user := model.User{
+ ID: userID,
+ Libraries: libraries,
+ }
+ return request.WithUser(ctx, user)
+}
+
+var _ = Describe("Browsing", func() {
+ var api *Router
+ var ctx context.Context
+ var ds model.DataStore
+
+ BeforeEach(func() {
+ ds = &tests.MockDataStore{}
+ auth.Init(ds)
+ api = &Router{ds: ds}
+ ctx = context.Background()
+ })
+
+ Describe("GetMusicFolders", func() {
+ It("should return all libraries the user has access", func() {
+ // Create mock user with libraries
+ ctx := contextWithUser(ctx, "user-id", 1, 2, 3)
+
+ // Create request
+ r := httptest.NewRequest("GET", "/rest/getMusicFolders", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.GetMusicFolders(r)
+
+ // Verify results
+ Expect(err).ToNot(HaveOccurred())
+ Expect(response).ToNot(BeNil())
+ Expect(response.MusicFolders).ToNot(BeNil())
+ Expect(response.MusicFolders.Folders).To(HaveLen(3))
+ Expect(response.MusicFolders.Folders[0].Name).To(Equal("Test Library 1"))
+ Expect(response.MusicFolders.Folders[1].Name).To(Equal("Test Library 2"))
+ Expect(response.MusicFolders.Folders[2].Name).To(Equal("Test Library 3"))
+ })
+ })
+
+ Describe("GetIndexes", func() {
+ It("should validate user access to the specified musicFolderId", func() {
+ // Create mock user with access to library 1 only
+ ctx = contextWithUser(ctx, "user-id", 1)
+
+ // Create request with musicFolderId=2 (not accessible)
+ r := httptest.NewRequest("GET", "/rest/getIndexes?musicFolderId=2", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.GetIndexes(r)
+
+ // Should return error due to lack of access
+ Expect(err).To(HaveOccurred())
+ Expect(response).To(BeNil())
+ })
+
+ It("should default to first accessible library when no musicFolderId specified", func() {
+ // Create mock user with access to libraries 2 and 3
+ ctx = contextWithUser(ctx, "user-id", 2, 3)
+
+ // Setup minimal mock library data for working tests
+ mockLibRepo := ds.Library(ctx).(*tests.MockLibraryRepo)
+ mockLibRepo.SetData(model.Libraries{
+ {ID: 2, Name: "Test Library 2", Path: "/music/library2"},
+ {ID: 3, Name: "Test Library 3", Path: "/music/library3"},
+ })
+
+ // Setup mock artist data
+ mockArtistRepo := ds.Artist(ctx).(*tests.MockArtistRepo)
+ mockArtistRepo.SetData(model.Artists{
+ {ID: "1", Name: "Test Artist 1"},
+ {ID: "2", Name: "Test Artist 2"},
+ })
+
+ // Create request without musicFolderId
+ r := httptest.NewRequest("GET", "/rest/getIndexes", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.GetIndexes(r)
+
+ // Should succeed and use first accessible library (2)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(response).ToNot(BeNil())
+ Expect(response.Indexes).ToNot(BeNil())
+ })
+ })
+
+ Describe("GetArtists", func() {
+ It("should validate user access to the specified musicFolderId", func() {
+ // Create mock user with access to library 1 only
+ ctx = contextWithUser(ctx, "user-id", 1)
+
+ // Create request with musicFolderId=3 (not accessible)
+ r := httptest.NewRequest("GET", "/rest/getArtists?musicFolderId=3", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.GetArtists(r)
+
+ // Should return error due to lack of access
+ Expect(err).To(HaveOccurred())
+ Expect(response).To(BeNil())
+ })
+
+ It("should default to first accessible library when no musicFolderId specified", func() {
+ // Create mock user with access to libraries 1 and 2
+ ctx = contextWithUser(ctx, "user-id", 1, 2)
+
+ // Setup minimal mock library data for working tests
+ mockLibRepo := ds.Library(ctx).(*tests.MockLibraryRepo)
+ mockLibRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Test Library 1", Path: "/music/library1"},
+ {ID: 2, Name: "Test Library 2", Path: "/music/library2"},
+ })
+
+ // Setup mock artist data
+ mockArtistRepo := ds.Artist(ctx).(*tests.MockArtistRepo)
+ mockArtistRepo.SetData(model.Artists{
+ {ID: "1", Name: "Test Artist 1"},
+ {ID: "2", Name: "Test Artist 2"},
+ })
+
+ // Create request without musicFolderId
+ r := httptest.NewRequest("GET", "/rest/getArtists", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.GetArtists(r)
+
+ // Should succeed and use first accessible library (1)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(response).ToNot(BeNil())
+ Expect(response.Artist).ToNot(BeNil())
+ })
+ })
+})
diff --git a/server/subsonic/filter/filters.go b/server/subsonic/filter/filters.go
index 4ab4f9642..8ba4f0ff9 100644
--- a/server/subsonic/filter/filters.go
+++ b/server/subsonic/filter/filters.go
@@ -108,14 +108,13 @@ func SongsByRandom(genre string, fromYear, toYear int) Options {
return addDefaultFilters(options)
}
-func SongWithLyrics(artist, title string) Options {
+func SongsByArtistTitleWithLyricsFirst(artist, title string) Options {
return addDefaultFilters(Options{
- Sort: "updated_at",
+ Sort: "lyrics, updated_at",
Order: "desc",
Max: 1,
Filters: And{
Eq{"title": title},
- NotEq{"lyrics": "[]"},
Or{
persistence.Exists("json_tree(participants, '$.albumartist')", Eq{"value": artist}),
persistence.Exists("json_tree(participants, '$.artist')", Eq{"value": artist}),
@@ -124,6 +123,38 @@ func SongWithLyrics(artist, title string) Options {
})
}
+func ApplyLibraryFilter(opts Options, musicFolderIds []int) Options {
+ if len(musicFolderIds) == 0 {
+ return opts
+ }
+
+ libraryFilter := Eq{"library_id": musicFolderIds}
+ if opts.Filters == nil {
+ opts.Filters = libraryFilter
+ } else {
+ opts.Filters = And{opts.Filters, libraryFilter}
+ }
+
+ return opts
+}
+
+// ApplyArtistLibraryFilter applies a filter to the given Options to ensure that only artists
+// that are associated with the specified music folders are included in the results.
+func ApplyArtistLibraryFilter(opts Options, musicFolderIds []int) Options {
+ if len(musicFolderIds) == 0 {
+ return opts
+ }
+
+ artistLibraryFilter := Eq{"library_artist.library_id": musicFolderIds}
+ if opts.Filters == nil {
+ opts.Filters = artistLibraryFilter
+ } else {
+ opts.Filters = And{opts.Filters, artistLibraryFilter}
+ }
+
+ return opts
+}
+
func ByGenre(genre string) Options {
return addDefaultFilters(Options{
Sort: "name",
@@ -132,7 +163,7 @@ func ByGenre(genre string) Options {
}
func filterByGenre(genre string) Sqlizer {
- return persistence.Exists("json_tree(tags)", And{
+ return persistence.Exists(`json_tree(tags, "$.genre")`, And{
Like{"value": genre},
NotEq{"atom": nil},
})
diff --git a/server/subsonic/helpers.go b/server/subsonic/helpers.go
index 39f324654..f9733bb3f 100644
--- a/server/subsonic/helpers.go
+++ b/server/subsonic/helpers.go
@@ -7,6 +7,7 @@ import (
"fmt"
"mime"
"net/http"
+ "slices"
"sort"
"strings"
@@ -17,6 +18,7 @@ import (
"github.com/navidrome/navidrome/server/public"
"github.com/navidrome/navidrome/server/subsonic/responses"
"github.com/navidrome/navidrome/utils/number"
+ "github.com/navidrome/navidrome/utils/req"
"github.com/navidrome/navidrome/utils/slice"
)
@@ -77,18 +79,16 @@ func sortName(sortName, orderName string) string {
return orderName
}
-func getArtistAlbumCount(a model.Artist) int32 {
- albumStats := a.Stats[model.RoleAlbumArtist]
-
+func getArtistAlbumCount(a *model.Artist) int32 {
// If ArtistParticipations are set, then `getArtist` will return albums
- // where the artist is an album artist OR artist. While it may be an underestimate,
- // guess the count by taking a max of the album artist and artist count. This is
- // guaranteed to be <= the actual count.
+ // where the artist is an album artist OR artist. Use the custom stat
+ // main credit for this calculation.
// Otherwise, return just the roles as album artist (precise)
if conf.Server.Subsonic.ArtistParticipations {
- artistStats := a.Stats[model.RoleArtist]
- return int32(max(artistStats.AlbumCount, albumStats.AlbumCount))
+ mainCreditStats := a.Stats[model.RoleMainCredit]
+ return int32(mainCreditStats.AlbumCount)
} else {
+ albumStats := a.Stats[model.RoleAlbumArtist]
return int32(albumStats.AlbumCount)
}
}
@@ -111,7 +111,7 @@ func toArtistID3(r *http.Request, a model.Artist) responses.ArtistID3 {
artist := responses.ArtistID3{
Id: a.ID,
Name: a.Name,
- AlbumCount: getArtistAlbumCount(a),
+ AlbumCount: getArtistAlbumCount(&a),
CoverArt: a.CoverArtID().String(),
ArtistImageUrl: public.ImageURL(r, a.CoverArtID(), 600),
UserRating: int32(a.Rating),
@@ -476,3 +476,40 @@ func buildLyricsList(mf *model.MediaFile, lyricsList model.LyricList) *responses
}
return res
}
+
+// getUserAccessibleLibraries returns the list of libraries the current user has access to.
+func getUserAccessibleLibraries(ctx context.Context) []model.Library {
+ user := getUser(ctx)
+ return user.Libraries
+}
+
+// selectedMusicFolderIds retrieves the music folder IDs from the request parameters.
+// If no IDs are provided, it returns all libraries the user has access to (based on the user found in the context).
+// If the parameter is required and not present, it returns an error.
+// If any of the provided library IDs are invalid (don't exist or user doesn't have access), returns ErrorDataNotFound.
+func selectedMusicFolderIds(r *http.Request, required bool) ([]int, error) {
+ p := req.Params(r)
+ musicFolderIds, err := p.Ints("musicFolderId")
+
+ // If the parameter is not present, it returns an error if it is required.
+ if errors.Is(err, req.ErrMissingParam) && required {
+ return nil, err
+ }
+
+ // Get user's accessible libraries for validation
+ libraries := getUserAccessibleLibraries(r.Context())
+ accessibleLibraryIds := slice.Map(libraries, func(lib model.Library) int { return lib.ID })
+
+ if len(musicFolderIds) > 0 {
+ // Validate all provided library IDs - if any are invalid, return an error
+ for _, id := range musicFolderIds {
+ if !slices.Contains(accessibleLibraryIds, id) {
+ return nil, newError(responses.ErrorDataNotFound, "Library %d not found or not accessible", id)
+ }
+ }
+ return musicFolderIds, nil
+ }
+
+ // If no musicFolderId is provided, return all libraries the user has access to.
+ return accessibleLibraryIds, nil
+}
diff --git a/server/subsonic/helpers_test.go b/server/subsonic/helpers_test.go
index d703607ba..a6508d4bb 100644
--- a/server/subsonic/helpers_test.go
+++ b/server/subsonic/helpers_test.go
@@ -1,10 +1,15 @@
package subsonic
import (
+ "context"
+ "net/http/httptest"
+
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/conf/configtest"
"github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
"github.com/navidrome/navidrome/server/subsonic/responses"
+ "github.com/navidrome/navidrome/utils/req"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
)
@@ -145,7 +150,7 @@ var _ = Describe("helpers", func() {
model.RoleAlbumArtist: {
AlbumCount: 3,
},
- model.RoleArtist: {
+ model.RoleMainCredit: {
AlbumCount: 4,
},
},
@@ -153,14 +158,118 @@ var _ = Describe("helpers", func() {
It("Handles album count without artist participations", func() {
conf.Server.Subsonic.ArtistParticipations = false
- result := getArtistAlbumCount(artist)
+ result := getArtistAlbumCount(&artist)
Expect(result).To(Equal(int32(3)))
})
It("Handles album count without with participations", func() {
conf.Server.Subsonic.ArtistParticipations = true
- result := getArtistAlbumCount(artist)
+ result := getArtistAlbumCount(&artist)
Expect(result).To(Equal(int32(4)))
})
})
+
+ Describe("selectedMusicFolderIds", func() {
+ var user model.User
+ var ctx context.Context
+
+ BeforeEach(func() {
+ user = model.User{
+ ID: "test-user",
+ Libraries: []model.Library{
+ {ID: 1, Name: "Library 1"},
+ {ID: 2, Name: "Library 2"},
+ {ID: 3, Name: "Library 3"},
+ },
+ }
+ ctx = request.WithUser(context.Background(), user)
+ })
+
+ Context("when musicFolderId parameter is provided", func() {
+ It("should return the specified musicFolderId values", func() {
+ r := httptest.NewRequest("GET", "/test?musicFolderId=1&musicFolderId=3", nil)
+ r = r.WithContext(ctx)
+
+ ids, err := selectedMusicFolderIds(r, false)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ids).To(Equal([]int{1, 3}))
+ })
+
+ It("should ignore invalid musicFolderId parameter values", func() {
+ r := httptest.NewRequest("GET", "/test?musicFolderId=invalid&musicFolderId=2", nil)
+ r = r.WithContext(ctx)
+
+ ids, err := selectedMusicFolderIds(r, false)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ids).To(Equal([]int{2})) // Only valid ID is returned
+ })
+
+ It("should return error when any library ID is not accessible", func() {
+ r := httptest.NewRequest("GET", "/test?musicFolderId=1&musicFolderId=5&musicFolderId=2&musicFolderId=99", nil)
+ r = r.WithContext(ctx)
+
+ ids, err := selectedMusicFolderIds(r, false)
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("Library 5 not found or not accessible"))
+ Expect(ids).To(BeNil())
+ })
+ })
+
+ Context("when musicFolderId parameter is not provided", func() {
+ Context("and required is false", func() {
+ It("should return all user's library IDs", func() {
+ r := httptest.NewRequest("GET", "/test", nil)
+ r = r.WithContext(ctx)
+
+ ids, err := selectedMusicFolderIds(r, false)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ids).To(Equal([]int{1, 2, 3}))
+ })
+
+ It("should return empty slice when user has no libraries", func() {
+ userWithoutLibs := model.User{ID: "no-libs-user", Libraries: []model.Library{}}
+ ctxWithoutLibs := request.WithUser(context.Background(), userWithoutLibs)
+ r := httptest.NewRequest("GET", "/test", nil)
+ r = r.WithContext(ctxWithoutLibs)
+
+ ids, err := selectedMusicFolderIds(r, false)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ids).To(Equal([]int{}))
+ })
+ })
+
+ Context("and required is true", func() {
+ It("should return ErrMissingParam error", func() {
+ r := httptest.NewRequest("GET", "/test", nil)
+ r = r.WithContext(ctx)
+
+ ids, err := selectedMusicFolderIds(r, true)
+ Expect(err).To(MatchError(req.ErrMissingParam))
+ Expect(ids).To(BeNil())
+ })
+ })
+ })
+
+ Context("when musicFolderId parameter is empty", func() {
+ It("should return all user's library IDs even when empty parameter is provided", func() {
+ r := httptest.NewRequest("GET", "/test?musicFolderId=", nil)
+ r = r.WithContext(ctx)
+
+ ids, err := selectedMusicFolderIds(r, false)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ids).To(Equal([]int{1, 2, 3}))
+ })
+ })
+
+ Context("when all musicFolderId parameters are invalid", func() {
+ It("should return all user libraries when all musicFolderId parameters are invalid", func() {
+ r := httptest.NewRequest("GET", "/test?musicFolderId=invalid&musicFolderId=notanumber", nil)
+ r = r.WithContext(ctx)
+
+ ids, err := selectedMusicFolderIds(r, false)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(ids).To(Equal([]int{1, 2, 3})) // Falls back to all user libraries
+ })
+ })
+ })
})
diff --git a/server/subsonic/library_scanning.go b/server/subsonic/library_scanning.go
index b6ccb9ae6..c9dd64968 100644
--- a/server/subsonic/library_scanning.go
+++ b/server/subsonic/library_scanning.go
@@ -1,10 +1,13 @@
package subsonic
import (
+ "fmt"
"net/http"
+ "slices"
"time"
"github.com/navidrome/navidrome/log"
+ "github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/model/request"
"github.com/navidrome/navidrome/server/subsonic/responses"
"github.com/navidrome/navidrome/utils/req"
@@ -44,15 +47,56 @@ func (api *Router) StartScan(r *http.Request) (*responses.Subsonic, error) {
p := req.Params(r)
fullScan := p.BoolOr("fullScan", false)
+ // Parse optional target parameters for selective scanning
+ var targets []model.ScanTarget
+ if targetParams, err := p.Strings("target"); err == nil && len(targetParams) > 0 {
+ targets, err = model.ParseTargets(targetParams)
+ if err != nil {
+ return nil, newError(responses.ErrorGeneric, fmt.Sprintf("Invalid target parameter: %v", err))
+ }
+
+ // Validate all libraries in targets exist and user has access to them
+ userLibraries, err := api.ds.User(ctx).GetUserLibraries(loggedUser.ID)
+ if err != nil {
+ return nil, newError(responses.ErrorGeneric, "Internal error")
+ }
+
+ // Check each target library
+ for _, target := range targets {
+ if !slices.ContainsFunc(userLibraries, func(lib model.Library) bool { return lib.ID == target.LibraryID }) {
+ return nil, newError(responses.ErrorDataNotFound, fmt.Sprintf("Library with ID %d not found", target.LibraryID))
+ }
+ }
+
+ // Special case: if single library with empty path and it's the only library in DB, call ScanAll
+ if len(targets) == 1 && targets[0].FolderPath == "" {
+ allLibs, err := api.ds.Library(ctx).GetAll()
+ if err != nil {
+ return nil, newError(responses.ErrorGeneric, "Internal error")
+ }
+ if len(allLibs) == 1 {
+ targets = nil // This will trigger ScanAll below
+ }
+ }
+ }
+
go func() {
start := time.Now()
- log.Info(ctx, "Triggering manual scan", "fullScan", fullScan, "user", loggedUser.UserName)
- _, err := api.scanner.ScanAll(ctx, fullScan)
+ var err error
+
+ if len(targets) > 0 {
+ log.Info(ctx, "Triggering on-demand scan", "fullScan", fullScan, "targets", len(targets), "user", loggedUser.UserName)
+ _, err = api.scanner.ScanFolders(ctx, fullScan, targets)
+ } else {
+ log.Info(ctx, "Triggering on-demand scan", "fullScan", fullScan, "user", loggedUser.UserName)
+ _, err = api.scanner.ScanAll(ctx, fullScan)
+ }
+
if err != nil {
log.Error(ctx, "Error scanning", err)
return
}
- log.Info(ctx, "Manual scan complete", "user", loggedUser.UserName, "elapsed", time.Since(start))
+ log.Info(ctx, "On-demand scan complete", "user", loggedUser.UserName, "elapsed", time.Since(start))
}()
return api.GetScanStatus(r)
diff --git a/server/subsonic/library_scanning_test.go b/server/subsonic/library_scanning_test.go
new file mode 100644
index 000000000..d8eba296b
--- /dev/null
+++ b/server/subsonic/library_scanning_test.go
@@ -0,0 +1,396 @@
+package subsonic
+
+import (
+ "context"
+ "errors"
+ "net/http/httptest"
+
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/server/subsonic/responses"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("LibraryScanning", func() {
+ var api *Router
+ var ms *tests.MockScanner
+
+ BeforeEach(func() {
+ ms = tests.NewMockScanner()
+ api = &Router{scanner: ms}
+ })
+
+ Describe("StartScan", func() {
+ It("requires admin authentication", func() {
+ // Create non-admin user
+ ctx := request.WithUser(context.Background(), model.User{
+ ID: "user-id",
+ IsAdmin: false,
+ })
+
+ // Create request
+ r := httptest.NewRequest("GET", "/rest/startScan", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.StartScan(r)
+
+ // Should return authorization error
+ Expect(err).To(HaveOccurred())
+ Expect(response).To(BeNil())
+ var subErr subError
+ ok := errors.As(err, &subErr)
+ Expect(ok).To(BeTrue())
+ Expect(subErr.code).To(Equal(responses.ErrorAuthorizationFail))
+ })
+
+ It("triggers a full scan with no parameters", func() {
+ // Create admin user
+ ctx := request.WithUser(context.Background(), model.User{
+ ID: "admin-id",
+ IsAdmin: true,
+ })
+
+ // Create request with no parameters
+ r := httptest.NewRequest("GET", "/rest/startScan", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.StartScan(r)
+
+ // Should succeed
+ Expect(err).ToNot(HaveOccurred())
+ Expect(response).ToNot(BeNil())
+
+ // Verify ScanAll was called (eventually, since it's in a goroutine)
+ Eventually(func() int {
+ return ms.GetScanAllCallCount()
+ }).Should(BeNumerically(">", 0))
+ calls := ms.GetScanAllCalls()
+ Expect(calls).To(HaveLen(1))
+ Expect(calls[0].FullScan).To(BeFalse())
+ })
+
+ It("triggers a full scan with fullScan=true", func() {
+ // Create admin user
+ ctx := request.WithUser(context.Background(), model.User{
+ ID: "admin-id",
+ IsAdmin: true,
+ })
+
+ // Create request with fullScan parameter
+ r := httptest.NewRequest("GET", "/rest/startScan?fullScan=true", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.StartScan(r)
+
+ // Should succeed
+ Expect(err).ToNot(HaveOccurred())
+ Expect(response).ToNot(BeNil())
+
+ // Verify ScanAll was called with fullScan=true
+ Eventually(func() int {
+ return ms.GetScanAllCallCount()
+ }).Should(BeNumerically(">", 0))
+ calls := ms.GetScanAllCalls()
+ Expect(calls).To(HaveLen(1))
+ Expect(calls[0].FullScan).To(BeTrue())
+ })
+
+ It("triggers a selective scan with single target parameter", func() {
+ // Setup mocks
+ mockUserRepo := tests.CreateMockUserRepo()
+ _ = mockUserRepo.SetUserLibraries("admin-id", []int{1, 2})
+ mockDS := &tests.MockDataStore{MockedUser: mockUserRepo}
+ api.ds = mockDS
+
+ // Create admin user
+ ctx := request.WithUser(context.Background(), model.User{
+ ID: "admin-id",
+ IsAdmin: true,
+ })
+
+ // Create request with single target parameter
+ r := httptest.NewRequest("GET", "/rest/startScan?target=1:Music/Rock", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.StartScan(r)
+
+ // Should succeed
+ Expect(err).ToNot(HaveOccurred())
+ Expect(response).ToNot(BeNil())
+
+ // Verify ScanFolders was called with correct targets
+ Eventually(func() int {
+ return ms.GetScanFoldersCallCount()
+ }).Should(BeNumerically(">", 0))
+ calls := ms.GetScanFoldersCalls()
+ Expect(calls).To(HaveLen(1))
+ targets := calls[0].Targets
+ Expect(targets).To(HaveLen(1))
+ Expect(targets[0].LibraryID).To(Equal(1))
+ Expect(targets[0].FolderPath).To(Equal("Music/Rock"))
+ })
+
+ It("triggers a selective scan with multiple target parameters", func() {
+ // Setup mocks
+ mockUserRepo := tests.CreateMockUserRepo()
+ _ = mockUserRepo.SetUserLibraries("admin-id", []int{1, 2})
+ mockDS := &tests.MockDataStore{MockedUser: mockUserRepo}
+ api.ds = mockDS
+
+ // Create admin user
+ ctx := request.WithUser(context.Background(), model.User{
+ ID: "admin-id",
+ IsAdmin: true,
+ })
+
+ // Create request with multiple target parameters
+ r := httptest.NewRequest("GET", "/rest/startScan?target=1:Music/Reggae&target=2:Classical/Bach", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.StartScan(r)
+
+ // Should succeed
+ Expect(err).ToNot(HaveOccurred())
+ Expect(response).ToNot(BeNil())
+
+ // Verify ScanFolders was called with correct targets
+ Eventually(func() int {
+ return ms.GetScanFoldersCallCount()
+ }).Should(BeNumerically(">", 0))
+ calls := ms.GetScanFoldersCalls()
+ Expect(calls).To(HaveLen(1))
+ targets := calls[0].Targets
+ Expect(targets).To(HaveLen(2))
+ Expect(targets[0].LibraryID).To(Equal(1))
+ Expect(targets[0].FolderPath).To(Equal("Music/Reggae"))
+ Expect(targets[1].LibraryID).To(Equal(2))
+ Expect(targets[1].FolderPath).To(Equal("Classical/Bach"))
+ })
+
+ It("triggers a selective full scan with target and fullScan parameters", func() {
+ // Setup mocks
+ mockUserRepo := tests.CreateMockUserRepo()
+ _ = mockUserRepo.SetUserLibraries("admin-id", []int{1})
+ mockDS := &tests.MockDataStore{MockedUser: mockUserRepo}
+ api.ds = mockDS
+
+ // Create admin user
+ ctx := request.WithUser(context.Background(), model.User{
+ ID: "admin-id",
+ IsAdmin: true,
+ })
+
+ // Create request with target and fullScan parameters
+ r := httptest.NewRequest("GET", "/rest/startScan?target=1:Music/Jazz&fullScan=true", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.StartScan(r)
+
+ // Should succeed
+ Expect(err).ToNot(HaveOccurred())
+ Expect(response).ToNot(BeNil())
+
+ // Verify ScanFolders was called with fullScan=true
+ Eventually(func() int {
+ return ms.GetScanFoldersCallCount()
+ }).Should(BeNumerically(">", 0))
+ calls := ms.GetScanFoldersCalls()
+ Expect(calls).To(HaveLen(1))
+ Expect(calls[0].FullScan).To(BeTrue())
+ targets := calls[0].Targets
+ Expect(targets).To(HaveLen(1))
+ })
+
+ It("returns error for invalid target format", func() {
+ // Create admin user
+ ctx := request.WithUser(context.Background(), model.User{
+ ID: "admin-id",
+ IsAdmin: true,
+ })
+
+ // Create request with invalid target format (missing colon)
+ r := httptest.NewRequest("GET", "/rest/startScan?target=1MusicRock", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.StartScan(r)
+
+ // Should return error
+ Expect(err).To(HaveOccurred())
+ Expect(response).To(BeNil())
+ var subErr subError
+ ok := errors.As(err, &subErr)
+ Expect(ok).To(BeTrue())
+ Expect(subErr.code).To(Equal(responses.ErrorGeneric))
+ })
+
+ It("returns error for invalid library ID in target", func() {
+ // Create admin user
+ ctx := request.WithUser(context.Background(), model.User{
+ ID: "admin-id",
+ IsAdmin: true,
+ })
+
+ // Create request with invalid library ID
+ r := httptest.NewRequest("GET", "/rest/startScan?target=0:Music/Rock", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.StartScan(r)
+
+ // Should return error
+ Expect(err).To(HaveOccurred())
+ Expect(response).To(BeNil())
+ var subErr subError
+ ok := errors.As(err, &subErr)
+ Expect(ok).To(BeTrue())
+ Expect(subErr.code).To(Equal(responses.ErrorGeneric))
+ })
+
+ It("returns error when library does not exist", func() {
+ // Setup mocks - user has access to library 1 and 2 only
+ mockUserRepo := tests.CreateMockUserRepo()
+ _ = mockUserRepo.SetUserLibraries("admin-id", []int{1, 2})
+ mockDS := &tests.MockDataStore{MockedUser: mockUserRepo}
+ api.ds = mockDS
+
+ // Create admin user
+ ctx := request.WithUser(context.Background(), model.User{
+ ID: "admin-id",
+ IsAdmin: true,
+ })
+
+ // Create request with library ID that doesn't exist
+ r := httptest.NewRequest("GET", "/rest/startScan?target=999:Music/Rock", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.StartScan(r)
+
+ // Should return ErrorDataNotFound
+ Expect(err).To(HaveOccurred())
+ Expect(response).To(BeNil())
+ var subErr subError
+ ok := errors.As(err, &subErr)
+ Expect(ok).To(BeTrue())
+ Expect(subErr.code).To(Equal(responses.ErrorDataNotFound))
+ })
+
+ It("calls ScanAll when single library with empty path and only one library exists", func() {
+ // Setup mocks - single library in DB
+ mockUserRepo := tests.CreateMockUserRepo()
+ _ = mockUserRepo.SetUserLibraries("admin-id", []int{1})
+ mockLibraryRepo := &tests.MockLibraryRepo{}
+ mockLibraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Music Library", Path: "/music"},
+ })
+ mockDS := &tests.MockDataStore{
+ MockedUser: mockUserRepo,
+ MockedLibrary: mockLibraryRepo,
+ }
+ api.ds = mockDS
+
+ // Create admin user
+ ctx := request.WithUser(context.Background(), model.User{
+ ID: "admin-id",
+ IsAdmin: true,
+ })
+
+ // Create request with single library and empty path
+ r := httptest.NewRequest("GET", "/rest/startScan?target=1:", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.StartScan(r)
+
+ // Should succeed
+ Expect(err).ToNot(HaveOccurred())
+ Expect(response).ToNot(BeNil())
+
+ // Verify ScanAll was called instead of ScanFolders
+ Eventually(func() int {
+ return ms.GetScanAllCallCount()
+ }).Should(BeNumerically(">", 0))
+ Expect(ms.GetScanFoldersCallCount()).To(Equal(0))
+ })
+
+ It("calls ScanFolders when single library with empty path but multiple libraries exist", func() {
+ // Setup mocks - multiple libraries in DB
+ mockUserRepo := tests.CreateMockUserRepo()
+ _ = mockUserRepo.SetUserLibraries("admin-id", []int{1, 2})
+ mockLibraryRepo := &tests.MockLibraryRepo{}
+ mockLibraryRepo.SetData(model.Libraries{
+ {ID: 1, Name: "Music Library", Path: "/music"},
+ {ID: 2, Name: "Audiobooks", Path: "/audiobooks"},
+ })
+ mockDS := &tests.MockDataStore{
+ MockedUser: mockUserRepo,
+ MockedLibrary: mockLibraryRepo,
+ }
+ api.ds = mockDS
+
+ // Create admin user
+ ctx := request.WithUser(context.Background(), model.User{
+ ID: "admin-id",
+ IsAdmin: true,
+ })
+
+ // Create request with single library and empty path
+ r := httptest.NewRequest("GET", "/rest/startScan?target=1:", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.StartScan(r)
+
+ // Should succeed
+ Expect(err).ToNot(HaveOccurred())
+ Expect(response).ToNot(BeNil())
+
+ // Verify ScanFolders was called (not ScanAll)
+ Eventually(func() int {
+ return ms.GetScanFoldersCallCount()
+ }).Should(BeNumerically(">", 0))
+ calls := ms.GetScanFoldersCalls()
+ Expect(calls).To(HaveLen(1))
+ targets := calls[0].Targets
+ Expect(targets).To(HaveLen(1))
+ Expect(targets[0].LibraryID).To(Equal(1))
+ Expect(targets[0].FolderPath).To(Equal(""))
+ })
+ })
+
+ Describe("GetScanStatus", func() {
+ It("returns scan status", func() {
+ // Setup mock scanner status
+ ms.SetStatusResponse(&model.ScannerStatus{
+ Scanning: false,
+ Count: 100,
+ FolderCount: 10,
+ })
+
+ // Create request
+ ctx := context.Background()
+ r := httptest.NewRequest("GET", "/rest/getScanStatus", nil)
+ r = r.WithContext(ctx)
+
+ // Call endpoint
+ response, err := api.GetScanStatus(r)
+
+ // Should succeed
+ Expect(err).ToNot(HaveOccurred())
+ Expect(response).ToNot(BeNil())
+ Expect(response.ScanStatus).ToNot(BeNil())
+ Expect(response.ScanStatus.Scanning).To(BeFalse())
+ Expect(response.ScanStatus.Count).To(Equal(int64(100)))
+ Expect(response.ScanStatus.FolderCount).To(Equal(int64(10)))
+ })
+ })
+})
diff --git a/server/subsonic/media_annotation.go b/server/subsonic/media_annotation.go
index 74000856f..39bc83fa9 100644
--- a/server/subsonic/media_annotation.go
+++ b/server/subsonic/media_annotation.go
@@ -165,6 +165,7 @@ func (api *Router) Scrobble(r *http.Request) (*responses.Subsonic, error) {
return nil, newError(responses.ErrorGeneric, "Wrong number of timestamps: %d, should be %d", len(times), len(ids))
}
submission := p.BoolOr("submission", true)
+ position := p.IntOr("position", 0)
ctx := r.Context()
if submission {
@@ -173,7 +174,7 @@ func (api *Router) Scrobble(r *http.Request) (*responses.Subsonic, error) {
log.Error(ctx, "Error registering scrobbles", "ids", ids, "times", times, err)
}
} else {
- err := api.scrobblerNowPlaying(ctx, ids[0])
+ err := api.scrobblerNowPlaying(ctx, ids[0], position)
if err != nil {
log.Error(ctx, "Error setting NowPlaying", "id", ids[0], err)
}
@@ -198,7 +199,7 @@ func (api *Router) scrobblerSubmit(ctx context.Context, ids []string, times []ti
return api.scrobbler.Submit(ctx, submissions)
}
-func (api *Router) scrobblerNowPlaying(ctx context.Context, trackId string) error {
+func (api *Router) scrobblerNowPlaying(ctx context.Context, trackId string, position int) error {
mf, err := api.ds.MediaFile(ctx).Get(trackId)
if err != nil {
return err
@@ -215,7 +216,7 @@ func (api *Router) scrobblerNowPlaying(ctx context.Context, trackId string) erro
clientId = player.ID
}
- log.Info(ctx, "Now Playing", "title", mf.Title, "artist", mf.Artist, "user", username, "player", player.Name)
- err = api.scrobbler.NowPlaying(ctx, clientId, client, trackId)
+ log.Info(ctx, "Now Playing", "title", mf.Title, "artist", mf.Artist, "user", username, "player", player.Name, "position", position)
+ err = api.scrobbler.NowPlaying(ctx, clientId, client, trackId, position)
return err
}
diff --git a/server/subsonic/media_annotation_test.go b/server/subsonic/media_annotation_test.go
index 1611250d9..6f09f5349 100644
--- a/server/subsonic/media_annotation_test.go
+++ b/server/subsonic/media_annotation_test.go
@@ -27,7 +27,7 @@ var _ = Describe("MediaAnnotationController", func() {
ds = &tests.MockDataStore{}
playTracker = &fakePlayTracker{}
eventBroker = &fakeEventBroker{}
- router = New(ds, nil, nil, nil, nil, nil, nil, eventBroker, nil, playTracker, nil, nil)
+ router = New(ds, nil, nil, nil, nil, nil, nil, eventBroker, nil, playTracker, nil, nil, nil)
})
Describe("Scrobble", func() {
@@ -104,7 +104,7 @@ type fakePlayTracker struct {
Error error
}
-func (f *fakePlayTracker) NowPlaying(_ context.Context, playerId string, _ string, trackId string) error {
+func (f *fakePlayTracker) NowPlaying(_ context.Context, playerId string, _ string, trackId string, position int) error {
if f.Error != nil {
return f.Error
}
@@ -138,4 +138,8 @@ func (f *fakeEventBroker) SendMessage(_ context.Context, event events.Event) {
f.Events = append(f.Events, event)
}
+func (f *fakeEventBroker) SendBroadcastMessage(_ context.Context, event events.Event) {
+ f.Events = append(f.Events, event)
+}
+
var _ events.Broker = (*fakeEventBroker)(nil)
diff --git a/server/subsonic/media_retrieval.go b/server/subsonic/media_retrieval.go
index 5cca74c30..a72e4865f 100644
--- a/server/subsonic/media_retrieval.go
+++ b/server/subsonic/media_retrieval.go
@@ -98,7 +98,7 @@ func (api *Router) GetLyrics(r *http.Request) (*responses.Subsonic, error) {
response := newResponse()
lyricsResponse := responses.Lyrics{}
response.Lyrics = &lyricsResponse
- mediaFiles, err := api.ds.MediaFile(r.Context()).GetAll(filter.SongWithLyrics(artist, title))
+ mediaFiles, err := api.ds.MediaFile(r.Context()).GetAll(filter.SongsByArtistTitleWithLyricsFirst(artist, title))
if err != nil {
return nil, err
diff --git a/server/subsonic/media_retrieval_test.go b/server/subsonic/media_retrieval_test.go
index a0e9754ce..351b4e591 100644
--- a/server/subsonic/media_retrieval_test.go
+++ b/server/subsonic/media_retrieval_test.go
@@ -2,17 +2,18 @@ package subsonic
import (
"bytes"
+ "cmp"
"context"
"encoding/json"
"errors"
"io"
"net/http/httptest"
+ "slices"
"time"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/conf/configtest"
"github.com/navidrome/navidrome/core/artwork"
- "github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/server/subsonic/responses"
"github.com/navidrome/navidrome/tests"
@@ -23,7 +24,7 @@ import (
var _ = Describe("MediaRetrievalController", func() {
var router *Router
var ds model.DataStore
- mockRepo := &mockedMediaFile{}
+ mockRepo := &mockedMediaFile{MockMediaFileRepo: tests.MockMediaFileRepo{}}
var artwork *fakeArtwork
var w *httptest.ResponseRecorder
@@ -31,8 +32,8 @@ var _ = Describe("MediaRetrievalController", func() {
ds = &tests.MockDataStore{
MockedMediaFile: mockRepo,
}
- artwork = &fakeArtwork{}
- router = New(ds, artwork, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil)
+ artwork = &fakeArtwork{data: "image data"}
+ router = New(ds, artwork, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil)
w = httptest.NewRecorder()
DeferCleanup(configtest.SetupConfig())
conf.Server.LyricsPriority = "embedded,.lrc"
@@ -40,27 +41,27 @@ var _ = Describe("MediaRetrievalController", func() {
Describe("GetCoverArt", func() {
It("should return data for that id", func() {
- artwork.data = "image data"
- r := newGetRequest("id=34", "size=128")
+ r := newGetRequest("id=34", "size=128", "square=true")
_, err := router.GetCoverArt(w, r)
- Expect(err).To(BeNil())
- Expect(artwork.recvId).To(Equal("34"))
+ Expect(err).ToNot(HaveOccurred())
Expect(artwork.recvSize).To(Equal(128))
+ Expect(artwork.recvSquare).To(BeTrue())
Expect(w.Body.String()).To(Equal(artwork.data))
})
It("should return placeholder if id parameter is missing (mimicking Subsonic)", func() {
- r := newGetRequest()
+ r := newGetRequest() // No id parameter
_, err := router.GetCoverArt(w, r)
Expect(err).To(BeNil())
+ Expect(artwork.recvId).To(BeEmpty())
Expect(w.Body.String()).To(Equal(artwork.data))
})
It("should fail when the file is not found", func() {
artwork.err = model.ErrNotFound
- r := newGetRequest("id=34", "size=128")
+ r := newGetRequest("id=34", "size=128", "square=true")
_, err := router.GetCoverArt(w, r)
Expect(err).To(MatchError("Artwork not found"))
@@ -73,6 +74,45 @@ var _ = Describe("MediaRetrievalController", func() {
Expect(err).To(MatchError("weird error"))
})
+
+ When("client disconnects (context is cancelled)", func() {
+ It("should not call the service if cancelled before the call", func() {
+ // Create a request
+ ctx, cancel := context.WithCancel(context.Background())
+ r := newGetRequest("id=34", "size=128", "square=true")
+ r = r.WithContext(ctx)
+ cancel() // Cancel the context before the call
+
+ // Call the GetCoverArt method
+ _, err := router.GetCoverArt(w, r)
+
+ // Expect no error and no call to the artwork service
+ Expect(err).ToNot(HaveOccurred())
+ Expect(artwork.recvId).To(Equal(""))
+ Expect(artwork.recvSize).To(Equal(0))
+ Expect(artwork.recvSquare).To(BeFalse())
+ Expect(w.Body.String()).To(BeEmpty())
+ })
+
+ It("should not return data if cancelled during the call", func() {
+ // Create a request with a context that will be cancelled
+ ctx, cancel := context.WithCancel(context.Background())
+ defer cancel() // Ensure the context is cancelled after the test (best practices)
+ r := newGetRequest("id=34", "size=128", "square=true")
+ r = r.WithContext(ctx)
+ artwork.ctxCancelFunc = cancel // Set the cancel function to simulate cancellation in the service
+
+ // Call the GetCoverArt method
+ _, err := router.GetCoverArt(w, r)
+
+ // Expect no error and the service to have been called
+ Expect(err).ToNot(HaveOccurred())
+ Expect(artwork.recvId).To(Equal("34"))
+ Expect(artwork.recvSize).To(Equal(128))
+ Expect(artwork.recvSquare).To(BeTrue())
+ Expect(w.Body.String()).To(BeEmpty())
+ })
+ })
})
Describe("GetLyrics", func() {
@@ -84,19 +124,32 @@ var _ = Describe("MediaRetrievalController", func() {
})
Expect(err).ToNot(HaveOccurred())
+ baseTime := time.Date(2025, 1, 1, 0, 0, 0, 0, time.UTC)
mockRepo.SetData(model.MediaFiles{
{
- ID: "1",
- Artist: "Rick Astley",
- Title: "Never Gonna Give You Up",
- Lyrics: string(lyricsJson),
+ ID: "2",
+ Artist: "Rick Astley",
+ Title: "Never Gonna Give You Up",
+ Lyrics: "[]",
+ UpdatedAt: baseTime.Add(2 * time.Hour), // No lyrics, newer
+ },
+ {
+ ID: "1",
+ Artist: "Rick Astley",
+ Title: "Never Gonna Give You Up",
+ Lyrics: string(lyricsJson),
+ UpdatedAt: baseTime.Add(1 * time.Hour), // Has lyrics, older
+ },
+ {
+ ID: "3",
+ Artist: "Rick Astley",
+ Title: "Never Gonna Give You Up",
+ Lyrics: "[]",
+ UpdatedAt: baseTime.Add(3 * time.Hour), // No lyrics, newest
},
})
response, err := router.GetLyrics(r)
- if err != nil {
- log.Error("You're missing something.", err)
- }
- Expect(err).To(BeNil())
+ Expect(err).ToNot(HaveOccurred())
Expect(response.Lyrics.Artist).To(Equal("Rick Astley"))
Expect(response.Lyrics.Title).To(Equal("Never Gonna Give You Up"))
Expect(response.Lyrics.Value).To(Equal("We're no strangers to love\nYou know the rules and so do I\n"))
@@ -105,10 +158,7 @@ var _ = Describe("MediaRetrievalController", func() {
r := newGetRequest("artist=Dheeraj", "title=Rinkiya+Ke+Papa")
mockRepo.SetData(model.MediaFiles{})
response, err := router.GetLyrics(r)
- if err != nil {
- log.Error("You're missing something.", err)
- }
- Expect(err).To(BeNil())
+ Expect(err).ToNot(HaveOccurred())
Expect(response.Lyrics.Artist).To(Equal(""))
Expect(response.Lyrics.Title).To(Equal(""))
Expect(response.Lyrics.Value).To(Equal(""))
@@ -122,16 +172,22 @@ var _ = Describe("MediaRetrievalController", func() {
Artist: "Rick Astley",
Title: "Never Gonna Give You Up",
},
+ {
+ Path: "tests/fixtures/test.mp3",
+ ID: "2",
+ Artist: "Rick Astley",
+ Title: "Never Gonna Give You Up",
+ },
})
response, err := router.GetLyrics(r)
- Expect(err).To(BeNil())
+ Expect(err).ToNot(HaveOccurred())
Expect(response.Lyrics.Artist).To(Equal("Rick Astley"))
Expect(response.Lyrics.Title).To(Equal("Never Gonna Give You Up"))
Expect(response.Lyrics.Value).To(Equal("We're no strangers to love\nYou know the rules and so do I\n"))
})
})
- Describe("getLyricsBySongId", func() {
+ Describe("GetLyricsBySongId", func() {
const syncedLyrics = "[00:18.80]We're no strangers to love\n[00:22.801]You know the rules and so do I"
const unsyncedLyrics = "We're no strangers to love\nYou know the rules and so do I"
const metadata = "[ar:Rick Astley]\n[ti:That one song]\n[offset:-100]"
@@ -271,10 +327,12 @@ var _ = Describe("MediaRetrievalController", func() {
type fakeArtwork struct {
artwork.Artwork
- data string
- err error
- recvId string
- recvSize int
+ data string
+ err error
+ ctxCancelFunc func()
+ recvId string
+ recvSize int
+ recvSquare bool
}
func (c *fakeArtwork) GetOrPlaceholder(_ context.Context, id string, size int, square bool) (io.ReadCloser, time.Time, error) {
@@ -283,27 +341,39 @@ func (c *fakeArtwork) GetOrPlaceholder(_ context.Context, id string, size int, s
}
c.recvId = id
c.recvSize = size
+ c.recvSquare = square
+ if c.ctxCancelFunc != nil {
+ c.ctxCancelFunc() // Simulate context cancellation
+ return nil, time.Time{}, context.Canceled
+ }
return io.NopCloser(bytes.NewReader([]byte(c.data))), time.Time{}, nil
}
type mockedMediaFile struct {
- model.MediaFileRepository
- data model.MediaFiles
+ tests.MockMediaFileRepo
}
-func (m *mockedMediaFile) SetData(mfs model.MediaFiles) {
- m.data = mfs
-}
-
-func (m *mockedMediaFile) GetAll(...model.QueryOptions) (model.MediaFiles, error) {
- return m.data, nil
-}
-
-func (m *mockedMediaFile) Get(id string) (*model.MediaFile, error) {
- for _, mf := range m.data {
- if mf.ID == id {
- return &mf, nil
- }
+func (m *mockedMediaFile) GetAll(opts ...model.QueryOptions) (model.MediaFiles, error) {
+ data, err := m.MockMediaFileRepo.GetAll(opts...)
+ if err != nil {
+ return nil, err
}
- return nil, model.ErrNotFound
+ if len(opts) == 0 || opts[0].Sort != "lyrics, updated_at" {
+ return data, nil
+ }
+
+ // Hardcoded support for lyrics sorting
+ result := slices.Clone(data)
+ // Sort by presence of lyrics, then by updated_at. Respect the order specified in opts.
+ slices.SortFunc(result, func(a, b model.MediaFile) int {
+ diff := cmp.Or(
+ cmp.Compare(a.Lyrics, b.Lyrics),
+ cmp.Compare(a.UpdatedAt.Unix(), b.UpdatedAt.Unix()),
+ )
+ if opts[0].Order == "desc" {
+ return -diff
+ }
+ return diff
+ })
+ return result, nil
}
diff --git a/server/subsonic/middlewares.go b/server/subsonic/middlewares.go
index 04c484791..af1ba448f 100644
--- a/server/subsonic/middlewares.go
+++ b/server/subsonic/middlewares.go
@@ -11,17 +11,21 @@ import (
"net/http"
"net/url"
"strings"
+ "time"
+ "github.com/go-chi/chi/v5/middleware"
ua "github.com/mileusna/useragent"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/consts"
"github.com/navidrome/navidrome/core"
"github.com/navidrome/navidrome/core/auth"
+ "github.com/navidrome/navidrome/core/metrics"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/model/request"
"github.com/navidrome/navidrome/server"
"github.com/navidrome/navidrome/server/subsonic/responses"
+ . "github.com/navidrome/navidrome/utils/gg"
"github.com/navidrome/navidrome/utils/req"
)
@@ -43,12 +47,24 @@ func postFormToQueryParams(next http.Handler) http.Handler {
})
}
+func fromInternalOrProxyAuth(r *http.Request) (string, bool) {
+ username := server.InternalAuth(r)
+
+ // If the username comes from internal auth, do not also do reverse proxy auth, as
+ // the request will have no reverse proxy IP
+ if username != "" {
+ return username, true
+ }
+
+ return server.UsernameFromReverseProxyHeader(r), false
+}
+
func checkRequiredParameters(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
var requiredParameters []string
- var username string
- if username = server.UsernameFromReverseProxyHeader(r); username != "" {
+ username, _ := fromInternalOrProxyAuth(r)
+ if username != "" {
requiredParameters = []string{"v", "c"}
} else {
requiredParameters = []string{"u", "v", "c"}
@@ -87,16 +103,18 @@ func authenticate(ds model.DataStore) func(next http.Handler) http.Handler {
var usr *model.User
var err error
- if username := server.UsernameFromReverseProxyHeader(r); username != "" {
+ username, isInternalAuth := fromInternalOrProxyAuth(r)
+ if username != "" {
+ authType := If(isInternalAuth, "internal", "reverse-proxy")
usr, err = ds.User(ctx).FindByUsername(username)
if errors.Is(err, context.Canceled) {
- log.Debug(ctx, "API: Request canceled when authenticating", "auth", "reverse-proxy", "username", username, "remoteAddr", r.RemoteAddr, err)
+ log.Debug(ctx, "API: Request canceled when authenticating", "auth", authType, "username", username, "remoteAddr", r.RemoteAddr, err)
return
}
if errors.Is(err, model.ErrNotFound) {
- log.Warn(ctx, "API: Invalid login", "auth", "reverse-proxy", "username", username, "remoteAddr", r.RemoteAddr, err)
+ log.Warn(ctx, "API: Invalid login", "auth", authType, "username", username, "remoteAddr", r.RemoteAddr, err)
} else if err != nil {
- log.Error(ctx, "API: Error authenticating username", "auth", "reverse-proxy", "username", username, "remoteAddr", r.RemoteAddr, err)
+ log.Error(ctx, "API: Error authenticating username", "auth", authType, "username", username, "remoteAddr", r.RemoteAddr, err)
}
} else {
p := req.Params(r)
@@ -218,3 +236,37 @@ func playerIDCookieName(userName string) string {
cookieName := fmt.Sprintf("nd-player-%x", userName)
return cookieName
}
+
+const subsonicErrorPointer = "subsonicErrorPointer"
+
+func recordStats(metrics metrics.Metrics) func(next http.Handler) http.Handler {
+ return func(next http.Handler) http.Handler {
+ fn := func(w http.ResponseWriter, r *http.Request) {
+ ww := middleware.NewWrapResponseWriter(w, r.ProtoMajor)
+
+ status := int32(-1)
+ contextWithStatus := context.WithValue(r.Context(), subsonicErrorPointer, &status)
+
+ start := time.Now()
+ defer func() {
+ elapsed := time.Since(start).Milliseconds()
+
+ // We want to get the client name (even if not present for certain endpoints)
+ p := req.Params(r)
+ client, _ := p.String("c")
+
+ // If there is no Subsonic status (e.g., HTTP 501 not implemented), fallback to HTTP
+ if status == -1 {
+ status = int32(ww.Status())
+ }
+
+ shortPath := strings.Replace(r.URL.Path, ".view", "", 1)
+
+ metrics.RecordRequest(r.Context(), shortPath, r.Method, client, status, elapsed)
+ }()
+
+ next.ServeHTTP(ww, r.WithContext(contextWithStatus))
+ }
+ return http.HandlerFunc(fn)
+ }
+}
diff --git a/server/subsonic/middlewares_test.go b/server/subsonic/middlewares_test.go
index 3fe577fad..a30d5b3af 100644
--- a/server/subsonic/middlewares_test.go
+++ b/server/subsonic/middlewares_test.go
@@ -281,6 +281,31 @@ var _ = Describe("Middlewares", func() {
Expect(next.called).To(BeFalse())
})
})
+
+ When("using internal authentication", func() {
+ It("passes authentication with correct internal credentials", func() {
+ // Simulate internal authentication by setting the context with WithInternalAuth
+ r := newGetRequest()
+ r = r.WithContext(request.WithInternalAuth(r.Context(), "admin"))
+ cp := authenticate(ds)(next)
+ cp.ServeHTTP(w, r)
+
+ Expect(next.called).To(BeTrue())
+ user, _ := request.UserFrom(next.req.Context())
+ Expect(user.UserName).To(Equal("admin"))
+ })
+
+ It("fails authentication with missing internal context", func() {
+ r := newGetRequest("u=admin")
+ // Do not set the internal auth context
+ cp := authenticate(ds)(next)
+ cp.ServeHTTP(w, r)
+
+ // Internal auth requires the context, so this should fail
+ Expect(w.Body.String()).To(ContainSubstring(`code="40"`))
+ Expect(next.called).To(BeFalse())
+ })
+ })
})
Describe("GetPlayer", func() {
diff --git a/server/subsonic/opensubsonic.go b/server/subsonic/opensubsonic.go
index 17ce3c2b0..a364651c5 100644
--- a/server/subsonic/opensubsonic.go
+++ b/server/subsonic/opensubsonic.go
@@ -12,6 +12,7 @@ func (api *Router) GetOpenSubsonicExtensions(_ *http.Request) (*responses.Subson
{Name: "transcodeOffset", Versions: []int32{1}},
{Name: "formPost", Versions: []int32{1}},
{Name: "songLyrics", Versions: []int32{1}},
+ {Name: "indexBasedQueue", Versions: []int32{1}},
}
return response, nil
}
diff --git a/server/subsonic/opensubsonic_test.go b/server/subsonic/opensubsonic_test.go
index d92ea4c67..58dca682c 100644
--- a/server/subsonic/opensubsonic_test.go
+++ b/server/subsonic/opensubsonic_test.go
@@ -19,7 +19,7 @@ var _ = Describe("GetOpenSubsonicExtensions", func() {
)
BeforeEach(func() {
- router = subsonic.New(nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil)
+ router = subsonic.New(nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil)
w = httptest.NewRecorder()
r = httptest.NewRequest("GET", "/getOpenSubsonicExtensions?f=json", nil)
})
@@ -35,10 +35,11 @@ var _ = Describe("GetOpenSubsonicExtensions", func() {
err := json.Unmarshal(w.Body.Bytes(), &response)
Expect(err).NotTo(HaveOccurred())
Expect(*response.Subsonic.OpenSubsonicExtensions).To(SatisfyAll(
- HaveLen(3),
+ HaveLen(4),
ContainElement(responses.OpenSubsonicExtension{Name: "transcodeOffset", Versions: []int32{1}}),
ContainElement(responses.OpenSubsonicExtension{Name: "formPost", Versions: []int32{1}}),
ContainElement(responses.OpenSubsonicExtension{Name: "songLyrics", Versions: []int32{1}}),
+ ContainElement(responses.OpenSubsonicExtension{Name: "indexBasedQueue", Versions: []int32{1}}),
))
})
})
diff --git a/server/subsonic/playlists.go b/server/subsonic/playlists.go
index 555c9eb48..23fac6814 100644
--- a/server/subsonic/playlists.go
+++ b/server/subsonic/playlists.go
@@ -76,7 +76,7 @@ func (api *Router) create(ctx context.Context, playlistId, name string, ids []st
pls.OwnerID = owner.ID
}
pls.Tracks = nil
- pls.AddTracks(ids)
+ pls.AddMediaFilesByID(ids)
err = tx.Playlist(ctx).Put(pls)
playlistId = pls.ID
@@ -131,14 +131,8 @@ func (api *Router) UpdatePlaylist(r *http.Request) (*responses.Subsonic, error)
if s, err := p.String("name"); err == nil {
plsName = &s
}
- var comment *string
- if s, err := p.String("comment"); err == nil {
- comment = &s
- }
- var public *bool
- if p, err := p.Bool("public"); err == nil {
- public = &p
- }
+ comment := p.StringPtr("comment")
+ public := p.BoolPtr("public")
log.Debug(r, "Updating playlist", "id", playlistId)
if plsName != nil {
diff --git a/server/subsonic/playlists_test.go b/server/subsonic/playlists_test.go
new file mode 100644
index 000000000..c0a007d6a
--- /dev/null
+++ b/server/subsonic/playlists_test.go
@@ -0,0 +1,88 @@
+package subsonic
+
+import (
+ "context"
+
+ "github.com/navidrome/navidrome/core"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ core.Playlists = (*fakePlaylists)(nil)
+
+var _ = Describe("UpdatePlaylist", func() {
+ var router *Router
+ var ds model.DataStore
+ var playlists *fakePlaylists
+
+ BeforeEach(func() {
+ ds = &tests.MockDataStore{}
+ playlists = &fakePlaylists{}
+ router = New(ds, nil, nil, nil, nil, nil, nil, nil, playlists, nil, nil, nil, nil)
+ })
+
+ It("clears the comment when parameter is empty", func() {
+ r := newGetRequest("playlistId=123", "comment=")
+ _, err := router.UpdatePlaylist(r)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(playlists.lastPlaylistID).To(Equal("123"))
+ Expect(playlists.lastComment).ToNot(BeNil())
+ Expect(*playlists.lastComment).To(Equal(""))
+ })
+
+ It("leaves comment unchanged when parameter is missing", func() {
+ r := newGetRequest("playlistId=123")
+ _, err := router.UpdatePlaylist(r)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(playlists.lastPlaylistID).To(Equal("123"))
+ Expect(playlists.lastComment).To(BeNil())
+ })
+
+ It("sets public to true when parameter is 'true'", func() {
+ r := newGetRequest("playlistId=123", "public=true")
+ _, err := router.UpdatePlaylist(r)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(playlists.lastPlaylistID).To(Equal("123"))
+ Expect(playlists.lastPublic).ToNot(BeNil())
+ Expect(*playlists.lastPublic).To(BeTrue())
+ })
+
+ It("sets public to false when parameter is 'false'", func() {
+ r := newGetRequest("playlistId=123", "public=false")
+ _, err := router.UpdatePlaylist(r)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(playlists.lastPlaylistID).To(Equal("123"))
+ Expect(playlists.lastPublic).ToNot(BeNil())
+ Expect(*playlists.lastPublic).To(BeFalse())
+ })
+
+ It("leaves public unchanged when parameter is missing", func() {
+ r := newGetRequest("playlistId=123")
+ _, err := router.UpdatePlaylist(r)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(playlists.lastPlaylistID).To(Equal("123"))
+ Expect(playlists.lastPublic).To(BeNil())
+ })
+})
+
+type fakePlaylists struct {
+ core.Playlists
+ lastPlaylistID string
+ lastName *string
+ lastComment *string
+ lastPublic *bool
+ lastAdd []string
+ lastRemove []int
+}
+
+func (f *fakePlaylists) Update(ctx context.Context, playlistID string, name *string, comment *string, public *bool, idsToAdd []string, idxToRemove []int) error {
+ f.lastPlaylistID = playlistID
+ f.lastName = name
+ f.lastComment = comment
+ f.lastPublic = public
+ f.lastAdd = idsToAdd
+ f.lastRemove = idxToRemove
+ return nil
+}
diff --git a/server/subsonic/responses/.snapshots/Responses AlbumWithSongsID3 with data should match .JSON b/server/subsonic/responses/.snapshots/Responses AlbumWithSongsID3 with data should match .JSON
index 78b5c6e7a..c2a29b22a 100644
--- a/server/subsonic/responses/.snapshots/Responses AlbumWithSongsID3 with data should match .JSON
+++ b/server/subsonic/responses/.snapshots/Responses AlbumWithSongsID3 with data should match .JSON
@@ -166,6 +166,52 @@
],
"displayComposer": "composer 1 \u0026 composer 2",
"explicitStatus": "clean"
+ },
+ {
+ "id": "2",
+ "isDir": true,
+ "title": "title",
+ "album": "album",
+ "artist": "artist",
+ "track": 1,
+ "year": 1985,
+ "genre": "Rock",
+ "coverArt": "1",
+ "size": 8421341,
+ "contentType": "audio/flac",
+ "suffix": "flac",
+ "starred": "2016-03-02T20:30:00Z",
+ "transcodedContentType": "audio/mpeg",
+ "transcodedSuffix": "mp3",
+ "duration": 146,
+ "bitRate": 320,
+ "isVideo": false,
+ "bpm": 0,
+ "comment": "",
+ "sortName": "",
+ "mediaType": "",
+ "musicBrainzId": "",
+ "isrc": [],
+ "genres": [],
+ "replayGain": {
+ "trackGain": 0,
+ "albumGain": 0,
+ "trackPeak": 0,
+ "albumPeak": 0,
+ "baseGain": 0,
+ "fallbackGain": 0
+ },
+ "channelCount": 0,
+ "samplingRate": 0,
+ "bitDepth": 0,
+ "moods": [],
+ "artists": [],
+ "displayArtist": "",
+ "albumArtists": [],
+ "displayAlbumArtist": "",
+ "contributors": [],
+ "displayComposer": "",
+ "explicitStatus": ""
}
]
}
diff --git a/server/subsonic/responses/.snapshots/Responses AlbumWithSongsID3 with data should match .XML b/server/subsonic/responses/.snapshots/Responses AlbumWithSongsID3 with data should match .XML
index f3281d9ee..1ad3e600c 100644
--- a/server/subsonic/responses/.snapshots/Responses AlbumWithSongsID3 with data should match .XML
+++ b/server/subsonic/responses/.snapshots/Responses AlbumWithSongsID3 with data should match .XML
@@ -33,5 +33,8 @@
+
+
+
diff --git a/server/subsonic/responses/.snapshots/Responses Child with data should match .JSON b/server/subsonic/responses/.snapshots/Responses Child with data should match .JSON
index d64ae9e7f..fde40646a 100644
--- a/server/subsonic/responses/.snapshots/Responses Child with data should match .JSON
+++ b/server/subsonic/responses/.snapshots/Responses Child with data should match .JSON
@@ -112,6 +112,37 @@
],
"displayComposer": "composer 1 \u0026 composer 2",
"explicitStatus": "clean"
+ },
+ {
+ "id": "",
+ "isDir": false,
+ "isVideo": false,
+ "bpm": 0,
+ "comment": "",
+ "sortName": "",
+ "mediaType": "",
+ "musicBrainzId": "",
+ "isrc": [],
+ "genres": [],
+ "replayGain": {
+ "trackGain": 0,
+ "albumGain": 0,
+ "trackPeak": 0,
+ "albumPeak": 0,
+ "baseGain": 0,
+ "fallbackGain": 0
+ },
+ "channelCount": 0,
+ "samplingRate": 0,
+ "bitDepth": 0,
+ "moods": [],
+ "artists": [],
+ "displayArtist": "",
+ "albumArtists": [],
+ "displayAlbumArtist": "",
+ "contributors": [],
+ "displayComposer": "",
+ "explicitStatus": ""
}
],
"id": "1",
diff --git a/server/subsonic/responses/.snapshots/Responses Child with data should match .XML b/server/subsonic/responses/.snapshots/Responses Child with data should match .XML
index 639fd3f60..faea8ee93 100644
--- a/server/subsonic/responses/.snapshots/Responses Child with data should match .XML
+++ b/server/subsonic/responses/.snapshots/Responses Child with data should match .XML
@@ -25,5 +25,8 @@
+
+
+
diff --git a/server/subsonic/responses/.snapshots/Responses PlayQueue without data should match .JSON b/server/subsonic/responses/.snapshots/Responses PlayQueue without data should match .JSON
index 88eebb276..70b10c059 100644
--- a/server/subsonic/responses/.snapshots/Responses PlayQueue without data should match .JSON
+++ b/server/subsonic/responses/.snapshots/Responses PlayQueue without data should match .JSON
@@ -6,6 +6,7 @@
"openSubsonic": true,
"playQueue": {
"username": "",
+ "changed": "0001-01-01T00:00:00Z",
"changedBy": ""
}
}
diff --git a/server/subsonic/responses/.snapshots/Responses PlayQueue without data should match .XML b/server/subsonic/responses/.snapshots/Responses PlayQueue without data should match .XML
index 5af3d9157..597781cbd 100644
--- a/server/subsonic/responses/.snapshots/Responses PlayQueue without data should match .XML
+++ b/server/subsonic/responses/.snapshots/Responses PlayQueue without data should match .XML
@@ -1,3 +1,3 @@
-
+
diff --git a/server/subsonic/responses/.snapshots/Responses PlayQueueByIndex with data should match .JSON b/server/subsonic/responses/.snapshots/Responses PlayQueueByIndex with data should match .JSON
new file mode 100644
index 000000000..efc032ca6
--- /dev/null
+++ b/server/subsonic/responses/.snapshots/Responses PlayQueueByIndex with data should match .JSON
@@ -0,0 +1,22 @@
+{
+ "status": "ok",
+ "version": "1.16.1",
+ "type": "navidrome",
+ "serverVersion": "v0.55.0",
+ "openSubsonic": true,
+ "playQueueByIndex": {
+ "entry": [
+ {
+ "id": "1",
+ "isDir": false,
+ "title": "title",
+ "isVideo": false
+ }
+ ],
+ "currentIndex": 0,
+ "position": 243,
+ "username": "user1",
+ "changed": "0001-01-01T00:00:00Z",
+ "changedBy": "a_client"
+ }
+}
diff --git a/server/subsonic/responses/.snapshots/Responses PlayQueueByIndex with data should match .XML b/server/subsonic/responses/.snapshots/Responses PlayQueueByIndex with data should match .XML
new file mode 100644
index 000000000..1d31b334e
--- /dev/null
+++ b/server/subsonic/responses/.snapshots/Responses PlayQueueByIndex with data should match .XML
@@ -0,0 +1,5 @@
+
+
+
+
+
diff --git a/server/subsonic/responses/.snapshots/Responses PlayQueueByIndex without data should match .JSON b/server/subsonic/responses/.snapshots/Responses PlayQueueByIndex without data should match .JSON
new file mode 100644
index 000000000..ad49a35e5
--- /dev/null
+++ b/server/subsonic/responses/.snapshots/Responses PlayQueueByIndex without data should match .JSON
@@ -0,0 +1,12 @@
+{
+ "status": "ok",
+ "version": "1.16.1",
+ "type": "navidrome",
+ "serverVersion": "v0.55.0",
+ "openSubsonic": true,
+ "playQueueByIndex": {
+ "username": "",
+ "changed": "0001-01-01T00:00:00Z",
+ "changedBy": ""
+ }
+}
diff --git a/server/subsonic/responses/.snapshots/Responses PlayQueueByIndex without data should match .XML b/server/subsonic/responses/.snapshots/Responses PlayQueueByIndex without data should match .XML
new file mode 100644
index 000000000..d99681f4c
--- /dev/null
+++ b/server/subsonic/responses/.snapshots/Responses PlayQueueByIndex without data should match .XML
@@ -0,0 +1,3 @@
+
+
+
diff --git a/server/subsonic/responses/responses.go b/server/subsonic/responses/responses.go
index 4a7ebbe83..0724d2fff 100644
--- a/server/subsonic/responses/responses.go
+++ b/server/subsonic/responses/responses.go
@@ -60,6 +60,7 @@ type Subsonic struct {
// OpenSubsonic extensions
OpenSubsonicExtensions *OpenSubsonicExtensions `xml:"openSubsonicExtensions,omitempty" json:"openSubsonicExtensions,omitempty"`
LyricsList *LyricsList `xml:"lyricsList,omitempty" json:"lyricsList,omitempty"`
+ PlayQueueByIndex *PlayQueueByIndex `xml:"playQueueByIndex,omitempty" json:"playQueueByIndex,omitempty"`
}
const (
@@ -439,12 +440,21 @@ type TopSongs struct {
}
type PlayQueue struct {
- Entry []Child `xml:"entry,omitempty" json:"entry,omitempty"`
- Current string `xml:"current,attr,omitempty" json:"current,omitempty"`
- Position int64 `xml:"position,attr,omitempty" json:"position,omitempty"`
- Username string `xml:"username,attr" json:"username"`
- Changed *time.Time `xml:"changed,attr,omitempty" json:"changed,omitempty"`
- ChangedBy string `xml:"changedBy,attr" json:"changedBy"`
+ Entry []Child `xml:"entry,omitempty" json:"entry,omitempty"`
+ Current string `xml:"current,attr,omitempty" json:"current,omitempty"`
+ Position int64 `xml:"position,attr,omitempty" json:"position,omitempty"`
+ Username string `xml:"username,attr" json:"username"`
+ Changed time.Time `xml:"changed,attr" json:"changed"`
+ ChangedBy string `xml:"changedBy,attr" json:"changedBy"`
+}
+
+type PlayQueueByIndex struct {
+ Entry []Child `xml:"entry,omitempty" json:"entry,omitempty"`
+ CurrentIndex *int `xml:"currentIndex,attr,omitempty" json:"currentIndex,omitempty"`
+ Position int64 `xml:"position,attr,omitempty" json:"position,omitempty"`
+ Username string `xml:"username,attr" json:"username"`
+ Changed time.Time `xml:"changed,attr" json:"changed"`
+ ChangedBy string `xml:"changedBy,attr" json:"changedBy"`
}
type Bookmark struct {
@@ -546,16 +556,16 @@ type ItemGenre struct {
}
type ReplayGain struct {
- TrackGain float64 `xml:"trackGain,omitempty,attr" json:"trackGain,omitempty"`
- AlbumGain float64 `xml:"albumGain,omitempty,attr" json:"albumGain,omitempty"`
- TrackPeak float64 `xml:"trackPeak,omitempty,attr" json:"trackPeak,omitempty"`
- AlbumPeak float64 `xml:"albumPeak,omitempty,attr" json:"albumPeak,omitempty"`
- BaseGain float64 `xml:"baseGain,omitempty,attr" json:"baseGain,omitempty"`
- FallbackGain float64 `xml:"fallbackGain,omitempty,attr" json:"fallbackGain,omitempty"`
+ TrackGain *float64 `xml:"trackGain,omitempty,attr" json:"trackGain,omitempty"`
+ AlbumGain *float64 `xml:"albumGain,omitempty,attr" json:"albumGain,omitempty"`
+ TrackPeak *float64 `xml:"trackPeak,omitempty,attr" json:"trackPeak,omitempty"`
+ AlbumPeak *float64 `xml:"albumPeak,omitempty,attr" json:"albumPeak,omitempty"`
+ BaseGain *float64 `xml:"baseGain,omitempty,attr" json:"baseGain,omitempty"`
+ FallbackGain *float64 `xml:"fallbackGain,omitempty,attr" json:"fallbackGain,omitempty"`
}
func (r ReplayGain) MarshalXML(e *xml.Encoder, start xml.StartElement) error {
- if r.TrackGain == 0 && r.AlbumGain == 0 && r.TrackPeak == 0 && r.AlbumPeak == 0 && r.BaseGain == 0 && r.FallbackGain == 0 {
+ if r.TrackGain == nil && r.AlbumGain == nil && r.TrackPeak == nil && r.AlbumPeak == nil && r.BaseGain == nil && r.FallbackGain == nil {
return nil
}
type replayGain ReplayGain
diff --git a/server/subsonic/responses/responses_test.go b/server/subsonic/responses/responses_test.go
index 9fcd6078e..2ee8e080d 100644
--- a/server/subsonic/responses/responses_test.go
+++ b/server/subsonic/responses/responses_test.go
@@ -12,6 +12,7 @@ import (
"github.com/navidrome/navidrome/consts"
. "github.com/navidrome/navidrome/server/subsonic/responses"
+ "github.com/navidrome/navidrome/utils/gg"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
)
@@ -213,7 +214,7 @@ var _ = Describe("Responses", func() {
Context("with data", func() {
BeforeEach(func() {
response.Directory = &Directory{Id: "1", Name: "N"}
- child := make([]Child, 1)
+ child := make([]Child, 2)
t := time.Date(2016, 03, 2, 20, 30, 0, 0, time.UTC)
child[0] = Child{
Id: "1", IsDir: true, Title: "title", Album: "album", Artist: "artist", Track: 1,
@@ -227,7 +228,7 @@ var _ = Describe("Responses", func() {
Isrc: []string{"ISRC-1", "ISRC-2"},
BPM: 127, ChannelCount: 2, SamplingRate: 44100, BitDepth: 16,
Moods: []string{"happy", "sad"},
- ReplayGain: ReplayGain{TrackGain: 1, AlbumGain: 2, TrackPeak: 3, AlbumPeak: 4, BaseGain: 5, FallbackGain: 6},
+ ReplayGain: ReplayGain{TrackGain: gg.P(1.0), AlbumGain: gg.P(2.0), TrackPeak: gg.P(3.0), AlbumPeak: gg.P(4.0), BaseGain: gg.P(5.0), FallbackGain: gg.P(6.0)},
DisplayArtist: "artist 1 & artist 2",
Artists: []ArtistID3Ref{
{Id: "1", Name: "artist1"},
@@ -247,6 +248,9 @@ var _ = Describe("Responses", func() {
},
ExplicitStatus: "clean",
}
+ child[1].OpenSubsonicChild = &OpenSubsonicChild{
+ ReplayGain: ReplayGain{TrackGain: gg.P(0.0), AlbumGain: gg.P(0.0), TrackPeak: gg.P(0.0), AlbumPeak: gg.P(0.0), BaseGain: gg.P(0.0), FallbackGain: gg.P(0.0)},
+ }
response.Directory.Child = child
})
@@ -309,13 +313,18 @@ var _ = Describe("Responses", func() {
Year: 1985, Genre: "Rock", CoverArt: "1", Size: 8421341, ContentType: "audio/flac",
Suffix: "flac", TranscodedContentType: "audio/mpeg", TranscodedSuffix: "mp3",
Duration: 146, BitRate: 320, Starred: &t,
+ }, {
+ Id: "2", IsDir: true, Title: "title", Album: "album", Artist: "artist", Track: 1,
+ Year: 1985, Genre: "Rock", CoverArt: "1", Size: 8421341, ContentType: "audio/flac",
+ Suffix: "flac", TranscodedContentType: "audio/mpeg", TranscodedSuffix: "mp3",
+ Duration: 146, BitRate: 320, Starred: &t,
}}
songs[0].OpenSubsonicChild = &OpenSubsonicChild{
Genres: []ItemGenre{{Name: "rock"}, {Name: "progressive"}},
Comment: "a comment", MediaType: MediaTypeSong, MusicBrainzId: "4321", SortName: "sorted song",
Isrc: []string{"ISRC-1"},
Moods: []string{"happy", "sad"},
- ReplayGain: ReplayGain{TrackGain: 1, AlbumGain: 2, TrackPeak: 3, AlbumPeak: 4, BaseGain: 5, FallbackGain: 6},
+ ReplayGain: ReplayGain{TrackGain: gg.P(1.0), AlbumGain: gg.P(2.0), TrackPeak: gg.P(3.0), AlbumPeak: gg.P(4.0), BaseGain: gg.P(5.0), FallbackGain: gg.P(6.0)},
BPM: 127, ChannelCount: 2, SamplingRate: 44100, BitDepth: 16,
DisplayArtist: "artist1 & artist2",
Artists: []ArtistID3Ref{
@@ -334,6 +343,9 @@ var _ = Describe("Responses", func() {
DisplayComposer: "composer 1 & composer 2",
ExplicitStatus: "clean",
}
+ songs[1].OpenSubsonicChild = &OpenSubsonicChild{
+ ReplayGain: ReplayGain{TrackGain: gg.P(0.0), AlbumGain: gg.P(0.0), TrackPeak: gg.P(0.0), AlbumPeak: gg.P(0.0), BaseGain: gg.P(0.0), FallbackGain: gg.P(0.0)},
+ }
response.AlbumWithSongsID3.AlbumID3 = album
response.AlbumWithSongsID3.Song = songs
})
@@ -756,7 +768,7 @@ var _ = Describe("Responses", func() {
response.PlayQueue.Username = "user1"
response.PlayQueue.Current = "111"
response.PlayQueue.Position = 243
- response.PlayQueue.Changed = &time.Time{}
+ response.PlayQueue.Changed = time.Time{}
response.PlayQueue.ChangedBy = "a_client"
child := make([]Child, 1)
child[0] = Child{Id: "1", Title: "title", IsDir: false}
@@ -771,6 +783,40 @@ var _ = Describe("Responses", func() {
})
})
+ Describe("PlayQueueByIndex", func() {
+ BeforeEach(func() {
+ response.PlayQueueByIndex = &PlayQueueByIndex{}
+ })
+
+ Context("without data", func() {
+ It("should match .XML", func() {
+ Expect(xml.MarshalIndent(response, "", " ")).To(MatchSnapshot())
+ })
+ It("should match .JSON", func() {
+ Expect(json.MarshalIndent(response, "", " ")).To(MatchSnapshot())
+ })
+ })
+
+ Context("with data", func() {
+ BeforeEach(func() {
+ response.PlayQueueByIndex.Username = "user1"
+ response.PlayQueueByIndex.CurrentIndex = gg.P(0)
+ response.PlayQueueByIndex.Position = 243
+ response.PlayQueueByIndex.Changed = time.Time{}
+ response.PlayQueueByIndex.ChangedBy = "a_client"
+ child := make([]Child, 1)
+ child[0] = Child{Id: "1", Title: "title", IsDir: false}
+ response.PlayQueueByIndex.Entry = child
+ })
+ It("should match .XML", func() {
+ Expect(xml.MarshalIndent(response, "", " ")).To(MatchSnapshot())
+ })
+ It("should match .JSON", func() {
+ Expect(json.MarshalIndent(response, "", " ")).To(MatchSnapshot())
+ })
+ })
+ })
+
Describe("Shares", func() {
BeforeEach(func() {
response.Shares = &Shares{}
diff --git a/server/subsonic/searching.go b/server/subsonic/searching.go
index f66846f35..ba1071320 100644
--- a/server/subsonic/searching.go
+++ b/server/subsonic/searching.go
@@ -8,6 +8,7 @@ import (
"strings"
"time"
+ . "github.com/Masterminds/squirrel"
"github.com/deluan/sanitize"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
@@ -41,9 +42,9 @@ func (api *Router) getSearchParams(r *http.Request) (*searchParams, error) {
return sp, nil
}
-type searchFunc[T any] func(q string, offset int, size int, includeMissing bool) (T, error)
+type searchFunc[T any] func(q string, offset int, size int, options ...model.QueryOptions) (T, error)
-func callSearch[T any](ctx context.Context, s searchFunc[T], q string, offset, size int, result *T) func() error {
+func callSearch[T any](ctx context.Context, s searchFunc[T], q string, offset, size int, result *T, options ...model.QueryOptions) func() error {
return func() error {
if size == 0 {
return nil
@@ -51,7 +52,7 @@ func callSearch[T any](ctx context.Context, s searchFunc[T], q string, offset, s
typ := strings.TrimPrefix(reflect.TypeOf(*result).String(), "model.")
var err error
start := time.Now()
- *result, err = s(q, offset, size, false)
+ *result, err = s(q, offset, size, options...)
if err != nil {
log.Error(ctx, "Error searching "+typ, "query", q, "elapsed", time.Since(start), err)
} else {
@@ -61,15 +62,31 @@ func callSearch[T any](ctx context.Context, s searchFunc[T], q string, offset, s
}
}
-func (api *Router) searchAll(ctx context.Context, sp *searchParams) (mediaFiles model.MediaFiles, albums model.Albums, artists model.Artists) {
+func (api *Router) searchAll(ctx context.Context, sp *searchParams, musicFolderIds []int) (mediaFiles model.MediaFiles, albums model.Albums, artists model.Artists) {
start := time.Now()
q := sanitize.Accents(strings.ToLower(strings.TrimSuffix(sp.query, "*")))
+ // Create query options for library filtering
+ var options []model.QueryOptions
+ var artistOptions []model.QueryOptions
+ if len(musicFolderIds) > 0 {
+ // For MediaFiles and Albums, use direct library_id filter
+ options = append(options, model.QueryOptions{
+ Filters: Eq{"library_id": musicFolderIds},
+ })
+ // For Artists, use the repository's built-in library filtering mechanism
+ // which properly handles the library_artist table joins
+ // TODO Revisit library filtering in sql_base_repository.go
+ artistOptions = append(artistOptions, model.QueryOptions{
+ Filters: Eq{"library_artist.library_id": musicFolderIds},
+ })
+ }
+
// Run searches in parallel
g, ctx := errgroup.WithContext(ctx)
- g.Go(callSearch(ctx, api.ds.MediaFile(ctx).Search, q, sp.songOffset, sp.songCount, &mediaFiles))
- g.Go(callSearch(ctx, api.ds.Album(ctx).Search, q, sp.albumOffset, sp.albumCount, &albums))
- g.Go(callSearch(ctx, api.ds.Artist(ctx).Search, q, sp.artistOffset, sp.artistCount, &artists))
+ g.Go(callSearch(ctx, api.ds.MediaFile(ctx).Search, q, sp.songOffset, sp.songCount, &mediaFiles, options...))
+ g.Go(callSearch(ctx, api.ds.Album(ctx).Search, q, sp.albumOffset, sp.albumCount, &albums, options...))
+ g.Go(callSearch(ctx, api.ds.Artist(ctx).Search, q, sp.artistOffset, sp.artistCount, &artists, artistOptions...))
err := g.Wait()
if err == nil {
log.Debug(ctx, fmt.Sprintf("Search resulted in %d songs, %d albums and %d artists",
@@ -86,7 +103,13 @@ func (api *Router) Search2(r *http.Request) (*responses.Subsonic, error) {
if err != nil {
return nil, err
}
- mfs, als, as := api.searchAll(ctx, sp)
+
+ // Get optional library IDs from musicFolderId parameter
+ musicFolderIds, err := selectedMusicFolderIds(r, false)
+ if err != nil {
+ return nil, err
+ }
+ mfs, als, as := api.searchAll(ctx, sp, musicFolderIds)
response := newResponse()
searchResult2 := &responses.SearchResult2{}
@@ -115,7 +138,13 @@ func (api *Router) Search3(r *http.Request) (*responses.Subsonic, error) {
if err != nil {
return nil, err
}
- mfs, als, as := api.searchAll(ctx, sp)
+
+ // Get optional library IDs from musicFolderId parameter
+ musicFolderIds, err := selectedMusicFolderIds(r, false)
+ if err != nil {
+ return nil, err
+ }
+ mfs, als, as := api.searchAll(ctx, sp, musicFolderIds)
response := newResponse()
searchResult3 := &responses.SearchResult3{}
diff --git a/server/subsonic/searching_test.go b/server/subsonic/searching_test.go
new file mode 100644
index 000000000..dfe3a45c4
--- /dev/null
+++ b/server/subsonic/searching_test.go
@@ -0,0 +1,208 @@
+package subsonic
+
+import (
+ "github.com/Masterminds/squirrel"
+ "github.com/navidrome/navidrome/core/auth"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/tests"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("Search", func() {
+ var router *Router
+ var ds model.DataStore
+ var mockAlbumRepo *tests.MockAlbumRepo
+ var mockArtistRepo *tests.MockArtistRepo
+ var mockMediaFileRepo *tests.MockMediaFileRepo
+
+ BeforeEach(func() {
+ ds = &tests.MockDataStore{}
+ auth.Init(ds)
+
+ router = New(ds, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil)
+
+ // Get references to the mock repositories so we can inspect their Options
+ mockAlbumRepo = ds.Album(nil).(*tests.MockAlbumRepo)
+ mockArtistRepo = ds.Artist(nil).(*tests.MockArtistRepo)
+ mockMediaFileRepo = ds.MediaFile(nil).(*tests.MockMediaFileRepo)
+ })
+
+ Context("musicFolderId parameter", func() {
+ assertQueryOptions := func(filter squirrel.Sqlizer, expectedQuery string, expectedArgs ...interface{}) {
+ GinkgoHelper()
+ query, args, err := filter.ToSql()
+ Expect(err).ToNot(HaveOccurred())
+ Expect(query).To(ContainSubstring(expectedQuery))
+ Expect(args).To(ContainElements(expectedArgs...))
+ }
+
+ Describe("Search2", func() {
+ It("should accept musicFolderId parameter", func() {
+ r := newGetRequest("query=test", "musicFolderId=1")
+ ctx := request.WithUser(r.Context(), model.User{
+ ID: "user1",
+ UserName: "testuser",
+ Libraries: []model.Library{{ID: 1, Name: "Library 1"}},
+ })
+ r = r.WithContext(ctx)
+
+ resp, err := router.Search2(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp).ToNot(BeNil())
+ Expect(resp.SearchResult2).ToNot(BeNil())
+
+ // Verify that library filter was applied to all repositories
+ assertQueryOptions(mockAlbumRepo.Options.Filters, "library_id IN (?)", 1)
+ assertQueryOptions(mockArtistRepo.Options.Filters, "library_id IN (?)", 1)
+ assertQueryOptions(mockMediaFileRepo.Options.Filters, "library_id IN (?)", 1)
+ })
+
+ It("should return results from all accessible libraries when musicFolderId is not provided", func() {
+ r := newGetRequest("query=test")
+ ctx := request.WithUser(r.Context(), model.User{
+ ID: "user1",
+ UserName: "testuser",
+ Libraries: []model.Library{
+ {ID: 1, Name: "Library 1"},
+ {ID: 2, Name: "Library 2"},
+ {ID: 3, Name: "Library 3"},
+ },
+ })
+ r = r.WithContext(ctx)
+
+ resp, err := router.Search2(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp).ToNot(BeNil())
+ Expect(resp.SearchResult2).ToNot(BeNil())
+
+ // Verify that library filter was applied to all repositories with all accessible libraries
+ assertQueryOptions(mockAlbumRepo.Options.Filters, "library_id IN (?,?,?)", 1, 2, 3)
+ assertQueryOptions(mockArtistRepo.Options.Filters, "library_id IN (?,?,?)", 1, 2, 3)
+ assertQueryOptions(mockMediaFileRepo.Options.Filters, "library_id IN (?,?,?)", 1, 2, 3)
+ })
+
+ It("should return empty results when user has no accessible libraries", func() {
+ r := newGetRequest("query=test")
+ ctx := request.WithUser(r.Context(), model.User{
+ ID: "user1",
+ UserName: "testuser",
+ Libraries: []model.Library{}, // No libraries
+ })
+ r = r.WithContext(ctx)
+
+ resp, err := router.Search2(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp).ToNot(BeNil())
+ Expect(resp.SearchResult2).ToNot(BeNil())
+ Expect(mockAlbumRepo.Options.Filters).To(BeNil())
+ Expect(mockArtistRepo.Options.Filters).To(BeNil())
+ Expect(mockMediaFileRepo.Options.Filters).To(BeNil())
+ })
+
+ It("should return error for inaccessible musicFolderId", func() {
+ r := newGetRequest("query=test", "musicFolderId=999")
+ ctx := request.WithUser(r.Context(), model.User{
+ ID: "user1",
+ UserName: "testuser",
+ Libraries: []model.Library{{ID: 1, Name: "Library 1"}},
+ })
+ r = r.WithContext(ctx)
+
+ resp, err := router.Search2(r)
+
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("Library 999 not found or not accessible"))
+ Expect(resp).To(BeNil())
+ })
+ })
+
+ Describe("Search3", func() {
+ It("should accept musicFolderId parameter", func() {
+ r := newGetRequest("query=test", "musicFolderId=1")
+ ctx := request.WithUser(r.Context(), model.User{
+ ID: "user1",
+ UserName: "testuser",
+ Libraries: []model.Library{{ID: 1, Name: "Library 1"}},
+ })
+ r = r.WithContext(ctx)
+
+ resp, err := router.Search3(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp).ToNot(BeNil())
+ Expect(resp.SearchResult3).ToNot(BeNil())
+
+ // Verify that library filter was applied to all repositories
+ assertQueryOptions(mockAlbumRepo.Options.Filters, "library_id IN (?)", 1)
+ assertQueryOptions(mockArtistRepo.Options.Filters, "library_id IN (?)", 1)
+ assertQueryOptions(mockMediaFileRepo.Options.Filters, "library_id IN (?)", 1)
+ })
+
+ It("should return results from all accessible libraries when musicFolderId is not provided", func() {
+ r := newGetRequest("query=test")
+ ctx := request.WithUser(r.Context(), model.User{
+ ID: "user1",
+ UserName: "testuser",
+ Libraries: []model.Library{
+ {ID: 1, Name: "Library 1"},
+ {ID: 2, Name: "Library 2"},
+ {ID: 3, Name: "Library 3"},
+ },
+ })
+ r = r.WithContext(ctx)
+
+ resp, err := router.Search3(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp).ToNot(BeNil())
+ Expect(resp.SearchResult3).ToNot(BeNil())
+
+ // Verify that library filter was applied to all repositories with all accessible libraries
+ assertQueryOptions(mockAlbumRepo.Options.Filters, "library_id IN (?,?,?)", 1, 2, 3)
+ assertQueryOptions(mockArtistRepo.Options.Filters, "library_id IN (?,?,?)", 1, 2, 3)
+ assertQueryOptions(mockMediaFileRepo.Options.Filters, "library_id IN (?,?,?)", 1, 2, 3)
+ })
+
+ It("should return empty results when user has no accessible libraries", func() {
+ r := newGetRequest("query=test")
+ ctx := request.WithUser(r.Context(), model.User{
+ ID: "user1",
+ UserName: "testuser",
+ Libraries: []model.Library{}, // No libraries
+ })
+ r = r.WithContext(ctx)
+
+ resp, err := router.Search3(r)
+
+ Expect(err).ToNot(HaveOccurred())
+ Expect(resp).ToNot(BeNil())
+ Expect(resp.SearchResult3).ToNot(BeNil())
+ Expect(mockAlbumRepo.Options.Filters).To(BeNil())
+ Expect(mockArtistRepo.Options.Filters).To(BeNil())
+ Expect(mockMediaFileRepo.Options.Filters).To(BeNil())
+ })
+
+ It("should return error for inaccessible musicFolderId", func() {
+ // Test that the endpoint returns an error when user tries to access a library they don't have access to
+ r := newGetRequest("query=test", "musicFolderId=999")
+ ctx := request.WithUser(r.Context(), model.User{
+ ID: "user1",
+ UserName: "testuser",
+ Libraries: []model.Library{{ID: 1, Name: "Library 1"}},
+ })
+ r = r.WithContext(ctx)
+
+ resp, err := router.Search3(r)
+
+ Expect(err).To(HaveOccurred())
+ Expect(err.Error()).To(ContainSubstring("Library 999 not found or not accessible"))
+ Expect(resp).To(BeNil())
+ })
+ })
+ })
+})
diff --git a/server/subsonic/users.go b/server/subsonic/users.go
index 0499b5ee0..733f3fddb 100644
--- a/server/subsonic/users.go
+++ b/server/subsonic/users.go
@@ -4,26 +4,41 @@ import (
"net/http"
"github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/model/request"
"github.com/navidrome/navidrome/server/subsonic/responses"
+ "github.com/navidrome/navidrome/utils/slice"
)
-// TODO This is a placeholder. The real one has to read this info from a config file or the database
+// buildUserResponse creates a User response object from a User model
+func buildUserResponse(user model.User) responses.User {
+ userResponse := responses.User{
+ Username: user.UserName,
+ AdminRole: user.IsAdmin,
+ Email: user.Email,
+ StreamRole: true,
+ ScrobblingEnabled: true,
+ DownloadRole: conf.Server.EnableDownloads,
+ ShareRole: conf.Server.EnableSharing,
+ Folder: slice.Map(user.Libraries, func(lib model.Library) int32 { return int32(lib.ID) }),
+ }
+
+ if conf.Server.Jukebox.Enabled {
+ userResponse.JukeboxRole = !conf.Server.Jukebox.AdminOnly || user.IsAdmin
+ }
+
+ return userResponse
+}
+
func (api *Router) GetUser(r *http.Request) (*responses.Subsonic, error) {
loggedUser, ok := request.UserFrom(r.Context())
if !ok {
return nil, newError(responses.ErrorGeneric, "Internal error")
}
+
response := newResponse()
- response.User = &responses.User{}
- response.User.Username = loggedUser.UserName
- response.User.AdminRole = loggedUser.IsAdmin
- response.User.Email = loggedUser.Email
- response.User.StreamRole = true
- response.User.ScrobblingEnabled = true
- response.User.DownloadRole = conf.Server.EnableDownloads
- response.User.ShareRole = conf.Server.EnableSharing
- response.User.JukeboxRole = conf.Server.Jukebox.Enabled
+ user := buildUserResponse(loggedUser)
+ response.User = &user
return response, nil
}
@@ -32,17 +47,8 @@ func (api *Router) GetUsers(r *http.Request) (*responses.Subsonic, error) {
if !ok {
return nil, newError(responses.ErrorGeneric, "Internal error")
}
- user := responses.User{}
- user.Username = loggedUser.Name
- user.AdminRole = loggedUser.IsAdmin
- user.Email = loggedUser.Email
- user.StreamRole = true
- user.ScrobblingEnabled = true
- user.DownloadRole = conf.Server.EnableDownloads
- user.ShareRole = conf.Server.EnableSharing
- if conf.Server.Jukebox.Enabled {
- user.JukeboxRole = !conf.Server.Jukebox.AdminOnly || loggedUser.IsAdmin
- }
+
+ user := buildUserResponse(loggedUser)
response := newResponse()
response.Users = &responses.Users{User: []responses.User{user}}
return response, nil
diff --git a/server/subsonic/users_test.go b/server/subsonic/users_test.go
new file mode 100644
index 000000000..e41c1af63
--- /dev/null
+++ b/server/subsonic/users_test.go
@@ -0,0 +1,119 @@
+package subsonic
+
+import (
+ "context"
+ "net/http/httptest"
+
+ "github.com/navidrome/navidrome/conf"
+ "github.com/navidrome/navidrome/conf/configtest"
+ "github.com/navidrome/navidrome/model"
+ "github.com/navidrome/navidrome/model/request"
+ "github.com/navidrome/navidrome/server/subsonic/responses"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("Users", func() {
+ var router *Router
+ var testUser model.User
+
+ BeforeEach(func() {
+ DeferCleanup(configtest.SetupConfig())
+ router = &Router{}
+
+ testUser = model.User{
+ ID: "user123",
+ UserName: "testuser",
+ Name: "Test User",
+ Email: "test@example.com",
+ IsAdmin: false,
+ }
+ })
+
+ Describe("Happy path", func() {
+ It("should return consistent user data in both GetUser and GetUsers", func() {
+ conf.Server.EnableDownloads = true
+ conf.Server.EnableSharing = true
+ conf.Server.Jukebox.Enabled = false
+
+ // Set up user with libraries
+ testUser.Libraries = model.Libraries{
+ {ID: 10, Name: "Music"},
+ {ID: 20, Name: "Podcasts"},
+ }
+
+ // Create request with user in context
+ req := httptest.NewRequest("GET", "/rest/getUser", nil)
+ ctx := request.WithUser(context.Background(), testUser)
+ req = req.WithContext(ctx)
+
+ userResponse, err1 := router.GetUser(req)
+ usersResponse, err2 := router.GetUsers(req)
+
+ Expect(err1).ToNot(HaveOccurred())
+ Expect(err2).ToNot(HaveOccurred())
+
+ // Verify GetUser response structure
+ Expect(userResponse.Status).To(Equal(responses.StatusOK))
+ Expect(userResponse.User).ToNot(BeNil())
+ Expect(userResponse.User.Username).To(Equal("testuser"))
+ Expect(userResponse.User.Email).To(Equal("test@example.com"))
+ Expect(userResponse.User.AdminRole).To(BeFalse())
+ Expect(userResponse.User.StreamRole).To(BeTrue())
+ Expect(userResponse.User.ScrobblingEnabled).To(BeTrue())
+ Expect(userResponse.User.DownloadRole).To(BeTrue())
+ Expect(userResponse.User.ShareRole).To(BeTrue())
+ Expect(userResponse.User.Folder).To(ContainElements(int32(10), int32(20)))
+
+ // Verify GetUsers response structure
+ Expect(usersResponse.Status).To(Equal(responses.StatusOK))
+ Expect(usersResponse.Users).ToNot(BeNil())
+ Expect(usersResponse.Users.User).To(HaveLen(1))
+
+ // Verify both methods return identical user data
+ singleUser := userResponse.User
+ userFromList := &usersResponse.Users.User[0]
+
+ Expect(singleUser.Username).To(Equal(userFromList.Username))
+ Expect(singleUser.Email).To(Equal(userFromList.Email))
+ Expect(singleUser.AdminRole).To(Equal(userFromList.AdminRole))
+ Expect(singleUser.StreamRole).To(Equal(userFromList.StreamRole))
+ Expect(singleUser.ScrobblingEnabled).To(Equal(userFromList.ScrobblingEnabled))
+ Expect(singleUser.DownloadRole).To(Equal(userFromList.DownloadRole))
+ Expect(singleUser.ShareRole).To(Equal(userFromList.ShareRole))
+ Expect(singleUser.JukeboxRole).To(Equal(userFromList.JukeboxRole))
+ Expect(singleUser.Folder).To(Equal(userFromList.Folder))
+ })
+ })
+
+ DescribeTable("Jukebox role permissions",
+ func(jukeboxEnabled, adminOnly, isAdmin, expectedJukeboxRole bool) {
+ conf.Server.Jukebox.Enabled = jukeboxEnabled
+ conf.Server.Jukebox.AdminOnly = adminOnly
+ testUser.IsAdmin = isAdmin
+
+ response := buildUserResponse(testUser)
+ Expect(response.JukeboxRole).To(Equal(expectedJukeboxRole))
+ },
+ Entry("jukebox disabled", false, false, false, false),
+ Entry("jukebox enabled, not admin-only, regular user", true, false, false, true),
+ Entry("jukebox enabled, not admin-only, admin user", true, false, true, true),
+ Entry("jukebox enabled, admin-only, regular user", true, true, false, false),
+ Entry("jukebox enabled, admin-only, admin user", true, true, true, true),
+ )
+
+ Describe("Folder list population", func() {
+ It("should populate Folder field with user's accessible library IDs", func() {
+ testUser.Libraries = model.Libraries{
+ {ID: 1, Name: "Music"},
+ {ID: 2, Name: "Podcasts"},
+ {ID: 5, Name: "Audiobooks"},
+ }
+
+ response := buildUserResponse(testUser)
+
+ Expect(response.Folder).To(HaveLen(3))
+ Expect(response.Folder).To(ContainElements(int32(1), int32(2), int32(5)))
+ })
+ })
+})
diff --git a/tests/fixtures/bom-test.lrc b/tests/fixtures/bom-test.lrc
new file mode 100644
index 000000000..223c37de0
--- /dev/null
+++ b/tests/fixtures/bom-test.lrc
@@ -0,0 +1,4 @@
+[00:00.00] 作曲 : 柏大輔
+NOTE: This file intentionally contains a UTF-8 BOM (Byte Order Mark) at byte 0.
+This tests BOM handling in lyrics parsing (GitHub issue #4631).
+The BOM bytes are: 0xEF 0xBB 0xBF
\ No newline at end of file
diff --git a/tests/fixtures/bom-utf16-test.lrc b/tests/fixtures/bom-utf16-test.lrc
new file mode 100644
index 000000000..e40ea3255
Binary files /dev/null and b/tests/fixtures/bom-utf16-test.lrc differ
diff --git a/tests/fixtures/deezer.artist.bio.json b/tests/fixtures/deezer.artist.bio.json
new file mode 100644
index 000000000..80e439bae
--- /dev/null
+++ b/tests/fixtures/deezer.artist.bio.json
@@ -0,0 +1,9 @@
+{
+ "data": {
+ "artist": {
+ "bio": {
+ "full": "Schoolmates Thomas and Guy-Manuel began their career in 1992 with the indie rock trio Darlin' (named after The Beach Boys song) but were scathingly dismissed by Melody Maker magazine as \"daft punk.\" Turning to house-inspired electronica, they used the put down as a name for their DJ-ing partnership and became a hugely successful and influential dance act.
"
+ }
+ }
+ }
+}
diff --git a/tests/fixtures/deezer.artist.related.json b/tests/fixtures/deezer.artist.related.json
new file mode 100644
index 000000000..2a55b303e
--- /dev/null
+++ b/tests/fixtures/deezer.artist.related.json
@@ -0,0 +1 @@
+{"data":[{"id":6404,"name":"Justice","link":"https:\/\/www.deezer.com\/artist\/6404","picture":"https:\/\/api.deezer.com\/artist\/6404\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/e5bf29cb99852f92a0079b184ace9479\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/e5bf29cb99852f92a0079b184ace9479\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/e5bf29cb99852f92a0079b184ace9479\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/e5bf29cb99852f92a0079b184ace9479\/1000x1000-000000-80-0-0.jpg","nb_album":41,"nb_fan":774236,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/6404\/top?limit=50","type":"artist"},{"id":2049,"name":"Cassius","link":"https:\/\/www.deezer.com\/artist\/2049","picture":"https:\/\/api.deezer.com\/artist\/2049\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/7ed91fa11a9785a82e63fd9058821d8a\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/7ed91fa11a9785a82e63fd9058821d8a\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/7ed91fa11a9785a82e63fd9058821d8a\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/7ed91fa11a9785a82e63fd9058821d8a\/1000x1000-000000-80-0-0.jpg","nb_album":25,"nb_fan":127692,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/2049\/top?limit=50","type":"artist"},{"id":2318,"name":"Etienne de Cr\u00e9cy","link":"https:\/\/www.deezer.com\/artist\/2318","picture":"https:\/\/api.deezer.com\/artist\/2318\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/b9efa1b51be3c2a506006a77517967f7\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/b9efa1b51be3c2a506006a77517967f7\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/b9efa1b51be3c2a506006a77517967f7\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/b9efa1b51be3c2a506006a77517967f7\/1000x1000-000000-80-0-0.jpg","nb_album":58,"nb_fan":104626,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/2318\/top?limit=50","type":"artist"},{"id":72041,"name":"Yuksek","link":"https:\/\/www.deezer.com\/artist\/72041","picture":"https:\/\/api.deezer.com\/artist\/72041\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/5e0fd4c6b670682861abcc825eb3db50\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/5e0fd4c6b670682861abcc825eb3db50\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/5e0fd4c6b670682861abcc825eb3db50\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/5e0fd4c6b670682861abcc825eb3db50\/1000x1000-000000-80-0-0.jpg","nb_album":102,"nb_fan":115772,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/72041\/top?limit=50","type":"artist"},{"id":81,"name":"The Chemical Brothers","link":"https:\/\/www.deezer.com\/artist\/81","picture":"https:\/\/api.deezer.com\/artist\/81\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/5294e68e9ef4f0359237935a4b8388f2\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/5294e68e9ef4f0359237935a4b8388f2\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/5294e68e9ef4f0359237935a4b8388f2\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/5294e68e9ef4f0359237935a4b8388f2\/1000x1000-000000-80-0-0.jpg","nb_album":83,"nb_fan":1433333,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/81\/top?limit=50","type":"artist"},{"id":3771,"name":"Mr. Oizo","link":"https:\/\/www.deezer.com\/artist\/3771","picture":"https:\/\/api.deezer.com\/artist\/3771\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/589305cb56ea3555b1f94341955bf875\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/589305cb56ea3555b1f94341955bf875\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/589305cb56ea3555b1f94341955bf875\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/589305cb56ea3555b1f94341955bf875\/1000x1000-000000-80-0-0.jpg","nb_album":31,"nb_fan":172085,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/3771\/top?limit=50","type":"artist"},{"id":9905,"name":"Alex Gopher","link":"https:\/\/www.deezer.com\/artist\/9905","picture":"https:\/\/api.deezer.com\/artist\/9905\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c4676bdbf3259decd69491523a1ace8f\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c4676bdbf3259decd69491523a1ace8f\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c4676bdbf3259decd69491523a1ace8f\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c4676bdbf3259decd69491523a1ace8f\/1000x1000-000000-80-0-0.jpg","nb_album":46,"nb_fan":10430,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/9905\/top?limit=50","type":"artist"},{"id":7914,"name":"Demon","link":"https:\/\/www.deezer.com\/artist\/7914","picture":"https:\/\/api.deezer.com\/artist\/7914\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c7c1f4d1f1cf6bdf04a6fc54ea65ef78\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c7c1f4d1f1cf6bdf04a6fc54ea65ef78\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c7c1f4d1f1cf6bdf04a6fc54ea65ef78\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c7c1f4d1f1cf6bdf04a6fc54ea65ef78\/1000x1000-000000-80-0-0.jpg","nb_album":21,"nb_fan":9286,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/7914\/top?limit=50","type":"artist"},{"id":8937,"name":"SebastiAn","link":"https:\/\/www.deezer.com\/artist\/8937","picture":"https:\/\/api.deezer.com\/artist\/8937\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0151acdea8a8d5f8e3aefabf6f034575\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0151acdea8a8d5f8e3aefabf6f034575\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0151acdea8a8d5f8e3aefabf6f034575\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0151acdea8a8d5f8e3aefabf6f034575\/1000x1000-000000-80-0-0.jpg","nb_album":48,"nb_fan":74884,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/8937\/top?limit=50","type":"artist"},{"id":2508,"name":"Digitalism","link":"https:\/\/www.deezer.com\/artist\/2508","picture":"https:\/\/api.deezer.com\/artist\/2508\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/21e56c7147508853dcdfd9997cd8d271\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/21e56c7147508853dcdfd9997cd8d271\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/21e56c7147508853dcdfd9997cd8d271\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/21e56c7147508853dcdfd9997cd8d271\/1000x1000-000000-80-0-0.jpg","nb_album":79,"nb_fan":158628,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/2508\/top?limit=50","type":"artist"},{"id":11703,"name":"Alan Braxe","link":"https:\/\/www.deezer.com\/artist\/11703","picture":"https:\/\/api.deezer.com\/artist\/11703\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0de64f7a63728f09520f987249630d7e\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0de64f7a63728f09520f987249630d7e\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0de64f7a63728f09520f987249630d7e\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0de64f7a63728f09520f987249630d7e\/1000x1000-000000-80-0-0.jpg","nb_album":25,"nb_fan":12595,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/11703\/top?limit=50","type":"artist"},{"id":574,"name":"Para One","link":"https:\/\/www.deezer.com\/artist\/574","picture":"https:\/\/api.deezer.com\/artist\/574\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/2560ff01d72f5e4a768e55892ad066e4\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/2560ff01d72f5e4a768e55892ad066e4\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/2560ff01d72f5e4a768e55892ad066e4\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/2560ff01d72f5e4a768e55892ad066e4\/1000x1000-000000-80-0-0.jpg","nb_album":40,"nb_fan":30828,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/574\/top?limit=50","type":"artist"},{"id":4397,"name":"Kojak","link":"https:\/\/www.deezer.com\/artist\/4397","picture":"https:\/\/api.deezer.com\/artist\/4397\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/817caeb7fa22eb511371ac260680d5fa\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/817caeb7fa22eb511371ac260680d5fa\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/817caeb7fa22eb511371ac260680d5fa\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/817caeb7fa22eb511371ac260680d5fa\/1000x1000-000000-80-0-0.jpg","nb_album":55,"nb_fan":1522,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/4397\/top?limit=50","type":"artist"},{"id":12439,"name":"Busy P","link":"https:\/\/www.deezer.com\/artist\/12439","picture":"https:\/\/api.deezer.com\/artist\/12439\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/6483408c3e46e8fa8872a805dfe6d5e0\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/6483408c3e46e8fa8872a805dfe6d5e0\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/6483408c3e46e8fa8872a805dfe6d5e0\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/6483408c3e46e8fa8872a805dfe6d5e0\/1000x1000-000000-80-0-0.jpg","nb_album":12,"nb_fan":65585,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/12439\/top?limit=50","type":"artist"},{"id":11656979,"name":"Mr Flash","link":"https:\/\/www.deezer.com\/artist\/11656979","picture":"https:\/\/api.deezer.com\/artist\/11656979\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/f953922c8caa74c5b48feaa07622b1ee\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/f953922c8caa74c5b48feaa07622b1ee\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/f953922c8caa74c5b48feaa07622b1ee\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/f953922c8caa74c5b48feaa07622b1ee\/1000x1000-000000-80-0-0.jpg","nb_album":7,"nb_fan":769,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/11656979\/top?limit=50","type":"artist"},{"id":76,"name":"Fatboy Slim","link":"https:\/\/www.deezer.com\/artist\/76","picture":"https:\/\/api.deezer.com\/artist\/76\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/f6ea7bd64ec1902feff17935fdfea263\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/f6ea7bd64ec1902feff17935fdfea263\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/f6ea7bd64ec1902feff17935fdfea263\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/f6ea7bd64ec1902feff17935fdfea263\/1000x1000-000000-80-0-0.jpg","nb_album":76,"nb_fan":1231355,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/76\/top?limit=50","type":"artist"},{"id":11265,"name":"Lifelike","link":"https:\/\/www.deezer.com\/artist\/11265","picture":"https:\/\/api.deezer.com\/artist\/11265\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/04f88199a69a646f6253808b58753629\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/04f88199a69a646f6253808b58753629\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/04f88199a69a646f6253808b58753629\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/04f88199a69a646f6253808b58753629\/1000x1000-000000-80-0-0.jpg","nb_album":38,"nb_fan":8316,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/11265\/top?limit=50","type":"artist"},{"id":2048,"name":"Groove Armada","link":"https:\/\/www.deezer.com\/artist\/2048","picture":"https:\/\/api.deezer.com\/artist\/2048\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0acbb71c44e4ecdefd102630f4cfc808\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0acbb71c44e4ecdefd102630f4cfc808\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0acbb71c44e4ecdefd102630f4cfc808\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0acbb71c44e4ecdefd102630f4cfc808\/1000x1000-000000-80-0-0.jpg","nb_album":92,"nb_fan":173879,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/2048\/top?limit=50","type":"artist"},{"id":71708,"name":"Surkin","link":"https:\/\/www.deezer.com\/artist\/71708","picture":"https:\/\/api.deezer.com\/artist\/71708\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/d2137c57fbdc9aa275b93e78b619b477\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/d2137c57fbdc9aa275b93e78b619b477\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/d2137c57fbdc9aa275b93e78b619b477\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/d2137c57fbdc9aa275b93e78b619b477\/1000x1000-000000-80-0-0.jpg","nb_album":15,"nb_fan":23101,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/71708\/top?limit=50","type":"artist"},{"id":166713,"name":"Fred Falke","link":"https:\/\/www.deezer.com\/artist\/166713","picture":"https:\/\/api.deezer.com\/artist\/166713\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/883821e7c5325cfa07fc660884a40624\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/883821e7c5325cfa07fc660884a40624\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/883821e7c5325cfa07fc660884a40624\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/883821e7c5325cfa07fc660884a40624\/1000x1000-000000-80-0-0.jpg","nb_album":67,"nb_fan":9688,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/166713\/top?limit=50","type":"artist"}],"total":20}
\ No newline at end of file
diff --git a/tests/fixtures/deezer.artist.top.json b/tests/fixtures/deezer.artist.top.json
new file mode 100644
index 000000000..e3f22a1aa
--- /dev/null
+++ b/tests/fixtures/deezer.artist.top.json
@@ -0,0 +1 @@
+{"data":[{"id":67238732,"readable":true,"title":"Instant Crush (feat. Julian Casablancas)","title_short":"Instant Crush","title_version":"(feat. Julian Casablancas)","link":"https:\/\/www.deezer.com\/track\/67238732","duration":337,"rank":944042,"explicit_lyrics":false,"explicit_content_lyrics":0,"explicit_content_cover":0,"preview":"https:\/\/cdnt-preview.dzcdn.net\/api\/1\/1\/d\/6\/b\/0\/d6bc80aadfa1d7625d59a6620f229371.mp3?hdnea=exp=1763672105~acl=\/api\/1\/1\/d\/6\/b\/0\/d6bc80aadfa1d7625d59a6620f229371.mp3*~data=user_id=0,application_id=42~hmac=66213cecf953c7ef8b4d89e3539a1355d318679c5ab54cac2007d4effa6c3bf4","contributors":[{"id":27,"name":"Daft Punk","link":"https:\/\/www.deezer.com\/artist\/27","share":"https:\/\/www.deezer.com\/artist\/27?utm_source=deezer&utm_content=artist-27&utm_term=0_1763671205&utm_medium=web","picture":"https:\/\/api.deezer.com\/artist\/27\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/1000x1000-000000-80-0-0.jpg","radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/27\/top?limit=50","type":"artist","role":"Main"},{"id":295821,"name":"Julian Casablancas","link":"https:\/\/www.deezer.com\/artist\/295821","share":"https:\/\/www.deezer.com\/artist\/295821?utm_source=deezer&utm_content=artist-295821&utm_term=0_1763671205&utm_medium=web","picture":"https:\/\/api.deezer.com\/artist\/295821\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/74e78538aefe2a6a49a851e569fc9f19\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/74e78538aefe2a6a49a851e569fc9f19\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/74e78538aefe2a6a49a851e569fc9f19\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/74e78538aefe2a6a49a851e569fc9f19\/1000x1000-000000-80-0-0.jpg","radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/295821\/top?limit=50","type":"artist","role":"Main"}],"md5_image":"311bba0fc112d15f72c8b5a65f0456c1","artist":{"id":27,"name":"Daft Punk","tracklist":"https:\/\/api.deezer.com\/artist\/27\/top?limit=50","type":"artist"},"album":{"id":6575789,"title":"Random Access Memories","cover":"https:\/\/api.deezer.com\/album\/6575789\/image","cover_small":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/311bba0fc112d15f72c8b5a65f0456c1\/56x56-000000-80-0-0.jpg","cover_medium":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/311bba0fc112d15f72c8b5a65f0456c1\/250x250-000000-80-0-0.jpg","cover_big":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/311bba0fc112d15f72c8b5a65f0456c1\/500x500-000000-80-0-0.jpg","cover_xl":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/311bba0fc112d15f72c8b5a65f0456c1\/1000x1000-000000-80-0-0.jpg","md5_image":"311bba0fc112d15f72c8b5a65f0456c1","tracklist":"https:\/\/api.deezer.com\/album\/6575789\/tracks","type":"album"},"type":"track"},{"id":3135553,"readable":true,"title":"One More Time","title_short":"One More Time","title_version":"","link":"https:\/\/www.deezer.com\/track\/3135553","duration":320,"rank":888570,"explicit_lyrics":false,"explicit_content_lyrics":0,"explicit_content_cover":0,"preview":"https:\/\/cdnt-preview.dzcdn.net\/api\/1\/1\/f\/8\/c\/0\/f8c5dc3837912dba37c9a1ab3170cc3f.mp3?hdnea=exp=1763672105~acl=\/api\/1\/1\/f\/8\/c\/0\/f8c5dc3837912dba37c9a1ab3170cc3f.mp3*~data=user_id=0,application_id=42~hmac=0824ec7ad045b82c04904fcd5f2a8ec2175acbe3d1649030d457023fdef45620","contributors":[{"id":27,"name":"Daft Punk","link":"https:\/\/www.deezer.com\/artist\/27","share":"https:\/\/www.deezer.com\/artist\/27?utm_source=deezer&utm_content=artist-27&utm_term=0_1763671205&utm_medium=web","picture":"https:\/\/api.deezer.com\/artist\/27\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/1000x1000-000000-80-0-0.jpg","radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/27\/top?limit=50","type":"artist","role":"Main"}],"md5_image":"5718f7c81c27e0b2417e2a4c45224f8a","artist":{"id":27,"name":"Daft Punk","tracklist":"https:\/\/api.deezer.com\/artist\/27\/top?limit=50","type":"artist"},"album":{"id":302127,"title":"Discovery","cover":"https:\/\/api.deezer.com\/album\/302127\/image","cover_small":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/5718f7c81c27e0b2417e2a4c45224f8a\/56x56-000000-80-0-0.jpg","cover_medium":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/5718f7c81c27e0b2417e2a4c45224f8a\/250x250-000000-80-0-0.jpg","cover_big":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/5718f7c81c27e0b2417e2a4c45224f8a\/500x500-000000-80-0-0.jpg","cover_xl":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/5718f7c81c27e0b2417e2a4c45224f8a\/1000x1000-000000-80-0-0.jpg","md5_image":"5718f7c81c27e0b2417e2a4c45224f8a","tracklist":"https:\/\/api.deezer.com\/album\/302127\/tracks","type":"album"},"type":"track"},{"id":66609426,"readable":true,"title":"Get Lucky (Radio Edit - feat. Pharrell Williams and Nile Rodgers)","title_short":"Get Lucky","title_version":"(Radio Edit - feat. Pharrell Williams and Nile Rodgers)","link":"https:\/\/www.deezer.com\/track\/66609426","duration":248,"rank":952197,"explicit_lyrics":false,"explicit_content_lyrics":0,"explicit_content_cover":0,"preview":"https:\/\/cdnt-preview.dzcdn.net\/api\/1\/1\/1\/b\/f\/0\/1bf80a82992903ff685ba1b7275223f8.mp3?hdnea=exp=1763672105~acl=\/api\/1\/1\/1\/b\/f\/0\/1bf80a82992903ff685ba1b7275223f8.mp3*~data=user_id=0,application_id=42~hmac=c6dfe58571df62f41e7b326dd9afebf87015541c06a521ebc88fc18671d8d06d","contributors":[{"id":27,"name":"Daft Punk","link":"https:\/\/www.deezer.com\/artist\/27","share":"https:\/\/www.deezer.com\/artist\/27?utm_source=deezer&utm_content=artist-27&utm_term=0_1763671205&utm_medium=web","picture":"https:\/\/api.deezer.com\/artist\/27\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/1000x1000-000000-80-0-0.jpg","radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/27\/top?limit=50","type":"artist","role":"Main"},{"id":103,"name":"Pharrell Williams","link":"https:\/\/www.deezer.com\/artist\/103","share":"https:\/\/www.deezer.com\/artist\/103?utm_source=deezer&utm_content=artist-103&utm_term=0_1763671205&utm_medium=web","picture":"https:\/\/api.deezer.com\/artist\/103\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/1267b8781c5bff065a20dca4a3c9fda7\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/1267b8781c5bff065a20dca4a3c9fda7\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/1267b8781c5bff065a20dca4a3c9fda7\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/1267b8781c5bff065a20dca4a3c9fda7\/1000x1000-000000-80-0-0.jpg","radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/103\/top?limit=50","type":"artist","role":"Main"},{"id":7207,"name":"Nile Rodgers","link":"https:\/\/www.deezer.com\/artist\/7207","share":"https:\/\/www.deezer.com\/artist\/7207?utm_source=deezer&utm_content=artist-7207&utm_term=0_1763671205&utm_medium=web","picture":"https:\/\/api.deezer.com\/artist\/7207\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/64f826f318c84ce50ff538c01f62f1ff\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/64f826f318c84ce50ff538c01f62f1ff\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/64f826f318c84ce50ff538c01f62f1ff\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/64f826f318c84ce50ff538c01f62f1ff\/1000x1000-000000-80-0-0.jpg","radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/7207\/top?limit=50","type":"artist","role":"Main"}],"md5_image":"bc49adb87758e0c8c4e508a9c5cce85d","artist":{"id":27,"name":"Daft Punk","tracklist":"https:\/\/api.deezer.com\/artist\/27\/top?limit=50","type":"artist"},"album":{"id":6516139,"title":"Get Lucky (Radio Edit - feat. Pharrell Williams and Nile Rodgers)","cover":"https:\/\/api.deezer.com\/album\/6516139\/image","cover_small":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/bc49adb87758e0c8c4e508a9c5cce85d\/56x56-000000-80-0-0.jpg","cover_medium":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/bc49adb87758e0c8c4e508a9c5cce85d\/250x250-000000-80-0-0.jpg","cover_big":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/bc49adb87758e0c8c4e508a9c5cce85d\/500x500-000000-80-0-0.jpg","cover_xl":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/bc49adb87758e0c8c4e508a9c5cce85d\/1000x1000-000000-80-0-0.jpg","md5_image":"bc49adb87758e0c8c4e508a9c5cce85d","tracklist":"https:\/\/api.deezer.com\/album\/6516139\/tracks","type":"album"},"type":"track"},{"id":67238735,"readable":true,"title":"Get Lucky (feat. Pharrell Williams and Nile Rodgers)","title_short":"Get Lucky","title_version":"(feat. Pharrell Williams and Nile Rodgers)","link":"https:\/\/www.deezer.com\/track\/67238735","duration":367,"rank":873875,"explicit_lyrics":false,"explicit_content_lyrics":0,"explicit_content_cover":0,"preview":"https:\/\/cdnt-preview.dzcdn.net\/api\/1\/1\/c\/8\/a\/0\/c8a61130657a2cf58e3ac751e7950617.mp3?hdnea=exp=1763672105~acl=\/api\/1\/1\/c\/8\/a\/0\/c8a61130657a2cf58e3ac751e7950617.mp3*~data=user_id=0,application_id=42~hmac=92002e6bade5ff82dd44751e8998beaa60844210df1d73b8f1bf7dafb02dc5c3","contributors":[{"id":27,"name":"Daft Punk","link":"https:\/\/www.deezer.com\/artist\/27","share":"https:\/\/www.deezer.com\/artist\/27?utm_source=deezer&utm_content=artist-27&utm_term=0_1763671205&utm_medium=web","picture":"https:\/\/api.deezer.com\/artist\/27\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/1000x1000-000000-80-0-0.jpg","radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/27\/top?limit=50","type":"artist","role":"Main"},{"id":103,"name":"Pharrell Williams","link":"https:\/\/www.deezer.com\/artist\/103","share":"https:\/\/www.deezer.com\/artist\/103?utm_source=deezer&utm_content=artist-103&utm_term=0_1763671205&utm_medium=web","picture":"https:\/\/api.deezer.com\/artist\/103\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/1267b8781c5bff065a20dca4a3c9fda7\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/1267b8781c5bff065a20dca4a3c9fda7\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/1267b8781c5bff065a20dca4a3c9fda7\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/1267b8781c5bff065a20dca4a3c9fda7\/1000x1000-000000-80-0-0.jpg","radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/103\/top?limit=50","type":"artist","role":"Main"},{"id":7207,"name":"Nile Rodgers","link":"https:\/\/www.deezer.com\/artist\/7207","share":"https:\/\/www.deezer.com\/artist\/7207?utm_source=deezer&utm_content=artist-7207&utm_term=0_1763671205&utm_medium=web","picture":"https:\/\/api.deezer.com\/artist\/7207\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/64f826f318c84ce50ff538c01f62f1ff\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/64f826f318c84ce50ff538c01f62f1ff\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/64f826f318c84ce50ff538c01f62f1ff\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/64f826f318c84ce50ff538c01f62f1ff\/1000x1000-000000-80-0-0.jpg","radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/7207\/top?limit=50","type":"artist","role":"Main"}],"md5_image":"311bba0fc112d15f72c8b5a65f0456c1","artist":{"id":27,"name":"Daft Punk","tracklist":"https:\/\/api.deezer.com\/artist\/27\/top?limit=50","type":"artist"},"album":{"id":6575789,"title":"Random Access Memories","cover":"https:\/\/api.deezer.com\/album\/6575789\/image","cover_small":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/311bba0fc112d15f72c8b5a65f0456c1\/56x56-000000-80-0-0.jpg","cover_medium":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/311bba0fc112d15f72c8b5a65f0456c1\/250x250-000000-80-0-0.jpg","cover_big":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/311bba0fc112d15f72c8b5a65f0456c1\/500x500-000000-80-0-0.jpg","cover_xl":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/311bba0fc112d15f72c8b5a65f0456c1\/1000x1000-000000-80-0-0.jpg","md5_image":"311bba0fc112d15f72c8b5a65f0456c1","tracklist":"https:\/\/api.deezer.com\/album\/6575789\/tracks","type":"album"},"type":"track"},{"id":3129775,"readable":true,"title":"Around the World","title_short":"Around the World","title_version":"","link":"https:\/\/www.deezer.com\/track\/3129775","duration":429,"rank":829911,"explicit_lyrics":false,"explicit_content_lyrics":0,"explicit_content_cover":0,"preview":"https:\/\/cdnt-preview.dzcdn.net\/api\/1\/1\/a\/4\/7\/0\/a47dbed01e6d9b0ac4e39a134f745ca2.mp3?hdnea=exp=1763672105~acl=\/api\/1\/1\/a\/4\/7\/0\/a47dbed01e6d9b0ac4e39a134f745ca2.mp3*~data=user_id=0,application_id=42~hmac=9b7aa12b647cabd3219779e0270e51e639dc326442071fceb6d723c331059a67","contributors":[{"id":27,"name":"Daft Punk","link":"https:\/\/www.deezer.com\/artist\/27","share":"https:\/\/www.deezer.com\/artist\/27?utm_source=deezer&utm_content=artist-27&utm_term=0_1763671205&utm_medium=web","picture":"https:\/\/api.deezer.com\/artist\/27\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/638e69b9caaf9f9f3f8826febea7b543\/1000x1000-000000-80-0-0.jpg","radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/27\/top?limit=50","type":"artist","role":"Main"}],"md5_image":"b870579c8650cd59b1cce656dde2ef17","artist":{"id":27,"name":"Daft Punk","tracklist":"https:\/\/api.deezer.com\/artist\/27\/top?limit=50","type":"artist"},"album":{"id":301775,"title":"Homework","cover":"https:\/\/api.deezer.com\/album\/301775\/image","cover_small":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/b870579c8650cd59b1cce656dde2ef17\/56x56-000000-80-0-0.jpg","cover_medium":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/b870579c8650cd59b1cce656dde2ef17\/250x250-000000-80-0-0.jpg","cover_big":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/b870579c8650cd59b1cce656dde2ef17\/500x500-000000-80-0-0.jpg","cover_xl":"https:\/\/cdn-images.dzcdn.net\/images\/cover\/b870579c8650cd59b1cce656dde2ef17\/1000x1000-000000-80-0-0.jpg","md5_image":"b870579c8650cd59b1cce656dde2ef17","tracklist":"https:\/\/api.deezer.com\/album\/301775\/tracks","type":"album"},"type":"track"}],"total":100,"next":"https:\/\/api.deezer.com\/artist\/27\/top?index=5"}
\ No newline at end of file
diff --git a/tests/fixtures/deezer.search.artist.json b/tests/fixtures/deezer.search.artist.json
new file mode 100644
index 000000000..29f138d34
--- /dev/null
+++ b/tests/fixtures/deezer.search.artist.json
@@ -0,0 +1 @@
+{"data":[{"id":259,"name":"Michael Jackson","link":"https:\/\/www.deezer.com\/artist\/259","picture":"https:\/\/api.deezer.com\/artist\/259\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/97fae13b2b30e4aec2e8c9e0c7839d92\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/97fae13b2b30e4aec2e8c9e0c7839d92\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/97fae13b2b30e4aec2e8c9e0c7839d92\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/97fae13b2b30e4aec2e8c9e0c7839d92\/1000x1000-000000-80-0-0.jpg","nb_album":43,"nb_fan":12074101,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/259\/top?limit=50","type":"artist"},{"id":719,"name":"Bob Marley & The Wailers","link":"https:\/\/www.deezer.com\/artist\/719","picture":"https:\/\/api.deezer.com\/artist\/719\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c8241e15efdefa9465c7b470643efb3b\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c8241e15efdefa9465c7b470643efb3b\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c8241e15efdefa9465c7b470643efb3b\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c8241e15efdefa9465c7b470643efb3b\/1000x1000-000000-80-0-0.jpg","nb_album":80,"nb_fan":12014466,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/719\/top?limit=50","type":"artist"},{"id":14031649,"name":"jay emcee, Micheal Jackson","link":"https:\/\/www.deezer.com\/artist\/14031649","picture":"https:\/\/api.deezer.com\/artist\/14031649\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/1000x1000-000000-80-0-0.jpg","nb_album":0,"nb_fan":104,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/14031649\/top?limit=50","type":"artist"},{"id":137159102,"name":"Micheal Collins The Mic Jackson Of Rap","link":"https:\/\/www.deezer.com\/artist\/137159102","picture":"https:\/\/api.deezer.com\/artist\/137159102\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/1000x1000-000000-80-0-0.jpg","nb_album":0,"nb_fan":13,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/137159102\/top?limit=50","type":"artist"},{"id":259786511,"name":"Consev","link":"https:\/\/www.deezer.com\/artist\/259786511","picture":"https:\/\/api.deezer.com\/artist\/259786511\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/342048482aae9fb1f1272510b1c1f54a\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/342048482aae9fb1f1272510b1c1f54a\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/342048482aae9fb1f1272510b1c1f54a\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/342048482aae9fb1f1272510b1c1f54a\/1000x1000-000000-80-0-0.jpg","nb_album":7,"nb_fan":1,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/259786511\/top?limit=50","type":"artist"},{"id":262255,"name":"Michael Jackson Tribute","link":"https:\/\/www.deezer.com\/artist\/262255","picture":"https:\/\/api.deezer.com\/artist\/262255\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c2f4036de3a412d9b84a213663163189\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c2f4036de3a412d9b84a213663163189\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c2f4036de3a412d9b84a213663163189\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/c2f4036de3a412d9b84a213663163189\/1000x1000-000000-80-0-0.jpg","nb_album":0,"nb_fan":9339,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/262255\/top?limit=50","type":"artist"},{"id":193820797,"name":"Michael Jackman","link":"https:\/\/www.deezer.com\/artist\/193820797","picture":"https:\/\/api.deezer.com\/artist\/193820797\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/da3f7e21c8e78bd2048f47ab60f5399c\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/da3f7e21c8e78bd2048f47ab60f5399c\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/da3f7e21c8e78bd2048f47ab60f5399c\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/da3f7e21c8e78bd2048f47ab60f5399c\/1000x1000-000000-80-0-0.jpg","nb_album":1,"nb_fan":0,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/193820797\/top?limit=50","type":"artist"},{"id":374060,"name":"Simply The Best Sax: The Hits Of Michael Jackson","link":"https:\/\/www.deezer.com\/artist\/374060","picture":"https:\/\/api.deezer.com\/artist\/374060\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/3bfd0455b269dd2ccb25776c6b26197c\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/3bfd0455b269dd2ccb25776c6b26197c\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/3bfd0455b269dd2ccb25776c6b26197c\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/3bfd0455b269dd2ccb25776c6b26197c\/1000x1000-000000-80-0-0.jpg","nb_album":1,"nb_fan":1507,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/374060\/top?limit=50","type":"artist"},{"id":4969823,"name":"Jackson Michael","link":"https:\/\/www.deezer.com\/artist\/4969823","picture":"https:\/\/api.deezer.com\/artist\/4969823\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/e25091f350716af9f4484939de692de6\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/e25091f350716af9f4484939de692de6\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/e25091f350716af9f4484939de692de6\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/e25091f350716af9f4484939de692de6\/1000x1000-000000-80-0-0.jpg","nb_album":1,"nb_fan":17,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/4969823\/top?limit=50","type":"artist"},{"id":1278001,"name":"David Michael Jackson","link":"https:\/\/www.deezer.com\/artist\/1278001","picture":"https:\/\/api.deezer.com\/artist\/1278001\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/fe2e46ba3c550a4747ca5fd9026d6f95\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/fe2e46ba3c550a4747ca5fd9026d6f95\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/fe2e46ba3c550a4747ca5fd9026d6f95\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/fe2e46ba3c550a4747ca5fd9026d6f95\/1000x1000-000000-80-0-0.jpg","nb_album":54,"nb_fan":178,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/1278001\/top?limit=50","type":"artist"},{"id":4142968,"name":"Cheyenne Jackson, Michael Feinstein","link":"https:\/\/www.deezer.com\/artist\/4142968","picture":"https:\/\/api.deezer.com\/artist\/4142968\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/1000x1000-000000-80-0-0.jpg","nb_album":0,"nb_fan":251,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/4142968\/top?limit=50","type":"artist"},{"id":766502,"name":"Michael Jackson Tribute Band","link":"https:\/\/www.deezer.com\/artist\/766502","picture":"https:\/\/api.deezer.com\/artist\/766502\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/82614ae5c3f98c571369f49aba675d1e\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/82614ae5c3f98c571369f49aba675d1e\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/82614ae5c3f98c571369f49aba675d1e\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/82614ae5c3f98c571369f49aba675d1e\/1000x1000-000000-80-0-0.jpg","nb_album":0,"nb_fan":623,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/766502\/top?limit=50","type":"artist"},{"id":1394615,"name":"Michael Jameson","link":"https:\/\/www.deezer.com\/artist\/1394615","picture":"https:\/\/api.deezer.com\/artist\/1394615\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0f680ba29a076b70d5027a68ad3a5c58\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0f680ba29a076b70d5027a68ad3a5c58\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0f680ba29a076b70d5027a68ad3a5c58\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/0f680ba29a076b70d5027a68ad3a5c58\/1000x1000-000000-80-0-0.jpg","nb_album":7,"nb_fan":78,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/1394615\/top?limit=50","type":"artist"},{"id":490836,"name":"Michael Blackson","link":"https:\/\/www.deezer.com\/artist\/490836","picture":"https:\/\/api.deezer.com\/artist\/490836\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/caa27786458e9459819f6ea28c4ed1cb\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/caa27786458e9459819f6ea28c4ed1cb\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/caa27786458e9459819f6ea28c4ed1cb\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/caa27786458e9459819f6ea28c4ed1cb\/1000x1000-000000-80-0-0.jpg","nb_album":1,"nb_fan":391,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/490836\/top?limit=50","type":"artist"},{"id":1229617,"name":"The Michael Jackson Tribute Band","link":"https:\/\/www.deezer.com\/artist\/1229617","picture":"https:\/\/api.deezer.com\/artist\/1229617\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/1000x1000-000000-80-0-0.jpg","nb_album":0,"nb_fan":344,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/1229617\/top?limit=50","type":"artist"},{"id":3662911,"name":"Fran London feat. Michael Jackson","link":"https:\/\/www.deezer.com\/artist\/3662911","picture":"https:\/\/api.deezer.com\/artist\/3662911\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/1000x1000-000000-80-0-0.jpg","nb_album":0,"nb_fan":247,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/3662911\/top?limit=50","type":"artist"},{"id":13014917,"name":"Scott Michael Bennett, Naomi Jackson, Gary Sewell & The Emmanuel Quartet","link":"https:\/\/www.deezer.com\/artist\/13014917","picture":"https:\/\/api.deezer.com\/artist\/13014917\/image","picture_small":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/56x56-000000-80-0-0.jpg","picture_medium":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/250x250-000000-80-0-0.jpg","picture_big":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/500x500-000000-80-0-0.jpg","picture_xl":"https:\/\/cdn-images.dzcdn.net\/images\/artist\/\/1000x1000-000000-80-0-0.jpg","nb_album":0,"nb_fan":66,"radio":true,"tracklist":"https:\/\/api.deezer.com\/artist\/13014917\/top?limit=50","type":"artist"}],"total":17}
\ No newline at end of file
diff --git a/tests/fixtures/lastfm.artist.page.html b/tests/fixtures/lastfm.artist.page.html
new file mode 100644
index 000000000..1922e313b
--- /dev/null
+++ b/tests/fixtures/lastfm.artist.page.html
@@ -0,0 +1,7 @@
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tests/fixtures/lastfm.artist.page.ignored.html b/tests/fixtures/lastfm.artist.page.ignored.html
new file mode 100644
index 000000000..96eda2377
--- /dev/null
+++ b/tests/fixtures/lastfm.artist.page.ignored.html
@@ -0,0 +1,7 @@
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tests/fixtures/lastfm.artist.page.no_meta.html b/tests/fixtures/lastfm.artist.page.no_meta.html
new file mode 100644
index 000000000..aa7b9c934
--- /dev/null
+++ b/tests/fixtures/lastfm.artist.page.no_meta.html
@@ -0,0 +1,6 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tests/fixtures/mixed-lyrics.flac b/tests/fixtures/mixed-lyrics.flac
new file mode 100644
index 000000000..d048234f5
Binary files /dev/null and b/tests/fixtures/mixed-lyrics.flac differ
diff --git a/tests/fixtures/no_replaygain.mp3 b/tests/fixtures/no_replaygain.mp3
new file mode 100644
index 000000000..45c2176e3
Binary files /dev/null and b/tests/fixtures/no_replaygain.mp3 differ
diff --git a/tests/fixtures/playlists/bom-test-utf16.m3u b/tests/fixtures/playlists/bom-test-utf16.m3u
new file mode 100644
index 000000000..9c2e9d599
Binary files /dev/null and b/tests/fixtures/playlists/bom-test-utf16.m3u differ
diff --git a/tests/fixtures/playlists/bom-test.m3u b/tests/fixtures/playlists/bom-test.m3u
new file mode 100644
index 000000000..f5a00806c
--- /dev/null
+++ b/tests/fixtures/playlists/bom-test.m3u
@@ -0,0 +1,6 @@
+#EXTM3U
+# NOTE: This file intentionally contains a UTF-8 BOM (Byte Order Mark) at the beginning
+# (bytes 0xEF 0xBB 0xBF) to test BOM handling in playlist parsing.
+#PLAYLIST:Test Playlist
+#EXTINF:123,Test Artist - Test Song
+test.mp3
diff --git a/tests/fixtures/zero_replaygain.mp3 b/tests/fixtures/zero_replaygain.mp3
new file mode 100644
index 000000000..96e6d21f0
Binary files /dev/null and b/tests/fixtures/zero_replaygain.mp3 differ
diff --git a/tests/mock_album_repo.go b/tests/mock_album_repo.go
index 58c33c97f..642ce6b41 100644
--- a/tests/mock_album_repo.go
+++ b/tests/mock_album_repo.go
@@ -16,10 +16,11 @@ func CreateMockAlbumRepo() *MockAlbumRepo {
type MockAlbumRepo struct {
model.AlbumRepository
- Data map[string]*model.Album
- All model.Albums
- Err bool
- Options model.QueryOptions
+ Data map[string]*model.Album
+ All model.Albums
+ Err bool
+ Options model.QueryOptions
+ ReassignAnnotationCalls map[string]string // prevID -> newID
}
func (m *MockAlbumRepo) SetError(err bool) {
@@ -117,4 +118,44 @@ func (m *MockAlbumRepo) UpdateExternalInfo(album *model.Album) error {
return nil
}
+func (m *MockAlbumRepo) Search(q string, offset int, size int, options ...model.QueryOptions) (model.Albums, error) {
+ if len(options) > 0 {
+ m.Options = options[0]
+ }
+ if m.Err {
+ return nil, errors.New("unexpected error")
+ }
+ // Simple mock implementation - just return all albums for testing
+ return m.All, nil
+}
+
+// ReassignAnnotation reassigns annotations from one album to another
+func (m *MockAlbumRepo) ReassignAnnotation(prevID string, newID string) error {
+ if m.Err {
+ return errors.New("unexpected error")
+ }
+ // Mock implementation - track the reassignment calls
+ if m.ReassignAnnotationCalls == nil {
+ m.ReassignAnnotationCalls = make(map[string]string)
+ }
+ m.ReassignAnnotationCalls[prevID] = newID
+ return nil
+}
+
+// SetRating sets the rating for an album
+func (m *MockAlbumRepo) SetRating(rating int, itemID string) error {
+ if m.Err {
+ return errors.New("unexpected error")
+ }
+ return nil
+}
+
+// SetStar sets the starred status for albums
+func (m *MockAlbumRepo) SetStar(starred bool, itemIDs ...string) error {
+ if m.Err {
+ return errors.New("unexpected error")
+ }
+ return nil
+}
+
var _ model.AlbumRepository = (*MockAlbumRepo)(nil)
diff --git a/tests/mock_artist_repo.go b/tests/mock_artist_repo.go
index 7058cead0..6d4792f83 100644
--- a/tests/mock_artist_repo.go
+++ b/tests/mock_artist_repo.go
@@ -16,8 +16,9 @@ func CreateMockArtistRepo() *MockArtistRepo {
type MockArtistRepo struct {
model.ArtistRepository
- Data map[string]*model.Artist
- Err bool
+ Data map[string]*model.Artist
+ Err bool
+ Options model.QueryOptions
}
func (m *MockArtistRepo) SetError(err bool) {
@@ -73,6 +74,9 @@ func (m *MockArtistRepo) IncPlayCount(id string, timestamp time.Time) error {
}
func (m *MockArtistRepo) GetAll(options ...model.QueryOptions) (model.Artists, error) {
+ if len(options) > 0 {
+ m.Options = options[0]
+ }
if m.Err {
return nil, errors.New("mock repo error")
}
@@ -94,4 +98,63 @@ func (m *MockArtistRepo) UpdateExternalInfo(artist *model.Artist) error {
return nil
}
+func (m *MockArtistRepo) RefreshStats(allArtists bool) (int64, error) {
+ if m.Err {
+ return 0, errors.New("mock repo error")
+ }
+ return int64(len(m.Data)), nil
+}
+
+func (m *MockArtistRepo) RefreshPlayCounts() (int64, error) {
+ if m.Err {
+ return 0, errors.New("mock repo error")
+ }
+ return int64(len(m.Data)), nil
+}
+
+func (m *MockArtistRepo) GetIndex(includeMissing bool, libraryIds []int, roles ...model.Role) (model.ArtistIndexes, error) {
+ if m.Err {
+ return nil, errors.New("mock repo error")
+ }
+
+ artists, err := m.GetAll()
+ if err != nil {
+ return nil, err
+ }
+
+ // For mock purposes, if no artists available, return empty result
+ if len(artists) == 0 {
+ return model.ArtistIndexes{}, nil
+ }
+
+ // Simple index grouping by first letter (simplified implementation for mocks)
+ indexMap := make(map[string]model.Artists)
+ for _, artist := range artists {
+ key := "#"
+ if len(artist.Name) > 0 {
+ key = string(artist.Name[0])
+ }
+ indexMap[key] = append(indexMap[key], artist)
+ }
+
+ var result model.ArtistIndexes
+ for k, artists := range indexMap {
+ result = append(result, model.ArtistIndex{ID: k, Artists: artists})
+ }
+
+ return result, nil
+}
+
+func (m *MockArtistRepo) Search(q string, offset int, size int, options ...model.QueryOptions) (model.Artists, error) {
+ if len(options) > 0 {
+ m.Options = options[0]
+ }
+ if m.Err {
+ return nil, errors.New("unexpected error")
+ }
+ // Simple mock implementation - just return all artists for testing
+ allArtists, err := m.GetAll()
+ return allArtists, err
+}
+
var _ model.ArtistRepository = (*MockArtistRepo)(nil)
diff --git a/tests/mock_data_store.go b/tests/mock_data_store.go
index f380755e0..ba586ab53 100644
--- a/tests/mock_data_store.go
+++ b/tests/mock_data_store.go
@@ -2,6 +2,7 @@ package tests
import (
"context"
+ "sync"
"github.com/navidrome/navidrome/model"
)
@@ -19,11 +20,18 @@ type MockDataStore struct {
MockedProperty model.PropertyRepository
MockedPlayer model.PlayerRepository
MockedPlaylist model.PlaylistRepository
+ MockedPlayQueue model.PlayQueueRepository
MockedShare model.ShareRepository
MockedTranscoding model.TranscodingRepository
MockedUserProps model.UserPropsRepository
MockedScrobbleBuffer model.ScrobbleBufferRepository
MockedRadio model.RadioRepository
+ scrobbleBufferMu sync.Mutex
+ repoMu sync.Mutex
+
+ // GC tracking
+ GCCalled bool
+ GCError error
}
func (db *MockDataStore) Library(ctx context.Context) model.LibraryRepository {
@@ -82,6 +90,8 @@ func (db *MockDataStore) Artist(ctx context.Context) model.ArtistRepository {
}
func (db *MockDataStore) MediaFile(ctx context.Context) model.MediaFileRepository {
+ db.repoMu.Lock()
+ defer db.repoMu.Unlock()
if db.MockedMediaFile == nil {
if db.RealDS != nil {
db.MockedMediaFile = db.RealDS.MediaFile(ctx)
@@ -115,10 +125,14 @@ func (db *MockDataStore) Playlist(ctx context.Context) model.PlaylistRepository
}
func (db *MockDataStore) PlayQueue(ctx context.Context) model.PlayQueueRepository {
- if db.RealDS != nil {
- return db.RealDS.PlayQueue(ctx)
+ if db.MockedPlayQueue == nil {
+ if db.RealDS != nil {
+ db.MockedPlayQueue = db.RealDS.PlayQueue(ctx)
+ } else {
+ db.MockedPlayQueue = &MockPlayQueueRepo{}
+ }
}
- return struct{ model.PlayQueueRepository }{}
+ return db.MockedPlayQueue
}
func (db *MockDataStore) UserProps(ctx context.Context) model.UserPropsRepository {
@@ -188,6 +202,8 @@ func (db *MockDataStore) Player(ctx context.Context) model.PlayerRepository {
}
func (db *MockDataStore) ScrobbleBuffer(ctx context.Context) model.ScrobbleBufferRepository {
+ db.scrobbleBufferMu.Lock()
+ defer db.scrobbleBufferMu.Unlock()
if db.MockedScrobbleBuffer == nil {
if db.RealDS != nil {
db.MockedScrobbleBuffer = db.RealDS.ScrobbleBuffer(ctx)
@@ -217,10 +233,39 @@ func (db *MockDataStore) WithTxImmediate(block func(tx model.DataStore) error, l
return block(db)
}
-func (db *MockDataStore) Resource(context.Context, any) model.ResourceRepository {
- return struct{ model.ResourceRepository }{}
+func (db *MockDataStore) Resource(ctx context.Context, m any) model.ResourceRepository {
+ switch m.(type) {
+ case model.MediaFile, *model.MediaFile:
+ return db.MediaFile(ctx).(model.ResourceRepository)
+ case model.Album, *model.Album:
+ return db.Album(ctx).(model.ResourceRepository)
+ case model.Artist, *model.Artist:
+ return db.Artist(ctx).(model.ResourceRepository)
+ case model.User, *model.User:
+ return db.User(ctx).(model.ResourceRepository)
+ case model.Playlist, *model.Playlist:
+ return db.Playlist(ctx).(model.ResourceRepository)
+ case model.Radio, *model.Radio:
+ return db.Radio(ctx).(model.ResourceRepository)
+ case model.Share, *model.Share:
+ return db.Share(ctx).(model.ResourceRepository)
+ case model.Genre, *model.Genre:
+ return db.Genre(ctx).(model.ResourceRepository)
+ case model.Tag, *model.Tag:
+ return db.Tag(ctx).(model.ResourceRepository)
+ case model.Transcoding, *model.Transcoding:
+ return db.Transcoding(ctx).(model.ResourceRepository)
+ case model.Player, *model.Player:
+ return db.Player(ctx).(model.ResourceRepository)
+ default:
+ return struct{ model.ResourceRepository }{}
+ }
}
-func (db *MockDataStore) GC(context.Context) error {
+func (db *MockDataStore) GC(context.Context, ...int) error {
+ db.GCCalled = true
+ if db.GCError != nil {
+ return db.GCError
+ }
return nil
}
diff --git a/tests/mock_library_repo.go b/tests/mock_library_repo.go
index 907a9d487..4d7539aa9 100644
--- a/tests/mock_library_repo.go
+++ b/tests/mock_library_repo.go
@@ -1,14 +1,22 @@
package tests
import (
+ "context"
+ "errors"
+ "fmt"
+ "slices"
+ "strconv"
+
+ "github.com/Masterminds/squirrel"
+ "github.com/deluan/rest"
"github.com/navidrome/navidrome/model"
- "golang.org/x/exp/maps"
)
type MockLibraryRepo struct {
model.LibraryRepository
- Data map[int]model.Library
- Err error
+ Data map[int]model.Library
+ Err error
+ PutFn func(*model.Library) error // Allow custom Put behavior for testing
}
func (m *MockLibraryRepo) SetData(data model.Libraries) {
@@ -22,7 +30,54 @@ func (m *MockLibraryRepo) GetAll(...model.QueryOptions) (model.Libraries, error)
if m.Err != nil {
return nil, m.Err
}
- return maps.Values(m.Data), nil
+ var libraries model.Libraries
+ for _, lib := range m.Data {
+ libraries = append(libraries, lib)
+ }
+ // Sort by ID for predictable order
+ slices.SortFunc(libraries, func(a, b model.Library) int {
+ return a.ID - b.ID
+ })
+ return libraries, nil
+}
+
+func (m *MockLibraryRepo) CountAll(qo ...model.QueryOptions) (int64, error) {
+ if m.Err != nil {
+ return 0, m.Err
+ }
+
+ // If no query options, return total count
+ if len(qo) == 0 || qo[0].Filters == nil {
+ return int64(len(m.Data)), nil
+ }
+
+ // Handle squirrel.Eq filter for ID validation
+ if eq, ok := qo[0].Filters.(squirrel.Eq); ok {
+ if idFilter, exists := eq["id"]; exists {
+ if ids, isSlice := idFilter.([]int); isSlice {
+ count := 0
+ for _, id := range ids {
+ if _, exists := m.Data[id]; exists {
+ count++
+ }
+ }
+ return int64(count), nil
+ }
+ }
+ }
+
+ // Default to total count for other filters
+ return int64(len(m.Data)), nil
+}
+
+func (m *MockLibraryRepo) Get(id int) (*model.Library, error) {
+ if m.Err != nil {
+ return nil, m.Err
+ }
+ if lib, ok := m.Data[id]; ok {
+ return &lib, nil
+ }
+ return nil, model.ErrNotFound
}
func (m *MockLibraryRepo) GetPath(id int) (string, error) {
@@ -35,4 +90,223 @@ func (m *MockLibraryRepo) GetPath(id int) (string, error) {
return "", model.ErrNotFound
}
-var _ model.LibraryRepository = &MockLibraryRepo{}
+func (m *MockLibraryRepo) Put(library *model.Library) error {
+ if m.PutFn != nil {
+ return m.PutFn(library)
+ }
+ if m.Err != nil {
+ return m.Err
+ }
+ if m.Data == nil {
+ m.Data = make(map[int]model.Library)
+ }
+ m.Data[library.ID] = *library
+ return nil
+}
+
+func (m *MockLibraryRepo) Delete(id int) error {
+ if m.Err != nil {
+ return m.Err
+ }
+ if _, ok := m.Data[id]; !ok {
+ return model.ErrNotFound
+ }
+ delete(m.Data, id)
+ return nil
+}
+
+func (m *MockLibraryRepo) StoreMusicFolder() error {
+ if m.Err != nil {
+ return m.Err
+ }
+ return nil
+}
+
+func (m *MockLibraryRepo) AddArtist(id int, artistID string) error {
+ if m.Err != nil {
+ return m.Err
+ }
+ return nil
+}
+
+func (m *MockLibraryRepo) ScanBegin(id int, fullScan bool) error {
+ if m.Err != nil {
+ return m.Err
+ }
+ return nil
+}
+
+func (m *MockLibraryRepo) ScanEnd(id int) error {
+ if m.Err != nil {
+ return m.Err
+ }
+ return nil
+}
+
+func (m *MockLibraryRepo) ScanInProgress() (bool, error) {
+ if m.Err != nil {
+ return false, m.Err
+ }
+ return false, nil
+}
+
+func (m *MockLibraryRepo) RefreshStats(id int) error {
+ return nil
+}
+
+// User-library association methods - mock implementations
+
+func (m *MockLibraryRepo) GetUsersWithLibraryAccess(libraryID int) (model.Users, error) {
+ if m.Err != nil {
+ return nil, m.Err
+ }
+ // Mock: return empty users for now
+ return model.Users{}, nil
+}
+
+func (m *MockLibraryRepo) Count(options ...rest.QueryOptions) (int64, error) {
+ return m.CountAll()
+}
+
+func (m *MockLibraryRepo) Read(id string) (interface{}, error) {
+ idInt, _ := strconv.Atoi(id)
+ mf, err := m.Get(idInt)
+ if errors.Is(err, model.ErrNotFound) {
+ return nil, rest.ErrNotFound
+ }
+ return mf, err
+}
+
+func (m *MockLibraryRepo) ReadAll(options ...rest.QueryOptions) (interface{}, error) {
+ return m.GetAll()
+}
+
+func (m *MockLibraryRepo) EntityName() string {
+ return "library"
+}
+
+func (m *MockLibraryRepo) NewInstance() interface{} {
+ return &model.Library{}
+}
+
+// REST Repository methods (string-based IDs)
+
+func (m *MockLibraryRepo) Save(entity interface{}) (string, error) {
+ lib := entity.(*model.Library)
+ if m.Err != nil {
+ return "", m.Err
+ }
+
+ // Validate required fields
+ if lib.Name == "" {
+ return "", &rest.ValidationError{Errors: map[string]string{"name": "library name is required"}}
+ }
+ if lib.Path == "" {
+ return "", &rest.ValidationError{Errors: map[string]string{"path": "library path is required"}}
+ }
+
+ // Generate ID if not set
+ if lib.ID == 0 {
+ lib.ID = len(m.Data) + 1
+ }
+ if m.Data == nil {
+ m.Data = make(map[int]model.Library)
+ }
+ m.Data[lib.ID] = *lib
+ return strconv.Itoa(lib.ID), nil
+}
+
+func (m *MockLibraryRepo) Update(id string, entity interface{}, cols ...string) error {
+ lib := entity.(*model.Library)
+ if m.Err != nil {
+ return m.Err
+ }
+
+ // Validate required fields
+ if lib.Name == "" {
+ return &rest.ValidationError{Errors: map[string]string{"name": "library name is required"}}
+ }
+ if lib.Path == "" {
+ return &rest.ValidationError{Errors: map[string]string{"path": "library path is required"}}
+ }
+
+ idInt, err := strconv.Atoi(id)
+ if err != nil {
+ return errors.New("invalid ID format")
+ }
+ if _, exists := m.Data[idInt]; !exists {
+ return rest.ErrNotFound
+ }
+ lib.ID = idInt
+ m.Data[idInt] = *lib
+ return nil
+}
+
+func (m *MockLibraryRepo) DeleteByStringID(id string) error {
+ if m.Err != nil {
+ return m.Err
+ }
+ idInt, err := strconv.Atoi(id)
+ if err != nil {
+ return errors.New("invalid ID format")
+ }
+ if _, exists := m.Data[idInt]; !exists {
+ return rest.ErrNotFound
+ }
+ delete(m.Data, idInt)
+ return nil
+}
+
+// Service-level methods for core.Library interface
+
+func (m *MockLibraryRepo) GetUserLibraries(ctx context.Context, userID string) (model.Libraries, error) {
+ if m.Err != nil {
+ return nil, m.Err
+ }
+ if userID == "non-existent" {
+ return nil, model.ErrNotFound
+ }
+ // Convert map to slice for return
+ var libraries model.Libraries
+ for _, lib := range m.Data {
+ libraries = append(libraries, lib)
+ }
+ // Sort by ID for predictable order
+ slices.SortFunc(libraries, func(a, b model.Library) int {
+ return a.ID - b.ID
+ })
+ return libraries, nil
+}
+
+func (m *MockLibraryRepo) SetUserLibraries(ctx context.Context, userID string, libraryIDs []int) error {
+ if m.Err != nil {
+ return m.Err
+ }
+ if userID == "non-existent" {
+ return model.ErrNotFound
+ }
+ if userID == "admin-1" {
+ return fmt.Errorf("%w: cannot manually assign libraries to admin users", model.ErrValidation)
+ }
+ if len(libraryIDs) == 0 {
+ return fmt.Errorf("%w: at least one library must be assigned to non-admin users", model.ErrValidation)
+ }
+ // Validate all library IDs exist
+ for _, id := range libraryIDs {
+ if _, exists := m.Data[id]; !exists {
+ return fmt.Errorf("%w: library ID %d does not exist", model.ErrValidation, id)
+ }
+ }
+ return nil
+}
+
+func (m *MockLibraryRepo) ValidateLibraryAccess(ctx context.Context, userID string, libraryID int) error {
+ if m.Err != nil {
+ return m.Err
+ }
+ // For testing purposes, allow access to all libraries
+ return nil
+}
+
+var _ model.LibraryRepository = (*MockLibraryRepo)(nil)
+var _ model.ResourceRepository = (*MockLibraryRepo)(nil)
diff --git a/tests/mock_mediafile_repo.go b/tests/mock_mediafile_repo.go
index 85adb8a25..5b38a7187 100644
--- a/tests/mock_mediafile_repo.go
+++ b/tests/mock_mediafile_repo.go
@@ -7,6 +7,7 @@ import (
"slices"
"time"
+ "github.com/deluan/rest"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/model/id"
"github.com/navidrome/navidrome/utils/slice"
@@ -26,6 +27,10 @@ type MockMediaFileRepo struct {
CountAllValue int64
CountAllOptions model.QueryOptions
DeleteAllMissingValue int64
+ Options model.QueryOptions
+ // Add fields for cross-library move detection tests
+ FindRecentFilesByMBZTrackIDFunc func(missing model.MediaFile, since time.Time) (model.MediaFiles, error)
+ FindRecentFilesByPropertiesFunc func(missing model.MediaFile, since time.Time) (model.MediaFiles, error)
}
func (m *MockMediaFileRepo) SetError(err bool) {
@@ -71,14 +76,22 @@ func (m *MockMediaFileRepo) GetWithParticipants(id string) (*model.MediaFile, er
return nil, model.ErrNotFound
}
-func (m *MockMediaFileRepo) GetAll(...model.QueryOptions) (model.MediaFiles, error) {
+func (m *MockMediaFileRepo) GetAll(qo ...model.QueryOptions) (model.MediaFiles, error) {
+ if len(qo) > 0 {
+ m.Options = qo[0]
+ }
if m.Err {
return nil, errors.New("error")
}
values := slices.Collect(maps.Values(m.Data))
- return slice.Map(values, func(p *model.MediaFile) model.MediaFile {
+ result := slice.Map(values, func(p *model.MediaFile) model.MediaFile {
return *p
- }), nil
+ })
+ // Sort by ID to ensure deterministic ordering for tests
+ slices.SortFunc(result, func(a, b model.MediaFile) int {
+ return cmp.Compare(a.ID, b.ID)
+ })
+ return result, nil
}
func (m *MockMediaFileRepo) Put(mf *model.MediaFile) error {
@@ -196,4 +209,91 @@ func (m *MockMediaFileRepo) DeleteAllMissing() (int64, error) {
return count, nil
}
+// ResourceRepository methods
+func (m *MockMediaFileRepo) Count(...rest.QueryOptions) (int64, error) {
+ return m.CountAll()
+}
+
+func (m *MockMediaFileRepo) Read(id string) (interface{}, error) {
+ mf, err := m.Get(id)
+ if errors.Is(err, model.ErrNotFound) {
+ return nil, rest.ErrNotFound
+ }
+ return mf, err
+}
+
+func (m *MockMediaFileRepo) ReadAll(...rest.QueryOptions) (interface{}, error) {
+ return m.GetAll()
+}
+
+func (m *MockMediaFileRepo) EntityName() string {
+ return "mediafile"
+}
+
+func (m *MockMediaFileRepo) NewInstance() interface{} {
+ return &model.MediaFile{}
+}
+
+func (m *MockMediaFileRepo) Search(q string, offset int, size int, options ...model.QueryOptions) (model.MediaFiles, error) {
+ if len(options) > 0 {
+ m.Options = options[0]
+ }
+ if m.Err {
+ return nil, errors.New("unexpected error")
+ }
+ // Simple mock implementation - just return all media files for testing
+ allFiles, err := m.GetAll()
+ return allFiles, err
+}
+
+// Cross-library move detection mock methods
+func (m *MockMediaFileRepo) FindRecentFilesByMBZTrackID(missing model.MediaFile, since time.Time) (model.MediaFiles, error) {
+ if m.Err {
+ return nil, errors.New("error")
+ }
+ if m.FindRecentFilesByMBZTrackIDFunc != nil {
+ return m.FindRecentFilesByMBZTrackIDFunc(missing, since)
+ }
+ // Default implementation: find files with same MBZ Track ID in other libraries
+ var result model.MediaFiles
+ for _, mf := range m.Data {
+ if mf.LibraryID != missing.LibraryID &&
+ mf.MbzReleaseTrackID == missing.MbzReleaseTrackID &&
+ mf.MbzReleaseTrackID != "" &&
+ mf.Suffix == missing.Suffix &&
+ mf.CreatedAt.After(since) &&
+ !mf.Missing {
+ result = append(result, *mf)
+ }
+ }
+ return result, nil
+}
+
+func (m *MockMediaFileRepo) FindRecentFilesByProperties(missing model.MediaFile, since time.Time) (model.MediaFiles, error) {
+ if m.Err {
+ return nil, errors.New("error")
+ }
+ if m.FindRecentFilesByPropertiesFunc != nil {
+ return m.FindRecentFilesByPropertiesFunc(missing, since)
+ }
+ // Default implementation: find files with same properties in other libraries
+ var result model.MediaFiles
+ for _, mf := range m.Data {
+ if mf.LibraryID != missing.LibraryID &&
+ mf.Title == missing.Title &&
+ mf.Size == missing.Size &&
+ mf.Suffix == missing.Suffix &&
+ mf.DiscNumber == missing.DiscNumber &&
+ mf.TrackNumber == missing.TrackNumber &&
+ mf.Album == missing.Album &&
+ mf.MbzReleaseTrackID == "" && // Exclude files with MBZ Track ID
+ mf.CreatedAt.After(since) &&
+ !mf.Missing {
+ result = append(result, *mf)
+ }
+ }
+ return result, nil
+}
+
var _ model.MediaFileRepository = (*MockMediaFileRepo)(nil)
+var _ model.ResourceRepository = (*MockMediaFileRepo)(nil)
diff --git a/tests/mock_playqueue_repo.go b/tests/mock_playqueue_repo.go
new file mode 100644
index 000000000..19976db57
--- /dev/null
+++ b/tests/mock_playqueue_repo.go
@@ -0,0 +1,65 @@
+package tests
+
+import (
+ "errors"
+
+ "github.com/navidrome/navidrome/model"
+)
+
+type MockPlayQueueRepo struct {
+ model.PlayQueueRepository
+ Queue *model.PlayQueue
+ Err bool
+ LastCols []string
+}
+
+func (m *MockPlayQueueRepo) Store(q *model.PlayQueue, cols ...string) error {
+ if m.Err {
+ return errors.New("error")
+ }
+ copyItems := make(model.MediaFiles, len(q.Items))
+ copy(copyItems, q.Items)
+ qCopy := *q
+ qCopy.Items = copyItems
+ m.Queue = &qCopy
+ m.LastCols = cols
+ return nil
+}
+
+func (m *MockPlayQueueRepo) RetrieveWithMediaFiles(userId string) (*model.PlayQueue, error) {
+ if m.Err {
+ return nil, errors.New("error")
+ }
+ if m.Queue == nil || m.Queue.UserID != userId {
+ return nil, model.ErrNotFound
+ }
+ copyItems := make(model.MediaFiles, len(m.Queue.Items))
+ copy(copyItems, m.Queue.Items)
+ qCopy := *m.Queue
+ qCopy.Items = copyItems
+ return &qCopy, nil
+}
+
+func (m *MockPlayQueueRepo) Retrieve(userId string) (*model.PlayQueue, error) {
+ if m.Err {
+ return nil, errors.New("error")
+ }
+ if m.Queue == nil || m.Queue.UserID != userId {
+ return nil, model.ErrNotFound
+ }
+ copyItems := make(model.MediaFiles, len(m.Queue.Items))
+ for i, t := range m.Queue.Items {
+ copyItems[i] = model.MediaFile{ID: t.ID}
+ }
+ qCopy := *m.Queue
+ qCopy.Items = copyItems
+ return &qCopy, nil
+}
+
+func (m *MockPlayQueueRepo) Clear(userId string) error {
+ if m.Err {
+ return errors.New("error")
+ }
+ m.Queue = nil
+ return nil
+}
diff --git a/tests/mock_scanner.go b/tests/mock_scanner.go
new file mode 100644
index 000000000..52396723f
--- /dev/null
+++ b/tests/mock_scanner.go
@@ -0,0 +1,120 @@
+package tests
+
+import (
+ "context"
+ "sync"
+
+ "github.com/navidrome/navidrome/model"
+)
+
+// MockScanner implements scanner.Scanner for testing with proper synchronization
+type MockScanner struct {
+ mu sync.Mutex
+ scanAllCalls []ScanAllCall
+ scanFoldersCalls []ScanFoldersCall
+ scanningStatus bool
+ statusResponse *model.ScannerStatus
+}
+
+type ScanAllCall struct {
+ FullScan bool
+}
+
+type ScanFoldersCall struct {
+ FullScan bool
+ Targets []model.ScanTarget
+}
+
+func NewMockScanner() *MockScanner {
+ return &MockScanner{
+ scanAllCalls: make([]ScanAllCall, 0),
+ scanFoldersCalls: make([]ScanFoldersCall, 0),
+ }
+}
+
+func (m *MockScanner) ScanAll(_ context.Context, fullScan bool) ([]string, error) {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+
+ m.scanAllCalls = append(m.scanAllCalls, ScanAllCall{FullScan: fullScan})
+
+ return nil, nil
+}
+
+func (m *MockScanner) ScanFolders(_ context.Context, fullScan bool, targets []model.ScanTarget) ([]string, error) {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+
+ // Make a copy of targets to avoid race conditions
+ targetsCopy := make([]model.ScanTarget, len(targets))
+ copy(targetsCopy, targets)
+
+ m.scanFoldersCalls = append(m.scanFoldersCalls, ScanFoldersCall{
+ FullScan: fullScan,
+ Targets: targetsCopy,
+ })
+
+ return nil, nil
+}
+
+func (m *MockScanner) Status(_ context.Context) (*model.ScannerStatus, error) {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+
+ if m.statusResponse != nil {
+ return m.statusResponse, nil
+ }
+
+ return &model.ScannerStatus{
+ Scanning: m.scanningStatus,
+ }, nil
+}
+
+func (m *MockScanner) GetScanAllCallCount() int {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+ return len(m.scanAllCalls)
+}
+
+func (m *MockScanner) GetScanAllCalls() []ScanAllCall {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+ // Return a copy to avoid race conditions
+ calls := make([]ScanAllCall, len(m.scanAllCalls))
+ copy(calls, m.scanAllCalls)
+ return calls
+}
+
+func (m *MockScanner) GetScanFoldersCallCount() int {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+ return len(m.scanFoldersCalls)
+}
+
+func (m *MockScanner) GetScanFoldersCalls() []ScanFoldersCall {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+ // Return a copy to avoid race conditions
+ calls := make([]ScanFoldersCall, len(m.scanFoldersCalls))
+ copy(calls, m.scanFoldersCalls)
+ return calls
+}
+
+func (m *MockScanner) Reset() {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+ m.scanAllCalls = make([]ScanAllCall, 0)
+ m.scanFoldersCalls = make([]ScanFoldersCall, 0)
+}
+
+func (m *MockScanner) SetScanning(scanning bool) {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+ m.scanningStatus = scanning
+}
+
+func (m *MockScanner) SetStatusResponse(status *model.ScannerStatus) {
+ m.mu.Lock()
+ defer m.mu.Unlock()
+ m.statusResponse = status
+}
diff --git a/tests/mock_scrobble_buffer_repo.go b/tests/mock_scrobble_buffer_repo.go
index 407c673eb..5865f423a 100644
--- a/tests/mock_scrobble_buffer_repo.go
+++ b/tests/mock_scrobble_buffer_repo.go
@@ -1,6 +1,7 @@
package tests
import (
+ "sync"
"time"
"github.com/navidrome/navidrome/model"
@@ -9,6 +10,7 @@ import (
type MockedScrobbleBufferRepo struct {
Error error
Data model.ScrobbleEntries
+ mu sync.RWMutex
}
func CreateMockedScrobbleBufferRepo() *MockedScrobbleBufferRepo {
@@ -19,6 +21,8 @@ func (m *MockedScrobbleBufferRepo) UserIDs(service string) ([]string, error) {
if m.Error != nil {
return nil, m.Error
}
+ m.mu.RLock()
+ defer m.mu.RUnlock()
userIds := make(map[string]struct{})
for _, e := range m.Data {
if e.Service == service {
@@ -36,6 +40,8 @@ func (m *MockedScrobbleBufferRepo) Enqueue(service, userId, mediaFileId string,
if m.Error != nil {
return m.Error
}
+ m.mu.Lock()
+ defer m.mu.Unlock()
m.Data = append(m.Data, model.ScrobbleEntry{
MediaFile: model.MediaFile{ID: mediaFileId},
Service: service,
@@ -50,6 +56,8 @@ func (m *MockedScrobbleBufferRepo) Next(service, userId string) (*model.Scrobble
if m.Error != nil {
return nil, m.Error
}
+ m.mu.RLock()
+ defer m.mu.RUnlock()
for _, e := range m.Data {
if e.Service == service && e.UserID == userId {
return &e, nil
@@ -62,6 +70,8 @@ func (m *MockedScrobbleBufferRepo) Dequeue(entry *model.ScrobbleEntry) error {
if m.Error != nil {
return m.Error
}
+ m.mu.Lock()
+ defer m.mu.Unlock()
newData := model.ScrobbleEntries{}
for _, e := range m.Data {
if e.Service == entry.Service && e.UserID == entry.UserID && e.PlayTime == entry.PlayTime && e.MediaFile.ID == entry.MediaFile.ID {
@@ -77,5 +87,7 @@ func (m *MockedScrobbleBufferRepo) Length() (int64, error) {
if m.Error != nil {
return 0, m.Error
}
+ m.mu.RLock()
+ defer m.mu.RUnlock()
return int64(len(m.Data)), nil
}
diff --git a/tests/mock_user_repo.go b/tests/mock_user_repo.go
index 09d804ccd..9f3dd672e 100644
--- a/tests/mock_user_repo.go
+++ b/tests/mock_user_repo.go
@@ -2,6 +2,7 @@ package tests
import (
"encoding/base64"
+ "fmt"
"strings"
"time"
@@ -11,14 +12,16 @@ import (
func CreateMockUserRepo() *MockedUserRepo {
return &MockedUserRepo{
- Data: map[string]*model.User{},
+ Data: map[string]*model.User{},
+ UserLibraries: map[string][]int{},
}
}
type MockedUserRepo struct {
model.UserRepository
- Error error
- Data map[string]*model.User
+ Error error
+ Data map[string]*model.User
+ UserLibraries map[string][]int // userID -> libraryIDs
}
func (u *MockedUserRepo) CountAll(qo ...model.QueryOptions) (int64, error) {
@@ -55,6 +58,18 @@ func (u *MockedUserRepo) FindByUsernameWithPassword(username string) (*model.Use
return u.FindByUsername(username)
}
+func (u *MockedUserRepo) Get(id string) (*model.User, error) {
+ if u.Error != nil {
+ return nil, u.Error
+ }
+ for _, usr := range u.Data {
+ if usr.ID == id {
+ return usr, nil
+ }
+ }
+ return nil, model.ErrNotFound
+}
+
func (u *MockedUserRepo) UpdateLastLoginAt(id string) error {
for _, usr := range u.Data {
if usr.ID == id {
@@ -74,3 +89,37 @@ func (u *MockedUserRepo) UpdateLastAccessAt(id string) error {
}
return u.Error
}
+
+// Library association methods - mock implementations
+
+func (u *MockedUserRepo) GetUserLibraries(userID string) (model.Libraries, error) {
+ if u.Error != nil {
+ return nil, u.Error
+ }
+ libraryIDs, exists := u.UserLibraries[userID]
+ if !exists {
+ return model.Libraries{}, nil
+ }
+
+ // Mock: Create libraries based on IDs
+ var libraries model.Libraries
+ for _, id := range libraryIDs {
+ libraries = append(libraries, model.Library{
+ ID: id,
+ Name: fmt.Sprintf("Test Library %d", id),
+ Path: fmt.Sprintf("/music/library%d", id),
+ })
+ }
+ return libraries, nil
+}
+
+func (u *MockedUserRepo) SetUserLibraries(userID string, libraryIDs []int) error {
+ if u.Error != nil {
+ return u.Error
+ }
+ if u.UserLibraries == nil {
+ u.UserLibraries = make(map[string][]int)
+ }
+ u.UserLibraries[userID] = libraryIDs
+ return nil
+}
diff --git a/tests/navidrome-test.toml b/tests/navidrome-test.toml
index 48f9f4c38..117178a76 100644
--- a/tests/navidrome-test.toml
+++ b/tests/navidrome-test.toml
@@ -1,5 +1,7 @@
User = "deluan"
Password = "wordpass"
DbPath = "file::memory:?cache=shared"
-DataFolder = "data/tests"
+DataFolder = "tmp/tests"
ScanSchedule="0"
+Plugins.Enabled = true
+Plugins.Folder = "plugins/testdata"
diff --git a/tests/test_helpers.go b/tests/test_helpers.go
index 1251c90cd..0a2cad4ad 100644
--- a/tests/test_helpers.go
+++ b/tests/test_helpers.go
@@ -6,7 +6,10 @@ import (
"path/filepath"
"github.com/navidrome/navidrome/db"
+ "github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model/id"
+ "github.com/sirupsen/logrus"
+ "github.com/sirupsen/logrus/hooks/test"
)
type testingT interface {
@@ -35,3 +38,23 @@ func ClearDB() error {
`)
return err
}
+
+// LogHook sets up a logrus test hook and configures the default logger to use it.
+// It returns the hook and a cleanup function to restore the default logger.
+// Example usage:
+//
+// hook, cleanup := LogHook()
+// defer cleanup()
+// // ... perform logging operations ...
+// Expect(hook.LastEntry()).ToNot(BeNil())
+// Expect(hook.LastEntry().Level).To(Equal(logrus.WarnLevel))
+// Expect(hook.LastEntry().Message).To(Equal("log message"))
+func LogHook() (*test.Hook, func()) {
+ l, hook := test.NewNullLogger()
+ log.SetLevel(log.LevelWarn)
+ log.SetDefaultLogger(l)
+ return hook, func() {
+ // Restore default logger after test
+ log.SetDefaultLogger(logrus.New())
+ }
+}
diff --git a/ui/package-lock.json b/ui/package-lock.json
index 9e449c5e0..c0901a73d 100644
--- a/ui/package-lock.json
+++ b/ui/package-lock.json
@@ -9,7 +9,7 @@
"dependencies": {
"@material-ui/core": "^4.12.4",
"@material-ui/icons": "^4.11.3",
- "@material-ui/lab": "^4.0.0-alpha.58",
+ "@material-ui/lab": "^4.0.0-alpha.61",
"@material-ui/styles": "^4.11.5",
"blueimp-md5": "^2.19.0",
"clsx": "^2.1.1",
@@ -37,8 +37,8 @@
"react-redux": "^7.2.9",
"react-router-dom": "^5.3.4",
"redux": "^4.2.1",
- "redux-saga": "^1.3.0",
- "uuid": "^11.1.0",
+ "redux-saga": "^1.4.2",
+ "uuid": "^13.0.0",
"workbox-cli": "^7.3.0"
},
"devDependencies": {
@@ -46,46 +46,35 @@
"@testing-library/react": "^12.1.5",
"@testing-library/react-hooks": "^7.0.2",
"@testing-library/user-event": "^14.6.1",
- "@types/node": "^22.15.21",
- "@types/react": "^17.0.86",
+ "@types/node": "^24.9.1",
+ "@types/react": "^17.0.89",
"@types/react-dom": "^17.0.26",
"@typescript-eslint/eslint-plugin": "^6.21.0",
"@typescript-eslint/parser": "^6.21.0",
- "@vitejs/plugin-react": "^4.5.0",
- "@vitest/coverage-v8": "^3.1.4",
+ "@vitejs/plugin-react": "^5.1.0",
+ "@vitest/coverage-v8": "^4.0.3",
"eslint": "^8.57.1",
- "eslint-config-prettier": "^10.1.5",
+ "eslint-config-prettier": "^10.1.8",
"eslint-plugin-jsx-a11y": "^6.10.2",
"eslint-plugin-react": "^7.37.5",
"eslint-plugin-react-hooks": "^5.2.0",
- "eslint-plugin-react-refresh": "^0.4.20",
- "happy-dom": "^17.4.7",
+ "eslint-plugin-react-refresh": "^0.4.24",
+ "happy-dom": "^20.0.8",
"jsdom": "^26.1.0",
- "prettier": "^3.5.3",
+ "prettier": "^3.6.2",
"ra-test": "^3.19.12",
"typescript": "^5.8.3",
- "vite": "^6.3.5",
- "vite-plugin-pwa": "^0.21.2",
- "vitest": "^3.1.4"
+ "vite": "^7.1.12",
+ "vite-plugin-pwa": "^1.1.0",
+ "vitest": "^4.0.3"
}
},
"node_modules/@adobe/css-tools": {
- "version": "4.4.3",
- "resolved": "https://registry.npmjs.org/@adobe/css-tools/-/css-tools-4.4.3.tgz",
- "integrity": "sha512-VQKMkwriZbaOgVCby1UDY/LDk5fIjhQicCvVPFqfe+69fWaPWydbWJ3wRt59/YzIwda1I81loas3oCoHxnqvdA==",
- "dev": true
- },
- "node_modules/@ampproject/remapping": {
- "version": "2.3.0",
- "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.3.0.tgz",
- "integrity": "sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==",
- "dependencies": {
- "@jridgewell/gen-mapping": "^0.3.5",
- "@jridgewell/trace-mapping": "^0.3.24"
- },
- "engines": {
- "node": ">=6.0.0"
- }
+ "version": "4.4.4",
+ "resolved": "https://registry.npmjs.org/@adobe/css-tools/-/css-tools-4.4.4.tgz",
+ "integrity": "sha512-Elp+iwUx5rN5+Y8xLt5/GRoG20WGoDCQ/1Fb+1LiGtvwbDavuSk0jhD/eZdckHAuzcDzccnkv+rEjyWfRx18gg==",
+ "dev": true,
+ "license": "MIT"
},
"node_modules/@asamuzakjp/css-color": {
"version": "3.2.0",
@@ -128,20 +117,20 @@
}
},
"node_modules/@babel/core": {
- "version": "7.27.1",
- "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.27.1.tgz",
- "integrity": "sha512-IaaGWsQqfsQWVLqMn9OB92MNN7zukfVA4s7KKAI0KfrrDsZ0yhi5uV4baBuLuN7n3vsZpwP8asPPcVwApxvjBQ==",
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.28.5.tgz",
+ "integrity": "sha512-e7jT4DxYvIDLk1ZHmU/m/mB19rex9sv0c2ftBtjSBv+kVM/902eh0fINUzD7UwLLNR+jU585GxUJ8/EBfAM5fw==",
"dependencies": {
- "@ampproject/remapping": "^2.2.0",
"@babel/code-frame": "^7.27.1",
- "@babel/generator": "^7.27.1",
- "@babel/helper-compilation-targets": "^7.27.1",
- "@babel/helper-module-transforms": "^7.27.1",
- "@babel/helpers": "^7.27.1",
- "@babel/parser": "^7.27.1",
- "@babel/template": "^7.27.1",
- "@babel/traverse": "^7.27.1",
- "@babel/types": "^7.27.1",
+ "@babel/generator": "^7.28.5",
+ "@babel/helper-compilation-targets": "^7.27.2",
+ "@babel/helper-module-transforms": "^7.28.3",
+ "@babel/helpers": "^7.28.4",
+ "@babel/parser": "^7.28.5",
+ "@babel/template": "^7.27.2",
+ "@babel/traverse": "^7.28.5",
+ "@babel/types": "^7.28.5",
+ "@jridgewell/remapping": "^2.3.5",
"convert-source-map": "^2.0.0",
"debug": "^4.1.0",
"gensync": "^1.0.0-beta.2",
@@ -165,14 +154,14 @@
}
},
"node_modules/@babel/generator": {
- "version": "7.27.1",
- "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.27.1.tgz",
- "integrity": "sha512-UnJfnIpc/+JO0/+KRVQNGU+y5taA5vCbwN8+azkX6beii/ZF+enZJSOKo11ZSzGJjlNfJHfQtmQT8H+9TXPG2w==",
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.28.5.tgz",
+ "integrity": "sha512-3EwLFhZ38J4VyIP6WNtt2kUdW9dokXA9Cr4IVIFHuCpZ3H8/YFOl5JjZHisrn1fATPBmKKqXzDFvh9fUwHz6CQ==",
"dependencies": {
- "@babel/parser": "^7.27.1",
- "@babel/types": "^7.27.1",
- "@jridgewell/gen-mapping": "^0.3.5",
- "@jridgewell/trace-mapping": "^0.3.25",
+ "@babel/parser": "^7.28.5",
+ "@babel/types": "^7.28.5",
+ "@jridgewell/gen-mapping": "^0.3.12",
+ "@jridgewell/trace-mapping": "^0.3.28",
"jsesc": "^3.0.2"
},
"engines": {
@@ -299,6 +288,14 @@
"url": "https://github.com/sponsors/ljharb"
}
},
+ "node_modules/@babel/helper-globals": {
+ "version": "7.28.0",
+ "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz",
+ "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
"node_modules/@babel/helper-member-expression-to-functions": {
"version": "7.27.1",
"resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.27.1.tgz",
@@ -324,13 +321,13 @@
}
},
"node_modules/@babel/helper-module-transforms": {
- "version": "7.27.1",
- "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.27.1.tgz",
- "integrity": "sha512-9yHn519/8KvTU5BjTVEEeIM3w9/2yXNKoD82JifINImhpKkARMJKPP59kLo+BafpdN5zgNeIcS4jsGDmd3l58g==",
+ "version": "7.28.3",
+ "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.3.tgz",
+ "integrity": "sha512-gytXUbs8k2sXS9PnQptz5o0QnpLL51SwASIORY6XaBKF88nsOT0Zw9szLqlSGQDP/4TljBAD5y98p2U1fqkdsw==",
"dependencies": {
"@babel/helper-module-imports": "^7.27.1",
"@babel/helper-validator-identifier": "^7.27.1",
- "@babel/traverse": "^7.27.1"
+ "@babel/traverse": "^7.28.3"
},
"engines": {
"node": ">=6.9.0"
@@ -411,9 +408,9 @@
}
},
"node_modules/@babel/helper-validator-identifier": {
- "version": "7.27.1",
- "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.27.1.tgz",
- "integrity": "sha512-D2hP9eA+Sqx1kBZgzxZh0y1trbuU+JoDkiEwqhQ36nodYqJwyEIhPSdMNd7lOm/4io72luTPWH20Yda0xOuUow==",
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz",
+ "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==",
"engines": {
"node": ">=6.9.0"
}
@@ -440,23 +437,23 @@
}
},
"node_modules/@babel/helpers": {
- "version": "7.27.1",
- "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.27.1.tgz",
- "integrity": "sha512-FCvFTm0sWV8Fxhpp2McP5/W53GPllQ9QeQ7SiqGWjMf/LVG07lFa5+pgK05IRhVwtvafT22KF+ZSnM9I545CvQ==",
+ "version": "7.28.4",
+ "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.28.4.tgz",
+ "integrity": "sha512-HFN59MmQXGHVyYadKLVumYsA9dBFun/ldYxipEjzA4196jpLZd8UjEEBLkbEkvfYreDqJhZxYAWFPtrfhNpj4w==",
"dependencies": {
- "@babel/template": "^7.27.1",
- "@babel/types": "^7.27.1"
+ "@babel/template": "^7.27.2",
+ "@babel/types": "^7.28.4"
},
"engines": {
"node": ">=6.9.0"
}
},
"node_modules/@babel/parser": {
- "version": "7.27.2",
- "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.27.2.tgz",
- "integrity": "sha512-QYLs8299NA7WM/bZAdp+CviYYkVoYXlDW2rzliy3chxd1PQjej7JORuMJDJXJUb9g0TT+B99EwaVLKmX+sPXWw==",
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.28.5.tgz",
+ "integrity": "sha512-KKBU1VGYR7ORr3At5HAtUQ+TV3SzRCXmA/8OdDZiLDBIZxVyzXuztPjfLd3BV1PRAQGCMWWSHYhL0F8d5uHBDQ==",
"dependencies": {
- "@babel/types": "^7.27.1"
+ "@babel/types": "^7.28.5"
},
"bin": {
"parser": "bin/babel-parser.js"
@@ -1464,9 +1461,10 @@
}
},
"node_modules/@babel/runtime": {
- "version": "7.27.1",
- "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.27.1.tgz",
- "integrity": "sha512-1x3D2xEk2fRo3PAhwQwu5UubzgiVWSXTBfWpVd2Mx2AzRqJuDJCsgaDVZ7HB5iGzDW1Hl1sWN2mFyKjmR9uAog==",
+ "version": "7.28.4",
+ "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.28.4.tgz",
+ "integrity": "sha512-Q/N6JNWvIvPnLDvjlE1OUBLPQHH6l3CltCEsHIujp45zQUSSh8K+gHnaEX45yAT1nyngnINhvWtzN+Nb9D8RAQ==",
+ "license": "MIT",
"engines": {
"node": ">=6.9.0"
}
@@ -1497,29 +1495,29 @@
}
},
"node_modules/@babel/traverse": {
- "version": "7.27.1",
- "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.27.1.tgz",
- "integrity": "sha512-ZCYtZciz1IWJB4U61UPu4KEaqyfj+r5T1Q5mqPo+IBpcG9kHv30Z0aD8LXPgC1trYa6rK0orRyAhqUgk4MjmEg==",
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.28.5.tgz",
+ "integrity": "sha512-TCCj4t55U90khlYkVV/0TfkJkAkUg3jZFA3Neb7unZT8CPok7iiRfaX0F+WnqWqt7OxhOn0uBKXCw4lbL8W0aQ==",
"dependencies": {
"@babel/code-frame": "^7.27.1",
- "@babel/generator": "^7.27.1",
- "@babel/parser": "^7.27.1",
- "@babel/template": "^7.27.1",
- "@babel/types": "^7.27.1",
- "debug": "^4.3.1",
- "globals": "^11.1.0"
+ "@babel/generator": "^7.28.5",
+ "@babel/helper-globals": "^7.28.0",
+ "@babel/parser": "^7.28.5",
+ "@babel/template": "^7.27.2",
+ "@babel/types": "^7.28.5",
+ "debug": "^4.3.1"
},
"engines": {
"node": ">=6.9.0"
}
},
"node_modules/@babel/types": {
- "version": "7.27.1",
- "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.27.1.tgz",
- "integrity": "sha512-+EzkxvLNfiUeKMgy/3luqfsCWFRXLb7U6wNQTk60tovuckwB15B191tJWvpp4HjiQWdJkCxO3Wbvc6jlk3Xb2Q==",
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.28.5.tgz",
+ "integrity": "sha512-qQ5m48eI/MFLQ5PxQj4PFaprjyCTLI37ElWMmNs0K8Lk3dVeOdNpB3ks8jc7yM5CDmVC73eMVk/trk3fgmrUpA==",
"dependencies": {
"@babel/helper-string-parser": "^7.27.1",
- "@babel/helper-validator-identifier": "^7.27.1"
+ "@babel/helper-validator-identifier": "^7.28.5"
},
"engines": {
"node": ">=6.9.0"
@@ -2100,10 +2098,11 @@
}
},
"node_modules/@eslint/eslintrc/node_modules/brace-expansion": {
- "version": "1.1.11",
- "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
- "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
+ "version": "1.1.12",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz",
+ "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==",
"dev": true,
+ "license": "MIT",
"dependencies": {
"balanced-match": "^1.0.0",
"concat-map": "0.0.1"
@@ -2173,10 +2172,11 @@
}
},
"node_modules/@humanwhocodes/config-array/node_modules/brace-expansion": {
- "version": "1.1.11",
- "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
- "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
+ "version": "1.1.12",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz",
+ "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==",
"dev": true,
+ "license": "MIT",
"dependencies": {
"balanced-match": "^1.0.0",
"concat-map": "0.0.1"
@@ -2214,76 +2214,6 @@
"deprecated": "Use @eslint/object-schema instead",
"dev": true
},
- "node_modules/@isaacs/cliui": {
- "version": "8.0.2",
- "resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz",
- "integrity": "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==",
- "dev": true,
- "dependencies": {
- "string-width": "^5.1.2",
- "string-width-cjs": "npm:string-width@^4.2.0",
- "strip-ansi": "^7.0.1",
- "strip-ansi-cjs": "npm:strip-ansi@^6.0.1",
- "wrap-ansi": "^8.1.0",
- "wrap-ansi-cjs": "npm:wrap-ansi@^7.0.0"
- },
- "engines": {
- "node": ">=12"
- }
- },
- "node_modules/@isaacs/cliui/node_modules/ansi-regex": {
- "version": "6.1.0",
- "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.1.0.tgz",
- "integrity": "sha512-7HSX4QQb4CspciLpVFwyRe79O3xsIZDDLER21kERQ71oaPodF8jL725AgJMFAYbooIqolJoRLuM81SpeUkpkvA==",
- "dev": true,
- "engines": {
- "node": ">=12"
- },
- "funding": {
- "url": "https://github.com/chalk/ansi-regex?sponsor=1"
- }
- },
- "node_modules/@isaacs/cliui/node_modules/string-width": {
- "version": "5.1.2",
- "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz",
- "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==",
- "dev": true,
- "dependencies": {
- "eastasianwidth": "^0.2.0",
- "emoji-regex": "^9.2.2",
- "strip-ansi": "^7.0.1"
- },
- "engines": {
- "node": ">=12"
- },
- "funding": {
- "url": "https://github.com/sponsors/sindresorhus"
- }
- },
- "node_modules/@isaacs/cliui/node_modules/strip-ansi": {
- "version": "7.1.0",
- "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz",
- "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==",
- "dev": true,
- "dependencies": {
- "ansi-regex": "^6.0.1"
- },
- "engines": {
- "node": ">=12"
- },
- "funding": {
- "url": "https://github.com/chalk/strip-ansi?sponsor=1"
- }
- },
- "node_modules/@istanbuljs/schema": {
- "version": "0.1.3",
- "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz",
- "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==",
- "dev": true,
- "engines": {
- "node": ">=8"
- }
- },
"node_modules/@jest/types": {
"version": "26.6.2",
"resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz",
@@ -2301,16 +2231,21 @@
}
},
"node_modules/@jridgewell/gen-mapping": {
- "version": "0.3.8",
- "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.8.tgz",
- "integrity": "sha512-imAbBGkb+ebQyxKgzv5Hu2nmROxoDOXHh80evxdoXNOrvAnVx7zimzc1Oo5h9RlfV4vPXaE2iM5pOFbvOCClWA==",
+ "version": "0.3.13",
+ "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz",
+ "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==",
"dependencies": {
- "@jridgewell/set-array": "^1.2.1",
- "@jridgewell/sourcemap-codec": "^1.4.10",
+ "@jridgewell/sourcemap-codec": "^1.5.0",
+ "@jridgewell/trace-mapping": "^0.3.24"
+ }
+ },
+ "node_modules/@jridgewell/remapping": {
+ "version": "2.3.5",
+ "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz",
+ "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==",
+ "dependencies": {
+ "@jridgewell/gen-mapping": "^0.3.5",
"@jridgewell/trace-mapping": "^0.3.24"
- },
- "engines": {
- "node": ">=6.0.0"
}
},
"node_modules/@jridgewell/resolve-uri": {
@@ -2321,14 +2256,6 @@
"node": ">=6.0.0"
}
},
- "node_modules/@jridgewell/set-array": {
- "version": "1.2.1",
- "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.2.1.tgz",
- "integrity": "sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==",
- "engines": {
- "node": ">=6.0.0"
- }
- },
"node_modules/@jridgewell/source-map": {
"version": "0.3.6",
"resolved": "https://registry.npmjs.org/@jridgewell/source-map/-/source-map-0.3.6.tgz",
@@ -2339,14 +2266,14 @@
}
},
"node_modules/@jridgewell/sourcemap-codec": {
- "version": "1.5.0",
- "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz",
- "integrity": "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ=="
+ "version": "1.5.5",
+ "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz",
+ "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og=="
},
"node_modules/@jridgewell/trace-mapping": {
- "version": "0.3.25",
- "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz",
- "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==",
+ "version": "0.3.31",
+ "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz",
+ "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==",
"dependencies": {
"@jridgewell/resolve-uri": "^3.1.0",
"@jridgewell/sourcemap-codec": "^1.4.14"
@@ -2596,16 +2523,6 @@
"node": ">= 8"
}
},
- "node_modules/@pkgjs/parseargs": {
- "version": "0.11.0",
- "resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz",
- "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==",
- "dev": true,
- "optional": true,
- "engines": {
- "node": ">=14"
- }
- },
"node_modules/@react-dnd/asap": {
"version": "4.0.1",
"resolved": "https://registry.npmjs.org/@react-dnd/asap/-/asap-4.0.1.tgz",
@@ -2630,16 +2547,17 @@
}
},
"node_modules/@redux-saga/core": {
- "version": "1.3.0",
- "resolved": "https://registry.npmjs.org/@redux-saga/core/-/core-1.3.0.tgz",
- "integrity": "sha512-L+i+qIGuyWn7CIg7k1MteHGfttKPmxwZR5E7OsGikCL2LzYA0RERlaUY00Y3P3ZV2EYgrsYlBrGs6cJP5OKKqA==",
+ "version": "1.4.2",
+ "resolved": "https://registry.npmjs.org/@redux-saga/core/-/core-1.4.2.tgz",
+ "integrity": "sha512-nIMLGKo6jV6Wc1sqtVQs1iqbB3Kq20udB/u9XEaZQisT6YZ0NRB8+4L6WqD/E+YziYutd27NJbG8EWUPkb7c6Q==",
+ "license": "MIT",
"dependencies": {
- "@babel/runtime": "^7.6.3",
- "@redux-saga/deferred": "^1.2.1",
- "@redux-saga/delay-p": "^1.2.1",
- "@redux-saga/is": "^1.1.3",
- "@redux-saga/symbols": "^1.1.3",
- "@redux-saga/types": "^1.2.1",
+ "@babel/runtime": "^7.28.4",
+ "@redux-saga/deferred": "^1.3.1",
+ "@redux-saga/delay-p": "^1.3.1",
+ "@redux-saga/is": "^1.2.1",
+ "@redux-saga/symbols": "^1.2.1",
+ "@redux-saga/types": "^1.3.1",
"typescript-tuple": "^2.2.1"
},
"funding": {
@@ -2648,41 +2566,46 @@
}
},
"node_modules/@redux-saga/deferred": {
- "version": "1.2.1",
- "resolved": "https://registry.npmjs.org/@redux-saga/deferred/-/deferred-1.2.1.tgz",
- "integrity": "sha512-cmin3IuuzMdfQjA0lG4B+jX+9HdTgHZZ+6u3jRAOwGUxy77GSlTi4Qp2d6PM1PUoTmQUR5aijlA39scWWPF31g=="
+ "version": "1.3.1",
+ "resolved": "https://registry.npmjs.org/@redux-saga/deferred/-/deferred-1.3.1.tgz",
+ "integrity": "sha512-0YZ4DUivWojXBqLB/TmuRRpDDz7tyq1I0AuDV7qi01XlLhM5m51W7+xYtIckH5U2cMlv9eAuicsfRAi1XHpXIg==",
+ "license": "MIT"
},
"node_modules/@redux-saga/delay-p": {
- "version": "1.2.1",
- "resolved": "https://registry.npmjs.org/@redux-saga/delay-p/-/delay-p-1.2.1.tgz",
- "integrity": "sha512-MdiDxZdvb1m+Y0s4/hgdcAXntpUytr9g0hpcOO1XFVyyzkrDu3SKPgBFOtHn7lhu7n24ZKIAT1qtKyQjHqRd+w==",
+ "version": "1.3.1",
+ "resolved": "https://registry.npmjs.org/@redux-saga/delay-p/-/delay-p-1.3.1.tgz",
+ "integrity": "sha512-597I7L5MXbD/1i3EmcaOOjL/5suxJD7p5tnbV1PiWnE28c2cYiIHqmSMK2s7us2/UrhOL2KTNBiD0qBg6KnImg==",
+ "license": "MIT",
"dependencies": {
- "@redux-saga/symbols": "^1.1.3"
+ "@redux-saga/symbols": "^1.2.1"
}
},
"node_modules/@redux-saga/is": {
- "version": "1.1.3",
- "resolved": "https://registry.npmjs.org/@redux-saga/is/-/is-1.1.3.tgz",
- "integrity": "sha512-naXrkETG1jLRfVfhOx/ZdLj0EyAzHYbgJWkXbB3qFliPcHKiWbv/ULQryOAEKyjrhiclmr6AMdgsXFyx7/yE6Q==",
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/@redux-saga/is/-/is-1.2.1.tgz",
+ "integrity": "sha512-x3aWtX3GmQfEvn8dh0ovPbsXgK9JjpiR24wKztpGbZP8JZUWWvUgKrvnWZ/T/4iphOBftyVc9VrIwhAnsM+OFA==",
+ "license": "MIT",
"dependencies": {
- "@redux-saga/symbols": "^1.1.3",
- "@redux-saga/types": "^1.2.1"
+ "@redux-saga/symbols": "^1.2.1",
+ "@redux-saga/types": "^1.3.1"
}
},
"node_modules/@redux-saga/symbols": {
- "version": "1.1.3",
- "resolved": "https://registry.npmjs.org/@redux-saga/symbols/-/symbols-1.1.3.tgz",
- "integrity": "sha512-hCx6ZvU4QAEUojETnX8EVg4ubNLBFl1Lps4j2tX7o45x/2qg37m3c6v+kSp8xjDJY+2tJw4QB3j8o8dsl1FDXg=="
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/@redux-saga/symbols/-/symbols-1.2.1.tgz",
+ "integrity": "sha512-3dh+uDvpBXi7EUp/eO+N7eFM4xKaU4yuGBXc50KnZGzIrR/vlvkTFQsX13zsY8PB6sCFYAgROfPSRUj8331QSA==",
+ "license": "MIT"
},
"node_modules/@redux-saga/types": {
- "version": "1.2.1",
- "resolved": "https://registry.npmjs.org/@redux-saga/types/-/types-1.2.1.tgz",
- "integrity": "sha512-1dgmkh+3so0+LlBWRhGA33ua4MYr7tUOj+a9Si28vUi0IUFNbff1T3sgpeDJI/LaC75bBYnQ0A3wXjn0OrRNBA=="
+ "version": "1.3.1",
+ "resolved": "https://registry.npmjs.org/@redux-saga/types/-/types-1.3.1.tgz",
+ "integrity": "sha512-YRCrJdhQLobGIQ8Cj1sta3nn6DrZDTSUnrIYhS2e5V590BmfVDleKoAquclAiKSBKWJwmuXTb+b4BL6rSHnahw==",
+ "license": "MIT"
},
"node_modules/@rolldown/pluginutils": {
- "version": "1.0.0-beta.9",
- "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.9.tgz",
- "integrity": "sha512-e9MeMtVWo186sgvFFJOPGy7/d2j2mZhLJIdVW0C/xDluuOvymEATqz6zKsP0ZmXGzQtqlyjz5sC1sYQUoJG98w==",
+ "version": "1.0.0-beta.43",
+ "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.43.tgz",
+ "integrity": "sha512-5Uxg7fQUCmfhax7FJke2+8B6cqgeUJUD9o2uXIKXhD+mG0mL6NObmVoi9wXEU1tY89mZKgAYA6fTbftx3q2ZPQ==",
"dev": true
},
"node_modules/@rollup/plugin-node-resolve": {
@@ -2793,6 +2716,12 @@
"node": ">=6"
}
},
+ "node_modules/@standard-schema/spec": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/@standard-schema/spec/-/spec-1.0.0.tgz",
+ "integrity": "sha512-m2bOd0f2RT9k8QJx1JN85cZYyH1RqFBdlwtkSlf4tBDYLCiiZnv1fIIwacK6cqwXavOydf0NPToMQgpKq+dVlA==",
+ "dev": true
+ },
"node_modules/@surma/rollup-plugin-off-main-thread": {
"version": "2.2.3",
"resolved": "https://registry.npmjs.org/@surma/rollup-plugin-off-main-thread/-/rollup-plugin-off-main-thread-2.2.3.tgz",
@@ -2844,17 +2773,17 @@
}
},
"node_modules/@testing-library/jest-dom": {
- "version": "6.6.3",
- "resolved": "https://registry.npmjs.org/@testing-library/jest-dom/-/jest-dom-6.6.3.tgz",
- "integrity": "sha512-IteBhl4XqYNkM54f4ejhLRJiZNqcSCoXUOG2CPK7qbD322KjQozM4kHQOfkG2oln9b9HTYqs+Sae8vBATubxxA==",
+ "version": "6.9.1",
+ "resolved": "https://registry.npmjs.org/@testing-library/jest-dom/-/jest-dom-6.9.1.tgz",
+ "integrity": "sha512-zIcONa+hVtVSSep9UT3jZ5rizo2BsxgyDYU7WFD5eICBE7no3881HGeb/QkGfsJs6JTkY1aQhT7rIPC7e+0nnA==",
"dev": true,
+ "license": "MIT",
"dependencies": {
"@adobe/css-tools": "^4.4.0",
"aria-query": "^5.0.0",
- "chalk": "^3.0.0",
"css.escape": "^1.5.1",
"dom-accessibility-api": "^0.6.3",
- "lodash": "^4.17.21",
+ "picocolors": "^1.1.1",
"redent": "^3.0.0"
},
"engines": {
@@ -2863,24 +2792,12 @@
"yarn": ">=1"
}
},
- "node_modules/@testing-library/jest-dom/node_modules/chalk": {
- "version": "3.0.0",
- "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz",
- "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==",
- "dev": true,
- "dependencies": {
- "ansi-styles": "^4.1.0",
- "supports-color": "^7.1.0"
- },
- "engines": {
- "node": ">=8"
- }
- },
"node_modules/@testing-library/jest-dom/node_modules/dom-accessibility-api": {
"version": "0.6.3",
"resolved": "https://registry.npmjs.org/dom-accessibility-api/-/dom-accessibility-api-0.6.3.tgz",
"integrity": "sha512-7ZgogeTnjuHbo+ct10G9Ffp0mif17idi0IyWNVA/wcwcm7NPOD/WEHVP3n7n3MhXqxoIYm8d6MuZohYWIZ4T3w==",
- "dev": true
+ "dev": true,
+ "license": "MIT"
},
"node_modules/@testing-library/react": {
"version": "12.1.5",
@@ -3017,6 +2934,22 @@
"@babel/types": "^7.20.7"
}
},
+ "node_modules/@types/chai": {
+ "version": "5.2.3",
+ "resolved": "https://registry.npmjs.org/@types/chai/-/chai-5.2.3.tgz",
+ "integrity": "sha512-Mw558oeA9fFbv65/y4mHtXDs9bPnFMZAL/jxdPFUpOHHIXX91mcgEHbS5Lahr+pwZFR8A7GQleRWeI6cGFC2UA==",
+ "dev": true,
+ "dependencies": {
+ "@types/deep-eql": "*",
+ "assertion-error": "^2.0.1"
+ }
+ },
+ "node_modules/@types/deep-eql": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/@types/deep-eql/-/deep-eql-4.0.2.tgz",
+ "integrity": "sha512-c9h9dVVMigMPc4bwTvC5dxqtqJZwQPePsWjPlpSOnojbor6pGqdk541lfA7AqFQr5pB1BRdq0juY9db81BwyFw==",
+ "dev": true
+ },
"node_modules/@types/estree": {
"version": "1.0.7",
"resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.7.tgz",
@@ -3067,12 +3000,13 @@
"integrity": "sha512-hov8bUuiLiyFPGyFPE1lwWhmzYbirOXQNNo40+y3zow8aFVTeyn3VWL0VFFfdNddA8S4Vf0Tc062rzyNr7Paag=="
},
"node_modules/@types/node": {
- "version": "22.15.21",
- "resolved": "https://registry.npmjs.org/@types/node/-/node-22.15.21.tgz",
- "integrity": "sha512-EV/37Td6c+MgKAbkcLG6vqZ2zEYHD7bvSrzqqs2RIhbA6w3x+Dqz8MZM3sP6kGTeLrdoOgKZe+Xja7tUB2DNkQ==",
+ "version": "24.9.1",
+ "resolved": "https://registry.npmjs.org/@types/node/-/node-24.9.1.tgz",
+ "integrity": "sha512-QoiaXANRkSXK6p0Duvt56W208du4P9Uye9hWLWgGMDTEoKPhuenzNcC4vGUmrNkiOKTlIrBoyNQYNpSwfEZXSg==",
"devOptional": true,
+ "license": "MIT",
"dependencies": {
- "undici-types": "~6.21.0"
+ "undici-types": "~7.16.0"
}
},
"node_modules/@types/normalize-package-data": {
@@ -3086,9 +3020,10 @@
"integrity": "sha512-gNMvNH49DJ7OJYv+KAKn0Xp45p8PLl6zo2YnvDIbTd4J6MER2BmWN49TG7n9LvkyihINxeKW8+3bfS2yDC9dzQ=="
},
"node_modules/@types/react": {
- "version": "17.0.86",
- "resolved": "https://registry.npmjs.org/@types/react/-/react-17.0.86.tgz",
- "integrity": "sha512-lPFuSjA85jecet6D4ZsPvCFuSrz6g2hkTSUw8MM0x5z2EndPV/itGnYQ39abjxd7F+cAcxLGtKQjnLn9cNUz3g==",
+ "version": "17.0.89",
+ "resolved": "https://registry.npmjs.org/@types/react/-/react-17.0.89.tgz",
+ "integrity": "sha512-I98SaDCar5lvEYl80ClRIUztH/hyWHR+I2f+5yTVp/MQ205HgYkA2b5mVdry/+nsEIrf8I65KA5V/PASx68MsQ==",
+ "license": "MIT",
"dependencies": {
"@types/prop-types": "*",
"@types/scheduler": "^0.16",
@@ -3158,6 +3093,13 @@
"resolved": "https://registry.npmjs.org/@types/trusted-types/-/trusted-types-2.0.7.tgz",
"integrity": "sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw=="
},
+ "node_modules/@types/whatwg-mimetype": {
+ "version": "3.0.2",
+ "resolved": "https://registry.npmjs.org/@types/whatwg-mimetype/-/whatwg-mimetype-3.0.2.tgz",
+ "integrity": "sha512-c2AKvDT8ToxLIOUlN51gTiHXflsfIFisS4pO7pDPoKouJCESkhZnEy623gwP9laCy5lnLDAw1vAzu2vM2YLOrA==",
+ "dev": true,
+ "license": "MIT"
+ },
"node_modules/@types/yargs": {
"version": "15.0.19",
"resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-15.0.19.tgz",
@@ -3370,50 +3312,49 @@
"dev": true
},
"node_modules/@vitejs/plugin-react": {
- "version": "4.5.0",
- "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-4.5.0.tgz",
- "integrity": "sha512-JuLWaEqypaJmOJPLWwO335Ig6jSgC1FTONCWAxnqcQthLTK/Yc9aH6hr9z/87xciejbQcnP3GnA1FWUSWeXaeg==",
+ "version": "5.1.0",
+ "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-5.1.0.tgz",
+ "integrity": "sha512-4LuWrg7EKWgQaMJfnN+wcmbAW+VSsCmqGohftWjuct47bv8uE4n/nPpq4XjJPsxgq00GGG5J8dvBczp8uxScew==",
"dev": true,
"dependencies": {
- "@babel/core": "^7.26.10",
- "@babel/plugin-transform-react-jsx-self": "^7.25.9",
- "@babel/plugin-transform-react-jsx-source": "^7.25.9",
- "@rolldown/pluginutils": "1.0.0-beta.9",
+ "@babel/core": "^7.28.4",
+ "@babel/plugin-transform-react-jsx-self": "^7.27.1",
+ "@babel/plugin-transform-react-jsx-source": "^7.27.1",
+ "@rolldown/pluginutils": "1.0.0-beta.43",
"@types/babel__core": "^7.20.5",
- "react-refresh": "^0.17.0"
+ "react-refresh": "^0.18.0"
},
"engines": {
- "node": "^14.18.0 || >=16.0.0"
+ "node": "^20.19.0 || >=22.12.0"
},
"peerDependencies": {
- "vite": "^4.2.0 || ^5.0.0 || ^6.0.0"
+ "vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0"
}
},
"node_modules/@vitest/coverage-v8": {
- "version": "3.1.4",
- "resolved": "https://registry.npmjs.org/@vitest/coverage-v8/-/coverage-v8-3.1.4.tgz",
- "integrity": "sha512-G4p6OtioySL+hPV7Y6JHlhpsODbJzt1ndwHAFkyk6vVjpK03PFsKnauZIzcd0PrK4zAbc5lc+jeZ+eNGiMA+iw==",
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/@vitest/coverage-v8/-/coverage-v8-4.0.3.tgz",
+ "integrity": "sha512-I+MlLwyJRBjmJr1kFYSxoseINbIdpxIAeK10jmXgB0FUtIfdYsvM3lGAvBu5yk8WPyhefzdmbCHCc1idFbNRcg==",
"dev": true,
"dependencies": {
- "@ampproject/remapping": "^2.3.0",
"@bcoe/v8-coverage": "^1.0.2",
- "debug": "^4.4.0",
+ "@vitest/utils": "4.0.3",
+ "ast-v8-to-istanbul": "^0.3.5",
+ "debug": "^4.4.3",
"istanbul-lib-coverage": "^3.2.2",
"istanbul-lib-report": "^3.0.1",
"istanbul-lib-source-maps": "^5.0.6",
- "istanbul-reports": "^3.1.7",
- "magic-string": "^0.30.17",
+ "istanbul-reports": "^3.2.0",
"magicast": "^0.3.5",
"std-env": "^3.9.0",
- "test-exclude": "^7.0.1",
- "tinyrainbow": "^2.0.0"
+ "tinyrainbow": "^3.0.3"
},
"funding": {
"url": "https://opencollective.com/vitest"
},
"peerDependencies": {
- "@vitest/browser": "3.1.4",
- "vitest": "3.1.4"
+ "@vitest/browser": "4.0.3",
+ "vitest": "4.0.3"
},
"peerDependenciesMeta": {
"@vitest/browser": {
@@ -3422,36 +3363,38 @@
}
},
"node_modules/@vitest/expect": {
- "version": "3.1.4",
- "resolved": "https://registry.npmjs.org/@vitest/expect/-/expect-3.1.4.tgz",
- "integrity": "sha512-xkD/ljeliyaClDYqHPNCiJ0plY5YIcM0OlRiZizLhlPmpXWpxnGMyTZXOHFhFeG7w9P5PBeL4IdtJ/HeQwTbQA==",
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/@vitest/expect/-/expect-4.0.3.tgz",
+ "integrity": "sha512-v3eSDx/bF25pzar6aEJrrdTXJduEBU3uSGXHslIdGIpJVP8tQQHV6x1ZfzbFQ/bLIomLSbR/2ZCfnaEGkWkiVQ==",
"dev": true,
"dependencies": {
- "@vitest/spy": "3.1.4",
- "@vitest/utils": "3.1.4",
- "chai": "^5.2.0",
- "tinyrainbow": "^2.0.0"
+ "@standard-schema/spec": "^1.0.0",
+ "@types/chai": "^5.2.2",
+ "@vitest/spy": "4.0.3",
+ "@vitest/utils": "4.0.3",
+ "chai": "^6.0.1",
+ "tinyrainbow": "^3.0.3"
},
"funding": {
"url": "https://opencollective.com/vitest"
}
},
"node_modules/@vitest/mocker": {
- "version": "3.1.4",
- "resolved": "https://registry.npmjs.org/@vitest/mocker/-/mocker-3.1.4.tgz",
- "integrity": "sha512-8IJ3CvwtSw/EFXqWFL8aCMu+YyYXG2WUSrQbViOZkWTKTVicVwZ/YiEZDSqD00kX+v/+W+OnxhNWoeVKorHygA==",
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/@vitest/mocker/-/mocker-4.0.3.tgz",
+ "integrity": "sha512-evZcRspIPbbiJEe748zI2BRu94ThCBE+RkjCpVF8yoVYuTV7hMe+4wLF/7K86r8GwJHSmAPnPbZhpXWWrg1qbA==",
"dev": true,
"dependencies": {
- "@vitest/spy": "3.1.4",
+ "@vitest/spy": "4.0.3",
"estree-walker": "^3.0.3",
- "magic-string": "^0.30.17"
+ "magic-string": "^0.30.19"
},
"funding": {
"url": "https://opencollective.com/vitest"
},
"peerDependencies": {
"msw": "^2.4.9",
- "vite": "^5.0.0 || ^6.0.0"
+ "vite": "^6.0.0 || ^7.0.0-0"
},
"peerDependenciesMeta": {
"msw": {
@@ -3463,24 +3406,24 @@
}
},
"node_modules/@vitest/pretty-format": {
- "version": "3.1.4",
- "resolved": "https://registry.npmjs.org/@vitest/pretty-format/-/pretty-format-3.1.4.tgz",
- "integrity": "sha512-cqv9H9GvAEoTaoq+cYqUTCGscUjKqlJZC7PRwY5FMySVj5J+xOm1KQcCiYHJOEzOKRUhLH4R2pTwvFlWCEScsg==",
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/@vitest/pretty-format/-/pretty-format-4.0.3.tgz",
+ "integrity": "sha512-N7gly/DRXzxa9w9sbDXwD9QNFYP2hw90LLLGDobPNwiWgyW95GMxsCt29/COIKKh3P7XJICR38PSDePenMBtsw==",
"dev": true,
"dependencies": {
- "tinyrainbow": "^2.0.0"
+ "tinyrainbow": "^3.0.3"
},
"funding": {
"url": "https://opencollective.com/vitest"
}
},
"node_modules/@vitest/runner": {
- "version": "3.1.4",
- "resolved": "https://registry.npmjs.org/@vitest/runner/-/runner-3.1.4.tgz",
- "integrity": "sha512-djTeF1/vt985I/wpKVFBMWUlk/I7mb5hmD5oP8K9ACRmVXgKTae3TUOtXAEBfslNKPzUQvnKhNd34nnRSYgLNQ==",
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/@vitest/runner/-/runner-4.0.3.tgz",
+ "integrity": "sha512-1/aK6fPM0lYXWyGKwop2Gbvz1plyTps/HDbIIJXYtJtspHjpXIeB3If07eWpVH4HW7Rmd3Rl+IS/+zEAXrRtXA==",
"dev": true,
"dependencies": {
- "@vitest/utils": "3.1.4",
+ "@vitest/utils": "4.0.3",
"pathe": "^2.0.3"
},
"funding": {
@@ -3488,13 +3431,13 @@
}
},
"node_modules/@vitest/snapshot": {
- "version": "3.1.4",
- "resolved": "https://registry.npmjs.org/@vitest/snapshot/-/snapshot-3.1.4.tgz",
- "integrity": "sha512-JPHf68DvuO7vilmvwdPr9TS0SuuIzHvxeaCkxYcCD4jTk67XwL45ZhEHFKIuCm8CYstgI6LZ4XbwD6ANrwMpFg==",
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/@vitest/snapshot/-/snapshot-4.0.3.tgz",
+ "integrity": "sha512-amnYmvZ5MTjNCP1HZmdeczAPLRD6iOm9+2nMRUGxbe/6sQ0Ymur0NnR9LIrWS8JA3wKE71X25D6ya/3LN9YytA==",
"dev": true,
"dependencies": {
- "@vitest/pretty-format": "3.1.4",
- "magic-string": "^0.30.17",
+ "@vitest/pretty-format": "4.0.3",
+ "magic-string": "^0.30.19",
"pathe": "^2.0.3"
},
"funding": {
@@ -3502,26 +3445,22 @@
}
},
"node_modules/@vitest/spy": {
- "version": "3.1.4",
- "resolved": "https://registry.npmjs.org/@vitest/spy/-/spy-3.1.4.tgz",
- "integrity": "sha512-Xg1bXhu+vtPXIodYN369M86K8shGLouNjoVI78g8iAq2rFoHFdajNvJJ5A/9bPMFcfQqdaCpOgWKEoMQg/s0Yg==",
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/@vitest/spy/-/spy-4.0.3.tgz",
+ "integrity": "sha512-82vVL8Cqz7rbXaNUl35V2G7xeNMAjBdNOVaHbrzznT9BmiCiPOzhf0FhU3eP41nP1bLDm/5wWKZqkG4nyU95DQ==",
"dev": true,
- "dependencies": {
- "tinyspy": "^3.0.2"
- },
"funding": {
"url": "https://opencollective.com/vitest"
}
},
"node_modules/@vitest/utils": {
- "version": "3.1.4",
- "resolved": "https://registry.npmjs.org/@vitest/utils/-/utils-3.1.4.tgz",
- "integrity": "sha512-yriMuO1cfFhmiGc8ataN51+9ooHRuURdfAZfwFd3usWynjzpLslZdYnRegTv32qdgtJTsj15FoeZe2g15fY1gg==",
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/@vitest/utils/-/utils-4.0.3.tgz",
+ "integrity": "sha512-qV6KJkq8W3piW6MDIbGOmn1xhvcW4DuA07alqaQ+vdx7YA49J85pnwnxigZVQFQw3tWnQNRKWwhz5wbP6iv/GQ==",
"dev": true,
"dependencies": {
- "@vitest/pretty-format": "3.1.4",
- "loupe": "^3.1.3",
- "tinyrainbow": "^2.0.0"
+ "@vitest/pretty-format": "4.0.3",
+ "tinyrainbow": "^3.0.3"
},
"funding": {
"url": "https://opencollective.com/vitest"
@@ -3813,6 +3752,23 @@
"integrity": "sha512-OH/2E5Fg20h2aPrbe+QL8JZQFko0YZaF+j4mnQ7BGhfavO7OpSLa8a0y9sBwomHdSbkhTS8TQNayBfnW5DwbvQ==",
"dev": true
},
+ "node_modules/ast-v8-to-istanbul": {
+ "version": "0.3.8",
+ "resolved": "https://registry.npmjs.org/ast-v8-to-istanbul/-/ast-v8-to-istanbul-0.3.8.tgz",
+ "integrity": "sha512-szgSZqUxI5T8mLKvS7WTjF9is+MVbOeLADU73IseOcrqhxr/VAvy6wfoVE39KnKzA7JRhjF5eUagNlHwvZPlKQ==",
+ "dev": true,
+ "dependencies": {
+ "@jridgewell/trace-mapping": "^0.3.31",
+ "estree-walker": "^3.0.3",
+ "js-tokens": "^9.0.1"
+ }
+ },
+ "node_modules/ast-v8-to-istanbul/node_modules/js-tokens": {
+ "version": "9.0.1",
+ "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-9.0.1.tgz",
+ "integrity": "sha512-mxa9E9ITFOt0ban3j6L5MpjwegGz6lBQmM1IJkWeBZGcMxto50+eWdjC/52xDbS2vy0k7vIMK0Fe2wfL9OQSpQ==",
+ "dev": true
+ },
"node_modules/async": {
"version": "3.2.6",
"resolved": "https://registry.npmjs.org/async/-/async-3.2.6.tgz",
@@ -4027,9 +3983,10 @@
}
},
"node_modules/brace-expansion": {
- "version": "2.0.1",
- "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz",
- "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==",
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.2.tgz",
+ "integrity": "sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==",
+ "license": "MIT",
"dependencies": {
"balanced-match": "^1.0.0"
}
@@ -4104,15 +4061,6 @@
"resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz",
"integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ=="
},
- "node_modules/cac": {
- "version": "6.7.14",
- "resolved": "https://registry.npmjs.org/cac/-/cac-6.7.14.tgz",
- "integrity": "sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==",
- "dev": true,
- "engines": {
- "node": ">=8"
- }
- },
"node_modules/cacheable-request": {
"version": "6.1.0",
"resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz",
@@ -4262,19 +4210,12 @@
]
},
"node_modules/chai": {
- "version": "5.2.0",
- "resolved": "https://registry.npmjs.org/chai/-/chai-5.2.0.tgz",
- "integrity": "sha512-mCuXncKXk5iCLhfhwTc0izo0gtEmpz5CtG2y8GiOINBlMVS6v8TMRc5TaLWKS6692m9+dVVfzgeVxR5UxWHTYw==",
+ "version": "6.2.0",
+ "resolved": "https://registry.npmjs.org/chai/-/chai-6.2.0.tgz",
+ "integrity": "sha512-aUTnJc/JipRzJrNADXVvpVqi6CO0dn3nx4EVPxijri+fj3LUUDyZQOgVeW54Ob3Y1Xh9Iz8f+CgaCl8v0mn9bA==",
"dev": true,
- "dependencies": {
- "assertion-error": "^2.0.1",
- "check-error": "^2.1.1",
- "deep-eql": "^5.0.1",
- "loupe": "^3.1.0",
- "pathval": "^2.0.0"
- },
"engines": {
- "node": ">=12"
+ "node": ">=18"
}
},
"node_modules/chalk": {
@@ -4297,15 +4238,6 @@
"resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz",
"integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA=="
},
- "node_modules/check-error": {
- "version": "2.1.1",
- "resolved": "https://registry.npmjs.org/check-error/-/check-error-2.1.1.tgz",
- "integrity": "sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw==",
- "dev": true,
- "engines": {
- "node": ">= 16"
- }
- },
"node_modules/chokidar": {
"version": "3.6.0",
"resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz",
@@ -4590,7 +4522,8 @@
"version": "1.5.1",
"resolved": "https://registry.npmjs.org/css.escape/-/css.escape-1.5.1.tgz",
"integrity": "sha512-YUifsXXuknHlUsmlgyY0PKzgPOr7/FjCePfHNt0jxm83wHZi44VDMQ7/fGNkjY3/jV1MC+1CmZbaHzugyeRtpg==",
- "dev": true
+ "dev": true,
+ "license": "MIT"
},
"node_modules/cssstyle": {
"version": "4.3.1",
@@ -4692,9 +4625,9 @@
"integrity": "sha512-hBSVCvSmWC+QypYObzwGOd9wqdDpOt+0wl0KbU+R+uuZBS1jN8VsD1ss3irQDknRj5NvxiTF6oj/nDRnN/UQNw=="
},
"node_modules/debug": {
- "version": "4.4.1",
- "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.1.tgz",
- "integrity": "sha512-KcKCqiftBJcZr++7ykoDIEwSa3XWowTfNPo92BYxjXiyYEVrUQh2aLyhxBCwww+heortUFxEJYcRzosstTEBYQ==",
+ "version": "4.4.3",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz",
+ "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==",
"dependencies": {
"ms": "^2.1.3"
},
@@ -4763,15 +4696,6 @@
"node": ">=4"
}
},
- "node_modules/deep-eql": {
- "version": "5.0.2",
- "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-5.0.2.tgz",
- "integrity": "sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==",
- "dev": true,
- "engines": {
- "node": ">=6"
- }
- },
"node_modules/deep-equal": {
"version": "2.2.3",
"resolved": "https://registry.npmjs.org/deep-equal/-/deep-equal-2.2.3.tgz",
@@ -5014,12 +4938,6 @@
"resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.5.tgz",
"integrity": "sha512-1A8za6ws41LQgv9HrE/66jyC5yuSjQ3L/KOpFtoBilsAK2iA2wuS5rTt1OCzIvtS2V7nVmedsUU+DGRcjBmOYA=="
},
- "node_modules/eastasianwidth": {
- "version": "0.2.0",
- "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz",
- "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==",
- "dev": true
- },
"node_modules/ejs": {
"version": "3.1.10",
"resolved": "https://registry.npmjs.org/ejs/-/ejs-3.1.10.tgz",
@@ -5390,9 +5308,9 @@
}
},
"node_modules/eslint-config-prettier": {
- "version": "10.1.5",
- "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-10.1.5.tgz",
- "integrity": "sha512-zc1UmCpNltmVY34vuLRV61r1K27sWuX39E+uyUnY8xS2Bex88VV9cugG+UZbRSRGtGyFboj+D8JODyme1plMpw==",
+ "version": "10.1.8",
+ "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-10.1.8.tgz",
+ "integrity": "sha512-82GZUjRS0p/jganf6q1rEO25VSoHH0hKPCTrgillPjdI/3bgBhAE1QzHrHTizjpRvy6pGAvKjDJtk2pF9NDq8w==",
"dev": true,
"license": "MIT",
"bin": {
@@ -5444,10 +5362,11 @@
}
},
"node_modules/eslint-plugin-jsx-a11y/node_modules/brace-expansion": {
- "version": "1.1.11",
- "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
- "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
+ "version": "1.1.12",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz",
+ "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==",
"dev": true,
+ "license": "MIT",
"dependencies": {
"balanced-match": "^1.0.0",
"concat-map": "0.0.1"
@@ -5510,19 +5429,21 @@
}
},
"node_modules/eslint-plugin-react-refresh": {
- "version": "0.4.20",
- "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.20.tgz",
- "integrity": "sha512-XpbHQ2q5gUF8BGOX4dHe+71qoirYMhApEPZ7sfhF/dNnOF1UXnCMGZf79SFTBO7Bz5YEIT4TMieSlJBWhP9WBA==",
+ "version": "0.4.24",
+ "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.24.tgz",
+ "integrity": "sha512-nLHIW7TEq3aLrEYWpVaJ1dRgFR+wLDPN8e8FpYAql/bMV2oBEfC37K0gLEGgv9fy66juNShSMV8OkTqzltcG/w==",
"dev": true,
+ "license": "MIT",
"peerDependencies": {
"eslint": ">=8.40"
}
},
"node_modules/eslint-plugin-react/node_modules/brace-expansion": {
- "version": "1.1.11",
- "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
- "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
+ "version": "1.1.12",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz",
+ "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==",
"dev": true,
+ "license": "MIT",
"dependencies": {
"balanced-match": "^1.0.0",
"concat-map": "0.0.1"
@@ -5590,10 +5511,11 @@
}
},
"node_modules/eslint/node_modules/brace-expansion": {
- "version": "1.1.11",
- "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
- "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
+ "version": "1.1.12",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz",
+ "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==",
"dev": true,
+ "license": "MIT",
"dependencies": {
"balanced-match": "^1.0.0",
"concat-map": "0.0.1"
@@ -5716,9 +5638,9 @@
"integrity": "sha512-Z+ktTxTwv9ILfgKCk32OX3n/doe+OcLTRtqK9pcL+JsP3J1/VW8Uvl4ZjLlKqeW4rzK4oesDOGMEMRIZqtP4Iw=="
},
"node_modules/expect-type": {
- "version": "1.2.1",
- "resolved": "https://registry.npmjs.org/expect-type/-/expect-type-1.2.1.tgz",
- "integrity": "sha512-/kP8CAwxzLVEeFrMm4kMmy4CCDlpipyA7MYLVrdJIkV0fYF0UaigQHRsxHiuY/GEea+bh4KSv3TIlgr+2UL6bw==",
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/expect-type/-/expect-type-1.2.2.tgz",
+ "integrity": "sha512-JhFGDVJ7tmDJItKhYgJCGLOWjuK9vPxiXoUFLwLDc99NlmklilbiQJwoctZtt13+xMw91MCk/REan6MWHqDjyA==",
"dev": true,
"engines": {
"node": ">=12.0.0"
@@ -5964,34 +5886,6 @@
"url": "https://github.com/sponsors/ljharb"
}
},
- "node_modules/foreground-child": {
- "version": "3.3.1",
- "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.1.tgz",
- "integrity": "sha512-gIXjKqtFuWEgzFRJA9WCQeSJLZDjgJUOMCMzxtvFq/37KojM1BFGufqsCy0r4qSQmYLsZYMeyRqzIWOMup03sw==",
- "dev": true,
- "dependencies": {
- "cross-spawn": "^7.0.6",
- "signal-exit": "^4.0.1"
- },
- "engines": {
- "node": ">=14"
- },
- "funding": {
- "url": "https://github.com/sponsors/isaacs"
- }
- },
- "node_modules/foreground-child/node_modules/signal-exit": {
- "version": "4.1.0",
- "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz",
- "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==",
- "dev": true,
- "engines": {
- "node": ">=14"
- },
- "funding": {
- "url": "https://github.com/sponsors/isaacs"
- }
- },
"node_modules/fs-extra": {
"version": "9.1.0",
"resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-9.1.0.tgz",
@@ -6172,9 +6066,10 @@
}
},
"node_modules/glob/node_modules/brace-expansion": {
- "version": "1.1.11",
- "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
- "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
+ "version": "1.1.12",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz",
+ "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==",
+ "license": "MIT",
"dependencies": {
"balanced-match": "^1.0.0",
"concat-map": "0.0.1"
@@ -6292,18 +6187,37 @@
"dev": true
},
"node_modules/happy-dom": {
- "version": "17.4.7",
- "resolved": "https://registry.npmjs.org/happy-dom/-/happy-dom-17.4.7.tgz",
- "integrity": "sha512-NZypxadhCiV5NT4A+Y86aQVVKQ05KDmueja3sz008uJfDRwz028wd0aTiJPwo4RQlvlz0fznkEEBBCHVNWc08g==",
+ "version": "20.0.8",
+ "resolved": "https://registry.npmjs.org/happy-dom/-/happy-dom-20.0.8.tgz",
+ "integrity": "sha512-TlYaNQNtzsZ97rNMBAm8U+e2cUQXNithgfCizkDgc11lgmN4j9CKMhO3FPGKWQYPwwkFcPpoXYF/CqEPLgzfOg==",
"dev": true,
+ "license": "MIT",
"dependencies": {
- "webidl-conversions": "^7.0.0",
+ "@types/node": "^20.0.0",
+ "@types/whatwg-mimetype": "^3.0.2",
"whatwg-mimetype": "^3.0.0"
},
"engines": {
- "node": ">=18.0.0"
+ "node": ">=20.0.0"
}
},
+ "node_modules/happy-dom/node_modules/@types/node": {
+ "version": "20.19.23",
+ "resolved": "https://registry.npmjs.org/@types/node/-/node-20.19.23.tgz",
+ "integrity": "sha512-yIdlVVVHXpmqRhtyovZAcSy0MiPcYWGkoO4CGe/+jpP0hmNuihm4XhHbADpK++MsiLHP5MVlv+bcgdF99kSiFQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "undici-types": "~6.21.0"
+ }
+ },
+ "node_modules/happy-dom/node_modules/undici-types": {
+ "version": "6.21.0",
+ "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz",
+ "integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==",
+ "dev": true,
+ "license": "MIT"
+ },
"node_modules/hard-rejection": {
"version": "2.1.0",
"resolved": "https://registry.npmjs.org/hard-rejection/-/hard-rejection-2.1.0.tgz",
@@ -7186,9 +7100,9 @@
}
},
"node_modules/istanbul-reports": {
- "version": "3.1.7",
- "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.1.7.tgz",
- "integrity": "sha512-BewmUXImeuRk2YY0PVbxgKAysvhRPUQE0h5QRM++nVWyubKGV0l8qQ5op8+B2DOmwSe63Jivj0BjkPQVf8fP5g==",
+ "version": "3.2.0",
+ "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.2.0.tgz",
+ "integrity": "sha512-HGYWWS/ehqTV3xN10i23tkPkpH46MLCIMFNCaaKNavAXTF1RkqxawEPtnjnGZ6XKSInBKkiOA5BKS+aZiY3AvA==",
"dev": true,
"dependencies": {
"html-escaper": "^2.0.0",
@@ -7215,21 +7129,6 @@
"node": ">= 0.4"
}
},
- "node_modules/jackspeak": {
- "version": "3.4.3",
- "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-3.4.3.tgz",
- "integrity": "sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==",
- "dev": true,
- "dependencies": {
- "@isaacs/cliui": "^8.0.2"
- },
- "funding": {
- "url": "https://github.com/sponsors/isaacs"
- },
- "optionalDependencies": {
- "@pkgjs/parseargs": "^0.11.0"
- }
- },
"node_modules/jake": {
"version": "10.9.2",
"resolved": "https://registry.npmjs.org/jake/-/jake-10.9.2.tgz",
@@ -7248,9 +7147,10 @@
}
},
"node_modules/jake/node_modules/brace-expansion": {
- "version": "1.1.11",
- "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
- "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
+ "version": "1.1.12",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz",
+ "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==",
+ "license": "MIT",
"dependencies": {
"balanced-match": "^1.0.0",
"concat-map": "0.0.1"
@@ -7663,12 +7563,6 @@
"loose-envify": "cli.js"
}
},
- "node_modules/loupe": {
- "version": "3.1.3",
- "resolved": "https://registry.npmjs.org/loupe/-/loupe-3.1.3.tgz",
- "integrity": "sha512-kkIp7XSkP78ZxJEsSxW3712C6teJVoeHHwgo9zJ380de7IYyJ2ISlxojcH2pC5OFLewESmnRi/+XCDIEEVyoug==",
- "dev": true
- },
"node_modules/lowercase-keys": {
"version": "1.0.1",
"resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz",
@@ -7695,12 +7589,12 @@
}
},
"node_modules/magic-string": {
- "version": "0.30.17",
- "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.17.tgz",
- "integrity": "sha512-sNPKHvyjVf7gyjwS4xGTaW/mCnF8wnjtifKBEhxfZ7E/S8tQ0rssrwGNn6q8JH/ohItJfSQp9mBtQYuTlH5QnA==",
+ "version": "0.30.21",
+ "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.21.tgz",
+ "integrity": "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==",
"dev": true,
"dependencies": {
- "@jridgewell/sourcemap-codec": "^1.5.0"
+ "@jridgewell/sourcemap-codec": "^1.5.5"
}
},
"node_modules/magicast": {
@@ -7865,15 +7759,6 @@
"node": ">= 6"
}
},
- "node_modules/minipass": {
- "version": "7.1.2",
- "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz",
- "integrity": "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==",
- "dev": true,
- "engines": {
- "node": ">=16 || 14 >=14.17"
- }
- },
"node_modules/ms": {
"version": "2.1.3",
"resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
@@ -8263,12 +8148,6 @@
"node": ">=8"
}
},
- "node_modules/package-json-from-dist": {
- "version": "1.0.1",
- "resolved": "https://registry.npmjs.org/package-json-from-dist/-/package-json-from-dist-1.0.1.tgz",
- "integrity": "sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw==",
- "dev": true
- },
"node_modules/package-json/node_modules/semver": {
"version": "6.3.1",
"resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz",
@@ -8348,28 +8227,6 @@
"resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz",
"integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw=="
},
- "node_modules/path-scurry": {
- "version": "1.11.1",
- "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.11.1.tgz",
- "integrity": "sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==",
- "dev": true,
- "dependencies": {
- "lru-cache": "^10.2.0",
- "minipass": "^5.0.0 || ^6.0.2 || ^7.0.0"
- },
- "engines": {
- "node": ">=16 || 14 >=14.18"
- },
- "funding": {
- "url": "https://github.com/sponsors/isaacs"
- }
- },
- "node_modules/path-scurry/node_modules/lru-cache": {
- "version": "10.4.3",
- "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-10.4.3.tgz",
- "integrity": "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==",
- "dev": true
- },
"node_modules/path-to-regexp": {
"version": "1.9.0",
"resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-1.9.0.tgz",
@@ -8393,15 +8250,6 @@
"integrity": "sha512-WUjGcAqP1gQacoQe+OBJsFA7Ld4DyXuUIjZ5cc75cLHvJ7dtNsTugphxIADwspS+AraAUePCKrSVtPLFj/F88w==",
"dev": true
},
- "node_modules/pathval": {
- "version": "2.0.0",
- "resolved": "https://registry.npmjs.org/pathval/-/pathval-2.0.0.tgz",
- "integrity": "sha512-vE7JKRyES09KiunauX7nd2Q9/L7lhok4smP9RZTDeD4MVs72Dp2qNFVz39Nz5a0FVEW0BJR6C0DYrq6unoziZA==",
- "dev": true,
- "engines": {
- "node": ">= 14.16"
- }
- },
"node_modules/picocolors": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
@@ -8432,9 +8280,9 @@
}
},
"node_modules/postcss": {
- "version": "8.5.3",
- "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.3.tgz",
- "integrity": "sha512-dle9A3yYxlBSrt8Fu+IpjGT8SY8hN0mlaA6GY8t0P5PjIOZemULz/E2Bnm/2dcUOena75OTNkHI76uZBNUUq3A==",
+ "version": "8.5.6",
+ "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.6.tgz",
+ "integrity": "sha512-3Ybi1tAuwAP9s0r1UQ2J4n5Y0G05bJkpUIO0/bI9MhwmD70S5aTWbXGBwxHrelT+XM1k6dM0pk+SwNkpTRN7Pg==",
"dev": true,
"funding": [
{
@@ -8451,7 +8299,7 @@
}
],
"dependencies": {
- "nanoid": "^3.3.8",
+ "nanoid": "^3.3.11",
"picocolors": "^1.1.1",
"source-map-js": "^1.2.1"
},
@@ -8477,9 +8325,9 @@
}
},
"node_modules/prettier": {
- "version": "3.5.3",
- "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.5.3.tgz",
- "integrity": "sha512-QQtaxnoDJeAkDvDKWCLiwIXkTgRhwYDEQCghU9Z6q03iyek/rxRh/2lC3HB7P8sWT2xC/y5JDctPLBIGzHKbhw==",
+ "version": "3.6.2",
+ "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.6.2.tgz",
+ "integrity": "sha512-I7AIg5boAr5R0FFtJ6rCfD+LFsWHp81dolrFD8S79U9tb8Az2nGrJncnMSnys+bpQJfRUzqs9hnA81OAA3hCuQ==",
"dev": true,
"bin": {
"prettier": "bin/prettier.cjs"
@@ -9258,9 +9106,9 @@
}
},
"node_modules/react-refresh": {
- "version": "0.17.0",
- "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.17.0.tgz",
- "integrity": "sha512-z6F7K9bV85EfseRCp2bzrpyQ0Gkw1uLoCel9XBVWPg/TjRj94SkJzUTGfOa4bs7iJvBWtQG0Wq7wnI0syw3EBQ==",
+ "version": "0.18.0",
+ "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.18.0.tgz",
+ "integrity": "sha512-QgT5//D3jfjJb6Gsjxv0Slpj23ip+HtOpnNgnb2S5zU3CB26G/IDPGoy4RJB42wzFE46DRsstbW6tKHoKbhAxw==",
"dev": true,
"engines": {
"node": ">=0.10.0"
@@ -9461,11 +9309,12 @@
}
},
"node_modules/redux-saga": {
- "version": "1.3.0",
- "resolved": "https://registry.npmjs.org/redux-saga/-/redux-saga-1.3.0.tgz",
- "integrity": "sha512-J9RvCeAZXSTAibFY0kGw6Iy4EdyDNW7k6Q+liwX+bsck7QVsU78zz8vpBRweEfANxnnlG/xGGeOvf6r8UXzNJQ==",
+ "version": "1.4.2",
+ "resolved": "https://registry.npmjs.org/redux-saga/-/redux-saga-1.4.2.tgz",
+ "integrity": "sha512-QLIn/q+7MX/B+MkGJ/K6R3//60eJ4QNy65eqPsJrfGezbxdh1Jx+37VRKE2K4PsJnNET5JufJtgWdT30WBa+6w==",
+ "license": "MIT",
"dependencies": {
- "@redux-saga/core": "^1.3.0"
+ "@redux-saga/core": "^1.4.2"
}
},
"node_modules/reflect.getprototypeof": {
@@ -10184,9 +10033,9 @@
"dev": true
},
"node_modules/std-env": {
- "version": "3.9.0",
- "resolved": "https://registry.npmjs.org/std-env/-/std-env-3.9.0.tgz",
- "integrity": "sha512-UGvjygr6F6tpH7o2qyqR6QYpwraIjKSdtzyBdyytFOHmPZY917kwdwLG0RbOjWOnKmnm3PeHjaoLLMie7kPLQw==",
+ "version": "3.10.0",
+ "resolved": "https://registry.npmjs.org/std-env/-/std-env-3.10.0.tgz",
+ "integrity": "sha512-5GS12FdOZNliM5mAOxFRg7Ir0pWz8MdpYm6AY6VPkGpbA7ZzmbzNcBJQ0GPvvyWgcY7QAhCgf9Uy89I03faLkg==",
"dev": true
},
"node_modules/stop-iteration-iterator": {
@@ -10231,27 +10080,6 @@
"node": ">=8"
}
},
- "node_modules/string-width-cjs": {
- "name": "string-width",
- "version": "4.2.3",
- "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
- "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
- "dev": true,
- "dependencies": {
- "emoji-regex": "^8.0.0",
- "is-fullwidth-code-point": "^3.0.0",
- "strip-ansi": "^6.0.1"
- },
- "engines": {
- "node": ">=8"
- }
- },
- "node_modules/string-width-cjs/node_modules/emoji-regex": {
- "version": "8.0.0",
- "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
- "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
- "dev": true
- },
"node_modules/string-width/node_modules/emoji-regex": {
"version": "8.0.0",
"resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
@@ -10384,19 +10212,6 @@
"node": ">=8"
}
},
- "node_modules/strip-ansi-cjs": {
- "name": "strip-ansi",
- "version": "6.0.1",
- "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
- "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
- "dev": true,
- "dependencies": {
- "ansi-regex": "^5.0.1"
- },
- "engines": {
- "node": ">=8"
- }
- },
"node_modules/strip-comments": {
"version": "2.0.1",
"resolved": "https://registry.npmjs.org/strip-comments/-/strip-comments-2.0.1.tgz",
@@ -10509,55 +10324,6 @@
"node": ">=10"
}
},
- "node_modules/test-exclude": {
- "version": "7.0.1",
- "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-7.0.1.tgz",
- "integrity": "sha512-pFYqmTw68LXVjeWJMST4+borgQP2AyMNbg1BpZh9LbyhUeNkeaPF9gzfPGUAnSMV3qPYdWUwDIjjCLiSDOl7vg==",
- "dev": true,
- "dependencies": {
- "@istanbuljs/schema": "^0.1.2",
- "glob": "^10.4.1",
- "minimatch": "^9.0.4"
- },
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/test-exclude/node_modules/glob": {
- "version": "10.4.5",
- "resolved": "https://registry.npmjs.org/glob/-/glob-10.4.5.tgz",
- "integrity": "sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg==",
- "dev": true,
- "dependencies": {
- "foreground-child": "^3.1.0",
- "jackspeak": "^3.1.2",
- "minimatch": "^9.0.4",
- "minipass": "^7.1.2",
- "package-json-from-dist": "^1.0.0",
- "path-scurry": "^1.11.1"
- },
- "bin": {
- "glob": "dist/esm/bin.mjs"
- },
- "funding": {
- "url": "https://github.com/sponsors/isaacs"
- }
- },
- "node_modules/test-exclude/node_modules/minimatch": {
- "version": "9.0.5",
- "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz",
- "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==",
- "dev": true,
- "dependencies": {
- "brace-expansion": "^2.0.1"
- },
- "engines": {
- "node": ">=16 || 14 >=14.17"
- },
- "funding": {
- "url": "https://github.com/sponsors/isaacs"
- }
- },
"node_modules/text-table": {
"version": "0.2.0",
"resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz",
@@ -10592,13 +10358,13 @@
"dev": true
},
"node_modules/tinyglobby": {
- "version": "0.2.13",
- "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.13.tgz",
- "integrity": "sha512-mEwzpUgrLySlveBwEVDMKk5B57bhLPYovRfPAXD5gA/98Opn0rCDj3GtLwFvCvH5RK9uPCExUROW5NjDwvqkxw==",
+ "version": "0.2.15",
+ "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.15.tgz",
+ "integrity": "sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ==",
"dev": true,
"dependencies": {
- "fdir": "^6.4.4",
- "picomatch": "^4.0.2"
+ "fdir": "^6.5.0",
+ "picomatch": "^4.0.3"
},
"engines": {
"node": ">=12.0.0"
@@ -10608,10 +10374,13 @@
}
},
"node_modules/tinyglobby/node_modules/fdir": {
- "version": "6.4.4",
- "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.4.4.tgz",
- "integrity": "sha512-1NZP+GK4GfuAv3PqKvxQRDMjdSRZjnkq7KfhlNrCNNlZ0ygQFpebfrnfnq/W7fpUnAv9aGWmY1zKx7FYL3gwhg==",
+ "version": "6.5.0",
+ "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz",
+ "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==",
"dev": true,
+ "engines": {
+ "node": ">=12.0.0"
+ },
"peerDependencies": {
"picomatch": "^3 || ^4"
},
@@ -10622,9 +10391,9 @@
}
},
"node_modules/tinyglobby/node_modules/picomatch": {
- "version": "4.0.2",
- "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.2.tgz",
- "integrity": "sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg==",
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz",
+ "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==",
"dev": true,
"engines": {
"node": ">=12"
@@ -10633,28 +10402,10 @@
"url": "https://github.com/sponsors/jonschlinkert"
}
},
- "node_modules/tinypool": {
- "version": "1.0.2",
- "resolved": "https://registry.npmjs.org/tinypool/-/tinypool-1.0.2.tgz",
- "integrity": "sha512-al6n+QEANGFOMf/dmUMsuS5/r9B06uwlyNjZZql/zv8J7ybHCgoihBNORZCY2mzUuAnomQa2JdhyHKzZxPCrFA==",
- "dev": true,
- "engines": {
- "node": "^18.0.0 || >=20.0.0"
- }
- },
"node_modules/tinyrainbow": {
- "version": "2.0.0",
- "resolved": "https://registry.npmjs.org/tinyrainbow/-/tinyrainbow-2.0.0.tgz",
- "integrity": "sha512-op4nsTR47R6p0vMUUoYl/a+ljLFVtlfaXkLQmqfLR1qHma1h/ysYk4hEXZ880bf2CYgTskvTa/e196Vd5dDQXw==",
- "dev": true,
- "engines": {
- "node": ">=14.0.0"
- }
- },
- "node_modules/tinyspy": {
- "version": "3.0.2",
- "resolved": "https://registry.npmjs.org/tinyspy/-/tinyspy-3.0.2.tgz",
- "integrity": "sha512-n1cw8k1k0x4pgA2+9XrOkFydTerNcJ1zWCO5Nn9scWHTD+5tp8dghT2x1uduQePZTZgd3Tupf+x9BxJjeJi77Q==",
+ "version": "3.0.3",
+ "resolved": "https://registry.npmjs.org/tinyrainbow/-/tinyrainbow-3.0.3.tgz",
+ "integrity": "sha512-PSkbLUoxOFRzJYjjxHJt9xro7D+iilgMX/C9lawzVuYiIdcihh9DXmVibBe8lmcFrRi/VzlPjBxbN7rH24q8/Q==",
"dev": true,
"engines": {
"node": ">=14.0.0"
@@ -10875,6 +10626,7 @@
"version": "0.0.2",
"resolved": "https://registry.npmjs.org/typescript-compare/-/typescript-compare-0.0.2.tgz",
"integrity": "sha512-8ja4j7pMHkfLJQO2/8tut7ub+J3Lw2S3061eJLFQcvs3tsmJKp8KG5NtpLn7KcY2w08edF74BSVN7qJS0U6oHA==",
+ "license": "MIT",
"dependencies": {
"typescript-logic": "^0.0.0"
}
@@ -10882,12 +10634,14 @@
"node_modules/typescript-logic": {
"version": "0.0.0",
"resolved": "https://registry.npmjs.org/typescript-logic/-/typescript-logic-0.0.0.tgz",
- "integrity": "sha512-zXFars5LUkI3zP492ls0VskH3TtdeHCqu0i7/duGt60i5IGPIpAHE/DWo5FqJ6EjQ15YKXrt+AETjv60Dat34Q=="
+ "integrity": "sha512-zXFars5LUkI3zP492ls0VskH3TtdeHCqu0i7/duGt60i5IGPIpAHE/DWo5FqJ6EjQ15YKXrt+AETjv60Dat34Q==",
+ "license": "MIT"
},
"node_modules/typescript-tuple": {
"version": "2.2.1",
"resolved": "https://registry.npmjs.org/typescript-tuple/-/typescript-tuple-2.2.1.tgz",
"integrity": "sha512-Zcr0lbt8z5ZdEzERHAMAniTiIKerFCMgd7yjq1fPnDJ43et/k9twIFQMUYff9k5oXcsQ0WpvFcgzK2ZKASoW6Q==",
+ "license": "MIT",
"dependencies": {
"typescript-compare": "^0.0.2"
}
@@ -10910,10 +10664,11 @@
}
},
"node_modules/undici-types": {
- "version": "6.21.0",
- "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz",
- "integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==",
- "devOptional": true
+ "version": "7.16.0",
+ "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.16.0.tgz",
+ "integrity": "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw==",
+ "devOptional": true,
+ "license": "MIT"
},
"node_modules/unicode-canonical-property-names-ecmascript": {
"version": "2.0.1",
@@ -11072,15 +10827,16 @@
"integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw=="
},
"node_modules/uuid": {
- "version": "11.1.0",
- "resolved": "https://registry.npmjs.org/uuid/-/uuid-11.1.0.tgz",
- "integrity": "sha512-0/A9rDy9P7cJ+8w1c9WD9V//9Wj15Ce2MPz8Ri6032usz+NfePxx5AcN3bN+r6ZL6jEo066/yNYB3tn4pQEx+A==",
+ "version": "13.0.0",
+ "resolved": "https://registry.npmjs.org/uuid/-/uuid-13.0.0.tgz",
+ "integrity": "sha512-XQegIaBTVUjSHliKqcnFqYypAd4S+WCYt5NIeRs6w/UAry7z8Y9j5ZwRRL4kzq9U3sD6v+85er9FvkEaBpji2w==",
"funding": [
"https://github.com/sponsors/broofa",
"https://github.com/sponsors/ctavan"
],
+ "license": "MIT",
"bin": {
- "uuid": "dist/esm/bin/uuid"
+ "uuid": "dist-node/bin/uuid"
}
},
"node_modules/validate-npm-package-license": {
@@ -11098,23 +10854,23 @@
"integrity": "sha512-NOJ6JZCAWr0zlxZt+xqCHNTEKOsrks2HQd4MqhP1qy4z1SkbEP467eNx6TgDKXMvUOb+OENfJCZwM+16n7fRfw=="
},
"node_modules/vite": {
- "version": "6.3.5",
- "resolved": "https://registry.npmjs.org/vite/-/vite-6.3.5.tgz",
- "integrity": "sha512-cZn6NDFE7wdTpINgs++ZJ4N49W2vRp8LCKrn3Ob1kYNtOo21vfDoaV5GzBfLU4MovSAB8uNRm4jgzVQZ+mBzPQ==",
+ "version": "7.1.12",
+ "resolved": "https://registry.npmjs.org/vite/-/vite-7.1.12.tgz",
+ "integrity": "sha512-ZWyE8YXEXqJrrSLvYgrRP7p62OziLW7xI5HYGWFzOvupfAlrLvURSzv/FyGyy0eidogEM3ujU+kUG1zuHgb6Ug==",
"dev": true,
"dependencies": {
"esbuild": "^0.25.0",
- "fdir": "^6.4.4",
- "picomatch": "^4.0.2",
- "postcss": "^8.5.3",
- "rollup": "^4.34.9",
- "tinyglobby": "^0.2.13"
+ "fdir": "^6.5.0",
+ "picomatch": "^4.0.3",
+ "postcss": "^8.5.6",
+ "rollup": "^4.43.0",
+ "tinyglobby": "^0.2.15"
},
"bin": {
"vite": "bin/vite.js"
},
"engines": {
- "node": "^18.0.0 || ^20.0.0 || >=22.0.0"
+ "node": "^20.19.0 || >=22.12.0"
},
"funding": {
"url": "https://github.com/vitejs/vite?sponsor=1"
@@ -11123,14 +10879,14 @@
"fsevents": "~2.3.3"
},
"peerDependencies": {
- "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0",
+ "@types/node": "^20.19.0 || >=22.12.0",
"jiti": ">=1.21.0",
- "less": "*",
+ "less": "^4.0.0",
"lightningcss": "^1.21.0",
- "sass": "*",
- "sass-embedded": "*",
- "stylus": "*",
- "sugarss": "*",
+ "sass": "^1.70.0",
+ "sass-embedded": "^1.70.0",
+ "stylus": ">=0.54.8",
+ "sugarss": "^5.0.0",
"terser": "^5.16.0",
"tsx": "^4.8.1",
"yaml": "^2.4.2"
@@ -11171,32 +10927,10 @@
}
}
},
- "node_modules/vite-node": {
- "version": "3.1.4",
- "resolved": "https://registry.npmjs.org/vite-node/-/vite-node-3.1.4.tgz",
- "integrity": "sha512-6enNwYnpyDo4hEgytbmc6mYWHXDHYEn0D1/rw4Q+tnHUGtKTJsn8T1YkX6Q18wI5LCrS8CTYlBaiCqxOy2kvUA==",
- "dev": true,
- "dependencies": {
- "cac": "^6.7.14",
- "debug": "^4.4.0",
- "es-module-lexer": "^1.7.0",
- "pathe": "^2.0.3",
- "vite": "^5.0.0 || ^6.0.0"
- },
- "bin": {
- "vite-node": "vite-node.mjs"
- },
- "engines": {
- "node": "^18.0.0 || ^20.0.0 || >=22.0.0"
- },
- "funding": {
- "url": "https://opencollective.com/vitest"
- }
- },
"node_modules/vite-plugin-pwa": {
- "version": "0.21.2",
- "resolved": "https://registry.npmjs.org/vite-plugin-pwa/-/vite-plugin-pwa-0.21.2.tgz",
- "integrity": "sha512-vFhH6Waw8itNu37hWUJxL50q+CBbNcMVzsKaYHQVrfxTt3ihk3PeLO22SbiP1UNWzcEPaTQv+YVxe4G0KOjAkg==",
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/vite-plugin-pwa/-/vite-plugin-pwa-1.1.0.tgz",
+ "integrity": "sha512-VsSpdubPzXhHWVINcSx6uHRMpOHVHQcHsef1QgkOlEoaIDAlssFEW88LBq1a59BuokAhsh2kUDJbaX1bZv4Bjw==",
"dev": true,
"dependencies": {
"debug": "^4.3.6",
@@ -11212,8 +10946,8 @@
"url": "https://github.com/sponsors/antfu"
},
"peerDependencies": {
- "@vite-pwa/assets-generator": "^0.2.6",
- "vite": "^3.1.0 || ^4.0.0 || ^5.0.0 || ^6.0.0",
+ "@vite-pwa/assets-generator": "^1.0.0",
+ "vite": "^3.1.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0",
"workbox-build": "^7.3.0",
"workbox-window": "^7.3.0"
},
@@ -11224,10 +10958,13 @@
}
},
"node_modules/vite/node_modules/fdir": {
- "version": "6.4.4",
- "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.4.4.tgz",
- "integrity": "sha512-1NZP+GK4GfuAv3PqKvxQRDMjdSRZjnkq7KfhlNrCNNlZ0ygQFpebfrnfnq/W7fpUnAv9aGWmY1zKx7FYL3gwhg==",
+ "version": "6.5.0",
+ "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz",
+ "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==",
"dev": true,
+ "engines": {
+ "node": ">=12.0.0"
+ },
"peerDependencies": {
"picomatch": "^3 || ^4"
},
@@ -11238,9 +10975,9 @@
}
},
"node_modules/vite/node_modules/picomatch": {
- "version": "4.0.2",
- "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.2.tgz",
- "integrity": "sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg==",
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz",
+ "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==",
"dev": true,
"engines": {
"node": ">=12"
@@ -11250,38 +10987,37 @@
}
},
"node_modules/vitest": {
- "version": "3.1.4",
- "resolved": "https://registry.npmjs.org/vitest/-/vitest-3.1.4.tgz",
- "integrity": "sha512-Ta56rT7uWxCSJXlBtKgIlApJnT6e6IGmTYxYcmxjJ4ujuZDI59GUQgVDObXXJujOmPDBYXHK1qmaGtneu6TNIQ==",
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/vitest/-/vitest-4.0.3.tgz",
+ "integrity": "sha512-IUSop8jgaT7w0g1yOM/35qVtKjr/8Va4PrjzH1OUb0YH4c3OXB2lCZDkMAB6glA8T5w8S164oJGsbcmAecr4sA==",
"dev": true,
"dependencies": {
- "@vitest/expect": "3.1.4",
- "@vitest/mocker": "3.1.4",
- "@vitest/pretty-format": "^3.1.4",
- "@vitest/runner": "3.1.4",
- "@vitest/snapshot": "3.1.4",
- "@vitest/spy": "3.1.4",
- "@vitest/utils": "3.1.4",
- "chai": "^5.2.0",
- "debug": "^4.4.0",
- "expect-type": "^1.2.1",
- "magic-string": "^0.30.17",
+ "@vitest/expect": "4.0.3",
+ "@vitest/mocker": "4.0.3",
+ "@vitest/pretty-format": "4.0.3",
+ "@vitest/runner": "4.0.3",
+ "@vitest/snapshot": "4.0.3",
+ "@vitest/spy": "4.0.3",
+ "@vitest/utils": "4.0.3",
+ "debug": "^4.4.3",
+ "es-module-lexer": "^1.7.0",
+ "expect-type": "^1.2.2",
+ "magic-string": "^0.30.19",
"pathe": "^2.0.3",
+ "picomatch": "^4.0.3",
"std-env": "^3.9.0",
"tinybench": "^2.9.0",
"tinyexec": "^0.3.2",
- "tinyglobby": "^0.2.13",
- "tinypool": "^1.0.2",
- "tinyrainbow": "^2.0.0",
- "vite": "^5.0.0 || ^6.0.0",
- "vite-node": "3.1.4",
+ "tinyglobby": "^0.2.15",
+ "tinyrainbow": "^3.0.3",
+ "vite": "^6.0.0 || ^7.0.0",
"why-is-node-running": "^2.3.0"
},
"bin": {
"vitest": "vitest.mjs"
},
"engines": {
- "node": "^18.0.0 || ^20.0.0 || >=22.0.0"
+ "node": "^20.0.0 || ^22.0.0 || >=24.0.0"
},
"funding": {
"url": "https://opencollective.com/vitest"
@@ -11289,9 +11025,11 @@
"peerDependencies": {
"@edge-runtime/vm": "*",
"@types/debug": "^4.1.12",
- "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0",
- "@vitest/browser": "3.1.4",
- "@vitest/ui": "3.1.4",
+ "@types/node": "^20.0.0 || ^22.0.0 || >=24.0.0",
+ "@vitest/browser-playwright": "4.0.3",
+ "@vitest/browser-preview": "4.0.3",
+ "@vitest/browser-webdriverio": "4.0.3",
+ "@vitest/ui": "4.0.3",
"happy-dom": "*",
"jsdom": "*"
},
@@ -11305,7 +11043,13 @@
"@types/node": {
"optional": true
},
- "@vitest/browser": {
+ "@vitest/browser-playwright": {
+ "optional": true
+ },
+ "@vitest/browser-preview": {
+ "optional": true
+ },
+ "@vitest/browser-webdriverio": {
"optional": true
},
"@vitest/ui": {
@@ -11319,6 +11063,18 @@
}
}
},
+ "node_modules/vitest/node_modules/picomatch": {
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz",
+ "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==",
+ "dev": true,
+ "engines": {
+ "node": ">=12"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/jonschlinkert"
+ }
+ },
"node_modules/w3c-xmlserializer": {
"version": "5.0.0",
"resolved": "https://registry.npmjs.org/w3c-xmlserializer/-/w3c-xmlserializer-5.0.0.tgz",
@@ -11868,97 +11624,6 @@
"workbox-core": "7.3.0"
}
},
- "node_modules/wrap-ansi": {
- "version": "8.1.0",
- "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-8.1.0.tgz",
- "integrity": "sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==",
- "dev": true,
- "dependencies": {
- "ansi-styles": "^6.1.0",
- "string-width": "^5.0.1",
- "strip-ansi": "^7.0.1"
- },
- "engines": {
- "node": ">=12"
- },
- "funding": {
- "url": "https://github.com/chalk/wrap-ansi?sponsor=1"
- }
- },
- "node_modules/wrap-ansi-cjs": {
- "name": "wrap-ansi",
- "version": "7.0.0",
- "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz",
- "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==",
- "dev": true,
- "dependencies": {
- "ansi-styles": "^4.0.0",
- "string-width": "^4.1.0",
- "strip-ansi": "^6.0.0"
- },
- "engines": {
- "node": ">=10"
- },
- "funding": {
- "url": "https://github.com/chalk/wrap-ansi?sponsor=1"
- }
- },
- "node_modules/wrap-ansi/node_modules/ansi-regex": {
- "version": "6.1.0",
- "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.1.0.tgz",
- "integrity": "sha512-7HSX4QQb4CspciLpVFwyRe79O3xsIZDDLER21kERQ71oaPodF8jL725AgJMFAYbooIqolJoRLuM81SpeUkpkvA==",
- "dev": true,
- "engines": {
- "node": ">=12"
- },
- "funding": {
- "url": "https://github.com/chalk/ansi-regex?sponsor=1"
- }
- },
- "node_modules/wrap-ansi/node_modules/ansi-styles": {
- "version": "6.2.1",
- "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.1.tgz",
- "integrity": "sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==",
- "dev": true,
- "engines": {
- "node": ">=12"
- },
- "funding": {
- "url": "https://github.com/chalk/ansi-styles?sponsor=1"
- }
- },
- "node_modules/wrap-ansi/node_modules/string-width": {
- "version": "5.1.2",
- "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz",
- "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==",
- "dev": true,
- "dependencies": {
- "eastasianwidth": "^0.2.0",
- "emoji-regex": "^9.2.2",
- "strip-ansi": "^7.0.1"
- },
- "engines": {
- "node": ">=12"
- },
- "funding": {
- "url": "https://github.com/sponsors/sindresorhus"
- }
- },
- "node_modules/wrap-ansi/node_modules/strip-ansi": {
- "version": "7.1.0",
- "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz",
- "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==",
- "dev": true,
- "dependencies": {
- "ansi-regex": "^6.0.1"
- },
- "engines": {
- "node": ">=12"
- },
- "funding": {
- "url": "https://github.com/chalk/strip-ansi?sponsor=1"
- }
- },
"node_modules/wrappy": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
diff --git a/ui/package.json b/ui/package.json
index b9c93316b..a3612aaf4 100644
--- a/ui/package.json
+++ b/ui/package.json
@@ -18,7 +18,7 @@
"dependencies": {
"@material-ui/core": "^4.12.4",
"@material-ui/icons": "^4.11.3",
- "@material-ui/lab": "^4.0.0-alpha.58",
+ "@material-ui/lab": "^4.0.0-alpha.61",
"@material-ui/styles": "^4.11.5",
"blueimp-md5": "^2.19.0",
"clsx": "^2.1.1",
@@ -46,8 +46,8 @@
"react-redux": "^7.2.9",
"react-router-dom": "^5.3.4",
"redux": "^4.2.1",
- "redux-saga": "^1.3.0",
- "uuid": "^11.1.0",
+ "redux-saga": "^1.4.2",
+ "uuid": "^13.0.0",
"workbox-cli": "^7.3.0"
},
"devDependencies": {
@@ -55,27 +55,27 @@
"@testing-library/react": "^12.1.5",
"@testing-library/react-hooks": "^7.0.2",
"@testing-library/user-event": "^14.6.1",
- "@types/node": "^22.15.21",
- "@types/react": "^17.0.86",
+ "@types/node": "^24.9.1",
+ "@types/react": "^17.0.89",
"@types/react-dom": "^17.0.26",
"@typescript-eslint/eslint-plugin": "^6.21.0",
"@typescript-eslint/parser": "^6.21.0",
- "@vitejs/plugin-react": "^4.5.0",
- "@vitest/coverage-v8": "^3.1.4",
+ "@vitejs/plugin-react": "^5.1.0",
+ "@vitest/coverage-v8": "^4.0.3",
"eslint": "^8.57.1",
- "eslint-config-prettier": "^10.1.5",
+ "eslint-config-prettier": "^10.1.8",
"eslint-plugin-jsx-a11y": "^6.10.2",
"eslint-plugin-react": "^7.37.5",
"eslint-plugin-react-hooks": "^5.2.0",
- "eslint-plugin-react-refresh": "^0.4.20",
- "happy-dom": "^17.4.7",
+ "eslint-plugin-react-refresh": "^0.4.24",
+ "happy-dom": "^20.0.8",
"jsdom": "^26.1.0",
- "prettier": "^3.5.3",
+ "prettier": "^3.6.2",
"ra-test": "^3.19.12",
"typescript": "^5.8.3",
- "vite": "^6.3.5",
- "vite-plugin-pwa": "^0.21.2",
- "vitest": "^3.1.4"
+ "vite": "^7.1.12",
+ "vite-plugin-pwa": "^1.1.0",
+ "vitest": "^4.0.3"
},
"overrides": {
"vite": {
diff --git a/ui/src/App.jsx b/ui/src/App.jsx
index 1b89f7b8c..dc4fe9b53 100644
--- a/ui/src/App.jsx
+++ b/ui/src/App.jsx
@@ -15,9 +15,11 @@ import artist from './artist'
import playlist from './playlist'
import radio from './radio'
import share from './share'
+import library from './library'
import { Player } from './audioplayer'
import customRoutes from './routes'
import {
+ libraryReducer,
themeReducer,
addToPlaylistDialogReducer,
expandInfoDialogReducer,
@@ -56,6 +58,7 @@ const adminStore = createAdminStore({
dataProvider,
history,
customReducers: {
+ library: libraryReducer,
player: playerReducer,
albumView: albumViewReducer,
theme: themeReducer,
@@ -122,7 +125,13 @@ const Admin = (props) => {
) : (
),
-
+ permissions === 'admin' ? (
+
+ ) : null,
permissions === 'admin' ? (
{
,
,
,
+ ,
,
]}
diff --git a/ui/src/actions/index.js b/ui/src/actions/index.js
index a319f7a69..9f35f86a9 100644
--- a/ui/src/actions/index.js
+++ b/ui/src/actions/index.js
@@ -1,3 +1,4 @@
+export * from './library'
export * from './player'
export * from './themes'
export * from './albumView'
diff --git a/ui/src/actions/library.js b/ui/src/actions/library.js
new file mode 100644
index 000000000..4653ec739
--- /dev/null
+++ b/ui/src/actions/library.js
@@ -0,0 +1,12 @@
+export const SET_SELECTED_LIBRARIES = 'SET_SELECTED_LIBRARIES'
+export const SET_USER_LIBRARIES = 'SET_USER_LIBRARIES'
+
+export const setSelectedLibraries = (libraryIds) => ({
+ type: SET_SELECTED_LIBRARIES,
+ data: libraryIds,
+})
+
+export const setUserLibraries = (libraries) => ({
+ type: SET_USER_LIBRARIES,
+ data: libraries,
+})
diff --git a/ui/src/actions/serverEvents.js b/ui/src/actions/serverEvents.js
index 7d89c5feb..995534550 100644
--- a/ui/src/actions/serverEvents.js
+++ b/ui/src/actions/serverEvents.js
@@ -1,6 +1,8 @@
export const EVENT_SCAN_STATUS = 'scanStatus'
export const EVENT_SERVER_START = 'serverStart'
export const EVENT_REFRESH_RESOURCE = 'refreshResource'
+export const EVENT_NOW_PLAYING_COUNT = 'nowPlayingCount'
+export const EVENT_STREAM_RECONNECTED = 'streamReconnected'
export const processEvent = (type, data) => ({
type,
@@ -11,7 +13,17 @@ export const scanStatusUpdate = (data) => ({
data: data,
})
+export const nowPlayingCountUpdate = (data) => ({
+ type: EVENT_NOW_PLAYING_COUNT,
+ data: data,
+})
+
export const serverDown = () => ({
type: EVENT_SERVER_START,
data: {},
})
+
+export const streamReconnected = () => ({
+ type: EVENT_STREAM_RECONNECTED,
+ data: {},
+})
diff --git a/ui/src/album/AlbumDatesField.jsx b/ui/src/album/AlbumDatesField.jsx
index e4cdeedce..ce1301380 100644
--- a/ui/src/album/AlbumDatesField.jsx
+++ b/ui/src/album/AlbumDatesField.jsx
@@ -10,6 +10,12 @@ export const AlbumDatesField = ({ className, ...rest }) => {
const releaseYear = releaseDate?.toString().substring(0, 4)
const yearRange =
formatRange(record, 'originalYear') || record['maxYear']?.toString()
+
+ // Don't show anything if the year starts with "0"
+ if (yearRange === '0' || releaseYear?.startsWith('0')) {
+ return null
+ }
+
let label = yearRange
if (releaseYear !== undefined && yearRange !== releaseYear) {
diff --git a/ui/src/album/AlbumDatesField.test.jsx b/ui/src/album/AlbumDatesField.test.jsx
new file mode 100644
index 000000000..9bcd41567
--- /dev/null
+++ b/ui/src/album/AlbumDatesField.test.jsx
@@ -0,0 +1,112 @@
+import { describe, test, expect, vi } from 'vitest'
+import { render } from '@testing-library/react'
+import { RecordContextProvider } from 'react-admin'
+import { AlbumDatesField } from './AlbumDatesField'
+import { formatRange } from '../common/index.js'
+
+// Mock the formatRange function
+vi.mock('../common/index.js', () => ({
+ formatRange: vi.fn(),
+}))
+
+describe('AlbumDatesField', () => {
+ test('renders nothing when yearRange is "0"', () => {
+ const record = {
+ maxYear: '0',
+ releaseDate: '2020-01-01',
+ }
+
+ vi.mocked(formatRange).mockReturnValue('0')
+
+ const { container } = render(
+
+
+ ,
+ )
+
+ expect(container.firstChild).toBeNull()
+ })
+
+ test('renders nothing when releaseYear is "0"', () => {
+ const record = {
+ maxYear: '2020',
+ releaseDate: '0-01-01',
+ }
+
+ vi.mocked(formatRange).mockReturnValue('2020')
+
+ const { container } = render(
+
+
+ ,
+ )
+
+ expect(container.firstChild).toBeNull()
+ })
+
+ test('renders only yearRange when releaseYear is undefined', () => {
+ const record = {
+ maxYear: '2020',
+ }
+
+ vi.mocked(formatRange).mockReturnValue('2020')
+
+ const { container } = render(
+
+
+ ,
+ )
+
+ expect(container.textContent).toBe('2020')
+ })
+
+ test('renders both years when they are different', () => {
+ const record = {
+ maxYear: '2018',
+ releaseDate: '2020-01-01',
+ }
+
+ vi.mocked(formatRange).mockReturnValue('2018')
+
+ const { container } = render(
+
+
+ ,
+ )
+
+ expect(container.textContent).toBe('♫ 2018 · ○ 2020')
+ })
+
+ test('renders only yearRange when both years are the same', () => {
+ const record = {
+ maxYear: '2020',
+ releaseDate: '2020-01-01',
+ }
+
+ vi.mocked(formatRange).mockReturnValue('2020')
+
+ const { container } = render(
+
+
+ ,
+ )
+
+ expect(container.textContent).toBe('2020')
+ })
+
+ test('applies className when provided', () => {
+ const record = {
+ maxYear: '2020',
+ }
+
+ vi.mocked(formatRange).mockReturnValue('2020')
+
+ const { container } = render(
+
+
+ ,
+ )
+
+ expect(container.firstChild).toHaveClass('test-class')
+ })
+})
diff --git a/ui/src/album/AlbumDetails.test.jsx b/ui/src/album/AlbumDetails.test.jsx
index e03022677..484045444 100644
--- a/ui/src/album/AlbumDetails.test.jsx
+++ b/ui/src/album/AlbumDetails.test.jsx
@@ -14,6 +14,24 @@ vi.mock('@material-ui/core', async () => {
}
})
+// Mock formatFullDate to return deterministic results
+vi.mock('../utils', async () => {
+ const actual = await import('../utils')
+ return {
+ ...actual,
+ formatFullDate: (date) => {
+ if (!date) return ''
+ // Use en-CA locale for consistent test results
+ return new Date(date).toLocaleDateString('en-CA', {
+ year: 'numeric',
+ month: 'short',
+ day: 'numeric',
+ timeZone: 'UTC',
+ })
+ },
+ }
+})
+
describe('Details component', () => {
describe('Desktop view', () => {
beforeEach(() => {
diff --git a/ui/src/album/AlbumInfo.jsx b/ui/src/album/AlbumInfo.jsx
index 453dbb167..e71cd3d33 100644
--- a/ui/src/album/AlbumInfo.jsx
+++ b/ui/src/album/AlbumInfo.jsx
@@ -38,6 +38,7 @@ const AlbumInfo = (props) => {
const record = useRecordContext(props)
const data = {
album: ,
+ libraryName: ,
albumArtist: (
),
diff --git a/ui/src/album/AlbumList.jsx b/ui/src/album/AlbumList.jsx
index 40b927a89..f10f8dbd3 100644
--- a/ui/src/album/AlbumList.jsx
+++ b/ui/src/album/AlbumList.jsx
@@ -42,6 +42,9 @@ const useStyles = makeStyles({
},
})
+const formatReleaseType = (record) =>
+ record?.tagValue ? humanize(record?.tagValue) : '-- None --'
+
const AlbumFilter = (props) => {
const classes = useStyles()
const translate = useTranslate()
@@ -142,9 +145,7 @@ const AlbumFilter = (props) => {
>
- record?.tagValue ? humanize(record?.tagValue) : '-- None --'
- }
+ optionText={formatReleaseType}
/>
diff --git a/ui/src/album/AlbumShow.jsx b/ui/src/album/AlbumShow.jsx
index 79aab6d6b..c9e944999 100644
--- a/ui/src/album/AlbumShow.jsx
+++ b/ui/src/album/AlbumShow.jsx
@@ -4,12 +4,13 @@ import {
ShowContextProvider,
useShowContext,
useShowController,
+ Title as RaTitle,
} from 'react-admin'
import { makeStyles } from '@material-ui/core/styles'
import AlbumSongs from './AlbumSongs'
import AlbumDetails from './AlbumDetails'
import AlbumActions from './AlbumActions'
-import { useResourceRefresh } from '../common'
+import { useResourceRefresh, Title } from '../common'
const useStyles = makeStyles(
(theme) => ({
@@ -30,6 +31,7 @@ const AlbumShowLayout = (props) => {
return (
<>
+ {record && } />}
{record && }
{record && (
({
+ toolbar: {
+ minHeight: 'auto',
+ padding: '0 !important',
+ background: 'transparent',
+ boxShadow: 'none',
+ '& .MuiToolbar-root': {
+ minHeight: 'auto',
+ padding: '0 !important',
+ background: 'transparent',
+ },
+ },
+ button: {
+ [theme.breakpoints.down('xs')]: {
+ minWidth: 'auto',
+ padding: '8px 12px',
+ fontSize: '0.75rem',
+ '& .MuiButton-startIcon': {
+ marginRight: '4px',
+ },
+ },
+ },
+ radioIcon: {
+ [theme.breakpoints.down('xs')]: {
+ fontSize: '1.5rem',
+ },
+ },
+}))
+
+const ArtistActions = ({ className, record, ...rest }) => {
+ const dispatch = useDispatch()
+ const translate = useTranslate()
+ const dataProvider = useDataProvider()
+ const notify = useNotify()
+ const classes = useStyles()
+ const isMobile = useMediaQuery((theme) => theme.breakpoints.down('xs'))
+
+ const handlePlay = React.useCallback(async () => {
+ try {
+ await playTopSongs(dispatch, notify, record.name)
+ } catch (e) {
+ // eslint-disable-next-line no-console
+ console.error('Error fetching top songs for artist:', e)
+ notify('ra.page.error', 'warning')
+ }
+ }, [dispatch, notify, record])
+
+ const handleShuffle = React.useCallback(async () => {
+ try {
+ await playShuffle(dataProvider, dispatch, record.id)
+ } catch (e) {
+ // eslint-disable-next-line no-console
+ console.error('Error fetching songs for shuffle:', e)
+ notify('ra.page.error', 'warning')
+ }
+ }, [dataProvider, dispatch, record, notify])
+
+ const handleRadio = React.useCallback(async () => {
+ try {
+ await playSimilar(dispatch, notify, record.id)
+ } catch (e) {
+ // eslint-disable-next-line no-console
+ console.error('Error starting radio for artist:', e)
+ notify('ra.page.error', 'warning')
+ }
+ }, [dispatch, notify, record])
+
+ return (
+
+
+
+
+
+
+
+
+
+
+
+ )
+}
+
+ArtistActions.propTypes = {
+ className: PropTypes.string,
+ record: PropTypes.object.isRequired,
+}
+
+ArtistActions.defaultProps = {
+ className: '',
+}
+
+export default ArtistActions
diff --git a/ui/src/artist/ArtistActions.test.jsx b/ui/src/artist/ArtistActions.test.jsx
new file mode 100644
index 000000000..a11ee50e3
--- /dev/null
+++ b/ui/src/artist/ArtistActions.test.jsx
@@ -0,0 +1,230 @@
+import React from 'react'
+import { render, fireEvent, waitFor, screen } from '@testing-library/react'
+import { TestContext } from 'ra-test'
+import { describe, it, expect, vi, beforeEach } from 'vitest'
+import ArtistActions from './ArtistActions'
+import subsonic from '../subsonic'
+import { ThemeProvider, createTheme } from '@material-ui/core/styles'
+
+const mockDispatch = vi.fn()
+vi.mock('react-redux', () => ({ useDispatch: () => mockDispatch }))
+
+vi.mock('../subsonic', () => ({
+ default: { getSimilarSongs2: vi.fn(), getTopSongs: vi.fn() },
+}))
+
+const mockNotify = vi.fn()
+const mockGetList = vi.fn().mockResolvedValue({ data: [{ id: 's1' }] })
+
+vi.mock('react-admin', async (importOriginal) => {
+ const actual = await importOriginal()
+ return {
+ ...actual,
+ useNotify: () => mockNotify,
+ useDataProvider: () => ({ getList: mockGetList }),
+ useTranslate: () => (x) => x,
+ }
+})
+
+describe('ArtistActions', () => {
+ const defaultRecord = { id: 'ar1', name: 'Artist' }
+
+ const renderArtistActions = (record = defaultRecord) => {
+ const theme = createTheme()
+ return render(
+
+
+
+
+ ,
+ )
+ }
+
+ const clickActionButton = (actionKey) => {
+ fireEvent.click(screen.getByText(`resources.artist.actions.${actionKey}`))
+ }
+
+ beforeEach(() => {
+ vi.clearAllMocks()
+ // Mock console.error to suppress error logging in tests
+ vi.spyOn(console, 'error').mockImplementation(() => {})
+
+ const songWithReplayGain = {
+ id: 'rec1',
+ replayGain: {
+ albumGain: -5,
+ albumPeak: 1,
+ trackGain: -6,
+ trackPeak: 0.8,
+ },
+ }
+
+ subsonic.getSimilarSongs2.mockResolvedValue({
+ json: {
+ 'subsonic-response': {
+ status: 'ok',
+ similarSongs2: { song: [songWithReplayGain] },
+ },
+ },
+ })
+ subsonic.getTopSongs.mockResolvedValue({
+ json: {
+ 'subsonic-response': {
+ status: 'ok',
+ topSongs: { song: [songWithReplayGain] },
+ },
+ },
+ })
+ })
+
+ describe('Shuffle action', () => {
+ it('shuffles songs when clicked', async () => {
+ renderArtistActions()
+ clickActionButton('shuffle')
+
+ await waitFor(() =>
+ expect(mockGetList).toHaveBeenCalledWith('song', {
+ pagination: { page: 1, perPage: 500 },
+ sort: { field: 'random', order: 'ASC' },
+ filter: { album_artist_id: 'ar1', missing: false },
+ }),
+ )
+ expect(mockDispatch).toHaveBeenCalled()
+ })
+ })
+
+ describe('Radio action', () => {
+ it('starts radio when clicked', async () => {
+ renderArtistActions()
+ clickActionButton('radio')
+
+ await waitFor(() =>
+ expect(subsonic.getSimilarSongs2).toHaveBeenCalledWith('ar1', 100),
+ )
+ expect(mockDispatch).toHaveBeenCalled()
+ })
+
+ it('maps replaygain info', async () => {
+ renderArtistActions()
+ clickActionButton('radio')
+
+ await waitFor(() =>
+ expect(subsonic.getSimilarSongs2).toHaveBeenCalledWith('ar1', 100),
+ )
+ const action = mockDispatch.mock.calls[0][0]
+ expect(action.data.rec1).toMatchObject({
+ rgAlbumGain: -5,
+ rgAlbumPeak: 1,
+ rgTrackGain: -6,
+ rgTrackPeak: 0.8,
+ })
+ })
+ })
+
+ describe('Play action', () => {
+ it('plays top songs when clicked', async () => {
+ renderArtistActions()
+ clickActionButton('topSongs')
+
+ await waitFor(() =>
+ expect(subsonic.getTopSongs).toHaveBeenCalledWith('Artist', 100),
+ )
+ expect(mockDispatch).toHaveBeenCalled()
+ })
+
+ it('maps replaygain info for top songs', async () => {
+ renderArtistActions()
+ clickActionButton('topSongs')
+
+ await waitFor(() =>
+ expect(subsonic.getTopSongs).toHaveBeenCalledWith('Artist', 100),
+ )
+ const action = mockDispatch.mock.calls[0][0]
+ expect(action.data.rec1).toMatchObject({
+ rgAlbumGain: -5,
+ rgAlbumPeak: 1,
+ rgTrackGain: -6,
+ rgTrackPeak: 0.8,
+ })
+ })
+
+ it('handles API rejection', async () => {
+ subsonic.getTopSongs.mockRejectedValue(new Error('Network error'))
+
+ renderArtistActions()
+ clickActionButton('topSongs')
+
+ await waitFor(() =>
+ expect(subsonic.getTopSongs).toHaveBeenCalledWith('Artist', 100),
+ )
+ expect(mockNotify).toHaveBeenCalledWith('ra.page.error', 'warning')
+ expect(mockDispatch).not.toHaveBeenCalled()
+ })
+
+ it('handles failed API response', async () => {
+ subsonic.getTopSongs.mockResolvedValue({
+ json: {
+ 'subsonic-response': {
+ status: 'failed',
+ error: { code: 40, message: 'Wrong username or password' },
+ },
+ },
+ })
+
+ renderArtistActions()
+ clickActionButton('topSongs')
+
+ await waitFor(() =>
+ expect(subsonic.getTopSongs).toHaveBeenCalledWith('Artist', 100),
+ )
+ expect(mockNotify).toHaveBeenCalledWith('ra.page.error', 'warning')
+ expect(mockDispatch).not.toHaveBeenCalled()
+ })
+
+ it('handles empty song list', async () => {
+ subsonic.getTopSongs.mockResolvedValue({
+ json: {
+ 'subsonic-response': {
+ status: 'ok',
+ topSongs: { song: [] },
+ },
+ },
+ })
+
+ renderArtistActions()
+ clickActionButton('topSongs')
+
+ await waitFor(() =>
+ expect(subsonic.getTopSongs).toHaveBeenCalledWith('Artist', 100),
+ )
+ expect(mockNotify).toHaveBeenCalledWith(
+ 'message.noTopSongsFound',
+ 'warning',
+ )
+ expect(mockDispatch).not.toHaveBeenCalled()
+ })
+
+ it('handles missing topSongs property', async () => {
+ subsonic.getTopSongs.mockResolvedValue({
+ json: {
+ 'subsonic-response': {
+ status: 'ok',
+ // topSongs property is missing
+ },
+ },
+ })
+
+ renderArtistActions()
+ clickActionButton('topSongs')
+
+ await waitFor(() =>
+ expect(subsonic.getTopSongs).toHaveBeenCalledWith('Artist', 100),
+ )
+ expect(mockNotify).toHaveBeenCalledWith(
+ 'message.noTopSongsFound',
+ 'warning',
+ )
+ expect(mockDispatch).not.toHaveBeenCalled()
+ })
+ })
+})
diff --git a/ui/src/artist/ArtistList.jsx b/ui/src/artist/ArtistList.jsx
index 7a14e9efe..e175763e3 100644
--- a/ui/src/artist/ArtistList.jsx
+++ b/ui/src/artist/ArtistList.jsx
@@ -132,8 +132,10 @@ const ArtistListView = ({ hasShow, hasEdit, hasList, width, ...rest }) => {
useResourceRefresh('artist')
const role = filterValues?.role
- const getCounter = (record, counter) =>
- role ? record?.stats[role]?.[counter] : record?.[counter]
+ const getCounter = (record, counter) => {
+ if (!record) return undefined
+ return role ? record?.stats?.[role]?.[counter] : record?.[counter]
+ }
const getAlbumCount = (record) => getCounter(record, 'albumCount')
const getSongCount = (record) => getCounter(record, 'songCount')
const getSize = (record) => {
diff --git a/ui/src/artist/ArtistShow.jsx b/ui/src/artist/ArtistShow.jsx
index 86f82bf35..c6dc832c1 100644
--- a/ui/src/artist/ArtistShow.jsx
+++ b/ui/src/artist/ArtistShow.jsx
@@ -7,12 +7,46 @@ import {
useShowContext,
ReferenceManyField,
Pagination,
+ Title as RaTitle,
} from 'react-admin'
import subsonic from '../subsonic'
import AlbumGridView from '../album/AlbumGridView'
import MobileArtistDetails from './MobileArtistDetails'
import DesktopArtistDetails from './DesktopArtistDetails'
-import { useAlbumsPerPage, useResourceRefresh } from '../common/index.js'
+import { useAlbumsPerPage, useResourceRefresh, Title } from '../common/index.js'
+import ArtistActions from './ArtistActions'
+import { makeStyles } from '@material-ui/core'
+
+const useStyles = makeStyles(
+ (theme) => ({
+ actions: {
+ width: '100%',
+ justifyContent: 'flex-start',
+ display: 'flex',
+ paddingTop: '0.25em',
+ paddingBottom: '0.25em',
+ paddingLeft: '1em',
+ paddingRight: '1em',
+ flexWrap: 'wrap',
+ overflowX: 'auto',
+ [theme.breakpoints.down('xs')]: {
+ paddingLeft: '0.5em',
+ paddingRight: '0.5em',
+ gap: '0.5em',
+ justifyContent: 'space-around',
+ },
+ },
+ actionsContainer: {
+ paddingLeft: '.75rem',
+ [theme.breakpoints.down('xs')]: {
+ padding: '.5rem',
+ },
+ },
+ }),
+ {
+ name: 'NDArtistShow',
+ },
+)
const ArtistDetails = (props) => {
const record = useRecordContext(props)
@@ -55,16 +89,17 @@ const ArtistShowLayout = (props) => {
const record = useRecordContext()
const { width } = props
const [, perPageOptions] = useAlbumsPerPage(width)
+ const classes = useStyles()
useResourceRefresh('artist', 'album')
const maxPerPage = 90
let perPage = 0
let pagination = null
- const count = Math.max(
- record?.stats?.['albumartist']?.albumCount || 0,
- record?.stats?.['artist']?.albumCount ?? 0,
- )
+ // Use the main credit count instead of total count, as this is a precise measure
+ // of the number of albums where the artist is credited as an album artist OR
+ // artist
+ const count = record?.stats?.['maincredit']?.albumCount || 0
if (count > maxPerPage) {
perPage = Math.trunc(maxPerPage / perPageOptions[0]) * perPageOptions[0]
@@ -76,7 +111,13 @@ const ArtistShowLayout = (props) => {
return (
<>
+ {record && } />}
{record && }
+ {record && (
+
+ )}
{record && (
{
+ const { replayGain: rg } = song
+ if (!rg) {
+ return song
+ }
+
+ return {
+ ...song,
+ ...(rg.albumGain !== undefined && { rgAlbumGain: rg.albumGain }),
+ ...(rg.albumPeak !== undefined && { rgAlbumPeak: rg.albumPeak }),
+ ...(rg.trackGain !== undefined && { rgTrackGain: rg.trackGain }),
+ ...(rg.trackPeak !== undefined && { rgTrackPeak: rg.trackPeak }),
+ }
+}
+
+const processSongsForPlayback = (songs) => {
+ const songData = {}
+ const ids = []
+ songs.forEach((s) => {
+ const song = mapReplayGain(s)
+ songData[song.id] = song
+ ids.push(song.id)
+ })
+ return { songData, ids }
+}
+
+export const playTopSongs = async (dispatch, notify, artistName) => {
+ const res = await subsonic.getTopSongs(artistName, 100)
+ const data = res.json['subsonic-response']
+
+ if (data.status !== 'ok') {
+ throw new Error(
+ `Error fetching top songs: ${data.error?.message || 'Unknown error'} (Code: ${data.error?.code || 'unknown'})`,
+ )
+ }
+
+ const songs = data.topSongs?.song || []
+ if (!songs.length) {
+ notify('message.noTopSongsFound', 'warning')
+ return
+ }
+
+ const { songData, ids } = processSongsForPlayback(songs)
+ dispatch(playTracks(songData, ids))
+}
+
+export const playSimilar = async (dispatch, notify, id) => {
+ const res = await subsonic.getSimilarSongs2(id, 100)
+ const data = res.json['subsonic-response']
+
+ if (data.status !== 'ok') {
+ throw new Error(
+ `Error fetching similar songs: ${data.error?.message || 'Unknown error'} (Code: ${data.error?.code || 'unknown'})`,
+ )
+ }
+
+ const songs = data.similarSongs2?.song || []
+ if (!songs.length) {
+ notify('message.noSimilarSongsFound', 'warning')
+ return
+ }
+
+ const { songData, ids } = processSongsForPlayback(songs)
+ dispatch(playTracks(songData, ids))
+}
+
+export const playShuffle = async (dataProvider, dispatch, id) => {
+ const res = await dataProvider.getList('song', {
+ pagination: { page: 1, perPage: 500 },
+ sort: { field: 'random', order: 'ASC' },
+ filter: { album_artist_id: id, missing: false },
+ })
+
+ const data = {}
+ const ids = []
+ res.data.forEach((s) => {
+ data[s.id] = s
+ ids.push(s.id)
+ })
+ dispatch(playTracks(data, ids))
+}
diff --git a/ui/src/audioplayer/AudioTitle.jsx b/ui/src/audioplayer/AudioTitle.jsx
index aebd37170..093bb53fb 100644
--- a/ui/src/audioplayer/AudioTitle.jsx
+++ b/ui/src/audioplayer/AudioTitle.jsx
@@ -38,16 +38,14 @@ const AudioTitle = React.memo(({ audioInfo, gainInfo, isMobile }) => {
const subtitle = song.tags?.['subtitle']
const title = song.title + (subtitle ? ` (${subtitle})` : '')
+ const linkTo = audioInfo.isRadio
+ ? `/radio/${audioInfo.trackId}/show`
+ : song.playlistId
+ ? `/playlist/${song.playlistId}/show`
+ : `/album/${song.albumId}/show`
+
return (
-
+
{title}
{isDesktop && (
diff --git a/ui/src/audioplayer/AudioTitle.test.jsx b/ui/src/audioplayer/AudioTitle.test.jsx
new file mode 100644
index 000000000..7b297c07e
--- /dev/null
+++ b/ui/src/audioplayer/AudioTitle.test.jsx
@@ -0,0 +1,58 @@
+import React from 'react'
+import { render, screen } from '@testing-library/react'
+import { describe, it, expect, beforeEach, vi } from 'vitest'
+import AudioTitle from './AudioTitle'
+
+vi.mock('@material-ui/core', async () => {
+ const actual = await import('@material-ui/core')
+ return {
+ ...actual,
+ useMediaQuery: vi.fn(),
+ }
+})
+
+vi.mock('react-router-dom', () => ({
+ // eslint-disable-next-line react/display-name
+ Link: React.forwardRef(({ to, children, ...props }, ref) => (
+
+ {children}
+
+ )),
+}))
+
+vi.mock('react-dnd', () => ({
+ useDrag: vi.fn(() => [null, () => {}]),
+}))
+
+describe(' ', () => {
+ const baseSong = {
+ id: 'song-1',
+ albumId: 'album-1',
+ playlistId: 'playlist-1',
+ title: 'Test Song',
+ artist: 'Artist',
+ album: 'Album',
+ year: '2020',
+ }
+
+ beforeEach(() => {
+ vi.clearAllMocks()
+ })
+
+ it('links to playlist when playlistId is provided', () => {
+ const audioInfo = { trackId: 'track-1', song: baseSong }
+ render( )
+ const link = screen.getByRole('link')
+ expect(link.getAttribute('href')).toBe('/playlist/playlist-1/show')
+ })
+
+ it('falls back to album link when no playlistId', () => {
+ const audioInfo = {
+ trackId: 'track-1',
+ song: { ...baseSong, playlistId: undefined },
+ }
+ render( )
+ const link = screen.getByRole('link')
+ expect(link.getAttribute('href')).toBe('/album/album-1/show')
+ })
+})
diff --git a/ui/src/audioplayer/Player.jsx b/ui/src/audioplayer/Player.jsx
index 1f57737d0..03419add3 100644
--- a/ui/src/audioplayer/Player.jsx
+++ b/ui/src/audioplayer/Player.jsx
@@ -127,6 +127,7 @@ const Player = () => {
/>
),
locale: locale(translate),
+ sortableOptions: { delay: 200, delayOnTouchOnly: true },
}),
[gainInfo, isDesktop, playerTheme, translate, playerState.mode],
)
@@ -214,7 +215,8 @@ const Player = () => {
const song = info.song
document.title = `${song.title} - ${song.artist} - Navidrome`
if (!info.isRadio) {
- subsonic.nowPlaying(info.trackId)
+ const pos = startTime === null ? null : Math.floor(info.currentTime)
+ subsonic.nowPlaying(info.trackId, pos)
}
setPreload(false)
if (config.gaTrackingId) {
diff --git a/ui/src/audioplayer/PlayerToolbar.jsx b/ui/src/audioplayer/PlayerToolbar.jsx
index 5230b30f2..4812141ab 100644
--- a/ui/src/audioplayer/PlayerToolbar.jsx
+++ b/ui/src/audioplayer/PlayerToolbar.jsx
@@ -57,7 +57,7 @@ const useStyles = makeStyles((theme) => ({
const PlayerToolbar = ({ id, isRadio }) => {
const dispatch = useDispatch()
- const { data, loading } = useGetOne('song', id, { enabled: !!id })
+ const { data, loading } = useGetOne('song', id, { enabled: !!id && !isRadio })
const [toggleLove, toggling] = useToggleLove('song', data)
const isDesktop = useMediaQuery('(min-width:810px)')
const classes = useStyles()
diff --git a/ui/src/common/LibrarySelector.jsx b/ui/src/common/LibrarySelector.jsx
new file mode 100644
index 000000000..1e89d3ec6
--- /dev/null
+++ b/ui/src/common/LibrarySelector.jsx
@@ -0,0 +1,221 @@
+import React, { useState, useEffect, useCallback } from 'react'
+import { useDispatch, useSelector } from 'react-redux'
+import { useDataProvider, useTranslate, useRefresh } from 'react-admin'
+import {
+ Box,
+ Chip,
+ ClickAwayListener,
+ FormControl,
+ FormGroup,
+ FormControlLabel,
+ Checkbox,
+ Typography,
+ Paper,
+ Popper,
+ makeStyles,
+} from '@material-ui/core'
+import { ExpandMore, ExpandLess, LibraryMusic } from '@material-ui/icons'
+import { setSelectedLibraries, setUserLibraries } from '../actions'
+import { useRefreshOnEvents } from './useRefreshOnEvents'
+
+const useStyles = makeStyles((theme) => ({
+ root: {
+ marginTop: theme.spacing(3),
+ marginBottom: theme.spacing(3),
+ paddingLeft: theme.spacing(2),
+ paddingRight: theme.spacing(2),
+ display: 'flex',
+ justifyContent: 'center',
+ },
+ chip: {
+ borderRadius: theme.spacing(1),
+ height: theme.spacing(4.8),
+ fontSize: '1rem',
+ fontWeight: 'normal',
+ minWidth: '210px',
+ justifyContent: 'flex-start',
+ paddingLeft: theme.spacing(1),
+ paddingRight: theme.spacing(1),
+ marginTop: theme.spacing(0.1),
+ '& .MuiChip-label': {
+ paddingLeft: theme.spacing(2),
+ paddingRight: theme.spacing(1),
+ },
+ '& .MuiChip-icon': {
+ fontSize: '1.2rem',
+ marginLeft: theme.spacing(0.5),
+ },
+ },
+ popper: {
+ zIndex: 1300,
+ },
+ paper: {
+ padding: theme.spacing(2),
+ marginTop: theme.spacing(1),
+ minWidth: 300,
+ maxWidth: 400,
+ },
+ headerContainer: {
+ display: 'flex',
+ alignItems: 'center',
+ marginBottom: 0,
+ },
+ masterCheckbox: {
+ padding: '7px',
+ marginLeft: '-9px',
+ marginRight: 0,
+ },
+}))
+
+const LibrarySelector = () => {
+ const classes = useStyles()
+ const dispatch = useDispatch()
+ const dataProvider = useDataProvider()
+ const translate = useTranslate()
+ const refresh = useRefresh()
+ const [anchorEl, setAnchorEl] = useState(null)
+ const [open, setOpen] = useState(false)
+
+ const { userLibraries, selectedLibraries } = useSelector(
+ (state) => state.library,
+ )
+
+ // Load user's libraries when component mounts
+ const loadUserLibraries = useCallback(async () => {
+ const userId = localStorage.getItem('userId')
+ if (userId) {
+ try {
+ const { data } = await dataProvider.getOne('user', { id: userId })
+ const libraries = data.libraries || []
+ dispatch(setUserLibraries(libraries))
+ } catch (error) {
+ // eslint-disable-next-line no-console
+ console.warn(
+ 'Could not load user libraries (this may be expected for non-admin users):',
+ error,
+ )
+ }
+ }
+ }, [dataProvider, dispatch])
+
+ // Initial load
+ useEffect(() => {
+ loadUserLibraries()
+ }, [loadUserLibraries])
+
+ // Reload user libraries when library changes occur
+ useRefreshOnEvents({
+ events: ['library', 'user'],
+ onRefresh: loadUserLibraries,
+ })
+
+ // Don't render if user has no libraries or only has one library
+ if (!userLibraries.length || userLibraries.length === 1) {
+ return null
+ }
+
+ const handleToggle = (event) => {
+ setAnchorEl(event.currentTarget)
+ const wasOpen = open
+ setOpen(!open)
+ // Refresh data when closing the dropdown
+ if (wasOpen) {
+ refresh()
+ }
+ }
+
+ const handleClose = () => {
+ setOpen(false)
+ refresh()
+ }
+
+ const handleLibraryToggle = (libraryId) => {
+ const newSelection = selectedLibraries.includes(libraryId)
+ ? selectedLibraries.filter((id) => id !== libraryId)
+ : [...selectedLibraries, libraryId]
+
+ dispatch(setSelectedLibraries(newSelection))
+ }
+
+ const handleMasterCheckboxChange = () => {
+ if (isAllSelected) {
+ dispatch(setSelectedLibraries([]))
+ } else {
+ const allIds = userLibraries.map((lib) => lib.id)
+ dispatch(setSelectedLibraries(allIds))
+ }
+ }
+
+ const selectedCount = selectedLibraries.length
+ const totalCount = userLibraries.length
+ const isAllSelected = selectedCount === totalCount
+ const isNoneSelected = selectedCount === 0
+ const isIndeterminate = selectedCount > 0 && selectedCount < totalCount
+
+ const displayText = isNoneSelected
+ ? translate('menu.librarySelector.none') + ` (0 of ${totalCount})`
+ : isAllSelected
+ ? translate('menu.librarySelector.allLibraries', { count: totalCount })
+ : translate('menu.librarySelector.multipleLibraries', {
+ selected: selectedCount,
+ total: totalCount,
+ })
+
+ return (
+
+ }
+ label={displayText}
+ onClick={handleToggle}
+ onDelete={open ? handleToggle : undefined}
+ deleteIcon={open ? : }
+ variant="outlined"
+ className={classes.chip}
+ />
+
+
+
+
+
+
+
+ {translate('menu.librarySelector.selectLibraries')}:
+
+
+
+
+
+ {userLibraries.map((library) => (
+ handleLibraryToggle(library.id)}
+ size="small"
+ />
+ }
+ label={library.name}
+ />
+ ))}
+
+
+
+
+
+
+ )
+}
+
+export default LibrarySelector
diff --git a/ui/src/common/LibrarySelector.test.jsx b/ui/src/common/LibrarySelector.test.jsx
new file mode 100644
index 000000000..13b607887
--- /dev/null
+++ b/ui/src/common/LibrarySelector.test.jsx
@@ -0,0 +1,517 @@
+import React from 'react'
+import { render, screen, fireEvent, waitFor } from '@testing-library/react'
+import userEvent from '@testing-library/user-event'
+import { describe, it, expect, beforeEach, vi } from 'vitest'
+import LibrarySelector from './LibrarySelector'
+
+// Mock dependencies
+const mockDispatch = vi.fn()
+const mockDataProvider = {
+ getOne: vi.fn(),
+}
+const mockIdentity = { username: 'testuser' }
+const mockRefresh = vi.fn()
+const mockTranslate = vi.fn((key, options = {}) => {
+ const translations = {
+ 'menu.librarySelector.allLibraries': `All Libraries (${options.count || 0})`,
+ 'menu.librarySelector.multipleLibraries': `${options.selected || 0} of ${options.total || 0} Libraries`,
+ 'menu.librarySelector.none': 'None',
+ 'menu.librarySelector.selectLibraries': 'Select Libraries',
+ }
+ return translations[key] || key
+})
+
+vi.mock('react-redux', () => ({
+ useDispatch: () => mockDispatch,
+ useSelector: vi.fn(),
+}))
+
+vi.mock('react-admin', () => ({
+ useDataProvider: () => mockDataProvider,
+ useGetIdentity: () => ({ identity: mockIdentity }),
+ useTranslate: () => mockTranslate,
+ useRefresh: () => mockRefresh,
+}))
+
+// Mock Material-UI components
+vi.mock('@material-ui/core', () => ({
+ Box: ({ children, className, ...props }) => (
+
+ {children}
+
+ ),
+ Chip: ({ label, onClick, onDelete, deleteIcon, icon, ...props }) => (
+
+ {icon}
+ {label}
+ {deleteIcon && {deleteIcon} }
+
+ ),
+ ClickAwayListener: ({ children, onClickAway }) => (
+
+ {children}
+
+ ),
+ Collapse: ({ children, in: inProp }) =>
+ inProp ? {children}
: null,
+ FormControl: ({ children }) => {children}
,
+ FormGroup: ({ children }) => {children}
,
+ FormControlLabel: ({ control, label }) => (
+
+ {control}
+ {label}
+
+ ),
+ Checkbox: ({
+ checked,
+ indeterminate,
+ onChange,
+ size,
+ className,
+ ...props
+ }) => (
+ {
+ if (el) el.indeterminate = indeterminate
+ }}
+ onChange={onChange}
+ className={className}
+ {...props}
+ />
+ ),
+ Typography: ({ children, variant, ...props }) => (
+ {children}
+ ),
+ Paper: ({ children, className }) => (
+ {children}
+ ),
+ Popper: ({ open, children, anchorEl, placement, className }) =>
+ open ? (
+
+ {children}
+
+ ) : null,
+ makeStyles: (styles) => () => {
+ if (typeof styles === 'function') {
+ return styles({
+ spacing: (value) => `${value * 8}px`,
+ palette: { divider: '#ccc' },
+ shape: { borderRadius: 4 },
+ })
+ }
+ return styles
+ },
+}))
+
+vi.mock('@material-ui/icons', () => ({
+ ExpandMore: () => ▼ ,
+ ExpandLess: () => ▲ ,
+ LibraryMusic: () => 🎵 ,
+}))
+
+// Mock actions
+vi.mock('../actions', () => ({
+ setSelectedLibraries: (libraries) => ({
+ type: 'SET_SELECTED_LIBRARIES',
+ data: libraries,
+ }),
+ setUserLibraries: (libraries) => ({
+ type: 'SET_USER_LIBRARIES',
+ data: libraries,
+ }),
+}))
+
+describe('LibrarySelector', () => {
+ const mockLibraries = [
+ { id: '1', name: 'Music Library', path: '/music' },
+ { id: '2', name: 'Podcasts', path: '/podcasts' },
+ { id: '3', name: 'Audiobooks', path: '/audiobooks' },
+ ]
+
+ const defaultState = {
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1'],
+ }
+
+ let mockUseSelector
+
+ beforeEach(async () => {
+ vi.clearAllMocks()
+ const { useSelector } = await import('react-redux')
+ mockUseSelector = vi.mocked(useSelector)
+ mockDataProvider.getOne.mockResolvedValue({
+ data: { libraries: mockLibraries },
+ })
+ // Setup localStorage mock
+ Object.defineProperty(window, 'localStorage', {
+ value: {
+ getItem: vi.fn(() => null), // Default to null to prevent API calls
+ setItem: vi.fn(),
+ },
+ writable: true,
+ })
+ })
+
+ const renderLibrarySelector = (selectorState = defaultState) => {
+ mockUseSelector.mockImplementation((selector) =>
+ selector({ library: selectorState }),
+ )
+
+ return render( )
+ }
+
+ describe('when user has no libraries', () => {
+ it('should not render anything', () => {
+ const { container } = renderLibrarySelector({
+ userLibraries: [],
+ selectedLibraries: [],
+ })
+ expect(container.firstChild).toBeNull()
+ })
+ })
+
+ describe('when user has only one library', () => {
+ it('should not render anything', () => {
+ const singleLibrary = [mockLibraries[0]]
+ const { container } = renderLibrarySelector({
+ userLibraries: singleLibrary,
+ selectedLibraries: ['1'],
+ })
+ expect(container.firstChild).toBeNull()
+ })
+ })
+
+ describe('when user has multiple libraries', () => {
+ it('should render the chip with correct label when one library is selected', () => {
+ renderLibrarySelector()
+
+ expect(screen.getByRole('button')).toBeInTheDocument()
+ expect(screen.getByText('1 of 3 Libraries')).toBeInTheDocument()
+ expect(screen.getByTestId('library-music')).toBeInTheDocument()
+ expect(screen.getByTestId('expand-more')).toBeInTheDocument()
+ })
+
+ it('should render the chip with "All Libraries" when all libraries are selected', () => {
+ renderLibrarySelector({
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1', '2', '3'],
+ })
+
+ expect(screen.getByText('All Libraries (3)')).toBeInTheDocument()
+ })
+
+ it('should render the chip with "None" when no libraries are selected', () => {
+ renderLibrarySelector({
+ userLibraries: mockLibraries,
+ selectedLibraries: [],
+ })
+
+ expect(screen.getByText('None (0 of 3)')).toBeInTheDocument()
+ })
+
+ it('should show expand less icon when dropdown is open', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector()
+
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ expect(screen.getByTestId('expand-less')).toBeInTheDocument()
+ })
+
+ it('should open dropdown when chip is clicked', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector()
+
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ expect(screen.getByTestId('popper')).toBeInTheDocument()
+ expect(screen.getByText('Select Libraries:')).toBeInTheDocument()
+ })
+
+ it('should display all library names in dropdown', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector()
+
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ expect(screen.getByText('Music Library')).toBeInTheDocument()
+ expect(screen.getByText('Podcasts')).toBeInTheDocument()
+ expect(screen.getByText('Audiobooks')).toBeInTheDocument()
+ })
+
+ it('should not display library paths', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector()
+
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ expect(screen.queryByText('/music')).not.toBeInTheDocument()
+ expect(screen.queryByText('/podcasts')).not.toBeInTheDocument()
+ expect(screen.queryByText('/audiobooks')).not.toBeInTheDocument()
+ })
+
+ describe('master checkbox', () => {
+ it('should be checked when all libraries are selected', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector({
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1', '2', '3'],
+ })
+
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ const checkboxes = screen.getAllByRole('checkbox')
+ const masterCheckbox = checkboxes[0] // First checkbox is the master checkbox
+ expect(masterCheckbox.checked).toBe(true)
+ expect(masterCheckbox.indeterminate).toBe(false)
+ })
+
+ it('should be unchecked when no libraries are selected', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector({
+ userLibraries: mockLibraries,
+ selectedLibraries: [],
+ })
+
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ const checkboxes = screen.getAllByRole('checkbox')
+ const masterCheckbox = checkboxes[0]
+ expect(masterCheckbox.checked).toBe(false)
+ expect(masterCheckbox.indeterminate).toBe(false)
+ })
+
+ it('should be indeterminate when some libraries are selected', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector({
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1', '2'],
+ })
+
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ const checkboxes = screen.getAllByRole('checkbox')
+ const masterCheckbox = checkboxes[0]
+ expect(masterCheckbox.checked).toBe(false)
+ expect(masterCheckbox.indeterminate).toBe(true)
+ })
+
+ it('should select all libraries when clicked and none are selected', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector({
+ userLibraries: mockLibraries,
+ selectedLibraries: [],
+ })
+
+ // Clear the dispatch mock after initial mount (it sets user libraries)
+ mockDispatch.mockClear()
+
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ const checkboxes = screen.getAllByRole('checkbox')
+ const masterCheckbox = checkboxes[0]
+
+ // Use fireEvent.click to trigger the onChange event
+ fireEvent.click(masterCheckbox)
+
+ expect(mockDispatch).toHaveBeenCalledWith({
+ type: 'SET_SELECTED_LIBRARIES',
+ data: ['1', '2', '3'],
+ })
+ })
+
+ it('should deselect all libraries when clicked and all are selected', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector({
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1', '2', '3'],
+ })
+
+ // Clear the dispatch mock after initial mount (it sets user libraries)
+ mockDispatch.mockClear()
+
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ const checkboxes = screen.getAllByRole('checkbox')
+ const masterCheckbox = checkboxes[0]
+
+ fireEvent.click(masterCheckbox)
+
+ expect(mockDispatch).toHaveBeenCalledWith({
+ type: 'SET_SELECTED_LIBRARIES',
+ data: [],
+ })
+ })
+
+ it('should select all libraries when clicked and some are selected', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector({
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1'],
+ })
+
+ // Clear the dispatch mock after initial mount (it sets user libraries)
+ mockDispatch.mockClear()
+
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ const checkboxes = screen.getAllByRole('checkbox')
+ const masterCheckbox = checkboxes[0]
+
+ fireEvent.click(masterCheckbox)
+
+ expect(mockDispatch).toHaveBeenCalledWith({
+ type: 'SET_SELECTED_LIBRARIES',
+ data: ['1', '2', '3'],
+ })
+ })
+ })
+
+ describe('individual library checkboxes', () => {
+ it('should show correct checked state for each library', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector({
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1', '3'],
+ })
+
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ const checkboxes = screen.getAllByRole('checkbox')
+ // Skip master checkbox (index 0)
+ expect(checkboxes[1].checked).toBe(true) // Music Library
+ expect(checkboxes[2].checked).toBe(false) // Podcasts
+ expect(checkboxes[3].checked).toBe(true) // Audiobooks
+ })
+
+ it('should toggle library selection when individual checkbox is clicked', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector()
+
+ // Clear the dispatch mock after initial mount (it sets user libraries)
+ mockDispatch.mockClear()
+
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ const checkboxes = screen.getAllByRole('checkbox')
+ const podcastsCheckbox = checkboxes[2] // Podcasts checkbox
+
+ fireEvent.click(podcastsCheckbox)
+
+ expect(mockDispatch).toHaveBeenCalledWith({
+ type: 'SET_SELECTED_LIBRARIES',
+ data: ['1', '2'],
+ })
+ })
+
+ it('should remove library from selection when clicking checked library', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector({
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1', '2'],
+ })
+
+ // Clear the dispatch mock after initial mount (it sets user libraries)
+ mockDispatch.mockClear()
+
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ const checkboxes = screen.getAllByRole('checkbox')
+ const musicCheckbox = checkboxes[1] // Music Library checkbox
+
+ fireEvent.click(musicCheckbox)
+
+ expect(mockDispatch).toHaveBeenCalledWith({
+ type: 'SET_SELECTED_LIBRARIES',
+ data: ['2'],
+ })
+ })
+ })
+
+ it('should close dropdown when clicking away', async () => {
+ const user = userEvent.setup()
+ renderLibrarySelector()
+
+ // Open dropdown
+ const chipButton = screen.getByRole('button')
+ await user.click(chipButton)
+
+ expect(screen.getByTestId('popper')).toBeInTheDocument()
+
+ // Click away
+ const clickAwayListener = screen.getByTestId('click-away-listener')
+ fireEvent.mouseDown(clickAwayListener)
+
+ await waitFor(() => {
+ expect(screen.queryByTestId('popper')).not.toBeInTheDocument()
+ })
+
+ // Should trigger refresh when closing
+ expect(mockRefresh).toHaveBeenCalledTimes(1)
+ })
+
+ it('should load user libraries on mount', async () => {
+ // Override localStorage mock to return a userId for this test
+ window.localStorage.getItem.mockReturnValue('user123')
+
+ mockDataProvider.getOne.mockResolvedValue({
+ data: { libraries: mockLibraries },
+ })
+
+ renderLibrarySelector({ userLibraries: [], selectedLibraries: [] })
+
+ await waitFor(() => {
+ expect(mockDataProvider.getOne).toHaveBeenCalledWith('user', {
+ id: 'user123',
+ })
+ })
+
+ expect(mockDispatch).toHaveBeenCalledWith({
+ type: 'SET_USER_LIBRARIES',
+ data: mockLibraries,
+ })
+ })
+
+ it('should handle API error gracefully', async () => {
+ // Override localStorage mock to return a userId for this test
+ window.localStorage.getItem.mockReturnValue('user123')
+
+ const consoleSpy = vi.spyOn(console, 'warn').mockImplementation(() => {})
+ mockDataProvider.getOne.mockRejectedValue(new Error('API Error'))
+
+ renderLibrarySelector({ userLibraries: [], selectedLibraries: [] })
+
+ await waitFor(() => {
+ expect(consoleSpy).toHaveBeenCalledWith(
+ 'Could not load user libraries (this may be expected for non-admin users):',
+ expect.any(Error),
+ )
+ })
+
+ consoleSpy.mockRestore()
+ })
+
+ it('should not load libraries when userId is not available', () => {
+ window.localStorage.getItem.mockReturnValue(null)
+
+ renderLibrarySelector({ userLibraries: [], selectedLibraries: [] })
+
+ expect(mockDataProvider.getOne).not.toHaveBeenCalled()
+ })
+ })
+})
diff --git a/ui/src/common/Linkify.test.jsx b/ui/src/common/Linkify.test.jsx
index cef50b228..cd19ffa03 100644
--- a/ui/src/common/Linkify.test.jsx
+++ b/ui/src/common/Linkify.test.jsx
@@ -1,6 +1,5 @@
import React from 'react'
import { render, screen } from '@testing-library/react'
-import '@testing-library/jest-dom'
import Linkify from './Linkify'
const URL = 'http://www.example.com'
diff --git a/ui/src/common/RatingField.jsx b/ui/src/common/RatingField.jsx
index 1b440c51e..b29c1eee8 100644
--- a/ui/src/common/RatingField.jsx
+++ b/ui/src/common/RatingField.jsx
@@ -38,15 +38,16 @@ export const RatingField = ({
const handleRating = useCallback(
(e, val) => {
- rate(val ?? 0, e.target.name)
+ const targetId = record.mediaFileId || record.id
+ rate(val ?? 0, targetId)
},
- [rate],
+ [rate, record.mediaFileId, record.id],
)
return (
stopPropagation(e)}>
({
+ root: {
+ width: '960px',
+ maxWidth: '100%',
+ },
+ headerContainer: {
+ display: 'flex',
+ alignItems: 'center',
+ marginBottom: theme.spacing(1),
+ paddingLeft: theme.spacing(1),
+ },
+ masterCheckbox: {
+ padding: '7px',
+ marginLeft: '-9px',
+ marginRight: theme.spacing(1),
+ },
+ libraryList: {
+ height: '120px',
+ overflow: 'auto',
+ border: `1px solid ${theme.palette.divider}`,
+ borderRadius: theme.shape.borderRadius,
+ backgroundColor: theme.palette.background.paper,
+ },
+ listItem: {
+ paddingTop: 0,
+ paddingBottom: 0,
+ },
+ emptyMessage: {
+ padding: theme.spacing(2),
+ textAlign: 'center',
+ color: theme.palette.text.secondary,
+ },
+}))
+
+const EmptyLibraryMessage = () => {
+ const classes = useStyles()
+
+ return (
+
+ No libraries available
+
+ )
+}
+
+const LibraryListItem = ({ library, isSelected, onToggle }) => {
+ const classes = useStyles()
+
+ return (
+ onToggle(library)}
+ dense
+ >
+
+ }
+ checkedIcon={ }
+ checked={isSelected}
+ tabIndex={-1}
+ disableRipple
+ />
+
+
+
+ )
+}
+
+export const SelectLibraryInput = ({
+ onChange,
+ value = [],
+ isNewUser = false,
+}) => {
+ const classes = useStyles()
+ const translate = useTranslate()
+ const [selectedLibraryIds, setSelectedLibraryIds] = useState([])
+ const [hasInitialized, setHasInitialized] = useState(false)
+
+ const { ids, data, isLoading } = useGetList(
+ 'library',
+ { page: 1, perPage: -1 },
+ { field: 'name', order: 'ASC' },
+ )
+
+ const options = useMemo(
+ () => (ids && ids.map((id) => data[id])) || [],
+ [ids, data],
+ )
+
+ // Reset initialization state when isNewUser changes
+ useEffect(() => {
+ if (isNewUser) {
+ setHasInitialized(false)
+ }
+ }, [isNewUser])
+
+ // Pre-select default libraries for new users
+ useEffect(() => {
+ if (
+ isNewUser &&
+ !isLoading &&
+ options.length > 0 &&
+ !hasInitialized &&
+ Array.isArray(value) &&
+ value.length === 0
+ ) {
+ const defaultLibraryIds = options
+ .filter((lib) => lib.defaultNewUsers)
+ .map((lib) => lib.id)
+
+ if (defaultLibraryIds.length > 0) {
+ setSelectedLibraryIds(defaultLibraryIds)
+ onChange(defaultLibraryIds)
+ }
+
+ setHasInitialized(true)
+ }
+ }, [isNewUser, isLoading, options, hasInitialized, value, onChange])
+
+ // Update selectedLibraryIds when value prop changes (for editing mode and pre-selection)
+ useEffect(() => {
+ // For new users, only sync from value prop if it has actual data
+ // This prevents empty initial state from overriding our pre-selection
+ if (isNewUser && Array.isArray(value) && value.length === 0) {
+ return
+ }
+
+ if (Array.isArray(value)) {
+ const libraryIds = value.map((item) =>
+ typeof item === 'object' ? item.id : item,
+ )
+ setSelectedLibraryIds(libraryIds)
+ } else if (value.length === 0) {
+ // Handle case where value is explicitly set to empty array (for existing users)
+ setSelectedLibraryIds([])
+ }
+ }, [value, isNewUser, hasInitialized])
+
+ const isLibrarySelected = (library) => selectedLibraryIds.includes(library.id)
+
+ const handleLibraryToggle = (library) => {
+ const isSelected = selectedLibraryIds.includes(library.id)
+ let newSelection
+
+ if (isSelected) {
+ newSelection = selectedLibraryIds.filter((id) => id !== library.id)
+ } else {
+ newSelection = [...selectedLibraryIds, library.id]
+ }
+
+ setSelectedLibraryIds(newSelection)
+ onChange(newSelection)
+ }
+
+ const handleMasterCheckboxChange = () => {
+ const isAllSelected = selectedLibraryIds.length === options.length
+ const newSelection = isAllSelected ? [] : options.map((lib) => lib.id)
+
+ setSelectedLibraryIds(newSelection)
+ onChange(newSelection)
+ }
+
+ const selectedCount = selectedLibraryIds.length
+ const totalCount = options.length
+ const isAllSelected = selectedCount === totalCount && totalCount > 0
+ const isIndeterminate = selectedCount > 0 && selectedCount < totalCount
+
+ return (
+
+ {options.length > 1 && (
+
+
+
+ {translate('resources.user.message.selectAllLibraries')}
+
+
+ )}
+
+ {options.length === 0 ? (
+
+ ) : (
+ options.map((library) => (
+
+ ))
+ )}
+
+
+ )
+}
+
+SelectLibraryInput.propTypes = {
+ onChange: PropTypes.func.isRequired,
+ value: PropTypes.array,
+ isNewUser: PropTypes.bool,
+}
+
+LibraryListItem.propTypes = {
+ library: PropTypes.object.isRequired,
+ isSelected: PropTypes.bool.isRequired,
+ onToggle: PropTypes.func.isRequired,
+}
diff --git a/ui/src/common/SelectLibraryInput.test.jsx b/ui/src/common/SelectLibraryInput.test.jsx
new file mode 100644
index 000000000..8a7e56d3e
--- /dev/null
+++ b/ui/src/common/SelectLibraryInput.test.jsx
@@ -0,0 +1,458 @@
+import * as React from 'react'
+import { render, screen, fireEvent, cleanup } from '@testing-library/react'
+import { SelectLibraryInput } from './SelectLibraryInput'
+import { useGetList } from 'react-admin'
+import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest'
+
+// Mock Material-UI components
+vi.mock('@material-ui/core', () => ({
+ List: ({ children }) => {children}
,
+ ListItem: ({ children, button, onClick, dense, className }) => (
+
+ {children}
+
+ ),
+ ListItemIcon: ({ children }) => {children} ,
+ ListItemText: ({ primary }) => {primary} ,
+ Typography: ({ children, variant }) => {children} ,
+ Box: ({ children, className }) => {children}
,
+ Checkbox: ({
+ checked,
+ indeterminate,
+ onChange,
+ size,
+ className,
+ ...props
+ }) => (
+ {
+ if (el) el.indeterminate = indeterminate
+ }}
+ onChange={onChange}
+ className={className}
+ {...props}
+ />
+ ),
+ makeStyles: () => () => ({}),
+}))
+
+// Mock Material-UI icons
+vi.mock('@material-ui/icons', () => ({
+ CheckBox: () => ☑ ,
+ CheckBoxOutlineBlank: () => ☐ ,
+}))
+
+// Mock the react-admin hook
+vi.mock('react-admin', () => ({
+ useGetList: vi.fn(),
+ useTranslate: vi.fn(() => (key) => key), // Simple translation mock
+}))
+
+describe(' ', () => {
+ const mockOnChange = vi.fn()
+
+ beforeEach(() => {
+ // Reset the mock before each test
+ mockOnChange.mockClear()
+ })
+
+ afterEach(cleanup)
+
+ it('should render empty message when no libraries available', () => {
+ // Mock empty library response
+ useGetList.mockReturnValue({
+ ids: [],
+ data: {},
+ })
+
+ render( )
+ expect(screen.getByText('No libraries available')).not.toBeNull()
+ })
+
+ it('should render libraries when available', () => {
+ // Mock libraries
+ const mockLibraries = {
+ 1: { id: '1', name: 'Library 1' },
+ 2: { id: '2', name: 'Library 2' },
+ }
+ useGetList.mockReturnValue({
+ ids: ['1', '2'],
+ data: mockLibraries,
+ })
+
+ render( )
+ expect(screen.getByText('Library 1')).not.toBeNull()
+ expect(screen.getByText('Library 2')).not.toBeNull()
+ })
+
+ it('should toggle selection when a library is clicked', () => {
+ // Mock libraries
+ const mockLibraries = {
+ 1: { id: '1', name: 'Library 1' },
+ 2: { id: '2', name: 'Library 2' },
+ }
+
+ // Test selecting an item
+ useGetList.mockReturnValue({
+ ids: ['1', '2'],
+ data: mockLibraries,
+ })
+ render( )
+
+ // Find the library buttons by their text content
+ const library1Button = screen.getByText('Library 1').closest('button')
+ fireEvent.click(library1Button)
+ expect(mockOnChange).toHaveBeenCalledWith(['1'])
+
+ // Clean up to avoid DOM duplication
+ cleanup()
+ mockOnChange.mockClear()
+
+ // Test deselecting an item
+ useGetList.mockReturnValue({
+ ids: ['1', '2'],
+ data: mockLibraries,
+ })
+ render( )
+
+ // Find the library button again and click to deselect
+ const library1ButtonDeselect = screen
+ .getByText('Library 1')
+ .closest('button')
+ fireEvent.click(library1ButtonDeselect)
+ expect(mockOnChange).toHaveBeenCalledWith([])
+ })
+
+ it('should correctly initialize with provided values', () => {
+ // Mock libraries
+ const mockLibraries = {
+ 1: { id: '1', name: 'Library 1' },
+ 2: { id: '2', name: 'Library 2' },
+ }
+ useGetList.mockReturnValue({
+ ids: ['1', '2'],
+ data: mockLibraries,
+ })
+
+ // Initial value as array of IDs
+ render( )
+
+ // Check that checkbox for Library 1 is checked
+ const checkboxes = screen.getAllByRole('checkbox')
+ // With master checkbox, individual checkboxes start at index 1
+ expect(checkboxes[1].checked).toBe(true) // Library 1
+ expect(checkboxes[2].checked).toBe(false) // Library 2
+ })
+
+ it('should handle value as array of objects', () => {
+ // Mock libraries
+ const mockLibraries = {
+ 1: { id: '1', name: 'Library 1' },
+ 2: { id: '2', name: 'Library 2' },
+ }
+ useGetList.mockReturnValue({
+ ids: ['1', '2'],
+ data: mockLibraries,
+ })
+
+ // Initial value as array of objects with id property
+ render( )
+
+ // Check that checkbox for Library 2 is checked
+ const checkboxes = screen.getAllByRole('checkbox')
+ // With master checkbox, index shifts by 1
+ expect(checkboxes[1].checked).toBe(false) // Library 1
+ expect(checkboxes[2].checked).toBe(true) // Library 2
+ })
+
+ it('should render master checkbox when there are multiple libraries', () => {
+ // Mock libraries
+ const mockLibraries = {
+ 1: { id: '1', name: 'Library 1' },
+ 2: { id: '2', name: 'Library 2' },
+ }
+ useGetList.mockReturnValue({
+ ids: ['1', '2'],
+ data: mockLibraries,
+ })
+
+ render( )
+
+ // Should render master checkbox plus individual checkboxes
+ const checkboxes = screen.getAllByRole('checkbox')
+ expect(checkboxes).toHaveLength(3) // 1 master + 2 individual
+ expect(
+ screen.getByText('resources.user.message.selectAllLibraries'),
+ ).not.toBeNull()
+ })
+
+ it('should not render master checkbox when there is only one library', () => {
+ // Mock single library
+ const mockLibraries = {
+ 1: { id: '1', name: 'Library 1' },
+ }
+ useGetList.mockReturnValue({
+ ids: ['1'],
+ data: mockLibraries,
+ })
+
+ render( )
+
+ // Should render only individual checkbox
+ const checkboxes = screen.getAllByRole('checkbox')
+ expect(checkboxes).toHaveLength(1) // Only 1 individual checkbox
+ })
+
+ it('should handle master checkbox selection and deselection', () => {
+ // Mock libraries
+ const mockLibraries = {
+ 1: { id: '1', name: 'Library 1' },
+ 2: { id: '2', name: 'Library 2' },
+ }
+ useGetList.mockReturnValue({
+ ids: ['1', '2'],
+ data: mockLibraries,
+ })
+
+ render( )
+
+ const checkboxes = screen.getAllByRole('checkbox')
+ const masterCheckbox = checkboxes[0] // Master is first
+
+ // Click master checkbox to select all
+ fireEvent.click(masterCheckbox)
+ expect(mockOnChange).toHaveBeenCalledWith(['1', '2'])
+
+ // Clean up and test deselect all
+ cleanup()
+ mockOnChange.mockClear()
+
+ render( )
+ const checkboxes2 = screen.getAllByRole('checkbox')
+ const masterCheckbox2 = checkboxes2[0]
+
+ // Click master checkbox to deselect all
+ fireEvent.click(masterCheckbox2)
+ expect(mockOnChange).toHaveBeenCalledWith([])
+ })
+
+ it('should show master checkbox as indeterminate when some libraries are selected', () => {
+ // Mock libraries
+ const mockLibraries = {
+ 1: { id: '1', name: 'Library 1' },
+ 2: { id: '2', name: 'Library 2' },
+ }
+ useGetList.mockReturnValue({
+ ids: ['1', '2'],
+ data: mockLibraries,
+ })
+
+ render( )
+
+ const checkboxes = screen.getAllByRole('checkbox')
+ const masterCheckbox = checkboxes[0] // Master is first
+
+ // Master checkbox should not be checked when only some libraries are selected
+ expect(masterCheckbox.checked).toBe(false)
+ // Note: Testing indeterminate property directly through JSDOM can be unreliable
+ // The important behavior is that it's not checked when only some are selected
+ })
+
+ describe('New User Default Library Selection', () => {
+ // Helper function to create mock libraries with configurable default settings
+ const createMockLibraries = (libraryConfigs) => {
+ const libraries = {}
+ const ids = []
+
+ libraryConfigs.forEach(({ id, name, defaultNewUsers }) => {
+ libraries[id] = {
+ id,
+ name,
+ ...(defaultNewUsers !== undefined && { defaultNewUsers }),
+ }
+ ids.push(id)
+ })
+
+ return { libraries, ids }
+ }
+
+ // Helper function to setup useGetList mock
+ const setupMockLibraries = (libraryConfigs, isLoading = false) => {
+ const { libraries, ids } = createMockLibraries(libraryConfigs)
+ useGetList.mockReturnValue({
+ ids,
+ data: libraries,
+ isLoading,
+ })
+ return { libraries, ids }
+ }
+
+ beforeEach(() => {
+ mockOnChange.mockClear()
+ })
+
+ it('should pre-select default libraries for new users', () => {
+ setupMockLibraries([
+ { id: '1', name: 'Library 1', defaultNewUsers: true },
+ { id: '2', name: 'Library 2', defaultNewUsers: false },
+ { id: '3', name: 'Library 3', defaultNewUsers: true },
+ ])
+
+ render(
+ ,
+ )
+
+ expect(mockOnChange).toHaveBeenCalledWith(['1', '3'])
+ })
+
+ it('should not pre-select default libraries if new user already has values', () => {
+ setupMockLibraries([
+ { id: '1', name: 'Library 1', defaultNewUsers: true },
+ { id: '2', name: 'Library 2', defaultNewUsers: false },
+ ])
+
+ render(
+ ,
+ )
+
+ expect(mockOnChange).not.toHaveBeenCalled()
+ })
+
+ it('should not pre-select libraries while data is still loading', () => {
+ setupMockLibraries(
+ [{ id: '1', name: 'Library 1', defaultNewUsers: true }],
+ true,
+ ) // isLoading = true
+
+ render(
+ ,
+ )
+
+ expect(mockOnChange).not.toHaveBeenCalled()
+ })
+
+ it('should not pre-select anything if no libraries have defaultNewUsers flag', () => {
+ setupMockLibraries([
+ { id: '1', name: 'Library 1', defaultNewUsers: false },
+ { id: '2', name: 'Library 2', defaultNewUsers: false },
+ ])
+
+ render(
+ ,
+ )
+
+ expect(mockOnChange).not.toHaveBeenCalled()
+ })
+
+ it('should reset initialization state when isNewUser prop changes', () => {
+ setupMockLibraries([
+ { id: '1', name: 'Library 1', defaultNewUsers: true },
+ ])
+
+ const { rerender } = render(
+ ,
+ )
+
+ expect(mockOnChange).not.toHaveBeenCalled()
+
+ // Change to new user
+ rerender(
+ ,
+ )
+
+ expect(mockOnChange).toHaveBeenCalledWith(['1'])
+ })
+
+ it('should not override pre-selection when value prop is empty for new users', () => {
+ setupMockLibraries([
+ { id: '1', name: 'Library 1', defaultNewUsers: true },
+ { id: '2', name: 'Library 2', defaultNewUsers: false },
+ ])
+
+ const { rerender } = render(
+ ,
+ )
+
+ expect(mockOnChange).toHaveBeenCalledWith(['1'])
+ mockOnChange.mockClear()
+
+ // Re-render with empty value prop (simulating form state update)
+ rerender(
+ ,
+ )
+
+ expect(mockOnChange).not.toHaveBeenCalled()
+ })
+
+ it('should sync from value prop for existing users even when empty', () => {
+ setupMockLibraries([
+ { id: '1', name: 'Library 1', defaultNewUsers: true },
+ ])
+
+ render(
+ ,
+ )
+
+ // Check that no libraries are selected (checkboxes should be unchecked)
+ const checkboxes = screen.getAllByRole('checkbox')
+ // Only one checkbox since there's only one library and no master checkbox for single library
+ expect(checkboxes[0].checked).toBe(false)
+ })
+
+ it('should handle libraries with missing defaultNewUsers property', () => {
+ setupMockLibraries([
+ { id: '1', name: 'Library 1', defaultNewUsers: true },
+ { id: '2', name: 'Library 2' }, // Missing defaultNewUsers property
+ { id: '3', name: 'Library 3', defaultNewUsers: false },
+ ])
+
+ render(
+ ,
+ )
+
+ expect(mockOnChange).toHaveBeenCalledWith(['1'])
+ })
+ })
+})
diff --git a/ui/src/common/SongContextMenu.jsx b/ui/src/common/SongContextMenu.jsx
index f2227dc72..f8b0bba5e 100644
--- a/ui/src/common/SongContextMenu.jsx
+++ b/ui/src/common/SongContextMenu.jsx
@@ -1,7 +1,12 @@
import React, { useState } from 'react'
import PropTypes from 'prop-types'
import { useDispatch } from 'react-redux'
-import { useNotify, usePermissions, useTranslate } from 'react-admin'
+import {
+ useNotify,
+ usePermissions,
+ useTranslate,
+ useDataProvider,
+} from 'react-admin'
import { IconButton, Menu, MenuItem } from '@material-ui/core'
import { makeStyles } from '@material-ui/core/styles'
import MoreVertIcon from '@material-ui/icons/MoreVert'
@@ -20,7 +25,7 @@ import {
import { LoveButton } from './LoveButton'
import config from '../config'
import { formatBytes } from '../utils'
-import { httpClient } from '../dataProvider'
+import { useRedirect } from 'react-admin'
const useStyles = makeStyles({
noWrap: {
@@ -57,8 +62,13 @@ export const SongContextMenu = ({
const dispatch = useDispatch()
const translate = useTranslate()
const notify = useNotify()
+ const dataProvider = useDataProvider()
const [anchorEl, setAnchorEl] = useState(null)
+ const [playlistAnchorEl, setPlaylistAnchorEl] = useState(null)
+ const [playlists, setPlaylists] = useState([])
+ const [playlistsLoaded, setPlaylistsLoaded] = useState(false)
const { permissions } = usePermissions()
+ const redirect = useRedirect()
const options = {
playNow: {
@@ -87,6 +97,15 @@ export const SongContextMenu = ({
}),
),
},
+ showInPlaylist: {
+ enabled: true,
+ label:
+ translate('resources.song.actions.showInPlaylist') +
+ (playlists.length > 0 ? ' ►' : ''),
+ action: (record, e) => {
+ setPlaylistAnchorEl(e.currentTarget)
+ },
+ },
share: {
enabled: config.enableSharing,
label: translate('ra.action.share'),
@@ -113,8 +132,8 @@ export const SongContextMenu = ({
if (permissions === 'admin' && !record.missing) {
try {
let id = record.mediaFileId ?? record.id
- const data = await httpClient(`/api/inspect?id=${id}`)
- fullRecord = { ...record, rawTags: data.json.rawTags }
+ const data = await dataProvider.inspect(id)
+ fullRecord = { ...record, rawTags: data.data.rawTags }
} catch (error) {
notify(
translate('ra.notification.http_error') + ': ' + error.message,
@@ -134,6 +153,21 @@ export const SongContextMenu = ({
const handleClick = (e) => {
setAnchorEl(e.currentTarget)
+ if (!playlistsLoaded) {
+ const id = record.mediaFileId || record.id
+ dataProvider
+ .getPlaylists(id)
+ .then((res) => {
+ setPlaylists(res.data)
+ setPlaylistsLoaded(true)
+ })
+ .catch((error) => {
+ // eslint-disable-next-line no-console
+ console.error('Failed to fetch playlists:', error)
+ setPlaylists([])
+ setPlaylistsLoaded(true)
+ })
+ }
e.stopPropagation()
}
@@ -144,12 +178,39 @@ export const SongContextMenu = ({
const handleItemClick = (e) => {
e.preventDefault()
- setAnchorEl(null)
const key = e.target.getAttribute('value')
- options[key].action(record)
+ const action = options[key].action
+
+ if (key === 'showInPlaylist') {
+ // For showInPlaylist, we keep the main menu open and show submenu
+ action(record, e)
+ } else {
+ // For other actions, close the main menu
+ setAnchorEl(null)
+ action(record)
+ }
e.stopPropagation()
}
+ const handlePlaylistClose = (e) => {
+ setPlaylistAnchorEl(null)
+ if (e) {
+ e.stopPropagation()
+ }
+ }
+
+ const handleMainMenuClose = (e) => {
+ setAnchorEl(null)
+ setPlaylistAnchorEl(null) // Close both menus
+ e.stopPropagation()
+ }
+
+ const handlePlaylistClick = (id, e) => {
+ e.stopPropagation()
+ redirect(`/playlist/${id}/show`)
+ handlePlaylistClose()
+ }
+
const open = Boolean(anchorEl)
if (!record) {
@@ -170,16 +231,50 @@ export const SongContextMenu = ({
id={'menu' + record.id}
anchorEl={anchorEl}
open={open}
- onClose={handleClose}
+ onClose={handleMainMenuClose}
>
- {Object.keys(options).map(
- (key) =>
+ {Object.keys(options).map((key) => {
+ const showInPlaylistDisabled =
+ key === 'showInPlaylist' && !playlists.length
+ return (
options[key].enabled && (
-
+ e.stopPropagation()
+ : handleItemClick
+ }
+ disabled={showInPlaylistDisabled}
+ style={
+ showInPlaylistDisabled ? { pointerEvents: 'auto' } : undefined
+ }
+ >
{options[key].label}
- ),
- )}
+ )
+ )
+ })}
+
+
+ {playlists.map((p) => (
+ handlePlaylistClick(p.id, e)}>
+ {p.name}
+
+ ))}
)
diff --git a/ui/src/common/SongContextMenu.test.jsx b/ui/src/common/SongContextMenu.test.jsx
new file mode 100644
index 000000000..a30da859f
--- /dev/null
+++ b/ui/src/common/SongContextMenu.test.jsx
@@ -0,0 +1,107 @@
+import React from 'react'
+import { render, fireEvent, screen, waitFor } from '@testing-library/react'
+import { TestContext } from 'ra-test'
+import { describe, it, expect, vi, beforeEach } from 'vitest'
+import { SongContextMenu } from './SongContextMenu'
+
+vi.mock('../dataProvider', () => ({
+ httpClient: vi.fn(),
+}))
+
+vi.mock('react-redux', () => ({ useDispatch: () => vi.fn() }))
+
+const getPlaylistsMock = vi.fn()
+
+vi.mock('react-admin', async (importOriginal) => {
+ const actual = await importOriginal()
+ return {
+ ...actual,
+ useRedirect: () => (url) => {
+ window.location.hash = `#${url}`
+ },
+ useDataProvider: () => ({
+ getPlaylists: getPlaylistsMock,
+ inspect: vi.fn().mockResolvedValue({
+ data: { rawTags: {} },
+ }),
+ }),
+ }
+})
+
+describe('SongContextMenu', () => {
+ beforeEach(() => {
+ vi.clearAllMocks()
+ window.location.hash = ''
+ getPlaylistsMock.mockResolvedValue({
+ data: [{ id: 'pl1', name: 'Pl 1' }],
+ })
+ })
+
+ it('navigates to playlist when selected', async () => {
+ render(
+
+
+ ,
+ )
+ fireEvent.click(screen.getAllByRole('button')[1])
+ await waitFor(() =>
+ screen.getByText(/resources\.song\.actions\.showInPlaylist/),
+ )
+ fireEvent.click(
+ screen.getByText(/resources\.song\.actions\.showInPlaylist/),
+ )
+ await waitFor(() => screen.getByText('Pl 1'))
+ fireEvent.click(screen.getByText('Pl 1'))
+ expect(window.location.hash).toBe('#/playlist/pl1/show')
+ })
+
+ it('stops event propagation when playlist submenu is closed', async () => {
+ const mockOnClick = vi.fn()
+ render(
+
+
+
+
+ ,
+ )
+
+ // Open main menu
+ fireEvent.click(screen.getAllByRole('button')[1])
+ await waitFor(() =>
+ screen.getByText(/resources\.song\.actions\.showInPlaylist/),
+ )
+
+ // Open playlist submenu
+ fireEvent.click(
+ screen.getByText(/resources\.song\.actions\.showInPlaylist/),
+ )
+ await waitFor(() => screen.getByText('Pl 1'))
+
+ // Click outside the playlist submenu (should close it without triggering parent click)
+ fireEvent.click(document.body)
+
+ expect(mockOnClick).not.toHaveBeenCalled()
+ })
+
+ it('does nothing when "Show in Playlist" is disabled', async () => {
+ getPlaylistsMock.mockResolvedValue({ data: [] })
+ const mockOnClick = vi.fn()
+ render(
+
+
+
+
+ ,
+ )
+
+ fireEvent.click(screen.getAllByRole('button')[1])
+ await waitFor(() =>
+ screen.getByText(/resources\.song\.actions\.showInPlaylist/),
+ )
+
+ fireEvent.click(
+ screen.getByText(/resources\.song\.actions\.showInPlaylist/),
+ )
+ expect(mockOnClick).not.toHaveBeenCalled()
+ })
+})
diff --git a/ui/src/common/SongInfo.jsx b/ui/src/common/SongInfo.jsx
index 77e91b653..1b1a014f1 100644
--- a/ui/src/common/SongInfo.jsx
+++ b/ui/src/common/SongInfo.jsx
@@ -59,6 +59,7 @@ export const SongInfo = (props) => {
]
const data = {
path: ,
+ libraryName: ,
album: (
),
@@ -138,7 +139,7 @@ export const SongInfo = (props) => {
)}
diff --git a/ui/src/common/index.js b/ui/src/common/index.js
index 1a43047c1..f64d4fe0c 100644
--- a/ui/src/common/index.js
+++ b/ui/src/common/index.js
@@ -27,6 +27,7 @@ export * from './useAlbumsPerPage'
export * from './useGetHandleArtistClick'
export * from './useInterval'
export * from './useResourceRefresh'
+export * from './useRefreshOnEvents'
export * from './useToggleLove'
export * from './useTraceUpdate'
export * from './Writable'
diff --git a/ui/src/common/useLibrarySelection.js b/ui/src/common/useLibrarySelection.js
new file mode 100644
index 000000000..c5d84a61f
--- /dev/null
+++ b/ui/src/common/useLibrarySelection.js
@@ -0,0 +1,44 @@
+import { useSelector } from 'react-redux'
+
+/**
+ * Hook to get the currently selected library IDs
+ * Returns an array of library IDs that should be used for filtering data
+ * If no libraries are selected (empty array), returns all user accessible libraries
+ */
+export const useSelectedLibraries = () => {
+ const { userLibraries, selectedLibraries } = useSelector(
+ (state) => state.library,
+ )
+
+ // If no specific selection, default to all accessible libraries
+ if (selectedLibraries.length === 0 && userLibraries.length > 0) {
+ return userLibraries.map((lib) => lib.id)
+ }
+
+ return selectedLibraries
+}
+
+/**
+ * Hook to get library filter parameters for data provider queries
+ * Returns an object that can be spread into query parameters
+ */
+export const useLibraryFilter = () => {
+ const selectedLibraryIds = useSelectedLibraries()
+
+ // If user has access to only one library or no specific selection, no filter needed
+ if (selectedLibraryIds.length <= 1) {
+ return {}
+ }
+
+ return {
+ libraryIds: selectedLibraryIds,
+ }
+}
+
+/**
+ * Hook to check if a specific library is currently selected
+ */
+export const useIsLibrarySelected = (libraryId) => {
+ const selectedLibraryIds = useSelectedLibraries()
+ return selectedLibraryIds.includes(libraryId)
+}
diff --git a/ui/src/common/useLibrarySelection.test.js b/ui/src/common/useLibrarySelection.test.js
new file mode 100644
index 000000000..30f109dc6
--- /dev/null
+++ b/ui/src/common/useLibrarySelection.test.js
@@ -0,0 +1,204 @@
+import { renderHook } from '@testing-library/react-hooks'
+import { describe, it, expect, beforeEach, vi } from 'vitest'
+import {
+ useSelectedLibraries,
+ useLibraryFilter,
+ useIsLibrarySelected,
+} from './useLibrarySelection'
+
+// Mock dependencies
+vi.mock('react-redux', () => ({
+ useSelector: vi.fn(),
+}))
+
+describe('Library Selection Hooks', () => {
+ const mockLibraries = [
+ { id: '1', name: 'Music Library' },
+ { id: '2', name: 'Podcasts' },
+ { id: '3', name: 'Audiobooks' },
+ ]
+
+ let mockUseSelector
+
+ beforeEach(async () => {
+ vi.clearAllMocks()
+ const { useSelector } = await import('react-redux')
+ mockUseSelector = vi.mocked(useSelector)
+ })
+
+ const setupSelector = (
+ userLibraries = mockLibraries,
+ selectedLibraries = [],
+ ) => {
+ mockUseSelector.mockImplementation((selector) =>
+ selector({
+ library: {
+ userLibraries,
+ selectedLibraries,
+ },
+ }),
+ )
+ }
+
+ describe('useSelectedLibraries', () => {
+ it('should return selected library IDs when libraries are explicitly selected', async () => {
+ setupSelector(mockLibraries, ['1', '2'])
+
+ const { result } = renderHook(() => useSelectedLibraries())
+
+ expect(result.current).toEqual(['1', '2'])
+ })
+
+ it('should return all user library IDs when no libraries are selected and user has libraries', async () => {
+ setupSelector(mockLibraries, [])
+
+ const { result } = renderHook(() => useSelectedLibraries())
+
+ expect(result.current).toEqual(['1', '2', '3'])
+ })
+
+ it('should return empty array when no libraries are selected and user has no libraries', async () => {
+ setupSelector([], [])
+
+ const { result } = renderHook(() => useSelectedLibraries())
+
+ expect(result.current).toEqual([])
+ })
+
+ it('should return selected libraries even if they are all user libraries', async () => {
+ setupSelector(mockLibraries, ['1', '2', '3'])
+
+ const { result } = renderHook(() => useSelectedLibraries())
+
+ expect(result.current).toEqual(['1', '2', '3'])
+ })
+
+ it('should return single selected library', async () => {
+ setupSelector(mockLibraries, ['2'])
+
+ const { result } = renderHook(() => useSelectedLibraries())
+
+ expect(result.current).toEqual(['2'])
+ })
+ })
+
+ describe('useLibraryFilter', () => {
+ it('should return empty object when user has only one library', async () => {
+ setupSelector([mockLibraries[0]], ['1'])
+
+ const { result } = renderHook(() => useLibraryFilter())
+
+ expect(result.current).toEqual({})
+ })
+
+ it('should return empty object when no libraries are selected (defaults to all)', async () => {
+ setupSelector([mockLibraries[0]], [])
+
+ const { result } = renderHook(() => useLibraryFilter())
+
+ expect(result.current).toEqual({})
+ })
+
+ it('should return libraryIds filter when multiple libraries are available and some are selected', async () => {
+ setupSelector(mockLibraries, ['1', '2'])
+
+ const { result } = renderHook(() => useLibraryFilter())
+
+ expect(result.current).toEqual({
+ libraryIds: ['1', '2'],
+ })
+ })
+
+ it('should return libraryIds filter when multiple libraries are available and all are selected', async () => {
+ setupSelector(mockLibraries, ['1', '2', '3'])
+
+ const { result } = renderHook(() => useLibraryFilter())
+
+ expect(result.current).toEqual({
+ libraryIds: ['1', '2', '3'],
+ })
+ })
+
+ it('should return empty object when user has no libraries', async () => {
+ setupSelector([], [])
+
+ const { result } = renderHook(() => useLibraryFilter())
+
+ expect(result.current).toEqual({})
+ })
+
+ it('should return libraryIds filter for default selection when multiple libraries available', async () => {
+ setupSelector(mockLibraries, []) // No explicit selection, should default to all
+
+ const { result } = renderHook(() => useLibraryFilter())
+
+ expect(result.current).toEqual({
+ libraryIds: ['1', '2', '3'],
+ })
+ })
+ })
+
+ describe('useIsLibrarySelected', () => {
+ it('should return true when library is explicitly selected', async () => {
+ setupSelector(mockLibraries, ['1', '3'])
+
+ const { result: result1 } = renderHook(() => useIsLibrarySelected('1'))
+ const { result: result2 } = renderHook(() => useIsLibrarySelected('3'))
+
+ expect(result1.current).toBe(true)
+ expect(result2.current).toBe(true)
+ })
+
+ it('should return false when library is not explicitly selected', async () => {
+ setupSelector(mockLibraries, ['1', '3'])
+
+ const { result } = renderHook(() => useIsLibrarySelected('2'))
+
+ expect(result.current).toBe(false)
+ })
+
+ it('should return true when no explicit selection (defaults to all) and library exists', async () => {
+ setupSelector(mockLibraries, [])
+
+ const { result: result1 } = renderHook(() => useIsLibrarySelected('1'))
+ const { result: result2 } = renderHook(() => useIsLibrarySelected('2'))
+ const { result: result3 } = renderHook(() => useIsLibrarySelected('3'))
+
+ expect(result1.current).toBe(true)
+ expect(result2.current).toBe(true)
+ expect(result3.current).toBe(true)
+ })
+
+ it('should return false when library does not exist in user libraries', async () => {
+ setupSelector(mockLibraries, [])
+
+ const { result } = renderHook(() => useIsLibrarySelected('999'))
+
+ expect(result.current).toBe(false)
+ })
+
+ it('should return false when user has no libraries', async () => {
+ setupSelector([], [])
+
+ const { result } = renderHook(() => useIsLibrarySelected('1'))
+
+ expect(result.current).toBe(false)
+ })
+
+ it('should handle undefined libraryId', async () => {
+ setupSelector(mockLibraries, ['1'])
+
+ const { result } = renderHook(() => useIsLibrarySelected(undefined))
+
+ expect(result.current).toBe(false)
+ })
+
+ it('should handle null libraryId', async () => {
+ setupSelector(mockLibraries, ['1'])
+
+ const { result } = renderHook(() => useIsLibrarySelected(null))
+
+ expect(result.current).toBe(false)
+ })
+ })
+})
diff --git a/ui/src/common/useRating.jsx b/ui/src/common/useRating.jsx
index f1d9a4fe9..2eb5d9eca 100644
--- a/ui/src/common/useRating.jsx
+++ b/ui/src/common/useRating.jsx
@@ -17,18 +17,42 @@ export const useRating = (resource, record) => {
}, [])
const refreshRating = useCallback(() => {
- dataProvider
- .getOne(resource, { id: record.id })
- .then(() => {
- if (mountedRef.current) {
- setLoading(false)
- }
- })
- .catch((e) => {
- // eslint-disable-next-line no-console
- console.log('Error encountered: ' + e)
- })
- }, [dataProvider, record, resource])
+ // For playlist tracks, refresh both resources to keep data in sync
+ if (record.mediaFileId) {
+ // This is a playlist track - refresh both the playlist track and the song
+ const promises = [
+ dataProvider.getOne('song', { id: record.mediaFileId }),
+ dataProvider.getOne('playlistTrack', {
+ id: record.id,
+ filter: { playlist_id: record.playlistId },
+ }),
+ ]
+
+ Promise.all(promises)
+ .catch((e) => {
+ // eslint-disable-next-line no-console
+ console.log('Error encountered: ' + e)
+ })
+ .finally(() => {
+ if (mountedRef.current) {
+ setLoading(false)
+ }
+ })
+ } else {
+ // Regular song or other resource
+ dataProvider
+ .getOne(resource, { id: record.id })
+ .catch((e) => {
+ // eslint-disable-next-line no-console
+ console.log('Error encountered: ' + e)
+ })
+ .finally(() => {
+ if (mountedRef.current) {
+ setLoading(false)
+ }
+ })
+ }
+ }, [dataProvider, record.id, record.mediaFileId, record.playlistId, resource])
const rate = (val, id) => {
setLoading(true)
diff --git a/ui/src/common/useRating.test.js b/ui/src/common/useRating.test.js
new file mode 100644
index 000000000..b1353512e
--- /dev/null
+++ b/ui/src/common/useRating.test.js
@@ -0,0 +1,165 @@
+import { renderHook, act } from '@testing-library/react-hooks'
+import { vi, describe, it, expect, beforeEach } from 'vitest'
+import { useRating } from './useRating'
+import subsonic from '../subsonic'
+import { useDataProvider } from 'react-admin'
+
+vi.mock('../subsonic', () => ({
+ default: {
+ setRating: vi.fn(() => Promise.resolve()),
+ },
+}))
+
+vi.mock('react-admin', async () => {
+ const actual = await vi.importActual('react-admin')
+ return {
+ ...actual,
+ useDataProvider: vi.fn(),
+ useNotify: vi.fn(() => vi.fn()),
+ }
+})
+
+describe('useRating', () => {
+ let getOne
+ beforeEach(() => {
+ getOne = vi.fn(() => Promise.resolve())
+ useDataProvider.mockReturnValue({ getOne })
+ vi.clearAllMocks()
+ })
+
+ it('returns rating value from record', () => {
+ const record = { id: 'sg-1', rating: 3 }
+ const { result } = renderHook(() => useRating('song', record))
+ const [rate, rating, loading] = result.current
+ expect(rating).toBe(3)
+ expect(loading).toBe(false)
+ expect(typeof rate).toBe('function')
+ })
+
+ it('sets rating using targetId and calls setRating API', async () => {
+ const record = { id: 'sg-1', rating: 0 }
+ const { result } = renderHook(() => useRating('song', record))
+ await act(async () => {
+ await result.current[0](4, 'sg-1')
+ })
+ expect(subsonic.setRating).toHaveBeenCalledWith('sg-1', 4)
+ expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-1' })
+ })
+
+ it('handles zero rating (unrate)', async () => {
+ const record = { id: 'sg-1', rating: 5 }
+ const { result } = renderHook(() => useRating('song', record))
+ await act(async () => {
+ await result.current[0](0, 'sg-1')
+ })
+ expect(subsonic.setRating).toHaveBeenCalledWith('sg-1', 0)
+ })
+
+ describe('playlist track scenarios', () => {
+ it('refreshes both playlist track and song for playlist tracks', async () => {
+ const record = {
+ id: 'pt-1',
+ mediaFileId: 'sg-1',
+ playlistId: 'pl-1',
+ rating: 2,
+ }
+ const { result } = renderHook(() => useRating('playlistTrack', record))
+ await act(async () => {
+ await result.current[0](5, 'sg-1')
+ })
+
+ // Should rate using the media file ID
+ expect(subsonic.setRating).toHaveBeenCalledWith('sg-1', 5)
+
+ // Should refresh both the playlist track and the song
+ expect(getOne).toHaveBeenCalledTimes(2)
+ expect(getOne).toHaveBeenCalledWith('playlistTrack', {
+ id: 'pt-1',
+ filter: { playlist_id: 'pl-1' },
+ })
+ expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-1' })
+ })
+
+ it('includes playlist_id filter when refreshing playlist tracks', async () => {
+ const record = {
+ id: 'pt-5',
+ mediaFileId: 'sg-10',
+ playlistId: 'pl-123',
+ rating: 1,
+ }
+ const { result } = renderHook(() => useRating('playlistTrack', record))
+ await act(async () => {
+ await result.current[0](3, 'sg-10')
+ })
+
+ // Should rate using the media file ID
+ expect(subsonic.setRating).toHaveBeenCalledWith('sg-10', 3)
+
+ // Should refresh playlist track with correct playlist_id filter
+ expect(getOne).toHaveBeenCalledWith('playlistTrack', {
+ id: 'pt-5',
+ filter: { playlist_id: 'pl-123' },
+ })
+ // Should also refresh the underlying song
+ expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-10' })
+ })
+
+ it('only refreshes original resource when no mediaFileId present', async () => {
+ const record = { id: 'sg-1', rating: 4 }
+ const { result } = renderHook(() => useRating('song', record))
+ await act(async () => {
+ await result.current[0](2, 'sg-1')
+ })
+
+ // Should only refresh the original resource (song)
+ expect(getOne).toHaveBeenCalledTimes(1)
+ expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-1' })
+ })
+
+ it('does not include playlist_id filter for non-playlist resources', async () => {
+ const record = { id: 'sg-1', rating: 0 }
+ const { result } = renderHook(() => useRating('song', record))
+ await act(async () => {
+ await result.current[0](5, 'sg-1')
+ })
+
+ // Should refresh without any filter
+ expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-1' })
+ })
+ })
+
+ describe('component integration scenarios', () => {
+ it('handles mediaFileId fallback correctly for playlist tracks', async () => {
+ const record = {
+ id: 'pt-1',
+ mediaFileId: 'sg-1',
+ playlistId: 'pl-1',
+ rating: 0,
+ }
+ const { result } = renderHook(() => useRating('playlistTrack', record))
+
+ // Simulate RatingField component behavior: uses mediaFileId || record.id
+ const targetId = record.mediaFileId || record.id
+ await act(async () => {
+ await result.current[0](4, targetId)
+ })
+
+ expect(subsonic.setRating).toHaveBeenCalledWith('sg-1', 4)
+ })
+
+ it('handles regular song rating without mediaFileId', async () => {
+ const record = { id: 'sg-1', rating: 2 }
+ const { result } = renderHook(() => useRating('song', record))
+
+ // Simulate RatingField component behavior: uses mediaFileId || record.id
+ const targetId = record.mediaFileId || record.id
+ await act(async () => {
+ await result.current[0](5, targetId)
+ })
+
+ expect(subsonic.setRating).toHaveBeenCalledWith('sg-1', 5)
+ expect(getOne).toHaveBeenCalledTimes(1)
+ expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-1' })
+ })
+ })
+})
diff --git a/ui/src/common/useRefreshOnEvents.jsx b/ui/src/common/useRefreshOnEvents.jsx
new file mode 100644
index 000000000..b5f1b1ede
--- /dev/null
+++ b/ui/src/common/useRefreshOnEvents.jsx
@@ -0,0 +1,109 @@
+import { useEffect, useState } from 'react'
+import { useSelector } from 'react-redux'
+
+/**
+ * A reusable hook for triggering custom reload logic when specific SSE events occur.
+ *
+ * This hook is ideal when:
+ * - Your component displays derived/related data that isn't directly managed by react-admin
+ * - You need custom loading logic that goes beyond simple dataProvider.getMany() calls
+ * - Your data comes from non-standard endpoints or requires special processing
+ * - You want to reload parent/related resources when child resources change
+ *
+ * @param {Object} options - Configuration options
+ * @param {Array
} options.events - Array of event types to listen for (e.g., ['library', 'user', '*'])
+ * @param {Function} options.onRefresh - Async function to call when events occur.
+ * Should be wrapped in useCallback with appropriate dependencies to avoid unnecessary re-renders.
+ *
+ * @example
+ * // Example 1: LibrarySelector - Reload user data when library changes
+ * const loadUserLibraries = useCallback(async () => {
+ * const userId = localStorage.getItem('userId')
+ * if (userId) {
+ * const { data } = await dataProvider.getOne('user', { id: userId })
+ * dispatch(setUserLibraries(data.libraries || []))
+ * }
+ * }, [dataProvider, dispatch])
+ *
+ * useRefreshOnEvents({
+ * events: ['library', 'user'],
+ * onRefresh: loadUserLibraries
+ * })
+ *
+ * @example
+ * // Example 2: Statistics Dashboard - Reload stats when any music data changes
+ * const loadStats = useCallback(async () => {
+ * const stats = await dataProvider.customRequest('GET', 'stats')
+ * setDashboardStats(stats)
+ * }, [dataProvider, setDashboardStats])
+ *
+ * useRefreshOnEvents({
+ * events: ['album', 'song', 'artist'],
+ * onRefresh: loadStats
+ * })
+ *
+ * @example
+ * // Example 3: Permission-based UI - Reload permissions when user changes
+ * const loadPermissions = useCallback(async () => {
+ * const authData = await authProvider.getPermissions()
+ * setUserPermissions(authData)
+ * }, [authProvider, setUserPermissions])
+ *
+ * useRefreshOnEvents({
+ * events: ['user'],
+ * onRefresh: loadPermissions
+ * })
+ *
+ * @example
+ * // Example 4: Listen to all events (use sparingly)
+ * const reloadAll = useCallback(async () => {
+ * // This will trigger on ANY refresh event
+ * await reloadEverything()
+ * }, [reloadEverything])
+ *
+ * useRefreshOnEvents({
+ * events: ['*'],
+ * onRefresh: reloadAll
+ * })
+ */
+export const useRefreshOnEvents = ({ events, onRefresh }) => {
+ const [lastRefreshTime, setLastRefreshTime] = useState(Date.now())
+
+ const refreshData = useSelector(
+ (state) => state.activity?.refresh || { lastReceived: lastRefreshTime },
+ )
+
+ useEffect(() => {
+ const { resources, lastReceived } = refreshData
+
+ // Only process if we have new events
+ if (lastReceived <= lastRefreshTime) {
+ return
+ }
+
+ // Check if any of the events we're interested in occurred
+ const shouldRefresh =
+ resources &&
+ // Global refresh event
+ (resources['*'] === '*' ||
+ // Check for specific events we're listening to
+ events.some((eventType) => {
+ if (eventType === '*') {
+ return true // Listen to all events
+ }
+ return resources[eventType] // Check if this specific event occurred
+ }))
+
+ if (shouldRefresh) {
+ setLastRefreshTime(lastReceived)
+
+ // Call the custom refresh function
+ if (onRefresh) {
+ onRefresh().catch((error) => {
+ // eslint-disable-next-line no-console
+ console.warn('Error in useRefreshOnEvents onRefresh callback:', error)
+ })
+ }
+ }
+ }, [refreshData, lastRefreshTime, events, onRefresh])
+}
diff --git a/ui/src/common/useRefreshOnEvents.test.js b/ui/src/common/useRefreshOnEvents.test.js
new file mode 100644
index 000000000..2306cd3c9
--- /dev/null
+++ b/ui/src/common/useRefreshOnEvents.test.js
@@ -0,0 +1,233 @@
+import { vi } from 'vitest'
+import * as React from 'react'
+import * as Redux from 'react-redux'
+import { useRefreshOnEvents } from './useRefreshOnEvents'
+
+vi.mock('react', async () => {
+ const actual = await vi.importActual('react')
+ return {
+ ...actual,
+ useState: vi.fn(),
+ useEffect: vi.fn(),
+ }
+})
+
+vi.mock('react-redux', async () => {
+ const actual = await vi.importActual('react-redux')
+ return {
+ ...actual,
+ useSelector: vi.fn(),
+ }
+})
+
+describe('useRefreshOnEvents', () => {
+ const setState = vi.fn()
+ const useStateMock = (initState) => [initState, setState]
+ const onRefresh = vi.fn().mockResolvedValue()
+ let lastTime
+ let mockUseEffect
+
+ beforeEach(() => {
+ vi.spyOn(React, 'useState').mockImplementation(useStateMock)
+ mockUseEffect = vi.spyOn(React, 'useEffect')
+ lastTime = new Date(new Date().valueOf() + 1000)
+ onRefresh.mockClear()
+ setState.mockClear()
+ })
+
+ afterEach(() => {
+ vi.clearAllMocks()
+ })
+
+ it('stores last time checked, to avoid redundant runs', () => {
+ const useSelectorMock = () => ({
+ lastReceived: lastTime,
+ resources: { library: ['lib-1'] }, // Need some resources to trigger the update
+ })
+ vi.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
+
+ // Mock useEffect to immediately call the effect callback
+ mockUseEffect.mockImplementation((callback) => callback())
+
+ useRefreshOnEvents({
+ events: ['library'],
+ onRefresh,
+ })
+
+ expect(setState).toHaveBeenCalledWith(lastTime)
+ })
+
+ it("does not run again if lastTime didn't change", () => {
+ vi.spyOn(React, 'useState').mockImplementation(() => [lastTime, setState])
+ const useSelectorMock = () => ({ lastReceived: lastTime })
+ vi.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
+
+ // Mock useEffect to immediately call the effect callback
+ mockUseEffect.mockImplementation((callback) => callback())
+
+ useRefreshOnEvents({
+ events: ['library'],
+ onRefresh,
+ })
+
+ expect(setState).not.toHaveBeenCalled()
+ expect(onRefresh).not.toHaveBeenCalled()
+ })
+
+ describe('Event listening and refresh triggering', () => {
+ beforeEach(() => {
+ // Mock useEffect to immediately call the effect callback
+ mockUseEffect.mockImplementation((callback) => callback())
+ })
+
+ it('triggers refresh when a watched event occurs', () => {
+ const useSelectorMock = () => ({
+ lastReceived: lastTime,
+ resources: { library: ['lib-1', 'lib-2'] },
+ })
+ vi.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
+
+ useRefreshOnEvents({
+ events: ['library'],
+ onRefresh,
+ })
+
+ expect(onRefresh).toHaveBeenCalledTimes(1)
+ expect(setState).toHaveBeenCalledWith(lastTime)
+ })
+
+ it('triggers refresh when multiple watched events occur', () => {
+ const useSelectorMock = () => ({
+ lastReceived: lastTime,
+ resources: {
+ library: ['lib-1'],
+ user: ['user-1'],
+ album: ['album-1'], // This shouldn't trigger since it's not watched
+ },
+ })
+ vi.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
+
+ useRefreshOnEvents({
+ events: ['library', 'user'],
+ onRefresh,
+ })
+
+ expect(onRefresh).toHaveBeenCalledTimes(1)
+ expect(setState).toHaveBeenCalledWith(lastTime)
+ })
+
+ it('does not trigger refresh when unwatched events occur', () => {
+ const useSelectorMock = () => ({
+ lastReceived: lastTime,
+ resources: { album: ['album-1'], song: ['song-1'] },
+ })
+ vi.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
+
+ useRefreshOnEvents({
+ events: ['library', 'user'],
+ onRefresh,
+ })
+
+ expect(onRefresh).not.toHaveBeenCalled()
+ expect(setState).not.toHaveBeenCalled()
+ })
+
+ it('triggers refresh on global refresh event', () => {
+ const useSelectorMock = () => ({
+ lastReceived: lastTime,
+ resources: { '*': '*' },
+ })
+ vi.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
+
+ useRefreshOnEvents({
+ events: ['library'],
+ onRefresh,
+ })
+
+ expect(onRefresh).toHaveBeenCalledTimes(1)
+ expect(setState).toHaveBeenCalledWith(lastTime)
+ })
+
+ it('triggers refresh when listening to all events with "*"', () => {
+ const useSelectorMock = () => ({
+ lastReceived: lastTime,
+ resources: { song: ['song-1'] },
+ })
+ vi.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
+
+ useRefreshOnEvents({
+ events: ['*'],
+ onRefresh,
+ })
+
+ expect(onRefresh).toHaveBeenCalledTimes(1)
+ expect(setState).toHaveBeenCalledWith(lastTime)
+ })
+
+ it('handles empty events array gracefully', () => {
+ const useSelectorMock = () => ({
+ lastReceived: lastTime,
+ resources: { library: ['lib-1'] },
+ })
+ vi.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
+
+ useRefreshOnEvents({
+ events: [],
+ onRefresh,
+ })
+
+ expect(onRefresh).not.toHaveBeenCalled()
+ expect(setState).not.toHaveBeenCalled()
+ })
+
+ it('handles missing onRefresh function gracefully', () => {
+ const useSelectorMock = () => ({
+ lastReceived: lastTime,
+ resources: { library: ['lib-1'] },
+ })
+ vi.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
+
+ expect(() => {
+ useRefreshOnEvents({
+ events: ['library'],
+ // onRefresh is undefined
+ })
+ }).not.toThrow()
+
+ expect(setState).toHaveBeenCalledWith(lastTime)
+ })
+
+ it('handles onRefresh errors gracefully', async () => {
+ const consoleWarnSpy = vi
+ .spyOn(console, 'warn')
+ .mockImplementation(() => {})
+ const failingRefresh = vi
+ .fn()
+ .mockRejectedValue(new Error('Refresh failed'))
+
+ const useSelectorMock = () => ({
+ lastReceived: lastTime,
+ resources: { library: ['lib-1'] },
+ })
+ vi.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
+
+ useRefreshOnEvents({
+ events: ['library'],
+ onRefresh: failingRefresh,
+ })
+
+ expect(failingRefresh).toHaveBeenCalledTimes(1)
+ expect(setState).toHaveBeenCalledWith(lastTime)
+
+ // Wait for the promise to be rejected and handled
+ await new Promise((resolve) => setTimeout(resolve, 10))
+
+ expect(consoleWarnSpy).toHaveBeenCalledWith(
+ 'Error in useRefreshOnEvents onRefresh callback:',
+ expect.any(Error),
+ )
+
+ consoleWarnSpy.mockRestore()
+ })
+ })
+})
diff --git a/ui/src/common/useResourceRefresh.jsx b/ui/src/common/useResourceRefresh.jsx
index d9f6aee52..eabff6f92 100644
--- a/ui/src/common/useResourceRefresh.jsx
+++ b/ui/src/common/useResourceRefresh.jsx
@@ -2,6 +2,67 @@ import { useSelector } from 'react-redux'
import { useState } from 'react'
import { useRefresh, useDataProvider } from 'react-admin'
+/**
+ * A hook that automatically refreshes react-admin managed resources when refresh events are received via SSE.
+ *
+ * This hook is designed for components that display react-admin managed resources (like lists, shows, edits)
+ * and need to stay in sync when those resources are modified elsewhere in the application.
+ *
+ * **When to use this hook:**
+ * - Your component displays react-admin resources (albums, songs, artists, playlists, etc.)
+ * - You want automatic refresh when those resources are created/updated/deleted
+ * - Your data comes from standard dataProvider.getMany() calls
+ * - You're using react-admin's data management (queries, mutations, caching)
+ *
+ * **When NOT to use this hook:**
+ * - Your component displays derived/custom data not directly managed by react-admin
+ * - You need custom reload logic beyond dataProvider.getMany()
+ * - Your data comes from non-standard endpoints
+ * - Use `useRefreshOnEvents` instead for these scenarios
+ *
+ * @param {...string} visibleResources - Resource names to watch for changes.
+ * If no resources specified, watches all resources.
+ * If '*' is included in resources, triggers full page refresh.
+ *
+ * @example
+ * // Example 1: Album list - refresh when albums change
+ * const AlbumList = () => {
+ * useResourceRefresh('album')
+ * return ...
+ * }
+ *
+ * @example
+ * // Example 2: Album show page - refresh when album or its songs change
+ * const AlbumShow = () => {
+ * useResourceRefresh('album', 'song')
+ * return ...
+ * }
+ *
+ * @example
+ * // Example 3: Dashboard - refresh when any resource changes
+ * const Dashboard = () => {
+ * useResourceRefresh() // No parameters = watch all resources
+ * return ...
+ * }
+ *
+ * @example
+ * // Example 4: Library management page - watch library resources
+ * const LibraryList = () => {
+ * useResourceRefresh('library')
+ * return ...
+ * }
+ *
+ * **How it works:**
+ * - Listens to refresh events from the SSE connection
+ * - When events arrive, checks if they match the specified visible resources
+ * - For specific resource IDs: calls dataProvider.getMany(resource, {ids: [...]})
+ * - For global refreshes: calls refresh() to reload the entire page
+ * - Uses react-admin's built-in data management and caching
+ *
+ * **Event format expected:**
+ * - Global refresh: { '*': '*' } or { someResource: ['*'] }
+ * - Specific resources: { album: ['id1', 'id2'], song: ['id3'] }
+ */
export const useResourceRefresh = (...visibleResources) => {
const [lastTime, setLastTime] = useState(Date.now())
const refresh = useRefresh()
diff --git a/ui/src/common/useToggleLove.jsx b/ui/src/common/useToggleLove.jsx
index 6379d587e..3f98a2e21 100644
--- a/ui/src/common/useToggleLove.jsx
+++ b/ui/src/common/useToggleLove.jsx
@@ -17,18 +17,38 @@ export const useToggleLove = (resource, record = {}) => {
const dataProvider = useDataProvider()
const refreshRecord = useCallback(() => {
- dataProvider.getOne(resource, { id: record.id }).then(() => {
- if (mountedRef.current) {
- setLoading(false)
- }
- })
- }, [dataProvider, record.id, resource])
+ const promises = []
+
+ // Always refresh the original resource
+ const params = { id: record.id }
+ if (record.playlistId) {
+ params.filter = { playlist_id: record.playlistId }
+ }
+ promises.push(dataProvider.getOne(resource, params))
+
+ // If we have a mediaFileId, also refresh the song
+ if (record.mediaFileId) {
+ promises.push(dataProvider.getOne('song', { id: record.mediaFileId }))
+ }
+
+ Promise.all(promises)
+ .catch((e) => {
+ // eslint-disable-next-line no-console
+ console.log('Error encountered: ' + e)
+ })
+ .finally(() => {
+ if (mountedRef.current) {
+ setLoading(false)
+ }
+ })
+ }, [dataProvider, record.mediaFileId, record.id, record.playlistId, resource])
const toggleLove = () => {
const toggle = record.starred ? subsonic.unstar : subsonic.star
+ const id = record.mediaFileId || record.id
setLoading(true)
- toggle(record.id)
+ toggle(id)
.then(refreshRecord)
.catch((e) => {
// eslint-disable-next-line no-console
diff --git a/ui/src/common/useToggleLove.test.js b/ui/src/common/useToggleLove.test.js
new file mode 100644
index 000000000..640e9ff89
--- /dev/null
+++ b/ui/src/common/useToggleLove.test.js
@@ -0,0 +1,136 @@
+import { renderHook, act } from '@testing-library/react-hooks'
+import { vi, describe, it, expect, beforeEach } from 'vitest'
+import { useToggleLove } from './useToggleLove'
+import subsonic from '../subsonic'
+import { useDataProvider } from 'react-admin'
+
+vi.mock('../subsonic', () => ({
+ default: {
+ star: vi.fn(() => Promise.resolve()),
+ unstar: vi.fn(() => Promise.resolve()),
+ },
+}))
+
+vi.mock('react-admin', async () => {
+ const actual = await vi.importActual('react-admin')
+ return {
+ ...actual,
+ useDataProvider: vi.fn(),
+ useNotify: vi.fn(() => vi.fn()),
+ }
+})
+
+describe('useToggleLove', () => {
+ let getOne
+ beforeEach(() => {
+ getOne = vi.fn(() => Promise.resolve())
+ useDataProvider.mockReturnValue({ getOne })
+ vi.clearAllMocks()
+ })
+
+ it('uses mediaFileId when present', async () => {
+ const record = { id: 'pt-1', mediaFileId: 'sg-1', starred: false }
+ const { result } = renderHook(() => useToggleLove('song', record))
+ await act(async () => {
+ await result.current[0]()
+ })
+ expect(subsonic.star).toHaveBeenCalledWith('sg-1')
+ expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-1' })
+ })
+
+ it('falls back to id when mediaFileId not present', async () => {
+ const record = { id: 'sg-1', starred: false }
+ const { result } = renderHook(() => useToggleLove('song', record))
+ await act(async () => {
+ await result.current[0]()
+ })
+ expect(subsonic.star).toHaveBeenCalledWith('sg-1')
+ expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-1' })
+ })
+
+ it('calls unstar when record is already loved', async () => {
+ const record = { id: 'sg-1', starred: true }
+ const { result } = renderHook(() => useToggleLove('song', record))
+ await act(async () => {
+ await result.current[0]()
+ })
+ expect(subsonic.unstar).toHaveBeenCalledWith('sg-1')
+ })
+
+ describe('playlist track scenarios', () => {
+ it('refreshes both playlist track and song for playlist tracks', async () => {
+ const record = {
+ id: 'pt-1',
+ mediaFileId: 'sg-1',
+ playlistId: 'pl-1',
+ starred: false,
+ }
+ const { result } = renderHook(() =>
+ useToggleLove('playlistTrack', record),
+ )
+ await act(async () => {
+ await result.current[0]()
+ })
+
+ // Should star using the media file ID
+ expect(subsonic.star).toHaveBeenCalledWith('sg-1')
+
+ // Should refresh both the playlist track and the song
+ expect(getOne).toHaveBeenCalledTimes(2)
+ expect(getOne).toHaveBeenCalledWith('playlistTrack', {
+ id: 'pt-1',
+ filter: { playlist_id: 'pl-1' },
+ })
+ expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-1' })
+ })
+
+ it('includes playlist_id filter when refreshing playlist tracks', async () => {
+ const record = {
+ id: 'pt-5',
+ mediaFileId: 'sg-10',
+ playlistId: 'pl-123',
+ starred: true,
+ }
+ const { result } = renderHook(() =>
+ useToggleLove('playlistTrack', record),
+ )
+ await act(async () => {
+ await result.current[0]()
+ })
+
+ // Should unstar using the media file ID
+ expect(subsonic.unstar).toHaveBeenCalledWith('sg-10')
+
+ // Should refresh playlist track with correct playlist_id filter
+ expect(getOne).toHaveBeenCalledWith('playlistTrack', {
+ id: 'pt-5',
+ filter: { playlist_id: 'pl-123' },
+ })
+ // Should also refresh the underlying song
+ expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-10' })
+ })
+
+ it('only refreshes original resource when no mediaFileId present', async () => {
+ const record = { id: 'sg-1', starred: false }
+ const { result } = renderHook(() => useToggleLove('song', record))
+ await act(async () => {
+ await result.current[0]()
+ })
+
+ // Should only refresh the original resource (song)
+ expect(getOne).toHaveBeenCalledTimes(1)
+ expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-1' })
+ })
+
+ it('does not include playlist_id filter for non-playlist resources', async () => {
+ const record = { id: 'sg-1', starred: false }
+ const { result } = renderHook(() => useToggleLove('song', record))
+ await act(async () => {
+ await result.current[0]()
+ })
+
+ // Should refresh without any filter
+ expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-1' })
+ })
+ })
+})
diff --git a/ui/src/config.js b/ui/src/config.js
index 92ce07893..a53a97de7 100644
--- a/ui/src/config.js
+++ b/ui/src/config.js
@@ -29,7 +29,10 @@ const defaultConfig = {
listenBrainzEnabled: true,
enableExternalServices: true,
enableCoverAnimation: true,
+ enableNowPlaying: true,
devShowArtistPage: true,
+ devUIShowConfig: true,
+ devNewEventStream: false,
enableReplayGain: true,
defaultDownsamplingFormat: 'opus',
publicBaseUrl: '/share',
diff --git a/ui/src/dataProvider/wrapperDataProvider.js b/ui/src/dataProvider/wrapperDataProvider.js
index bf487dc7c..268d3668d 100644
--- a/ui/src/dataProvider/wrapperDataProvider.js
+++ b/ui/src/dataProvider/wrapperDataProvider.js
@@ -9,25 +9,73 @@ const isAdmin = () => {
return role === 'admin'
}
+const getSelectedLibraries = () => {
+ try {
+ const state = JSON.parse(localStorage.getItem('state'))
+ const selectedLibraries = state?.library?.selectedLibraries || []
+ const userLibraries = state?.library?.userLibraries || []
+
+ // Validate selected libraries against current user libraries
+ const userLibraryIds = userLibraries.map((lib) => lib.id)
+ const validatedSelection = selectedLibraries.filter((id) =>
+ userLibraryIds.includes(id),
+ )
+
+ // If user has only one library, return empty array (no filter needed)
+ if (userLibraryIds.length === 1) {
+ return []
+ }
+
+ return validatedSelection
+ } catch (err) {
+ return []
+ }
+}
+
+// Function to apply library filtering to appropriate resources
+const applyLibraryFilter = (resource, params) => {
+ // Content resources that should be filtered by selected libraries
+ const filteredResources = ['album', 'song', 'artist', 'playlistTrack', 'tag']
+
+ // Get selected libraries from localStorage
+ const selectedLibraries = getSelectedLibraries()
+
+ // Add library filter for content resources if libraries are selected
+ if (filteredResources.includes(resource) && selectedLibraries.length > 0) {
+ if (!params.filter) {
+ params.filter = {}
+ }
+ params.filter.library_id = selectedLibraries
+ }
+
+ return params
+}
+
const mapResource = (resource, params) => {
switch (resource) {
+ // /api/playlistTrack?playlist_id=123 => /api/playlist/123/tracks
case 'playlistTrack': {
- // /api/playlistTrack?playlist_id=123 => /api/playlist/123/tracks
+ params.filter = params.filter || {}
+
let plsId = '0'
- if (params.filter) {
- plsId = params.filter.playlist_id
- if (!isAdmin()) {
- params.filter.missing = false
- }
+ plsId = params.filter.playlist_id
+ if (!isAdmin()) {
+ params.filter.missing = false
}
+ params = applyLibraryFilter(resource, params)
+
return [`playlist/${plsId}/tracks`, params]
}
case 'album':
case 'song':
- case 'artist': {
- if (params.filter && !isAdmin()) {
+ case 'artist':
+ case 'tag': {
+ params.filter = params.filter || {}
+ if (!isAdmin()) {
params.filter.missing = false
}
+ params = applyLibraryFilter(resource, params)
+
return [resource, params]
}
default:
@@ -43,6 +91,60 @@ const callDeleteMany = (resource, params) => {
}).then((response) => ({ data: response.json.ids || [] }))
}
+// Helper function to handle user-library associations
+const handleUserLibraryAssociation = async (userId, libraryIds) => {
+ if (!libraryIds || libraryIds.length === 0) {
+ return // Admin users or users without library assignments
+ }
+
+ try {
+ await httpClient(`${REST_URL}/user/${userId}/library`, {
+ method: 'PUT',
+ body: JSON.stringify({ libraryIds }),
+ })
+ } catch (error) {
+ console.error('Error setting user libraries:', error) //eslint-disable-line no-console
+ throw error
+ }
+}
+
+// Enhanced user creation that handles library associations
+const createUser = async (params) => {
+ const { data } = params
+ const { libraryIds, ...userData } = data
+
+ // First create the user
+ const userResponse = await dataProvider.create('user', { data: userData })
+ const userId = userResponse.data.id
+
+ // Then set library associations for non-admin users
+ if (!userData.isAdmin && libraryIds && libraryIds.length > 0) {
+ await handleUserLibraryAssociation(userId, libraryIds)
+ }
+
+ return userResponse
+}
+
+// Enhanced user update that handles library associations
+const updateUser = async (params) => {
+ const { data } = params
+ const { libraryIds, ...userData } = data
+ const userId = params.id
+
+ // First update the user
+ const userResponse = await dataProvider.update('user', {
+ ...params,
+ data: userData,
+ })
+
+ // Then handle library associations for non-admin users
+ if (!userData.isAdmin && libraryIds !== undefined) {
+ await handleUserLibraryAssociation(userId, libraryIds)
+ }
+
+ return userResponse
+}
+
const wrapperDataProvider = {
...dataProvider,
getList: (resource, params) => {
@@ -51,7 +153,19 @@ const wrapperDataProvider = {
},
getOne: (resource, params) => {
const [r, p] = mapResource(resource, params)
- return dataProvider.getOne(r, p)
+ const response = dataProvider.getOne(r, p)
+
+ // Transform user data to ensure libraryIds is present for form compatibility
+ if (resource === 'user') {
+ return response.then((result) => {
+ if (result.data.libraries && Array.isArray(result.data.libraries)) {
+ result.data.libraryIds = result.data.libraries.map((lib) => lib.id)
+ }
+ return result
+ })
+ }
+
+ return response
},
getMany: (resource, params) => {
const [r, p] = mapResource(resource, params)
@@ -62,6 +176,9 @@ const wrapperDataProvider = {
return dataProvider.getManyReference(r, p)
},
update: (resource, params) => {
+ if (resource === 'user') {
+ return updateUser(params)
+ }
const [r, p] = mapResource(resource, params)
return dataProvider.update(r, p)
},
@@ -70,6 +187,9 @@ const wrapperDataProvider = {
return dataProvider.updateMany(r, p)
},
create: (resource, params) => {
+ if (resource === 'user') {
+ return createUser(params)
+ }
const [r, p] = mapResource(resource, params)
return dataProvider.create(r, p)
},
@@ -90,6 +210,16 @@ const wrapperDataProvider = {
body: JSON.stringify(data),
}).then(({ json }) => ({ data: json }))
},
+ getPlaylists: (songId) => {
+ return httpClient(`${REST_URL}/song/${songId}/playlists`).then(
+ ({ json }) => ({ data: json }),
+ )
+ },
+ inspect: (songId) => {
+ return httpClient(`${REST_URL}/inspect?id=${songId}`).then(({ json }) => ({
+ data: json,
+ }))
+ },
}
export default wrapperDataProvider
diff --git a/ui/src/dialogs/AboutDialog.jsx b/ui/src/dialogs/AboutDialog.jsx
index 4f074002b..661462b9b 100644
--- a/ui/src/dialogs/AboutDialog.jsx
+++ b/ui/src/dialogs/AboutDialog.jsx
@@ -10,14 +10,63 @@ import TableRow from '@material-ui/core/TableRow'
import TableCell from '@material-ui/core/TableCell'
import Paper from '@material-ui/core/Paper'
import FavoriteBorderIcon from '@material-ui/icons/FavoriteBorder'
+import FileCopyIcon from '@material-ui/icons/FileCopy'
+import Button from '@material-ui/core/Button'
import { humanize, underscore } from 'inflection'
-import { useGetOne, usePermissions, useTranslate } from 'react-admin'
+import { useGetOne, usePermissions, useTranslate, useNotify } from 'react-admin'
+import { Tabs, Tab } from '@material-ui/core'
+import { makeStyles } from '@material-ui/core/styles'
import config from '../config'
import { DialogTitle } from './DialogTitle'
import { DialogContent } from './DialogContent'
import { INSIGHTS_DOC_URL } from '../consts.js'
import subsonic from '../subsonic/index.js'
import { Typography } from '@material-ui/core'
+import TableHead from '@material-ui/core/TableHead'
+import { configToToml, separateAndSortConfigs } from './aboutUtils'
+
+const useStyles = makeStyles((theme) => ({
+ configNameColumn: {
+ maxWidth: '200px',
+ width: '200px',
+ wordWrap: 'break-word',
+ overflowWrap: 'break-word',
+ },
+ envVarColumn: {
+ maxWidth: '250px',
+ width: '250px',
+ fontFamily: 'monospace',
+ wordWrap: 'break-word',
+ overflowWrap: 'break-word',
+ },
+ copyButton: {
+ marginBottom: theme.spacing(2),
+ marginTop: theme.spacing(1),
+ },
+ devSectionHeader: {
+ '& td': {
+ paddingTop: theme.spacing(2),
+ paddingBottom: theme.spacing(2),
+ borderTop: `2px solid ${theme.palette.divider}`,
+ borderBottom: `1px solid ${theme.palette.divider}`,
+ textAlign: 'left',
+ fontWeight: 600,
+ },
+ },
+ configContainer: {
+ paddingTop: theme.spacing(1),
+ },
+ tableContainer: {
+ maxHeight: '60vh',
+ overflow: 'auto',
+ },
+ devFlagsTitle: {
+ fontWeight: 600,
+ },
+ expandableDialog: {
+ transition: 'max-width 300ms ease',
+ },
+}))
const links = {
homepage: 'navidrome.org',
@@ -54,7 +103,6 @@ const LinkToVersion = ({ version }) => {
const ShowVersion = ({ uiVersion, serverVersion }) => {
const translate = useTranslate()
-
const showRefresh = uiVersion !== serverVersion
return (
@@ -73,12 +121,16 @@ const ShowVersion = ({ uiVersion, serverVersion }) => {
UI {translate('menu.version')}:
-
- window.location.reload()}>
-
- {' ' + translate('ra.notification.new_version')}
-
-
+
+
+
+
+ window.location.reload()}>
+
+ {translate('ra.notification.new_version')}
+
+
+
)}
@@ -86,11 +138,285 @@ const ShowVersion = ({ uiVersion, serverVersion }) => {
)
}
-const AboutDialog = ({ open, onClose }) => {
+const AboutTabContent = ({
+ uiVersion,
+ serverVersion,
+ insightsData,
+ loading,
+ permissions,
+}) => {
const translate = useTranslate()
+
+ const lastRun = !loading && insightsData?.lastRun
+ let insightsStatus = 'N/A'
+ if (lastRun === 'disabled') {
+ insightsStatus = translate('about.links.insights.disabled')
+ } else if (lastRun && lastRun?.startsWith('1969-12-31')) {
+ insightsStatus = translate('about.links.insights.waiting')
+ } else if (lastRun) {
+ insightsStatus = lastRun
+ }
+
+ return (
+
+
+
+ {Object.keys(links).map((key) => {
+ return (
+
+
+ {translate(`about.links.${key}`, {
+ _: humanize(underscore(key)),
+ })}
+ :
+
+
+
+ {links[key]}
+
+
+
+ )
+ })}
+ {permissions === 'admin' ? (
+
+
+ {translate(`about.links.lastInsightsCollection`)}:
+
+
+ {insightsStatus}
+
+
+ ) : null}
+
+
+
+
+
+
+
+
+
+
+ ko-fi.com/deluan
+
+
+
+
+
+ )
+}
+
+const ConfigTabContent = ({ configData }) => {
+ const classes = useStyles()
+ const translate = useTranslate()
+ const notify = useNotify()
+
+ if (!configData || !configData.config) {
+ return null
+ }
+
+ // Use the shared separation and sorting logic
+ const { regularConfigs, devConfigs } = separateAndSortConfigs(
+ configData.config,
+ )
+
+ const handleCopyToml = async () => {
+ try {
+ const tomlContent = configToToml(configData, translate)
+ await navigator.clipboard.writeText(tomlContent)
+ notify(translate('about.config.exportSuccess'), 'info')
+ } catch (err) {
+ // eslint-disable-next-line no-console
+ console.error('Failed to copy TOML:', err)
+ notify(translate('about.config.exportFailed'), 'error')
+ }
+ }
+
+ return (
+
+
}
+ onClick={handleCopyToml}
+ className={classes.copyButton}
+ disabled={!configData}
+ size="small"
+ >
+ {translate('about.config.exportToml')}
+
+
+
+
+
+
+ {translate('about.config.configName')}
+
+
+ {translate('about.config.environmentVariable')}
+
+
+ {translate('about.config.currentValue')}
+
+
+
+
+ {configData?.configFile && (
+
+
+ {translate('about.config.configurationFile')}
+
+
+ ND_CONFIGFILE
+
+ {configData.configFile}
+
+ )}
+ {regularConfigs.map(({ key, envVar, value }) => (
+
+
+ {key}
+
+
+ {envVar}
+
+ {String(value)}
+
+ ))}
+ {devConfigs.length > 0 && (
+
+
+
+ 🚧 {translate('about.config.devFlagsHeader')}
+
+
+
+ )}
+ {devConfigs.map(({ key, envVar, value }) => (
+
+
+ {key}
+
+
+ {envVar}
+
+ {String(value)}
+
+ ))}
+
+
+
+
+ )
+}
+
+const TabContent = ({
+ tab,
+ setTab,
+ showConfigTab,
+ uiVersion,
+ serverVersion,
+ insightsData,
+ loading,
+ permissions,
+ configData,
+}) => {
+ const translate = useTranslate()
+
+ return (
+
+ {showConfigTab && (
+ setTab(value)}>
+
+
+
+ )}
+
+ {showConfigTab && (
+
+
+
+ )}
+
+ )
+}
+
+const AboutDialog = ({ open, onClose }) => {
+ const classes = useStyles()
const { permissions } = usePermissions()
- const { data, loading } = useGetOne('insights', 'insights_status')
+ const { data: insightsData, loading } = useGetOne(
+ 'insights',
+ 'insights_status',
+ )
const [serverVersion, setServerVersion] = useState('')
+ const showConfigTab = permissions === 'admin' && config.devUIShowConfig
+ const [tab, setTab] = useState(0)
+ const { data: configData } = useGetOne('config', 'config', {
+ enabled: showConfigTab,
+ })
+ const expanded = showConfigTab && tab === 1
const uiVersion = config.version
useEffect(() => {
@@ -108,85 +434,30 @@ const AboutDialog = ({ open, onClose }) => {
})
}, [setServerVersion])
- const lastRun = !loading && data?.lastRun
- let insightsStatus = 'N/A'
- if (lastRun === 'disabled') {
- insightsStatus = translate('about.links.insights.disabled')
- } else if (lastRun && lastRun?.startsWith('1969-12-31')) {
- insightsStatus = translate('about.links.insights.waiting')
- } else if (lastRun) {
- insightsStatus = lastRun
- }
-
return (
-
+
Navidrome Music Server
-
-
-
-
- {Object.keys(links).map((key) => {
- return (
-
-
- {translate(`about.links.${key}`, {
- _: humanize(underscore(key)),
- })}
- :
-
-
-
- {links[key]}
-
-
-
- )
- })}
- {permissions === 'admin' ? (
-
-
- {translate(`about.links.lastInsightsCollection`)}:
-
-
- {insightsStatus}
-
-
- ) : null}
-
-
-
-
-
-
-
-
-
-
- ko-fi.com/deluan
-
-
-
-
-
-
+
)
diff --git a/ui/src/dialogs/AddToPlaylistDialog.jsx b/ui/src/dialogs/AddToPlaylistDialog.jsx
index d1f80a432..91521d1ce 100644
--- a/ui/src/dialogs/AddToPlaylistDialog.jsx
+++ b/ui/src/dialogs/AddToPlaylistDialog.jsx
@@ -12,6 +12,7 @@ import {
DialogActions,
DialogContent,
DialogTitle,
+ makeStyles,
} from '@material-ui/core'
import {
closeAddToPlaylist,
@@ -23,7 +24,21 @@ import DuplicateSongDialog from './DuplicateSongDialog'
import { httpClient } from '../dataProvider'
import { REST_URL } from '../consts'
+const useStyles = makeStyles({
+ dialogPaper: {
+ height: '26em',
+ maxHeight: '26em',
+ },
+ dialogContent: {
+ height: '17.5em',
+ overflowY: 'auto',
+ paddingTop: '0.5em',
+ paddingBottom: '0.5em',
+ },
+})
+
export const AddToPlaylistDialog = () => {
+ const classes = useStyles()
const { open, selectedIds, onSuccess, duplicateSong, duplicateIds } =
useSelector((state) => state.addToPlaylistDialog)
const dispatch = useDispatch()
@@ -145,11 +160,14 @@ export const AddToPlaylistDialog = () => {
aria-labelledby="form-dialog-new-playlist"
fullWidth={true}
maxWidth={'sm'}
+ classes={{
+ paper: classes.dialogPaper,
+ }}
>
{translate('resources.playlist.actions.selectPlaylist')}
-
+
diff --git a/ui/src/dialogs/AddToPlaylistDialog.test.jsx b/ui/src/dialogs/AddToPlaylistDialog.test.jsx
index a019ee03c..60d3cca0d 100644
--- a/ui/src/dialogs/AddToPlaylistDialog.test.jsx
+++ b/ui/src/dialogs/AddToPlaylistDialog.test.jsx
@@ -88,12 +88,18 @@ describe('AddToPlaylistDialog', () => {
createTestUtils(mockDataProvider)
+ // Filter to see sample playlists
let textBox = screen.getByRole('textbox')
fireEvent.change(textBox, { target: { value: 'sample' } })
- fireEvent.keyDown(textBox, { key: 'ArrowDown' })
- fireEvent.keyDown(textBox, { key: 'Enter' })
- fireEvent.keyDown(textBox, { key: 'ArrowDown' })
- fireEvent.keyDown(textBox, { key: 'Enter' })
+
+ // Click on first playlist
+ const firstPlaylist = screen.getByText('sample playlist 1')
+ fireEvent.click(firstPlaylist)
+
+ // Click on second playlist
+ const secondPlaylist = screen.getByText('sample playlist 2')
+ fireEvent.click(secondPlaylist)
+
await waitFor(() => {
expect(screen.getByTestId('playlist-add')).not.toBeDisabled()
})
@@ -133,12 +139,11 @@ describe('AddToPlaylistDialog', () => {
createTestUtils(mockDataProvider)
+ // Type a new playlist name and press Enter to create it
let textBox = screen.getByRole('textbox')
fireEvent.change(textBox, { target: { value: 'sample' } })
- fireEvent.keyDown(textBox, { key: 'ArrowDown' })
- fireEvent.keyDown(textBox, { key: 'ArrowDown' })
- fireEvent.keyDown(textBox, { key: 'ArrowDown' })
fireEvent.keyDown(textBox, { key: 'Enter' })
+
await waitFor(() => {
expect(screen.getByTestId('playlist-add')).not.toBeDisabled()
})
@@ -171,14 +176,15 @@ describe('AddToPlaylistDialog', () => {
createTestUtils(mockDataProvider)
+ // Create first playlist
let textBox = screen.getByRole('textbox')
fireEvent.change(textBox, { target: { value: 'sample' } })
- fireEvent.keyDown(textBox, { key: 'ArrowDown' })
- fireEvent.keyDown(textBox, { key: 'ArrowDown' })
- fireEvent.keyDown(textBox, { key: 'ArrowDown' })
fireEvent.keyDown(textBox, { key: 'Enter' })
+
+ // Create second playlist
fireEvent.change(textBox, { target: { value: 'new playlist' } })
fireEvent.keyDown(textBox, { key: 'Enter' })
+
await waitFor(() => {
expect(screen.getByTestId('playlist-add')).not.toBeDisabled()
})
diff --git a/ui/src/dialogs/SaveQueueDialog.jsx b/ui/src/dialogs/SaveQueueDialog.jsx
index 69f07dab7..f916a0793 100644
--- a/ui/src/dialogs/SaveQueueDialog.jsx
+++ b/ui/src/dialogs/SaveQueueDialog.jsx
@@ -57,7 +57,10 @@ export const SaveQueueDialog = () => {
return res
})
.then((res) => {
- notify('ra.notification.created', 'info', { smart_count: 1 })
+ notify('ra.notification.created', {
+ type: 'info',
+ messageArgs: { smart_count: 1 },
+ })
dispatch(closeSaveQueueDialog())
refresh()
history.push(`/playlist/${res.data.id}/show`)
diff --git a/ui/src/dialogs/SelectPlaylistInput.jsx b/ui/src/dialogs/SelectPlaylistInput.jsx
index 372d5f408..d401dd822 100644
--- a/ui/src/dialogs/SelectPlaylistInput.jsx
+++ b/ui/src/dialogs/SelectPlaylistInput.jsx
@@ -1,26 +1,265 @@
-import React from 'react'
+import React, { useState } from 'react'
import TextField from '@material-ui/core/TextField'
import Checkbox from '@material-ui/core/Checkbox'
import CheckBoxIcon from '@material-ui/icons/CheckBox'
import CheckBoxOutlineBlankIcon from '@material-ui/icons/CheckBoxOutlineBlank'
-import Autocomplete, {
- createFilterOptions,
-} from '@material-ui/lab/Autocomplete'
+import {
+ List,
+ ListItem,
+ ListItemIcon,
+ ListItemText,
+ Typography,
+ Box,
+ InputAdornment,
+ IconButton,
+} from '@material-ui/core'
+import AddIcon from '@material-ui/icons/Add'
import { useGetList, useTranslate } from 'react-admin'
import PropTypes from 'prop-types'
import { isWritable } from '../common'
import { makeStyles } from '@material-ui/core'
-const filter = createFilterOptions()
+const useStyles = makeStyles((theme) => ({
+ root: {
+ width: '100%',
+ height: '100%',
+ display: 'flex',
+ flexDirection: 'column',
+ },
+ searchField: {
+ marginBottom: theme.spacing(2),
+ width: '100%',
+ flexShrink: 0,
+ },
+ playlistList: {
+ flex: 1,
+ minHeight: 0,
+ overflow: 'auto',
+ border: `1px solid ${theme.palette.divider}`,
+ borderRadius: theme.shape.borderRadius,
+ backgroundColor: theme.palette.background.paper,
+ },
+ listItem: {
+ paddingTop: 0,
+ paddingBottom: 0,
+ },
+ createIcon: {
+ fontSize: '1.25rem',
+ margin: '9px',
+ },
+ selectedPlaylistsContainer: {
+ marginTop: theme.spacing(2),
+ flexShrink: 0,
+ maxHeight: '30%',
+ overflow: 'auto',
+ },
+ selectedPlaylist: {
+ display: 'inline-flex',
+ alignItems: 'center',
+ margin: theme.spacing(0.5),
+ padding: theme.spacing(0.5, 1),
+ backgroundColor: theme.palette.primary.main,
+ color: theme.palette.primary.contrastText,
+ borderRadius: theme.shape.borderRadius,
+ fontSize: '0.875rem',
+ },
+ removeButton: {
+ marginLeft: theme.spacing(0.5),
+ padding: 2,
+ color: 'inherit',
+ },
+ emptyMessage: {
+ padding: theme.spacing(2),
+ textAlign: 'center',
+ color: theme.palette.text.secondary,
+ },
+}))
-const useStyles = makeStyles({
- root: { width: '100%' },
- checkbox: { marginRight: 8 },
-})
+const PlaylistSearchField = ({
+ searchText,
+ onSearchChange,
+ onCreateNew,
+ onKeyDown,
+ canCreateNew,
+}) => {
+ const classes = useStyles()
+ const translate = useTranslate()
+
+ return (
+ onSearchChange(e.target.value)}
+ onKeyDown={onKeyDown}
+ placeholder={translate('resources.playlist.actions.searchOrCreate')}
+ InputProps={{
+ endAdornment: canCreateNew && (
+
+
+
+
+
+ ),
+ }}
+ />
+ )
+}
+
+const EmptyPlaylistMessage = ({ searchText, canCreateNew }) => {
+ const classes = useStyles()
+ const translate = useTranslate()
+
+ return (
+
+
+ {searchText
+ ? translate('resources.playlist.message.noPlaylistsFound')
+ : translate('resources.playlist.message.noPlaylists')}
+
+ {canCreateNew && (
+
+ {translate('resources.playlist.actions.pressEnterToCreate')}
+
+ )}
+
+ )
+}
+
+const PlaylistListItem = ({ playlist, isSelected, onToggle }) => {
+ const classes = useStyles()
+
+ return (
+ onToggle(playlist)}
+ dense
+ >
+
+ }
+ checkedIcon={ }
+ checked={isSelected}
+ tabIndex={-1}
+ disableRipple
+ />
+
+
+
+ )
+}
+
+const CreatePlaylistItem = ({ searchText, onCreateNew }) => {
+ const classes = useStyles()
+ const translate = useTranslate()
+
+ return (
+
+
+
+
+
+
+ )
+}
+
+const PlaylistList = ({
+ filteredOptions,
+ selectedPlaylists,
+ onPlaylistToggle,
+ searchText,
+ canCreateNew,
+ onCreateNew,
+}) => {
+ const classes = useStyles()
+
+ const isPlaylistSelected = (playlist) =>
+ selectedPlaylists.some((p) => p.id === playlist.id)
+
+ return (
+
+ {filteredOptions.length === 0 ? (
+
+ ) : (
+ filteredOptions.map((playlist) => (
+
+ ))
+ )}
+ {canCreateNew && filteredOptions.length > 0 && (
+
+ )}
+
+ )
+}
+
+const SelectedPlaylistChip = ({ playlist, onRemove }) => {
+ const classes = useStyles()
+ const translate = useTranslate()
+
+ return (
+
+ {playlist.name}
+ onRemove(playlist)}
+ title={translate('resources.playlist.actions.removeFromSelection')}
+ >
+ {'×'}
+
+
+ )
+}
+
+const SelectedPlaylistsDisplay = ({ selectedPlaylists, onRemoveSelected }) => {
+ const classes = useStyles()
+ const translate = useTranslate()
+
+ if (selectedPlaylists.length === 0) {
+ return null
+ }
+
+ return (
+
+
+ {selectedPlaylists.map((playlist, index) => (
+
+ ))}
+
+
+ )
+}
export const SelectPlaylistInput = ({ onChange }) => {
const classes = useStyles()
- const translate = useTranslate()
+ const [searchText, setSearchText] = useState('')
+ const [selectedPlaylists, setSelectedPlaylists] = useState([])
+
const { ids, data } = useGetList(
'playlist',
{ page: 1, perPage: -1 },
@@ -32,92 +271,131 @@ export const SelectPlaylistInput = ({ onChange }) => {
ids &&
ids.map((id) => data[id]).filter((option) => isWritable(option.ownerId))
- const handleOnChange = (event, newValue) => {
- let newState = []
- if (newValue && newValue.length) {
- newValue.forEach((playlistObject) => {
- if (playlistObject.inputValue) {
- newState.push({
- name: playlistObject.inputValue,
- })
- } else if (typeof playlistObject === 'string') {
- newState.push({
- name: playlistObject,
- })
- } else {
- newState.push(playlistObject)
- }
- })
+ // Filter playlists based on search text
+ const filteredOptions =
+ options?.filter((option) =>
+ option.name.toLowerCase().includes(searchText.toLowerCase()),
+ ) || []
+
+ const handlePlaylistToggle = (playlist) => {
+ const isSelected = selectedPlaylists.some((p) => p.id === playlist.id)
+ let newSelection
+
+ if (isSelected) {
+ newSelection = selectedPlaylists.filter((p) => p.id !== playlist.id)
+ } else {
+ newSelection = [...selectedPlaylists, playlist]
}
- onChange(newState)
+
+ setSelectedPlaylists(newSelection)
+ onChange(newSelection)
}
- const icon =
- const checkedIcon =
+ const handleRemoveSelected = (playlistToRemove) => {
+ const newSelection = selectedPlaylists.filter(
+ (p) => p.id !== playlistToRemove.id,
+ )
+ setSelectedPlaylists(newSelection)
+ onChange(newSelection)
+ }
+
+ const handleCreateNew = () => {
+ if (searchText.trim()) {
+ const newPlaylist = { name: searchText.trim() }
+ const newSelection = [...selectedPlaylists, newPlaylist]
+ setSelectedPlaylists(newSelection)
+ onChange(newSelection)
+ setSearchText('')
+ }
+ }
+
+ const handleKeyDown = (e) => {
+ if (e.key === 'Enter' && searchText.trim()) {
+ e.preventDefault()
+ handleCreateNew()
+ }
+ }
+
+ const canCreateNew = Boolean(
+ searchText.trim() &&
+ !filteredOptions.some(
+ (option) =>
+ option.name.toLowerCase() === searchText.toLowerCase().trim(),
+ ) &&
+ !selectedPlaylists.some((p) => p.name === searchText.trim()),
+ )
return (
- {
- const filtered = filter(options, params)
+
+
- // Suggest the creation of a new value
- if (params.inputValue !== '') {
- filtered.push({
- inputValue: params.inputValue,
- name: translate('resources.playlist.actions.addNewPlaylist', {
- name: params.inputValue,
- }),
- })
- }
+
- return filtered
- }}
- clearOnBlur
- handleHomeEndKeys
- openOnFocus
- selectOnFocus
- id="select-playlist-input"
- options={options}
- getOptionLabel={(option) => {
- // Value selected with enter, right from the input
- if (typeof option === 'string') {
- return option
- }
- // Add "xxx" option created dynamically
- if (option.inputValue) {
- return option.inputValue
- }
- // Regular option
- return option.name
- }}
- renderOption={(option, { selected }) => (
-
-
- {option.name}
-
- )}
- className={classes.root}
- freeSolo
- renderInput={(params) => (
-
- )}
- />
+
+
)
}
SelectPlaylistInput.propTypes = {
onChange: PropTypes.func.isRequired,
}
+
+// PropTypes for sub-components
+PlaylistSearchField.propTypes = {
+ searchText: PropTypes.string.isRequired,
+ onSearchChange: PropTypes.func.isRequired,
+ onCreateNew: PropTypes.func.isRequired,
+ onKeyDown: PropTypes.func.isRequired,
+ canCreateNew: PropTypes.bool.isRequired,
+}
+
+EmptyPlaylistMessage.propTypes = {
+ searchText: PropTypes.string.isRequired,
+ canCreateNew: PropTypes.bool.isRequired,
+}
+
+PlaylistListItem.propTypes = {
+ playlist: PropTypes.object.isRequired,
+ isSelected: PropTypes.bool.isRequired,
+ onToggle: PropTypes.func.isRequired,
+}
+
+CreatePlaylistItem.propTypes = {
+ searchText: PropTypes.string.isRequired,
+ onCreateNew: PropTypes.func.isRequired,
+}
+
+PlaylistList.propTypes = {
+ filteredOptions: PropTypes.array.isRequired,
+ selectedPlaylists: PropTypes.array.isRequired,
+ onPlaylistToggle: PropTypes.func.isRequired,
+ searchText: PropTypes.string.isRequired,
+ canCreateNew: PropTypes.bool.isRequired,
+ onCreateNew: PropTypes.func.isRequired,
+}
+
+SelectedPlaylistChip.propTypes = {
+ playlist: PropTypes.object.isRequired,
+ onRemove: PropTypes.func.isRequired,
+}
+
+SelectedPlaylistsDisplay.propTypes = {
+ selectedPlaylists: PropTypes.array.isRequired,
+ onRemoveSelected: PropTypes.func.isRequired,
+}
diff --git a/ui/src/dialogs/SelectPlaylistInput.test.jsx b/ui/src/dialogs/SelectPlaylistInput.test.jsx
index 727126965..4ffcdf0b6 100644
--- a/ui/src/dialogs/SelectPlaylistInput.test.jsx
+++ b/ui/src/dialogs/SelectPlaylistInput.test.jsx
@@ -11,115 +11,479 @@ import {
import { SelectPlaylistInput } from './SelectPlaylistInput'
import { describe, beforeAll, afterEach, it, expect, vi } from 'vitest'
-describe('SelectPlaylistInput', () => {
- beforeAll(() => localStorage.setItem('userId', 'admin'))
- afterEach(cleanup)
- const onChangeHandler = vi.fn()
+const mockPlaylists = [
+ { id: 'playlist-1', name: 'Rock Classics', ownerId: 'admin' },
+ { id: 'playlist-2', name: 'Jazz Collection', ownerId: 'admin' },
+ { id: 'playlist-3', name: 'Electronic Beats', ownerId: 'admin' },
+ { id: 'playlist-4', name: 'Chill Vibes', ownerId: 'user2' }, // Not writable by admin
+]
- it('should call the handler with the selections', async () => {
- const mockData = [
- { id: 'sample-id1', name: 'sample playlist 1', ownerId: 'admin' },
- { id: 'sample-id2', name: 'sample playlist 2', ownerId: 'admin' },
- ]
- const mockIndexedData = {
- 'sample-id1': {
- id: 'sample-id1',
- name: 'sample playlist 1',
- ownerId: 'admin',
- },
- 'sample-id2': {
- id: 'sample-id2',
- name: 'sample playlist 2',
- ownerId: 'admin',
- },
- }
+const mockIndexedData = {
+ 'playlist-1': { id: 'playlist-1', name: 'Rock Classics', ownerId: 'admin' },
+ 'playlist-2': { id: 'playlist-2', name: 'Jazz Collection', ownerId: 'admin' },
+ 'playlist-3': {
+ id: 'playlist-3',
+ name: 'Electronic Beats',
+ ownerId: 'admin',
+ },
+ 'playlist-4': { id: 'playlist-4', name: 'Chill Vibes', ownerId: 'user2' },
+}
- const mockDataProvider = {
- getList: vi
- .fn()
- .mockResolvedValue({ data: mockData, total: mockData.length }),
- }
+const createTestComponent = (
+ mockDataProvider = null,
+ onChangeMock = vi.fn(),
+ playlists = mockPlaylists,
+ indexedData = mockIndexedData,
+) => {
+ const dataProvider = mockDataProvider || {
+ getList: vi.fn().mockResolvedValue({
+ data: playlists,
+ total: playlists.length,
+ }),
+ }
- render(
-
-
+
-
-
- ,
- )
+ },
+ }}
+ >
+
+
+ ,
+ )
+}
- await waitFor(() => {
- expect(mockDataProvider.getList).toHaveBeenCalledWith('playlist', {
- filter: { smart: false },
- pagination: { page: 1, perPage: -1 },
- sort: { field: 'name', order: 'ASC' },
+describe('SelectPlaylistInput', () => {
+ beforeAll(() => localStorage.setItem('userId', 'admin'))
+ afterEach(cleanup)
+
+ describe('Basic Functionality', () => {
+ it('should render search field and playlist list', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByRole('textbox')).toBeInTheDocument()
+ expect(screen.getByText('Rock Classics')).toBeInTheDocument()
+ expect(screen.getByText('Jazz Collection')).toBeInTheDocument()
+ expect(screen.getByText('Electronic Beats')).toBeInTheDocument()
+ })
+
+ // Should not show playlists not owned by admin (not writable)
+ expect(screen.queryByText('Chill Vibes')).not.toBeInTheDocument()
+ })
+
+ it('should filter playlists based on search input', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByText('Rock Classics')).toBeInTheDocument()
+ })
+
+ const searchInput = screen.getByRole('textbox')
+ fireEvent.change(searchInput, { target: { value: 'rock' } })
+
+ await waitFor(() => {
+ expect(screen.getByText('Rock Classics')).toBeInTheDocument()
+ expect(screen.queryByText('Jazz Collection')).not.toBeInTheDocument()
+ expect(screen.queryByText('Electronic Beats')).not.toBeInTheDocument()
})
})
- let textBox = screen.getByRole('textbox')
- fireEvent.change(textBox, { target: { value: 'sample' } })
- fireEvent.keyDown(textBox, { key: 'ArrowDown' })
- fireEvent.keyDown(textBox, { key: 'Enter' })
- await waitFor(() => {
- expect(onChangeHandler).toHaveBeenCalledWith([
- { id: 'sample-id1', name: 'sample playlist 1', ownerId: 'admin' },
- ])
+ it('should handle case-insensitive search', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByText('Jazz Collection')).toBeInTheDocument()
+ })
+
+ const searchInput = screen.getByRole('textbox')
+ fireEvent.change(searchInput, { target: { value: 'JAZZ' } })
+
+ await waitFor(() => {
+ expect(screen.getByText('Jazz Collection')).toBeInTheDocument()
+ expect(screen.queryByText('Rock Classics')).not.toBeInTheDocument()
+ })
+ })
+ })
+
+ describe('Playlist Selection', () => {
+ it('should select and deselect playlists by clicking', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByText('Rock Classics')).toBeInTheDocument()
+ })
+
+ // Select first playlist
+ const rockPlaylist = screen.getByText('Rock Classics')
+ fireEvent.click(rockPlaylist)
+
+ await waitFor(() => {
+ expect(onChangeMock).toHaveBeenCalledWith([
+ { id: 'playlist-1', name: 'Rock Classics', ownerId: 'admin' },
+ ])
+ })
+
+ // Select second playlist
+ const jazzPlaylist = screen.getByText('Jazz Collection')
+ fireEvent.click(jazzPlaylist)
+
+ await waitFor(() => {
+ expect(onChangeMock).toHaveBeenCalledWith([
+ { id: 'playlist-1', name: 'Rock Classics', ownerId: 'admin' },
+ { id: 'playlist-2', name: 'Jazz Collection', ownerId: 'admin' },
+ ])
+ })
+
+ // Deselect first playlist
+ fireEvent.click(rockPlaylist)
+
+ await waitFor(() => {
+ expect(onChangeMock).toHaveBeenCalledWith([
+ { id: 'playlist-2', name: 'Jazz Collection', ownerId: 'admin' },
+ ])
+ })
})
- fireEvent.keyDown(textBox, { key: 'ArrowDown' })
- fireEvent.keyDown(textBox, { key: 'Enter' })
- await waitFor(() => {
- expect(onChangeHandler).toHaveBeenCalledWith([
- { id: 'sample-id1', name: 'sample playlist 1', ownerId: 'admin' },
- { id: 'sample-id2', name: 'sample playlist 2', ownerId: 'admin' },
- ])
+ it('should show selected playlists as chips', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByText('Rock Classics')).toBeInTheDocument()
+ })
+
+ // Select a playlist
+ const rockPlaylist = screen.getByText('Rock Classics')
+ fireEvent.click(rockPlaylist)
+
+ await waitFor(() => {
+ // Should show the selected playlist as a chip
+ const chips = screen.getAllByText('Rock Classics')
+ expect(chips.length).toBeGreaterThan(1) // One in list, one in chip
+ })
})
- fireEvent.change(textBox, {
- target: { value: 'new playlist' },
+ it('should remove selected playlists via chip remove button', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByText('Rock Classics')).toBeInTheDocument()
+ })
+
+ // Select a playlist
+ const rockPlaylist = screen.getByText('Rock Classics')
+ fireEvent.click(rockPlaylist)
+
+ await waitFor(() => {
+ // Should show selected playlist as chip
+ const chips = screen.getAllByText('Rock Classics')
+ expect(chips.length).toBeGreaterThan(1)
+ })
+
+ // Find and click the remove button (translation key)
+ const removeButton = screen.getByText('×')
+ fireEvent.click(removeButton)
+
+ await waitFor(() => {
+ expect(onChangeMock).toHaveBeenCalledWith([])
+ // Should only have one instance (in the list) after removal
+ const remainingChips = screen.getAllByText('Rock Classics')
+ expect(remainingChips.length).toBe(1)
+ })
})
- fireEvent.keyDown(textBox, { key: 'ArrowDown' })
- fireEvent.keyDown(textBox, { key: 'Enter' })
- await waitFor(() => {
- expect(onChangeHandler).toHaveBeenCalledWith([
- { id: 'sample-id1', name: 'sample playlist 1', ownerId: 'admin' },
- { id: 'sample-id2', name: 'sample playlist 2', ownerId: 'admin' },
- { name: 'new playlist' },
- ])
+ })
+
+ describe('Create New Playlist', () => {
+ it('should create new playlist by pressing Enter', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByRole('textbox')).toBeInTheDocument()
+ })
+
+ const searchInput = screen.getByRole('textbox')
+ fireEvent.change(searchInput, { target: { value: 'My New Playlist' } })
+ fireEvent.keyDown(searchInput, { key: 'Enter' })
+
+ await waitFor(() => {
+ expect(onChangeMock).toHaveBeenCalledWith([{ name: 'My New Playlist' }])
+ })
+
+ // Input should be cleared after creating
+ expect(searchInput.value).toBe('')
})
- fireEvent.change(textBox, {
- target: { value: 'another new playlist' },
+ it('should create new playlist by clicking add button', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByRole('textbox')).toBeInTheDocument()
+ })
+
+ const searchInput = screen.getByRole('textbox')
+ fireEvent.change(searchInput, { target: { value: 'Another Playlist' } })
+
+ // Find the add button by the translation key title
+ const addButton = screen.getByTitle(
+ 'resources.playlist.actions.addNewPlaylist',
+ )
+ fireEvent.click(addButton)
+
+ await waitFor(() => {
+ expect(onChangeMock).toHaveBeenCalledWith([
+ { name: 'Another Playlist' },
+ ])
+ })
})
- fireEvent.keyDown(textBox, { key: 'Enter' })
- await waitFor(() => {
- expect(onChangeHandler).toHaveBeenCalledWith([
- { id: 'sample-id1', name: 'sample playlist 1', ownerId: 'admin' },
- { id: 'sample-id2', name: 'sample playlist 2', ownerId: 'admin' },
- { name: 'new playlist' },
- { name: 'another new playlist' },
- ])
+
+ it('should not show create option for existing playlist names', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByRole('textbox')).toBeInTheDocument()
+ })
+
+ const searchInput = screen.getByRole('textbox')
+ fireEvent.change(searchInput, { target: { value: 'Rock Classics' } })
+
+ await waitFor(() => {
+ expect(
+ screen.queryByText('resources.playlist.actions.addNewPlaylist'),
+ ).not.toBeInTheDocument()
+ })
+ })
+
+ it('should not create playlist with empty name', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByRole('textbox')).toBeInTheDocument()
+ })
+
+ const searchInput = screen.getByRole('textbox')
+ fireEvent.change(searchInput, { target: { value: ' ' } }) // Only spaces
+ fireEvent.keyDown(searchInput, { key: 'Enter' })
+
+ // Should not call onChange
+ expect(onChangeMock).not.toHaveBeenCalled()
+ })
+
+ it('should show create options in appropriate contexts', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByRole('textbox')).toBeInTheDocument()
+ })
+
+ const searchInput = screen.getByRole('textbox')
+
+ // When typing a new name, should show create options
+ fireEvent.change(searchInput, { target: { value: 'My New Playlist' } })
+
+ await waitFor(() => {
+ // Should show the add button in the search field
+ expect(
+ screen.getByTitle('resources.playlist.actions.addNewPlaylist'),
+ ).toBeInTheDocument()
+ // Should also show hint in empty message when no matches
+ expect(
+ screen.getByText('resources.playlist.actions.pressEnterToCreate'),
+ ).toBeInTheDocument()
+ })
+ })
+ })
+
+ describe('Mixed Operations', () => {
+ it('should handle selecting existing playlists and creating new ones', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByText('Rock Classics')).toBeInTheDocument()
+ })
+
+ // Select existing playlist
+ const rockPlaylist = screen.getByText('Rock Classics')
+ fireEvent.click(rockPlaylist)
+
+ await waitFor(() => {
+ expect(onChangeMock).toHaveBeenCalledWith([
+ { id: 'playlist-1', name: 'Rock Classics', ownerId: 'admin' },
+ ])
+ })
+
+ // Create new playlist
+ const searchInput = screen.getByRole('textbox')
+ fireEvent.change(searchInput, { target: { value: 'New Mix' } })
+ fireEvent.keyDown(searchInput, { key: 'Enter' })
+
+ await waitFor(() => {
+ expect(onChangeMock).toHaveBeenCalledWith([
+ { id: 'playlist-1', name: 'Rock Classics', ownerId: 'admin' },
+ { name: 'New Mix' },
+ ])
+ })
+ })
+
+ it('should maintain selections when searching', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByText('Rock Classics')).toBeInTheDocument()
+ })
+
+ // Select a playlist
+ const rockPlaylist = screen.getByText('Rock Classics')
+ fireEvent.click(rockPlaylist)
+
+ // Filter the list
+ const searchInput = screen.getByRole('textbox')
+ fireEvent.change(searchInput, { target: { value: 'jazz' } })
+
+ await waitFor(() => {
+ // Should still show selected playlists section
+ // Rock Classics should still be visible as a selected chip even though filtered out
+ expect(screen.getByText('Rock Classics')).toBeInTheDocument() // In selected chips
+ expect(screen.getByText('Jazz Collection')).toBeInTheDocument()
+ })
+ })
+ })
+
+ describe('Empty States', () => {
+ it('should show empty message when no playlists exist', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock, [], {})
+
+ await waitFor(() => {
+ expect(
+ screen.getByText('resources.playlist.message.noPlaylists'),
+ ).toBeInTheDocument()
+ })
+ })
+
+ it('should show "no results" message when search returns no matches', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByRole('textbox')).toBeInTheDocument()
+ })
+
+ const searchInput = screen.getByRole('textbox')
+ fireEvent.change(searchInput, {
+ target: { value: 'NonExistentPlaylist' },
+ })
+
+ await waitFor(() => {
+ expect(
+ screen.getByText('resources.playlist.message.noPlaylistsFound'),
+ ).toBeInTheDocument()
+ expect(
+ screen.getByText('resources.playlist.actions.pressEnterToCreate'),
+ ).toBeInTheDocument()
+ })
+ })
+ })
+
+ describe('Keyboard Navigation', () => {
+ it('should not create playlist on Enter if input is empty', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByRole('textbox')).toBeInTheDocument()
+ })
+
+ const searchInput = screen.getByRole('textbox')
+ fireEvent.keyDown(searchInput, { key: 'Enter' })
+
+ expect(onChangeMock).not.toHaveBeenCalled()
+ })
+
+ it('should handle other keys without side effects', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByRole('textbox')).toBeInTheDocument()
+ })
+
+ const searchInput = screen.getByRole('textbox')
+ fireEvent.change(searchInput, { target: { value: 'test' } })
+ fireEvent.keyDown(searchInput, { key: 'ArrowDown' })
+ fireEvent.keyDown(searchInput, { key: 'Tab' })
+ fireEvent.keyDown(searchInput, { key: 'Escape' })
+
+ // Should not create playlist or trigger onChange
+ expect(onChangeMock).not.toHaveBeenCalled()
+ expect(searchInput.value).toBe('test')
+ })
+ })
+
+ describe('Integration Scenarios', () => {
+ it('should handle complex workflow: search, select, create, remove', async () => {
+ const onChangeMock = vi.fn()
+ createTestComponent(null, onChangeMock)
+
+ await waitFor(() => {
+ expect(screen.getByText('Rock Classics')).toBeInTheDocument()
+ })
+
+ // Search and select existing playlist
+ const searchInput = screen.getByRole('textbox')
+ fireEvent.change(searchInput, { target: { value: 'rock' } })
+
+ const rockPlaylist = screen.getByText('Rock Classics')
+ fireEvent.click(rockPlaylist)
+
+ // Clear search and create new playlist
+ fireEvent.change(searchInput, { target: { value: 'My Custom Mix' } })
+ fireEvent.keyDown(searchInput, { key: 'Enter' })
+
+ await waitFor(() => {
+ expect(onChangeMock).toHaveBeenCalledWith([
+ { id: 'playlist-1', name: 'Rock Classics', ownerId: 'admin' },
+ { name: 'My Custom Mix' },
+ ])
+ })
+
+ // Remove the first selected playlist via chip
+ const removeButtons = screen.getAllByText('×')
+ fireEvent.click(removeButtons[0])
+
+ await waitFor(() => {
+ expect(onChangeMock).toHaveBeenCalledWith([{ name: 'My Custom Mix' }])
+ })
})
})
})
diff --git a/ui/src/dialogs/aboutUtils.js b/ui/src/dialogs/aboutUtils.js
new file mode 100644
index 000000000..7c92692de
--- /dev/null
+++ b/ui/src/dialogs/aboutUtils.js
@@ -0,0 +1,278 @@
+/**
+ * TOML utility functions for configuration export
+ */
+
+/**
+ * Flattens nested configuration object and generates environment variable names
+ * @param {Object} config - The nested configuration object from the backend
+ * @param {string} prefix - The current prefix for nested keys
+ * @returns {Array} - Array of config objects with key, envVar, and value properties
+ */
+export const flattenConfig = (config, prefix = '') => {
+ const result = []
+
+ if (!config || typeof config !== 'object') {
+ return result
+ }
+
+ Object.keys(config).forEach((key) => {
+ const value = config[key]
+ const currentKey = prefix ? `${prefix}.${key}` : key
+
+ if (value && typeof value === 'object' && !Array.isArray(value)) {
+ // Recursively flatten nested objects
+ result.push(...flattenConfig(value, currentKey))
+ } else {
+ // Generate environment variable name: ND_ + uppercase with dots replaced by underscores
+ const envVar = 'ND_' + currentKey.toUpperCase().replace(/\./g, '_')
+
+ // Convert value to string for display
+ let displayValue = value
+ if (
+ Array.isArray(value) ||
+ (typeof value === 'object' && value !== null)
+ ) {
+ displayValue = JSON.stringify(value)
+ } else {
+ displayValue = String(value)
+ }
+
+ result.push({
+ key: currentKey,
+ envVar: envVar,
+ value: displayValue,
+ })
+ }
+ })
+
+ return result
+}
+
+/**
+ * Separates and sorts configuration entries into regular and dev configs
+ * @param {Array|Object} configEntries - Array of config objects with key and value, or nested config object
+ * @returns {Object} - Object with regularConfigs and devConfigs arrays, both sorted
+ */
+export const separateAndSortConfigs = (configEntries) => {
+ const regularConfigs = []
+ const devConfigs = []
+
+ // Handle both the old array format and new nested object format
+ let flattenedConfigs
+ if (Array.isArray(configEntries)) {
+ // Old format - already flattened
+ flattenedConfigs = configEntries
+ } else {
+ // New format - need to flatten
+ flattenedConfigs = flattenConfig(configEntries)
+ }
+
+ flattenedConfigs?.forEach((config) => {
+ // Skip configFile as it's displayed separately
+ if (config.key === 'ConfigFile') {
+ return
+ }
+
+ if (config.key.startsWith('Dev')) {
+ devConfigs.push(config)
+ } else {
+ regularConfigs.push(config)
+ }
+ })
+
+ // Sort configurations alphabetically
+ regularConfigs.sort((a, b) => a.key.localeCompare(b.key))
+ devConfigs.sort((a, b) => a.key.localeCompare(b.key))
+
+ return { regularConfigs, devConfigs }
+}
+
+/**
+ * Escapes TOML keys that contain special characters
+ * @param {string} key - The key to potentially escape
+ * @returns {string} - The escaped key if needed, or the original key
+ */
+export const escapeTomlKey = (key) => {
+ // Convert to string first to handle null/undefined
+ const keyStr = String(key)
+
+ // Empty strings always need quotes
+ if (keyStr === '') {
+ return '""'
+ }
+
+ // TOML bare keys can only contain letters, numbers, underscores, and hyphens
+ // If the key contains other characters, it needs to be quoted
+ if (/^[a-zA-Z0-9_-]+$/.test(keyStr)) {
+ return keyStr
+ }
+
+ // Escape quotes in the key and wrap in quotes
+ return `"${keyStr.replace(/\\/g, '\\\\').replace(/"/g, '\\"')}"`
+}
+
+/**
+ * Converts a value to proper TOML format
+ * @param {*} value - The value to format
+ * @returns {string} - The TOML-formatted value
+ */
+export const formatTomlValue = (value) => {
+ if (value === null || value === undefined) {
+ return '""'
+ }
+
+ const str = String(value)
+
+ // Boolean values
+ if (str === 'true' || str === 'false') {
+ return str
+ }
+
+ // Numbers (integers and floats)
+ if (/^-?\d+$/.test(str)) {
+ return str // Integer
+ }
+ if (/^-?\d*\.\d+$/.test(str)) {
+ return str // Float
+ }
+
+ // Duration values (like "300ms", "1s", "5m")
+ if (/^\d+(\.\d+)?(ns|us|µs|ms|s|m|h)$/.test(str)) {
+ return `"${str}"`
+ }
+
+ // Handle arrays and objects
+ if (str.startsWith('[') || str.startsWith('{')) {
+ try {
+ const parsed = JSON.parse(str)
+
+ // If it's an array, format as TOML array
+ if (Array.isArray(parsed)) {
+ const formattedItems = parsed.map((item) => {
+ if (typeof item === 'string') {
+ return `"${item.replace(/\\/g, '\\\\').replace(/"/g, '\\"')}"`
+ } else if (typeof item === 'number' || typeof item === 'boolean') {
+ return String(item)
+ } else {
+ return `"${String(item).replace(/\\/g, '\\\\').replace(/"/g, '\\"')}"`
+ }
+ })
+
+ if (formattedItems.length === 0) {
+ return '[ ]'
+ }
+ return `[ ${formattedItems.join(', ')} ]`
+ }
+
+ // For objects, keep the JSON string format with triple quotes
+ return `"""${str}"""`
+ } catch {
+ return `"${str.replace(/\\/g, '\\\\').replace(/"/g, '\\"')}"`
+ }
+ }
+
+ // String values (escape backslashes and quotes)
+ return `"${str.replace(/\\/g, '\\\\').replace(/"/g, '\\"')}"`
+}
+
+/**
+ * Converts nested keys to TOML sections
+ * @param {Array} configs - Array of config objects with key and value
+ * @returns {Object} - Object with sections and rootKeys
+ */
+export const buildTomlSections = (configs) => {
+ const sections = {}
+ const rootKeys = []
+
+ configs.forEach(({ key, value }) => {
+ if (key.includes('.')) {
+ const parts = key.split('.')
+ const sectionName = parts[0]
+ const keyName = parts.slice(1).join('.')
+
+ if (!sections[sectionName]) {
+ sections[sectionName] = []
+ }
+ sections[sectionName].push({ key: keyName, value })
+ } else {
+ rootKeys.push({ key, value })
+ }
+ })
+
+ return { sections, rootKeys }
+}
+
+/**
+ * Converts configuration data to TOML format
+ * @param {Object} configData - The configuration data object
+ * @param {Function} translate - Translation function for internationalization
+ * @returns {string} - The TOML-formatted configuration
+ */
+export const configToToml = (configData, translate = (key) => key) => {
+ let tomlContent = `# Navidrome Configuration\n# Generated on ${new Date().toISOString()}\n\n`
+
+ // Handle both old array format (configData.config is array) and new nested format (configData.config is object)
+ let configs
+ if (Array.isArray(configData.config)) {
+ // Old format - already flattened
+ configs = configData.config
+ } else {
+ // New format - need to flatten
+ configs = flattenConfig(configData.config)
+ }
+
+ const { regularConfigs, devConfigs } = separateAndSortConfigs(configs)
+
+ // Process regular configs
+ const { sections: regularSections, rootKeys: regularRootKeys } =
+ buildTomlSections(regularConfigs)
+
+ // Add root-level keys first
+ if (regularRootKeys.length > 0) {
+ regularRootKeys.forEach(({ key, value }) => {
+ tomlContent += `${key} = ${formatTomlValue(value)}\n`
+ })
+ tomlContent += '\n'
+ }
+
+ // Add dev configs if any
+ if (devConfigs.length > 0) {
+ tomlContent += `# ${translate('about.config.devFlagsHeader')}\n`
+ tomlContent += `# ${translate('about.config.devFlagsComment')}\n\n`
+
+ const { sections: devSections, rootKeys: devRootKeys } =
+ buildTomlSections(devConfigs)
+
+ // Add dev root-level keys
+ devRootKeys.forEach(({ key, value }) => {
+ tomlContent += `${key} = ${formatTomlValue(value)}\n`
+ })
+ if (devRootKeys.length > 0) {
+ tomlContent += '\n'
+ }
+
+ // Add dev sections
+ Object.keys(devSections)
+ .sort()
+ .forEach((sectionName) => {
+ tomlContent += `[${sectionName}]\n`
+ devSections[sectionName].forEach(({ key, value }) => {
+ tomlContent += `${escapeTomlKey(key)} = ${formatTomlValue(value)}\n`
+ })
+ tomlContent += '\n'
+ })
+ }
+
+ // Add sections
+ Object.keys(regularSections)
+ .sort()
+ .forEach((sectionName) => {
+ tomlContent += `[${sectionName}]\n`
+ regularSections[sectionName].forEach(({ key, value }) => {
+ tomlContent += `${escapeTomlKey(key)} = ${formatTomlValue(value)}\n`
+ })
+ tomlContent += '\n'
+ })
+
+ return tomlContent
+}
diff --git a/ui/src/dialogs/aboutUtils.test.js b/ui/src/dialogs/aboutUtils.test.js
new file mode 100644
index 000000000..4632c5aed
--- /dev/null
+++ b/ui/src/dialogs/aboutUtils.test.js
@@ -0,0 +1,737 @@
+import { describe, it, expect } from 'vitest'
+import {
+ formatTomlValue,
+ buildTomlSections,
+ configToToml,
+ separateAndSortConfigs,
+ flattenConfig,
+ escapeTomlKey,
+} from './aboutUtils'
+
+describe('formatTomlValue', () => {
+ it('handles null and undefined values', () => {
+ expect(formatTomlValue(null)).toBe('""')
+ expect(formatTomlValue(undefined)).toBe('""')
+ })
+
+ it('handles boolean values', () => {
+ expect(formatTomlValue('true')).toBe('true')
+ expect(formatTomlValue('false')).toBe('false')
+ expect(formatTomlValue(true)).toBe('true')
+ expect(formatTomlValue(false)).toBe('false')
+ })
+
+ it('handles integer values', () => {
+ expect(formatTomlValue('123')).toBe('123')
+ expect(formatTomlValue('-456')).toBe('-456')
+ expect(formatTomlValue('0')).toBe('0')
+ expect(formatTomlValue(789)).toBe('789')
+ })
+
+ it('handles float values', () => {
+ expect(formatTomlValue('123.45')).toBe('123.45')
+ expect(formatTomlValue('-67.89')).toBe('-67.89')
+ expect(formatTomlValue('0.0')).toBe('0.0')
+ expect(formatTomlValue(12.34)).toBe('12.34')
+ })
+
+ it('handles duration values', () => {
+ expect(formatTomlValue('300ms')).toBe('"300ms"')
+ expect(formatTomlValue('5s')).toBe('"5s"')
+ expect(formatTomlValue('10m')).toBe('"10m"')
+ expect(formatTomlValue('2h')).toBe('"2h"')
+ expect(formatTomlValue('1.5s')).toBe('"1.5s"')
+ })
+
+ it('handles JSON arrays and objects', () => {
+ expect(formatTomlValue('["item1", "item2"]')).toBe('[ "item1", "item2" ]')
+ expect(formatTomlValue('{"key": "value"}')).toBe('"""{"key": "value"}"""')
+ })
+
+ it('formats different types of arrays correctly', () => {
+ // String array
+ expect(formatTomlValue('["genre", "tcon", "©gen"]')).toBe(
+ '[ "genre", "tcon", "©gen" ]',
+ )
+ // Mixed array with numbers and strings
+ expect(formatTomlValue('[42, "test", true]')).toBe('[ 42, "test", true ]')
+ // Empty array
+ expect(formatTomlValue('[]')).toBe('[ ]')
+ // Array with special characters in strings
+ expect(
+ formatTomlValue('["item with spaces", "item\\"with\\"quotes"]'),
+ ).toBe('[ "item with spaces", "item\\"with\\"quotes" ]')
+ })
+
+ it('handles invalid JSON as regular strings', () => {
+ expect(formatTomlValue('[invalid json')).toBe('"[invalid json"')
+ expect(formatTomlValue('{broken')).toBe('"{broken"')
+ })
+
+ it('handles regular strings with quote escaping', () => {
+ expect(formatTomlValue('simple string')).toBe('"simple string"')
+ expect(formatTomlValue('string with "quotes"')).toBe(
+ '"string with \\"quotes\\""',
+ )
+ expect(formatTomlValue('/path/to/file')).toBe('"/path/to/file"')
+ })
+
+ it('handles strings with backslashes and quotes', () => {
+ expect(formatTomlValue('C:\\Program Files\\app')).toBe(
+ '"C:\\\\Program Files\\\\app"',
+ )
+ expect(formatTomlValue('path\\to"file')).toBe('"path\\\\to\\"file"')
+ expect(formatTomlValue('backslash\\ and "quote"')).toBe(
+ '"backslash\\\\ and \\"quote\\""',
+ )
+ expect(formatTomlValue('single\\backslash')).toBe('"single\\\\backslash"')
+ })
+
+ it('handles empty strings', () => {
+ expect(formatTomlValue('')).toBe('""')
+ })
+})
+
+describe('buildTomlSections', () => {
+ it('separates root keys from nested keys', () => {
+ const configs = [
+ { key: 'RootKey1', value: 'value1' },
+ { key: 'Section.NestedKey', value: 'value2' },
+ { key: 'RootKey2', value: 'value3' },
+ { key: 'Section.AnotherKey', value: 'value4' },
+ { key: 'AnotherSection.Key', value: 'value5' },
+ ]
+
+ const result = buildTomlSections(configs)
+
+ expect(result.rootKeys).toEqual([
+ { key: 'RootKey1', value: 'value1' },
+ { key: 'RootKey2', value: 'value3' },
+ ])
+
+ expect(result.sections).toEqual({
+ Section: [
+ { key: 'NestedKey', value: 'value2' },
+ { key: 'AnotherKey', value: 'value4' },
+ ],
+ AnotherSection: [{ key: 'Key', value: 'value5' }],
+ })
+ })
+
+ it('handles deeply nested keys', () => {
+ const configs = [{ key: 'Section.SubSection.DeepKey', value: 'deepValue' }]
+
+ const result = buildTomlSections(configs)
+
+ expect(result.rootKeys).toEqual([])
+ expect(result.sections).toEqual({
+ Section: [{ key: 'SubSection.DeepKey', value: 'deepValue' }],
+ })
+ })
+
+ it('handles empty input', () => {
+ const result = buildTomlSections([])
+
+ expect(result.rootKeys).toEqual([])
+ expect(result.sections).toEqual({})
+ })
+})
+
+describe('configToToml', () => {
+ const mockTranslate = (key) => {
+ const translations = {
+ 'about.config.devFlagsHeader':
+ 'Development Flags (subject to change/removal)',
+ 'about.config.devFlagsComment':
+ 'These are experimental settings and may be removed in future versions',
+ }
+ return translations[key] || key
+ }
+
+ it('generates TOML with header and timestamp', () => {
+ const configData = {
+ config: [{ key: 'TestKey', value: 'testValue' }],
+ }
+
+ const result = configToToml(configData, mockTranslate)
+
+ expect(result).toContain('# Navidrome Configuration')
+ expect(result).toContain('# Generated on')
+ expect(result).toContain('TestKey = "testValue"')
+ })
+
+ it('separates and sorts regular and dev configs', () => {
+ const configData = {
+ config: [
+ { key: 'ZRegularKey', value: 'regularValue' },
+ { key: 'DevTestFlag', value: 'true' },
+ { key: 'ARegularKey', value: 'anotherValue' },
+ { key: 'DevAnotherFlag', value: 'false' },
+ ],
+ }
+
+ const result = configToToml(configData, mockTranslate)
+
+ // Check that regular configs come first and are sorted
+ const lines = result.split('\n')
+ const aRegularIndex = lines.findIndex((line) =>
+ line.includes('ARegularKey'),
+ )
+ const zRegularIndex = lines.findIndex((line) =>
+ line.includes('ZRegularKey'),
+ )
+ const devHeaderIndex = lines.findIndex((line) =>
+ line.includes('Development Flags'),
+ )
+ const devAnotherIndex = lines.findIndex((line) =>
+ line.includes('DevAnotherFlag'),
+ )
+ const devTestIndex = lines.findIndex((line) => line.includes('DevTestFlag'))
+
+ expect(aRegularIndex).toBeLessThan(zRegularIndex)
+ expect(zRegularIndex).toBeLessThan(devHeaderIndex)
+ expect(devHeaderIndex).toBeLessThan(devAnotherIndex)
+ expect(devAnotherIndex).toBeLessThan(devTestIndex)
+ })
+
+ it('skips ConfigFile entries', () => {
+ const configData = {
+ config: [
+ { key: 'ConfigFile', value: '/path/to/config.toml' },
+ { key: 'TestKey', value: 'testValue' },
+ ],
+ }
+
+ const result = configToToml(configData, mockTranslate)
+
+ expect(result).not.toContain('ConfigFile =')
+ expect(result).toContain('TestKey = "testValue"')
+ })
+
+ it('handles sections correctly', () => {
+ const configData = {
+ config: [
+ { key: 'RootKey', value: 'rootValue' },
+ { key: 'Section.NestedKey', value: 'nestedValue' },
+ { key: 'Section.AnotherKey', value: 'anotherValue' },
+ { key: 'DevA', value: 'DevValue' },
+ ],
+ }
+
+ const result = configToToml(configData, mockTranslate)
+ // Fields in a section are sorted alphabetically
+ const fields = [
+ 'RootKey = "rootValue"',
+ 'DevA = "DevValue"',
+ '[Section]',
+ 'AnotherKey = "anotherValue"',
+ 'NestedKey = "nestedValue"',
+ ]
+
+ for (let idx = 0; idx < fields.length - 1; idx++) {
+ expect(result).toContain(fields[idx])
+
+ const idxA = result.indexOf(fields[idx])
+ const idxB = result.indexOf(fields[idx + 1])
+
+ expect(idxA).toBeLessThan(idxB)
+ }
+
+ expect(result).toContain(fields[fields.length - 1])
+ })
+
+ it('includes dev flags header when dev configs exist', () => {
+ const configData = {
+ config: [
+ { key: 'RegularKey', value: 'regularValue' },
+ { key: 'DevTestFlag', value: 'true' },
+ ],
+ }
+
+ const result = configToToml(configData, mockTranslate)
+
+ expect(result).toContain('# Development Flags (subject to change/removal)')
+ expect(result).toContain(
+ '# These are experimental settings and may be removed in future versions',
+ )
+ expect(result).toContain('DevTestFlag = true')
+ })
+
+ it('does not include dev flags header when no dev configs exist', () => {
+ const configData = {
+ config: [{ key: 'RegularKey', value: 'regularValue' }],
+ }
+
+ const result = configToToml(configData, mockTranslate)
+
+ expect(result).not.toContain('Development Flags')
+ expect(result).toContain('RegularKey = "regularValue"')
+ })
+
+ it('handles empty config data', () => {
+ const configData = { config: [] }
+
+ const result = configToToml(configData, mockTranslate)
+
+ expect(result).toContain('# Navidrome Configuration')
+ expect(result).not.toContain('Development Flags')
+ })
+
+ it('handles missing config array', () => {
+ const configData = {}
+
+ const result = configToToml(configData, mockTranslate)
+
+ expect(result).toContain('# Navidrome Configuration')
+ expect(result).not.toContain('Development Flags')
+ })
+
+ it('works without translate function', () => {
+ const configData = {
+ config: [{ key: 'DevTestFlag', value: 'true' }],
+ }
+
+ const result = configToToml(configData)
+
+ expect(result).toContain('# about.config.devFlagsHeader')
+ expect(result).toContain('# about.config.devFlagsComment')
+ expect(result).toContain('DevTestFlag = true')
+ })
+
+ it('handles various data types correctly', () => {
+ const configData = {
+ config: [
+ { key: 'StringValue', value: 'test string' },
+ { key: 'BooleanValue', value: 'true' },
+ { key: 'IntegerValue', value: '42' },
+ { key: 'FloatValue', value: '3.14' },
+ { key: 'DurationValue', value: '5s' },
+ { key: 'ArrayValue', value: '["item1", "item2"]' },
+ ],
+ }
+
+ const result = configToToml(configData, mockTranslate)
+
+ expect(result).toContain('StringValue = "test string"')
+ expect(result).toContain('BooleanValue = true')
+ expect(result).toContain('IntegerValue = 42')
+ expect(result).toContain('FloatValue = 3.14')
+ expect(result).toContain('DurationValue = "5s"')
+ expect(result).toContain('ArrayValue = [ "item1", "item2" ]')
+ })
+
+ it('handles nested config object format correctly', () => {
+ const configData = {
+ config: {
+ Address: '127.0.0.1',
+ Port: 4533,
+ EnableDownloads: true,
+ DevLogSourceLine: false,
+ LastFM: {
+ Enabled: true,
+ ApiKey: 'secret123',
+ Language: 'en',
+ },
+ Scanner: {
+ Schedule: 'daily',
+ Enabled: true,
+ },
+ },
+ }
+
+ const result = configToToml(configData, mockTranslate)
+
+ // Should contain regular configs
+ expect(result).toContain('Address = "127.0.0.1"')
+ expect(result).toContain('Port = 4533')
+ expect(result).toContain('EnableDownloads = true')
+
+ // Should contain dev configs with header
+ expect(result).toContain('# Development Flags (subject to change/removal)')
+ expect(result).toContain('DevLogSourceLine = false')
+
+ // Should contain sections
+ expect(result).toContain('[LastFM]')
+ expect(result).toContain('Enabled = true')
+ expect(result).toContain('ApiKey = "secret123"')
+ expect(result).toContain('Language = "en"')
+
+ expect(result).toContain('[Scanner]')
+ expect(result).toContain('Schedule = "daily"')
+ })
+
+ it('handles mixed nested and flat structure', () => {
+ const configData = {
+ config: {
+ MusicFolder: '/music',
+ DevAutoLoginUsername: 'testuser',
+ Jukebox: {
+ Enabled: false,
+ AdminOnly: true,
+ },
+ },
+ }
+
+ const result = configToToml(configData, mockTranslate)
+
+ expect(result).toContain('MusicFolder = "/music"')
+ expect(result).toContain('DevAutoLoginUsername = "testuser"')
+ expect(result).toContain('[Jukebox]')
+ expect(result).toContain('Enabled = false')
+ expect(result).toContain('AdminOnly = true')
+ })
+
+ it('properly escapes keys with special characters in sections', () => {
+ const configData = {
+ config: [
+ { key: 'DevLogLevels.persistence/sql_base_repository', value: 'trace' },
+ { key: 'DevLogLevels.core/scanner', value: 'debug' },
+ { key: 'DevLogLevels.regular_key', value: 'info' },
+ { key: 'Tags.genre.Aliases', value: '["tcon","genre","©gen"]' },
+ ],
+ }
+
+ const result = configToToml(configData, mockTranslate)
+
+ // Keys with forward slashes should be quoted
+ expect(result).toContain('"persistence/sql_base_repository" = "trace"')
+ expect(result).toContain('"core/scanner" = "debug"')
+
+ // Regular keys should not be quoted
+ expect(result).toContain('regular_key = "info"')
+
+ // Arrays should be formatted correctly
+ expect(result).toContain('"genre.Aliases" = [ "tcon", "genre", "©gen" ]')
+
+ // Should contain proper sections
+ expect(result).toContain('[DevLogLevels]')
+ expect(result).toContain('[Tags]')
+ })
+})
+
+describe('flattenConfig', () => {
+ it('flattens simple nested objects correctly', () => {
+ const config = {
+ Address: '0.0.0.0',
+ Port: 4533,
+ EnableDownloads: true,
+ LastFM: {
+ Enabled: true,
+ ApiKey: 'secret123',
+ Language: 'en',
+ },
+ }
+
+ const result = flattenConfig(config)
+
+ expect(result).toContainEqual({
+ key: 'Address',
+ envVar: 'ND_ADDRESS',
+ value: '0.0.0.0',
+ })
+
+ expect(result).toContainEqual({
+ key: 'Port',
+ envVar: 'ND_PORT',
+ value: '4533',
+ })
+
+ expect(result).toContainEqual({
+ key: 'EnableDownloads',
+ envVar: 'ND_ENABLEDOWNLOADS',
+ value: 'true',
+ })
+
+ expect(result).toContainEqual({
+ key: 'LastFM.Enabled',
+ envVar: 'ND_LASTFM_ENABLED',
+ value: 'true',
+ })
+
+ expect(result).toContainEqual({
+ key: 'LastFM.ApiKey',
+ envVar: 'ND_LASTFM_APIKEY',
+ value: 'secret123',
+ })
+
+ expect(result).toContainEqual({
+ key: 'LastFM.Language',
+ envVar: 'ND_LASTFM_LANGUAGE',
+ value: 'en',
+ })
+ })
+
+ it('handles deeply nested objects', () => {
+ const config = {
+ Scanner: {
+ Schedule: 'daily',
+ Options: {
+ ExtractorType: 'taglib',
+ ArtworkPriority: 'cover.jpg',
+ },
+ },
+ }
+
+ const result = flattenConfig(config)
+
+ expect(result).toContainEqual({
+ key: 'Scanner.Schedule',
+ envVar: 'ND_SCANNER_SCHEDULE',
+ value: 'daily',
+ })
+
+ expect(result).toContainEqual({
+ key: 'Scanner.Options.ExtractorType',
+ envVar: 'ND_SCANNER_OPTIONS_EXTRACTORTYPE',
+ value: 'taglib',
+ })
+
+ expect(result).toContainEqual({
+ key: 'Scanner.Options.ArtworkPriority',
+ envVar: 'ND_SCANNER_OPTIONS_ARTWORKPRIORITY',
+ value: 'cover.jpg',
+ })
+ })
+
+ it('handles arrays correctly', () => {
+ const config = {
+ DeviceList: ['device1', 'device2'],
+ Settings: {
+ EnabledFormats: ['mp3', 'flac', 'ogg'],
+ },
+ }
+
+ const result = flattenConfig(config)
+
+ expect(result).toContainEqual({
+ key: 'DeviceList',
+ envVar: 'ND_DEVICELIST',
+ value: '["device1","device2"]',
+ })
+
+ expect(result).toContainEqual({
+ key: 'Settings.EnabledFormats',
+ envVar: 'ND_SETTINGS_ENABLEDFORMATS',
+ value: '["mp3","flac","ogg"]',
+ })
+ })
+
+ it('handles null and undefined values', () => {
+ const config = {
+ NullValue: null,
+ UndefinedValue: undefined,
+ EmptyString: '',
+ ZeroValue: 0,
+ }
+
+ const result = flattenConfig(config)
+
+ expect(result).toContainEqual({
+ key: 'NullValue',
+ envVar: 'ND_NULLVALUE',
+ value: 'null',
+ })
+
+ expect(result).toContainEqual({
+ key: 'UndefinedValue',
+ envVar: 'ND_UNDEFINEDVALUE',
+ value: 'undefined',
+ })
+
+ expect(result).toContainEqual({
+ key: 'EmptyString',
+ envVar: 'ND_EMPTYSTRING',
+ value: '',
+ })
+
+ expect(result).toContainEqual({
+ key: 'ZeroValue',
+ envVar: 'ND_ZEROVALUE',
+ value: '0',
+ })
+ })
+
+ it('handles empty object', () => {
+ const result = flattenConfig({})
+ expect(result).toEqual([])
+ })
+
+ it('handles null/undefined input', () => {
+ expect(flattenConfig(null)).toEqual([])
+ expect(flattenConfig(undefined)).toEqual([])
+ })
+
+ it('handles non-object input', () => {
+ expect(flattenConfig('string')).toEqual([])
+ expect(flattenConfig(123)).toEqual([])
+ expect(flattenConfig(true)).toEqual([])
+ })
+})
+
+describe('separateAndSortConfigs', () => {
+ it('separates regular and dev configs correctly with array input', () => {
+ const configs = [
+ { key: 'RegularKey1', value: 'value1' },
+ { key: 'DevTestFlag', value: 'true' },
+ { key: 'AnotherRegular', value: 'value2' },
+ { key: 'DevAnotherFlag', value: 'false' },
+ ]
+
+ const result = separateAndSortConfigs(configs)
+
+ expect(result.regularConfigs).toEqual([
+ { key: 'AnotherRegular', value: 'value2' },
+ { key: 'RegularKey1', value: 'value1' },
+ ])
+
+ expect(result.devConfigs).toEqual([
+ { key: 'DevAnotherFlag', value: 'false' },
+ { key: 'DevTestFlag', value: 'true' },
+ ])
+ })
+
+ it('separates regular and dev configs correctly with nested object input', () => {
+ const config = {
+ Address: '127.0.0.1',
+ Port: 4533,
+ DevAutoLoginUsername: 'testuser',
+ DevLogSourceLine: true,
+ LastFM: {
+ Enabled: true,
+ ApiKey: 'secret123',
+ },
+ }
+
+ const result = separateAndSortConfigs(config)
+
+ expect(result.regularConfigs).toEqual([
+ { key: 'Address', envVar: 'ND_ADDRESS', value: '127.0.0.1' },
+ { key: 'LastFM.ApiKey', envVar: 'ND_LASTFM_APIKEY', value: 'secret123' },
+ { key: 'LastFM.Enabled', envVar: 'ND_LASTFM_ENABLED', value: 'true' },
+ { key: 'Port', envVar: 'ND_PORT', value: '4533' },
+ ])
+
+ expect(result.devConfigs).toEqual([
+ {
+ key: 'DevAutoLoginUsername',
+ envVar: 'ND_DEVAUTOLOGINUSERNAME',
+ value: 'testuser',
+ },
+ { key: 'DevLogSourceLine', envVar: 'ND_DEVLOGSOURCELINE', value: 'true' },
+ ])
+ })
+
+ it('skips ConfigFile entries', () => {
+ const configs = [
+ { key: 'ConfigFile', value: '/path/to/config.toml' },
+ { key: 'RegularKey', value: 'value' },
+ { key: 'DevFlag', value: 'true' },
+ ]
+
+ const result = separateAndSortConfigs(configs)
+
+ expect(result.regularConfigs).toEqual([
+ { key: 'RegularKey', value: 'value' },
+ ])
+ expect(result.devConfigs).toEqual([{ key: 'DevFlag', value: 'true' }])
+ })
+
+ it('skips ConfigFile entries with nested object input', () => {
+ const config = {
+ ConfigFile: '/path/to/config.toml',
+ RegularKey: 'value',
+ DevFlag: true,
+ }
+
+ const result = separateAndSortConfigs(config)
+
+ expect(result.regularConfigs).toEqual([
+ { key: 'RegularKey', envVar: 'ND_REGULARKEY', value: 'value' },
+ ])
+ expect(result.devConfigs).toEqual([
+ { key: 'DevFlag', envVar: 'ND_DEVFLAG', value: 'true' },
+ ])
+ })
+
+ it('handles empty input', () => {
+ const result = separateAndSortConfigs([])
+
+ expect(result.regularConfigs).toEqual([])
+ expect(result.devConfigs).toEqual([])
+ })
+
+ it('handles null/undefined input', () => {
+ const result1 = separateAndSortConfigs(null)
+ const result2 = separateAndSortConfigs(undefined)
+
+ expect(result1.regularConfigs).toEqual([])
+ expect(result1.devConfigs).toEqual([])
+ expect(result2.regularConfigs).toEqual([])
+ expect(result2.devConfigs).toEqual([])
+ })
+
+ it('sorts configs alphabetically', () => {
+ const configs = [
+ { key: 'ZRegular', value: 'z' },
+ { key: 'ARegular', value: 'a' },
+ { key: 'DevZ', value: 'z' },
+ { key: 'DevA', value: 'a' },
+ ]
+
+ const result = separateAndSortConfigs(configs)
+
+ expect(result.regularConfigs[0].key).toBe('ARegular')
+ expect(result.regularConfigs[1].key).toBe('ZRegular')
+ expect(result.devConfigs[0].key).toBe('DevA')
+ expect(result.devConfigs[1].key).toBe('DevZ')
+ })
+})
+
+describe('escapeTomlKey', () => {
+ it('does not escape valid bare keys', () => {
+ expect(escapeTomlKey('RegularKey')).toBe('RegularKey')
+ expect(escapeTomlKey('regular_key')).toBe('regular_key')
+ expect(escapeTomlKey('regular-key')).toBe('regular-key')
+ expect(escapeTomlKey('key123')).toBe('key123')
+ expect(escapeTomlKey('Key_with_underscores')).toBe('Key_with_underscores')
+ expect(escapeTomlKey('Key-with-hyphens')).toBe('Key-with-hyphens')
+ })
+
+ it('escapes keys with special characters', () => {
+ // Keys with forward slashes (like DevLogLevels keys)
+ expect(escapeTomlKey('persistence/sql_base_repository')).toBe(
+ '"persistence/sql_base_repository"',
+ )
+ expect(escapeTomlKey('core/scanner')).toBe('"core/scanner"')
+
+ // Keys with dots
+ expect(escapeTomlKey('Section.NestedKey')).toBe('"Section.NestedKey"')
+
+ // Keys with spaces
+ expect(escapeTomlKey('key with spaces')).toBe('"key with spaces"')
+
+ // Keys with other special characters
+ expect(escapeTomlKey('key@with@symbols')).toBe('"key@with@symbols"')
+ expect(escapeTomlKey('key+with+plus')).toBe('"key+with+plus"')
+ })
+
+ it('escapes quotes in keys', () => {
+ expect(escapeTomlKey('key"with"quotes')).toBe('"key\\"with\\"quotes"')
+ expect(escapeTomlKey('key with "quotes" inside')).toBe(
+ '"key with \\"quotes\\" inside"',
+ )
+ })
+
+ it('escapes backslashes in keys', () => {
+ expect(escapeTomlKey('key\\with\\backslashes')).toBe(
+ '"key\\\\with\\\\backslashes"',
+ )
+ expect(escapeTomlKey('path\\to\\file')).toBe('"path\\\\to\\\\file"')
+ })
+
+ it('handles empty and null keys', () => {
+ expect(escapeTomlKey('')).toBe('""')
+ expect(escapeTomlKey(null)).toBe('null')
+ expect(escapeTomlKey(undefined)).toBe('undefined')
+ })
+})
diff --git a/ui/src/eventStream.js b/ui/src/eventStream.js
index 34463e19c..c91dae875 100644
--- a/ui/src/eventStream.js
+++ b/ui/src/eventStream.js
@@ -1,7 +1,8 @@
import { baseUrl } from './utils'
import throttle from 'lodash.throttle'
-import { processEvent, serverDown } from './actions'
+import { processEvent, serverDown, streamReconnected } from './actions'
import { REST_URL } from './consts'
+import config from './config'
const newEventStream = async () => {
let url = baseUrl(`${REST_URL}/events`)
@@ -11,6 +12,51 @@ const newEventStream = async () => {
return new EventSource(url)
}
+let eventStream
+let reconnectTimer
+const RECONNECT_DELAY = 5000
+
+const setupHandlers = (stream, dispatchFn) => {
+ stream.addEventListener('serverStart', eventHandler(dispatchFn))
+ stream.addEventListener('scanStatus', throttledEventHandler(dispatchFn))
+ stream.addEventListener('refreshResource', eventHandler(dispatchFn))
+ if (config.enableNowPlaying) {
+ stream.addEventListener('nowPlayingCount', eventHandler(dispatchFn))
+ }
+ stream.addEventListener('keepAlive', eventHandler(dispatchFn))
+ stream.onerror = (e) => {
+ // eslint-disable-next-line no-console
+ console.log('EventStream error', e)
+ dispatchFn(serverDown())
+ if (stream) stream.close()
+ scheduleReconnect(dispatchFn)
+ }
+}
+
+const scheduleReconnect = (dispatchFn) => {
+ if (!reconnectTimer) {
+ reconnectTimer = setTimeout(() => {
+ reconnectTimer = null
+ connect(dispatchFn)
+ }, RECONNECT_DELAY)
+ }
+}
+
+const connect = async (dispatchFn) => {
+ try {
+ const stream = await newEventStream()
+ eventStream = stream
+ setupHandlers(stream, dispatchFn)
+ // Dispatch reconnection event to refresh critical data
+ dispatchFn(streamReconnected())
+ return stream
+ } catch (e) {
+ // eslint-disable-next-line no-console
+ console.log(`Error connecting to server:`, e)
+ scheduleReconnect(dispatchFn)
+ }
+}
+
const eventHandler = (dispatchFn) => (event) => {
const data = JSON.parse(event.data)
if (event.type !== 'keepAlive') {
@@ -21,10 +67,7 @@ const eventHandler = (dispatchFn) => (event) => {
const throttledEventHandler = (dispatchFn) =>
throttle(eventHandler(dispatchFn), 100, { trailing: true })
-const startEventStream = async (dispatchFn) => {
- if (!localStorage.getItem('is-authenticated')) {
- return Promise.resolve()
- }
+const startEventStreamLegacy = async (dispatchFn) => {
return newEventStream()
.then((newStream) => {
newStream.addEventListener('serverStart', eventHandler(dispatchFn))
@@ -33,6 +76,9 @@ const startEventStream = async (dispatchFn) => {
throttledEventHandler(dispatchFn),
)
newStream.addEventListener('refreshResource', eventHandler(dispatchFn))
+ if (config.enableNowPlaying) {
+ newStream.addEventListener('nowPlayingCount', eventHandler(dispatchFn))
+ }
newStream.addEventListener('keepAlive', eventHandler(dispatchFn))
newStream.onerror = (e) => {
// eslint-disable-next-line no-console
@@ -47,4 +93,22 @@ const startEventStream = async (dispatchFn) => {
})
}
+const startEventStreamNew = async (dispatchFn) => {
+ if (eventStream) {
+ eventStream.close()
+ eventStream = null
+ }
+ return connect(dispatchFn)
+}
+
+const startEventStream = async (dispatchFn) => {
+ if (!localStorage.getItem('is-authenticated')) {
+ return Promise.resolve()
+ }
+ if (config.devNewEventStream) {
+ return startEventStreamNew(dispatchFn)
+ }
+ return startEventStreamLegacy(dispatchFn)
+}
+
export { startEventStream }
diff --git a/ui/src/eventStream.test.js b/ui/src/eventStream.test.js
new file mode 100644
index 000000000..27f53c872
--- /dev/null
+++ b/ui/src/eventStream.test.js
@@ -0,0 +1,51 @@
+import { describe, it, beforeEach, vi, expect } from 'vitest'
+import { startEventStream } from './eventStream'
+import { serverDown } from './actions'
+import config from './config'
+
+class MockEventSource {
+ constructor(url) {
+ this.url = url
+ this.readyState = 1
+ this.listeners = {}
+ this.onerror = null
+ }
+ addEventListener(type, handler) {
+ this.listeners[type] = handler
+ }
+ close() {
+ this.readyState = 2
+ }
+}
+
+describe('startEventStream', () => {
+ vi.useFakeTimers()
+ let dispatch
+ let instance
+
+ beforeEach(() => {
+ dispatch = vi.fn()
+ global.EventSource = vi.fn().mockImplementation(function (url) {
+ instance = new MockEventSource(url)
+ return instance
+ })
+ localStorage.setItem('is-authenticated', 'true')
+ localStorage.setItem('token', 'abc')
+ config.devNewEventStream = true
+ // Mock console.log to suppress output during tests
+ vi.spyOn(console, 'log').mockImplementation(() => {})
+ })
+
+ afterEach(() => {
+ config.devNewEventStream = false
+ })
+
+ it('reconnects after an error', async () => {
+ await startEventStream(dispatch)
+ expect(global.EventSource).toHaveBeenCalledTimes(1)
+ instance.onerror(new Event('error'))
+ expect(dispatch).toHaveBeenCalledWith(serverDown())
+ vi.advanceTimersByTime(5000)
+ expect(global.EventSource).toHaveBeenCalledTimes(2)
+ })
+})
diff --git a/ui/src/i18n/en.json b/ui/src/i18n/en.json
index c40197277..a8d807ba9 100644
--- a/ui/src/i18n/en.json
+++ b/ui/src/i18n/en.json
@@ -12,6 +12,7 @@
"artist": "Artist",
"album": "Album",
"path": "File path",
+ "libraryName": "Library",
"genre": "Genre",
"compilation": "Compilation",
"year": "Year",
@@ -41,6 +42,7 @@
"addToQueue": "Play Later",
"playNow": "Play Now",
"addToPlaylist": "Add to Playlist",
+ "showInPlaylist": "Show in Playlist",
"shuffleAll": "Shuffle All",
"download": "Download",
"playNext": "Play Next",
@@ -57,6 +59,7 @@
"playCount": "Plays",
"size": "Size",
"name": "Name",
+ "libraryName": "Library",
"genre": "Genre",
"compilation": "Compilation",
"year": "Year",
@@ -123,7 +126,13 @@
"mixer": "Mixer |||| Mixers",
"remixer": "Remixer |||| Remixers",
"djmixer": "DJ Mixer |||| DJ Mixers",
- "performer": "Performer |||| Performers"
+ "performer": "Performer |||| Performers",
+ "maincredit": "Album Artist or Artist |||| Album Artists or Artists"
+ },
+ "actions": {
+ "topSongs": "Top Songs",
+ "shuffle": "Shuffle",
+ "radio": "Radio"
}
},
"user": {
@@ -140,19 +149,26 @@
"changePassword": "Change Password?",
"currentPassword": "Current Password",
"newPassword": "New Password",
- "token": "Token"
+ "token": "Token",
+ "libraries": "Libraries"
},
"helperTexts": {
- "name": "Changes to your name will only be reflected on next login"
+ "name": "Changes to your name will only be reflected on next login",
+ "libraries": "Select specific libraries for this user, or leave empty to use default libraries"
},
"notifications": {
"created": "User created",
"updated": "User updated",
"deleted": "User deleted"
},
+ "validation": {
+ "librariesRequired": "At least one library must be selected for non-admin users"
+ },
"message": {
"listenBrainzToken": "Enter your ListenBrainz user token.",
- "clickHereForToken": "Click here to get your token"
+ "clickHereForToken": "Click here to get your token",
+ "selectAllLibraries": "Select all libraries",
+ "adminAutoLibraries": "Admin users automatically have access to all libraries"
}
},
"player": {
@@ -198,11 +214,16 @@
"export": "Export",
"saveQueue": "Save Queue to Playlist",
"makePublic": "Make Public",
- "makePrivate": "Make Private"
+ "makePrivate": "Make Private",
+ "searchOrCreate": "Search playlists or type to create new...",
+ "pressEnterToCreate": "Press Enter to create new playlist",
+ "removeFromSelection": "Remove from selection"
},
"message": {
"duplicate_song": "Add duplicated songs",
- "song_exist": "There are duplicates being added to the playlist. Would you like to add the duplicates or skip them?"
+ "song_exist": "There are duplicates being added to the playlist. Would you like to add the duplicates or skip them?",
+ "noPlaylistsFound": "No playlists found",
+ "noPlaylists": "No playlists available"
}
},
"radio": {
@@ -243,6 +264,7 @@
"fields": {
"path": "Path",
"size": "Size",
+ "libraryName": "Library",
"updatedAt": "Disappeared on"
},
"actions": {
@@ -252,6 +274,63 @@
"notifications": {
"removed": "Missing file(s) removed"
}
+ },
+ "library": {
+ "name": "Library |||| Libraries",
+ "fields": {
+ "name": "Name",
+ "path": "Path",
+ "remotePath": "Remote Path",
+ "lastScanAt": "Last Scan",
+ "songCount": "Songs",
+ "albumCount": "Albums",
+ "artistCount": "Artists",
+ "totalSongs": "Songs",
+ "totalAlbums": "Albums",
+ "totalArtists": "Artists",
+ "totalFolders": "Folders",
+ "totalFiles": "Files",
+ "totalMissingFiles": "Missing Files",
+ "totalSize": "Total Size",
+ "totalDuration": "Duration",
+ "defaultNewUsers": "Default for New Users",
+ "createdAt": "Created",
+ "updatedAt": "Updated"
+ },
+ "sections": {
+ "basic": "Basic Information",
+ "statistics": "Statistics"
+ },
+ "actions": {
+ "scan": "Scan Library",
+ "quickScan": "Quick Scan",
+ "fullScan": "Full Scan",
+ "manageUsers": "Manage User Access",
+ "viewDetails": "View Details"
+ },
+ "notifications": {
+ "created": "Library created successfully",
+ "updated": "Library updated successfully",
+ "deleted": "Library deleted successfully",
+ "scanStarted": "Library scan started",
+ "quickScanStarted": "Quick scan started",
+ "fullScanStarted": "Full scan started",
+ "scanError": "Error starting scan. Check logs",
+ "scanCompleted": "Library scan completed"
+ },
+ "validation": {
+ "nameRequired": "Library name is required",
+ "pathRequired": "Library path is required",
+ "pathNotDirectory": "Library path must be a directory",
+ "pathNotFound": "Library path not found",
+ "pathNotAccessible": "Library path is not accessible",
+ "pathInvalid": "Invalid library path"
+ },
+ "messages": {
+ "deleteConfirm": "Are you sure you want to delete this library? This will remove all associated data and user access.",
+ "scanInProgress": "Scan in progress...",
+ "noLibrariesAssigned": "No libraries assigned to this user"
+ }
}
},
"ra": {
@@ -404,6 +483,8 @@
"transcodingDisabled": "Changing the transcoding configuration through the web interface is disabled for security reasons. If you would like to change (edit or add) transcoding options, restart the server with the %{config} configuration option.",
"transcodingEnabled": "Navidrome is currently running with %{config}, making it possible to run system commands from the transcoding settings using the web interface. We recommend to disable it for security reasons and only enable it when configuring Transcoding options.",
"songsAddedToPlaylist": "Added 1 song to playlist |||| Added %{smart_count} songs to playlist",
+ "noSimilarSongsFound": "No similar songs found",
+ "noTopSongsFound": "No top songs found",
"noPlaylistsAvailable": "None available",
"delete_user_title": "Delete user '%{name}'",
"delete_user_content": "Are you sure you want to delete this user and all their data (including playlists and preferences)?",
@@ -437,6 +518,12 @@
},
"menu": {
"library": "Library",
+ "librarySelector": {
+ "allLibraries": "All Libraries (%{count})",
+ "multipleLibraries": "%{selected} of %{total} Libraries",
+ "selectLibraries": "Select Libraries",
+ "none": "None"
+ },
"settings": "Settings",
"version": "Version",
"theme": "Theme",
@@ -499,19 +586,40 @@
"disabled": "Disabled",
"waiting": "Waiting"
}
+ },
+ "tabs": {
+ "about": "About",
+ "config": "Configuration"
+ },
+ "config": {
+ "configName": "Config Name",
+ "environmentVariable": "Environment Variable",
+ "currentValue": "Current Value",
+ "configurationFile": "Configuration File",
+ "exportToml": "Export Configuration (TOML)",
+ "exportSuccess": "Configuration exported to clipboard in TOML format",
+ "exportFailed": "Failed to copy configuration",
+ "devFlagsHeader": "Development Flags (subject to change/removal)",
+ "devFlagsComment": "These are experimental settings and may be removed in future versions"
}
},
"activity": {
"title": "Activity",
"totalScanned": "Total Folders Scanned",
- "quickScan": "Quick Scan",
- "fullScan": "Full Scan",
+ "quickScan": "Quick",
+ "fullScan": "Full",
+ "selectiveScan": "Selective",
"serverUptime": "Server Uptime",
"serverDown": "OFFLINE",
- "scanType": "Type",
+ "scanType": "Last Scan",
"status": "Scan Error",
"elapsedTime": "Elapsed Time"
},
+ "nowPlaying": {
+ "title": "Now Playing",
+ "empty": "Nothing playing",
+ "minutesAgo": "%{smart_count} minute ago |||| %{smart_count} minutes ago"
+ },
"help": {
"title": "Navidrome Hotkeys",
"hotkeys": {
diff --git a/ui/src/layout/ActivityPanel.jsx b/ui/src/layout/ActivityPanel.jsx
index 6b50cee0c..6d5d32d31 100644
--- a/ui/src/layout/ActivityPanel.jsx
+++ b/ui/src/layout/ActivityPanel.jsx
@@ -75,14 +75,25 @@ const ActivityPanel = () => {
scanStatus.scanning,
scanStatus.elapsedTime,
)
- const classes = useStyles({ up: up && !scanStatus.error })
+ const [acknowledgedError, setAcknowledgedError] = useState(null)
+ const isErrorVisible =
+ scanStatus.error && scanStatus.error !== acknowledgedError
+ const classes = useStyles({
+ up: up && (!scanStatus.error || !isErrorVisible),
+ })
const translate = useTranslate()
const notify = useNotify()
const [anchorEl, setAnchorEl] = useState(null)
const open = Boolean(anchorEl)
useInitialScanStatus()
- const handleMenuOpen = (event) => setAnchorEl(event.currentTarget)
+ const handleMenuOpen = (event) => {
+ if (scanStatus.error) {
+ setAcknowledgedError(scanStatus.error)
+ }
+ setAnchorEl(event.currentTarget)
+ }
+
const handleMenuClose = () => setAnchorEl(null)
const triggerScan = (full) => () => subsonic.startScan({ fullScan: full })
@@ -102,6 +113,9 @@ const ActivityPanel = () => {
return translate('activity.fullScan')
case 'quick':
return translate('activity.quickScan')
+ case 'full-selective':
+ case 'quick-selective':
+ return translate('activity.selectiveScan')
default:
return ''
}
@@ -111,10 +125,10 @@ const ActivityPanel = () => {
- {!up || scanStatus.error ? (
-
+ {!up || isErrorVisible ? (
+
) : (
-
+
)}
diff --git a/ui/src/layout/ActivityPanel.test.jsx b/ui/src/layout/ActivityPanel.test.jsx
new file mode 100644
index 000000000..c506fd08b
--- /dev/null
+++ b/ui/src/layout/ActivityPanel.test.jsx
@@ -0,0 +1,61 @@
+import React from 'react'
+import { render, screen, fireEvent } from '@testing-library/react'
+import { Provider } from 'react-redux'
+import { createStore, combineReducers } from 'redux'
+import { describe, it, beforeEach } from 'vitest'
+
+import ActivityPanel from './ActivityPanel'
+import { activityReducer } from '../reducers'
+import config from '../config'
+import subsonic from '../subsonic'
+
+vi.mock('../subsonic', () => ({
+ default: {
+ getScanStatus: vi.fn(() =>
+ Promise.resolve({
+ json: {
+ 'subsonic-response': {
+ status: 'ok',
+ scanStatus: { error: 'Scan failed' },
+ },
+ },
+ }),
+ ),
+ startScan: vi.fn(),
+ },
+}))
+
+describe('
', () => {
+ let store
+
+ beforeEach(() => {
+ store = createStore(combineReducers({ activity: activityReducer }), {
+ activity: {
+ scanStatus: {
+ scanning: false,
+ folderCount: 0,
+ count: 0,
+ error: 'Scan failed',
+ elapsedTime: 0,
+ },
+ serverStart: { version: config.version, startTime: Date.now() },
+ },
+ })
+ })
+
+ it('clears the error icon after opening the panel', () => {
+ render(
+
+
+ ,
+ )
+
+ const button = screen.getByRole('button')
+ expect(screen.getByTestId('activity-error-icon')).toBeInTheDocument()
+
+ fireEvent.click(button)
+
+ expect(screen.getByTestId('activity-ok-icon')).toBeInTheDocument()
+ expect(screen.getByText('Scan failed')).toBeInTheDocument()
+ })
+})
diff --git a/ui/src/layout/AppBar.jsx b/ui/src/layout/AppBar.jsx
index a8c36cd14..561701dce 100644
--- a/ui/src/layout/AppBar.jsx
+++ b/ui/src/layout/AppBar.jsx
@@ -14,6 +14,7 @@ import { Dialogs } from '../dialogs/Dialogs'
import { AboutDialog } from '../dialogs'
import PersonalMenu from './PersonalMenu'
import ActivityPanel from './ActivityPanel'
+import NowPlayingPanel from './NowPlayingPanel'
import UserMenu from './UserMenu'
import config from '../config'
@@ -49,7 +50,7 @@ const AboutMenuItem = forwardRef(({ onClick, ...rest }, ref) => {
<>
-
+
{label}
@@ -119,6 +120,9 @@ const CustomUserMenu = ({ onClick, ...rest }) => {
return (
<>
+ {config.devActivityPanel &&
+ permissions === 'admin' &&
+ config.enableNowPlaying &&
}
{config.devActivityPanel && permissions === 'admin' &&
}
diff --git a/ui/src/layout/AppBar.test.jsx b/ui/src/layout/AppBar.test.jsx
new file mode 100644
index 000000000..f39dd75cb
--- /dev/null
+++ b/ui/src/layout/AppBar.test.jsx
@@ -0,0 +1,65 @@
+import React from 'react'
+import { render, screen } from '@testing-library/react'
+import { describe, it, beforeEach, vi } from 'vitest'
+import { Provider } from 'react-redux'
+import { createStore, combineReducers } from 'redux'
+import { activityReducer } from '../reducers'
+import AppBar from './AppBar'
+import config from '../config'
+
+let store
+
+vi.mock('react-admin', () => ({
+ AppBar: ({ userMenu }) => {userMenu}
,
+ useTranslate: () => (x) => x,
+ usePermissions: () => ({ permissions: 'admin' }),
+ getResources: () => [],
+}))
+
+vi.mock('./NowPlayingPanel', () => ({
+ default: () =>
,
+}))
+vi.mock('./ActivityPanel', () => ({
+ default: () =>
,
+}))
+vi.mock('./PersonalMenu', () => ({
+ default: () =>
,
+}))
+vi.mock('./UserMenu', () => ({
+ default: ({ children }) => {children}
,
+}))
+vi.mock('../dialogs/Dialogs', () => ({
+ Dialogs: () =>
,
+}))
+vi.mock('../dialogs', () => ({
+ AboutDialog: () =>
,
+}))
+
+describe(' ', () => {
+ beforeEach(() => {
+ config.devActivityPanel = true
+ config.enableNowPlaying = true
+ store = createStore(combineReducers({ activity: activityReducer }), {
+ activity: { nowPlayingCount: 0 },
+ })
+ })
+
+ it('renders NowPlayingPanel when enabled', () => {
+ render(
+
+
+ ,
+ )
+ expect(screen.getByTestId('now-playing-panel')).toBeInTheDocument()
+ })
+
+ it('hides NowPlayingPanel when disabled', () => {
+ config.enableNowPlaying = false
+ render(
+
+
+ ,
+ )
+ expect(screen.queryByTestId('now-playing-panel')).toBeNull()
+ })
+})
diff --git a/ui/src/layout/Menu.jsx b/ui/src/layout/Menu.jsx
index bd1e37ee0..45f40b26d 100644
--- a/ui/src/layout/Menu.jsx
+++ b/ui/src/layout/Menu.jsx
@@ -9,6 +9,7 @@ import SubMenu from './SubMenu'
import { humanize, pluralize } from 'inflection'
import albumLists from '../album/albumLists'
import PlaylistsSubMenu from './PlaylistsSubMenu'
+import LibrarySelector from '../common/LibrarySelector'
import config from '../config'
const useStyles = makeStyles((theme) => ({
@@ -111,6 +112,7 @@ const Menu = ({ dense = false }) => {
[classes.closed]: !open,
})}
>
+ {open && }
handleToggle('menuAlbumList')}
isOpen={state.menuAlbumList}
diff --git a/ui/src/layout/NowPlayingPanel.jsx b/ui/src/layout/NowPlayingPanel.jsx
new file mode 100644
index 000000000..4aaee1bee
--- /dev/null
+++ b/ui/src/layout/NowPlayingPanel.jsx
@@ -0,0 +1,353 @@
+import React, { useState, useEffect, useCallback } from 'react'
+import PropTypes from 'prop-types'
+import { useSelector, useDispatch } from 'react-redux'
+import { useTranslate, Link, useNotify } from 'react-admin'
+import {
+ Popover,
+ IconButton,
+ makeStyles,
+ Tooltip,
+ List,
+ ListItem,
+ ListItemText,
+ ListItemAvatar,
+ Avatar,
+ Badge,
+ Card,
+ CardContent,
+ Typography,
+ useTheme,
+ useMediaQuery,
+} from '@material-ui/core'
+import { FaRegCirclePlay } from 'react-icons/fa6'
+import subsonic from '../subsonic'
+import { useInterval } from '../common'
+import { nowPlayingCountUpdate } from '../actions'
+import config from '../config'
+
+const useStyles = makeStyles((theme) => ({
+ button: { color: 'inherit' },
+ list: {
+ width: '30em',
+ maxHeight: (props) => {
+ // Calculate height for up to 4 entries before scrolling
+ const entryHeight = 80
+ const maxEntries = Math.min(props.entryCount || 0, 4)
+ return maxEntries > 0 ? `${maxEntries * entryHeight}px` : '12em'
+ },
+ overflowY: 'auto',
+ padding: 0,
+ },
+ card: {
+ padding: 0,
+ },
+ cardContent: {
+ padding: `${theme.spacing(1)}px !important`, // Minimal padding, override default
+ '&:last-child': {
+ paddingBottom: `${theme.spacing(1)}px !important`, // Override Material-UI's last-child padding
+ },
+ },
+ listItem: {
+ paddingTop: theme.spacing(0.5),
+ paddingBottom: theme.spacing(0.5),
+ paddingLeft: theme.spacing(1),
+ paddingRight: theme.spacing(1),
+ },
+ avatar: {
+ width: theme.spacing(6),
+ height: theme.spacing(6),
+ cursor: 'pointer',
+ '&:hover': {
+ opacity: 0.8,
+ },
+ },
+ badge: {
+ '& .MuiBadge-badge': {
+ backgroundColor: theme.palette.primary.main,
+ color: theme.palette.primary.contrastText,
+ },
+ },
+ artistLink: {
+ cursor: 'pointer',
+ '&:hover': {
+ textDecoration: 'underline',
+ },
+ },
+ primaryText: {
+ display: 'flex',
+ alignItems: 'center',
+ flexWrap: 'wrap',
+ },
+}))
+
+// NowPlayingButton component - handles the button with badge
+const NowPlayingButton = React.memo(({ count, onClick }) => {
+ const classes = useStyles()
+ const translate = useTranslate()
+
+ return (
+
+
+
+
+
+
+
+ )
+})
+
+NowPlayingButton.displayName = 'NowPlayingButton'
+
+NowPlayingButton.propTypes = {
+ count: PropTypes.number.isRequired,
+ onClick: PropTypes.func.isRequired,
+}
+
+// NowPlayingItem component - individual list item
+const NowPlayingItem = React.memo(
+ ({ nowPlayingEntry, onLinkClick, getArtistLink }) => {
+ const classes = useStyles()
+ const translate = useTranslate()
+
+ return (
+
+
+
+
+
+
+
+ {nowPlayingEntry.albumArtistId || nowPlayingEntry.artistId ? (
+
+ {nowPlayingEntry.albumArtist || nowPlayingEntry.artist}
+
+ ) : (
+
+ {nowPlayingEntry.albumArtist || nowPlayingEntry.artist}
+
+ )}
+ - {nowPlayingEntry.title}
+
+ }
+ secondary={`${nowPlayingEntry.username}${nowPlayingEntry.playerName ? ` (${nowPlayingEntry.playerName})` : ''} • ${translate('nowPlaying.minutesAgo', { smart_count: nowPlayingEntry.minutesAgo })}`}
+ />
+
+ )
+ },
+)
+
+NowPlayingItem.displayName = 'NowPlayingItem'
+
+NowPlayingItem.propTypes = {
+ nowPlayingEntry: PropTypes.shape({
+ playerId: PropTypes.oneOfType([PropTypes.string, PropTypes.number])
+ .isRequired,
+ albumId: PropTypes.oneOfType([PropTypes.string, PropTypes.number])
+ .isRequired,
+ albumArtistId: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
+ artistId: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
+ albumArtist: PropTypes.string,
+ artist: PropTypes.string,
+ title: PropTypes.string.isRequired,
+ username: PropTypes.string.isRequired,
+ playerName: PropTypes.string,
+ minutesAgo: PropTypes.number.isRequired,
+ album: PropTypes.string,
+ }).isRequired,
+ onLinkClick: PropTypes.func.isRequired,
+ getArtistLink: PropTypes.func.isRequired,
+}
+
+// NowPlayingList component - handles the popover content
+const NowPlayingList = React.memo(
+ ({ anchorEl, open, onClose, entries, onLinkClick, getArtistLink }) => {
+ const classes = useStyles({ entryCount: entries.length })
+ const translate = useTranslate()
+
+ return (
+
+
+
+ {entries.length === 0 ? (
+
+ {translate('nowPlaying.empty')}
+
+ ) : (
+
+ {entries.map((nowPlayingEntry) => (
+
+ ))}
+
+ )}
+
+
+
+ )
+ },
+)
+
+NowPlayingList.displayName = 'NowPlayingList'
+
+NowPlayingList.propTypes = {
+ anchorEl: PropTypes.object,
+ open: PropTypes.bool.isRequired,
+ onClose: PropTypes.func.isRequired,
+ entries: PropTypes.arrayOf(PropTypes.object).isRequired,
+ onLinkClick: PropTypes.func.isRequired,
+ getArtistLink: PropTypes.func.isRequired,
+}
+
+// Main NowPlayingPanel component
+const NowPlayingPanel = () => {
+ const dispatch = useDispatch()
+ const count = useSelector((state) => state.activity.nowPlayingCount)
+ const streamReconnected = useSelector(
+ (state) => state.activity.streamReconnected,
+ )
+ const serverUp = useSelector(
+ (state) => !!state.activity.serverStart.startTime,
+ )
+ const translate = useTranslate()
+ const notify = useNotify()
+ const theme = useTheme()
+ const isSmallScreen = useMediaQuery(theme.breakpoints.down('sm'))
+
+ const [anchorEl, setAnchorEl] = useState(null)
+ const [entries, setEntries] = useState([])
+ const open = Boolean(anchorEl)
+
+ const handleMenuOpen = useCallback((event) => {
+ setAnchorEl(event.currentTarget)
+ }, [])
+
+ const handleMenuClose = useCallback(() => {
+ setAnchorEl(null)
+ }, [])
+
+ // Close panel when link is clicked on small screens
+ const handleLinkClick = useCallback(() => {
+ if (isSmallScreen) {
+ handleMenuClose()
+ }
+ }, [isSmallScreen, handleMenuClose])
+
+ const getArtistLink = useCallback((artistId) => {
+ if (!artistId) return null
+ return config.devShowArtistPage && artistId !== config.variousArtistsId
+ ? `/artist/${artistId}/show`
+ : `/album?filter={"artist_id":"${artistId}"}&order=ASC&sort=max_year&displayedFilters={"compilation":true}&perPage=15`
+ }, [])
+
+ const fetchList = useCallback(
+ () =>
+ subsonic
+ .getNowPlaying()
+ .then((resp) => resp.json['subsonic-response'])
+ .then((data) => {
+ if (data.status === 'ok') {
+ const nowPlayingEntries = data.nowPlaying?.entry || []
+ setEntries(nowPlayingEntries)
+ // Also update the count in Redux store
+ dispatch(nowPlayingCountUpdate({ count: nowPlayingEntries.length }))
+ } else {
+ throw new Error(
+ data.error?.message || 'Failed to fetch now playing data',
+ )
+ }
+ })
+ .catch((error) => {
+ notify('ra.page.error', 'warning', {
+ messageArgs: { error: error.message || 'Unknown error' },
+ })
+ }),
+ [dispatch, notify],
+ )
+
+ // Initialize count and entries on mount, and refresh on server/stream changes
+ useEffect(() => {
+ if (serverUp) fetchList()
+ }, [fetchList, serverUp, streamReconnected])
+
+ // Refresh when count changes from WebSocket events (if panel is open)
+ useEffect(() => {
+ if (open && serverUp) fetchList()
+ }, [count, open, fetchList, serverUp])
+
+ // Periodic refresh when panel is open (10 seconds)
+ useInterval(
+ () => {
+ if (open && serverUp) fetchList()
+ },
+ open ? 10000 : null,
+ )
+
+ // Periodic refresh when panel is closed (60 seconds) to keep badge accurate
+ useInterval(
+ () => {
+ if (!open && serverUp) fetchList()
+ },
+ !open ? 60000 : null,
+ )
+
+ return (
+
+
+
+
+ )
+}
+
+NowPlayingPanel.propTypes = {}
+
+export default NowPlayingPanel
diff --git a/ui/src/layout/NowPlayingPanel.test.jsx b/ui/src/layout/NowPlayingPanel.test.jsx
new file mode 100644
index 000000000..4dd5dac8b
--- /dev/null
+++ b/ui/src/layout/NowPlayingPanel.test.jsx
@@ -0,0 +1,367 @@
+import React from 'react'
+import { render, screen, fireEvent, waitFor } from '@testing-library/react'
+import { describe, it, beforeEach, vi } from 'vitest'
+import { Provider } from 'react-redux'
+import { createStore, combineReducers } from 'redux'
+import { activityReducer } from '../reducers'
+import NowPlayingPanel from './NowPlayingPanel'
+import subsonic from '../subsonic'
+
+vi.mock('../subsonic', () => ({
+ default: {
+ getNowPlaying: vi.fn(),
+ getAvatarUrl: vi.fn(() => '/avatar'),
+ getCoverArtUrl: vi.fn(() => '/cover'),
+ },
+}))
+
+// Create a mock for useMediaQuery
+const mockUseMediaQuery = vi.fn()
+
+vi.mock('react-admin', async (importOriginal) => {
+ const actual = await importOriginal()
+ const redux = await import('react-redux')
+ return {
+ ...actual,
+ useTranslate: () => (x) => x,
+ useSelector: redux.useSelector,
+ useDispatch: redux.useDispatch,
+ Link: ({ to, children, onClick, ...props }) => (
+ {
+ e.preventDefault() // Prevent navigation in tests
+ if (onClick) onClick(e)
+ }}
+ {...props}
+ >
+ {children}
+
+ ),
+ }
+})
+
+// Mock the specific Material-UI hooks we need
+vi.mock('@material-ui/core/useMediaQuery', () => ({
+ default: () => mockUseMediaQuery(),
+}))
+
+vi.mock('@material-ui/core/styles/useTheme', () => ({
+ default: () => ({
+ breakpoints: {
+ down: () => '(max-width:959.95px)', // Mock breakpoint string
+ },
+ }),
+}))
+
+describe(' ', () => {
+ const createMockStore = (overrides = {}) => {
+ const defaultState = {
+ activity: {
+ nowPlayingCount: 1,
+ serverStart: { startTime: Date.now() }, // Server is up by default
+ streamReconnected: 0,
+ ...overrides,
+ },
+ }
+ return createStore(
+ combineReducers({ activity: activityReducer }),
+ defaultState,
+ )
+ }
+
+ beforeEach(() => {
+ vi.clearAllMocks()
+ mockUseMediaQuery.mockReturnValue(false) // Default to large screen
+
+ subsonic.getNowPlaying.mockResolvedValue({
+ json: {
+ 'subsonic-response': {
+ status: 'ok',
+ nowPlaying: {
+ entry: [
+ {
+ playerId: 1,
+ username: 'u1',
+ playerName: 'Chrome Browser',
+ title: 'Song',
+ albumArtist: 'Artist',
+ albumId: 'album1',
+ albumArtistId: 'artist1',
+ minutesAgo: 2,
+ },
+ ],
+ },
+ },
+ },
+ })
+ })
+
+ it('fetches and displays entries when opened', async () => {
+ const store = createMockStore()
+ render(
+
+
+ ,
+ )
+
+ // Wait for initial fetch to complete
+ await waitFor(() => {
+ expect(subsonic.getNowPlaying).toHaveBeenCalled()
+ })
+
+ fireEvent.click(screen.getByRole('button'))
+ await waitFor(() => {
+ expect(screen.getByText('Artist')).toBeInTheDocument()
+ expect(screen.getByRole('link', { name: 'Artist' })).toHaveAttribute(
+ 'href',
+ '/artist/artist1/show',
+ )
+ })
+ })
+
+ it('displays player name after username', async () => {
+ const store = createMockStore()
+ render(
+
+
+ ,
+ )
+
+ // Wait for initial fetch to complete
+ await waitFor(() => {
+ expect(subsonic.getNowPlaying).toHaveBeenCalled()
+ })
+
+ fireEvent.click(screen.getByRole('button'))
+ await waitFor(() => {
+ expect(
+ screen.getByText('u1 (Chrome Browser) • nowPlaying.minutesAgo'),
+ ).toBeInTheDocument()
+ })
+ })
+
+ it('handles entries without player name', async () => {
+ subsonic.getNowPlaying.mockResolvedValueOnce({
+ json: {
+ 'subsonic-response': {
+ status: 'ok',
+ nowPlaying: {
+ entry: [
+ {
+ playerId: 1,
+ username: 'u1',
+ title: 'Song',
+ albumArtist: 'Artist',
+ albumId: 'album1',
+ albumArtistId: 'artist1',
+ minutesAgo: 2,
+ },
+ ],
+ },
+ },
+ },
+ })
+
+ const store = createMockStore()
+ render(
+
+
+ ,
+ )
+
+ // Wait for initial fetch to complete
+ await waitFor(() => {
+ expect(subsonic.getNowPlaying).toHaveBeenCalled()
+ })
+
+ fireEvent.click(screen.getByRole('button'))
+ await waitFor(() => {
+ expect(screen.getByText('u1 • nowPlaying.minutesAgo')).toBeInTheDocument()
+ })
+ })
+
+ it('shows empty message when no entries', async () => {
+ subsonic.getNowPlaying.mockResolvedValueOnce({
+ json: {
+ 'subsonic-response': { status: 'ok', nowPlaying: { entry: [] } },
+ },
+ })
+ const store = createMockStore({ nowPlayingCount: 0 })
+ render(
+
+
+ ,
+ )
+
+ // Wait for initial fetch
+ await waitFor(() => {
+ expect(subsonic.getNowPlaying).toHaveBeenCalled()
+ })
+
+ fireEvent.click(screen.getByRole('button'))
+ await waitFor(() => {
+ expect(screen.getByText('nowPlaying.empty')).toBeInTheDocument()
+ })
+ })
+
+ it('does not close panel when artist link is clicked on large screens', async () => {
+ mockUseMediaQuery.mockReturnValue(false) // Simulate large screen
+
+ const store = createMockStore()
+ render(
+
+
+ ,
+ )
+
+ // Wait for initial fetch to complete
+ await waitFor(() => {
+ expect(subsonic.getNowPlaying).toHaveBeenCalled()
+ })
+
+ // Open the panel
+ fireEvent.click(screen.getByRole('button'))
+ await waitFor(() => {
+ expect(screen.getByText('Artist')).toBeInTheDocument()
+ })
+
+ // Check that the popover is open
+ expect(screen.getByRole('presentation')).toBeInTheDocument()
+
+ // Click the artist link
+ fireEvent.click(screen.getByRole('link', { name: 'Artist' }))
+
+ // Panel should remain open (popover should still be in document)
+ expect(screen.getByRole('presentation')).toBeInTheDocument()
+ expect(screen.getByText('Artist')).toBeInTheDocument()
+ })
+
+ it('does not fetch on mount when server is down', () => {
+ const store = createMockStore({
+ nowPlayingCount: 1,
+ serverStart: { startTime: null }, // Server is down
+ })
+ render(
+
+
+ ,
+ )
+
+ // Should not have made initial fetch request due to server being down
+ expect(subsonic.getNowPlaying).not.toHaveBeenCalled()
+ })
+
+ it('does not fetch on stream reconnection when server is down', () => {
+ const store = createMockStore({
+ nowPlayingCount: 1,
+ serverStart: { startTime: null }, // Server is down
+ streamReconnected: Date.now(), // Stream reconnected
+ })
+ render(
+
+
+ ,
+ )
+
+ // Should not have made fetch request due to server being down
+ expect(subsonic.getNowPlaying).not.toHaveBeenCalled()
+ })
+
+ it('does not double-fetch on server reconnection', () => {
+ const initialStore = createMockStore({
+ nowPlayingCount: 1,
+ serverStart: { startTime: null }, // Server initially down
+ streamReconnected: 0,
+ })
+ const { rerender } = render(
+
+
+ ,
+ )
+
+ // Clear initial (empty) calls
+ vi.clearAllMocks()
+
+ // Simulate server coming back up with stream reconnection (both state changes happen)
+ const reconnectedStore = createMockStore({
+ nowPlayingCount: 1,
+ serverStart: { startTime: Date.now() }, // Server back up
+ streamReconnected: Date.now(), // Stream reconnected
+ })
+ rerender(
+
+
+ ,
+ )
+
+ // Should only make one call despite both serverUp and streamReconnected changing
+ expect(subsonic.getNowPlaying).toHaveBeenCalledTimes(1)
+ })
+
+ it('skips polling when server is down', () => {
+ vi.useFakeTimers()
+
+ const store = createMockStore({
+ nowPlayingCount: 1,
+ serverStart: { startTime: null }, // Server is down
+ })
+ render(
+
+
+ ,
+ )
+
+ // Clear initial mount fetch
+ vi.clearAllMocks()
+
+ // Advance time by 70 seconds to trigger polling interval
+ vi.advanceTimersByTime(70000)
+
+ // Should not have made any additional requests due to server being down
+ expect(subsonic.getNowPlaying).not.toHaveBeenCalled()
+
+ vi.useRealTimers()
+ })
+
+ it('resumes polling when server comes back up', () => {
+ vi.useFakeTimers()
+
+ const store = createMockStore({
+ nowPlayingCount: 1,
+ serverStart: { startTime: null }, // Server is down
+ })
+ const { rerender } = render(
+
+
+ ,
+ )
+
+ // Clear initial mount fetch
+ vi.clearAllMocks()
+
+ // Advance time - should not poll when server is down
+ vi.advanceTimersByTime(70000)
+ expect(subsonic.getNowPlaying).not.toHaveBeenCalled()
+
+ // Update state to indicate server is back up
+ const updatedStore = createMockStore({
+ nowPlayingCount: 1,
+ serverStart: { startTime: Date.now() }, // Server is back up
+ })
+ rerender(
+
+
+ ,
+ )
+
+ // Clear the fetch that happens due to initial mount of rerender
+ vi.clearAllMocks()
+
+ // Advance time again - should now poll since server is up
+ vi.advanceTimersByTime(70000)
+ expect(subsonic.getNowPlaying).toHaveBeenCalled()
+
+ vi.useRealTimers()
+ })
+})
diff --git a/ui/src/library/DeleteLibraryButton.jsx b/ui/src/library/DeleteLibraryButton.jsx
new file mode 100644
index 000000000..8d9ff6ed2
--- /dev/null
+++ b/ui/src/library/DeleteLibraryButton.jsx
@@ -0,0 +1,80 @@
+import React from 'react'
+import DeleteIcon from '@material-ui/icons/Delete'
+import { makeStyles, alpha } from '@material-ui/core/styles'
+import clsx from 'clsx'
+import {
+ useNotify,
+ useDeleteWithConfirmController,
+ Button,
+ Confirm,
+ useTranslate,
+ useRedirect,
+} from 'react-admin'
+
+const useStyles = makeStyles(
+ (theme) => ({
+ deleteButton: {
+ color: theme.palette.error.main,
+ '&:hover': {
+ backgroundColor: alpha(theme.palette.error.main, 0.12),
+ // Reset on mouse devices
+ '@media (hover: none)': {
+ backgroundColor: 'transparent',
+ },
+ },
+ },
+ }),
+ { name: 'RaDeleteWithConfirmButton' },
+)
+
+const DeleteLibraryButton = ({
+ record,
+ resource,
+ basePath,
+ className,
+ ...props
+}) => {
+ const translate = useTranslate()
+ const notify = useNotify()
+ const redirect = useRedirect()
+
+ const onSuccess = () => {
+ notify('resources.library.notifications.deleted', 'info', {
+ smart_count: 1,
+ })
+ redirect('/library')
+ }
+
+ const { open, loading, handleDialogOpen, handleDialogClose, handleDelete } =
+ useDeleteWithConfirmController({
+ resource,
+ record,
+ basePath,
+ onSuccess,
+ })
+
+ const classes = useStyles(props)
+ return (
+ <>
+
+
+
+
+ >
+ )
+}
+
+export default DeleteLibraryButton
diff --git a/ui/src/library/LibraryCreate.jsx b/ui/src/library/LibraryCreate.jsx
new file mode 100644
index 000000000..0e69964b6
--- /dev/null
+++ b/ui/src/library/LibraryCreate.jsx
@@ -0,0 +1,84 @@
+import React, { useCallback } from 'react'
+import {
+ Create,
+ SimpleForm,
+ TextInput,
+ BooleanInput,
+ required,
+ useTranslate,
+ useMutation,
+ useNotify,
+ useRedirect,
+} from 'react-admin'
+import { Title } from '../common'
+
+const LibraryCreate = (props) => {
+ const translate = useTranslate()
+ const [mutate] = useMutation()
+ const notify = useNotify()
+ const redirect = useRedirect()
+ const resourceName = translate('resources.library.name', { smart_count: 1 })
+ const title = translate('ra.page.create', {
+ name: `${resourceName}`,
+ })
+
+ const save = useCallback(
+ async (values) => {
+ try {
+ await mutate(
+ {
+ type: 'create',
+ resource: 'library',
+ payload: { data: values },
+ },
+ { returnPromise: true },
+ )
+ notify('resources.library.notifications.created', 'info', {
+ smart_count: 1,
+ })
+ redirect('/library')
+ } catch (error) {
+ // Handle validation errors with proper field mapping
+ if (error.body && error.body.errors) {
+ return error.body.errors
+ }
+
+ // Handle other structured errors from the server
+ if (error.body && error.body.error) {
+ const errorMsg = error.body.error
+
+ // Handle database constraint violations
+ if (errorMsg.includes('UNIQUE constraint failed: library.name')) {
+ return { name: 'ra.validation.unique' }
+ }
+ if (errorMsg.includes('UNIQUE constraint failed: library.path')) {
+ return { path: 'ra.validation.unique' }
+ }
+
+ // Show a general notification for other server errors
+ notify(errorMsg, 'error')
+ return
+ }
+
+ // Fallback for unexpected error formats
+ const fallbackMessage =
+ error.message ||
+ (typeof error === 'string' ? error : 'An unexpected error occurred')
+ notify(fallbackMessage, 'error')
+ }
+ },
+ [mutate, notify, redirect],
+ )
+
+ return (
+ } {...props}>
+
+
+
+
+
+
+ )
+}
+
+export default LibraryCreate
diff --git a/ui/src/library/LibraryEdit.jsx b/ui/src/library/LibraryEdit.jsx
new file mode 100644
index 000000000..7e89c892c
--- /dev/null
+++ b/ui/src/library/LibraryEdit.jsx
@@ -0,0 +1,273 @@
+import React, { useCallback } from 'react'
+import {
+ Edit,
+ FormWithRedirect,
+ TextInput,
+ BooleanInput,
+ required,
+ SaveButton,
+ DateField,
+ useTranslate,
+ useMutation,
+ useNotify,
+ useRedirect,
+ Toolbar,
+} from 'react-admin'
+import { Typography, Box } from '@material-ui/core'
+import { makeStyles } from '@material-ui/core/styles'
+import DeleteLibraryButton from './DeleteLibraryButton'
+import { Title } from '../common'
+import { formatBytes, formatDuration2, formatNumber } from '../utils/index.js'
+
+const useStyles = makeStyles({
+ toolbar: {
+ display: 'flex',
+ justifyContent: 'space-between',
+ },
+})
+
+const LibraryTitle = ({ record }) => {
+ const translate = useTranslate()
+ const resourceName = translate('resources.library.name', { smart_count: 1 })
+ return (
+
+ )
+}
+
+const CustomToolbar = ({ showDelete, ...props }) => (
+
+
+ {showDelete && (
+
+ )}
+
+)
+
+const LibraryEdit = (props) => {
+ const translate = useTranslate()
+ const [mutate] = useMutation()
+ const notify = useNotify()
+ const redirect = useRedirect()
+
+ // Library ID 1 is protected (main library)
+ const canDelete = props.id !== '1'
+ const canEditPath = props.id !== '1'
+
+ const save = useCallback(
+ async (values) => {
+ try {
+ await mutate(
+ {
+ type: 'update',
+ resource: 'library',
+ payload: { id: values.id, data: values },
+ },
+ { returnPromise: true },
+ )
+ notify('resources.library.notifications.updated', 'info', {
+ smart_count: 1,
+ })
+ redirect('/library')
+ } catch (error) {
+ if (error.body && error.body.errors) {
+ return error.body.errors
+ }
+ }
+ },
+ [mutate, notify, redirect],
+ )
+
+ return (
+ } undoable={false} {...props}>
+ (
+
+ )}
+ />
+
+ )
+}
+
+export default LibraryEdit
diff --git a/ui/src/library/LibraryList.jsx b/ui/src/library/LibraryList.jsx
new file mode 100644
index 000000000..aa1294882
--- /dev/null
+++ b/ui/src/library/LibraryList.jsx
@@ -0,0 +1,56 @@
+import React from 'react'
+import {
+ Datagrid,
+ Filter,
+ SearchInput,
+ SimpleList,
+ TextField,
+ NumberField,
+ BooleanField,
+} from 'react-admin'
+import { useMediaQuery } from '@material-ui/core'
+import { List, DateField, useResourceRefresh, SizeField } from '../common'
+import LibraryListBulkActions from './LibraryListBulkActions'
+import LibraryListActions from './LibraryListActions'
+
+const LibraryFilter = (props) => (
+
+
+
+)
+
+const LibraryList = (props) => {
+ const isXsmall = useMediaQuery((theme) => theme.breakpoints.down('xs'))
+ useResourceRefresh('library')
+
+ return (
+
}
+ filters={ }
+ actions={ }
+ >
+ {isXsmall ? (
+ record.name}
+ secondaryText={(record) => record.path}
+ />
+ ) : (
+
+
+
+
+
+
+
+
+
+
+ )}
+
+ )
+}
+
+export default LibraryList
diff --git a/ui/src/library/LibraryListActions.jsx b/ui/src/library/LibraryListActions.jsx
new file mode 100644
index 000000000..f4d0913df
--- /dev/null
+++ b/ui/src/library/LibraryListActions.jsx
@@ -0,0 +1,31 @@
+import React, { cloneElement } from 'react'
+import { sanitizeListRestProps, TopToolbar, CreateButton } from 'react-admin'
+import LibraryScanButton from './LibraryScanButton'
+
+const LibraryListActions = ({
+ className,
+ filters,
+ resource,
+ showFilter,
+ displayedFilters,
+ filterValues,
+ ...rest
+}) => {
+ return (
+
+ {filters &&
+ cloneElement(filters, {
+ resource,
+ showFilter,
+ displayedFilters,
+ filterValues,
+ context: 'button',
+ })}
+
+
+
+
+ )
+}
+
+export default LibraryListActions
diff --git a/ui/src/library/LibraryListBulkActions.jsx b/ui/src/library/LibraryListBulkActions.jsx
new file mode 100644
index 000000000..8862a4f51
--- /dev/null
+++ b/ui/src/library/LibraryListBulkActions.jsx
@@ -0,0 +1,11 @@
+import React from 'react'
+import LibraryScanButton from './LibraryScanButton'
+
+const LibraryListBulkActions = (props) => (
+ <>
+
+
+ >
+)
+
+export default LibraryListBulkActions
diff --git a/ui/src/library/LibraryScanButton.jsx b/ui/src/library/LibraryScanButton.jsx
new file mode 100644
index 000000000..50d90e615
--- /dev/null
+++ b/ui/src/library/LibraryScanButton.jsx
@@ -0,0 +1,77 @@
+import React, { useState } from 'react'
+import PropTypes from 'prop-types'
+import {
+ Button,
+ useNotify,
+ useRefresh,
+ useTranslate,
+ useUnselectAll,
+} from 'react-admin'
+import { useSelector } from 'react-redux'
+import SyncIcon from '@material-ui/icons/Sync'
+import CachedIcon from '@material-ui/icons/Cached'
+import subsonic from '../subsonic'
+
+const LibraryScanButton = ({ fullScan, selectedIds, className }) => {
+ const [loading, setLoading] = useState(false)
+ const notify = useNotify()
+ const refresh = useRefresh()
+ const translate = useTranslate()
+ const unselectAll = useUnselectAll()
+ const scanStatus = useSelector((state) => state.activity.scanStatus)
+
+ const handleClick = async () => {
+ setLoading(true)
+ try {
+ // Build scan options
+ const options = { fullScan }
+
+ // If specific libraries are selected, scan only those
+ // Format: "libraryID:" to scan entire library (no folder path specified)
+ if (selectedIds && selectedIds.length > 0) {
+ options.target = selectedIds.map((id) => `${id}:`)
+ }
+
+ await subsonic.startScan(options)
+ const notificationKey = fullScan
+ ? 'resources.library.notifications.fullScanStarted'
+ : 'resources.library.notifications.quickScanStarted'
+ notify(notificationKey, 'info')
+ refresh()
+
+ // Unselect all items after successful scan
+ unselectAll('library')
+ } catch (error) {
+ notify('resources.library.notifications.scanError', 'warning')
+ } finally {
+ setLoading(false)
+ }
+ }
+
+ const isDisabled = loading || scanStatus.scanning
+
+ const label = fullScan
+ ? translate('resources.library.actions.fullScan')
+ : translate('resources.library.actions.quickScan')
+
+ const icon = fullScan ? :
+
+ return (
+
+ {icon}
+
+ )
+}
+
+LibraryScanButton.propTypes = {
+ fullScan: PropTypes.bool.isRequired,
+ selectedIds: PropTypes.array,
+ className: PropTypes.string,
+}
+
+export default LibraryScanButton
diff --git a/ui/src/library/index.js b/ui/src/library/index.js
new file mode 100644
index 000000000..3a8b71b52
--- /dev/null
+++ b/ui/src/library/index.js
@@ -0,0 +1,11 @@
+import { MdLibraryMusic } from 'react-icons/md'
+import LibraryList from './LibraryList'
+import LibraryEdit from './LibraryEdit'
+import LibraryCreate from './LibraryCreate'
+
+export default {
+ icon: MdLibraryMusic,
+ list: LibraryList,
+ edit: LibraryEdit,
+ create: LibraryCreate,
+}
diff --git a/ui/src/missing/MissingFilesList.jsx b/ui/src/missing/MissingFilesList.jsx
index 74711eed0..87d9f629f 100644
--- a/ui/src/missing/MissingFilesList.jsx
+++ b/ui/src/missing/MissingFilesList.jsx
@@ -5,10 +5,15 @@ import {
TextField,
downloadCSV,
Pagination,
+ Filter,
+ ReferenceInput,
+ useTranslate,
+ SelectInput,
} from 'react-admin'
import jsonExport from 'jsonexport/dist'
import DeleteMissingFilesButton from './DeleteMissingFilesButton.jsx'
import MissingListActions from './MissingListActions.jsx'
+import React from 'react'
const exporter = (files) => {
const filesToExport = files.map((file) => {
@@ -20,6 +25,24 @@ const exporter = (files) => {
})
}
+const MissingFilesFilter = (props) => {
+ const translate = useTranslate()
+ return (
+
+ ({ name: [searchText] })}
+ alwaysOn
+ >
+
+
+
+ )
+}
+
const BulkActionButtons = (props) => (
<>
@@ -38,11 +61,13 @@ const MissingFilesList = (props) => {
sort={{ field: 'updated_at', order: 'DESC' }}
exporter={exporter}
actions={ }
+ filters={ }
bulkActionButtons={ }
perPage={50}
pagination={ }
>
+
diff --git a/ui/src/playlist/PlaylistDetails.jsx b/ui/src/playlist/PlaylistDetails.jsx
index 634804d4f..23372a9e9 100644
--- a/ui/src/playlist/PlaylistDetails.jsx
+++ b/ui/src/playlist/PlaylistDetails.jsx
@@ -69,9 +69,13 @@ const useStyles = makeStyles(
opacity: 0.5,
},
title: {
- whiteSpace: 'nowrap',
overflow: 'hidden',
textOverflow: 'ellipsis',
+ wordBreak: 'break-word',
+ },
+ stats: {
+ marginTop: '1em',
+ marginBottom: '0.5em',
},
}),
{
@@ -144,7 +148,7 @@ const PlaylistDetails = (props) => {
>
{record.name || translate('ra.page.loading')}
-
+
{record.songCount ? (
{record.songCount}{' '}
diff --git a/ui/src/playlist/PlaylistShow.jsx b/ui/src/playlist/PlaylistShow.jsx
index 64364897f..f0cb472b1 100644
--- a/ui/src/playlist/PlaylistShow.jsx
+++ b/ui/src/playlist/PlaylistShow.jsx
@@ -5,6 +5,7 @@ import {
useShowContext,
useShowController,
Pagination,
+ Title as RaTitle,
} from 'react-admin'
import { makeStyles } from '@material-ui/core/styles'
import PlaylistDetails from './PlaylistDetails'
@@ -31,6 +32,7 @@ const PlaylistShowLayout = (props) => {
return (
<>
+ {record && } />}
{record && }
{record && (
({
@@ -66,11 +68,17 @@ const useStyles = makeStyles(
'& $contextMenu': {
visibility: 'visible',
},
+ '& $ratingField': {
+ visibility: 'visible',
+ },
},
},
contextMenu: {
visibility: (props) => (props.isDesktop ? 'hidden' : 'visible'),
},
+ ratingField: {
+ visibility: 'hidden',
+ },
}),
{ name: 'RaList' },
)
@@ -84,7 +92,8 @@ const ReorderableList = ({ readOnly, children, ...rest }) => {
const PlaylistSongs = ({ playlistId, readOnly, actions, ...props }) => {
const listContext = useListContext()
- const { data, ids, selectedIds, onUnselectItems, refetch } = listContext
+ const { data, ids, selectedIds, onUnselectItems, refetch, setPage } =
+ listContext
const isDesktop = useMediaQuery((theme) => theme.breakpoints.up('md'))
const classes = useStyles({ isDesktop })
const dispatch = useDispatch()
@@ -93,6 +102,11 @@ const PlaylistSongs = ({ playlistId, readOnly, actions, ...props }) => {
const version = useVersion()
useResourceRefresh('song', 'playlist')
+ useEffect(() => {
+ setPage(1)
+ window.scrollTo({ top: 0, behavior: 'smooth' })
+ }, [playlistId, setPage])
+
const onAddToPlaylist = useCallback(
(pls) => {
if (pls.id === playlistId) {
@@ -149,12 +163,23 @@ const PlaylistSongs = ({ playlistId, readOnly, actions, ...props }) => {
playCount: isDesktop && (
),
- playDate: ,
+ playDate: isDesktop && (
+
+ ),
quality: isDesktop && ,
channels: isDesktop && ,
bpm: isDesktop && ,
+ genre: ,
+ rating: config.enableStarRating && (
+
+ ),
}
- }, [isDesktop, classes.draggable])
+ }, [isDesktop, classes.draggable, classes.ratingField])
const columns = useSelectedFields({
resource: 'playlistTrack',
@@ -166,6 +191,8 @@ const PlaylistSongs = ({ playlistId, readOnly, actions, ...props }) => {
'playCount',
'playDate',
'albumArtist',
+ 'genre',
+ 'rating',
],
})
@@ -205,7 +232,7 @@ const PlaylistSongs = ({ playlistId, readOnly, actions, ...props }) => {
{columns}
diff --git a/ui/src/radio/RadioList.jsx b/ui/src/radio/RadioList.jsx
index 551c4c691..3d1adacc9 100644
--- a/ui/src/radio/RadioList.jsx
+++ b/ui/src/radio/RadioList.jsx
@@ -6,7 +6,6 @@ import {
DateField,
EditButton,
Filter,
- List,
sanitizeListRestProps,
SearchInput,
SimpleList,
@@ -15,6 +14,7 @@ import {
UrlField,
useTranslate,
} from 'react-admin'
+import { List } from '../common'
import { ToggleFieldsMenu, useSelectedFields } from '../common'
import { StreamField } from './StreamField'
import { setTrack } from '../actions'
diff --git a/ui/src/reducers/activityReducer.js b/ui/src/reducers/activityReducer.js
index 2b6d2741c..8238e395a 100644
--- a/ui/src/reducers/activityReducer.js
+++ b/ui/src/reducers/activityReducer.js
@@ -2,6 +2,8 @@ import {
EVENT_REFRESH_RESOURCE,
EVENT_SCAN_STATUS,
EVENT_SERVER_START,
+ EVENT_NOW_PLAYING_COUNT,
+ EVENT_STREAM_RECONNECTED,
} from '../actions'
import config from '../config'
@@ -14,6 +16,8 @@ const initialState = {
elapsedTime: 0,
},
serverStart: { version: config.version },
+ nowPlayingCount: 0,
+ streamReconnected: 0, // Timestamp of last reconnection
}
export const activityReducer = (previousState = initialState, payload) => {
@@ -40,6 +44,10 @@ export const activityReducer = (previousState = initialState, payload) => {
resources: data,
},
}
+ case EVENT_NOW_PLAYING_COUNT:
+ return { ...previousState, nowPlayingCount: data.count }
+ case EVENT_STREAM_RECONNECTED:
+ return { ...previousState, streamReconnected: Date.now() }
default:
return previousState
}
diff --git a/ui/src/reducers/activityReducer.test.js b/ui/src/reducers/activityReducer.test.js
index a1389e3d2..c9db38dbb 100644
--- a/ui/src/reducers/activityReducer.test.js
+++ b/ui/src/reducers/activityReducer.test.js
@@ -1,5 +1,10 @@
import { activityReducer } from './activityReducer'
-import { EVENT_SCAN_STATUS, EVENT_SERVER_START } from '../actions'
+import {
+ EVENT_SCAN_STATUS,
+ EVENT_SERVER_START,
+ EVENT_NOW_PLAYING_COUNT,
+ EVENT_STREAM_RECONNECTED,
+} from '../actions'
import config from '../config'
describe('activityReducer', () => {
@@ -12,6 +17,8 @@ describe('activityReducer', () => {
elapsedTime: 0,
},
serverStart: { version: config.version },
+ nowPlayingCount: 0,
+ streamReconnected: 0,
}
it('returns the initial state when no action is specified', () => {
@@ -116,4 +123,26 @@ describe('activityReducer', () => {
startTime: Date.parse('2023-01-01T00:00:00Z'),
})
})
+
+ it('handles EVENT_NOW_PLAYING_COUNT', () => {
+ const action = {
+ type: EVENT_NOW_PLAYING_COUNT,
+ data: { count: 5 },
+ }
+ const newState = activityReducer(initialState, action)
+ expect(newState.nowPlayingCount).toEqual(5)
+ })
+
+ it('handles EVENT_STREAM_RECONNECTED', () => {
+ const action = {
+ type: EVENT_STREAM_RECONNECTED,
+ data: {},
+ }
+ const beforeTimestamp = Date.now()
+ const newState = activityReducer(initialState, action)
+ const afterTimestamp = Date.now()
+
+ expect(newState.streamReconnected).toBeGreaterThanOrEqual(beforeTimestamp)
+ expect(newState.streamReconnected).toBeLessThanOrEqual(afterTimestamp)
+ })
})
diff --git a/ui/src/reducers/index.js b/ui/src/reducers/index.js
index b9414c864..3db0b1dff 100644
--- a/ui/src/reducers/index.js
+++ b/ui/src/reducers/index.js
@@ -1,3 +1,4 @@
+export * from './libraryReducer'
export * from './themeReducer'
export * from './dialogReducer'
export * from './playerReducer'
diff --git a/ui/src/reducers/libraryReducer.js b/ui/src/reducers/libraryReducer.js
new file mode 100644
index 000000000..ef613260f
--- /dev/null
+++ b/ui/src/reducers/libraryReducer.js
@@ -0,0 +1,52 @@
+import { SET_SELECTED_LIBRARIES, SET_USER_LIBRARIES } from '../actions'
+
+const initialState = {
+ userLibraries: [],
+ selectedLibraries: [], // Empty means "all accessible libraries"
+}
+
+export const libraryReducer = (previousState = initialState, payload) => {
+ const { type, data } = payload
+ switch (type) {
+ case SET_USER_LIBRARIES: {
+ const newUserLibraryIds = data.map((lib) => lib.id)
+
+ // Validate and filter selected libraries to only include IDs that exist in new user libraries
+ const validatedSelection = previousState.selectedLibraries.filter((id) =>
+ newUserLibraryIds.includes(id),
+ )
+
+ // Determine the final selection:
+ // 1. If first time setting libraries (no previous user libraries), select all
+ // 2. If user now has only one library, reset to empty (no filter needed)
+ // 3. Otherwise, use validated selection (may be empty if all previous selections were invalid)
+ let finalSelection
+ if (
+ previousState.selectedLibraries.length === 0 &&
+ previousState.userLibraries.length === 0
+ ) {
+ // First time: select all libraries
+ finalSelection = newUserLibraryIds
+ } else if (newUserLibraryIds.length === 1) {
+ // Single library: reset selection (empty means "all accessible")
+ finalSelection = []
+ } else {
+ // Multiple libraries: use validated selection
+ finalSelection = validatedSelection
+ }
+
+ return {
+ ...previousState,
+ userLibraries: data,
+ selectedLibraries: finalSelection,
+ }
+ }
+ case SET_SELECTED_LIBRARIES:
+ return {
+ ...previousState,
+ selectedLibraries: data,
+ }
+ default:
+ return previousState
+ }
+}
diff --git a/ui/src/reducers/libraryReducer.test.js b/ui/src/reducers/libraryReducer.test.js
new file mode 100644
index 000000000..b962c1036
--- /dev/null
+++ b/ui/src/reducers/libraryReducer.test.js
@@ -0,0 +1,186 @@
+import { describe, it, expect } from 'vitest'
+import { libraryReducer } from './libraryReducer'
+import { SET_SELECTED_LIBRARIES, SET_USER_LIBRARIES } from '../actions'
+
+describe('libraryReducer', () => {
+ const mockLibraries = [
+ { id: '1', name: 'Music Library' },
+ { id: '2', name: 'Podcasts' },
+ { id: '3', name: 'Audiobooks' },
+ ]
+
+ const initialState = {
+ userLibraries: [],
+ selectedLibraries: [],
+ }
+
+ describe('SET_USER_LIBRARIES', () => {
+ it('should set user libraries and select all on first load', () => {
+ const action = {
+ type: SET_USER_LIBRARIES,
+ data: mockLibraries,
+ }
+
+ const result = libraryReducer(initialState, action)
+
+ expect(result.userLibraries).toEqual(mockLibraries)
+ expect(result.selectedLibraries).toEqual(['1', '2', '3'])
+ })
+
+ it('should reset selection to empty when user has only one library', () => {
+ const previousState = {
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1', '2'],
+ }
+
+ const action = {
+ type: SET_USER_LIBRARIES,
+ data: [mockLibraries[0]], // Only one library now
+ }
+
+ const result = libraryReducer(previousState, action)
+
+ expect(result.userLibraries).toEqual([mockLibraries[0]])
+ expect(result.selectedLibraries).toEqual([]) // Reset for single library
+ })
+
+ it('should filter out invalid library IDs from selection', () => {
+ const previousState = {
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1', '2', '3'],
+ }
+
+ const action = {
+ type: SET_USER_LIBRARIES,
+ data: [mockLibraries[0], mockLibraries[1]], // Only libraries 1 and 2 remain
+ }
+
+ const result = libraryReducer(previousState, action)
+
+ expect(result.userLibraries).toEqual([mockLibraries[0], mockLibraries[1]])
+ expect(result.selectedLibraries).toEqual(['1', '2']) // Library 3 removed
+ })
+
+ it('should keep valid selection when libraries change', () => {
+ const previousState = {
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1'],
+ }
+
+ const action = {
+ type: SET_USER_LIBRARIES,
+ data: mockLibraries, // Same libraries
+ }
+
+ const result = libraryReducer(previousState, action)
+
+ expect(result.userLibraries).toEqual(mockLibraries)
+ expect(result.selectedLibraries).toEqual(['1']) // Selection preserved
+ })
+
+ it('should handle selection becoming empty after filtering invalid IDs', () => {
+ const previousState = {
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1', '2'],
+ }
+
+ const newLibraries = [{ id: '4', name: 'New Library' }]
+ const action = {
+ type: SET_USER_LIBRARIES,
+ data: newLibraries,
+ }
+
+ const result = libraryReducer(previousState, action)
+
+ expect(result.userLibraries).toEqual(newLibraries)
+ expect(result.selectedLibraries).toEqual([]) // All selected IDs were invalid
+ })
+
+ it('should handle transition from multiple to single library with invalid selection', () => {
+ const previousState = {
+ userLibraries: mockLibraries,
+ selectedLibraries: ['2', '3'], // User had libraries 2 and 3 selected
+ }
+
+ const action = {
+ type: SET_USER_LIBRARIES,
+ data: [mockLibraries[0]], // Now only has access to library 1
+ }
+
+ const result = libraryReducer(previousState, action)
+
+ expect(result.userLibraries).toEqual([mockLibraries[0]])
+ expect(result.selectedLibraries).toEqual([]) // Reset for single library
+ })
+
+ it('should handle empty library list', () => {
+ const previousState = {
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1', '2'],
+ }
+
+ const action = {
+ type: SET_USER_LIBRARIES,
+ data: [],
+ }
+
+ const result = libraryReducer(previousState, action)
+
+ expect(result.userLibraries).toEqual([])
+ expect(result.selectedLibraries).toEqual([]) // All selections filtered out
+ })
+ })
+
+ describe('SET_SELECTED_LIBRARIES', () => {
+ it('should update selected libraries', () => {
+ const previousState = {
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1'],
+ }
+
+ const action = {
+ type: SET_SELECTED_LIBRARIES,
+ data: ['2', '3'],
+ }
+
+ const result = libraryReducer(previousState, action)
+
+ expect(result.selectedLibraries).toEqual(['2', '3'])
+ expect(result.userLibraries).toEqual(mockLibraries) // Unchanged
+ })
+
+ it('should allow setting empty selection', () => {
+ const previousState = {
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1', '2'],
+ }
+
+ const action = {
+ type: SET_SELECTED_LIBRARIES,
+ data: [],
+ }
+
+ const result = libraryReducer(previousState, action)
+
+ expect(result.selectedLibraries).toEqual([])
+ })
+ })
+
+ describe('unknown action', () => {
+ it('should return previous state for unknown action', () => {
+ const previousState = {
+ userLibraries: mockLibraries,
+ selectedLibraries: ['1'],
+ }
+
+ const action = {
+ type: 'UNKNOWN_ACTION',
+ data: null,
+ }
+
+ const result = libraryReducer(previousState, action)
+
+ expect(result).toBe(previousState) // Same reference
+ })
+ })
+})
diff --git a/ui/src/share/ShareList.jsx b/ui/src/share/ShareList.jsx
index 183d6af6b..48ab44e18 100644
--- a/ui/src/share/ShareList.jsx
+++ b/ui/src/share/ShareList.jsx
@@ -2,13 +2,13 @@ import {
Datagrid,
FunctionField,
BooleanField,
- List,
NumberField,
SimpleList,
TextField,
useNotify,
useTranslate,
} from 'react-admin'
+import { List } from '../common'
import React from 'react'
import { IconButton, Link, useMediaQuery } from '@material-ui/core'
import ShareIcon from '@material-ui/icons/Share'
diff --git a/ui/src/song/SongList.jsx b/ui/src/song/SongList.jsx
index 2a2807964..f067e11d2 100644
--- a/ui/src/song/SongList.jsx
+++ b/ui/src/song/SongList.jsx
@@ -182,7 +182,9 @@ const SongList = (props) => {
),
comment: ,
path: ,
- createdAt: ,
+ createdAt: (
+
+ ),
}
}, [isDesktop, classes.ratingField])
diff --git a/ui/src/store/createAdminStore.js b/ui/src/store/createAdminStore.js
index e4877eb14..4888e49e4 100644
--- a/ui/src/store/createAdminStore.js
+++ b/ui/src/store/createAdminStore.js
@@ -57,6 +57,7 @@ const createAdminStore = ({
const state = store.getState()
saveState({
theme: state.theme,
+ library: state.library,
player: (({ queue, volume, savedPlayIndex }) => ({
queue,
volume,
diff --git a/ui/src/subsonic/index.js b/ui/src/subsonic/index.js
index 857e33f3c..cfcc01043 100644
--- a/ui/src/subsonic/index.js
+++ b/ui/src/subsonic/index.js
@@ -23,7 +23,13 @@ const url = (command, id, options) => {
delete options.ts
}
Object.keys(options).forEach((k) => {
- params.append(k, options[k])
+ const value = options[k]
+ // Handle array parameters by appending each value separately
+ if (Array.isArray(value)) {
+ value.forEach((v) => params.append(k, v))
+ } else {
+ params.append(k, value)
+ }
})
}
return `/rest/${command}?${params.toString()}`
@@ -31,15 +37,16 @@ const url = (command, id, options) => {
const ping = () => httpClient(url('ping'))
-const scrobble = (id, time, submission = true) =>
+const scrobble = (id, time, submission = true, position = null) =>
httpClient(
url('scrobble', id, {
...(submission && time && { time }),
submission,
+ ...(!submission && position !== null && { position }),
}),
)
-const nowPlaying = (id) => scrobble(id, null, false)
+const nowPlaying = (id, position = null) => scrobble(id, null, false, position)
const star = (id) => httpClient(url('star', id))
@@ -54,6 +61,16 @@ const startScan = (options) => httpClient(url('startScan', null, options))
const getScanStatus = () => httpClient(url('getScanStatus'))
+const getNowPlaying = () => httpClient(url('getNowPlaying'))
+
+const getAvatarUrl = (username, size) =>
+ baseUrl(
+ url('getAvatar', null, {
+ username,
+ ...(size && { size }),
+ }),
+ )
+
const getCoverArtUrl = (record, size, square) => {
const options = {
...(record.updatedAt && { _: record.updatedAt }),
@@ -82,6 +99,14 @@ const getAlbumInfo = (id) => {
return httpClient(url('getAlbumInfo', id))
}
+const getSimilarSongs2 = (id, count = 100) => {
+ return httpClient(url('getSimilarSongs2', id, { count }))
+}
+
+const getTopSongs = (artist, count = 50) => {
+ return httpClient(url('getTopSongs', null, { artist, count }))
+}
+
const streamUrl = (id, options) => {
return baseUrl(
url('stream', id, {
@@ -102,8 +127,12 @@ export default {
setRating,
startScan,
getScanStatus,
+ getNowPlaying,
getCoverArtUrl,
+ getAvatarUrl,
streamUrl,
getAlbumInfo,
getArtistInfo,
+ getTopSongs,
+ getSimilarSongs2,
}
diff --git a/ui/src/subsonic/index.test.js b/ui/src/subsonic/index.test.js
index 6b902dfb1..1e0fbeaa6 100644
--- a/ui/src/subsonic/index.test.js
+++ b/ui/src/subsonic/index.test.js
@@ -104,3 +104,26 @@ describe('getCoverArtUrl', () => {
expect(url).not.toContain('_=')
})
})
+
+describe('getAvatarUrl', () => {
+ beforeEach(() => {
+ // Mock localStorage values required by subsonic
+ const localStorageMock = {
+ getItem: vi.fn((key) => {
+ const values = {
+ username: 'testuser',
+ 'subsonic-token': 'testtoken',
+ 'subsonic-salt': 'testsalt',
+ }
+ return values[key] || null
+ }),
+ }
+ Object.defineProperty(window, 'localStorage', { value: localStorageMock })
+ })
+
+ it('should include username parameter', () => {
+ const url = subsonic.getAvatarUrl('john')
+ expect(url).toContain('getAvatar')
+ expect(url).toContain('username=john')
+ })
+})
diff --git a/ui/src/themes/dark.js b/ui/src/themes/dark.js
index 2f06b4337..15d8aa365 100644
--- a/ui/src/themes/dark.js
+++ b/ui/src/themes/dark.js
@@ -16,6 +16,11 @@ export default {
color: 'white',
},
},
+ MuiButton: {
+ textPrimary: {
+ color: '#fff',
+ },
+ },
NDLogin: {
systemNameLink: {
color: '#0085ff',
diff --git a/ui/src/themes/gruvboxDark.js b/ui/src/themes/gruvboxDark.js
index b576e7713..b1a2e4c90 100644
--- a/ui/src/themes/gruvboxDark.js
+++ b/ui/src/themes/gruvboxDark.js
@@ -40,6 +40,11 @@ export default {
color: '#ebdbb2',
},
},
+ MuiIconButton: {
+ root: {
+ color: '#ebdbb2',
+ },
+ },
MuiChip: {
clickable: {
background: '#49483e',
diff --git a/ui/src/themes/ligera.js b/ui/src/themes/ligera.js
index 824cf7e67..363a379bc 100644
--- a/ui/src/themes/ligera.js
+++ b/ui/src/themes/ligera.js
@@ -70,7 +70,7 @@ export default {
},
background: {
default: '#f0f2f5',
- paper: 'inherit',
+ paper: bLight['500'],
},
text: {
secondary: '#232323',
@@ -448,15 +448,28 @@ export default {
backgroundColor: bLight['500'],
},
},
+ RaButton: {
+ button: {
+ margin: '0 5px 0 5px',
+ },
+ },
RaPaginationActions: {
button: {
- backgroundColor: 'inherit',
+ backgroundColor: '#fff',
+ color: '#000',
minWidth: 48,
margin: '0 4px',
- border: '1px solid #282828',
+ border: '1px solid #cccccc',
'@global': {
'> .MuiButton-label': {
padding: 0,
+ color: '#656565',
+ '&:hover': {
+ color: '#fff !important',
+ },
+ },
+ '> .MuiButton-label > svg': {
+ color: '#656565',
},
},
},
diff --git a/ui/src/themes/nord.js b/ui/src/themes/nord.js
index 8c346eefe..5420bbc60 100644
--- a/ui/src/themes/nord.js
+++ b/ui/src/themes/nord.js
@@ -259,7 +259,6 @@ export default {
},
details: {
fontSize: '.875rem',
- minWidth: '75vw',
color: 'rgba(255,255,255, 0.8)',
},
},
diff --git a/ui/src/themes/spotify.js b/ui/src/themes/spotify.js
index 703d8159e..725831cc7 100644
--- a/ui/src/themes/spotify.js
+++ b/ui/src/themes/spotify.js
@@ -204,7 +204,6 @@ export default {
},
details: {
fontSize: '.875rem',
- minWidth: '75vw',
color: 'rgba(255,255,255, 0.8)',
},
},
@@ -243,6 +242,64 @@ export default {
NDPlaylistShow: {
playlistActions: musicListActions,
},
+ NDArtistShow: {
+ actions: {
+ padding: '2rem 0',
+ alignItems: 'center',
+ overflow: 'visible',
+ minHeight: '120px',
+ '@global': {
+ button: {
+ border: '1px solid transparent',
+ backgroundColor: 'inherit',
+ color: '#b3b3b3',
+ margin: '0 0.5rem',
+ '&:hover': {
+ border: '1px solid #b3b3b3',
+ backgroundColor: 'inherit !important',
+ },
+ },
+ // Hide shuffle button label (first button)
+ 'button:first-child>span:first-child>span': {
+ display: 'none',
+ },
+ // Style shuffle button (first button)
+ 'button:first-child': {
+ '@media screen and (max-width: 720px)': {
+ transform: 'scale(1.5)',
+ margin: '1rem',
+ '&:hover': {
+ transform: 'scale(1.6) !important',
+ },
+ },
+ transform: 'scale(2)',
+ margin: '1.5rem',
+ minWidth: 0,
+ padding: 5,
+ transition: 'transform .3s ease',
+ background: spotifyGreen['500'],
+ color: '#fff',
+ borderRadius: 500,
+ border: 0,
+ '&:hover': {
+ transform: 'scale(2.1)',
+ backgroundColor: `${spotifyGreen['500']} !important`,
+ border: 0,
+ },
+ },
+ 'button:first-child>span:first-child': {
+ padding: 0,
+ },
+ 'button>span:first-child>span, button:not(:first-child)>span:first-child>svg':
+ {
+ color: '#b3b3b3',
+ },
+ },
+ },
+ actionsContainer: {
+ overflow: 'visible',
+ },
+ },
NDAudioPlayer: {
audioTitle: {
color: '#fff',
@@ -332,6 +389,11 @@ export default {
marginRight: '1rem',
},
},
+ RaButton: {
+ button: {
+ margin: '0 5px 0 5px',
+ },
+ },
RaPaginationActions: {
currentPageButton: {
border: '1px solid #b3b3b3',
diff --git a/ui/src/themes/theme.test.js b/ui/src/themes/theme.test.js
new file mode 100644
index 000000000..b65c3a5fe
--- /dev/null
+++ b/ui/src/themes/theme.test.js
@@ -0,0 +1,14 @@
+import themes from './index'
+import { describe, it, expect } from 'vitest'
+
+describe('NDPlaylistDetails styles', () => {
+ const themeEntries = Object.entries(themes)
+
+ it.each(themeEntries)(
+ '%s should not set minWidth on details',
+ (themeName, theme) => {
+ const details = theme.overrides?.NDPlaylistDetails?.details
+ expect(details?.minWidth).toBeUndefined()
+ },
+ )
+})
diff --git a/ui/src/user/LibrarySelectionField.jsx b/ui/src/user/LibrarySelectionField.jsx
new file mode 100644
index 000000000..4967720cd
--- /dev/null
+++ b/ui/src/user/LibrarySelectionField.jsx
@@ -0,0 +1,55 @@
+import { useInput, useTranslate, useRecordContext } from 'react-admin'
+import { Box, FormControl, FormLabel, Typography } from '@material-ui/core'
+import { SelectLibraryInput } from '../common/SelectLibraryInput.jsx'
+import React, { useMemo } from 'react'
+
+export const LibrarySelectionField = () => {
+ const translate = useTranslate()
+ const record = useRecordContext()
+
+ const {
+ input: { name, onChange, value },
+ meta: { error, touched },
+ } = useInput({ source: 'libraryIds' })
+
+ // Extract library IDs from either 'libraries' array or 'libraryIds' array
+ const libraryIds = useMemo(() => {
+ // First check if form has libraryIds (create mode or already transformed)
+ if (value && Array.isArray(value)) {
+ return value
+ }
+
+ // Then check if record has libraries array (edit mode from backend)
+ if (record?.libraries && Array.isArray(record.libraries)) {
+ return record.libraries.map((lib) => lib.id)
+ }
+
+ return []
+ }, [value, record])
+
+ // Determine if this is a new user (no ID means new record)
+ const isNewUser = !record?.id
+
+ return (
+
+
+ {translate('resources.user.fields.libraries')}
+
+
+
+
+ {touched && error && (
+
+ {error}
+
+ )}
+
+ {translate('resources.user.helperTexts.libraries')}
+
+
+ )
+}
diff --git a/ui/src/user/LibrarySelectionField.test.jsx b/ui/src/user/LibrarySelectionField.test.jsx
new file mode 100644
index 000000000..9777bab99
--- /dev/null
+++ b/ui/src/user/LibrarySelectionField.test.jsx
@@ -0,0 +1,168 @@
+import * as React from 'react'
+import { render, screen, cleanup } from '@testing-library/react'
+import { LibrarySelectionField } from './LibrarySelectionField'
+import { useInput, useTranslate, useRecordContext } from 'react-admin'
+import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest'
+import { SelectLibraryInput } from '../common/SelectLibraryInput'
+
+// Mock the react-admin hooks
+vi.mock('react-admin', () => ({
+ useInput: vi.fn(),
+ useTranslate: vi.fn(),
+ useRecordContext: vi.fn(),
+}))
+
+// Mock the SelectLibraryInput component
+vi.mock('../common/SelectLibraryInput.jsx', () => ({
+ SelectLibraryInput: vi.fn(() =>
),
+}))
+
+describe(' ', () => {
+ const defaultProps = {
+ input: {
+ name: 'libraryIds',
+ value: [],
+ onChange: vi.fn(),
+ },
+ meta: {
+ touched: false,
+ error: undefined,
+ },
+ }
+
+ const mockTranslate = vi.fn((key) => key)
+
+ beforeEach(() => {
+ useInput.mockReturnValue(defaultProps)
+ useTranslate.mockReturnValue(mockTranslate)
+ useRecordContext.mockReturnValue({})
+ SelectLibraryInput.mockClear()
+ })
+
+ afterEach(cleanup)
+
+ it('should render field label from translations', () => {
+ render( )
+ expect(screen.getByText('resources.user.fields.libraries')).not.toBeNull()
+ })
+
+ it('should render helper text from translations', () => {
+ render( )
+ expect(
+ screen.getByText('resources.user.helperTexts.libraries'),
+ ).not.toBeNull()
+ })
+
+ it('should render SelectLibraryInput with correct props', () => {
+ render( )
+ expect(screen.getByTestId('select-library-input')).not.toBeNull()
+ expect(SelectLibraryInput).toHaveBeenCalledWith(
+ expect.objectContaining({
+ onChange: defaultProps.input.onChange,
+ value: defaultProps.input.value,
+ }),
+ expect.anything(),
+ )
+ })
+
+ it('should render error message when touched and has error', () => {
+ useInput.mockReturnValue({
+ ...defaultProps,
+ meta: {
+ touched: true,
+ error: 'This field is required',
+ },
+ })
+
+ render( )
+ expect(screen.getByText('This field is required')).not.toBeNull()
+ })
+
+ it('should not render error message when not touched', () => {
+ useInput.mockReturnValue({
+ ...defaultProps,
+ meta: {
+ touched: false,
+ error: 'This field is required',
+ },
+ })
+
+ render( )
+ expect(screen.queryByText('This field is required')).toBeNull()
+ })
+
+ it('should initialize with empty array when value is null', () => {
+ useInput.mockReturnValue({
+ ...defaultProps,
+ input: {
+ ...defaultProps.input,
+ value: null,
+ },
+ })
+
+ render( )
+ expect(SelectLibraryInput).toHaveBeenCalledWith(
+ expect.objectContaining({
+ value: [],
+ }),
+ expect.anything(),
+ )
+ })
+
+ it('should extract library IDs from record libraries array when editing user', () => {
+ // Mock a record with libraries array (from backend during edit)
+ useRecordContext.mockReturnValue({
+ id: 'user123',
+ name: 'John Doe',
+ libraries: [
+ { id: 1, name: 'Music Library 1', path: '/music1' },
+ { id: 3, name: 'Music Library 3', path: '/music3' },
+ ],
+ })
+
+ // Mock input without libraryIds (edit mode scenario)
+ useInput.mockReturnValue({
+ ...defaultProps,
+ input: {
+ ...defaultProps.input,
+ value: undefined,
+ },
+ })
+
+ render( )
+ expect(SelectLibraryInput).toHaveBeenCalledWith(
+ expect.objectContaining({
+ value: [1, 3], // Should extract IDs from libraries array
+ }),
+ expect.anything(),
+ )
+ })
+
+ it('should prefer libraryIds when both libraryIds and libraries are present', () => {
+ // Mock a record with libraries array
+ useRecordContext.mockReturnValue({
+ id: 'user123',
+ libraries: [
+ { id: 1, name: 'Music Library 1', path: '/music1' },
+ { id: 3, name: 'Music Library 3', path: '/music3' },
+ ],
+ })
+
+ // Mock input with explicit libraryIds (create mode or already transformed)
+ useInput.mockReturnValue({
+ ...defaultProps,
+ input: {
+ ...defaultProps.input,
+ value: [2, 4], // Different IDs than in libraries
+ },
+ })
+
+ render( )
+ expect(SelectLibraryInput).toHaveBeenCalledWith(
+ expect.objectContaining({
+ value: [2, 4], // Should prefer libraryIds over libraries
+ }),
+ expect.anything(),
+ )
+ })
+})
diff --git a/ui/src/user/UserCreate.jsx b/ui/src/user/UserCreate.jsx
index 42ea1ce94..ce69b6542 100644
--- a/ui/src/user/UserCreate.jsx
+++ b/ui/src/user/UserCreate.jsx
@@ -2,17 +2,20 @@ import React, { useCallback } from 'react'
import {
BooleanInput,
Create,
- TextInput,
+ email,
+ FormDataConsumer,
PasswordInput,
required,
- email,
SimpleForm,
- useTranslate,
+ TextInput,
useMutation,
useNotify,
useRedirect,
+ useTranslate,
} from 'react-admin'
+import { Typography } from '@material-ui/core'
import { Title } from '../common'
+import { LibrarySelectionField } from './LibrarySelectionField.jsx'
const UserCreate = (props) => {
const translate = useTranslate()
@@ -48,9 +51,17 @@ const UserCreate = (props) => {
[mutate, notify, redirect],
)
+ // Custom validation function
+ const validateUserForm = (values) => {
+ const errors = {}
+ // Library selection is optional for non-admin users since they will be auto-assigned to default libraries
+ // No validation required for library selection
+ return errors
+ }
+
return (
} {...props}>
-
+
{
validate={[required()]}
/>
+
+ {/* Conditional Library Selection */}
+
+ {({ formData }) => (
+ <>
+ {!formData.isAdmin && }
+
+ {formData.isAdmin && (
+
+ {translate('resources.user.message.adminAutoLibraries')}
+
+ )}
+ >
+ )}
+
)
diff --git a/ui/src/user/UserEdit.jsx b/ui/src/user/UserEdit.jsx
index 445f9c6fd..2283dd8bc 100644
--- a/ui/src/user/UserEdit.jsx
+++ b/ui/src/user/UserEdit.jsx
@@ -18,9 +18,13 @@ import {
useRefresh,
FormDataConsumer,
usePermissions,
+ useRecordContext,
} from 'react-admin'
+import { Typography } from '@material-ui/core'
import { Title } from '../common'
import DeleteUserButton from './DeleteUserButton'
+import { LibrarySelectionField } from './LibrarySelectionField.jsx'
+import { validateUserForm } from './userValidation'
const useStyles = makeStyles({
toolbar: {
@@ -100,12 +104,18 @@ const UserEdit = (props) => {
[mutate, notify, permissions, redirect, refresh],
)
+ // Custom validation function
+ const validateForm = (values) => {
+ return validateUserForm(values, translate)
+ }
+
return (
} undoable={false} {...props}>
}
save={save}
+ validate={validateForm}
>
{permissions === 'admin' && (
{
{permissions === 'admin' && (
)}
+
+ {/* Conditional Library Selection for Admin Users Only */}
+ {permissions === 'admin' && (
+
+ {({ formData }) => (
+ <>
+ {!formData.isAdmin && }
+
+ {formData.isAdmin && (
+
+ {translate('resources.user.message.adminAutoLibraries')}
+
+ )}
+ >
+ )}
+
+ )}
+
diff --git a/ui/src/user/UserEdit.test.jsx b/ui/src/user/UserEdit.test.jsx
new file mode 100644
index 000000000..75a9a1ada
--- /dev/null
+++ b/ui/src/user/UserEdit.test.jsx
@@ -0,0 +1,130 @@
+import * as React from 'react'
+import { render, screen } from '@testing-library/react'
+import UserEdit from './UserEdit'
+import { describe, it, expect, vi } from 'vitest'
+
+const defaultUser = {
+ id: 'user1',
+ userName: 'testuser',
+ name: 'Test User',
+ email: 'test@example.com',
+ isAdmin: false,
+ libraries: [
+ { id: 1, name: 'Library 1', path: '/music1' },
+ { id: 2, name: 'Library 2', path: '/music2' },
+ ],
+ lastLoginAt: '2023-01-01T12:00:00Z',
+ lastAccessAt: '2023-01-02T12:00:00Z',
+ updatedAt: '2023-01-03T12:00:00Z',
+ createdAt: '2023-01-04T12:00:00Z',
+}
+
+const adminUser = {
+ ...defaultUser,
+ id: 'admin1',
+ userName: 'admin',
+ name: 'Admin User',
+ isAdmin: true,
+}
+
+// Mock React-Admin completely with simpler implementations
+vi.mock('react-admin', () => ({
+ Edit: ({ children, title }) => (
+
+ {title}
+ {children}
+
+ ),
+ SimpleForm: ({ children }) => (
+
+ ),
+ TextInput: ({ source }) => ,
+ BooleanInput: ({ source }) => (
+
+ ),
+ DateField: ({ source }) => (
+ Date
+ ),
+ PasswordInput: ({ source }) => (
+
+ ),
+ Toolbar: ({ children }) => {children}
,
+ SaveButton: () => Save ,
+ FormDataConsumer: ({ children }) => children({ formData: {} }),
+ Typography: ({ children }) => {children}
,
+ required: () => () => null,
+ email: () => () => null,
+ useMutation: () => [vi.fn()],
+ useNotify: () => vi.fn(),
+ useRedirect: () => vi.fn(),
+ useRefresh: () => vi.fn(),
+ usePermissions: () => ({ permissions: 'admin' }),
+ useTranslate: () => (key) => key,
+}))
+
+vi.mock('./LibrarySelectionField.jsx', () => ({
+ LibrarySelectionField: () =>
,
+}))
+
+vi.mock('./DeleteUserButton', () => ({
+ __esModule: true,
+ default: () => Delete ,
+}))
+
+vi.mock('../common', () => ({
+ Title: ({ subTitle }) => {subTitle}
,
+}))
+
+// Mock Material-UI
+vi.mock('@material-ui/core/styles', () => ({
+ makeStyles: () => () => ({}),
+}))
+
+vi.mock('@material-ui/core', () => ({
+ Typography: ({ children }) => {children}
,
+}))
+
+describe(' ', () => {
+ it('should render the user edit form', () => {
+ render( )
+
+ // Check if the edit component renders
+ expect(screen.getByTestId('edit-component')).toBeInTheDocument()
+ expect(screen.getByTestId('simple-form')).toBeInTheDocument()
+ })
+
+ it('should render text inputs for admin users', () => {
+ render( )
+
+ // Should render username input for admin
+ expect(screen.getByTestId('text-input-userName')).toBeInTheDocument()
+ expect(screen.getByTestId('text-input-name')).toBeInTheDocument()
+ expect(screen.getByTestId('text-input-email')).toBeInTheDocument()
+ })
+
+ it('should render admin checkbox for admin permissions', () => {
+ render( )
+
+ // Should render isAdmin checkbox for admin users
+ expect(screen.getByTestId('boolean-input-isAdmin')).toBeInTheDocument()
+ })
+
+ it('should render date fields', () => {
+ render( )
+
+ expect(screen.getByTestId('date-field-lastLoginAt')).toBeInTheDocument()
+ expect(screen.getByTestId('date-field-lastAccessAt')).toBeInTheDocument()
+ expect(screen.getByTestId('date-field-updatedAt')).toBeInTheDocument()
+ expect(screen.getByTestId('date-field-createdAt')).toBeInTheDocument()
+ })
+
+ it('should not render username input for non-admin users', () => {
+ render( )
+
+ // Should not render username input for non-admin
+ expect(screen.queryByTestId('text-input-userName')).not.toBeInTheDocument()
+ // But should still render name and email
+ expect(screen.getByTestId('text-input-name')).toBeInTheDocument()
+ expect(screen.getByTestId('text-input-email')).toBeInTheDocument()
+ })
+})
diff --git a/ui/src/user/userValidation.js b/ui/src/user/userValidation.js
new file mode 100644
index 000000000..e90fd2acb
--- /dev/null
+++ b/ui/src/user/userValidation.js
@@ -0,0 +1,19 @@
+// User form validation utilities
+export const validateUserForm = (values, translate) => {
+ const errors = {}
+
+ // Only require library selection for non-admin users
+ if (!values.isAdmin) {
+ // Check both libraryIds (array of IDs) and libraries (array of objects)
+ const hasLibraryIds = values.libraryIds && values.libraryIds.length > 0
+ const hasLibraries = values.libraries && values.libraries.length > 0
+
+ if (!hasLibraryIds && !hasLibraries) {
+ errors.libraryIds = translate(
+ 'resources.user.validation.librariesRequired',
+ )
+ }
+ }
+
+ return errors
+}
diff --git a/ui/src/user/userValidation.test.js b/ui/src/user/userValidation.test.js
new file mode 100644
index 000000000..2ee473910
--- /dev/null
+++ b/ui/src/user/userValidation.test.js
@@ -0,0 +1,70 @@
+import { describe, it, expect, vi } from 'vitest'
+import { validateUserForm } from './userValidation'
+
+describe('User Validation Utilities', () => {
+ const mockTranslate = vi.fn((key) => key)
+
+ describe('validateUserForm', () => {
+ it('should not return errors for admin users', () => {
+ const values = {
+ isAdmin: true,
+ libraryIds: [],
+ }
+ const errors = validateUserForm(values, mockTranslate)
+ expect(errors).toEqual({})
+ })
+
+ it('should not return errors for non-admin users with libraries', () => {
+ const values = {
+ isAdmin: false,
+ libraryIds: [1, 2, 3],
+ }
+ const errors = validateUserForm(values, mockTranslate)
+ expect(errors).toEqual({})
+ })
+
+ it('should return error for non-admin users without libraries', () => {
+ const values = {
+ isAdmin: false,
+ libraryIds: [],
+ }
+ const errors = validateUserForm(values, mockTranslate)
+ expect(errors.libraryIds).toBe(
+ 'resources.user.validation.librariesRequired',
+ )
+ })
+
+ it('should return error for non-admin users with undefined libraryIds', () => {
+ const values = {
+ isAdmin: false,
+ }
+ const errors = validateUserForm(values, mockTranslate)
+ expect(errors.libraryIds).toBe(
+ 'resources.user.validation.librariesRequired',
+ )
+ })
+
+ it('should not return errors for non-admin users with libraries array', () => {
+ const values = {
+ isAdmin: false,
+ libraries: [
+ { id: 1, name: 'Library 1' },
+ { id: 2, name: 'Library 2' },
+ ],
+ }
+ const errors = validateUserForm(values, mockTranslate)
+ expect(errors).toEqual({})
+ })
+
+ it('should return error for non-admin users with empty libraries array', () => {
+ const values = {
+ isAdmin: false,
+ libraries: [],
+ }
+ const errors = validateUserForm(values, mockTranslate)
+ expect(errors.libraryIds).toBe(
+ 'resources.user.validation.librariesRequired',
+ )
+ })
+ })
+})
diff --git a/ui/src/utils/formatters.js b/ui/src/utils/formatters.js
index ae27f230f..cfcb84b05 100644
--- a/ui/src/utils/formatters.js
+++ b/ui/src/utils/formatters.js
@@ -25,6 +25,42 @@ export const formatDuration = (d) => {
return `${days > 0 ? days + ':' : ''}${f}`
}
+export const formatDuration2 = (totalSeconds) => {
+ if (totalSeconds == null || totalSeconds < 0) {
+ return '0s'
+ }
+ const days = Math.floor(totalSeconds / 86400)
+ const hours = Math.floor((totalSeconds % 86400) / 3600)
+ const minutes = Math.floor((totalSeconds % 3600) / 60)
+ const seconds = Math.floor(totalSeconds % 60)
+
+ const parts = []
+
+ if (days > 0) {
+ // When days are present, show only d h m (3 levels max)
+ parts.push(`${days}d`)
+ if (hours > 0) {
+ parts.push(`${hours}h`)
+ }
+ if (minutes > 0) {
+ parts.push(`${minutes}m`)
+ }
+ } else {
+ // When no days, show h m s (3 levels max)
+ if (hours > 0) {
+ parts.push(`${hours}h`)
+ }
+ if (minutes > 0) {
+ parts.push(`${minutes}m`)
+ }
+ if (seconds > 0 || parts.length === 0) {
+ parts.push(`${seconds}s`)
+ }
+ }
+
+ return parts.join(' ')
+}
+
export const formatShortDuration = (ns) => {
// Convert nanoseconds to seconds
const seconds = ns / 1e9
@@ -58,3 +94,8 @@ export const formatFullDate = (date, locale) => {
}
return new Date(date).toLocaleDateString(locale, options)
}
+
+export const formatNumber = (value, locale) => {
+ if (value === null || value === undefined) return '0'
+ return value.toLocaleString(locale)
+}
diff --git a/ui/src/utils/formatters.test.js b/ui/src/utils/formatters.test.js
index 87b40f16b..d633e96f2 100644
--- a/ui/src/utils/formatters.test.js
+++ b/ui/src/utils/formatters.test.js
@@ -1,7 +1,9 @@
import {
formatBytes,
formatDuration,
+ formatDuration2,
formatFullDate,
+ formatNumber,
formatShortDuration,
} from './formatters'
@@ -64,11 +66,90 @@ describe('formatShortDuration', () => {
})
})
+describe('formatDuration2', () => {
+ it('handles null and undefined values', () => {
+ expect(formatDuration2(null)).toEqual('0s')
+ expect(formatDuration2(undefined)).toEqual('0s')
+ })
+
+ it('handles negative values', () => {
+ expect(formatDuration2(-10)).toEqual('0s')
+ expect(formatDuration2(-1)).toEqual('0s')
+ })
+
+ it('formats zero seconds', () => {
+ expect(formatDuration2(0)).toEqual('0s')
+ })
+
+ it('formats seconds only', () => {
+ expect(formatDuration2(1)).toEqual('1s')
+ expect(formatDuration2(30)).toEqual('30s')
+ expect(formatDuration2(59)).toEqual('59s')
+ })
+
+ it('formats minutes and seconds', () => {
+ expect(formatDuration2(60)).toEqual('1m')
+ expect(formatDuration2(90)).toEqual('1m 30s')
+ expect(formatDuration2(119)).toEqual('1m 59s')
+ expect(formatDuration2(120)).toEqual('2m')
+ })
+
+ it('formats hours, minutes and seconds', () => {
+ expect(formatDuration2(3600)).toEqual('1h')
+ expect(formatDuration2(3661)).toEqual('1h 1m 1s')
+ expect(formatDuration2(7200)).toEqual('2h')
+ expect(formatDuration2(7260)).toEqual('2h 1m')
+ expect(formatDuration2(7261)).toEqual('2h 1m 1s')
+ })
+
+ it('handles decimal values by flooring', () => {
+ expect(formatDuration2(59.9)).toEqual('59s')
+ expect(formatDuration2(60.1)).toEqual('1m')
+ expect(formatDuration2(3600.9)).toEqual('1h')
+ })
+
+ it('formats days with maximum 3 levels (d h m)', () => {
+ expect(formatDuration2(86400)).toEqual('1d')
+ expect(formatDuration2(86461)).toEqual('1d 1m') // seconds dropped when days present
+ expect(formatDuration2(90061)).toEqual('1d 1h 1m') // seconds dropped when days present
+ expect(formatDuration2(172800)).toEqual('2d')
+ expect(formatDuration2(176400)).toEqual('2d 1h')
+ expect(formatDuration2(176460)).toEqual('2d 1h 1m')
+ expect(formatDuration2(176461)).toEqual('2d 1h 1m') // seconds dropped when days present
+ })
+})
+
+describe('formatNumber', () => {
+ it('handles null and undefined values', () => {
+ expect(formatNumber(null, 'en-CA')).toEqual('0')
+ expect(formatNumber(undefined, 'en-CA')).toEqual('0')
+ })
+
+ it('formats integers', () => {
+ expect(formatNumber(0, 'en-CA')).toEqual('0')
+ expect(formatNumber(1, 'en-CA')).toEqual('1')
+ expect(formatNumber(123, 'en-CA')).toEqual('123')
+ expect(formatNumber(1000, 'en-CA')).toEqual('1,000')
+ expect(formatNumber(1234567, 'en-CA')).toEqual('1,234,567')
+ })
+
+ it('formats decimal numbers', () => {
+ expect(formatNumber(123.45, 'en-CA')).toEqual('123.45')
+ expect(formatNumber(1234.567, 'en-CA')).toEqual('1,234.567')
+ })
+
+ it('formats negative numbers', () => {
+ expect(formatNumber(-123, 'en-CA')).toEqual('-123')
+ expect(formatNumber(-1234, 'en-CA')).toEqual('-1,234')
+ expect(formatNumber(-123.45, 'en-CA')).toEqual('-123.45')
+ })
+})
+
describe('formatFullDate', () => {
it('format dates', () => {
- expect(formatFullDate('2011', 'en-US')).toEqual('2011')
- expect(formatFullDate('2011-06', 'en-US')).toEqual('Jun 2011')
- expect(formatFullDate('1985-01-01', 'en-US')).toEqual('Jan 1, 1985')
+ expect(formatFullDate('2011', 'en-CA')).toEqual('2011')
+ expect(formatFullDate('2011-06', 'en-CA')).toEqual('Jun 2011')
+ expect(formatFullDate('1985-01-01', 'en-CA')).toEqual('Jan 1, 1985')
expect(formatFullDate('199704')).toEqual('')
})
})
diff --git a/utils/cache/file_caches.go b/utils/cache/file_caches.go
index ffa9f5488..5edc533f8 100644
--- a/utils/cache/file_caches.go
+++ b/utils/cache/file_caches.go
@@ -174,7 +174,6 @@ func (fc *fileCache) Get(ctx context.Context, arg Item) (*CachedStream, error) {
go func() {
if err := copyAndClose(w, reader); err != nil {
log.Debug(ctx, "Error storing file in cache", "cache", fc.name, "key", key, err)
- _ = r.Close()
_ = fc.invalidate(ctx, key)
} else {
log.Trace(ctx, "File successfully stored in cache", "cache", fc.name, "key", key)
diff --git a/utils/cache/file_caches_test.go b/utils/cache/file_caches_test.go
index a8511d16d..72f4463d1 100644
--- a/utils/cache/file_caches_test.go
+++ b/utils/cache/file_caches_test.go
@@ -116,16 +116,18 @@ var _ = Describe("File Caches", func() {
})
})
When("reader returns error", func() {
- It("does not cache and closes the stream", func() {
+ It("does not cache", func() {
fc := callNewFileCache("test", "1KB", "test", 0, func(ctx context.Context, arg Item) (io.Reader, error) {
- return &errFakeReader{data: []byte("data"), err: errors.New("read failure")}, nil
+ return errFakeReader{errors.New("read failure")}, nil
})
s, err := fc.Get(context.Background(), &testArg{"test"})
Expect(err).ToNot(HaveOccurred())
- _, err = io.ReadAll(s)
- Expect(err.Error()).To(ContainSubstring("file already closed"))
+ _, _ = io.Copy(io.Discard, s)
+ // TODO How to make the fscache reader return the underlying reader error?
+ //Expect(err).To(MatchError("read failure"))
+ // Data should not be cached (or eventually be removed from cache)
Eventually(func() bool {
s, _ = fc.Get(context.Background(), &testArg{"test"})
if s != nil {
@@ -143,17 +145,6 @@ type testArg struct{ s string }
func (t *testArg) Key() string { return t.s }
-type errFakeReader struct {
- data []byte
- err error
- off int
-}
+type errFakeReader struct{ err error }
-func (e *errFakeReader) Read(b []byte) (int, error) {
- if e.off < len(e.data) {
- n := copy(b, e.data[e.off:])
- e.off += n
- return n, nil
- }
- return 0, e.err
-}
+func (e errFakeReader) Read([]byte) (int, error) { return 0, e.err }
diff --git a/utils/cache/simple_cache.go b/utils/cache/simple_cache.go
index 182d1d12a..cac41be7b 100644
--- a/utils/cache/simple_cache.go
+++ b/utils/cache/simple_cache.go
@@ -1,8 +1,10 @@
package cache
import (
+ "context"
"errors"
"fmt"
+ "runtime"
"sync/atomic"
"time"
@@ -17,6 +19,8 @@ type SimpleCache[K comparable, V any] interface {
GetWithLoader(key K, loader func(key K) (V, time.Duration, error)) (V, error)
Keys() []K
Values() []V
+ Len() int
+ OnExpiration(fn func(K, V)) func()
}
type Options struct {
@@ -39,9 +43,17 @@ func NewSimpleCache[K comparable, V any](options ...Options) SimpleCache[K, V] {
}
c := ttlcache.New[K, V](opts...)
- return &simpleCache[K, V]{
+ cache := &simpleCache[K, V]{
data: c,
}
+ go cache.data.Start()
+
+ // Automatic cleanup to prevent goroutine leak when cache is garbage collected
+ runtime.AddCleanup(cache, func(ttlCache *ttlcache.Cache[K, V]) {
+ ttlCache.Stop()
+ }, cache.data)
+
+ return cache
}
const evictionTimeout = 1 * time.Hour
@@ -127,3 +139,15 @@ func (c *simpleCache[K, V]) Values() []V {
})
return res
}
+
+func (c *simpleCache[K, V]) Len() int {
+ return c.data.Len()
+}
+
+func (c *simpleCache[K, V]) OnExpiration(fn func(K, V)) func() {
+ return c.data.OnEviction(func(_ context.Context, reason ttlcache.EvictionReason, item *ttlcache.Item[K, V]) {
+ if reason == ttlcache.EvictionReasonExpired {
+ fn(item.Key(), item.Value())
+ }
+ })
+}
diff --git a/utils/cache/simple_cache_test.go b/utils/cache/simple_cache_test.go
index 88dab5e07..45ba2c966 100644
--- a/utils/cache/simple_cache_test.go
+++ b/utils/cache/simple_cache_test.go
@@ -143,5 +143,19 @@ var _ = Describe("SimpleCache", func() {
Expect(cache.Get("key0")).To(Equal("value0"))
})
})
+
+ Describe("OnExpiration", func() {
+ It("should call callback when item expires", func() {
+ cache = NewSimpleCache[string, string]()
+ expired := make(chan struct{})
+ cache.OnExpiration(func(k, v string) { close(expired) })
+ Expect(cache.AddWithTTL("key", "value", 10*time.Millisecond)).To(Succeed())
+ select {
+ case <-expired:
+ case <-time.After(100 * time.Millisecond):
+ Fail("expiration callback not called")
+ }
+ })
+ })
})
})
diff --git a/utils/chain/chain_test.go b/utils/chain/chain_test.go
deleted file mode 100644
index 1c6010fb3..000000000
--- a/utils/chain/chain_test.go
+++ /dev/null
@@ -1,51 +0,0 @@
-package chain_test
-
-import (
- "errors"
- "testing"
-
- "github.com/navidrome/navidrome/utils/chain"
- . "github.com/onsi/ginkgo/v2"
- . "github.com/onsi/gomega"
-)
-
-func TestChain(t *testing.T) {
- RegisterFailHandler(Fail)
- RunSpecs(t, "chain Suite")
-}
-
-var _ = Describe("RunSequentially", func() {
- It("should return nil if no functions are provided", func() {
- err := chain.RunSequentially()
- Expect(err).To(BeNil())
- })
-
- It("should return nil if all functions succeed", func() {
- err := chain.RunSequentially(
- func() error { return nil },
- func() error { return nil },
- )
- Expect(err).To(BeNil())
- })
-
- It("should return the error from the first failing function", func() {
- expectedErr := errors.New("error in function 2")
- err := chain.RunSequentially(
- func() error { return nil },
- func() error { return expectedErr },
- func() error { return errors.New("error in function 3") },
- )
- Expect(err).To(Equal(expectedErr))
- })
-
- It("should not run functions after the first failing function", func() {
- expectedErr := errors.New("error in function 1")
- var runCount int
- err := chain.RunSequentially(
- func() error { runCount++; return expectedErr },
- func() error { runCount++; return nil },
- )
- Expect(err).To(Equal(expectedErr))
- Expect(runCount).To(Equal(1))
- })
-})
diff --git a/utils/files.go b/utils/files.go
index 59988340c..9bdc262c5 100644
--- a/utils/files.go
+++ b/utils/files.go
@@ -17,3 +17,9 @@ func BaseName(filePath string) string {
p := path.Base(filePath)
return strings.TrimSuffix(p, path.Ext(p))
}
+
+// FileExists checks if a file or directory exists
+func FileExists(path string) bool {
+ _, err := os.Stat(path)
+ return err == nil || !os.IsNotExist(err)
+}
diff --git a/utils/files_test.go b/utils/files_test.go
new file mode 100644
index 000000000..dcb28aafb
--- /dev/null
+++ b/utils/files_test.go
@@ -0,0 +1,178 @@
+package utils_test
+
+import (
+ "os"
+ "path/filepath"
+ "strings"
+
+ "github.com/navidrome/navidrome/utils"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("TempFileName", func() {
+ It("creates a temporary file name with prefix and suffix", func() {
+ prefix := "test-"
+ suffix := ".tmp"
+ result := utils.TempFileName(prefix, suffix)
+
+ Expect(result).To(ContainSubstring(prefix))
+ Expect(result).To(HaveSuffix(suffix))
+ Expect(result).To(ContainSubstring(os.TempDir()))
+ })
+
+ It("creates unique file names on multiple calls", func() {
+ prefix := "unique-"
+ suffix := ".test"
+
+ result1 := utils.TempFileName(prefix, suffix)
+ result2 := utils.TempFileName(prefix, suffix)
+
+ Expect(result1).NotTo(Equal(result2))
+ })
+
+ It("handles empty prefix and suffix", func() {
+ result := utils.TempFileName("", "")
+
+ Expect(result).To(ContainSubstring(os.TempDir()))
+ Expect(len(result)).To(BeNumerically(">", len(os.TempDir())))
+ })
+
+ It("creates proper file path separators", func() {
+ prefix := "path-test-"
+ suffix := ".ext"
+ result := utils.TempFileName(prefix, suffix)
+
+ expectedDir := os.TempDir()
+ Expect(result).To(HavePrefix(expectedDir))
+ Expect(strings.Count(result, string(filepath.Separator))).To(BeNumerically(">=", strings.Count(expectedDir, string(filepath.Separator))))
+ })
+})
+
+var _ = Describe("BaseName", func() {
+ It("extracts basename from a simple filename", func() {
+ result := utils.BaseName("test.mp3")
+ Expect(result).To(Equal("test"))
+ })
+
+ It("extracts basename from a file path", func() {
+ result := utils.BaseName("/path/to/file.txt")
+ Expect(result).To(Equal("file"))
+ })
+
+ It("handles files without extension", func() {
+ result := utils.BaseName("/path/to/filename")
+ Expect(result).To(Equal("filename"))
+ })
+
+ It("handles files with multiple dots", func() {
+ result := utils.BaseName("archive.tar.gz")
+ Expect(result).To(Equal("archive.tar"))
+ })
+
+ It("handles hidden files", func() {
+ // For hidden files without additional extension, path.Ext returns the entire name
+ // So basename becomes empty string after TrimSuffix
+ result := utils.BaseName(".hidden")
+ Expect(result).To(Equal(""))
+ })
+
+ It("handles hidden files with extension", func() {
+ result := utils.BaseName(".config.json")
+ Expect(result).To(Equal(".config"))
+ })
+
+ It("handles empty string", func() {
+ // The actual behavior returns empty string for empty input
+ result := utils.BaseName("")
+ Expect(result).To(Equal(""))
+ })
+
+ It("handles path ending with separator", func() {
+ result := utils.BaseName("/path/to/dir/")
+ Expect(result).To(Equal("dir"))
+ })
+
+ It("handles complex nested path", func() {
+ result := utils.BaseName("/very/long/path/to/my/favorite/song.mp3")
+ Expect(result).To(Equal("song"))
+ })
+})
+
+var _ = Describe("FileExists", func() {
+ var tempFile *os.File
+ var tempDir string
+
+ BeforeEach(func() {
+ var err error
+ tempFile, err = os.CreateTemp("", "fileexists-test-*.txt")
+ Expect(err).NotTo(HaveOccurred())
+
+ tempDir, err = os.MkdirTemp("", "fileexists-test-dir-*")
+ Expect(err).NotTo(HaveOccurred())
+ })
+
+ AfterEach(func() {
+ if tempFile != nil {
+ os.Remove(tempFile.Name())
+ tempFile.Close()
+ }
+ if tempDir != "" {
+ os.RemoveAll(tempDir)
+ }
+ })
+
+ It("returns true for existing file", func() {
+ Expect(utils.FileExists(tempFile.Name())).To(BeTrue())
+ })
+
+ It("returns true for existing directory", func() {
+ Expect(utils.FileExists(tempDir)).To(BeTrue())
+ })
+
+ It("returns false for non-existing file", func() {
+ nonExistentPath := filepath.Join(tempDir, "does-not-exist.txt")
+ Expect(utils.FileExists(nonExistentPath)).To(BeFalse())
+ })
+
+ It("returns false for empty path", func() {
+ Expect(utils.FileExists("")).To(BeFalse())
+ })
+
+ It("handles nested non-existing path", func() {
+ nonExistentPath := "/this/path/definitely/does/not/exist/file.txt"
+ Expect(utils.FileExists(nonExistentPath)).To(BeFalse())
+ })
+
+ Context("when file is deleted after creation", func() {
+ It("returns false after file deletion", func() {
+ filePath := tempFile.Name()
+ Expect(utils.FileExists(filePath)).To(BeTrue())
+
+ err := os.Remove(filePath)
+ Expect(err).NotTo(HaveOccurred())
+ tempFile = nil // Prevent cleanup attempt
+
+ Expect(utils.FileExists(filePath)).To(BeFalse())
+ })
+ })
+
+ Context("when directory is deleted after creation", func() {
+ It("returns false after directory deletion", func() {
+ dirPath := tempDir
+ Expect(utils.FileExists(dirPath)).To(BeTrue())
+
+ err := os.RemoveAll(dirPath)
+ Expect(err).NotTo(HaveOccurred())
+ tempDir = "" // Prevent cleanup attempt
+
+ Expect(utils.FileExists(dirPath)).To(BeFalse())
+ })
+ })
+
+ It("handles permission denied scenarios gracefully", func() {
+ // This test might be platform specific, but we test the general case
+ result := utils.FileExists("/root/.ssh/id_rsa") // Likely to not exist or be inaccessible
+ Expect(result).To(Or(BeTrue(), BeFalse())) // Should not panic
+ })
+})
diff --git a/utils/ioutils/ioutils.go b/utils/ioutils/ioutils.go
new file mode 100644
index 000000000..89d3997f3
--- /dev/null
+++ b/utils/ioutils/ioutils.go
@@ -0,0 +1,33 @@
+package ioutils
+
+import (
+ "io"
+ "os"
+
+ "golang.org/x/text/encoding/unicode"
+ "golang.org/x/text/transform"
+)
+
+// UTF8Reader wraps an io.Reader to handle Byte Order Mark (BOM) properly.
+// It strips UTF-8 BOM if present, and converts UTF-16 (LE/BE) to UTF-8.
+// This is particularly useful for reading user-provided text files (like LRC lyrics,
+// playlists) that may have been created on Windows, which often adds BOM markers.
+//
+// Reference: https://en.wikipedia.org/wiki/Byte_order_mark
+func UTF8Reader(r io.Reader) io.Reader {
+ return transform.NewReader(r, unicode.BOMOverride(unicode.UTF8.NewDecoder()))
+}
+
+// UTF8ReadFile reads the named file and returns its contents as a byte slice,
+// automatically handling BOM markers. It's similar to os.ReadFile but strips
+// UTF-8 BOM and converts UTF-16 encoded files to UTF-8.
+func UTF8ReadFile(filename string) ([]byte, error) {
+ file, err := os.Open(filename)
+ if err != nil {
+ return nil, err
+ }
+ defer file.Close()
+
+ reader := UTF8Reader(file)
+ return io.ReadAll(reader)
+}
diff --git a/utils/ioutils/ioutils_test.go b/utils/ioutils/ioutils_test.go
new file mode 100644
index 000000000..7f5483879
--- /dev/null
+++ b/utils/ioutils/ioutils_test.go
@@ -0,0 +1,117 @@
+package ioutils
+
+import (
+ "bytes"
+ "io"
+ "testing"
+
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+func TestIOUtils(t *testing.T) {
+ RegisterFailHandler(Fail)
+ RunSpecs(t, "IO Utils Suite")
+}
+
+var _ = Describe("UTF8Reader", func() {
+ Context("when reading text with UTF-8 BOM", func() {
+ It("strips the UTF-8 BOM marker", func() {
+ // UTF-8 BOM is EF BB BF
+ input := []byte{0xEF, 0xBB, 0xBF, 'h', 'e', 'l', 'l', 'o'}
+ reader := UTF8Reader(bytes.NewReader(input))
+
+ output, err := io.ReadAll(reader)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(string(output)).To(Equal("hello"))
+ })
+
+ It("strips UTF-8 BOM from multi-line text", func() {
+ // Test with the actual LRC file format
+ input := []byte{0xEF, 0xBB, 0xBF, '[', '0', '0', ':', '0', '0', '.', '0', '0', ']', ' ', 't', 'e', 's', 't'}
+ reader := UTF8Reader(bytes.NewReader(input))
+
+ output, err := io.ReadAll(reader)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(string(output)).To(Equal("[00:00.00] test"))
+ })
+ })
+
+ Context("when reading text without BOM", func() {
+ It("passes through unchanged", func() {
+ input := []byte("hello world")
+ reader := UTF8Reader(bytes.NewReader(input))
+
+ output, err := io.ReadAll(reader)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(string(output)).To(Equal("hello world"))
+ })
+ })
+
+ Context("when reading UTF-16 LE encoded text", func() {
+ It("converts to UTF-8 and strips BOM", func() {
+ // UTF-16 LE BOM (FF FE) followed by "hi" in UTF-16 LE
+ input := []byte{0xFF, 0xFE, 'h', 0x00, 'i', 0x00}
+ reader := UTF8Reader(bytes.NewReader(input))
+
+ output, err := io.ReadAll(reader)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(string(output)).To(Equal("hi"))
+ })
+ })
+
+ Context("when reading UTF-16 BE encoded text", func() {
+ It("converts to UTF-8 and strips BOM", func() {
+ // UTF-16 BE BOM (FE FF) followed by "hi" in UTF-16 BE
+ input := []byte{0xFE, 0xFF, 0x00, 'h', 0x00, 'i'}
+ reader := UTF8Reader(bytes.NewReader(input))
+
+ output, err := io.ReadAll(reader)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(string(output)).To(Equal("hi"))
+ })
+ })
+
+ Context("when reading empty content", func() {
+ It("returns empty string", func() {
+ reader := UTF8Reader(bytes.NewReader([]byte{}))
+
+ output, err := io.ReadAll(reader)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(string(output)).To(Equal(""))
+ })
+ })
+})
+
+var _ = Describe("UTF8ReadFile", func() {
+ Context("when reading a file with UTF-8 BOM", func() {
+ It("strips the BOM marker", func() {
+ // Use the actual fixture from issue #4631
+ contents, err := UTF8ReadFile("../../tests/fixtures/bom-test.lrc")
+ Expect(err).ToNot(HaveOccurred())
+
+ // Should NOT start with BOM
+ Expect(contents[0]).ToNot(Equal(byte(0xEF)))
+ // Should start with '['
+ Expect(contents[0]).To(Equal(byte('[')))
+ Expect(string(contents)).To(HavePrefix("[00:00.00]"))
+ })
+ })
+
+ Context("when reading a file without BOM", func() {
+ It("reads the file normally", func() {
+ contents, err := UTF8ReadFile("../../tests/fixtures/test.lrc")
+ Expect(err).ToNot(HaveOccurred())
+
+ // Should contain the expected content
+ Expect(string(contents)).To(ContainSubstring("We're no strangers to love"))
+ })
+ })
+
+ Context("when reading a non-existent file", func() {
+ It("returns an error", func() {
+ _, err := UTF8ReadFile("../../tests/fixtures/nonexistent.lrc")
+ Expect(err).To(HaveOccurred())
+ })
+ })
+})
diff --git a/utils/req/req.go b/utils/req/req.go
index cf498f322..f9fa5724b 100644
--- a/utils/req/req.go
+++ b/utils/req/req.go
@@ -35,6 +35,25 @@ func (r *Values) String(param string) (string, error) {
return v, nil
}
+func (r *Values) StringPtr(param string) *string {
+ var v *string
+ if _, exists := r.URL.Query()[param]; exists {
+ s := r.URL.Query().Get(param)
+ v = &s
+ }
+ return v
+}
+
+func (r *Values) BoolPtr(param string) *bool {
+ var v *bool
+ if _, exists := r.URL.Query()[param]; exists {
+ s := r.URL.Query().Get(param)
+ b := strings.Contains("/true/on/1/", "/"+strings.ToLower(s)+"/")
+ v = &b
+ }
+ return v
+}
+
func (r *Values) StringOr(param, def string) string {
v, _ := r.String(param)
if v == "" {
diff --git a/utils/req/req_test.go b/utils/req/req_test.go
index 041aca220..e710365bd 100644
--- a/utils/req/req_test.go
+++ b/utils/req/req_test.go
@@ -219,4 +219,59 @@ var _ = Describe("Request Helpers", func() {
})
})
})
+
+ Describe("ParamStringPtr", func() {
+ BeforeEach(func() {
+ r = req.Params(httptest.NewRequest("GET", "/ping?a=123", nil))
+ })
+
+ It("returns pointer to string if param exists", func() {
+ ptr := r.StringPtr("a")
+ Expect(ptr).ToNot(BeNil())
+ Expect(*ptr).To(Equal("123"))
+ })
+
+ It("returns nil if param does not exist", func() {
+ ptr := r.StringPtr("xx")
+ Expect(ptr).To(BeNil())
+ })
+
+ It("returns pointer to empty string if param exists but is empty", func() {
+ r = req.Params(httptest.NewRequest("GET", "/ping?a=", nil))
+ ptr := r.StringPtr("a")
+ Expect(ptr).ToNot(BeNil())
+ Expect(*ptr).To(Equal(""))
+ })
+ })
+
+ Describe("ParamBoolPtr", func() {
+ Context("value is true", func() {
+ BeforeEach(func() {
+ r = req.Params(httptest.NewRequest("GET", "/ping?b=true", nil))
+ })
+
+ It("returns pointer to true if param is 'true'", func() {
+ ptr := r.BoolPtr("b")
+ Expect(ptr).ToNot(BeNil())
+ Expect(*ptr).To(BeTrue())
+ })
+ })
+
+ Context("value is false", func() {
+ BeforeEach(func() {
+ r = req.Params(httptest.NewRequest("GET", "/ping?b=false", nil))
+ })
+
+ It("returns pointer to false if param is 'false'", func() {
+ ptr := r.BoolPtr("b")
+ Expect(ptr).ToNot(BeNil())
+ Expect(*ptr).To(BeFalse())
+ })
+ })
+
+ It("returns nil if param does not exist", func() {
+ ptr := r.BoolPtr("xx")
+ Expect(ptr).To(BeNil())
+ })
+ })
})
diff --git a/utils/chain/chain.go b/utils/run/run.go
similarity index 68%
rename from utils/chain/chain.go
rename to utils/run/run.go
index b93dbd93d..182eec42c 100644
--- a/utils/chain/chain.go
+++ b/utils/run/run.go
@@ -1,11 +1,11 @@
-package chain
+package run
import "golang.org/x/sync/errgroup"
-// RunSequentially runs the given functions sequentially,
+// Sequentially runs the given functions sequentially,
// If any function returns an error, it stops the execution and returns that error.
// If all functions return nil, it returns nil.
-func RunSequentially(fs ...func() error) error {
+func Sequentially(fs ...func() error) error {
for _, f := range fs {
if err := f(); err != nil {
return err
@@ -14,9 +14,9 @@ func RunSequentially(fs ...func() error) error {
return nil
}
-// RunParallel runs the given functions in parallel,
+// Parallel runs the given functions in parallel,
// It waits for all functions to finish and returns the first error encountered.
-func RunParallel(fs ...func() error) func() error {
+func Parallel(fs ...func() error) func() error {
return func() error {
g := errgroup.Group{}
for _, f := range fs {
diff --git a/utils/run/run_test.go b/utils/run/run_test.go
new file mode 100644
index 000000000..07d2d3994
--- /dev/null
+++ b/utils/run/run_test.go
@@ -0,0 +1,171 @@
+package run_test
+
+import (
+ "errors"
+ "sync/atomic"
+ "testing"
+ "time"
+
+ "github.com/navidrome/navidrome/utils/run"
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+func TestRun(t *testing.T) {
+ RegisterFailHandler(Fail)
+ RunSpecs(t, "Run Suite")
+}
+
+var _ = Describe("Sequentially", func() {
+ It("should return nil if no functions are provided", func() {
+ err := run.Sequentially()
+ Expect(err).To(BeNil())
+ })
+
+ It("should return nil if all functions succeed", func() {
+ err := run.Sequentially(
+ func() error { return nil },
+ func() error { return nil },
+ )
+ Expect(err).To(BeNil())
+ })
+
+ It("should return the error from the first failing function", func() {
+ expectedErr := errors.New("error in function 2")
+ err := run.Sequentially(
+ func() error { return nil },
+ func() error { return expectedErr },
+ func() error { return errors.New("error in function 3") },
+ )
+ Expect(err).To(Equal(expectedErr))
+ })
+
+ It("should not run functions after the first failing function", func() {
+ expectedErr := errors.New("error in function 1")
+ var runCount int
+ err := run.Sequentially(
+ func() error { runCount++; return expectedErr },
+ func() error { runCount++; return nil },
+ )
+ Expect(err).To(Equal(expectedErr))
+ Expect(runCount).To(Equal(1))
+ })
+})
+
+var _ = Describe("Parallel", func() {
+ It("should return a function that returns nil if no functions are provided", func() {
+ parallelFunc := run.Parallel()
+ err := parallelFunc()
+ Expect(err).To(BeNil())
+ })
+
+ It("should return a function that returns nil if all functions succeed", func() {
+ parallelFunc := run.Parallel(
+ func() error { return nil },
+ func() error { return nil },
+ func() error { return nil },
+ )
+ err := parallelFunc()
+ Expect(err).To(BeNil())
+ })
+
+ It("should return the first error encountered when functions fail", func() {
+ expectedErr := errors.New("parallel error")
+ parallelFunc := run.Parallel(
+ func() error { return nil },
+ func() error { return expectedErr },
+ func() error { return errors.New("another error") },
+ )
+ err := parallelFunc()
+ Expect(err).To(HaveOccurred())
+ // Note: We can't guarantee which error will be returned first in parallel execution
+ // but we can ensure an error is returned
+ })
+
+ It("should run all functions in parallel", func() {
+ var runCount atomic.Int32
+ sync := make(chan struct{})
+
+ parallelFunc := run.Parallel(
+ func() error {
+ runCount.Add(1)
+ <-sync
+ runCount.Add(-1)
+ return nil
+ },
+ func() error {
+ runCount.Add(1)
+ <-sync
+ runCount.Add(-1)
+ return nil
+ },
+ func() error {
+ runCount.Add(1)
+ <-sync
+ runCount.Add(-1)
+ return nil
+ },
+ )
+
+ // Run the parallel function in a goroutine
+ go func() {
+ Expect(parallelFunc()).To(Succeed())
+ }()
+
+ // Wait for all functions to start running
+ Eventually(func() int32 { return runCount.Load() }).Should(Equal(int32(3)))
+
+ // Release the functions to complete
+ close(sync)
+
+ // Wait for all functions to finish
+ Eventually(func() int32 { return runCount.Load() }).Should(Equal(int32(0)))
+ })
+
+ It("should wait for all functions to complete before returning", func() {
+ var completedCount atomic.Int32
+
+ parallelFunc := run.Parallel(
+ func() error {
+ completedCount.Add(1)
+ return nil
+ },
+ func() error {
+ completedCount.Add(1)
+ return nil
+ },
+ func() error {
+ completedCount.Add(1)
+ return nil
+ },
+ )
+
+ Expect(parallelFunc()).To(Succeed())
+ Expect(completedCount.Load()).To(Equal(int32(3)))
+ })
+
+ It("should return an error even if other functions are still running", func() {
+ expectedErr := errors.New("fast error")
+ var slowFunctionCompleted bool
+
+ parallelFunc := run.Parallel(
+ func() error {
+ return expectedErr // Return error immediately
+ },
+ func() error {
+ time.Sleep(50 * time.Millisecond) // Slow function
+ slowFunctionCompleted = true
+ return nil
+ },
+ )
+
+ start := time.Now()
+ err := parallelFunc()
+ duration := time.Since(start)
+
+ Expect(err).To(HaveOccurred())
+ // Should wait for all functions to complete, even if one fails early
+ Expect(duration).To(BeNumerically(">=", 50*time.Millisecond))
+ Expect(slowFunctionCompleted).To(BeTrue())
+ })
+})
diff --git a/utils/singleton/singleton.go b/utils/singleton/singleton.go
index 7f5c6a4e0..1066ae610 100644
--- a/utils/singleton/singleton.go
+++ b/utils/singleton/singleton.go
@@ -9,36 +9,61 @@ import (
)
var (
- instances = make(map[string]any)
+ instances = map[string]interface{}{}
+ pending = map[string]chan struct{}{}
lock sync.RWMutex
)
-// GetInstance returns an existing instance of object. If it is not yet created, calls `constructor`, stores the
-// result for future calls and returns it
func GetInstance[T any](constructor func() T) T {
var v T
name := reflect.TypeOf(v).String()
- v, available := func() (T, bool) {
+ // First check with read lock
+ lock.RLock()
+ if instance, ok := instances[name]; ok {
+ defer lock.RUnlock()
+ return instance.(T)
+ }
+ lock.RUnlock()
+
+ // Now check if someone is already creating this type
+ lock.Lock()
+
+ // Check again with the write lock - someone might have created it
+ if instance, ok := instances[name]; ok {
+ lock.Unlock()
+ return instance.(T)
+ }
+
+ // Check if creation is pending
+ wait, isPending := pending[name]
+ if !isPending {
+ // We'll be the one creating it
+ pending[name] = make(chan struct{})
+ wait = pending[name]
+ }
+ lock.Unlock()
+
+ // If someone else is creating it, wait for them
+ if isPending {
+ <-wait // Wait for creation to complete
+
+ // Now it should be in the instances map
lock.RLock()
defer lock.RUnlock()
- v, available := instances[name].(T)
- return v, available
- }()
-
- if available {
- return v
+ return instances[name].(T)
}
+ // We're responsible for creating the instance
+ newInstance := constructor()
+
+ // Store it and signal other goroutines
lock.Lock()
- defer lock.Unlock()
- v, available = instances[name].(T)
- if available {
- return v
- }
+ instances[name] = newInstance
+ close(wait) // Signal that creation is complete
+ delete(pending, name) // Clean up
+ log.Trace("Created new singleton", "type", name, "instance", fmt.Sprintf("%+v", newInstance))
+ lock.Unlock()
- v = constructor()
- log.Trace("Created new singleton", "type", name, "instance", fmt.Sprintf("%+v", v))
- instances[name] = v
- return v
+ return newInstance
}
diff --git a/utils/slice/slice.go b/utils/slice/slice.go
index 1d7c64f50..b1f50afcc 100644
--- a/utils/slice/slice.go
+++ b/utils/slice/slice.go
@@ -171,3 +171,14 @@ func SeqFunc[I, O any](s []I, f func(I) O) iter.Seq[O] {
}
}
}
+
+// Filter returns a new slice containing only the elements of s for which filterFunc returns true
+func Filter[T any](s []T, filterFunc func(T) bool) []T {
+ var result []T
+ for _, item := range s {
+ if filterFunc(item) {
+ result = append(result, item)
+ }
+ }
+ return result
+}
diff --git a/utils/slice/slice_test.go b/utils/slice/slice_test.go
index c6d4be1e0..65e5f0934 100644
--- a/utils/slice/slice_test.go
+++ b/utils/slice/slice_test.go
@@ -172,4 +172,42 @@ var _ = Describe("Slice Utils", func() {
Expect(result).To(ConsistOf("2", "4", "6", "8"))
})
})
+
+ Describe("Filter", func() {
+ It("returns empty slice for an empty input", func() {
+ filterFunc := func(v int) bool { return v > 0 }
+ result := slice.Filter([]int{}, filterFunc)
+ Expect(result).To(BeEmpty())
+ })
+
+ It("returns all elements when filter matches all", func() {
+ filterFunc := func(v int) bool { return v > 0 }
+ result := slice.Filter([]int{1, 2, 3, 4}, filterFunc)
+ Expect(result).To(HaveExactElements(1, 2, 3, 4))
+ })
+
+ It("returns empty slice when filter matches none", func() {
+ filterFunc := func(v int) bool { return v > 10 }
+ result := slice.Filter([]int{1, 2, 3, 4}, filterFunc)
+ Expect(result).To(BeEmpty())
+ })
+
+ It("returns only matching elements", func() {
+ filterFunc := func(v int) bool { return v%2 == 0 }
+ result := slice.Filter([]int{1, 2, 3, 4, 5, 6}, filterFunc)
+ Expect(result).To(HaveExactElements(2, 4, 6))
+ })
+
+ It("works with string slices", func() {
+ filterFunc := func(s string) bool { return len(s) > 3 }
+ result := slice.Filter([]string{"a", "abc", "abcd", "ab", "abcde"}, filterFunc)
+ Expect(result).To(HaveExactElements("abcd", "abcde"))
+ })
+
+ It("preserves order of elements", func() {
+ filterFunc := func(v int) bool { return v%2 == 1 }
+ result := slice.Filter([]int{9, 8, 7, 6, 5, 4, 3, 2, 1}, filterFunc)
+ Expect(result).To(HaveExactElements(9, 7, 5, 3, 1))
+ })
+ })
})
diff --git a/utils/str/str.go b/utils/str/str.go
index 8a94488de..f662473da 100644
--- a/utils/str/str.go
+++ b/utils/str/str.go
@@ -2,6 +2,7 @@ package str
import (
"strings"
+ "unicode/utf8"
)
var utf8ToAscii = func() *strings.Replacer {
@@ -39,3 +40,25 @@ func LongestCommonPrefix(list []string) string {
}
return list[0]
}
+
+// TruncateRunes truncates a string to a maximum number of runes, adding a suffix if truncated.
+// The suffix is included in the rune count, so if maxRunes is 30 and suffix is "...", the actual
+// string content will be truncated to fit within the maxRunes limit including the suffix.
+func TruncateRunes(s string, maxRunes int, suffix string) string {
+ if utf8.RuneCountInString(s) <= maxRunes {
+ return s
+ }
+
+ suffixRunes := utf8.RuneCountInString(suffix)
+ truncateAt := maxRunes - suffixRunes
+ if truncateAt < 0 {
+ truncateAt = 0
+ }
+
+ runes := []rune(s)
+ if truncateAt >= len(runes) {
+ return s + suffix
+ }
+
+ return string(runes[:truncateAt]) + suffix
+}
diff --git a/utils/str/str_test.go b/utils/str/str_test.go
index 0c3524e4e..511805831 100644
--- a/utils/str/str_test.go
+++ b/utils/str/str_test.go
@@ -31,6 +31,72 @@ var _ = Describe("String Utils", func() {
Expect(str.LongestCommonPrefix(albums)).To(Equal("/artist/album"))
})
})
+
+ Describe("TruncateRunes", func() {
+ It("returns string unchanged if under max runes", func() {
+ Expect(str.TruncateRunes("hello", 10, "...")).To(Equal("hello"))
+ })
+
+ It("returns string unchanged if exactly at max runes", func() {
+ Expect(str.TruncateRunes("hello", 5, "...")).To(Equal("hello"))
+ })
+
+ It("truncates and adds suffix when over max runes", func() {
+ Expect(str.TruncateRunes("hello world", 8, "...")).To(Equal("hello..."))
+ })
+
+ It("handles unicode characters correctly", func() {
+ // 6 emoji characters, maxRunes=5, suffix="..." (3 runes)
+ // So content gets 5-3=2 runes
+ Expect(str.TruncateRunes("😀😁😂😃😄😅", 5, "...")).To(Equal("😀😁..."))
+ })
+
+ It("handles multi-byte UTF-8 characters", func() {
+ // Characters like é are single runes
+ Expect(str.TruncateRunes("Café au Lait", 5, "...")).To(Equal("Ca..."))
+ })
+
+ It("works with empty suffix", func() {
+ Expect(str.TruncateRunes("hello world", 5, "")).To(Equal("hello"))
+ })
+
+ It("accounts for suffix length in truncation", func() {
+ // maxRunes=10, suffix="..." (3 runes) -> leaves 7 runes for content
+ result := str.TruncateRunes("hello world this is long", 10, "...")
+ Expect(result).To(Equal("hello w..."))
+ // Verify total rune count is <= maxRunes
+ runeCount := len([]rune(result))
+ Expect(runeCount).To(BeNumerically("<=", 10))
+ })
+
+ It("handles very long suffix gracefully", func() {
+ // If suffix is longer than maxRunes, we still add it
+ // but the content will be truncated to 0
+ result := str.TruncateRunes("hello world", 5, "... (truncated)")
+ // Result will be just the suffix (since truncateAt=0)
+ Expect(result).To(Equal("... (truncated)"))
+ })
+
+ It("handles empty string", func() {
+ Expect(str.TruncateRunes("", 10, "...")).To(Equal(""))
+ })
+
+ It("uses custom suffix", func() {
+ // maxRunes=11, suffix=" [...]" (6 runes) -> content gets 5 runes
+ // "hello world" is 11 runes exactly, so we need a longer string
+ Expect(str.TruncateRunes("hello world extra", 11, " [...]")).To(Equal("hello [...]"))
+ })
+
+ DescribeTable("truncates at rune boundaries (not byte boundaries)",
+ func(input string, maxRunes int, suffix string, expected string) {
+ Expect(str.TruncateRunes(input, maxRunes, suffix)).To(Equal(expected))
+ },
+ Entry("ASCII", "abcdefghij", 5, "...", "ab..."),
+ Entry("Mixed ASCII and Unicode", "ab😀cd", 4, ".", "ab😀."),
+ Entry("All emoji", "😀😁😂😃😄", 3, "…", "😀😁…"),
+ Entry("Japanese", "こんにちは世界", 3, "…", "こん…"),
+ )
+ })
})
var testPaths = []string{