using System; using System.Collections.Concurrent; using System.IO; using System.Linq; using System.Runtime.InteropServices; using System.Threading.Tasks; using RomRepoMgr.Database; using RomRepoMgr.Database.Models; namespace RomRepoMgr.Core.Filesystem { public class Vfs : IDisposable { readonly ConcurrentDictionary> _machinesStatCache; readonly ConcurrentDictionary _romSetsCache; Fuse _fuse; ConcurrentDictionary _rootDirectoryCache; Winfsp _winfsp; public Vfs() { _rootDirectoryCache = new ConcurrentDictionary(); _romSetsCache = new ConcurrentDictionary(); _machinesStatCache = new ConcurrentDictionary>(); } public static bool IsAvailable => Winfsp.IsAvailable || Fuse.IsAvailable; public void Dispose() => _fuse?.Dispose(); public event EventHandler Umounted; public void MountTo(string mountPoint) { if(Fuse.IsAvailable) { _fuse = new Fuse(this) { MountPoint = mountPoint }; Task.Run(() => { _fuse.Start(); Umounted?.Invoke(this, System.EventArgs.Empty); }); } else if(Winfsp.IsAvailable) { _winfsp = new Winfsp(this); bool ret = _winfsp.Mount(mountPoint); if(ret) return; _winfsp = null; Umounted?.Invoke(this, System.EventArgs.Empty); } else Umounted?.Invoke(this, System.EventArgs.Empty); } public void Umount() { _fuse?.Umount(); _fuse = null; _winfsp?.Umount(); _winfsp = null; Umounted?.Invoke(this, System.EventArgs.Empty); } internal void GetInfo(out ulong files, out ulong totalSize) { using var ctx = Context.Create(Settings.Settings.Current.DatabasePath); totalSize = (ulong)ctx.Files.Where(f => f.IsInRepo).Sum(f => (double)f.Size); files = (ulong)ctx.Files.Count(f => f.IsInRepo); } internal string[] SplitPath(string path) => path.Split(RuntimeInformation.IsOSPlatform(OSPlatform.Windows) ? "\\" : "/", StringSplitOptions.RemoveEmptyEntries); void FillRootDirectoryCache() { using var ctx = Context.Create(Settings.Settings.Current.DatabasePath); ConcurrentDictionary rootCache = new ConcurrentDictionary(); foreach(RomSet set in ctx.RomSets) { string name; if(RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) { name = set.Name.Replace('/', '∕').Replace('<', '\uFF1C').Replace('>', '\uFF1E'). Replace(':', '\uFF1A').Replace('"', '\u2033').Replace('\\', '\').Replace('|', '|'). Replace('?', '?').Replace('*', '*'); if(rootCache.ContainsKey(name)) name = Path.GetFileNameWithoutExtension(set.Filename)?.Replace('/', '∕').Replace('<', '\uFF1C'). Replace('>', '\uFF1E').Replace(':', '\uFF1A').Replace('"', '\u2033'). Replace('\\', '\').Replace('|', '|').Replace('?', '?').Replace('*', '*'); } else { name = set.Name.Replace('/', '∕'); if(rootCache.ContainsKey(name)) name = Path.GetFileNameWithoutExtension(set.Filename)?.Replace('/', '∕'); } if(name == null || rootCache.ContainsKey(name)) name = Path.GetFileNameWithoutExtension(set.Sha384); if(name == null) continue; rootCache[name] = set.Id; _romSetsCache[set.Id] = set; } _rootDirectoryCache = rootCache; } internal long GetRomSetId(string name) { if(_rootDirectoryCache.Count == 0) FillRootDirectoryCache(); if(!_rootDirectoryCache.TryGetValue(name, out long romSetId)) return -1; return romSetId; } internal RomSet GetRomSet(long id) { if(_romSetsCache.TryGetValue(id, out RomSet romSet)) return romSet; using var ctx = Context.Create(Settings.Settings.Current.DatabasePath); romSet = ctx.RomSets.Find(id); if(romSet == null) return null; _romSetsCache[id] = romSet; return romSet; } internal ConcurrentDictionary GetMachinesFromRomSet(long id) { _machinesStatCache.TryGetValue(id, out ConcurrentDictionary cachedMachines); if(cachedMachines != null) return cachedMachines; cachedMachines = new ConcurrentDictionary(); using var ctx = Context.Create(Settings.Settings.Current.DatabasePath); foreach(Machine mach in ctx.Machines.Where(m => m.RomSet.Id == id)) { cachedMachines[mach.Name] = new CachedMachine { Id = mach.Id, CreationDate = mach.CreatedOn, ModificationDate = mach.UpdatedOn }; } _machinesStatCache[id] = cachedMachines; return cachedMachines; } internal CachedMachine GetMachine(long romSetId, string name) { ConcurrentDictionary cachedMachines = GetMachinesFromRomSet(romSetId); if(cachedMachines == null || !cachedMachines.TryGetValue(name, out CachedMachine machine)) return null; return machine; } } internal sealed class CachedMachine { public ulong Id { get; set; } public DateTime CreationDate { get; set; } public DateTime ModificationDate { get; set; } } }