using System.Reflection; using System.Text.Json; using System.Text.Json.Serialization; using Core.Annotations; using Core.Utils.Json.Converters; namespace Core.Utils; [Injectable(InjectionType.Singleton)] public class ImporterUtil { private readonly FileUtil _fileUtil; private readonly HashSet filesToIgnore = ["bearsuits.json", "usecsuits.json", "archivedquests.json"]; private readonly JsonSerializerOptions jsonSerializerOptions = new JsonSerializerOptions { UnmappedMemberHandling = JsonUnmappedMemberHandling.Disallow, Converters = { new ListOrTConverterFactory(), new DictionaryOrListConverter() } }; public ImporterUtil(FileUtil fileUtil) { _fileUtil = fileUtil; } /** * Load files into js objects recursively (asynchronous) * @param filepath Path to folder with files * @returns Promise return T type associated with this class */ public Task LoadRecursiveAsync( string filepath, Type loadedType, Action? onReadCallback = null, Action? onObjectDeserialized = null ) { var tasks = new List(); var result = Activator.CreateInstance(loadedType); // get all filepaths var files = _fileUtil.GetFiles(filepath); var directories = _fileUtil.GetDirectories(filepath); // add file content to result foreach (var file in files) { if (_fileUtil.GetFileExtension(file) != "json") continue; if (filesToIgnore.Contains(Path.GetFileName(file).ToLower())) continue; tasks.Add( Task.Factory.StartNew(() => { // const filename = this.vfs.stripExtension(file); // const filePathAndName = `${filepath}${file}`; var fileData = File.ReadAllText(file); if (onReadCallback != null) onReadCallback(file, fileData); var setMethod = GetSetMethod( Path.GetFileNameWithoutExtension(file).ToLower(), loadedType, out var propertyType, out var isDictionary ); try { var fileDeserialized = JsonSerializer.Deserialize(fileData, propertyType, jsonSerializerOptions); if (onObjectDeserialized != null) onObjectDeserialized(file, fileDeserialized); setMethod.Invoke(result, isDictionary ? [Path.GetFileNameWithoutExtension(file), fileDeserialized] : [fileDeserialized]); } catch (Exception e) { throw new Exception($"Unable to deserialize or set properties for file '{file}'", e); } }) ); } // deep tree search foreach (var directory in directories) { var dictionaryLock = new object(); tasks.Add( Task.Factory.StartNew(() => { var setMethod = GetSetMethod(directory.Split("/").Last().Replace("_", ""), loadedType, out var matchedProperty, out var isDictionary); var loadTask = LoadRecursiveAsync($"{directory}/", matchedProperty); loadTask.Wait(); lock (dictionaryLock) { setMethod.Invoke(result, isDictionary ? [directory, loadTask.Result] : [loadTask.Result]); } }) ); } // return the result of all async fetch return Task.WhenAll(tasks).ContinueWith((t) => { if (t.IsCanceled || t.IsFaulted) tasks.Where(t => t.IsFaulted || t.IsCanceled).ToList().ForEach(t => Console.WriteLine(t.Exception)); }).ContinueWith(_ => result); } public MethodInfo GetSetMethod(string propertyName, Type type, out Type propertyType, out bool isDictionary) { MethodInfo setMethod; isDictionary = false; if (type.IsGenericType && type.GetGenericTypeDefinition() == typeof(Dictionary<,>)) { propertyType = type.GetGenericArguments()[1]; setMethod = type.GetMethod("Add"); isDictionary = true; } else { var matchedProperty = type.GetProperties() .FirstOrDefault(prop => prop.Name.ToLower() == Path.GetFileNameWithoutExtension(propertyName).ToLower()); if (matchedProperty == null) throw new Exception( $"Unable to find property '{Path.GetFileNameWithoutExtension(propertyName)}' for type '{type.Name}'"); propertyType = matchedProperty.PropertyType; setMethod = matchedProperty.GetSetMethod(); } return setMethod; } /** * Load files into js objects recursively (synchronous) * @param filepath Path to folder with files * @returns */ public object LoadRecursive( string filepath, Type loadedType, Action? onReadCallback = null, Action? onObjectDeserialized = null ) { var result = Activator.CreateInstance(loadedType); // get all filepaths var files = Directory.GetFiles(filepath); var directories = Directory.GetDirectories(filepath); foreach (var file in files) if (Path.GetExtension(file) == "json") { // const filename = this.vfs.stripExtension(file); // const filePathAndName = `${filepath}${file}`; var fileData = File.ReadAllText(file); onReadCallback(file, fileData); var matchedProperty = loadedType.GetProperties() .FirstOrDefault(prop => prop.Name.ToLower() == Path.GetFileNameWithoutExtension(file).ToLower()); if (matchedProperty == null) throw new Exception($"Unable to find property '{Path.GetFileNameWithoutExtension(file)}' for type '{loadedType.Name}'"); var propertyType = matchedProperty.PropertyType; var fileDeserialized = JsonSerializer.Deserialize(fileData, propertyType); onObjectDeserialized(file, fileDeserialized); matchedProperty.GetSetMethod().Invoke(result, [fileDeserialized]); } // deep tree search foreach (var directory in directories) { var matchedProperty = loadedType.GetProperties().FirstOrDefault(prop => prop.Name.ToLower() == directory.ToLower()); if (matchedProperty == null) throw new Exception($"Unable to find property '{directory}' for type '{loadedType.Name}'"); matchedProperty.GetSetMethod().Invoke(result, [LoadRecursive($"{filepath}{directory}/", matchedProperty.PropertyType)]); } // return the result of all async fetch return result; } public async Task LoadAsync( string filepath, Type loadedType, string strippablePath = "", Action? onReadCallback = null, Action? onObjectDeserialized = null ) { var result = Activator.CreateInstance(loadedType); var promises = new List>(); var filesToProcess = new Queue(_fileUtil.GetFiles(filepath, true)); while (filesToProcess.Count != 0) { var fileNode = filesToProcess.Dequeue(); if (fileNode == null || _fileUtil.GetFileExtension(fileNode) != "json") continue; promises.Add(File.ReadAllTextAsync(fileNode).ContinueWith(fd => { onReadCallback(fileNode, fd.Result); return JsonSerializer.Deserialize(fd.Result, typeof(object)); })); /* this.vfs .readFileAsync(filePathAndName) .then(async (fileData) => { onReadCallback(filePathAndName, fileData); return this.jsonUtil.deserializeWithCacheCheckAsync(fileData, filePathAndName); }) .then(async (fileDeserialized) => { onObjectDeserialized(filePathAndName, fileDeserialized); const strippedFilePath = this.vfs.stripExtension(filePathAndName).replace(filepath, ""); this.placeObject(fileDeserialized, strippedFilePath, result, strippablePath); }) .then(() => progressWriter.increment()), );*/ } //await JSType.Promise<>.all(promises).catch((e) => console.error(e)); return result; } /* protected placeObject(fileDeserialized: any, strippedFilePath: string, result: T, strippablePath: string): void { const strippedFinalPath = strippedFilePath.replace(strippablePath, ""); let temp = result; const propertiesToVisit = strippedFinalPath.split("/"); for (let i = 0; i < propertiesToVisit.length; i++) { const property = propertiesToVisit[i]; if (i === propertiesToVisit.length - 1) { temp[property] = fileDeserialized; } else { if (!temp[property]) { temp[property] = {}; } temp = temp[property]; } } }*/ }