Files
KianaBH3/Common/Database/DatabaseHelper.cs
2025-06-14 11:15:32 +08:00

307 lines
9.3 KiB
C#
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

using KianaBH.Database.Account;
using KianaBH.Internationalization;
using KianaBH.Util;
using SqlSugar;
using System.Collections.Concurrent;
using System.Globalization;
namespace KianaBH.Database;
public class DatabaseHelper
{
public static Logger logger = new("Database");
public static SqlSugarScope? sqlSugarScope;
public static readonly ConcurrentDictionary<int, List<BaseDatabaseDataHelper>> UidInstanceMap = [];
public static readonly List<int> ToSaveUidList = [];
public static long LastSaveTick = DateTime.UtcNow.Ticks;
public static Thread? SaveThread;
public static bool LoadAccount;
public static bool LoadAllData;
public void Initialize()
{
logger.Info(I18NManager.Translate("Server.ServerInfo.LoadingItem", I18NManager.Translate("Word.Database")));
var f = new FileInfo(ConfigManager.Config.Path.DatabasePath + "/" + ConfigManager.Config.GameServer.DatabaseName);
if (!f.Exists && f.Directory != null) f.Directory.Create();
sqlSugarScope = new SqlSugarScope(new ConnectionConfig
{
ConnectionString = $"Data Source={f.FullName};",
DbType = DbType.Sqlite,
IsAutoCloseConnection = true,
ConfigureExternalServices = new ConfigureExternalServices
{
SerializeService = new CustomSerializeService()
}
});
InitializeSqlite();
var baseType = typeof(BaseDatabaseDataHelper);
var assembly = typeof(BaseDatabaseDataHelper).Assembly;
var types = assembly.GetTypes().Where(t => t.IsSubclassOf(baseType));
var list = sqlSugarScope.Queryable<AccountData>().ToList();
foreach (var inst in list)
{
if (!UidInstanceMap.TryGetValue(inst.Uid, out var value))
{
value = [];
UidInstanceMap[inst.Uid] = value;
}
value.Add(inst); // add to the map
}
// start dispatch server
LoadAccount = true;
var res = Parallel.ForEach(list, account =>
{
Parallel.ForEach(types, t =>
{
if (t == typeof(AccountData)) return; // skip the account data
try
{
typeof(DatabaseHelper).GetMethod(nameof(InitializeTable))?.MakeGenericMethod(t)
.Invoke(null, [account.Uid]);
}
catch (Exception e)
{
logger.Error("Database initialization error: ", e);
}
}); // cache the data
});
while (!res.IsCompleted)
{
}
LastSaveTick = DateTime.UtcNow.Ticks;
SaveThread = new Thread(() =>
{
while (true) CalcSaveDatabase();
});
SaveThread.Start();
LoadAllData = true;
}
public static void InitializeTable<T>(int uid) where T : BaseDatabaseDataHelper, new()
{
var list = sqlSugarScope?.Queryable<T>()
.Select(x => x)
.Select<T>()
.Where(x => x.Uid == uid)
.ToList();
foreach (var inst in list!.Select(instance => (instance as BaseDatabaseDataHelper)!))
{
if (!UidInstanceMap.TryGetValue(inst.Uid, out var value))
{
value = [];
UidInstanceMap[inst.Uid] = value;
}
value.Add(inst); // add to the map
}
}
public static void InitializeSqlite()
{
var baseType = typeof(BaseDatabaseDataHelper);
var assembly = typeof(BaseDatabaseDataHelper).Assembly;
var types = assembly.GetTypes().Where(t => t.IsSubclassOf(baseType));
foreach (var type in types)
typeof(DatabaseHelper).GetMethod("InitializeSqliteTable")?.MakeGenericMethod(type).Invoke(null, null);
}
// DO NOT DEL ReSharper disable once UnusedMember.Global
public static void InitializeSqliteTable<T>() where T : BaseDatabaseDataHelper, new()
{
try
{
sqlSugarScope?.CodeFirst.InitTables<T>();
}
catch
{
// ignored
}
}
public static T? GetInstance<T>(int uid, bool forceReload = false) where T : BaseDatabaseDataHelper, new()
{
try
{
if (!forceReload && UidInstanceMap.TryGetValue(uid, out var value))
{
var instance = value.OfType<T>().FirstOrDefault();
if (instance != null) return instance;
}
var t = sqlSugarScope?.Queryable<T>()
.Where(x => x.Uid == uid)
.ToList();
if (t is { Count: > 0 })
{
var instance = t[0];
if (!UidInstanceMap.TryGetValue(uid, out var list))
{
list = new List<BaseDatabaseDataHelper>();
UidInstanceMap[uid] = list;
}
else
{
list.RemoveAll(i => i is T);
}
list.Add(instance);
return instance;
}
return null;
return null;
}
catch (Exception e)
{
logger.Error("Unsupported type", e);
return null;
}
}
public static T GetInstanceOrCreateNew<T>(int uid) where T : BaseDatabaseDataHelper, new()
{
var instance = GetInstance<T>(uid);
if (instance != null) return instance;
instance = new T
{
Uid = uid
};
CreateInstance(instance);
return instance;
}
public static List<T>? GetAllInstance<T>() where T : BaseDatabaseDataHelper, new()
{
try
{
return sqlSugarScope?.Queryable<T>()
.Select(x => x)
.ToList();
}
catch (Exception e)
{
logger.Error("Unsupported type", e);
return null;
}
}
public static void UpdateInstance<T>(T instance) where T : BaseDatabaseDataHelper, new()
{
sqlSugarScope?.Updateable(instance).ExecuteCommand();
}
public static void CreateInstance<T>(T instance) where T : BaseDatabaseDataHelper, new()
{
sqlSugarScope?.Insertable(instance).ExecuteCommand();
if (!UidInstanceMap.TryGetValue(instance.Uid, out var value))
{
value = [];
UidInstanceMap[instance.Uid] = value;
}
value.Add(instance);
}
public static void DeleteInstance<T>(int key) where T : BaseDatabaseDataHelper, new()
{
try
{
sqlSugarScope?.Deleteable<T>().Where(x => x.Uid == key).ExecuteCommand();
}
catch (Exception e)
{
logger.Error("An error occurred while delete the database", e);
}
}
public static void DeleteAllInstance(int key)
{
var value = UidInstanceMap[key];
var baseType = typeof(BaseDatabaseDataHelper);
var assembly = typeof(BaseDatabaseDataHelper).Assembly;
var types = assembly.GetTypes().Where(t => t.IsSubclassOf(baseType));
foreach (var type in types)
{
var instance = value.Find(x => x.GetType() == type);
if (instance != null)
typeof(DatabaseHelper).GetMethod("DeleteInstance")?.MakeGenericMethod(type)
.Invoke(null, [key]);
}
if (UidInstanceMap.TryRemove(key, out var instances))
ToSaveUidList.RemoveAll(x => x == key);
}
// Auto save per 5 min
public static void CalcSaveDatabase()
{
if (LastSaveTick + TimeSpan.TicksPerMinute * 5 > DateTime.UtcNow.Ticks) return;
SaveDatabase();
}
public static void SaveDatabase()
{
try
{
var prev = DateTime.Now;
var list = ToSaveUidList.ToList(); // copy the list to avoid the exception
foreach (var uid in list)
{
var value = UidInstanceMap[uid];
var baseType = typeof(BaseDatabaseDataHelper);
var assembly = typeof(BaseDatabaseDataHelper).Assembly;
var types = assembly.GetTypes().Where(t => t.IsSubclassOf(baseType));
foreach (var type in types)
{
var instance = value.Find(x => x.GetType() == type);
if (instance != null)
typeof(DatabaseHelper).GetMethod("SaveDatabaseType")?.MakeGenericMethod(type)
.Invoke(null, [instance]);
}
}
var t = (DateTime.Now - prev).TotalSeconds;
logger.Info(I18NManager.Translate("Server.ServerInfo.SaveDatabase",
Math.Round(t, 2).ToString(CultureInfo.InvariantCulture)));
ToSaveUidList.Clear();
}
catch (Exception e)
{
logger.Error("An error occurred while saving the database", e);
}
LastSaveTick = DateTime.UtcNow.Ticks;
}
// DO NOT DEL ReSharper save database from cache
public static void SaveDatabaseType<T>(T instance) where T : BaseDatabaseDataHelper, new()
{
try
{
sqlSugarScope?.Updateable(instance).ExecuteCommand();
}
catch (Exception e)
{
logger.Error("An error occurred while saving the database", e);
}
}
}