mirror of
https://git.sr.ht/~phw/scotty
synced 2025-06-01 19:38:34 +02:00
206 lines
4.9 KiB
Go
206 lines
4.9 KiB
Go
/*
|
|
Copyright © 2025 Philipp Wolfer <phw@uploadedlobster.com>
|
|
|
|
This file is part of Scotty.
|
|
|
|
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 deezerhistory
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/xuri/excelize/v2"
|
|
"go.uploadedlobster.com/mbtypes"
|
|
"go.uploadedlobster.com/scotty/internal/config"
|
|
"go.uploadedlobster.com/scotty/internal/i18n"
|
|
"go.uploadedlobster.com/scotty/internal/models"
|
|
)
|
|
|
|
const (
|
|
sheetListeningHistory = "10_listeningHistory"
|
|
sheetFavoriteSongs = "8_favoriteSong"
|
|
)
|
|
|
|
type DeezerHistoryBackend struct {
|
|
filePath string
|
|
}
|
|
|
|
func (b *DeezerHistoryBackend) Name() string { return "deezer-history" }
|
|
|
|
func (b *DeezerHistoryBackend) Options() []models.BackendOption {
|
|
return []models.BackendOption{{
|
|
Name: "file-path",
|
|
Label: i18n.Tr("File path"),
|
|
Type: models.String,
|
|
Default: "",
|
|
}}
|
|
}
|
|
|
|
func (b *DeezerHistoryBackend) InitConfig(config *config.ServiceConfig) error {
|
|
b.filePath = config.GetString("file-path")
|
|
return nil
|
|
}
|
|
|
|
func (b *DeezerHistoryBackend) ExportListens(ctx context.Context, oldestTimestamp time.Time, results chan models.ListensResult, progress chan models.TransferProgress) {
|
|
p := models.TransferProgress{
|
|
Export: &models.Progress{},
|
|
}
|
|
|
|
rows, err := ReadXLSXSheet(b.filePath, sheetListeningHistory)
|
|
if err != nil {
|
|
p.Export.Abort()
|
|
progress <- p
|
|
results <- models.ListensResult{Error: err}
|
|
return
|
|
}
|
|
|
|
count := len(rows) - 1 // Exclude the header row
|
|
p.Export.TotalItems = count
|
|
p.Export.Total = int64(count)
|
|
|
|
listens := make(models.ListensList, 0, count)
|
|
for i, row := range models.IterExportProgress(rows, &p, progress) {
|
|
// Skip header row
|
|
if i == 0 {
|
|
continue
|
|
}
|
|
|
|
l, err := RowAsListen(row)
|
|
if err != nil {
|
|
p.Export.Abort()
|
|
progress <- p
|
|
results <- models.ListensResult{Error: err}
|
|
return
|
|
}
|
|
listens = append(listens, *l)
|
|
}
|
|
|
|
sort.Sort(listens)
|
|
results <- models.ListensResult{Items: listens}
|
|
p.Export.Complete()
|
|
progress <- p
|
|
}
|
|
|
|
func (b *DeezerHistoryBackend) ExportLoves(ctx context.Context, oldestTimestamp time.Time, results chan models.LovesResult, progress chan models.TransferProgress) {
|
|
p := models.TransferProgress{
|
|
Export: &models.Progress{},
|
|
}
|
|
|
|
rows, err := ReadXLSXSheet(b.filePath, sheetFavoriteSongs)
|
|
if err != nil {
|
|
p.Export.Abort()
|
|
progress <- p
|
|
results <- models.LovesResult{Error: err}
|
|
return
|
|
}
|
|
|
|
count := len(rows) - 1 // Exclude the header row
|
|
p.Export.TotalItems = count
|
|
p.Export.Total = int64(count)
|
|
|
|
love := make(models.LovesList, 0, count)
|
|
for i, row := range models.IterExportProgress(rows, &p, progress) {
|
|
// Skip header row
|
|
if i == 0 {
|
|
continue
|
|
}
|
|
|
|
l, err := RowAsLove(row)
|
|
if err != nil {
|
|
p.Export.Abort()
|
|
progress <- p
|
|
results <- models.LovesResult{Error: err}
|
|
return
|
|
}
|
|
love = append(love, *l)
|
|
}
|
|
|
|
sort.Sort(love)
|
|
results <- models.LovesResult{Items: love}
|
|
p.Export.Complete()
|
|
progress <- p
|
|
}
|
|
|
|
func ReadXLSXSheet(path string, sheet string) ([][]string, error) {
|
|
exc, err := excelize.OpenFile(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Get all the rows in the Sheet1.
|
|
return exc.GetRows(sheet)
|
|
}
|
|
|
|
func RowAsListen(row []string) (*models.Listen, error) {
|
|
if len(row) < 9 {
|
|
err := fmt.Errorf("Invalid row, expected 9 columns, got %d", len(row))
|
|
return nil, err
|
|
}
|
|
|
|
listenedAt, err := time.Parse(time.DateTime, row[8])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
listen := models.Listen{
|
|
ListenedAt: listenedAt,
|
|
Track: models.Track{
|
|
TrackName: row[0],
|
|
ArtistNames: []string{row[1]},
|
|
ReleaseName: row[3],
|
|
ISRC: mbtypes.ISRC(row[2]),
|
|
AdditionalInfo: map[string]any{
|
|
"music_service": "deezer.com",
|
|
},
|
|
},
|
|
}
|
|
|
|
if duration, err := strconv.Atoi(row[5]); err == nil {
|
|
listen.PlaybackDuration = time.Duration(duration) * time.Second
|
|
}
|
|
|
|
return &listen, nil
|
|
}
|
|
|
|
func RowAsLove(row []string) (*models.Love, error) {
|
|
if len(row) < 5 {
|
|
err := fmt.Errorf("Invalid row, expected 5 columns, got %d", len(row))
|
|
return nil, err
|
|
}
|
|
|
|
url := row[4]
|
|
if !strings.HasPrefix(url, "http://") || !strings.HasPrefix(url, "https") {
|
|
url = "https://" + url
|
|
}
|
|
|
|
love := models.Love{
|
|
Track: models.Track{
|
|
TrackName: row[0],
|
|
ArtistNames: []string{row[1]},
|
|
ReleaseName: row[2],
|
|
ISRC: mbtypes.ISRC(row[3]),
|
|
AdditionalInfo: map[string]any{
|
|
"music_service": "deezer.com",
|
|
"origin_url": url,
|
|
"deezer_id": url,
|
|
},
|
|
},
|
|
}
|
|
|
|
return &love, nil
|
|
}
|