mirror of
				https://github.com/velopack/velopack.git
				synced 2025-10-25 15:19:22 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			305 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			305 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| // Parts of this file have been used from
 | |
| // https://github.com/icsharpcode/ILSpy/blob/f7460a041ea8fb8b0abf8527b97a5b890eb94eea/ICSharpCode.Decompiler/SingleFileBundle.cs
 | |
| 
 | |
| using Microsoft.NET.HostModel.AppHost;
 | |
| using Microsoft.NET.HostModel.Bundle;
 | |
| using NuGet.Versioning;
 | |
| using Squirrel.SimpleSplat;
 | |
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Collections.Immutable;
 | |
| using System.IO;
 | |
| using System.IO.Compression;
 | |
| using System.IO.MemoryMappedFiles;
 | |
| using System.Linq;
 | |
| using System.Runtime.InteropServices;
 | |
| using System.Text;
 | |
| using System.Threading.Tasks;
 | |
| 
 | |
| namespace Squirrel.CommandLine
 | |
| {
 | |
|     internal class DotnetUtil
 | |
|     {
 | |
|         private static IFullLogger Log = SquirrelLocator.CurrentMutable.GetService<ILogManager>().GetLogger(typeof(DotnetUtil));
 | |
| 
 | |
|         public static void CheckDotnetReferences(string filePath, PeNet.PeFile pe, IEnumerable<Runtimes.RuntimeInfo> dependencies)
 | |
|         {
 | |
|             try {
 | |
|                 var name = Path.GetFileName(filePath);
 | |
|                 var baseDir = Path.GetDirectoryName(filePath);
 | |
| 
 | |
|                 if (pe == null) {
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 Log.Debug($"Verifying dependencies for {name}");
 | |
| 
 | |
|                 var probablyBitness = pe.IsExe && pe.Is32Bit ? "-x86" : ""; // used to construct a suggested reference later
 | |
| 
 | |
|                 // might be an app host or single file bundle, lets look for a dotnet "dll"
 | |
|                 if (pe.IsExe && !pe.IsDotNet) {
 | |
|                     if (IsSingleFileBundle(filePath)) {
 | |
|                         // TODO
 | |
|                         Log.Info("Checking dependencies in SingleFileBundle's is not supported.");
 | |
|                         return;
 | |
|                     } else {
 | |
|                         var st = pe.Resources.VsVersionInfo.StringFileInfo.StringTable;
 | |
|                         if (st.Length > 0) {
 | |
|                             var original = st[0].OriginalFilename;
 | |
|                             if (original != null && original.EndsWith(".dll", StringComparison.InvariantCultureIgnoreCase) && File.Exists(Path.Combine(baseDir, original))) {
 | |
|                                 pe = new PeNet.PeFile(Path.Combine(baseDir, original));
 | |
|                                 Log.Debug($"Redirecting to {original}");
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 var refs = pe.MetaDataStreamTablesHeader?.Tables?.AssemblyRef ?? new();
 | |
|                 if (!pe.IsDotNet || refs.Count == 0) {
 | |
|                     Log.Info($"{name} is not a CLR assembly or no assembly references were found.");
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 var lookup = refs.ToDictionary(x => pe.MetaDataStreamString.GetStringAtIndex(x.Name), x => x, StringComparer.InvariantCultureIgnoreCase);
 | |
| 
 | |
|                 // base lib is "mscorlib" for full framework or "System.Runtime" for dotnet
 | |
|                 if (lookup.ContainsKey("mscorlib")) {
 | |
|                     Log.Debug($"{name} references the full framework, skipping....");
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 var foundcore = lookup.TryGetValue("System.Runtime", out var corelib);
 | |
|                 if (!foundcore) {
 | |
|                     Log.Debug($"{name} has no reference to a known core lib, skipping...");
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 // don't bother checking any assemblies (exe or dll) that are present in the package
 | |
|                 foreach (var k in Directory.GetFiles(baseDir)) {
 | |
|                     if (Utility.FileIsLikelyPEImage(k)) {
 | |
|                         if (lookup.ContainsKey(Path.GetFileNameWithoutExtension(k))) {
 | |
|                             Log.Debug($"Reference {Path.GetFileName(k)} found in local directory.");
 | |
|                             lookup.Remove(Path.GetFileNameWithoutExtension(k));
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if (!lookup.Any())
 | |
|                     return;
 | |
| 
 | |
|                 var runtime = dependencies.OfType<Runtimes.DotnetInfo>()
 | |
|                     .FirstOrDefault(f => f.MinVersion.Major == corelib.MajorVersion && f.MinVersion.Minor == corelib.MinorVersion);
 | |
| 
 | |
|                 if (runtime == null) {
 | |
|                     var suggestedArg = $"--framework net{corelib.MajorVersion}.{corelib.MinorVersion}" + probablyBitness;
 | |
|                     Log.Warn($"{name} has {lookup.Count} unresolved references, and no matching runtimes were found. (Are you missing the '{suggestedArg}' argument?)");
 | |
|                     foreach (var f in lookup)
 | |
|                         Log.Debug($"{name} has unresolved reference {f.Key}.");
 | |
| 
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 foreach (var f in lookup) {
 | |
|                     var fver = new NuGetVersion(f.Value.MajorVersion, f.Value.MinorVersion, f.Value.BuildNumber, f.Value.RevisionNumber);
 | |
|                     if (fver > runtime.MinVersion) {
 | |
|                         Log.Warn($"{name} references {f.Key},Version={fver} - which is higher than the current runtime version ({runtime.MinVersion}).");
 | |
|                     }
 | |
|                 }
 | |
|             } catch (Exception ex) {
 | |
|                 Log.WarnException("Failed to verify dependencies.", ex);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public static bool IsSingleFileBundle(string peFile)
 | |
|         {
 | |
|             return HostWriter.IsBundle(peFile, out var offset) && offset > 0;
 | |
|         }
 | |
| 
 | |
|         public static void UpdateSingleFileBundleIcon(string sourceFile, string destinationFile, string iconPath)
 | |
|         {
 | |
|             using var _ = Utility.GetTempDirectory(out var tmpdir);
 | |
|             var sourceName = Path.GetFileNameWithoutExtension(sourceFile);
 | |
|             var newAppHost = Path.Combine(tmpdir, sourceName + ".exe");
 | |
| 
 | |
|             // extract bundled Update.exe assets to tmp dir
 | |
|             Log.Info("Extracting Update.exe resources");
 | |
|             File.Copy(sourceFile, newAppHost);
 | |
|             DumpPackageAssemblies(sourceFile, tmpdir);
 | |
| 
 | |
|             // set new icon (which eradicates bundled assets)
 | |
|             Log.Info("Patching Update.exe icon");
 | |
|             HelperExe.SetExeIcon(newAppHost, iconPath);
 | |
|             HostWriter.ResetBundle(newAppHost);
 | |
| 
 | |
|             // re-append bundled assets to exe
 | |
|             Log.Info("Re-packing Update.exe bundle");
 | |
|             var bundlerOutput = Path.Combine(tmpdir, "output");
 | |
|             Directory.CreateDirectory(bundlerOutput);
 | |
|             var bundler = new Bundler(
 | |
|                 sourceName + ".exe",
 | |
|                 bundlerOutput,
 | |
|                 BundleOptions.EnableCompression,
 | |
|                 OSPlatform.Windows,
 | |
|                 Architecture.X86,
 | |
|                 new Version(6, 0),
 | |
|                 false,
 | |
|                 sourceName
 | |
|             );
 | |
|             var singleFile = GenerateBundle(bundler, tmpdir, bundlerOutput);
 | |
| 
 | |
|             // copy to requested location
 | |
|             File.Copy(singleFile, destinationFile);
 | |
|         }
 | |
| 
 | |
|         private static void DumpPackageAssemblies(string packageFileName, string outputDirectory)
 | |
|         {
 | |
|             if (!HostWriter.IsBundle(packageFileName, out long bundleHeaderOffset)) {
 | |
|                 throw new InvalidOperationException($"Cannot dump assembiles for {packageFileName}, because it is not a single file bundle.");
 | |
|             }
 | |
| 
 | |
|             using (var memoryMappedPackage = MemoryMappedFile.CreateFromFile(packageFileName, FileMode.Open, null, 0, MemoryMappedFileAccess.Read)) {
 | |
|                 using (var packageView = memoryMappedPackage.CreateViewAccessor(0, 0, MemoryMappedFileAccess.Read)) {
 | |
|                     var manifest = ReadManifest(packageView, bundleHeaderOffset);
 | |
|                     foreach (var entry in manifest.Entries) {
 | |
|                         Stream contents;
 | |
| 
 | |
|                         if (entry.CompressedSize == 0) {
 | |
|                             contents = new UnmanagedMemoryStream(packageView.SafeMemoryMappedViewHandle, entry.Offset, entry.Size);
 | |
|                         } else {
 | |
|                             Stream compressedStream = new UnmanagedMemoryStream(packageView.SafeMemoryMappedViewHandle, entry.Offset, entry.CompressedSize);
 | |
|                             Stream decompressedStream = new MemoryStream((int) entry.Size);
 | |
|                             using (var deflateStream = new DeflateStream(compressedStream, CompressionMode.Decompress)) {
 | |
|                                 deflateStream.CopyTo(decompressedStream);
 | |
|                             }
 | |
| 
 | |
|                             if (decompressedStream.Length != entry.Size) {
 | |
|                                 throw new Exception($"Corrupted single-file entry '${entry.RelativePath}'. Declared decompressed size '${entry.Size}' is not the same as actual decompressed size '${decompressedStream.Length}'.");
 | |
|                             }
 | |
| 
 | |
|                             decompressedStream.Seek(0, SeekOrigin.Begin);
 | |
|                             contents = decompressedStream;
 | |
|                         }
 | |
| 
 | |
|                         using (var fileStream = File.Create(Path.Combine(outputDirectory, entry.RelativePath))) {
 | |
|                             contents.CopyTo(fileStream);
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private static string GenerateBundle(Bundler bundler, string sourceDir, string outputDir)
 | |
|         {
 | |
|             // Convert sourceDir to absolute path
 | |
|             sourceDir = Path.GetFullPath(sourceDir);
 | |
| 
 | |
|             // Get all files in the source directory and all sub-directories.
 | |
|             string[] sources = Directory.GetFiles(sourceDir, searchPattern: "*", searchOption: SearchOption.AllDirectories);
 | |
| 
 | |
|             // Sort the file names to keep the bundle construction deterministic.
 | |
|             Array.Sort(sources, StringComparer.Ordinal);
 | |
| 
 | |
|             List<FileSpec> fileSpecs = new List<FileSpec>(sources.Length);
 | |
|             foreach (var file in sources) {
 | |
|                 fileSpecs.Add(new FileSpec(file, Path.GetRelativePath(sourceDir, file)));
 | |
|             }
 | |
| 
 | |
|             return bundler.GenerateBundle(fileSpecs);
 | |
|         }
 | |
| 
 | |
|         public struct Header
 | |
|         {
 | |
|             public uint MajorVersion;
 | |
|             public uint MinorVersion;
 | |
|             public int FileCount;
 | |
|             public string BundleID;
 | |
| 
 | |
|             // Fields introduced with v2:
 | |
|             public long DepsJsonOffset;
 | |
|             public long DepsJsonSize;
 | |
|             public long RuntimeConfigJsonOffset;
 | |
|             public long RuntimeConfigJsonSize;
 | |
|             public ulong Flags;
 | |
| 
 | |
|             public ImmutableArray<Entry> Entries;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// FileType: Identifies the type of file embedded into the bundle.
 | |
|         ///
 | |
|         /// The bundler differentiates a few kinds of files via the manifest,
 | |
|         /// with respect to the way in which they'll be used by the runtime.
 | |
|         /// </summary>
 | |
|         public enum FileType : byte
 | |
|         {
 | |
|             Unknown,           // Type not determined.
 | |
|             Assembly,          // IL and R2R Assemblies
 | |
|             NativeBinary,      // NativeBinaries
 | |
|             DepsJson,          // .deps.json configuration file
 | |
|             RuntimeConfigJson, // .runtimeconfig.json configuration file
 | |
|             Symbols            // PDB Files
 | |
|         };
 | |
| 
 | |
|         public struct Entry
 | |
|         {
 | |
|             public long Offset;
 | |
|             public long Size;
 | |
|             public long CompressedSize; // 0 if not compressed, otherwise the compressed size in the bundle
 | |
|             public FileType Type;
 | |
|             public string RelativePath; // Path of an embedded file, relative to the Bundle source-directory.
 | |
|         }
 | |
| 
 | |
|         static UnmanagedMemoryStream AsStream(MemoryMappedViewAccessor view)
 | |
|         {
 | |
|             long size = checked((long) view.SafeMemoryMappedViewHandle.ByteLength);
 | |
|             return new UnmanagedMemoryStream(view.SafeMemoryMappedViewHandle, 0, size);
 | |
|         }
 | |
| 
 | |
|         public static Header ReadManifest(MemoryMappedViewAccessor view, long bundleHeaderOffset)
 | |
|         {
 | |
|             using var stream = AsStream(view);
 | |
|             stream.Seek(bundleHeaderOffset, SeekOrigin.Begin);
 | |
|             return ReadManifest(stream);
 | |
|         }
 | |
| 
 | |
|         public static Header ReadManifest(Stream stream)
 | |
|         {
 | |
|             var header = new Header();
 | |
|             using var reader = new BinaryReader(stream, Encoding.UTF8, leaveOpen: true);
 | |
|             header.MajorVersion = reader.ReadUInt32();
 | |
|             header.MinorVersion = reader.ReadUInt32();
 | |
| 
 | |
|             // Major versions 3, 4 and 5 were skipped to align bundle versioning with .NET versioning scheme
 | |
|             if (header.MajorVersion < 1 || header.MajorVersion > 6) {
 | |
|                 throw new InvalidDataException($"Unsupported manifest version: {header.MajorVersion}.{header.MinorVersion}");
 | |
|             }
 | |
|             header.FileCount = reader.ReadInt32();
 | |
|             header.BundleID = reader.ReadString();
 | |
|             if (header.MajorVersion >= 2) {
 | |
|                 header.DepsJsonOffset = reader.ReadInt64();
 | |
|                 header.DepsJsonSize = reader.ReadInt64();
 | |
|                 header.RuntimeConfigJsonOffset = reader.ReadInt64();
 | |
|                 header.RuntimeConfigJsonSize = reader.ReadInt64();
 | |
|                 header.Flags = reader.ReadUInt64();
 | |
|             }
 | |
|             var entries = ImmutableArray.CreateBuilder<Entry>(header.FileCount);
 | |
|             for (int i = 0; i < header.FileCount; i++) {
 | |
|                 entries.Add(ReadEntry(reader, header.MajorVersion));
 | |
|             }
 | |
|             header.Entries = entries.MoveToImmutable();
 | |
|             return header;
 | |
|         }
 | |
| 
 | |
|         private static Entry ReadEntry(BinaryReader reader, uint bundleMajorVersion)
 | |
|         {
 | |
|             Entry entry;
 | |
|             entry.Offset = reader.ReadInt64();
 | |
|             entry.Size = reader.ReadInt64();
 | |
|             entry.CompressedSize = bundleMajorVersion >= 6 ? reader.ReadInt64() : 0;
 | |
|             entry.Type = (FileType) reader.ReadByte();
 | |
|             entry.RelativePath = reader.ReadString();
 | |
|             return entry;
 | |
|         }
 | |
|     }
 | |
| }
 |