mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 13:07:36 +03:00
Remove non-album artist_ids from the DB
This commit is contained in:
parent
0d9dcebf32
commit
1c82bf5179
5 changed files with 53 additions and 1 deletions
22
db/migration/20211105162746_remove_invalid_artist_ids.go
Normal file
22
db/migration/20211105162746_remove_invalid_artist_ids.go
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
package migrations
|
||||||
|
|
||||||
|
import (
|
||||||
|
"database/sql"
|
||||||
|
|
||||||
|
"github.com/pressly/goose"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
goose.AddMigration(upRemoveInvalidArtistIds, downRemoveInvalidArtistIds)
|
||||||
|
}
|
||||||
|
|
||||||
|
func upRemoveInvalidArtistIds(tx *sql.Tx) error {
|
||||||
|
_, err := tx.Exec(`
|
||||||
|
update media_file set artist_id = '' where not exists(select 1 from artist where id = artist_id)
|
||||||
|
`)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func downRemoveInvalidArtistIds(tx *sql.Tx) error {
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -37,17 +37,25 @@ func toSnakeCase(str string) string {
|
||||||
}
|
}
|
||||||
|
|
||||||
func exists(subTable string, cond squirrel.Sqlizer) existsCond {
|
func exists(subTable string, cond squirrel.Sqlizer) existsCond {
|
||||||
return existsCond{subTable: subTable, cond: cond}
|
return existsCond{subTable: subTable, cond: cond, not: false}
|
||||||
|
}
|
||||||
|
|
||||||
|
func notExists(subTable string, cond squirrel.Sqlizer) existsCond {
|
||||||
|
return existsCond{subTable: subTable, cond: cond, not: true}
|
||||||
}
|
}
|
||||||
|
|
||||||
type existsCond struct {
|
type existsCond struct {
|
||||||
subTable string
|
subTable string
|
||||||
cond squirrel.Sqlizer
|
cond squirrel.Sqlizer
|
||||||
|
not bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e existsCond) ToSql() (string, []interface{}, error) {
|
func (e existsCond) ToSql() (string, []interface{}, error) {
|
||||||
sql, args, err := e.cond.ToSql()
|
sql, args, err := e.cond.ToSql()
|
||||||
sql = fmt.Sprintf("exists (select 1 from %s where %s)", e.subTable, sql)
|
sql = fmt.Sprintf("exists (select 1 from %s where %s)", e.subTable, sql)
|
||||||
|
if e.not {
|
||||||
|
sql = "not " + sql
|
||||||
|
}
|
||||||
return sql, args, err
|
return sql, args, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -59,6 +59,16 @@ var _ = Describe("Helpers", func() {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
Describe("notExists", func() {
|
||||||
|
It("constructs the correct NOT EXISTS query", func() {
|
||||||
|
e := notExists("artist", squirrel.ConcatExpr("id = artist_id"))
|
||||||
|
sql, args, err := e.ToSql()
|
||||||
|
Expect(sql).To(Equal("not exists (select 1 from artist where id = artist_id)"))
|
||||||
|
Expect(args).To(BeEmpty())
|
||||||
|
Expect(err).To(BeNil())
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
Describe("getMostFrequentMbzID", func() {
|
Describe("getMostFrequentMbzID", func() {
|
||||||
It(`returns "" when no ids are passed`, func() {
|
It(`returns "" when no ids are passed`, func() {
|
||||||
Expect(getMostFrequentMbzID(context.TODO(), " ", "", "")).To(Equal(""))
|
Expect(getMostFrequentMbzID(context.TODO(), " ", "", "")).To(Equal(""))
|
||||||
|
|
|
@ -177,6 +177,13 @@ func (r *mediaFileRepository) DeleteByPath(basePath string) (int64, error) {
|
||||||
return r.executeSQL(del)
|
return r.executeSQL(del)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *mediaFileRepository) removeNonAlbumArtistIds() error {
|
||||||
|
upd := Update(r.tableName).Set("artist_id", "").Where(notExists("artist", ConcatExpr("id = artist_id")))
|
||||||
|
log.Debug(r.ctx, "Removing non-album artist_id")
|
||||||
|
_, err := r.executeSQL(upd)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
func (r *mediaFileRepository) Search(q string, offset int, size int) (model.MediaFiles, error) {
|
func (r *mediaFileRepository) Search(q string, offset int, size int) (model.MediaFiles, error) {
|
||||||
results := model.MediaFiles{}
|
results := model.MediaFiles{}
|
||||||
err := r.doSearch(q, offset, size, &results, "title")
|
err := r.doSearch(q, offset, size, &results, "title")
|
||||||
|
|
|
@ -135,6 +135,11 @@ func (s *SQLStore) GC(ctx context.Context, rootFolder string) error {
|
||||||
log.Error(ctx, "Error removing dangling tracks", err)
|
log.Error(ctx, "Error removing dangling tracks", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
err = s.MediaFile(ctx).(*mediaFileRepository).removeNonAlbumArtistIds()
|
||||||
|
if err != nil {
|
||||||
|
log.Error(ctx, "Error removing non-album artist_ids", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
err = s.Album(ctx).(*albumRepository).purgeEmpty()
|
err = s.Album(ctx).(*albumRepository).purgeEmpty()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error(ctx, "Error removing empty albums", err)
|
log.Error(ctx, "Error removing empty albums", err)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue