mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 13:07:36 +03:00
Another big refactor: Back to a single folder for persistence implementation
This commit is contained in:
parent
08e096c569
commit
a99c3a8af3
27 changed files with 177 additions and 171 deletions
2
.gitignore
vendored
2
.gitignore
vendored
|
@ -1,6 +1,5 @@
|
|||
/sonic-server
|
||||
/iTunes*.xml
|
||||
devDb*
|
||||
/tmp
|
||||
vendor/*/
|
||||
wiki
|
||||
|
@ -11,3 +10,4 @@ sonic.toml
|
|||
master.zip
|
||||
Jamstash-master
|
||||
storm.db
|
||||
sonic.db
|
||||
|
|
|
@ -11,7 +11,7 @@ import (
|
|||
type sonic struct {
|
||||
Port string `default:"4533"`
|
||||
MusicFolder string `default:"./iTunes1.xml"`
|
||||
DbPath string `default:"./devDb"`
|
||||
DbPath string `default:"./sonic.db"`
|
||||
|
||||
IgnoredArticles string `default:"The El La Los Las Le Les Os As O A"`
|
||||
IndexGroups string `default:"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]([)"`
|
||||
|
|
|
@ -32,7 +32,7 @@ type AlbumRepository interface {
|
|||
Get(id string) (*Album, error)
|
||||
FindByArtist(artistId string) (Albums, error)
|
||||
GetAll(...QueryOptions) (Albums, error)
|
||||
PurgeInactive(active Albums) ([]string, error)
|
||||
PurgeInactive(active Albums) error
|
||||
GetAllIds() ([]string, error)
|
||||
GetStarred(...QueryOptions) (Albums, error)
|
||||
Search(q string, offset int, size int) (Albums, error)
|
||||
|
|
|
@ -10,7 +10,7 @@ type ArtistRepository interface {
|
|||
BaseRepository
|
||||
Put(m *Artist) error
|
||||
Get(id string) (*Artist, error)
|
||||
PurgeInactive(active Artists) ([]string, error)
|
||||
PurgeInactive(active Artists) error
|
||||
Search(q string, offset int, size int) (Artists, error)
|
||||
}
|
||||
|
||||
|
|
|
@ -51,7 +51,7 @@ type MediaFileRepository interface {
|
|||
Get(id string) (*MediaFile, error)
|
||||
FindByAlbum(albumId string) (MediaFiles, error)
|
||||
GetStarred(options ...QueryOptions) (MediaFiles, error)
|
||||
PurgeInactive(active MediaFiles) ([]string, error)
|
||||
PurgeInactive(active MediaFiles) error
|
||||
GetAllIds() ([]string, error)
|
||||
Search(q string, offset int, size int) (MediaFiles, error)
|
||||
}
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"time"
|
||||
|
@ -30,7 +30,7 @@ type Album struct {
|
|||
}
|
||||
|
||||
type albumRepository struct {
|
||||
sqlRepository
|
||||
searchableRepository
|
||||
}
|
||||
|
||||
func NewAlbumRepository() domain.AlbumRepository {
|
||||
|
@ -41,12 +41,8 @@ func NewAlbumRepository() domain.AlbumRepository {
|
|||
|
||||
func (r *albumRepository) Put(a *domain.Album) error {
|
||||
ta := Album(*a)
|
||||
return WithTx(func(o orm.Ormer) error {
|
||||
err := r.put(o, a.ID, &ta)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return r.searcher.Index(o, r.tableName, a.ID, a.Name)
|
||||
return withTx(func(o orm.Ormer) error {
|
||||
return r.put(o, a.ID, a.Name, &ta)
|
||||
})
|
||||
}
|
||||
|
||||
|
@ -89,10 +85,14 @@ func (r *albumRepository) toAlbums(all []Album) domain.Albums {
|
|||
return result
|
||||
}
|
||||
|
||||
func (r *albumRepository) PurgeInactive(activeList domain.Albums) ([]string, error) {
|
||||
return r.purgeInactive(activeList, func(item interface{}) string {
|
||||
// TODO Remove []string from return
|
||||
func (r *albumRepository) PurgeInactive(activeList domain.Albums) error {
|
||||
return withTx(func(o orm.Ormer) error {
|
||||
_, err := r.purgeInactive(o, activeList, func(item interface{}) string {
|
||||
return item.(domain.Album).ID
|
||||
})
|
||||
return err
|
||||
})
|
||||
}
|
||||
|
||||
func (r *albumRepository) GetStarred(options ...domain.QueryOptions) (domain.Albums, error) {
|
||||
|
@ -110,7 +110,7 @@ func (r *albumRepository) Search(q string, offset int, size int) (domain.Albums,
|
|||
}
|
||||
|
||||
var results []Album
|
||||
err := r.searcher.Search(r.tableName, q, offset, size, &results, "rating desc", "starred desc", "play_count desc", "name")
|
||||
err := r.doSearch(r.tableName, q, offset, size, &results, "rating desc", "starred desc", "play_count desc", "name")
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
|
@ -1,4 +1,4 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"github.com/cloudsonic/sonic-server/domain"
|
|
@ -1,4 +1,4 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"github.com/astaxie/beego/orm"
|
||||
|
@ -13,7 +13,7 @@ type Artist struct {
|
|||
}
|
||||
|
||||
type artistRepository struct {
|
||||
sqlRepository
|
||||
searchableRepository
|
||||
}
|
||||
|
||||
func NewArtistRepository() domain.ArtistRepository {
|
||||
|
@ -24,12 +24,8 @@ func NewArtistRepository() domain.ArtistRepository {
|
|||
|
||||
func (r *artistRepository) Put(a *domain.Artist) error {
|
||||
ta := Artist(*a)
|
||||
return WithTx(func(o orm.Ormer) error {
|
||||
err := r.put(o, a.ID, &ta)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return r.searcher.Index(o, r.tableName, a.ID, a.Name)
|
||||
return withTx(func(o orm.Ormer) error {
|
||||
return r.put(o, a.ID, a.Name, &ta)
|
||||
})
|
||||
}
|
||||
|
||||
|
@ -46,10 +42,13 @@ func (r *artistRepository) Get(id string) (*domain.Artist, error) {
|
|||
return &a, nil
|
||||
}
|
||||
|
||||
func (r *artistRepository) PurgeInactive(activeList domain.Artists) ([]string, error) {
|
||||
return r.purgeInactive(activeList, func(item interface{}) string {
|
||||
func (r *artistRepository) PurgeInactive(activeList domain.Artists) error {
|
||||
return withTx(func(o orm.Ormer) error {
|
||||
_, err := r.purgeInactive(o, activeList, func(item interface{}) string {
|
||||
return item.(domain.Artist).ID
|
||||
})
|
||||
return err
|
||||
})
|
||||
}
|
||||
|
||||
func (r *artistRepository) Search(q string, offset int, size int) (domain.Artists, error) {
|
||||
|
@ -58,7 +57,7 @@ func (r *artistRepository) Search(q string, offset int, size int) (domain.Artist
|
|||
}
|
||||
|
||||
var results []Artist
|
||||
err := r.searcher.Search(r.tableName, q, offset, size, &results, "name")
|
||||
err := r.doSearch(r.tableName, q, offset, size, &results, "name")
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
|
@ -1,4 +1,4 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"github.com/cloudsonic/sonic-server/domain"
|
||||
|
@ -36,14 +36,18 @@ var _ = Describe("ArtistRepository", func() {
|
|||
|
||||
It("purges inactive records", func() {
|
||||
active := domain.Artists{{ID: "1"}, {ID: "3"}}
|
||||
Expect(repo.PurgeInactive(active)).To(Equal([]string{"2"}))
|
||||
|
||||
Expect(repo.PurgeInactive(active)).To(BeNil())
|
||||
|
||||
Expect(repo.CountAll()).To(Equal(int64(2)))
|
||||
Expect(repo.Exists("2")).To(BeFalse())
|
||||
})
|
||||
|
||||
It("doesn't delete anything if all is active", func() {
|
||||
active := domain.Artists{{ID: "1"}, {ID: "2"}, {ID: "3"}}
|
||||
Expect(repo.PurgeInactive(active)).To(BeEmpty())
|
||||
|
||||
Expect(repo.PurgeInactive(active)).To(BeNil())
|
||||
|
||||
Expect(repo.CountAll()).To(Equal(int64(3)))
|
||||
Expect(repo.Exists("1")).To(BeTrue())
|
||||
})
|
|
@ -1,4 +1,4 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"github.com/astaxie/beego/orm"
|
||||
|
@ -51,7 +51,7 @@ func (r *checkSumRepository) Get(id string) (string, error) {
|
|||
}
|
||||
|
||||
func (r *checkSumRepository) SetData(newSums map[string]string) error {
|
||||
err := WithTx(func(o orm.Ormer) error {
|
||||
err := withTx(func(o orm.Ormer) error {
|
||||
_, err := Db().Raw("delete from checksum").Exec()
|
||||
if err != nil {
|
||||
return err
|
|
@ -1,4 +1,4 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"github.com/cloudsonic/sonic-server/scanner"
|
|
@ -1,48 +0,0 @@
|
|||
package db_sql
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/cloudsonic/sonic-server/conf"
|
||||
"github.com/cloudsonic/sonic-server/domain"
|
||||
"github.com/cloudsonic/sonic-server/log"
|
||||
. "github.com/onsi/ginkgo"
|
||||
. "github.com/onsi/gomega"
|
||||
)
|
||||
|
||||
func TestSQLitePersistence(t *testing.T) {
|
||||
log.SetLevel(log.LevelDebug)
|
||||
RegisterFailHandler(Fail)
|
||||
RunSpecs(t, "SQLite Persistence Suite")
|
||||
}
|
||||
|
||||
var testAlbums = domain.Albums{
|
||||
{ID: "1", Name: "Sgt Peppers", Artist: "The Beatles", ArtistID: "1"},
|
||||
{ID: "2", Name: "Abbey Road", Artist: "The Beatles", ArtistID: "1"},
|
||||
{ID: "3", Name: "Radioactivity", Artist: "Kraftwerk", ArtistID: "2", Starred: true},
|
||||
}
|
||||
var testArtists = domain.Artists{
|
||||
{ID: "1", Name: "Saara Saara", AlbumCount: 2},
|
||||
{ID: "2", Name: "Kraftwerk"},
|
||||
{ID: "3", Name: "The Beatles"},
|
||||
}
|
||||
|
||||
var _ = Describe("Initialize test DB", func() {
|
||||
BeforeSuite(func() {
|
||||
//conf.Sonic.DbPath, _ = ioutil.TempDir("", "cloudsonic_tests")
|
||||
//os.MkdirAll(conf.Sonic.DbPath, 0700)
|
||||
conf.Sonic.DbPath = ":memory:"
|
||||
Db()
|
||||
artistRepo := NewArtistRepository()
|
||||
for _, a := range testArtists {
|
||||
artistRepo.Put(&a)
|
||||
}
|
||||
albumRepository := NewAlbumRepository()
|
||||
for _, a := range testAlbums {
|
||||
err := albumRepository.Put(&a)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
}
|
||||
})
|
||||
})
|
|
@ -1,4 +1,4 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"sort"
|
||||
|
@ -35,7 +35,7 @@ func (r *artistIndexRepository) CountAll() (int64, error) {
|
|||
}
|
||||
|
||||
func (r *artistIndexRepository) Put(idx *domain.ArtistIndex) error {
|
||||
return WithTx(func(o orm.Ormer) error {
|
||||
return withTx(func(o orm.Ormer) error {
|
||||
_, err := r.newQuery(o).Filter("idx", idx.ID).Delete()
|
||||
if err != nil {
|
||||
return err
|
|
@ -1,4 +1,4 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"github.com/cloudsonic/sonic-server/domain"
|
|
@ -1,4 +1,4 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"time"
|
||||
|
@ -36,7 +36,7 @@ type MediaFile struct {
|
|||
}
|
||||
|
||||
type mediaFileRepository struct {
|
||||
sqlRepository
|
||||
searchableRepository
|
||||
}
|
||||
|
||||
func NewMediaFileRepository() domain.MediaFileRepository {
|
||||
|
@ -47,12 +47,8 @@ func NewMediaFileRepository() domain.MediaFileRepository {
|
|||
|
||||
func (r *mediaFileRepository) Put(m *domain.MediaFile) error {
|
||||
tm := MediaFile(*m)
|
||||
return WithTx(func(o orm.Ormer) error {
|
||||
err := r.put(o, m.ID, &tm)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return r.searcher.Index(o, r.tableName, m.ID, m.Title)
|
||||
return withTx(func(o orm.Ormer) error {
|
||||
return r.put(o, m.ID, m.Title, &tm)
|
||||
})
|
||||
}
|
||||
|
||||
|
@ -95,10 +91,13 @@ func (r *mediaFileRepository) GetStarred(options ...domain.QueryOptions) (domain
|
|||
return r.toMediaFiles(starred), nil
|
||||
}
|
||||
|
||||
func (r *mediaFileRepository) PurgeInactive(activeList domain.MediaFiles) ([]string, error) {
|
||||
return r.purgeInactive(activeList, func(item interface{}) string {
|
||||
func (r *mediaFileRepository) PurgeInactive(activeList domain.MediaFiles) error {
|
||||
return withTx(func(o orm.Ormer) error {
|
||||
_, err := r.purgeInactive(o, activeList, func(item interface{}) string {
|
||||
return item.(domain.MediaFile).ID
|
||||
})
|
||||
return err
|
||||
})
|
||||
}
|
||||
|
||||
func (r *mediaFileRepository) Search(q string, offset int, size int) (domain.MediaFiles, error) {
|
||||
|
@ -107,7 +106,7 @@ func (r *mediaFileRepository) Search(q string, offset int, size int) (domain.Med
|
|||
}
|
||||
|
||||
var results []MediaFile
|
||||
err := r.searcher.Search(r.tableName, q, offset, size, &results, "rating desc", "starred desc", "play_count desc", "title")
|
||||
err := r.doSearch(r.tableName, q, offset, size, &results, "rating desc", "starred desc", "play_count desc", "title")
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
|
@ -2,9 +2,7 @@ package persistence
|
|||
|
||||
import "reflect"
|
||||
|
||||
type ProviderIdentifier string
|
||||
|
||||
func CollectValue(collection interface{}, getValue func(item interface{}) string) []string {
|
||||
func collectField(collection interface{}, getValue func(item interface{}) string) []string {
|
||||
s := reflect.ValueOf(collection)
|
||||
result := make([]string, s.Len())
|
||||
|
||||
|
|
|
@ -3,12 +3,46 @@ package persistence
|
|||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/cloudsonic/sonic-server/conf"
|
||||
"github.com/cloudsonic/sonic-server/domain"
|
||||
"github.com/cloudsonic/sonic-server/log"
|
||||
. "github.com/onsi/ginkgo"
|
||||
. "github.com/onsi/gomega"
|
||||
)
|
||||
|
||||
func TestPersistence(t *testing.T) {
|
||||
//log.SetLevel(log.LevelCritical)
|
||||
log.SetLevel(log.LevelCritical)
|
||||
RegisterFailHandler(Fail)
|
||||
RunSpecs(t, "Common Persistence Suite")
|
||||
RunSpecs(t, "Persistence Suite")
|
||||
}
|
||||
|
||||
var testAlbums = domain.Albums{
|
||||
{ID: "1", Name: "Sgt Peppers", Artist: "The Beatles", ArtistID: "1"},
|
||||
{ID: "2", Name: "Abbey Road", Artist: "The Beatles", ArtistID: "1"},
|
||||
{ID: "3", Name: "Radioactivity", Artist: "Kraftwerk", ArtistID: "2", Starred: true},
|
||||
}
|
||||
var testArtists = domain.Artists{
|
||||
{ID: "1", Name: "Saara Saara", AlbumCount: 2},
|
||||
{ID: "2", Name: "Kraftwerk"},
|
||||
{ID: "3", Name: "The Beatles"},
|
||||
}
|
||||
|
||||
var _ = Describe("Initialize test DB", func() {
|
||||
BeforeSuite(func() {
|
||||
//conf.Sonic.DbPath, _ = ioutil.TempDir("", "cloudsonic_tests")
|
||||
//os.MkdirAll(conf.Sonic.DbPath, 0700)
|
||||
conf.Sonic.DbPath = ":memory:"
|
||||
Db()
|
||||
artistRepo := NewArtistRepository()
|
||||
for _, a := range testArtists {
|
||||
artistRepo.Put(&a)
|
||||
}
|
||||
albumRepository := NewAlbumRepository()
|
||||
for _, a := range testAlbums {
|
||||
err := albumRepository.Put(&a)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
}
|
||||
})
|
||||
})
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"strings"
|
||||
|
@ -30,7 +30,7 @@ func NewPlaylistRepository() domain.PlaylistRepository {
|
|||
|
||||
func (r *playlistRepository) Put(p *domain.Playlist) error {
|
||||
tp := r.fromDomain(p)
|
||||
return WithTx(func(o orm.Ormer) error {
|
||||
return withTx(func(o orm.Ormer) error {
|
||||
return r.put(o, p.ID, &tp)
|
||||
})
|
||||
}
|
||||
|
@ -66,9 +66,12 @@ func (r *playlistRepository) toPlaylists(all []Playlist) (domain.Playlists, erro
|
|||
}
|
||||
|
||||
func (r *playlistRepository) PurgeInactive(activeList domain.Playlists) ([]string, error) {
|
||||
return r.purgeInactive(activeList, func(item interface{}) string {
|
||||
return nil, withTx(func(o orm.Ormer) error {
|
||||
_, err := r.purgeInactive(o, activeList, func(item interface{}) string {
|
||||
return item.(domain.Playlist).ID
|
||||
})
|
||||
return err
|
||||
})
|
||||
}
|
||||
|
||||
func (r *playlistRepository) toDomain(p *Playlist) domain.Playlist {
|
|
@ -1,4 +1,4 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"github.com/astaxie/beego/orm"
|
|
@ -1,4 +1,4 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"github.com/cloudsonic/sonic-server/domain"
|
|
@ -1,4 +1,4 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"strings"
|
||||
|
@ -15,9 +15,45 @@ type Search struct {
|
|||
FullText string `orm:"type(text)"`
|
||||
}
|
||||
|
||||
type sqlSearcher struct{}
|
||||
type searchableRepository struct {
|
||||
sqlRepository
|
||||
}
|
||||
|
||||
func (s *sqlSearcher) Index(o orm.Ormer, table, id, text string) error {
|
||||
func (r *searchableRepository) DeleteAll() error {
|
||||
return withTx(func(o orm.Ormer) error {
|
||||
_, err := r.newQuery(Db()).Filter("id__isnull", false).Delete()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return r.removeAllFromIndex(o, r.tableName)
|
||||
})
|
||||
}
|
||||
|
||||
func (r *searchableRepository) put(o orm.Ormer, id string, textToIndex string, a interface{}) error {
|
||||
c, err := r.newQuery(o).Filter("id", id).Count()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if c == 0 {
|
||||
_, err = o.Insert(a)
|
||||
} else {
|
||||
_, err = o.Update(a)
|
||||
}
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return r.addToIndex(o, r.tableName, id, textToIndex)
|
||||
}
|
||||
|
||||
func (r *searchableRepository) purgeInactive(o orm.Ormer, activeList interface{}, getId func(item interface{}) string) ([]string, error) {
|
||||
idsToDelete, err := r.sqlRepository.purgeInactive(o, activeList, getId)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return idsToDelete, r.removeFromIndex(o, r.tableName, idsToDelete)
|
||||
}
|
||||
|
||||
func (r *searchableRepository) addToIndex(o orm.Ormer, table, id, text string) error {
|
||||
item := Search{ID: id, Table: table}
|
||||
err := o.Read(&item)
|
||||
if err != nil && err != orm.ErrNoRows {
|
||||
|
@ -33,7 +69,7 @@ func (s *sqlSearcher) Index(o orm.Ormer, table, id, text string) error {
|
|||
return err
|
||||
}
|
||||
|
||||
func (s *sqlSearcher) Remove(o orm.Ormer, table string, ids []string) error {
|
||||
func (r *searchableRepository) removeFromIndex(o orm.Ormer, table string, ids []string) error {
|
||||
var offset int
|
||||
for {
|
||||
var subset = paginateSlice(ids, offset, batchSize)
|
||||
|
@ -50,12 +86,12 @@ func (s *sqlSearcher) Remove(o orm.Ormer, table string, ids []string) error {
|
|||
return nil
|
||||
}
|
||||
|
||||
func (s *sqlSearcher) DeleteAll(o orm.Ormer, table string) error {
|
||||
func (r *searchableRepository) removeAllFromIndex(o orm.Ormer, table string) error {
|
||||
_, err := o.QueryTable(&Search{}).Filter("table", table).Delete()
|
||||
return err
|
||||
}
|
||||
|
||||
func (s *sqlSearcher) Search(table string, q string, offset, size int, results interface{}, orderBys ...string) error {
|
||||
func (r *searchableRepository) doSearch(table string, q string, offset, size int, results interface{}, orderBys ...string) error {
|
||||
q = strings.TrimSpace(sanitize.Accents(strings.ToLower(strings.TrimSuffix(q, "*"))))
|
||||
if len(q) <= 2 {
|
||||
return nil
|
|
@ -1,8 +1,6 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"os"
|
||||
"path"
|
||||
"sync"
|
||||
|
||||
"github.com/astaxie/beego/orm"
|
||||
|
@ -20,12 +18,6 @@ func Db() orm.Ormer {
|
|||
dbPath := conf.Sonic.DbPath
|
||||
if dbPath == ":memory:" {
|
||||
dbPath = "file::memory:?cache=shared"
|
||||
} else {
|
||||
err := os.MkdirAll(conf.Sonic.DbPath, 0700)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
dbPath = path.Join(conf.Sonic.DbPath, "sqlite.db")
|
||||
}
|
||||
err := initORM(dbPath)
|
||||
if err != nil {
|
||||
|
@ -36,7 +28,7 @@ func Db() orm.Ormer {
|
|||
return orm.NewOrm()
|
||||
}
|
||||
|
||||
func WithTx(block func(orm.Ormer) error) error {
|
||||
func withTx(block func(orm.Ormer) error) error {
|
||||
o := orm.NewOrm()
|
||||
err := o.Begin()
|
||||
if err != nil {
|
|
@ -1,15 +1,13 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"github.com/astaxie/beego/orm"
|
||||
"github.com/cloudsonic/sonic-server/domain"
|
||||
"github.com/cloudsonic/sonic-server/log"
|
||||
"github.com/cloudsonic/sonic-server/persistence"
|
||||
)
|
||||
|
||||
type sqlRepository struct {
|
||||
tableName string
|
||||
searcher sqlSearcher
|
||||
}
|
||||
|
||||
func (r *sqlRepository) newQuery(o orm.Ormer, options ...domain.QueryOptions) orm.QuerySeter {
|
||||
|
@ -49,7 +47,7 @@ func (r *sqlRepository) GetAllIds() ([]string, error) {
|
|||
return nil, err
|
||||
}
|
||||
|
||||
result := persistence.CollectValue(values, func(item interface{}) string {
|
||||
result := collectField(values, func(item interface{}) string {
|
||||
return item.(orm.Params)["ID"].(string)
|
||||
})
|
||||
|
||||
|
@ -103,28 +101,24 @@ func difference(slice1 []string, slice2 []string) []string {
|
|||
}
|
||||
|
||||
func (r *sqlRepository) DeleteAll() error {
|
||||
return WithTx(func(o orm.Ormer) error {
|
||||
return withTx(func(o orm.Ormer) error {
|
||||
_, err := r.newQuery(Db()).Filter("id__isnull", false).Delete()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return r.searcher.DeleteAll(o, r.tableName)
|
||||
})
|
||||
}
|
||||
|
||||
func (r *sqlRepository) purgeInactive(activeList interface{}, getId func(item interface{}) string) ([]string, error) {
|
||||
func (r *sqlRepository) purgeInactive(o orm.Ormer, activeList interface{}, getId func(item interface{}) string) ([]string, error) {
|
||||
allIds, err := r.GetAllIds()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
activeIds := persistence.CollectValue(activeList, getId)
|
||||
activeIds := collectField(activeList, getId)
|
||||
idsToDelete := difference(allIds, activeIds)
|
||||
if len(idsToDelete) == 0 {
|
||||
return nil, nil
|
||||
}
|
||||
log.Debug("Purging inactive records", "table", r.tableName, "total", len(idsToDelete))
|
||||
|
||||
err = WithTx(func(o orm.Ormer) error {
|
||||
var offset int
|
||||
for {
|
||||
var subset = paginateSlice(idsToDelete, offset, batchSize)
|
||||
|
@ -135,10 +129,8 @@ func (r *sqlRepository) purgeInactive(activeList interface{}, getId func(item in
|
|||
offset += len(subset)
|
||||
_, err := r.newQuery(o).Filter("id__in", subset).Delete()
|
||||
if err != nil {
|
||||
return err
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
return r.searcher.Remove(o, r.tableName, idsToDelete)
|
||||
})
|
||||
return idsToDelete, err
|
||||
return idsToDelete, nil
|
||||
}
|
|
@ -1,7 +1,6 @@
|
|||
package db_sql
|
||||
package persistence
|
||||
|
||||
import (
|
||||
"github.com/cloudsonic/sonic-server/persistence"
|
||||
"github.com/google/wire"
|
||||
)
|
||||
|
||||
|
@ -13,7 +12,6 @@ var Set = wire.NewSet(
|
|||
NewCheckSumRepository,
|
||||
NewPropertyRepository,
|
||||
NewPlaylistRepository,
|
||||
persistence.NewNowPlayingRepository,
|
||||
persistence.NewMediaFolderRepository,
|
||||
wire.Value(persistence.ProviderIdentifier("sql")),
|
||||
NewNowPlayingRepository,
|
||||
NewMediaFolderRepository,
|
||||
)
|
|
@ -134,13 +134,13 @@ func (i *Importer) importLibrary() (err error) {
|
|||
i.importArtistIndex()
|
||||
|
||||
log.Debug("Purging old data")
|
||||
if _, err := i.mfRepo.PurgeInactive(mfs); err != nil {
|
||||
if err := i.mfRepo.PurgeInactive(mfs); err != nil {
|
||||
log.Error(err)
|
||||
}
|
||||
if _, err := i.albumRepo.PurgeInactive(als); err != nil {
|
||||
if err := i.albumRepo.PurgeInactive(als); err != nil {
|
||||
log.Error(err)
|
||||
}
|
||||
if _, err := i.artistRepo.PurgeInactive(ars); err != nil {
|
||||
if err := i.artistRepo.PurgeInactive(ars); err != nil {
|
||||
log.Error("Deleting inactive artists", err)
|
||||
}
|
||||
if _, err := i.plsRepo.PurgeInactive(pls); err != nil {
|
||||
|
|
15
wire_gen.go
15
wire_gen.go
|
@ -11,7 +11,6 @@ import (
|
|||
"github.com/cloudsonic/sonic-server/engine"
|
||||
"github.com/cloudsonic/sonic-server/itunesbridge"
|
||||
"github.com/cloudsonic/sonic-server/persistence"
|
||||
"github.com/cloudsonic/sonic-server/persistence/db_sql"
|
||||
"github.com/cloudsonic/sonic-server/scanner"
|
||||
"github.com/cloudsonic/sonic-server/server"
|
||||
"github.com/google/wire"
|
||||
|
@ -57,15 +56,15 @@ func CreateSubsonicAPIRouter() *api.Router {
|
|||
}
|
||||
|
||||
func createPersistenceProvider() *Provider {
|
||||
albumRepository := db_sql.NewAlbumRepository()
|
||||
artistRepository := db_sql.NewArtistRepository()
|
||||
checkSumRepository := db_sql.NewCheckSumRepository()
|
||||
artistIndexRepository := db_sql.NewArtistIndexRepository()
|
||||
mediaFileRepository := db_sql.NewMediaFileRepository()
|
||||
albumRepository := persistence.NewAlbumRepository()
|
||||
artistRepository := persistence.NewArtistRepository()
|
||||
checkSumRepository := persistence.NewCheckSumRepository()
|
||||
artistIndexRepository := persistence.NewArtistIndexRepository()
|
||||
mediaFileRepository := persistence.NewMediaFileRepository()
|
||||
mediaFolderRepository := persistence.NewMediaFolderRepository()
|
||||
nowPlayingRepository := persistence.NewNowPlayingRepository()
|
||||
playlistRepository := db_sql.NewPlaylistRepository()
|
||||
propertyRepository := db_sql.NewPropertyRepository()
|
||||
playlistRepository := persistence.NewPlaylistRepository()
|
||||
propertyRepository := persistence.NewPropertyRepository()
|
||||
provider := &Provider{
|
||||
AlbumRepository: albumRepository,
|
||||
ArtistRepository: artistRepository,
|
||||
|
|
|
@ -7,7 +7,7 @@ import (
|
|||
"github.com/cloudsonic/sonic-server/domain"
|
||||
"github.com/cloudsonic/sonic-server/engine"
|
||||
"github.com/cloudsonic/sonic-server/itunesbridge"
|
||||
"github.com/cloudsonic/sonic-server/persistence/db_sql"
|
||||
"github.com/cloudsonic/sonic-server/persistence"
|
||||
"github.com/cloudsonic/sonic-server/scanner"
|
||||
"github.com/cloudsonic/sonic-server/server"
|
||||
"github.com/google/wire"
|
||||
|
@ -51,7 +51,7 @@ func CreateSubsonicAPIRouter() *api.Router {
|
|||
// to conditionally select which function to use
|
||||
func createPersistenceProvider() *Provider {
|
||||
panic(wire.Build(
|
||||
db_sql.Set,
|
||||
persistence.Set,
|
||||
wire.Struct(new(Provider), "*"),
|
||||
))
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue