scotty/internal/backends/lastfm/lastfm.go
2023-11-27 14:48:00 +01:00

132 lines
3.4 KiB
Go

/*
Copyright © 2023 Philipp Wolfer <phw@uploadedlobster.com>
Scotty is free software: you can redistribute it and/or modify it under the
terms of the GNU General Public License as published by the Free Software
Foundation, either version 3 of the License, or (at your option) any later version.
Scotty is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with
Scotty. If not, see <https://www.gnu.org/licenses/>.
*/
package lastfm
import (
"net/url"
"sort"
"strconv"
"time"
"github.com/shkh/lastfm-go/lastfm"
"github.com/spf13/viper"
"go.uploadedlobster.com/scotty/internal/auth"
"go.uploadedlobster.com/scotty/internal/models"
"golang.org/x/oauth2"
)
const MaxItemsPerGet = 50
type LastfmApiBackend struct {
client *lastfm.Api
username string
}
func (b *LastfmApiBackend) Name() string { return "lastfm" }
func (b *LastfmApiBackend) FromConfig(config *viper.Viper) models.Backend {
clientId := config.GetString("client-id")
clientSecret := config.GetString("client-secret")
b.client = lastfm.New(clientId, clientSecret)
b.username = config.GetString("username")
return b
}
func (b *LastfmApiBackend) OAuth2Strategy(redirectUrl *url.URL) auth.OAuth2Strategy {
return lastfmStrategy{
client: b.client,
redirectUrl: redirectUrl,
}
}
func (b *LastfmApiBackend) OAuth2Setup(token oauth2.TokenSource) error {
t, err := token.Token()
if err != nil {
return err
}
return b.client.LoginWithToken(t.AccessToken)
}
func (b *LastfmApiBackend) ExportLoves(oldestTimestamp time.Time, results chan models.LovesResult, progress chan models.Progress) {
// Choose a high offset, we attempt to search the loves backwards starting
// at the oldest one.
page := 1
perPage := MaxItemsPerGet
defer close(results)
loves := make(models.LovesList, 0, 2*MaxItemsPerGet)
p := models.Progress{Total: int64(perPage)}
var totalCount int
out:
for {
result, err := b.client.User.GetLovedTracks(lastfm.P{
"user": b.username,
"limit": MaxItemsPerGet,
"page": page,
})
if err != nil {
progress <- p.Complete()
results <- models.LovesResult{Error: err}
return
}
p.Total = int64(result.Total)
count := len(result.Tracks)
if count == 0 {
break out
}
for _, track := range result.Tracks {
timestamp, err := strconv.ParseInt(track.Date.Uts, 10, 64)
if err != nil {
progress <- p.Complete()
results <- models.LovesResult{Error: err}
return
}
if timestamp > oldestTimestamp.Unix() {
totalCount += 1
love := models.Love{
Created: time.Unix(timestamp, 0),
UserName: result.User,
RecordingMbid: models.MBID(track.Mbid),
Track: models.Track{
TrackName: track.Name,
ArtistNames: []string{track.Artist.Name},
RecordingMbid: models.MBID(track.Mbid),
ArtistMbids: []models.MBID{models.MBID(track.Artist.Mbid)},
AdditionalInfo: models.AdditionalInfo{
"lastfm_url": track.Url,
},
},
}
loves = append(loves, love)
} else {
break out
}
}
p.Elapsed += int64(count)
progress <- p
page += 1
}
sort.Sort(loves)
results <- models.LovesResult{Loves: loves, Total: totalCount}
progress <- p.Complete()
}