using HashidsNet; using Microsoft.EntityFrameworkCore; using Microsoft.Extensions.Logging; using Roadie.Library; using Roadie.Library.Caching; using Roadie.Library.Configuration; using Roadie.Library.Data.Context; using Roadie.Library.Encoding; using Roadie.Library.Enums; using Roadie.Library.Identity; using Roadie.Library.Utility; using System; using System.Collections.Generic; using System.Data.Common; using System.Diagnostics; using System.Linq; using System.Threading.Tasks; using data = Roadie.Library.Data; namespace Roadie.Api.Services { public abstract class ServiceBase { public static string TrackTokenSalt = "B0246908-FBD6-4E12-A96C-AF5B086115B3"; protected readonly ICacheManager _cacheManager; protected readonly IRoadieSettings _configuration; protected readonly IRoadieDbContext _dbContext; protected readonly IHttpContext _httpContext; protected readonly IHttpEncoder _httpEncoder; protected readonly ILogger _logger; private static readonly Lazy hashIds = new Lazy(() => { return new Hashids(TrackTokenSalt); }); protected ICacheManager CacheManager => _cacheManager; protected IRoadieSettings Configuration => _configuration; protected IRoadieDbContext DbContext => _dbContext; protected IHttpContext HttpContext => _httpContext; protected IHttpEncoder HttpEncoder => _httpEncoder; protected ILogger Logger => _logger; public ServiceBase(IRoadieSettings configuration, IHttpEncoder httpEncoder, IRoadieDbContext context, ICacheManager cacheManager, ILogger logger, IHttpContext httpContext) { _configuration = configuration; _httpEncoder = httpEncoder; _dbContext = context; _cacheManager = cacheManager; _logger = logger; _httpContext = httpContext; } public static bool ConfirmTrackPlayToken(User user, Guid trackRoadieId, string token) { if (string.IsNullOrEmpty(token)) { return false; } return TrackPlayToken(user, trackRoadieId).Equals(token); } public static string MakeTrackPlayUrl(User user, string baseUrl, Guid trackRoadieId) => $"{baseUrl}/play/track/{user.Id}/{TrackPlayToken(user, trackRoadieId)}/{trackRoadieId}.mp3"; public static string TrackPlayToken(User user, Guid trackId) { var trackIdPart = BitConverter.ToInt32(trackId.ToByteArray(), 6); if (trackIdPart < 0) { trackIdPart *= -1; } var token = hashIds.Value.Encode(user.Id, SafeParser.ToNumber(user.CreatedDate.Value.ToString("DDHHmmss")), trackIdPart); return token; } protected async Task GetArtist(string artistName) { if (string.IsNullOrEmpty(artistName)) { return null; } var artistByName = await CacheManager.GetAsync(data.Artist.CacheUrnByName(artistName), async () => { return await DbContext.Artists.FirstOrDefaultAsync(x => x.Name == artistName).ConfigureAwait(false); }, null).ConfigureAwait(false); if (artistByName == null) { return null; } return await GetArtist(artistByName.RoadieId); } protected Task GetArtist(Guid id) { if (id == Guid.Empty) { return Task.FromResult(null); } return CacheManager.GetAsync(data.Artist.CacheUrn(id), () => { return DbContext.Artists .Include(x => x.Genres) .Include("Genres.Genre") .FirstOrDefaultAsync(x => x.RoadieId == id); }, data.Artist.CacheRegionUrn(id)); } protected Task GetCollection(Guid id) { if (id == Guid.Empty) { return Task.FromResult(null); } return CacheManager.GetAsync(data.Collection.CacheUrn(id), () => { return DbContext.Collections.FirstOrDefaultAsync(x => x.RoadieId == id); }, data.Collection.CacheRegionUrn(id)); } protected Task GetGenre(Guid id) { if (id == Guid.Empty) { return Task.FromResult(null); } return CacheManager.GetAsync(data.Genre.CacheUrn(id), () => { return DbContext.Genres.FirstOrDefaultAsync(x => x.RoadieId == id); }, data.Genre.CacheRegionUrn(id)); } protected Task GetLabel(Guid id) { if (id == Guid.Empty) { return Task.FromResult(null); } return CacheManager.GetAsync(data.Label.CacheUrn(id), () => { return DbContext.Labels.FirstOrDefaultAsync(x => x.RoadieId == id); }, data.Label.CacheRegionUrn(id)); } protected Task GetPlaylist(Guid id) { if (id == Guid.Empty) { return Task.FromResult(null); } return CacheManager.GetAsync(data.Playlist.CacheUrn(id), () => { return DbContext.Playlists .Include(x => x.User) .FirstOrDefaultAsync(x => x.RoadieId == id); }, data.Playlist.CacheRegionUrn(id)); } protected Task GetRelease(Guid id) { if (id == Guid.Empty) { return Task.FromResult(null); } return CacheManager.GetAsync(data.Release.CacheUrn(id), () => { return DbContext.Releases .Include(x => x.Artist) .Include(x => x.Genres) .Include("Genres.Genre") .Include(x => x.Medias) .Include("Medias.Tracks") .Include("Medias.Tracks.TrackArtist") .FirstOrDefaultAsync(x => x.RoadieId == id); }, data.Release.CacheRegionUrn(id)); } /// /// Get Track by Subsonic Id ("T:guid") /// protected async Task GetTrack(string id) { if (Guid.TryParse(id, out Guid trackId)) { return await GetTrack(trackId); } return null; } // Only read operations protected Task GetTrack(Guid id) { if(id == Guid.Empty) { return Task.FromResult(null); } return CacheManager.GetAsync(data.Track.CacheUrn(id), () => { return DbContext.Tracks .Include(x => x.ReleaseMedia) .Include(x => x.ReleaseMedia.Release) .Include(x => x.ReleaseMedia.Release.Artist) .Include(x => x.TrackArtist) .FirstOrDefaultAsync(x => x.RoadieId == id); }, data.Track.CacheRegionUrn(id)); } protected async Task GetUser(string username) { if (string.IsNullOrEmpty(username)) { return null; } var userByUsername = await CacheManager.GetAsync(User.CacheUrnByUsername(username), () => { return DbContext.Users.FirstOrDefaultAsync(x => x.UserName == username); }, null); return await GetUser(userByUsername?.RoadieId).ConfigureAwait(false); } protected Task GetUser(Guid? id) { if (!id.HasValue) { return Task.FromResult(null); } return CacheManager.GetAsync(User.CacheUrn(id.Value), () => { return DbContext.Users .Include(x => x.UserRoles) .Include("UserRoles.Role") .Include("UserRoles.Role.RoleClaims") .Include(x => x.UserClaims) .Include(x => x.UserQues) .Include("UserQues.Track") .FirstOrDefaultAsync(x => x.RoadieId == id); }, User.CacheRegionUrn(id.Value)); } protected string MakeLastFmUrl(string artistName, string releaseTitle) => "http://www.last.fm/music/" + HttpEncoder.UrlEncode($"{artistName}/{releaseTitle}"); protected async Task> SetArtistRating(Guid artistId, User user, short rating) { var artist = DbContext.Artists .Include(x => x.Genres) .Include("Genres.Genre") .FirstOrDefault(x => x.RoadieId == artistId); if (artist == null) return new OperationResult(true, $"Invalid Artist Id [{artistId}]"); var now = DateTime.UtcNow; var userArtist = DbContext.UserArtists.FirstOrDefault(x => x.ArtistId == artist.Id && x.UserId == user.Id); if (userArtist == null) { userArtist = new data.UserArtist { Rating = rating, UserId = user.Id, ArtistId = artist.Id }; DbContext.UserArtists.Add(userArtist); } else { userArtist.Rating = rating; userArtist.LastUpdated = now; } await DbContext.SaveChangesAsync().ConfigureAwait(false); var ratings = await DbContext.UserArtists.Where(x => x.ArtistId == artist.Id && x.Rating > 0).Select(x => x.Rating).ToListAsync().ConfigureAwait(false); if (ratings != null && ratings.Any()) { artist.Rating = (short)ratings.Average(x => (decimal)x); } else { artist.Rating = 0; } artist.LastUpdated = now; await DbContext.SaveChangesAsync().ConfigureAwait(false); await UpdateArtistRank(artist.Id); CacheManager.ClearRegion(user.CacheRegion); CacheManager.ClearRegion(artist.CacheRegion); artist = await GetArtist(artistId).ConfigureAwait(false); return new OperationResult { IsSuccess = true, Data = artist.Rating ?? 0 }; } protected async Task> SetReleaseRating(Guid releaseId, User user, short rating) { var release = DbContext.Releases .Include(x => x.Artist) .Include(x => x.Genres) .Include("Genres.Genre") .Include(x => x.Medias) .Include("Medias.Tracks") .Include("Medias.Tracks.TrackArtist") .FirstOrDefault(x => x.RoadieId == releaseId); if (release == null) { return new OperationResult(true, $"Invalid Release Id [{releaseId}]"); } var userRelease = DbContext.UserReleases.FirstOrDefault(x => x.ReleaseId == release.Id && x.UserId == user.Id); var now = DateTime.UtcNow; if (userRelease == null) { userRelease = new data.UserRelease { Rating = rating, UserId = user.Id, ReleaseId = release.Id }; DbContext.UserReleases.Add(userRelease); } else { userRelease.Rating = rating; userRelease.LastUpdated = now; } await DbContext.SaveChangesAsync(); var ratings = await DbContext.UserReleases.Where(x => x.ReleaseId == release.Id && x.Rating > 0).Select(x => x.Rating).ToListAsync(); if (ratings != null && ratings.Any()) { release.Rating = (short)ratings.Average(x => (decimal)x); } else { release.Rating = 0; } release.LastUpdated = now; await DbContext.SaveChangesAsync(); await UpdateReleaseRank(release.Id); CacheManager.ClearRegion(user.CacheRegion); CacheManager.ClearRegion(release.CacheRegion); CacheManager.ClearRegion(release.Artist.CacheRegion); release = await GetRelease(releaseId); return new OperationResult { IsSuccess = true, Data = release.Rating ?? 0 }; } protected async Task> SetTrackRating(Guid trackId, User user, short rating) { var sw = Stopwatch.StartNew(); var track = DbContext.Tracks .Include(x => x.ReleaseMedia) .Include(x => x.ReleaseMedia.Release) .Include(x => x.ReleaseMedia.Release.Artist) .Include(x => x.TrackArtist) .FirstOrDefault(x => x.RoadieId == trackId); if (track == null) { return new OperationResult(true, $"Invalid Track Id [{trackId}]"); } var now = DateTime.UtcNow; var userTrack = DbContext.UserTracks.FirstOrDefault(x => x.TrackId == track.Id && x.UserId == user.Id); if (userTrack == null) { userTrack = new data.UserTrack { Rating = rating, UserId = user.Id, TrackId = track.Id }; DbContext.UserTracks.Add(userTrack); } else { userTrack.Rating = rating; userTrack.LastUpdated = now; } await DbContext.SaveChangesAsync(); var ratings = await DbContext.UserTracks.Where(x => x.TrackId == track.Id && x.Rating > 0).Select(x => x.Rating).ToListAsync(); if (ratings != null && ratings.Any()) { track.Rating = (short)ratings.Average(x => (double)x); } else { track.Rating = 0; } track.LastUpdated = now; await DbContext.SaveChangesAsync(); await UpdateReleaseRank(track.ReleaseMedia.Release.Id); CacheManager.ClearRegion(user.CacheRegion); CacheManager.ClearRegion(track.CacheRegion); CacheManager.ClearRegion(track.ReleaseMedia.Release.CacheRegion); CacheManager.ClearRegion(track.ReleaseMedia.Release.Artist.CacheRegion); if (track.TrackArtist != null) { CacheManager.ClearRegion(track.TrackArtist.CacheRegion); } sw.Stop(); return new OperationResult { IsSuccess = true, Data = track.Rating, OperationTime = sw.ElapsedMilliseconds }; } protected async Task> ToggleArtistDisliked(Guid artistId, User user, bool isDisliked) { var result = false; try { var artist = DbContext.Artists .Include(x => x.Genres) .Include("Genres.Genre") .FirstOrDefault(x => x.RoadieId == artistId); if (artist == null) return new OperationResult(true, $"Invalid Artist Id [{artistId}]"); var userArtist = DbContext.UserArtists.FirstOrDefault(x => x.ArtistId == artist.Id && x.UserId == user.Id); if (userArtist == null) { userArtist = new data.UserArtist { IsDisliked = isDisliked, UserId = user.Id, ArtistId = artist.Id }; DbContext.UserArtists.Add(userArtist); } else { userArtist.IsDisliked = isDisliked; userArtist.LastUpdated = DateTime.UtcNow; } await DbContext.SaveChangesAsync(); CacheManager.ClearRegion(user.CacheRegion); CacheManager.ClearRegion(artist.CacheRegion); result = true; } catch (DbException ex) { Logger.LogError($"ToggleArtistDisliked DbException [{ ex }]"); } return new OperationResult { IsSuccess = result, Data = result }; } protected async Task> ToggleArtistFavorite(Guid artistId, User user, bool isFavorite) { var result = false; try { var artist = DbContext.Artists .Include(x => x.Genres) .Include("Genres.Genre") .FirstOrDefault(x => x.RoadieId == artistId); if (artist == null) return new OperationResult(true, $"Invalid Artist Id [{artistId}]"); var userArtist = DbContext.UserArtists.FirstOrDefault(x => x.ArtistId == artist.Id && x.UserId == user.Id); if (userArtist == null) { userArtist = new data.UserArtist { IsFavorite = isFavorite, UserId = user.Id, ArtistId = artist.Id }; DbContext.UserArtists.Add(userArtist); } else { userArtist.IsFavorite = isFavorite; userArtist.LastUpdated = DateTime.UtcNow; } await DbContext.SaveChangesAsync(); CacheManager.ClearRegion(user.CacheRegion); CacheManager.ClearRegion(artist.CacheRegion); result = true; } catch (DbException ex) { Logger.LogError($"ToggleArtistFavorite DbException [{ ex }]"); } return new OperationResult { IsSuccess = result, Data = result }; } protected async Task> ToggleReleaseDisliked(Guid releaseId, User user, bool isDisliked) { var result = false; try { var release = DbContext.Releases .Include(x => x.Artist) .Include(x => x.Genres) .Include("Genres.Genre") .Include(x => x.Medias) .Include("Medias.Tracks") .Include("Medias.Tracks.TrackArtist") .FirstOrDefault(x => x.RoadieId == releaseId); if (release == null) { return new OperationResult(true, $"Invalid Release Id [{releaseId}]"); } var userRelease = DbContext.UserReleases.FirstOrDefault(x => x.ReleaseId == release.Id && x.UserId == user.Id); if (userRelease == null) { userRelease = new data.UserRelease { IsDisliked = isDisliked, UserId = user.Id, ReleaseId = release.Id }; DbContext.UserReleases.Add(userRelease); } else { userRelease.IsDisliked = isDisliked; userRelease.LastUpdated = DateTime.UtcNow; } await DbContext.SaveChangesAsync(); CacheManager.ClearRegion(user.CacheRegion); CacheManager.ClearRegion(release.CacheRegion); CacheManager.ClearRegion(release.Artist.CacheRegion); result = true; } catch (DbException ex) { Logger.LogError($"ToggleReleaseDisliked DbException [{ ex }]"); } return new OperationResult { IsSuccess = result, Data = result }; } protected async Task> ToggleReleaseFavorite(Guid releaseId, User user, bool isFavorite) { var result = false; try { var release = DbContext.Releases .Include(x => x.Artist) .Include(x => x.Genres) .Include("Genres.Genre") .Include(x => x.Medias) .Include("Medias.Tracks") .Include("Medias.Tracks.TrackArtist") .FirstOrDefault(x => x.RoadieId == releaseId); if (release == null) { return new OperationResult(true, $"Invalid Release Id [{releaseId}]"); } var userRelease = DbContext.UserReleases.FirstOrDefault(x => x.ReleaseId == release.Id && x.UserId == user.Id); if (userRelease == null) { userRelease = new data.UserRelease { IsFavorite = isFavorite, UserId = user.Id, ReleaseId = release.Id }; DbContext.UserReleases.Add(userRelease); } else { userRelease.IsFavorite = isFavorite; userRelease.LastUpdated = DateTime.UtcNow; } await DbContext.SaveChangesAsync(); CacheManager.ClearRegion(user.CacheRegion); CacheManager.ClearRegion(release.CacheRegion); CacheManager.ClearRegion(release.Artist.CacheRegion); result = true; } catch (DbException ex) { Logger.LogError($"ToggleReleaseFavorite DbException [{ ex }]"); } return new OperationResult { IsSuccess = result, Data = result }; } protected async Task> ToggleTrackDisliked(Guid trackId, User user, bool isDisliked) { var result = false; try { var track = DbContext.Tracks .Include(x => x.ReleaseMedia) .Include(x => x.ReleaseMedia.Release) .Include(x => x.ReleaseMedia.Release.Artist) .Include(x => x.TrackArtist) .FirstOrDefault(x => x.RoadieId == trackId); if (track == null) { return new OperationResult(true, $"Invalid Track Id [{trackId}]"); } var userTrack = DbContext.UserTracks.FirstOrDefault(x => x.TrackId == track.Id && x.UserId == user.Id); if (userTrack == null) { userTrack = new data.UserTrack { IsDisliked = isDisliked, UserId = user.Id, TrackId = track.Id }; DbContext.UserTracks.Add(userTrack); } else { userTrack.IsDisliked = isDisliked; userTrack.LastUpdated = DateTime.UtcNow; } await DbContext.SaveChangesAsync(); CacheManager.ClearRegion(user.CacheRegion); CacheManager.ClearRegion(track.CacheRegion); CacheManager.ClearRegion(track.ReleaseMedia.Release.CacheRegion); CacheManager.ClearRegion(track.ReleaseMedia.Release.Artist.CacheRegion); result = true; } catch (DbException ex) { Logger.LogError($"ToggleTrackDisliked DbException [{ ex }]"); } return new OperationResult { IsSuccess = result, Data = result }; } protected async Task> ToggleTrackFavorite(Guid trackId, User user, bool isFavorite) { var result = false; try { var track = DbContext.Tracks .Include(x => x.ReleaseMedia) .Include(x => x.ReleaseMedia.Release) .Include(x => x.ReleaseMedia.Release.Artist) .Include(x => x.TrackArtist) .FirstOrDefault(x => x.RoadieId == trackId); if (track == null) { return new OperationResult(true, $"Invalid Track Id [{trackId}]"); } var userTrack = DbContext.UserTracks.FirstOrDefault(x => x.TrackId == track.Id && x.UserId == user.Id); if (userTrack == null) { userTrack = new data.UserTrack { IsFavorite = isFavorite, UserId = user.Id, TrackId = track.Id }; DbContext.UserTracks.Add(userTrack); } else { userTrack.IsFavorite = isFavorite; userTrack.LastUpdated = DateTime.UtcNow; } await DbContext.SaveChangesAsync(); CacheManager.ClearRegion(user.CacheRegion); CacheManager.ClearRegion(track.CacheRegion); CacheManager.ClearRegion(track.ReleaseMedia.Release.CacheRegion); CacheManager.ClearRegion(track.ReleaseMedia.Release.Artist.CacheRegion); Logger.LogDebug($"ToggleTrackFavorite: User `{ user }`, Track `{ track }`, isFavorite: [{ isFavorite }]"); result = true; } catch (DbException ex) { Logger.LogError($"ToggleTrackFavorite DbException [{ ex }]"); } return new OperationResult { IsSuccess = result, Data = result }; } protected async Task UpdateArtistCounts(int artistId, DateTime now) { var artist = DbContext.Artists.FirstOrDefault(x => x.Id == artistId); if (artist != null) { artist.ReleaseCount = await DbContext.Releases.Where(x => x.ArtistId == artistId).CountAsync(); artist.TrackCount = await (from r in DbContext.Releases join rm in DbContext.ReleaseMedias on r.Id equals rm.ReleaseId join tr in DbContext.Tracks on rm.Id equals tr.ReleaseMediaId where tr.ArtistId == artistId || r.ArtistId == artistId select tr).CountAsync(); artist.LastUpdated = now; await DbContext.SaveChangesAsync(); CacheManager.ClearRegion(artist.CacheRegion); } } /// /// Update any playlist counts for any playlists with tracks for any release for this artist /// protected async Task UpdatePlaylistCountsForArtist(int artistId, DateTime now) { var playlistIds = await (from r in DbContext.Releases join rm in DbContext.ReleaseMedias on r.Id equals rm.ReleaseId join tr in DbContext.Tracks on rm.Id equals tr.ReleaseMediaId join plt in DbContext.PlaylistTracks on tr.Id equals plt.TrackId where r.ArtistId == artistId select plt.PlayListId).ToListAsync(); foreach (var playlistId in playlistIds.Distinct()) { await UpdatePlaylistCounts(playlistId, now); } } /// /// Update any playlist counts for any playlists with tracks on this release /// protected async Task UpdatePlaylistCountsForRelease(int releaseId, DateTime now) { var playlistIds = await (from r in DbContext.Releases join rm in DbContext.ReleaseMedias on r.Id equals rm.ReleaseId join tr in DbContext.Tracks on rm.Id equals tr.ReleaseMediaId join plt in DbContext.PlaylistTracks on tr.Id equals plt.TrackId where r.Id == releaseId select plt.PlayListId).ToListAsync(); foreach(var playlistId in playlistIds.Distinct()) { await UpdatePlaylistCounts(playlistId, now); } } /// /// Update the counts for all artists on a release (both track and release artists) /// protected async Task UpdateArtistCountsForRelease(int releaseId, DateTime now) { var trackArtistIds = await (from r in DbContext.Releases join rm in DbContext.ReleaseMedias on r.Id equals rm.ReleaseId join tr in DbContext.Tracks on rm.Id equals tr.ReleaseMediaId where r.Id == releaseId where tr.ArtistId != null select tr.ArtistId.Value).ToListAsync(); trackArtistIds.Add(DbContext.Releases.FirstOrDefault(x => x.Id == releaseId).ArtistId); foreach (var artistId in trackArtistIds.Distinct()) { await UpdateArtistCounts(artistId, now); } } /// /// Update Artist Rank /// Artist Rank is a sum of the artists release ranks + artist tracks rating + artist user rating /// protected async Task UpdateArtistRank(int artistId, bool updateReleaseRanks = false) { try { var artist = DbContext.Artists.FirstOrDefault(x => x.Id == artistId); if (artist != null) { if (updateReleaseRanks) { var artistReleaseIds = await DbContext.Releases.Where(x => x.ArtistId == artistId).Select(x => x.Id).ToArrayAsync(); foreach (var artistReleaseId in artistReleaseIds) { await UpdateReleaseRank(artistReleaseId, false); } } var artistTrackAverage = (await (from t in DbContext.Tracks join rm in DbContext.ReleaseMedias on t.ReleaseMediaId equals rm.Id join ut in DbContext.UserTracks on t.Id equals ut.TrackId where t.ArtistId == artist.Id select ut.Rating).Select(x => (decimal?)x).ToArrayAsync()).Average(); var artistReleaseRatingRating = (await (from r in DbContext.Releases join ur in DbContext.UserReleases on r.Id equals ur.ReleaseId where r.ArtistId == artist.Id select ur.Rating).Select(x => (decimal?)x).ToArrayAsync()).Average(); var artistReleaseRankSum = (await (from r in DbContext.Releases where r.ArtistId == artist.Id select r.Rank).ToArrayAsync()).Sum(x => x) ?? 0; artist.Rank = SafeParser.ToNumber(artistTrackAverage + artistReleaseRatingRating) + artistReleaseRankSum + artist.Rating; await DbContext.SaveChangesAsync(); CacheManager.ClearRegion(artist.CacheRegion); Logger.LogTrace("UpdatedArtistRank For Artist `{0}`", artist); } } catch (Exception ex) { Logger.LogError(ex, "Error in UpdateArtistRank ArtistId [{0}], UpdateReleaseRanks [{1}]", artistId, updateReleaseRanks); } } /// /// Find all artists involved with release and update their rank /// protected async Task UpdateArtistsRankForRelease(data.Release release) { if (release != null) { var artistsForRelease = new List { release.ArtistId }; var trackArtistsForRelease = await (from t in DbContext.Tracks join rm in DbContext.ReleaseMedias on t.ReleaseMediaId equals rm.Id where rm.ReleaseId == release.Id where t.ArtistId.HasValue select t.ArtistId.Value).ToArrayAsync(); artistsForRelease.AddRange(trackArtistsForRelease); foreach (var artistId in artistsForRelease.Distinct()) { await UpdateArtistRank(artistId); } } } protected async Task UpdateLabelCounts(int labelId, DateTime now) { var label = DbContext.Labels.FirstOrDefault(x => x.Id == labelId); if (label != null) { label.ReleaseCount = await DbContext.ReleaseLabels.Where(x => x.LabelId == label.Id).CountAsync(); label.ArtistCount = await (from r in DbContext.Releases join rl in DbContext.ReleaseLabels on r.Id equals rl.ReleaseId join a in DbContext.Artists on r.ArtistId equals a.Id where rl.LabelId == label.Id group a by a.Id into artists select artists).Select(x => x.Key).CountAsync(); label.TrackCount = await (from r in DbContext.Releases join rl in DbContext.ReleaseLabels on r.Id equals rl.ReleaseId join rm in DbContext.ReleaseMedias on r.Id equals rm.ReleaseId join t in DbContext.Tracks on rm.Id equals t.ReleaseMediaId where rl.LabelId == label.Id select t).CountAsync(); label.LastUpdated = now; await DbContext.SaveChangesAsync(); CacheManager.ClearRegion(label.CacheRegion); } } protected async Task UpdatePlaylistCounts(int playlistId, DateTime now) { var playlist = DbContext.Playlists.FirstOrDefault(x => x.Id == playlistId); if (playlist != null) { var playlistTracks = await DbContext.PlaylistTracks .Include(x => x.Track) .Include("Track.ReleaseMedia") .Where(x => x.PlayListId == playlist.Id).ToArrayAsync(); playlist.TrackCount = (short)playlistTracks.Count(); playlist.Duration = playlistTracks.Sum(x => x.Track.Duration); playlist.ReleaseCount = (short)playlistTracks.Select(x => x.Track.ReleaseMedia.ReleaseId).Distinct().Count(); playlist.LastUpdated = now; await DbContext.SaveChangesAsync(); CacheManager.ClearRegion(playlist.CacheRegion); } } protected async Task UpdateReleaseCounts(int releaseId, DateTime now) { var release = DbContext.Releases.FirstOrDefault(x => x.Id == releaseId); if (release != null) { release.PlayedCount = await (from t in DbContext.Tracks join rm in DbContext.ReleaseMedias on t.ReleaseMediaId equals rm.Id where rm.ReleaseId == releaseId where t.PlayedCount.HasValue select t).SumAsync(x => x.PlayedCount); release.Duration = await (from t in DbContext.Tracks join rm in DbContext.ReleaseMedias on t.ReleaseMediaId equals rm.Id where rm.ReleaseId == releaseId select t).SumAsync(x => x.Duration); release.LastUpdated = now; await DbContext.SaveChangesAsync(); CacheManager.ClearRegion(release.CacheRegion); } } /// /// Update Relase Rank /// Release Rank Calculation = Average of Track User Ratings + (User Rating of Release / Release Track Count) + /// Collection Rank Value /// protected async Task UpdateReleaseRank(int releaseId, bool updateArtistRank = true) { try { var release = DbContext.Releases.FirstOrDefault(x => x.Id == releaseId); if (release != null) { var releaseTrackAverage = (await (from ut in DbContext.UserTracks join t in DbContext.Tracks on ut.TrackId equals t.Id join rm in DbContext.ReleaseMedias on t.ReleaseMediaId equals rm.Id where rm.ReleaseId == releaseId select ut.Rating).Select(x => (decimal?)x).ToArrayAsync().ConfigureAwait(false)).Average(); var releaseUserRatingRank = release.Rating > 0 ? release.Rating / (decimal?)release.TrackCount : 0; var collectionsWithRelease = from c in DbContext.Collections join cr in DbContext.CollectionReleases on c.Id equals cr.CollectionId where c.CollectionType != CollectionType.Chart where cr.ReleaseId == release.Id select new { c.CollectionCount, cr.ListNumber }; decimal releaseCollectionRank = 0; foreach (var collectionWithRelease in collectionsWithRelease) { var rank = (decimal)(collectionWithRelease.CollectionCount * .01 - (collectionWithRelease.ListNumber - 1) * .01); releaseCollectionRank += rank; } release.Rank = SafeParser.ToNumber(releaseTrackAverage) + releaseUserRatingRank + releaseCollectionRank; await DbContext.SaveChangesAsync().ConfigureAwait(false); CacheManager.ClearRegion(release.CacheRegion); Logger.LogTrace("UpdateReleaseRank For Release `{0}`", release); if (updateArtistRank) { await UpdateArtistsRankForRelease(release).ConfigureAwait(false); } } } catch (Exception ex) { Logger.LogError(ex, "Error UpdateReleaseRank RelaseId [{0}], UpdateArtistRank [{1}]", releaseId, updateArtistRank); } } } }