mirror of
https://github.com/NoiTheCat/BirthdayBot.git
synced 2024-11-24 01:14:12 +00:00
Noi
2f0fe8641a
The BirthdayBot class has been split up into ShardInstance and ShardManager. Several other things have been reorganized so that shards may act independently. The overall goal of these changes made is to limit failures to sections that can easily be discarded and replaced.
94 lines
3.9 KiB
C#
94 lines
3.9 KiB
C#
using BirthdayBot.Data;
|
|
using NpgsqlTypes;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
|
|
namespace BirthdayBot.BackgroundServices
|
|
{
|
|
/// <summary>
|
|
/// Automatically removes database information for guilds that have not been accessed in a long time.
|
|
/// </summary>
|
|
class StaleDataCleaner : BackgroundService
|
|
{
|
|
private int _tickCount = 0;
|
|
|
|
public StaleDataCleaner(ShardInstance instance) : base(instance) { }
|
|
|
|
public override async Task OnTick(CancellationToken token)
|
|
{
|
|
if (++_tickCount % 20 != 0)
|
|
{
|
|
// Do not process on every tick.
|
|
return;
|
|
}
|
|
|
|
// Build a list of all values to update
|
|
var updateList = new Dictionary<ulong, List<ulong>>();
|
|
foreach (var g in ShardInstance.DiscordClient.Guilds)
|
|
{
|
|
// Get list of IDs for all users who exist in the database and currently exist in the guild
|
|
var savedUserIds = from cu in await GuildUserConfiguration.LoadAllAsync(g.Id) select cu.UserId;
|
|
var guildUserIds = from gu in g.Users select gu.Id;
|
|
var existingCachedIds = savedUserIds.Intersect(guildUserIds);
|
|
updateList[g.Id] = new List<ulong>(existingCachedIds);
|
|
}
|
|
|
|
using var db = await Database.OpenConnectionAsync();
|
|
|
|
// Statement for updating last_seen in guilds
|
|
var cUpdateGuild = db.CreateCommand();
|
|
cUpdateGuild.CommandText = $"update {GuildConfiguration.BackingTable} set last_seen = now() "
|
|
+ "where guild_id = @Gid";
|
|
var pUpdateG = cUpdateGuild.Parameters.Add("@Gid", NpgsqlDbType.Bigint);
|
|
cUpdateGuild.Prepare();
|
|
|
|
// Statement for updating last_seen in guild users
|
|
var cUpdateGuildUser = db.CreateCommand();
|
|
cUpdateGuildUser.CommandText = $"update {GuildUserConfiguration.BackingTable} set last_seen = now() "
|
|
+ "where guild_id = @Gid and user_id = @Uid";
|
|
var pUpdateGU_g = cUpdateGuildUser.Parameters.Add("@Gid", NpgsqlDbType.Bigint);
|
|
var pUpdateGU_u = cUpdateGuildUser.Parameters.Add("@Uid", NpgsqlDbType.Bigint);
|
|
cUpdateGuildUser.Prepare();
|
|
|
|
// Do actual updates
|
|
int updatedGuilds = 0;
|
|
int updatedUsers = 0;
|
|
foreach (var item in updateList)
|
|
{
|
|
var guild = item.Key;
|
|
var userlist = item.Value;
|
|
|
|
pUpdateG.Value = (long)guild;
|
|
updatedGuilds += await cUpdateGuild.ExecuteNonQueryAsync(token);
|
|
|
|
pUpdateGU_g.Value = (long)guild;
|
|
foreach (var userid in userlist)
|
|
{
|
|
pUpdateGU_u.Value = (long)userid;
|
|
updatedUsers += await cUpdateGuildUser.ExecuteNonQueryAsync(token);
|
|
}
|
|
}
|
|
Log($"Updated last-seen records: {updatedGuilds} guilds, {updatedUsers} users");
|
|
|
|
// Delete all old values - expects referencing tables to have 'on delete cascade'
|
|
using var t = db.BeginTransaction();
|
|
int staleGuilds, staleUsers;
|
|
using (var c = db.CreateCommand())
|
|
{
|
|
// Delete data for guilds not seen in 4 weeks
|
|
c.CommandText = $"delete from {GuildConfiguration.BackingTable} where (now() - interval '28 days') > last_seen";
|
|
staleGuilds = c.ExecuteNonQuery();
|
|
}
|
|
using (var c = db.CreateCommand())
|
|
{
|
|
// Delete data for users not seen in 8 weeks
|
|
c.CommandText = $"delete from {GuildUserConfiguration.BackingTable} where (now() - interval '56 days') > last_seen";
|
|
staleUsers = c.ExecuteNonQuery();
|
|
}
|
|
Log($"Will remove {staleGuilds} guilds, {staleUsers} users.");
|
|
t.Commit();
|
|
}
|
|
}
|
|
}
|