mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-05 05:27:37 +03:00
Using checksums to detect modified stats in the iTunes Library
This commit is contained in:
parent
bb5d4c920d
commit
12aedc0996
3 changed files with 103 additions and 9 deletions
|
@ -28,9 +28,11 @@ func init() {
|
||||||
utils.DefineSingleton(new(engine.Playlists), engine.NewPlaylists)
|
utils.DefineSingleton(new(engine.Playlists), engine.NewPlaylists)
|
||||||
utils.DefineSingleton(new(engine.Search), engine.NewSearch)
|
utils.DefineSingleton(new(engine.Search), engine.NewSearch)
|
||||||
|
|
||||||
|
utils.DefineSingleton(new(scanner.CheckSumRepository), persistence.NewCheckSumRepository)
|
||||||
|
utils.DefineSingleton(new(scanner.Scanner), scanner.NewItunesScanner)
|
||||||
|
|
||||||
// Other dependencies
|
// Other dependencies
|
||||||
utils.DefineSingleton(new(itunesbridge.ItunesControl), itunesbridge.NewItunesControl)
|
utils.DefineSingleton(new(itunesbridge.ItunesControl), itunesbridge.NewItunesControl)
|
||||||
utils.DefineSingleton(new(scanner.Scanner), scanner.NewItunesScanner)
|
|
||||||
utils.DefineSingleton(new(gomate.DB), func() gomate.DB {
|
utils.DefineSingleton(new(gomate.DB), func() gomate.DB {
|
||||||
return gomate.NewLedisEmbeddedDB(persistence.Db())
|
return gomate.NewLedisEmbeddedDB(persistence.Db())
|
||||||
})
|
})
|
||||||
|
|
61
persistence/checksum_repository.go
Normal file
61
persistence/checksum_repository.go
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
package persistence
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
|
||||||
|
"github.com/astaxie/beego"
|
||||||
|
"github.com/deluan/gosonic/scanner"
|
||||||
|
"github.com/siddontang/ledisdb/ledis"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
keyName = []byte("checksums")
|
||||||
|
)
|
||||||
|
|
||||||
|
type checkSumRepository struct {
|
||||||
|
data map[string]string
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewCheckSumRepository() scanner.CheckSumRepository {
|
||||||
|
r := &checkSumRepository{}
|
||||||
|
r.loadData()
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *checkSumRepository) loadData() {
|
||||||
|
r.data = make(map[string]string)
|
||||||
|
|
||||||
|
pairs, err := Db().HGetAll(keyName)
|
||||||
|
if err != nil {
|
||||||
|
beego.Error("Error loading CheckSums:", err)
|
||||||
|
}
|
||||||
|
for _, p := range pairs {
|
||||||
|
r.data[string(p.Field)] = string(p.Value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *checkSumRepository) Put(id, sum string) error {
|
||||||
|
if id == "" {
|
||||||
|
return errors.New("Id is required")
|
||||||
|
}
|
||||||
|
_, err := Db().HSet(keyName, []byte(id), []byte(sum))
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *checkSumRepository) Get(id string) (string, error) {
|
||||||
|
return r.data[id], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *checkSumRepository) SetData(newSums map[string]string) error {
|
||||||
|
Db().HClear(keyName)
|
||||||
|
pairs := make([]ledis.FVPair, len(newSums))
|
||||||
|
r.data = make(map[string]string)
|
||||||
|
i := 0
|
||||||
|
for id, sum := range newSums {
|
||||||
|
p := ledis.FVPair{Field: []byte(id), Value: []byte(sum)}
|
||||||
|
pairs[i] = p
|
||||||
|
r.data[id] = sum
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
return Db().HMset(keyName, pairs...)
|
||||||
|
}
|
|
@ -3,19 +3,16 @@ package scanner
|
||||||
import (
|
import (
|
||||||
"crypto/md5"
|
"crypto/md5"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"html"
|
||||||
|
"mime"
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"regexp"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"html"
|
|
||||||
|
|
||||||
"regexp"
|
|
||||||
|
|
||||||
"mime"
|
|
||||||
|
|
||||||
"github.com/astaxie/beego"
|
"github.com/astaxie/beego"
|
||||||
"github.com/deluan/gosonic/domain"
|
"github.com/deluan/gosonic/domain"
|
||||||
"github.com/deluan/itl"
|
"github.com/deluan/itl"
|
||||||
|
@ -29,10 +26,18 @@ type ItunesScanner struct {
|
||||||
playlists map[string]*domain.Playlist
|
playlists map[string]*domain.Playlist
|
||||||
pplaylists map[string]plsRelation
|
pplaylists map[string]plsRelation
|
||||||
lastModifiedSince time.Time
|
lastModifiedSince time.Time
|
||||||
|
checksumRepo CheckSumRepository
|
||||||
|
newSums map[string]string
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewItunesScanner() *ItunesScanner {
|
func NewItunesScanner(checksumRepo CheckSumRepository) *ItunesScanner {
|
||||||
return &ItunesScanner{}
|
return &ItunesScanner{checksumRepo: checksumRepo}
|
||||||
|
}
|
||||||
|
|
||||||
|
type CheckSumRepository interface {
|
||||||
|
Put(id, sum string) error
|
||||||
|
Get(id string) (string, error)
|
||||||
|
SetData(newSums map[string]string) error
|
||||||
}
|
}
|
||||||
|
|
||||||
type plsRelation struct {
|
type plsRelation struct {
|
||||||
|
@ -57,6 +62,7 @@ func (s *ItunesScanner) ScanLibrary(lastModifiedSince time.Time, path string) (i
|
||||||
s.artists = make(map[string]*domain.Artist)
|
s.artists = make(map[string]*domain.Artist)
|
||||||
s.playlists = make(map[string]*domain.Playlist)
|
s.playlists = make(map[string]*domain.Playlist)
|
||||||
s.pplaylists = make(map[string]plsRelation)
|
s.pplaylists = make(map[string]plsRelation)
|
||||||
|
s.newSums = make(map[string]string)
|
||||||
|
|
||||||
i := 0
|
i := 0
|
||||||
for _, t := range l.Tracks {
|
for _, t := range l.Tracks {
|
||||||
|
@ -71,6 +77,12 @@ func (s *ItunesScanner) ScanLibrary(lastModifiedSince time.Time, path string) (i
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if err := s.checksumRepo.SetData(s.newSums); err != nil {
|
||||||
|
beego.Error("Error saving checksums:", err)
|
||||||
|
} else {
|
||||||
|
beego.Debug("Saved", len(s.newSums), "checksums")
|
||||||
|
}
|
||||||
|
|
||||||
ignFolders, _ := beego.AppConfig.Bool("plsIgnoreFolders")
|
ignFolders, _ := beego.AppConfig.Bool("plsIgnoreFolders")
|
||||||
ignPatterns := beego.AppConfig.Strings("plsIgnoredPatterns")
|
ignPatterns := beego.AppConfig.Strings("plsIgnoredPatterns")
|
||||||
for _, p := range l.Playlists {
|
for _, p := range l.Playlists {
|
||||||
|
@ -158,6 +170,9 @@ func (s *ItunesScanner) fullPath(pID string) string {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *ItunesScanner) lastChangedDate(t *itl.Track) time.Time {
|
func (s *ItunesScanner) lastChangedDate(t *itl.Track) time.Time {
|
||||||
|
if s.hasChanged(t) {
|
||||||
|
return time.Now()
|
||||||
|
}
|
||||||
allDates := []time.Time{t.DateModified, t.PlayDateUTC}
|
allDates := []time.Time{t.DateModified, t.PlayDateUTC}
|
||||||
c := time.Time{}
|
c := time.Time{}
|
||||||
for _, d := range allDates {
|
for _, d := range allDates {
|
||||||
|
@ -247,6 +262,14 @@ func (s *ItunesScanner) collectAlbums(t *itl.Track, mf *domain.MediaFile, ar *do
|
||||||
return al
|
return al
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *ItunesScanner) hasChanged(t *itl.Track) bool {
|
||||||
|
id := strconv.Itoa(t.TrackID)
|
||||||
|
oldSum, _ := s.checksumRepo.Get(id)
|
||||||
|
newSum := calcCheckSum(t)
|
||||||
|
s.newSums[id] = newSum
|
||||||
|
return oldSum != newSum
|
||||||
|
}
|
||||||
|
|
||||||
func (s *ItunesScanner) collectArtists(t *itl.Track) *domain.Artist {
|
func (s *ItunesScanner) collectArtists(t *itl.Track) *domain.Artist {
|
||||||
id := artistId(t)
|
id := artistId(t)
|
||||||
_, found := s.artists[id]
|
_, found := s.artists[id]
|
||||||
|
@ -317,4 +340,12 @@ func realArtistName(t *itl.Track) string {
|
||||||
return t.Artist
|
return t.Artist
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Calc sum of stats fields (whose changes are not reflected in DataModified)
|
||||||
|
func calcCheckSum(t *itl.Track) string {
|
||||||
|
data := fmt.Sprint(t.DateModified, t.PlayCount, t.PlayDate, t.ArtworkCount, t.Loved, t.AlbumLoved,
|
||||||
|
t.Rating, t.AlbumRating, t.SkipCount, t.SkipDate)
|
||||||
|
return fmt.Sprintf("%x", md5.Sum([]byte(data)))
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
var _ Scanner = (*ItunesScanner)(nil)
|
var _ Scanner = (*ItunesScanner)(nil)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue