mirror of
https://git.sr.ht/~phw/scotty
synced 2025-04-16 10:09:28 +02:00
ListenBrainz submit listens
This commit is contained in:
parent
d55114ed3f
commit
e8fdfb95a6
4 changed files with 118 additions and 6 deletions
|
@ -32,9 +32,12 @@ import (
|
||||||
|
|
||||||
const listenBrainzBaseURL = "https://api.listenbrainz.org/1/"
|
const listenBrainzBaseURL = "https://api.listenbrainz.org/1/"
|
||||||
|
|
||||||
const DefaultItemsPerGet = 25
|
const (
|
||||||
const MaxItemsPerGet = 1000
|
DefaultItemsPerGet = 25
|
||||||
const DefaultRateLimitWaitSeconds = 5
|
MaxItemsPerGet = 1000
|
||||||
|
MaxListensPerRequest = 1000
|
||||||
|
DefaultRateLimitWaitSeconds = 5
|
||||||
|
)
|
||||||
|
|
||||||
type Client struct {
|
type Client struct {
|
||||||
HttpClient *resty.Client
|
HttpClient *resty.Client
|
||||||
|
@ -96,6 +99,22 @@ func (c Client) GetListens(user string, maxTime time.Time, minTime time.Time) (r
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c Client) SubmitListens(listens ListenSubmission) (result StatusResult, err error) {
|
||||||
|
const path = "/submit-listens"
|
||||||
|
errorResult := ErrorResult{}
|
||||||
|
response, err := c.HttpClient.R().
|
||||||
|
SetBody(listens).
|
||||||
|
SetResult(&result).
|
||||||
|
SetError(&errorResult).
|
||||||
|
Post(path)
|
||||||
|
|
||||||
|
if response.StatusCode() != 200 {
|
||||||
|
err = errors.New(errorResult.Error)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
func (c Client) GetFeedback(user string, status int, offset int) (result GetFeedbackResult, err error) {
|
func (c Client) GetFeedback(user string, status int, offset int) (result GetFeedbackResult, err error) {
|
||||||
const path = "/feedback/user/{username}/get-feedback"
|
const path = "/feedback/user/{username}/get-feedback"
|
||||||
errorResult := ErrorResult{}
|
errorResult := ErrorResult{}
|
||||||
|
|
|
@ -57,6 +57,44 @@ func TestGetListens(t *testing.T) {
|
||||||
assert.Equal("Shadowplay", result.Payload.Listens[0].TrackMetadata.TrackName)
|
assert.Equal("Shadowplay", result.Payload.Listens[0].TrackMetadata.TrackName)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestSubmitListens(t *testing.T) {
|
||||||
|
client := listenbrainz.NewClient("thetoken")
|
||||||
|
httpmock.ActivateNonDefault(client.HttpClient.GetClient())
|
||||||
|
|
||||||
|
responder, err := httpmock.NewJsonResponder(200, listenbrainz.StatusResult{
|
||||||
|
Status: "ok",
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
url := "https://api.listenbrainz.org/1/submit-listens"
|
||||||
|
httpmock.RegisterResponder("POST", url, responder)
|
||||||
|
|
||||||
|
listens := listenbrainz.ListenSubmission{
|
||||||
|
ListenType: listenbrainz.Import,
|
||||||
|
Payload: []listenbrainz.Listen{
|
||||||
|
{
|
||||||
|
ListenedAt: time.Now().Unix(),
|
||||||
|
TrackMetadata: listenbrainz.Track{
|
||||||
|
TrackName: "Oweynagat",
|
||||||
|
ArtistName: "Dool",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
ListenedAt: time.Now().Add(-2 * time.Minute).Unix(),
|
||||||
|
TrackMetadata: listenbrainz.Track{
|
||||||
|
TrackName: "Say Just Words",
|
||||||
|
ArtistName: "Paradise Lost",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
result, err := client.SubmitListens(listens)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
assert.Equal(t, "ok", result.Status)
|
||||||
|
}
|
||||||
|
|
||||||
func TestGetFeedback(t *testing.T) {
|
func TestGetFeedback(t *testing.T) {
|
||||||
defer httpmock.DeactivateAndReset()
|
defer httpmock.DeactivateAndReset()
|
||||||
|
|
||||||
|
|
|
@ -95,6 +95,48 @@ out:
|
||||||
results <- models.ListensResult{Listens: listens, OldestTimestamp: oldestTimestamp}
|
results <- models.ListensResult{Listens: listens, OldestTimestamp: oldestTimestamp}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b *ListenBrainzApiBackend) ImportListens(export models.ListensResult, importResult models.ImportResult, progress chan models.Progress) (models.ImportResult, error) {
|
||||||
|
total := len(export.Listens)
|
||||||
|
for i := 0; i < total; i += MaxListensPerRequest {
|
||||||
|
listens := export.Listens[i:min(i+MaxItemsPerGet, total)]
|
||||||
|
count := len(listens)
|
||||||
|
if count == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
submission := ListenSubmission{
|
||||||
|
ListenType: Import,
|
||||||
|
Payload: make([]Listen, 0, count),
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, l := range listens {
|
||||||
|
l.FillAdditionalInfo()
|
||||||
|
listen := Listen{
|
||||||
|
ListenedAt: l.ListenedAt.Unix(),
|
||||||
|
TrackMetadata: Track{
|
||||||
|
TrackName: l.TrackName,
|
||||||
|
ReleaseName: l.ReleaseName,
|
||||||
|
ArtistName: l.ArtistName(),
|
||||||
|
AdditionalInfo: l.AdditionalInfo,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
listen.TrackMetadata.AdditionalInfo["submission_client"] = "Scotty"
|
||||||
|
submission.Payload = append(submission.Payload, listen)
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := b.client.SubmitListens(submission)
|
||||||
|
if err != nil {
|
||||||
|
return importResult, err
|
||||||
|
}
|
||||||
|
|
||||||
|
importResult.UpdateTimestamp(listens[count-1].ListenedAt)
|
||||||
|
importResult.ImportCount += count
|
||||||
|
progress <- models.Progress{}.FromImportResult(importResult)
|
||||||
|
}
|
||||||
|
|
||||||
|
return importResult, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (b *ListenBrainzApiBackend) ExportLoves(oldestTimestamp time.Time, results chan models.LovesResult, progress chan models.Progress) {
|
func (b *ListenBrainzApiBackend) ExportLoves(oldestTimestamp time.Time, results chan models.LovesResult, progress chan models.Progress) {
|
||||||
offset := 0
|
offset := 0
|
||||||
defer close(results)
|
defer close(results)
|
||||||
|
|
|
@ -39,11 +39,24 @@ type GetListenPayload struct {
|
||||||
Listens []Listen `json:"listens"`
|
Listens []Listen `json:"listens"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type listenType string
|
||||||
|
|
||||||
|
const (
|
||||||
|
PlayingNow listenType = "playing_now"
|
||||||
|
Single listenType = "single"
|
||||||
|
Import listenType = "import"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ListenSubmission struct {
|
||||||
|
ListenType listenType `json:"listen_type"`
|
||||||
|
Payload []Listen `json:"payload"`
|
||||||
|
}
|
||||||
|
|
||||||
type Listen struct {
|
type Listen struct {
|
||||||
InsertedAt int64 `json:"inserted_at"`
|
InsertedAt int64 `json:"inserted_at,omitempty"`
|
||||||
ListenedAt int64 `json:"listened_at"`
|
ListenedAt int64 `json:"listened_at"`
|
||||||
RecordingMsid string `json:"recording_msid"`
|
RecordingMsid string `json:"recording_msid,omitempty"`
|
||||||
UserName string `json:"user_name"`
|
UserName string `json:"user_name,omitempty"`
|
||||||
TrackMetadata Track `json:"track_metadata"`
|
TrackMetadata Track `json:"track_metadata"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue