#region Copyright & License Information /* * Copyright 2007-2015 The OpenRA Developers (see AUTHORS) * This file is part of OpenRA, which is free software. It is made * available to you under the terms of the GNU General Public License * as published by the Free Software Foundation. For more information, * see COPYING. */ #endregion using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Reflection; using OpenRA.Primitives; namespace OpenRA.FileSystem { public class FileSystem { public readonly List MountedPackages = new List(); static readonly Dictionary AssemblyCache = new Dictionary(); int order; Cache> fileIndex = new Cache>(_ => new List()); public IReadWritePackage CreatePackage(string filename, int order, Dictionary content) { if (filename.EndsWith(".zip", StringComparison.InvariantCultureIgnoreCase)) return new ZipFile(this, filename, order, content); if (filename.EndsWith(".oramap", StringComparison.InvariantCultureIgnoreCase)) return new ZipFile(this, filename, order, content); return new Folder(filename, order, content); } public IReadOnlyPackage OpenPackage(string filename, string annotation, int order) { if (filename.EndsWith(".mix", StringComparison.InvariantCultureIgnoreCase)) { var type = string.IsNullOrEmpty(annotation) ? PackageHashType.Classic : FieldLoader.GetValue("(value)", annotation); return new MixFile(this, filename, type, order); } if (filename.EndsWith(".zip", StringComparison.InvariantCultureIgnoreCase)) return new ZipFile(this, filename, order); if (filename.EndsWith(".oramap", StringComparison.InvariantCultureIgnoreCase)) return new ZipFile(this, filename, order); if (filename.EndsWith(".RS", StringComparison.InvariantCultureIgnoreCase)) return new D2kSoundResources(this, filename, order); if (filename.EndsWith(".Z", StringComparison.InvariantCultureIgnoreCase)) return new InstallShieldPackage(this, filename, order); if (filename.EndsWith(".PAK", StringComparison.InvariantCultureIgnoreCase)) return new PakFile(this, filename, order); if (filename.EndsWith(".big", StringComparison.InvariantCultureIgnoreCase)) return new BigFile(this, filename, order); if (filename.EndsWith(".bag", StringComparison.InvariantCultureIgnoreCase)) return new BagFile(this, filename, order); if (filename.EndsWith(".hdr", StringComparison.InvariantCultureIgnoreCase)) return new InstallShieldCABExtractor(this, filename, order); return new Folder(filename, order); } public IReadWritePackage OpenWritablePackage(string filename, int order) { if (filename.EndsWith(".zip", StringComparison.InvariantCultureIgnoreCase)) return new ZipFile(this, filename, order); if (filename.EndsWith(".oramap", StringComparison.InvariantCultureIgnoreCase)) return new ZipFile(this, filename, order); return new Folder(filename, order); } public void Mount(IReadOnlyPackage mount) { if (!MountedPackages.Contains(mount)) MountedPackages.Add(mount); } public void Mount(string name, string annotation = null) { var optional = name.StartsWith("~"); if (optional) name = name.Substring(1); name = Platform.ResolvePath(name); Action a = () => MountInner(OpenPackage(name, annotation, order++)); if (optional) try { a(); } catch { } else a(); } void MountInner(IReadOnlyPackage package) { MountedPackages.Add(package); foreach (var filename in package.AllFileNames()) { var packageList = fileIndex[filename]; if (!packageList.Contains(package)) packageList.Add(package); } } public bool Unmount(IReadOnlyPackage package) { foreach (var packagesForFile in fileIndex.Values) packagesForFile.RemoveAll(p => p == package); if (MountedPackages.Contains(package)) package.Dispose(); return MountedPackages.RemoveAll(p => p == package) > 0; } public void UnmountAll() { foreach (var package in MountedPackages) package.Dispose(); MountedPackages.Clear(); fileIndex = new Cache>(_ => new List()); } public void LoadFromManifest(Manifest manifest) { UnmountAll(); foreach (var dir in manifest.Folders) Mount(dir); foreach (var pkg in manifest.Packages) Mount(pkg.Key, pkg.Value); } Stream GetFromCache(string filename) { var package = fileIndex[filename] .Where(x => x.Exists(filename)) .MinByOrDefault(x => x.Priority); if (package != null) return package.GetContent(filename); return null; } public Stream Open(string filename) { Stream s; if (!TryOpen(filename, out s)) throw new FileNotFoundException("File not found: {0}".F(filename), filename); return s; } public bool TryOpen(string name, out Stream s) { var filename = name; var packageName = string.Empty; // Used for faction specific packages; rule out false positive on Windows C:\ drive notation var explicitPackage = name.Contains(':') && !Directory.Exists(Path.GetDirectoryName(name)); if (explicitPackage) { var divide = name.Split(':'); packageName = divide.First(); filename = divide.Last(); } // Check the cache for a quick lookup if the package name is unknown // TODO: This disables caching for explicit package requests if (filename.IndexOfAny(new[] { '/', '\\' }) == -1 && !explicitPackage) { s = GetFromCache(filename); if (s != null) return true; } // Ask each package individually IReadOnlyPackage package; if (explicitPackage && !string.IsNullOrEmpty(packageName)) package = MountedPackages.Where(x => x.Name == packageName).MaxByOrDefault(x => x.Priority); else package = MountedPackages.Where(x => x.Exists(filename)).MaxByOrDefault(x => x.Priority); if (package != null) { s = package.GetContent(filename); return true; } s = null; return false; } public bool Exists(string name) { var explicitPackage = name.Contains(':') && !Directory.Exists(Path.GetDirectoryName(name)); if (explicitPackage) { var divide = name.Split(':'); var packageName = divide.First(); var filename = divide.Last(); return MountedPackages.Where(n => n.Name == packageName).Any(f => f.Exists(filename)); } else return MountedPackages.Any(f => f.Exists(name)); } public static Assembly ResolveAssembly(object sender, ResolveEventArgs e) { foreach (var assembly in AppDomain.CurrentDomain.GetAssemblies()) if (assembly.FullName == e.Name) return assembly; var frags = e.Name.Split(','); var filename = frags[0] + ".dll"; Assembly a; if (AssemblyCache.TryGetValue(filename, out a)) return a; if (Game.ModData.ModFiles.Exists(filename)) using (var s = Game.ModData.ModFiles.Open(filename)) { var buf = s.ReadBytes((int)s.Length); a = Assembly.Load(buf); AssemblyCache.Add(filename, a); return a; } return null; } } }