using DysonNetwork.Sphere.Storage; using Microsoft.EntityFrameworkCore; namespace DysonNetwork.Sphere.Sticker; public class StickerService(AppDatabase db, FileService fs, ICacheService cache) { private static readonly TimeSpan CacheDuration = TimeSpan.FromMinutes(15); public async Task CreateStickerAsync(Sticker sticker) { db.Stickers.Add(sticker); await db.SaveChangesAsync(); await fs.MarkUsageAsync(sticker.Image, 1); return sticker; } public async Task UpdateStickerAsync(Sticker sticker, CloudFile? newImage) { if (newImage != null) { await fs.MarkUsageAsync(sticker.Image, -1); sticker.Image = newImage; await fs.MarkUsageAsync(sticker.Image, 1); } db.Stickers.Update(sticker); await db.SaveChangesAsync(); // Invalidate cache for this sticker await PurgeStickerCache(sticker); return sticker; } public async Task DeleteStickerAsync(Sticker sticker) { db.Stickers.Remove(sticker); await db.SaveChangesAsync(); await fs.MarkUsageAsync(sticker.Image, -1); // Invalidate cache for this sticker await PurgeStickerCache(sticker); } public async Task DeleteStickerPackAsync(StickerPack pack) { var stickers = await db.Stickers .Include(s => s.Image) .Where(s => s.PackId == pack.Id) .ToListAsync(); var images = stickers.Select(s => s.Image).ToList(); db.Stickers.RemoveRange(stickers); db.StickerPacks.Remove(pack); await db.SaveChangesAsync(); await fs.MarkUsageRangeAsync(images, -1); // Invalidate cache for all stickers in this pack foreach (var sticker in stickers) await PurgeStickerCache(sticker); } public async Task LookupStickerByIdentifierAsync(string identifier) { identifier = identifier.ToLower(); // Try to get from the cache first var cacheKey = $"StickerLookup_{identifier}"; var cachedSticker = await cache.GetAsync(cacheKey); if (cachedSticker is not null) return cachedSticker; // If not in cache, fetch from the database IQueryable query = db.Stickers .Include(e => e.Pack) .Include(e => e.Image); query = Guid.TryParse(identifier, out var guid) ? query.Where(e => e.Id == guid) : query.Where(e => (e.Pack.Prefix + e.Slug).ToLower() == identifier); var sticker = await query.FirstOrDefaultAsync(); // Store in cache if found if (sticker != null) await cache.SetAsync(cacheKey, sticker, CacheDuration); return sticker; } private async Task PurgeStickerCache(Sticker sticker) { // Remove both possible cache entries await cache.RemoveAsync($"StickerLookup_{sticker.Id}"); await cache.RemoveAsync($"StickerLookup_{sticker.Pack.Prefix}{sticker.Slug}"); } }