sharp-chat/SharpChatCommon/EventStorage/MariaDBEventStorage.cs

139 lines
6.8 KiB
C#
Raw Normal View History

2022-08-30 15:00:58 +00:00
using MySqlConnector;
2024-05-24 03:44:20 +00:00
using SharpChat.Events;
2022-08-30 15:00:58 +00:00
using System;
using System.Collections.Generic;
2024-05-24 03:44:20 +00:00
using System.Reflection;
2022-08-30 15:00:58 +00:00
using System.Text;
using System.Text.Json;
namespace SharpChat.EventStorage {
public partial class MariaDBEventStorage : IEventStorage {
private string ConnectionString { get; }
2022-08-30 15:00:58 +00:00
2024-05-24 03:44:20 +00:00
private readonly JsonSerializerOptions jsonOpts = new() {
DefaultIgnoreCondition = System.Text.Json.Serialization.JsonIgnoreCondition.WhenWritingDefault,
};
private static readonly Dictionary<string, Type> EventDataTypes = new();
static MariaDBEventStorage() {
foreach(Type type in Assembly.GetExecutingAssembly().GetTypes()) {
ChatEventDataForAttribute? forAttr = type.GetCustomAttribute<ChatEventDataForAttribute>();
if(forAttr == null)
continue;
if(EventDataTypes.ContainsKey(forAttr.EventName))
throw new InvalidOperationException($"Attempted to register more than one data type for {forAttr.EventName}!");
EventDataTypes.Add(forAttr.EventName, type);
}
}
public MariaDBEventStorage(string connString) {
ConnectionString = connString;
2022-08-30 15:00:58 +00:00
}
2024-05-24 03:44:20 +00:00
public void HandleEvent(ChatEventInfo info) {
2022-08-30 15:00:58 +00:00
RunCommand(
"INSERT INTO sqc_events (event_id, event_type, event_created, event_channel, event_data"
+ ", event_sender, event_sender_name, event_sender_colour, event_sender_rank, event_sender_nick, event_sender_perms"
+ ") VALUES (@id, @type, FROM_UNIXTIME(@created), @channel, @data"
+ ", @senderId, @senderName, @senderColour, @senderRank, @senderNickName, @senderPerms)",
2024-05-24 03:44:20 +00:00
new MySqlParameter("id", info.Id),
new MySqlParameter("type", info.Type),
new MySqlParameter("created", info.Created.ToUnixTimeSeconds()),
new MySqlParameter("channel", string.IsNullOrWhiteSpace(info.ChannelName) ? null : info.ChannelName),
new MySqlParameter("data", JsonSerializer.Serialize(info.Data, info.Data.GetType(), jsonOpts)),
new MySqlParameter("senderId", info.SenderId < 1 ? null : info.SenderId),
new MySqlParameter("senderName", info.SenderName),
new MySqlParameter("senderColour", info.SenderColour.ToMisuzu()),
new MySqlParameter("senderRank", info.SenderRank),
new MySqlParameter("senderNickName", string.IsNullOrWhiteSpace(info.SenderNickName) ? null : info.SenderNickName),
new MySqlParameter("senderPerms", info.SenderPerms)
);
2024-05-24 03:44:20 +00:00
if(info.Type.Equals("msg:delete") && info.Data is MessageDeleteEventData msgDelete)
RunCommand(
"UPDATE IGNORE sqc_events SET event_deleted = NOW() WHERE event_id = @id AND event_deleted IS NULL",
2024-05-24 03:44:20 +00:00
new MySqlParameter("id", msgDelete.MessageId)
);
}
2024-05-24 03:44:20 +00:00
public ChatEventInfo? GetEvent(long eventId) {
try {
2024-05-10 19:18:55 +00:00
using MySqlDataReader? reader = RunQuery(
"SELECT event_id, event_type, event_channel, event_data"
+ ", event_sender, event_sender_name, event_sender_colour, event_sender_rank, event_sender_nick, event_sender_perms"
+ ", UNIX_TIMESTAMP(event_created) AS event_created, UNIX_TIMESTAMP(event_deleted) AS event_deleted"
+ " FROM sqc_events WHERE event_id = @id",
2024-05-24 03:44:20 +00:00
new MySqlParameter("id", eventId)
);
2024-05-10 19:18:55 +00:00
if(reader != null)
while(reader.Read()) {
2024-05-24 03:44:20 +00:00
ChatEventInfo evt = ReadEvent(reader);
2024-05-10 19:18:55 +00:00
if(evt != null)
return evt;
}
} catch(MySqlException ex) {
Logger.Write(ex);
}
return null;
2022-08-30 15:00:58 +00:00
}
2024-05-24 03:44:20 +00:00
private static ChatEventInfo ReadEvent(MySqlDataReader reader) {
string eventType = reader.GetString("event_type");
2024-05-24 03:44:20 +00:00
ChatEventData eventData = EventDataTypes.ContainsKey(eventType)
? (ChatEventData)(JsonSerializer.Deserialize(reader.GetString("event_data"), EventDataTypes[eventType]) ?? ChatEventData.EmptyInstance)
2024-05-24 03:44:20 +00:00
: ChatEventData.EmptyInstance;
return new ChatEventInfo(
reader.GetInt64("event_id"),
2024-05-24 03:44:20 +00:00
eventType,
DateTimeOffset.FromUnixTimeSeconds(reader.GetInt32("event_created")),
reader.GetString("event_channel"),
2024-05-24 03:44:20 +00:00
reader.IsDBNull(reader.GetOrdinal("event_sender")) ? -1 : reader.GetInt64("event_sender"),
reader.IsDBNull(reader.GetOrdinal("event_sender_name")) ? string.Empty : reader.GetString("event_sender_name"),
reader.IsDBNull(reader.GetOrdinal("event_sender_colour")) ? Colour.None : Colour.FromMisuzu(reader.GetInt32("event_sender_colour")),
reader.IsDBNull(reader.GetOrdinal("event_sender_rank")) ? 0 : reader.GetInt32("event_sender_rank"),
2024-05-24 03:44:20 +00:00
reader.IsDBNull(reader.GetOrdinal("event_sender_nick")) ? null : reader.GetString("event_sender_nick"),
(UserPermissions)(reader.IsDBNull(reader.GetOrdinal("event_sender_perms")) ? 0 : reader.GetInt32("event_sender_perms")),
2024-05-24 03:44:20 +00:00
eventData
);
2022-08-30 15:00:58 +00:00
}
public IEnumerable<ChatEventInfo> GetChannelEventLog(string channelName, int amount = 20, int after = 0) {
2024-05-24 03:44:20 +00:00
List<ChatEventInfo> events = new();
2022-08-30 15:00:58 +00:00
try {
2024-05-10 19:18:55 +00:00
using MySqlDataReader? reader = RunQuery(
"SELECT event_id, event_type, event_channel, event_data"
+ ", event_sender, event_sender_name, event_sender_colour, event_sender_rank, event_sender_nick, event_sender_perms"
+ ", UNIX_TIMESTAMP(event_created) AS event_created"
+ " FROM sqc_events WHERE event_deleted IS NULL"
+ " AND (event_channel IS NULL OR event_channel = @channel)"
+ " AND event_id > @after"
+ " ORDER BY event_id DESC"
+ " LIMIT @amount",
new MySqlParameter("channel", channelName),
new MySqlParameter("amount", amount),
new MySqlParameter("after", after)
2022-08-30 15:00:58 +00:00
);
2024-05-10 19:18:55 +00:00
if(reader != null)
while(reader.Read()) {
2024-05-24 03:44:20 +00:00
ChatEventInfo evt = ReadEvent(reader);
2024-05-10 19:18:55 +00:00
if(evt != null)
events.Add(evt);
}
2022-08-30 15:00:58 +00:00
} catch(MySqlException ex) {
Logger.Write(ex);
}
events.Reverse();
2022-08-30 15:00:58 +00:00
return events;
}
}
}