using System; using System.Collections.Generic; using System.Linq; using System.Text.Json; using System.Threading; using System.Threading.Tasks; using Data.Abstractions; using Data.Models; using Microsoft.EntityFrameworkCore; using Microsoft.Extensions.Logging; using SharedLibraryCore.Dtos; using SharedLibraryCore.Interfaces; using SharedLibraryCore.QueryHelper; using ILogger = Microsoft.Extensions.Logging.ILogger; namespace IW4MAdmin.Application.Misc; public class MetaServiceV2 : IMetaServiceV2 { private readonly IDictionary> _metaActions; private readonly IDatabaseContextFactory _contextFactory; private readonly ILogger _logger; public MetaServiceV2(ILogger logger, IDatabaseContextFactory contextFactory) { _logger = logger; _metaActions = new Dictionary>(); _contextFactory = contextFactory; } public async Task SetPersistentMeta(string metaKey, string metaValue, int clientId, CancellationToken token = default) { if (!ValidArgs(metaKey, clientId)) { return; } await using var context = _contextFactory.CreateContext(); var existingMeta = await context.EFMeta .Where(meta => meta.Key == metaKey) .Where(meta => meta.ClientId == clientId) .FirstOrDefaultAsync(token); if (existingMeta != null) { _logger.LogDebug("Updating existing meta with key {Key} and id {Id}", existingMeta.Key, existingMeta.MetaId); existingMeta.Value = metaValue; existingMeta.Updated = DateTime.UtcNow; } else { _logger.LogDebug("Adding new meta with key {Key}", metaKey); context.EFMeta.Add(new EFMeta { ClientId = clientId, Created = DateTime.UtcNow, Key = metaKey, Value = metaValue, }); } await context.SaveChangesAsync(token); } public async Task SetPersistentMetaValue(string metaKey, T metaValue, int clientId, CancellationToken token = default) where T : class { if (!ValidArgs(metaKey, clientId)) { return; } string serializedValue; try { serializedValue = JsonSerializer.Serialize(metaValue); } catch (Exception ex) { _logger.LogError(ex, "Could not serialize meta with key {Key}", metaKey); return; } await SetPersistentMeta(metaKey, serializedValue, clientId, token); } public async Task SetPersistentMetaForLookupKey(string metaKey, string lookupKey, int lookupId, int clientId, CancellationToken token = default) { if (!ValidArgs(metaKey, clientId)) { return; } await using var context = _contextFactory.CreateContext(); var lookupMeta = await context.EFMeta.FirstOrDefaultAsync(meta => meta.Key == lookupKey, token); if (lookupMeta is null) { _logger.LogWarning("No lookup meta exists for metaKey {MetaKey} and lookupKey {LookupKey}", metaKey, lookupKey); return; } var lookupValues = JsonSerializer.Deserialize>>(lookupMeta.Value); if (lookupValues is null) { return; } var foundLookup = lookupValues.FirstOrDefault(value => value.Id == lookupId); if (foundLookup is null) { _logger.LogWarning("No lookup meta found for provided lookup id {MetaKey}, {LookupKey}, {LookupId}", metaKey, lookupKey, lookupId); return; } _logger.LogDebug("Setting meta for lookup {MetaKey}, {LookupKey}, {LookupId}", metaKey, lookupKey, lookupId); await SetPersistentMeta(metaKey, lookupId.ToString(), clientId, token); } public async Task IncrementPersistentMeta(string metaKey, int incrementAmount, int clientId, CancellationToken token = default) { if (!ValidArgs(metaKey, clientId)) { return; } var existingMeta = await GetPersistentMeta(metaKey, clientId, token); if (!long.TryParse(existingMeta?.Value ?? "0", out var existingValue)) { existingValue = 0; } var newValue = existingValue + incrementAmount; await SetPersistentMeta(metaKey, newValue.ToString(), clientId, token); } public async Task DecrementPersistentMeta(string metaKey, int decrementAmount, int clientId, CancellationToken token = default) { await IncrementPersistentMeta(metaKey, -decrementAmount, clientId, token); } public async Task GetPersistentMeta(string metaKey, int clientId, CancellationToken token = default) { if (!ValidArgs(metaKey, clientId)) { return null; } await using var ctx = _contextFactory.CreateContext(enableTracking: false); return await ctx.EFMeta .Where(meta => meta.Key == metaKey) .Where(meta => meta.ClientId == clientId) .Select(meta => new EFMeta { MetaId = meta.MetaId, Key = meta.Key, ClientId = meta.ClientId, Value = meta.Value, }) .FirstOrDefaultAsync(token); } public async Task GetPersistentMetaValue(string metaKey, int clientId, CancellationToken token = default) where T : class { var meta = await GetPersistentMeta(metaKey, clientId, token); if (meta is null) { return default; } try { return JsonSerializer.Deserialize(meta.Value); } catch (Exception ex) { _logger.LogError(ex, "Could not deserialize meta with key {Key} and value {Value}", metaKey, meta.Value); return default; } } public async Task GetPersistentMetaByLookup(string metaKey, string lookupKey, int clientId, CancellationToken token = default) { await using var context = _contextFactory.CreateContext(); var metaValue = await GetPersistentMeta(metaKey, clientId, token); if (metaValue is null) { _logger.LogDebug("No meta exists for key {Key}, clientId {ClientId}", metaKey, clientId); return default; } var lookupMeta = await context.EFMeta.FirstOrDefaultAsync(meta => meta.Key == lookupKey, token); if (lookupMeta is null) { _logger.LogWarning("No lookup meta exists for metaKey {MetaKey} and lookupKey {LookupKey}", metaKey, lookupKey); return default; } var lookupId = int.Parse(metaValue.Value); var lookupValues = JsonSerializer.Deserialize>>(lookupMeta.Value); if (lookupValues is null) { return default; } var foundLookup = lookupValues.FirstOrDefault(value => value.Id == lookupId); if (foundLookup is not null) { return new EFMeta { Created = metaValue.Created, Updated = metaValue.Updated, Extra = metaValue.Extra, MetaId = metaValue.MetaId, Value = foundLookup.Value }; } _logger.LogWarning("No lookup meta found for provided lookup id {MetaKey}, {LookupKey}, {LookupId}", metaKey, lookupKey, lookupId); return default; } public async Task RemovePersistentMeta(string metaKey, int clientId, CancellationToken token = default) { if (!ValidArgs(metaKey, clientId)) { return; } await using var context = _contextFactory.CreateContext(); var existingMeta = await context.EFMeta .FirstOrDefaultAsync(meta => meta.Key == metaKey && meta.ClientId == clientId, token); if (existingMeta == null) { _logger.LogDebug("No meta with key {Key} found for client id {Id}", metaKey, clientId); return; } _logger.LogDebug("Removing meta for key {Key} with id {Id}", metaKey, existingMeta.MetaId); context.EFMeta.Remove(existingMeta); await context.SaveChangesAsync(token); } public async Task SetPersistentMeta(string metaKey, string metaValue, CancellationToken token = default) { if (string.IsNullOrWhiteSpace(metaKey)) { _logger.LogWarning("Cannot save meta with no key"); return; } await using var ctx = _contextFactory.CreateContext(); var existingMeta = await ctx.EFMeta .Where(meta => meta.Key == metaKey) .Where(meta => meta.ClientId == null) .FirstOrDefaultAsync(token); if (existingMeta is not null) { _logger.LogDebug("Updating existing meta with key {Key} and id {Id}", existingMeta.Key, existingMeta.MetaId); existingMeta.Value = metaValue; existingMeta.Updated = DateTime.UtcNow; await ctx.SaveChangesAsync(token); } else { _logger.LogDebug("Adding new meta with key {Key}", metaKey); ctx.EFMeta.Add(new EFMeta { Created = DateTime.UtcNow, Key = metaKey, Value = metaValue }); await ctx.SaveChangesAsync(token); } } public async Task SetPersistentMetaValue(string metaKey, T metaValue, CancellationToken token = default) where T : class { if (string.IsNullOrWhiteSpace(metaKey)) { _logger.LogWarning("Meta key is null, not setting"); return; } if (metaValue is null) { _logger.LogWarning("Meta value is null, not setting"); return; } string serializedMeta; try { serializedMeta = JsonSerializer.Serialize(metaValue); } catch (Exception ex) { _logger.LogError(ex, "Could not serialize meta with {Key} and value {Value}", metaKey, metaValue); return; } await SetPersistentMeta(metaKey, serializedMeta, token); } public async Task GetPersistentMeta(string metaKey, CancellationToken token = default) { if (string.IsNullOrWhiteSpace(metaKey)) { return null; } await using var context = _contextFactory.CreateContext(false); return await context.EFMeta.FirstOrDefaultAsync(meta => meta.Key == metaKey, token); } public async Task GetPersistentMetaValue(string metaKey, CancellationToken token = default) where T : class { if (string.IsNullOrWhiteSpace(metaKey)) { return default; } var meta = await GetPersistentMeta(metaKey, token); if (meta is null) { return default; } try { return JsonSerializer.Deserialize(meta.Value); } catch (Exception ex) { _logger.LogError(ex, "Could not serialize meta with key {Key} and value {Value}", metaKey, meta.Value); return default; } } public async Task RemovePersistentMeta(string metaKey, CancellationToken token = default) { if (string.IsNullOrWhiteSpace(metaKey)) { return; } await using var context = _contextFactory.CreateContext(enableTracking: false); var existingMeta = await context.EFMeta .Where(meta => meta.Key == metaKey) .Where(meta => meta.ClientId == null) .FirstOrDefaultAsync(token); if (existingMeta != null) { _logger.LogDebug("Removing meta for key {Key} with id {Id}", metaKey, existingMeta.MetaId); context.Remove(existingMeta); await context.SaveChangesAsync(token); } } public void AddRuntimeMeta(MetaType metaKey, Func>> metaAction) where T : PaginationRequest where TReturnType : IClientMeta { if (!_metaActions.ContainsKey(metaKey)) { _metaActions.Add(metaKey, new List { metaAction }); } else { _metaActions[metaKey].Add(metaAction); } } public async Task> GetRuntimeMeta(ClientPaginationRequest request, CancellationToken token = default) { var metas = await Task.WhenAll(_metaActions.Where(kvp => kvp.Key != MetaType.Information) .Select(async kvp => await kvp.Value[0](request, token))); return metas.SelectMany(m => (IEnumerable)m) .OrderByDescending(m => m.When) .Take(request.Count) .ToList(); } public async Task> GetRuntimeMeta(ClientPaginationRequest request, MetaType metaType, CancellationToken token = default) where T : IClientMeta { if (metaType == MetaType.Information) { var allMeta = new List(); var completedMeta = await Task.WhenAll(_metaActions[metaType].Select(async individualMetaRegistration => (IEnumerable)await individualMetaRegistration(request, token))); allMeta.AddRange(completedMeta.SelectMany(meta => meta)); return ProcessInformationMeta(allMeta); } var meta = await _metaActions[metaType][0](request, token) as IEnumerable; return meta; } private static IEnumerable ProcessInformationMeta(IEnumerable meta) where T : IClientMeta { var metaList = meta.ToList(); var metaWithColumn = metaList .Where(m => m.Column != null) .ToList(); var columnGrouping = metaWithColumn .GroupBy(m => m.Column) .ToList(); var metaToSort = metaList.Except(metaWithColumn).ToList(); var table = columnGrouping.Select(metaItem => new List(metaItem)).ToList(); while (metaToSort.Count > 0) { var sortingMeta = metaToSort.First(); int IndexOfSmallestColumn() { var index = 0; var smallestColumnSize = int.MaxValue; for (var i = 0; i < table.Count; i++) { if (table[i].Count >= smallestColumnSize) { continue; } smallestColumnSize = table[i].Count; index = i; } return index; } var columnIndex = IndexOfSmallestColumn(); sortingMeta.Column = columnIndex; sortingMeta.Order = columnGrouping .First(group => group.Key == columnIndex) .Count(); table[columnIndex].Add(sortingMeta); metaToSort.Remove(sortingMeta); } return metaList; } private static bool ValidArgs(string key, int clientId) => !string.IsNullOrWhiteSpace(key) && clientId > 0; }