2022-02-16 23:38:45 +00:00
|
|
|
|
using IF.Lastfm.Core.Api;
|
|
|
|
|
using IF.Lastfm.Core.Api.Helpers;
|
|
|
|
|
using IF.Lastfm.Core.Objects;
|
|
|
|
|
using Microsoft.Extensions.DependencyInjection;
|
|
|
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
|
using Selector.Model;
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
2022-02-18 00:08:42 +00:00
|
|
|
|
using System.Diagnostics;
|
2022-02-16 23:38:45 +00:00
|
|
|
|
using System.Linq;
|
2022-02-18 19:47:11 +00:00
|
|
|
|
using System.Text;
|
2022-02-16 23:38:45 +00:00
|
|
|
|
using System.Threading;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
|
|
|
|
namespace Selector
|
|
|
|
|
{
|
|
|
|
|
public class ScrobbleSaverConfig
|
|
|
|
|
{
|
|
|
|
|
public ApplicationUser User { get; set; }
|
|
|
|
|
public TimeSpan InterRequestDelay { get; set; }
|
|
|
|
|
public DateTime? From { get; set; }
|
|
|
|
|
public DateTime? To { get; set; }
|
|
|
|
|
public int PageSize { get; set; } = 100;
|
2022-02-18 00:08:42 +00:00
|
|
|
|
public int Retries { get; set; } = 5;
|
|
|
|
|
public bool DontAdd { get; set; } = false;
|
|
|
|
|
public bool DontRemove { get; set; } = false;
|
2022-02-16 23:38:45 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class ScrobbleSaver
|
|
|
|
|
{
|
|
|
|
|
private readonly ILogger<ScrobbleSaver> logger;
|
|
|
|
|
|
|
|
|
|
private readonly IUserApi userClient;
|
|
|
|
|
private readonly ScrobbleSaverConfig config;
|
2022-02-18 00:08:42 +00:00
|
|
|
|
private readonly ApplicationDbContext db;
|
2022-02-16 23:38:45 +00:00
|
|
|
|
|
2022-02-18 00:08:42 +00:00
|
|
|
|
public ScrobbleSaver(IUserApi _userClient, ScrobbleSaverConfig _config, ApplicationDbContext _db, ILogger<ScrobbleSaver> _logger)
|
2022-02-16 23:38:45 +00:00
|
|
|
|
{
|
|
|
|
|
userClient = _userClient;
|
|
|
|
|
config = _config;
|
2022-02-18 00:08:42 +00:00
|
|
|
|
db = _db;
|
2022-02-16 23:38:45 +00:00
|
|
|
|
logger = _logger;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task Execute(CancellationToken token)
|
|
|
|
|
{
|
2022-02-18 00:08:42 +00:00
|
|
|
|
logger.LogInformation("Saving scrobbles for {0}/{1}", config.User.UserName, config.User.LastFmUsername);
|
2022-02-16 23:38:45 +00:00
|
|
|
|
|
|
|
|
|
var page1 = await userClient.GetRecentScrobbles(config.User.LastFmUsername, count: config.PageSize, from: config.From, to: config.To);
|
|
|
|
|
|
|
|
|
|
if(page1.Success)
|
|
|
|
|
{
|
|
|
|
|
var scrobbles = page1.Content.ToList();
|
|
|
|
|
|
2022-02-18 00:08:42 +00:00
|
|
|
|
if (page1.TotalPages > 1)
|
2022-02-16 23:38:45 +00:00
|
|
|
|
{
|
2022-02-18 00:08:42 +00:00
|
|
|
|
var tasks = await GetScrobblesFromPageNumbers(2, page1.TotalPages, token);
|
2022-02-16 23:38:45 +00:00
|
|
|
|
var taskResults = await Task.WhenAll(tasks);
|
|
|
|
|
|
|
|
|
|
foreach (var result in taskResults)
|
|
|
|
|
{
|
|
|
|
|
if (result.Success)
|
|
|
|
|
{
|
|
|
|
|
scrobbles.AddRange(result.Content);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2022-02-18 00:08:42 +00:00
|
|
|
|
logger.LogWarning("Failed to get a subset of scrobbles for {0}/{1}", config.User.UserName, config.User.LastFmUsername);
|
2022-02-16 23:38:45 +00:00
|
|
|
|
}
|
2022-02-18 00:08:42 +00:00
|
|
|
|
}
|
2022-02-16 23:38:45 +00:00
|
|
|
|
}
|
|
|
|
|
|
2022-02-18 19:47:11 +00:00
|
|
|
|
IdentifyDuplicates(scrobbles);
|
|
|
|
|
|
2022-02-18 00:08:42 +00:00
|
|
|
|
logger.LogDebug("Ordering and filtering pulled scrobbles");
|
|
|
|
|
|
2022-02-16 23:38:45 +00:00
|
|
|
|
var nativeScrobbles = scrobbles
|
2022-02-18 00:08:42 +00:00
|
|
|
|
.DistinctBy(s => s.TimePlayed?.UtcDateTime)
|
2022-02-16 23:38:45 +00:00
|
|
|
|
.Select(s =>
|
|
|
|
|
{
|
|
|
|
|
var nativeScrobble = (UserScrobble) s;
|
|
|
|
|
nativeScrobble.UserId = config.User.Id;
|
|
|
|
|
return nativeScrobble;
|
|
|
|
|
});
|
2022-02-18 00:08:42 +00:00
|
|
|
|
|
|
|
|
|
logger.LogDebug("Pulling currently stored scrobbles");
|
|
|
|
|
|
2022-02-16 23:38:45 +00:00
|
|
|
|
var currentScrobbles = db.Scrobble
|
|
|
|
|
.AsEnumerable()
|
|
|
|
|
.Where(s => s.UserId == config.User.Id);
|
|
|
|
|
|
|
|
|
|
if (config.From is not null)
|
|
|
|
|
{
|
|
|
|
|
currentScrobbles = currentScrobbles.Where(s => s.Timestamp > config.From);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (config.To is not null)
|
|
|
|
|
{
|
|
|
|
|
currentScrobbles = currentScrobbles.Where(s => s.Timestamp < config.To);
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-18 00:08:42 +00:00
|
|
|
|
logger.LogInformation("Completed scrobble pulling for {0}, pulled {1:n0}", config.User.UserName, nativeScrobbles.Count());
|
|
|
|
|
|
|
|
|
|
logger.LogDebug("Identifying difference sets");
|
|
|
|
|
var time = Stopwatch.StartNew();
|
2022-02-16 23:38:45 +00:00
|
|
|
|
|
2022-02-18 00:08:42 +00:00
|
|
|
|
(var toAdd, var toRemove) = ScrobbleMatcher.IdentifyDiffs(currentScrobbles, nativeScrobbles);
|
|
|
|
|
|
|
|
|
|
var toAddUser = toAdd.Cast<UserScrobble>().ToList();
|
|
|
|
|
var toRemoveUser = toRemove.Cast<UserScrobble>().ToList();
|
|
|
|
|
|
|
|
|
|
time.Stop();
|
|
|
|
|
logger.LogTrace("Finished diffing: {0:n}ms", time.ElapsedMilliseconds);
|
|
|
|
|
|
|
|
|
|
var timeDbOps = Stopwatch.StartNew();
|
|
|
|
|
|
|
|
|
|
if(!config.DontAdd)
|
|
|
|
|
{
|
|
|
|
|
await db.Scrobble.AddRangeAsync(toAddUser);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.LogInformation("Skipping adding of {0} scrobbles", toAddUser.Count);
|
|
|
|
|
}
|
|
|
|
|
if (!config.DontRemove)
|
|
|
|
|
{
|
|
|
|
|
db.Scrobble.RemoveRange(toRemoveUser);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.LogInformation("Skipping removal of {0} scrobbles", toRemoveUser.Count);
|
|
|
|
|
}
|
2022-02-16 23:38:45 +00:00
|
|
|
|
await db.SaveChangesAsync();
|
2022-02-18 00:08:42 +00:00
|
|
|
|
|
|
|
|
|
timeDbOps.Stop();
|
|
|
|
|
logger.LogTrace("DB ops: {0:n}ms", timeDbOps.ElapsedMilliseconds);
|
|
|
|
|
|
|
|
|
|
logger.LogInformation("Completed scrobble pulling for {0}, +{1:n0}, -{2:n0}", config.User.UserName, toAddUser.Count(), toRemoveUser.Count());
|
2022-02-16 23:38:45 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.LogError("Failed to pull first scrobble page for {0}/{1}", config.User.UserName, config.User.LastFmUsername);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-18 00:08:42 +00:00
|
|
|
|
private async Task<List<Task<PageResponse<LastTrack>>>> GetScrobblesFromPageNumbers(int start, int totalPages, CancellationToken token)
|
2022-02-16 23:38:45 +00:00
|
|
|
|
{
|
|
|
|
|
var tasks = new List<Task<PageResponse<LastTrack>>>();
|
|
|
|
|
|
2022-02-18 00:08:42 +00:00
|
|
|
|
foreach (var pageNumber in Enumerable.Range(start, totalPages - 1))
|
2022-02-16 23:38:45 +00:00
|
|
|
|
{
|
2022-02-18 00:08:42 +00:00
|
|
|
|
logger.LogInformation("Pulling page {2:n0}/{3:n0} for {0}/{1}", config.User.UserName, config.User.LastFmUsername, pageNumber, totalPages);
|
2022-02-16 23:38:45 +00:00
|
|
|
|
|
|
|
|
|
tasks.Add(userClient.GetRecentScrobbles(config.User.LastFmUsername, pagenumber: pageNumber, count: config.PageSize, from: config.From, to: config.To));
|
|
|
|
|
await Task.Delay(config.InterRequestDelay, token);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return tasks;
|
|
|
|
|
}
|
2022-02-18 19:47:11 +00:00
|
|
|
|
|
|
|
|
|
private void IdentifyDuplicates(IEnumerable<LastTrack> tracks)
|
|
|
|
|
{
|
|
|
|
|
logger.LogDebug("Identifying duplicates");
|
|
|
|
|
|
|
|
|
|
var duplicates = tracks
|
|
|
|
|
.GroupBy(t => t.TimePlayed?.UtcDateTime)
|
|
|
|
|
.Where(g => g.Count() > 1);
|
|
|
|
|
|
|
|
|
|
foreach(var dupe in duplicates)
|
|
|
|
|
{
|
|
|
|
|
var dupeString = new StringBuilder();
|
|
|
|
|
|
|
|
|
|
foreach(var scrobble in dupe)
|
|
|
|
|
{
|
|
|
|
|
dupeString.Append("(");
|
|
|
|
|
dupeString.Append(scrobble.Name);
|
|
|
|
|
dupeString.Append(", ");
|
|
|
|
|
dupeString.Append(scrobble.AlbumName);
|
|
|
|
|
dupeString.Append(", ");
|
|
|
|
|
dupeString.Append(scrobble.ArtistName);
|
|
|
|
|
dupeString.Append(")");
|
|
|
|
|
|
|
|
|
|
dupeString.Append(" ");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
logger.LogInformation("Duplicate at {0}: {1}", dupe.Key, dupeString.ToString());
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-02-16 23:38:45 +00:00
|
|
|
|
}
|
|
|
|
|
}
|