mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 13:07:36 +03:00
Import comments and lyrics
This commit is contained in:
parent
aee4eb71c4
commit
98af68ac99
7 changed files with 64 additions and 1 deletions
32
db/migration/20201110205344_add_comments_and_lyrics.go
Normal file
32
db/migration/20201110205344_add_comments_and_lyrics.go
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
package migration
|
||||||
|
|
||||||
|
import (
|
||||||
|
"database/sql"
|
||||||
|
|
||||||
|
"github.com/pressly/goose"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
goose.AddMigration(Up20201110205344, Down20201110205344)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Up20201110205344(tx *sql.Tx) error {
|
||||||
|
_, err := tx.Exec(`
|
||||||
|
alter table media_file
|
||||||
|
add comment varchar;
|
||||||
|
alter table media_file
|
||||||
|
add lyrics varchar;
|
||||||
|
|
||||||
|
alter table album
|
||||||
|
add comment varchar;
|
||||||
|
`)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
notice(tx, "A full rescan will be performed to import comments and lyrics")
|
||||||
|
return forceFullRescan(tx)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Down20201110205344(tx *sql.Tx) error {
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -16,6 +16,7 @@ type Album struct {
|
||||||
MaxYear int `json:"maxYear"`
|
MaxYear int `json:"maxYear"`
|
||||||
MinYear int `json:"minYear"`
|
MinYear int `json:"minYear"`
|
||||||
Compilation bool `json:"compilation"`
|
Compilation bool `json:"compilation"`
|
||||||
|
Comment string `json:"comment"`
|
||||||
SongCount int `json:"songCount"`
|
SongCount int `json:"songCount"`
|
||||||
Duration float32 `json:"duration"`
|
Duration float32 `json:"duration"`
|
||||||
Genre string `json:"genre"`
|
Genre string `json:"genre"`
|
||||||
|
|
|
@ -37,6 +37,8 @@ type MediaFile struct {
|
||||||
OrderArtistName string `json:"orderArtistName"`
|
OrderArtistName string `json:"orderArtistName"`
|
||||||
OrderAlbumArtistName string `json:"orderAlbumArtistName"`
|
OrderAlbumArtistName string `json:"orderAlbumArtistName"`
|
||||||
Compilation bool `json:"compilation"`
|
Compilation bool `json:"compilation"`
|
||||||
|
Comment string `json:"comment"`
|
||||||
|
Lyrics string `json:"lyrics"`
|
||||||
CatalogNum string `json:"catalogNum"`
|
CatalogNum string `json:"catalogNum"`
|
||||||
MbzTrackID string `json:"mbzTrackId" orm:"column(mbz_track_id)"`
|
MbzTrackID string `json:"mbzTrackId" orm:"column(mbz_track_id)"`
|
||||||
MbzAlbumID string `json:"mbzAlbumId" orm:"column(mbz_album_id)"`
|
MbzAlbumID string `json:"mbzAlbumId" orm:"column(mbz_album_id)"`
|
||||||
|
|
|
@ -155,12 +155,14 @@ func (r *albumRepository) refresh(ids ...string) error {
|
||||||
SongArtists string
|
SongArtists string
|
||||||
Years string
|
Years string
|
||||||
DiscSubtitles string
|
DiscSubtitles string
|
||||||
|
Comments string
|
||||||
Path string
|
Path string
|
||||||
}
|
}
|
||||||
var albums []refreshAlbum
|
var albums []refreshAlbum
|
||||||
|
const zwsp = string('\u200b')
|
||||||
sel := Select(`f.album_id as id, f.album as name, f.artist, f.album_artist, f.artist_id, f.album_artist_id,
|
sel := Select(`f.album_id as id, f.album as name, f.artist, f.album_artist, f.artist_id, f.album_artist_id,
|
||||||
f.sort_album_name, f.sort_artist_name, f.sort_album_artist_name,
|
f.sort_album_name, f.sort_artist_name, f.sort_album_artist_name,
|
||||||
f.order_album_name, f.order_album_artist_name, f.path,
|
f.order_album_name, f.order_album_artist_name, f.path, group_concat(f.comment, "` + zwsp + `") as comments,
|
||||||
group_concat(f.mbz_album_id, ' ') as mbz_album_id, f.mbz_album_artist_id, f.mbz_album_type, f.mbz_album_comment,
|
group_concat(f.mbz_album_id, ' ') as mbz_album_id, f.mbz_album_artist_id, f.mbz_album_type, f.mbz_album_comment,
|
||||||
f.catalog_num, f.compilation, f.genre, max(f.year) as max_year, sum(f.duration) as duration,
|
f.catalog_num, f.compilation, f.genre, max(f.year) as max_year, sum(f.duration) as duration,
|
||||||
count(f.id) as song_count, a.id as current_id,
|
count(f.id) as song_count, a.id as current_id,
|
||||||
|
@ -212,6 +214,7 @@ func (r *albumRepository) refresh(ids ...string) error {
|
||||||
}
|
}
|
||||||
al.MinYear = getMinYear(al.Years)
|
al.MinYear = getMinYear(al.Years)
|
||||||
al.MbzAlbumID = getMbzId(r.ctx, al.MbzAlbumID, r.tableName, al.Name)
|
al.MbzAlbumID = getMbzId(r.ctx, al.MbzAlbumID, r.tableName, al.Name)
|
||||||
|
al.Comment = getComment(al.Comments, zwsp)
|
||||||
al.UpdatedAt = time.Now()
|
al.UpdatedAt = time.Now()
|
||||||
if al.CurrentId != "" {
|
if al.CurrentId != "" {
|
||||||
toUpdate++
|
toUpdate++
|
||||||
|
@ -235,6 +238,17 @@ func (r *albumRepository) refresh(ids ...string) error {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Return the first non empty comment, if any
|
||||||
|
func getComment(comments string, separator string) string {
|
||||||
|
cs := strings.Split(comments, separator)
|
||||||
|
for _, c := range cs {
|
||||||
|
if c != "" {
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
func getMinYear(years string) int {
|
func getMinYear(years string) int {
|
||||||
ys := strings.Fields(years)
|
ys := strings.Fields(years)
|
||||||
sort.Strings(ys)
|
sort.Strings(ys)
|
||||||
|
|
|
@ -91,6 +91,16 @@ var _ = Describe("AlbumRepository", func() {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
Describe("getComment", func() {
|
||||||
|
const zwsp = string('\u200b')
|
||||||
|
It("returns empty string if there are no comments", func() {
|
||||||
|
Expect(getComment("", "")).To(Equal(""))
|
||||||
|
})
|
||||||
|
It("returns first occurrence of non-empty comment", func() {
|
||||||
|
Expect(getComment(zwsp+zwsp+"first"+zwsp+"second", zwsp)).To(Equal("first"))
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
Describe("getCoverFromPath", func() {
|
Describe("getCoverFromPath", func() {
|
||||||
testFolder, _ := ioutil.TempDir("", "album_persistence_tests")
|
testFolder, _ := ioutil.TempDir("", "album_persistence_tests")
|
||||||
if err := os.MkdirAll(testFolder, 0777); err != nil {
|
if err := os.MkdirAll(testFolder, 0777); err != nil {
|
||||||
|
|
|
@ -59,6 +59,8 @@ func (s *mediaFileMapper) toMediaFile(md metadata.Metadata) model.MediaFile {
|
||||||
mf.MbzAlbumArtistID = md.MbzAlbumArtistID()
|
mf.MbzAlbumArtistID = md.MbzAlbumArtistID()
|
||||||
mf.MbzAlbumType = md.MbzAlbumType()
|
mf.MbzAlbumType = md.MbzAlbumType()
|
||||||
mf.MbzAlbumComment = md.MbzAlbumComment()
|
mf.MbzAlbumComment = md.MbzAlbumComment()
|
||||||
|
mf.Comment = md.Comment()
|
||||||
|
mf.Lyrics = md.Lyrics()
|
||||||
|
|
||||||
// TODO Get Creation time. https://github.com/djherbis/times ?
|
// TODO Get Creation time. https://github.com/djherbis/times ?
|
||||||
mf.CreatedAt = md.ModificationTime()
|
mf.CreatedAt = md.ModificationTime()
|
||||||
|
|
|
@ -51,6 +51,7 @@ type Metadata interface {
|
||||||
DiscSubtitle() string
|
DiscSubtitle() string
|
||||||
HasPicture() bool
|
HasPicture() bool
|
||||||
Comment() string
|
Comment() string
|
||||||
|
Lyrics() string
|
||||||
Compilation() bool
|
Compilation() bool
|
||||||
CatalogNum() string
|
CatalogNum() string
|
||||||
MbzTrackID() string
|
MbzTrackID() string
|
||||||
|
@ -89,6 +90,7 @@ func (m *baseMetadata) Composer() string { return m.getTag("composer", "t
|
||||||
func (m *baseMetadata) Genre() string { return m.getTag("genre") }
|
func (m *baseMetadata) Genre() string { return m.getTag("genre") }
|
||||||
func (m *baseMetadata) Year() int { return m.parseYear("date") }
|
func (m *baseMetadata) Year() int { return m.parseYear("date") }
|
||||||
func (m *baseMetadata) Comment() string { return m.getTag("comment") }
|
func (m *baseMetadata) Comment() string { return m.getTag("comment") }
|
||||||
|
func (m *baseMetadata) Lyrics() string { return m.getTag("lyrics", "lyrics-eng") }
|
||||||
func (m *baseMetadata) Compilation() bool { return m.parseBool("tcmp", "compilation") }
|
func (m *baseMetadata) Compilation() bool { return m.parseBool("tcmp", "compilation") }
|
||||||
func (m *baseMetadata) TrackNumber() (int, int) { return m.parseTuple("track", "tracknumber") }
|
func (m *baseMetadata) TrackNumber() (int, int) { return m.parseTuple("track", "tracknumber") }
|
||||||
func (m *baseMetadata) DiscNumber() (int, int) { return m.parseTuple("disc", "discnumber") }
|
func (m *baseMetadata) DiscNumber() (int, int) { return m.parseTuple("disc", "discnumber") }
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue