Big LedisDB refactoring
This commit is contained in:
+1
-1
@@ -8,7 +8,7 @@ apiVersion = 1.0.0
|
|||||||
ignoredArticles="The El La Los Las Le Les Os As O A"
|
ignoredArticles="The El La Los Las Le Les Os As O A"
|
||||||
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)
|
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)
|
||||||
|
|
||||||
musicFolder=./iTunesFull.xml
|
musicFolder=./iTunes.xml
|
||||||
user=deluan
|
user=deluan
|
||||||
password=wordpass
|
password=wordpass
|
||||||
dbPath = ./devDb
|
dbPath = ./devDb
|
||||||
|
|||||||
+1
-24
@@ -3,32 +3,9 @@ package models
|
|||||||
type Album struct {
|
type Album struct {
|
||||||
Id string
|
Id string
|
||||||
Name string
|
Name string
|
||||||
ArtistId string
|
ArtistId string `parent:"artist"`
|
||||||
CoverArtPath string
|
CoverArtPath string
|
||||||
Year int
|
Year int
|
||||||
Compilation bool
|
Compilation bool
|
||||||
Rating int
|
Rating int
|
||||||
MediaFiles map[string]bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *Album) AdMediaFiles(mfs ...*MediaFile) {
|
|
||||||
for _, mf := range mfs {
|
|
||||||
a.MediaFiles[mf.Id] = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *Album) AddMediaFiles(mfs ...interface{}) {
|
|
||||||
if a.MediaFiles == nil {
|
|
||||||
a.MediaFiles = make(map[string]bool)
|
|
||||||
}
|
|
||||||
for _, v := range mfs {
|
|
||||||
switch v := v.(type) {
|
|
||||||
case *MediaFile:
|
|
||||||
a.MediaFiles[v.Id] = true
|
|
||||||
case map[string]bool:
|
|
||||||
for k, _ := range v {
|
|
||||||
a.MediaFiles[k] = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
@@ -20,20 +20,4 @@ func NoArticle(name string) string {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
return name
|
return name
|
||||||
}
|
|
||||||
|
|
||||||
func (a *Artist) AddAlbums(albums ...interface{}) {
|
|
||||||
if a.Albums == nil {
|
|
||||||
a.Albums = make(map[string]bool)
|
|
||||||
}
|
|
||||||
for _, v := range albums {
|
|
||||||
switch v := v.(type) {
|
|
||||||
case *Album:
|
|
||||||
a.Albums[v.Id] = true
|
|
||||||
case map[string]bool:
|
|
||||||
for k, _ := range v {
|
|
||||||
a.Albums[k] = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
@@ -11,6 +11,7 @@ type MediaFile struct {
|
|||||||
Album string
|
Album string
|
||||||
Artist string
|
Artist string
|
||||||
AlbumArtist string
|
AlbumArtist string
|
||||||
|
AlbumId string `parent:"album"`
|
||||||
Compilation bool
|
Compilation bool
|
||||||
CreatedAt time.Time
|
CreatedAt time.Time
|
||||||
UpdatedAt time.Time
|
UpdatedAt time.Time
|
||||||
|
|||||||
@@ -10,20 +10,20 @@ type Album struct {
|
|||||||
|
|
||||||
func NewAlbumRepository() *Album {
|
func NewAlbumRepository() *Album {
|
||||||
r := &Album{}
|
r := &Album{}
|
||||||
r.key = "album"
|
r.table = "album"
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *Album) Put(m *models.Album) (*models.Album, error) {
|
func (r *Album) Put(m *models.Album) error {
|
||||||
if m.Id == "" {
|
if m.Id == "" {
|
||||||
m.Id = r.NewId(m.Name)
|
m.Id = r.NewId(m.Name)
|
||||||
}
|
}
|
||||||
return m, r.saveOrUpdate(m.Id, m)
|
return r.saveOrUpdate(m.Id, m)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *Album) Get(id string) (*models.Album, error) {
|
func (r *Album) Get(id string) (*models.Album, error) {
|
||||||
rec := &models.Album{}
|
rec := &models.Album{}
|
||||||
err := readStruct(r.key, id, rec)
|
err := r.loadEntity(id, rec)
|
||||||
return rec, err
|
return rec, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -10,20 +10,20 @@ type Artist struct {
|
|||||||
|
|
||||||
func NewArtistRepository() *Artist {
|
func NewArtistRepository() *Artist {
|
||||||
r := &Artist{}
|
r := &Artist{}
|
||||||
r.key = "artist"
|
r.table = "artist"
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *Artist) Put(m *models.Artist) (*models.Artist, error) { // TODO Return only error
|
func (r *Artist) Put(m *models.Artist) error {
|
||||||
if m.Id == "" {
|
if m.Id == "" {
|
||||||
m.Id = r.NewId(m.Name)
|
m.Id = r.NewId(m.Name)
|
||||||
}
|
}
|
||||||
return m, r.saveOrUpdate(m.Id, m)
|
return r.saveOrUpdate(m.Id, m)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *Artist) Get(id string) (*models.Artist, error) {
|
func (r *Artist) Get(id string) (*models.Artist, error) {
|
||||||
rec := &models.Artist{}
|
rec := &models.Artist{}
|
||||||
err := readStruct(r.key, id, rec)
|
err := r.loadEntity(id, rec)
|
||||||
return rec, err
|
return rec, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -4,26 +4,100 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"crypto/md5"
|
"crypto/md5"
|
||||||
"strings"
|
"strings"
|
||||||
|
"github.com/deluan/gosonic/utils"
|
||||||
|
"encoding/json"
|
||||||
|
"reflect"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
type BaseRepository struct {
|
type BaseRepository struct {
|
||||||
key string // TODO Rename to 'table'
|
table string
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *BaseRepository) NewId(fields ...string) string {
|
func (r *BaseRepository) NewId(fields ...string) string {
|
||||||
s := fmt.Sprintf("%s\\%s", strings.ToUpper(r.key), strings.Join(fields, ""))
|
s := fmt.Sprintf("%s\\%s", strings.ToUpper(r.table), strings.Join(fields, ""))
|
||||||
return fmt.Sprintf("%x", md5.Sum([]byte(s)))
|
return fmt.Sprintf("%x", md5.Sum([]byte(s)))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *BaseRepository) CountAll() (int, error) {
|
func (r *BaseRepository) CountAll() (int, error) {
|
||||||
return count(r.key)
|
ids, err := db().SMembers([]byte(r.table + "s:all"))
|
||||||
|
return len(ids), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *BaseRepository) saveOrUpdate(id string, rec interface{}) error {
|
func (r *BaseRepository) saveOrUpdate(id string, rec interface{}) error {
|
||||||
return saveStruct(r.key, id, rec)
|
return r.saveEntity(id, rec)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *BaseRepository) Dump() {
|
func (r *BaseRepository) Dump() {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *BaseRepository) saveEntity(id string, entity interface{}) error {
|
||||||
|
recordPrefix := fmt.Sprintf("%s:%s:", r.table, id)
|
||||||
|
allKey := r.table + "s:all"
|
||||||
|
|
||||||
|
h, err := utils.ToMap(entity)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for f, v := range h {
|
||||||
|
key := recordPrefix + f
|
||||||
|
value, _ := json.Marshal(v)
|
||||||
|
if err := db().Set([]byte(key), value); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err = db().SAdd([]byte(allKey), []byte(id)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if parentTable, parentId := r.getParent(entity); parentTable != "" {
|
||||||
|
parentCollectionKey := fmt.Sprintf("%s:%s:%ss", parentTable, parentId, r.table)
|
||||||
|
_, err = db().SAdd([]byte(parentCollectionKey), []byte(id))
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO Optimize
|
||||||
|
func (r *BaseRepository) getParent(entity interface{}) (table string, id string) {
|
||||||
|
dt := reflect.TypeOf(entity).Elem()
|
||||||
|
for i := 0; i < dt.NumField(); i++ {
|
||||||
|
f := dt.Field(i)
|
||||||
|
table := f.Tag.Get("parent")
|
||||||
|
if table != "" {
|
||||||
|
dv := reflect.ValueOf(entity).Elem()
|
||||||
|
return table, dv.FieldByName(f.Name).String()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return "", ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *BaseRepository) loadEntity(id string, entity interface{}) error {
|
||||||
|
recordPrefix := fmt.Sprintf("%s:%s:", r.table, id)
|
||||||
|
|
||||||
|
h, _ := utils.ToMap(entity)
|
||||||
|
var fieldKeys = make([][]byte, len(h))
|
||||||
|
var fieldNames = make([]string, len(h))
|
||||||
|
i := 0
|
||||||
|
for k, _ := range h {
|
||||||
|
fieldNames[i] = k
|
||||||
|
fieldKeys[i] = []byte(recordPrefix + k)
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
|
||||||
|
res, err := db().MGet(fieldKeys...)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
var record = make(map[string]interface{}, len(res))
|
||||||
|
for i, v := range res {
|
||||||
|
var value interface{}
|
||||||
|
if err := json.Unmarshal(v, &value); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
record[string(fieldNames[i])] = value
|
||||||
|
}
|
||||||
|
|
||||||
|
return utils.ToStruct(record, entity)
|
||||||
|
}
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ type ArtistIndex struct {
|
|||||||
|
|
||||||
func NewArtistIndexRepository() *ArtistIndex {
|
func NewArtistIndexRepository() *ArtistIndex {
|
||||||
r := &ArtistIndex{}
|
r := &ArtistIndex{}
|
||||||
r.key = "index"
|
r.table = "index"
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,11 +2,9 @@ package repositories
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"sync"
|
"sync"
|
||||||
"encoding/json"
|
|
||||||
"github.com/astaxie/beego"
|
"github.com/astaxie/beego"
|
||||||
"github.com/siddontang/ledisdb/ledis"
|
"github.com/siddontang/ledisdb/ledis"
|
||||||
"github.com/siddontang/ledisdb/config"
|
"github.com/siddontang/ledisdb/config"
|
||||||
"github.com/deluan/gosonic/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@@ -26,40 +24,4 @@ func db() *ledis.DB {
|
|||||||
_dbInstance = instance
|
_dbInstance = instance
|
||||||
})
|
})
|
||||||
return _dbInstance
|
return _dbInstance
|
||||||
}
|
|
||||||
|
|
||||||
func saveStruct(key, id string, data interface{}) error {
|
|
||||||
h, err := utils.ToMap(data)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
var fvList = make([]ledis.FVPair, len(h))
|
|
||||||
i := 0
|
|
||||||
for f, v := range h {
|
|
||||||
fvList[i].Field = []byte(f)
|
|
||||||
fvList[i].Value, _ = json.Marshal(v)
|
|
||||||
i++
|
|
||||||
}
|
|
||||||
kh := key + "_id_" + id
|
|
||||||
ks := key + "_ids"
|
|
||||||
db().SAdd([]byte(ks), []byte(id))
|
|
||||||
return db().HMset([]byte(kh), fvList...)
|
|
||||||
}
|
|
||||||
|
|
||||||
func readStruct(key, id string, rec interface{}) error {
|
|
||||||
kh := key + "_id_" + id
|
|
||||||
fvs, _ := db().HGetAll([]byte(kh))
|
|
||||||
var m = make(map[string]interface{}, len(fvs))
|
|
||||||
for _, fv := range fvs {
|
|
||||||
var v interface{}
|
|
||||||
json.Unmarshal(fv.Value, &v)
|
|
||||||
m[string(fv.Field)] = v
|
|
||||||
}
|
|
||||||
|
|
||||||
return utils.ToStruct(m, rec)
|
|
||||||
}
|
|
||||||
|
|
||||||
func count(key string) (int, error) {
|
|
||||||
ids, err := db().SMembers([]byte(key + "_ids"))
|
|
||||||
return len(ids), err
|
|
||||||
}
|
}
|
||||||
@@ -10,7 +10,7 @@ type MediaFile struct {
|
|||||||
|
|
||||||
func NewMediaFileRepository() *MediaFile {
|
func NewMediaFileRepository() *MediaFile {
|
||||||
r := &MediaFile{}
|
r := &MediaFile{}
|
||||||
r.key = "mediafile"
|
r.table = "mediafile"
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
+14
-30
@@ -5,8 +5,6 @@ import (
|
|||||||
"github.com/deluan/gosonic/repositories"
|
"github.com/deluan/gosonic/repositories"
|
||||||
"github.com/deluan/gosonic/models"
|
"github.com/deluan/gosonic/models"
|
||||||
"strings"
|
"strings"
|
||||||
"fmt"
|
|
||||||
"encoding/json"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type Scanner interface {
|
type Scanner interface {
|
||||||
@@ -33,10 +31,8 @@ func importLibrary(files []Track) {
|
|||||||
var artistIndex = make(map[string]tempIndex)
|
var artistIndex = make(map[string]tempIndex)
|
||||||
|
|
||||||
for _, t := range files {
|
for _, t := range files {
|
||||||
mf, album, artist := processTrack(&t)
|
mf, album, artist := parseTrack(&t)
|
||||||
mergeInfo(mfRepo, mf, albumRepo, album, artistRepo, artist)
|
persist(mfRepo, mf, albumRepo, album, artistRepo, artist)
|
||||||
fmt.Printf("%#v\n", album)
|
|
||||||
fmt.Printf("%#v\n\n", artist)
|
|
||||||
collectIndex(artist, artistIndex)
|
collectIndex(artist, artistIndex)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -44,14 +40,15 @@ func importLibrary(files []Track) {
|
|||||||
beego.Error(err)
|
beego.Error(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
j,_ := json.MarshalIndent(artistIndex, "", " ")
|
c, _ := artistRepo.CountAll()
|
||||||
fmt.Println(string(j))
|
beego.Info("Total Artists in database:", c)
|
||||||
|
c, _ = albumRepo.CountAll()
|
||||||
c, _ := mfRepo.CountAll()
|
beego.Info("Total Albums in database:", c)
|
||||||
beego.Info("Total mediafiles in database:", c)
|
c, _ = mfRepo.CountAll()
|
||||||
|
beego.Info("Total MediaFiles in database:", c)
|
||||||
}
|
}
|
||||||
|
|
||||||
func processTrack(t *Track) (*models.MediaFile, *models.Album, *models.Artist) {
|
func parseTrack(t *Track) (*models.MediaFile, *models.Album, *models.Artist) {
|
||||||
mf := &models.MediaFile{
|
mf := &models.MediaFile{
|
||||||
Id: t.Id,
|
Id: t.Id,
|
||||||
Album: t.Album,
|
Album: t.Album,
|
||||||
@@ -77,30 +74,17 @@ func processTrack(t *Track) (*models.MediaFile, *models.Album, *models.Artist) {
|
|||||||
return mf, album, artist
|
return mf, album, artist
|
||||||
}
|
}
|
||||||
|
|
||||||
func mergeInfo(mfRepo *repositories.MediaFile, mf *models.MediaFile, albumRepo *repositories.Album, album *models.Album, artistRepo *repositories.Artist, artist *models.Artist) {
|
func persist(mfRepo *repositories.MediaFile, mf *models.MediaFile, albumRepo *repositories.Album, album *models.Album, artistRepo *repositories.Artist, artist *models.Artist) {
|
||||||
artist.Id = artistRepo.NewId(artist.Name)
|
if err := artistRepo.Put(artist); err != nil {
|
||||||
|
|
||||||
sAlbum, err := albumRepo.GetByName(album.Name)
|
|
||||||
if err != nil {
|
|
||||||
beego.Error(err)
|
beego.Error(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
album.ArtistId = artist.Id
|
album.ArtistId = artist.Id
|
||||||
album.AddMediaFiles(mf, sAlbum.MediaFiles)
|
if err := albumRepo.Put(album); err != nil {
|
||||||
sAlbum, err = albumRepo.Put(album)
|
|
||||||
if err != nil {
|
|
||||||
beego.Error(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
sArtist, err := artistRepo.GetByName(artist.Name)
|
|
||||||
if err != nil {
|
|
||||||
beego.Error(err)
|
|
||||||
}
|
|
||||||
artist.AddAlbums(sAlbum, sArtist.Albums)
|
|
||||||
_, err = artistRepo.Put(artist)
|
|
||||||
if err != nil {
|
|
||||||
beego.Error(err)
|
beego.Error(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
mf.AlbumId = album.Id
|
||||||
if err := mfRepo.Put(mf); err != nil {
|
if err := mfRepo.Put(mf); err != nil {
|
||||||
beego.Error(err)
|
beego.Error(err)
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user