mirror of
https://github.com/raphaeIl/Novaria.git
synced 2025-12-12 14:34:38 +01:00
113 lines
5.0 KiB
C#
113 lines
5.0 KiB
C#
using Google.Protobuf;
|
|
using Google.Protobuf.Collections;
|
|
using Ionic.Zip;
|
|
using Nova.Client;
|
|
using Novaria.Common.Util;
|
|
using Serilog;
|
|
using System.IO;
|
|
using System.Reflection;
|
|
using System.Runtime.Serialization.Formatters;
|
|
|
|
namespace Novaria.GameServer.Services
|
|
{
|
|
// note that all operations in this class MUST be thread-safe, idk if it is currently but if it's not, there will be race condition problems
|
|
public class TableService(ILogger<TableService> _logger)
|
|
{
|
|
private readonly ILogger<TableService> logger = _logger;
|
|
|
|
// example: maps Achievement (type) -> table_Achievement (instance), table_Achievement has a list that stores table
|
|
private readonly Dictionary<Type, IMessage> caches = [];
|
|
public static string ResourceDir = Path.Join(Path.GetDirectoryName(AppContext.BaseDirectory), "Resources");
|
|
|
|
private Type currentTableTypeCache; // this is the type that we're currently loading, ex. Achievement (type), reason for caching it in a field member? Nova uses hardcoded params Action<int, byte[]> so im too lazy to change
|
|
|
|
/// <summary>
|
|
/// Please <b>only</b> use this to get table that <b>have a respective file</b> (i.e. <c>CharacterExcelTable</c> have <c>characterexceltable.bytes</c>)
|
|
/// </summary>
|
|
/// <typeparam name="T"></typeparam>
|
|
/// <returns></returns>
|
|
/// <exception cref="FileNotFoundException"></exception>
|
|
public IMessage GetTable<T>(bool bypassCache = false) where T : IMessage
|
|
{
|
|
var type = typeof(T);
|
|
currentTableTypeCache = type;
|
|
|
|
if (!bypassCache && caches.TryGetValue(type, out var cache))
|
|
return (T)cache;
|
|
|
|
var tableDir = Path.Combine(ResourceDir, "Tables");
|
|
var bytesFilePath = Path.Combine(tableDir, $"{typeof(T).Name}.bytes");
|
|
|
|
if (!File.Exists(bytesFilePath))
|
|
{
|
|
throw new FileNotFoundException($"bytes files for {type.Name} not found");
|
|
}
|
|
|
|
//TableEncryptionService.XOR(type.Name, bytes);
|
|
|
|
this.LoadCommonBin<T>(bytesFilePath); // after this, loaded table will be in the cache
|
|
|
|
logger.LogDebug("{Excel} loaded and cached", type.Name);
|
|
|
|
return caches[currentTableTypeCache];
|
|
}
|
|
|
|
private void LoadCommonBin<T>(string bytesFilePath)
|
|
{
|
|
currentTableTypeCache = typeof(T); // too lazy to change actions params, so do this
|
|
|
|
// get the table_XXX type
|
|
Type table_Type = Assembly.GetAssembly(typeof(table_Achievement)).GetTypes().Where(t => t.Name == $"table_{typeof(T).Name}").FirstOrDefault();
|
|
|
|
if (table_Type == null)
|
|
{
|
|
Log.Error($"table_{typeof(T).Name} type was not found.");
|
|
return;
|
|
}
|
|
|
|
var inst = (IMessage)Activator.CreateInstance(table_Type);
|
|
caches[currentTableTypeCache] = inst;
|
|
|
|
GameDataController.Instance.LoadCommonBinData(bytesFilePath, new Action<int, byte[]>(this.AddCommonBin),
|
|
new Action<string, byte[]>(this.AddCommonBin),
|
|
new Action<long, byte[]>(this.AddCommonBin),
|
|
new Action<byte[]>(this.AddCommonBin));
|
|
}
|
|
|
|
private void AddCommonBin(int _, byte[] data) { AddCommonBin_internal(_, data); }
|
|
private void AddCommonBin(string _, byte[] data) { AddCommonBin_internal(_, data); }
|
|
private void AddCommonBin(long _, byte[] data) { AddCommonBin_internal(_, data); }
|
|
private void AddCommonBin(byte[] data) { AddCommonBin_internal(null, data); }
|
|
|
|
private void AddCommonBin_internal(object _, byte[] data)
|
|
{
|
|
var parserProperty = currentTableTypeCache.GetProperties().Where(p => p.Name == "Parser").SingleOrDefault();
|
|
var parserInstance = parserProperty.GetValue(null);
|
|
|
|
var parsedData = typeof(MessageParser).GetMethods().Where(m => m.Name == "ParseFrom").FirstOrDefault().Invoke(parserInstance, new object[] { data }); ;
|
|
// add to target table, very inefficient rn
|
|
IMessage targetTable = caches[currentTableTypeCache];
|
|
var __ = currentTableTypeCache.GetProperties();
|
|
|
|
|
|
var listField = targetTable.GetType().GetProperties().Where(p => p.Name == "List").SingleOrDefault();
|
|
|
|
object repeatedFieldInstance = listField.GetValue(targetTable);
|
|
|
|
var addMethod = repeatedFieldInstance.GetType().GetMethods().Where(m => m.Name == "Add").FirstOrDefault();
|
|
|
|
addMethod.Invoke(repeatedFieldInstance, new object[] { parsedData });
|
|
|
|
//Log.Information($"Added {_}");
|
|
}
|
|
}
|
|
|
|
internal static class ExcelTableServiceExtensions
|
|
{
|
|
public static void AddExcelTableService(this IServiceCollection services)
|
|
{
|
|
services.AddSingleton<TableService>();
|
|
}
|
|
}
|
|
}
|