mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 21:17:37 +03:00
Add bookmark in persistence layer
This commit is contained in:
parent
3d4f4b4e2b
commit
2d3ed85311
7 changed files with 172 additions and 25 deletions
|
@ -14,13 +14,13 @@ func upCreatePlayQueuesTable(tx *sql.Tx) error {
|
||||||
_, err := tx.Exec(`
|
_, err := tx.Exec(`
|
||||||
create table playqueue
|
create table playqueue
|
||||||
(
|
(
|
||||||
id varchar(255) not null,
|
id varchar(255) not null primary key,
|
||||||
user_id varchar(255) not null
|
user_id varchar(255) not null
|
||||||
references user (id)
|
references user (id)
|
||||||
on update cascade on delete cascade,
|
on update cascade on delete cascade,
|
||||||
comment varchar(255),
|
comment varchar(255),
|
||||||
current varchar(255),
|
current varchar(255) not null,
|
||||||
position real,
|
position integer,
|
||||||
changed_by varchar(255),
|
changed_by varchar(255),
|
||||||
items varchar(255),
|
items varchar(255),
|
||||||
created_at datetime,
|
created_at datetime,
|
||||||
|
|
|
@ -9,7 +9,7 @@ type PlayQueue struct {
|
||||||
UserID string `json:"userId" orm:"column(user_id)"`
|
UserID string `json:"userId" orm:"column(user_id)"`
|
||||||
Comment string `json:"comment"`
|
Comment string `json:"comment"`
|
||||||
Current string `json:"current"`
|
Current string `json:"current"`
|
||||||
Position float32 `json:"position"`
|
Position int64 `json:"position"`
|
||||||
ChangedBy string `json:"changedBy"`
|
ChangedBy string `json:"changedBy"`
|
||||||
Items MediaFiles `json:"items,omitempty"`
|
Items MediaFiles `json:"items,omitempty"`
|
||||||
CreatedAt time.Time `json:"createdAt"`
|
CreatedAt time.Time `json:"createdAt"`
|
||||||
|
@ -21,4 +21,16 @@ type PlayQueues []PlayQueue
|
||||||
type PlayQueueRepository interface {
|
type PlayQueueRepository interface {
|
||||||
Store(queue *PlayQueue) error
|
Store(queue *PlayQueue) error
|
||||||
Retrieve(userId string) (*PlayQueue, error)
|
Retrieve(userId string) (*PlayQueue, error)
|
||||||
|
AddBookmark(userId, id, comment string, position int64) error
|
||||||
|
GetBookmarks(userId string) (Bookmarks, error)
|
||||||
|
DeleteBookmark(userId, id string) error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Bookmark struct {
|
||||||
|
ID string `json:"id" orm:"column(id)"`
|
||||||
|
Comment string `json:"comment"`
|
||||||
|
Position int64 `json:"position"`
|
||||||
|
CreatedAt time.Time `json:"createdAt"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Bookmarks []Bookmark
|
||||||
|
|
|
@ -9,6 +9,7 @@ import (
|
||||||
"github.com/astaxie/beego/orm"
|
"github.com/astaxie/beego/orm"
|
||||||
"github.com/deluan/navidrome/log"
|
"github.com/deluan/navidrome/log"
|
||||||
"github.com/deluan/navidrome/model"
|
"github.com/deluan/navidrome/model"
|
||||||
|
"github.com/deluan/navidrome/model/request"
|
||||||
)
|
)
|
||||||
|
|
||||||
type playQueueRepository struct {
|
type playQueueRepository struct {
|
||||||
|
@ -28,7 +29,7 @@ type playQueue struct {
|
||||||
UserID string `orm:"column(user_id)"`
|
UserID string `orm:"column(user_id)"`
|
||||||
Comment string
|
Comment string
|
||||||
Current string
|
Current string
|
||||||
Position float32
|
Position int64
|
||||||
ChangedBy string
|
ChangedBy string
|
||||||
Items string
|
Items string
|
||||||
CreatedAt time.Time
|
CreatedAt time.Time
|
||||||
|
@ -63,6 +64,66 @@ func (r *playQueueRepository) Retrieve(userId string) (*model.PlayQueue, error)
|
||||||
return &pls, err
|
return &pls, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *playQueueRepository) AddBookmark(userId, id, comment string, position int64) error {
|
||||||
|
u := loggedUser(r.ctx)
|
||||||
|
client, _ := request.ClientFrom(r.ctx)
|
||||||
|
bm := &playQueue{
|
||||||
|
UserID: userId,
|
||||||
|
Comment: comment,
|
||||||
|
Current: id,
|
||||||
|
Position: position,
|
||||||
|
ChangedBy: client,
|
||||||
|
CreatedAt: time.Now(),
|
||||||
|
UpdatedAt: time.Now(),
|
||||||
|
}
|
||||||
|
|
||||||
|
sel := r.newSelect().Column("id").Where(And{
|
||||||
|
Eq{"user_id": userId},
|
||||||
|
Eq{"items": ""},
|
||||||
|
Eq{"current": id},
|
||||||
|
})
|
||||||
|
var prev model.PlayQueue
|
||||||
|
err := r.queryOne(sel, &prev)
|
||||||
|
if err != nil && err != model.ErrNotFound {
|
||||||
|
log.Error(r.ctx, "Error retrieving previous bookmark", "user", u.UserName, err, "mediaFileId", id, err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = r.put(prev.ID, bm)
|
||||||
|
if err != nil {
|
||||||
|
log.Error(r.ctx, "Error saving bookmark", "user", u.UserName, err, "mediaFileId", id, err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *playQueueRepository) GetBookmarks(userId string) (model.Bookmarks, error) {
|
||||||
|
u := loggedUser(r.ctx)
|
||||||
|
sel := r.newSelect().Column("*").Where(And{Eq{"user_id": userId}, Eq{"items": ""}})
|
||||||
|
var pqs model.PlayQueues
|
||||||
|
err := r.queryAll(sel, &pqs)
|
||||||
|
if err != nil {
|
||||||
|
log.Error(r.ctx, "Error retrieving bookmarks", "user", u.UserName, err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
bms := make(model.Bookmarks, len(pqs))
|
||||||
|
for i := range pqs {
|
||||||
|
bms[i].ID = pqs[i].Current
|
||||||
|
bms[i].Comment = pqs[i].Comment
|
||||||
|
bms[i].Position = int64(pqs[i].Position)
|
||||||
|
bms[i].CreatedAt = pqs[i].CreatedAt
|
||||||
|
}
|
||||||
|
return bms, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *playQueueRepository) DeleteBookmark(userId, id string) error {
|
||||||
|
return r.delete(And{
|
||||||
|
Eq{"user_id": userId},
|
||||||
|
Eq{"items": ""},
|
||||||
|
Eq{"current": id},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
func (r *playQueueRepository) fromModel(q *model.PlayQueue) playQueue {
|
func (r *playQueueRepository) fromModel(q *model.PlayQueue) playQueue {
|
||||||
pq := playQueue{
|
pq := playQueue{
|
||||||
ID: q.ID,
|
ID: q.ID,
|
||||||
|
|
|
@ -23,32 +23,76 @@ var _ = Describe("PlayQueueRepository", func() {
|
||||||
repo = NewPlayQueueRepository(ctx, orm.NewOrm())
|
repo = NewPlayQueueRepository(ctx, orm.NewOrm())
|
||||||
})
|
})
|
||||||
|
|
||||||
It("returns notfound error if there's no playqueue for the user", func() {
|
Describe("PlayQueues", func() {
|
||||||
_, err := repo.Retrieve("user999")
|
It("returns notfound error if there's no playqueue for the user", func() {
|
||||||
Expect(err).To(MatchError(model.ErrNotFound))
|
_, err := repo.Retrieve("user999")
|
||||||
|
Expect(err).To(MatchError(model.ErrNotFound))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("stores and retrieves the playqueue for the user", func() {
|
||||||
|
By("Storing a playqueue for the user")
|
||||||
|
|
||||||
|
expected := aPlayQueue("user1", songDayInALife.ID, 123, songComeTogether, songDayInALife)
|
||||||
|
Expect(repo.Store(expected)).To(BeNil())
|
||||||
|
|
||||||
|
actual, err := repo.Retrieve("user1")
|
||||||
|
Expect(err).To(BeNil())
|
||||||
|
|
||||||
|
AssertPlayQueue(expected, actual)
|
||||||
|
|
||||||
|
By("Storing a new playqueue for the same user")
|
||||||
|
|
||||||
|
new := aPlayQueue("user1", songRadioactivity.ID, 321, songAntenna, songRadioactivity)
|
||||||
|
Expect(repo.Store(new)).To(BeNil())
|
||||||
|
|
||||||
|
actual, err = repo.Retrieve("user1")
|
||||||
|
Expect(err).To(BeNil())
|
||||||
|
|
||||||
|
AssertPlayQueue(new, actual)
|
||||||
|
Expect(countPlayQueues(repo, "user1")).To(Equal(1))
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
It("stores and retrieves the playqueue for the user", func() {
|
Describe("Bookmarks", func() {
|
||||||
By("Storing a playqueue for the user")
|
It("returns an empty collection if there are no bookmarks", func() {
|
||||||
|
Expect(repo.GetBookmarks("user999")).To(BeEmpty())
|
||||||
|
})
|
||||||
|
|
||||||
expected := aPlayQueue("user1", songDayInALife.ID, 123, songComeTogether, songDayInALife)
|
It("saves and overrides bookmarks", func() {
|
||||||
Expect(repo.Store(expected)).To(BeNil())
|
By("Saving the bookmark")
|
||||||
|
Expect(repo.AddBookmark("user5", songAntenna.ID, "this is a comment", 123)).To(BeNil())
|
||||||
|
|
||||||
actual, err := repo.Retrieve("user1")
|
bms, err := repo.GetBookmarks("user5")
|
||||||
Expect(err).To(BeNil())
|
Expect(err).To(BeNil())
|
||||||
|
|
||||||
AssertPlayQueue(expected, actual)
|
Expect(bms).To(HaveLen(1))
|
||||||
|
Expect(bms[0].ID).To(Equal(songAntenna.ID))
|
||||||
|
Expect(bms[0].Comment).To(Equal("this is a comment"))
|
||||||
|
Expect(bms[0].Position).To(Equal(int64(123)))
|
||||||
|
|
||||||
By("Storing a new playqueue for the same user")
|
By("Overriding the bookmark")
|
||||||
|
Expect(repo.AddBookmark("user5", songAntenna.ID, "another comment", 333)).To(BeNil())
|
||||||
|
|
||||||
new := aPlayQueue("user1", songRadioactivity.ID, 321, songAntenna, songRadioactivity)
|
bms, err = repo.GetBookmarks("user5")
|
||||||
Expect(repo.Store(new)).To(BeNil())
|
Expect(err).To(BeNil())
|
||||||
|
|
||||||
actual, err = repo.Retrieve("user1")
|
Expect(bms[0].ID).To(Equal(songAntenna.ID))
|
||||||
Expect(err).To(BeNil())
|
Expect(bms[0].Comment).To(Equal("another comment"))
|
||||||
|
Expect(bms[0].Position).To(Equal(int64(333)))
|
||||||
|
|
||||||
AssertPlayQueue(new, actual)
|
By("Saving another bookmark")
|
||||||
Expect(countPlayQueues(repo, "user1")).To(Equal(1))
|
Expect(repo.AddBookmark("user5", songComeTogether.ID, "one more comment", 444)).To(BeNil())
|
||||||
|
bms, err = repo.GetBookmarks("user5")
|
||||||
|
Expect(err).To(BeNil())
|
||||||
|
Expect(bms).To(HaveLen(2))
|
||||||
|
|
||||||
|
By("Delete bookmark")
|
||||||
|
Expect(repo.DeleteBookmark("user5", songAntenna.ID))
|
||||||
|
bms, err = repo.GetBookmarks("user5")
|
||||||
|
Expect(err).To(BeNil())
|
||||||
|
Expect(bms).To(HaveLen(1))
|
||||||
|
Expect(bms[0].ID).To(Equal(songComeTogether.ID))
|
||||||
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -74,7 +118,7 @@ func AssertPlayQueue(expected, actual *model.PlayQueue) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func aPlayQueue(userId, current string, position float32, items ...model.MediaFile) *model.PlayQueue {
|
func aPlayQueue(userId, current string, position int64, items ...model.MediaFile) *model.PlayQueue {
|
||||||
createdAt := time.Now()
|
createdAt := time.Now()
|
||||||
updatedAt := createdAt.Add(time.Minute)
|
updatedAt := createdAt.Add(time.Minute)
|
||||||
id, _ := uuid.NewRandom()
|
id, _ := uuid.NewRandom()
|
||||||
|
|
|
@ -46,7 +46,7 @@ func (c *BookmarksController) SavePlayQueue(w http.ResponseWriter, r *http.Reque
|
||||||
}
|
}
|
||||||
|
|
||||||
current := utils.ParamString(r, "current")
|
current := utils.ParamString(r, "current")
|
||||||
position := utils.ParamInt(r, "position", 0)
|
position := utils.ParamInt64(r, "position", 0)
|
||||||
|
|
||||||
user, _ := request.UserFrom(r.Context())
|
user, _ := request.UserFrom(r.Context())
|
||||||
client, _ := request.ClientFrom(r.Context())
|
client, _ := request.ClientFrom(r.Context())
|
||||||
|
@ -59,7 +59,7 @@ func (c *BookmarksController) SavePlayQueue(w http.ResponseWriter, r *http.Reque
|
||||||
pq := &model.PlayQueue{
|
pq := &model.PlayQueue{
|
||||||
UserID: user.ID,
|
UserID: user.ID,
|
||||||
Current: current,
|
Current: current,
|
||||||
Position: float32(position),
|
Position: position,
|
||||||
ChangedBy: client,
|
ChangedBy: client,
|
||||||
Items: items,
|
Items: items,
|
||||||
CreatedAt: time.Time{},
|
CreatedAt: time.Time{},
|
||||||
|
|
|
@ -51,6 +51,18 @@ func ParamInt(r *http.Request, param string, def int) int {
|
||||||
return int(value)
|
return int(value)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func ParamInt64(r *http.Request, param string, def int64) int64 {
|
||||||
|
v := ParamString(r, param)
|
||||||
|
if v == "" {
|
||||||
|
return def
|
||||||
|
}
|
||||||
|
value, err := strconv.ParseInt(v, 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return def
|
||||||
|
}
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
|
||||||
func ParamInts(r *http.Request, param string) []int {
|
func ParamInts(r *http.Request, param string) []int {
|
||||||
pStr := ParamStrings(r, param)
|
pStr := ParamStrings(r, param)
|
||||||
ints := make([]int, 0, len(pStr))
|
ints := make([]int, 0, len(pStr))
|
||||||
|
|
|
@ -98,6 +98,24 @@ var _ = Describe("Request Helpers", func() {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
Describe("ParamInt64", func() {
|
||||||
|
BeforeEach(func() {
|
||||||
|
r = httptest.NewRequest("GET", "/ping?i=123&inv=123.45", nil)
|
||||||
|
})
|
||||||
|
|
||||||
|
It("returns default value if param does not exist", func() {
|
||||||
|
Expect(ParamInt64(r, "xx", 999)).To(Equal(int64(999)))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("returns default value if param is an invalid int", func() {
|
||||||
|
Expect(ParamInt64(r, "inv", 999)).To(Equal(int64(999)))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("returns parsed time", func() {
|
||||||
|
Expect(ParamInt64(r, "i", 999)).To(Equal(int64(123)))
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
Describe("ParamInts", func() {
|
Describe("ParamInts", func() {
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
r = httptest.NewRequest("GET", "/ping?i=123&i=456", nil)
|
r = httptest.NewRequest("GET", "/ping?i=123&i=456", nil)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue