mirror of
https://github.com/SabreTools/BinaryObjectScanner.git
synced 2026-02-13 21:31:04 +00:00
794 lines
27 KiB
C#
794 lines
27 KiB
C#
using System;
|
|
using System.Collections.Concurrent;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Security.Cryptography;
|
|
using BurnOutSharp.Interfaces;
|
|
using BurnOutSharp.Wrappers;
|
|
|
|
namespace BurnOutSharp.Tools
|
|
{
|
|
internal static class Utilities
|
|
{
|
|
#region Dictionary Manipulation
|
|
|
|
/// <summary>
|
|
/// Append one result to a results dictionary
|
|
/// </summary>
|
|
/// <param name="original">Dictionary to append to</param>
|
|
/// <param name="key">Key to add information to</param>
|
|
/// <param name="value">String value to add</param>
|
|
public static void AppendToDictionary(ConcurrentDictionary<string, ConcurrentQueue<string>> original, string key, string value)
|
|
{
|
|
// If the value is empty, don't add it
|
|
if (string.IsNullOrWhiteSpace(value))
|
|
return;
|
|
|
|
var values = new ConcurrentQueue<string>();
|
|
values.Enqueue(value);
|
|
AppendToDictionary(original, key, values);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Append one result to a results dictionary
|
|
/// </summary>
|
|
/// <param name="original">Dictionary to append to</param>
|
|
/// <param name="key">Key to add information to</param>
|
|
/// <param name="value">String value to add</param>
|
|
public static void AppendToDictionary(ConcurrentDictionary<string, ConcurrentQueue<string>> original, string key, ConcurrentQueue<string> values)
|
|
{
|
|
// If the dictionary is null, just return
|
|
if (original == null)
|
|
return;
|
|
|
|
// Use a placeholder value if the key is null
|
|
key = key ?? "NO FILENAME";
|
|
|
|
// Add the key if needed and then append the lists
|
|
original.TryAdd(key, new ConcurrentQueue<string>());
|
|
original[key].AddRange(values);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Append one results dictionary to another
|
|
/// </summary>
|
|
/// <param name="original">Dictionary to append to</param>
|
|
/// <param name="addition">Dictionary to pull from</param>
|
|
public static void AppendToDictionary(ConcurrentDictionary<string, ConcurrentQueue<string>> original, ConcurrentDictionary<string, ConcurrentQueue<string>> addition)
|
|
{
|
|
// If either dictionary is missing, just return
|
|
if (original == null || addition == null)
|
|
return;
|
|
|
|
// Loop through each of the addition keys and add accordingly
|
|
foreach (string key in addition.Keys)
|
|
{
|
|
original.TryAdd(key, new ConcurrentQueue<string>());
|
|
original[key].AddRange(addition[key]);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Remove empty or null keys from a results dictionary
|
|
/// </summary>
|
|
/// <param name="original">Dictionary to clean</param>
|
|
public static void ClearEmptyKeys(ConcurrentDictionary<string, ConcurrentQueue<string>> original)
|
|
{
|
|
// If the dictionary is missing, we can't do anything
|
|
if (original == null)
|
|
return;
|
|
|
|
// Get a list of all of the keys
|
|
var keys = original.Keys.ToList();
|
|
|
|
// Iterate and reset keys
|
|
for (int i = 0; i < keys.Count; i++)
|
|
{
|
|
// Get the current key
|
|
string key = keys[i];
|
|
|
|
// If the key is empty, remove it
|
|
if (original[key] == null || !original[key].Any())
|
|
original.TryRemove(key, out _);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Prepend a parent path from dictionary keys, if possible
|
|
/// </summary>
|
|
/// <param name="original">Dictionary to strip values from</param>
|
|
/// <param name="pathToPrepend">Path to strip from the keys</param>
|
|
public static void PrependToKeys(ConcurrentDictionary<string, ConcurrentQueue<string>> original, string pathToPrepend)
|
|
{
|
|
// If the dictionary is missing, we can't do anything
|
|
if (original == null)
|
|
return;
|
|
|
|
// Use a placeholder value if the path is null
|
|
pathToPrepend = (pathToPrepend ?? "ARCHIVE").TrimEnd(Path.DirectorySeparatorChar);
|
|
|
|
// Get a list of all of the keys
|
|
var keys = original.Keys.ToList();
|
|
|
|
// Iterate and reset keys
|
|
for (int i = 0; i < keys.Count; i++)
|
|
{
|
|
// Get the current key
|
|
string currentKey = keys[i];
|
|
|
|
// Otherwise, get the new key name and transfer over
|
|
string newKey = $"{pathToPrepend}{Path.DirectorySeparatorChar}{currentKey.Trim(Path.DirectorySeparatorChar)}";
|
|
original[newKey] = original[currentKey];
|
|
original.TryRemove(currentKey, out _);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Strip a parent path from dictionary keys, if possible
|
|
/// </summary>
|
|
/// <param name="original">Dictionary to strip values from</param>
|
|
/// <param name="pathToStrip">Path to strip from the keys</param>
|
|
public static void StripFromKeys(ConcurrentDictionary<string, ConcurrentQueue<string>> original, string pathToStrip)
|
|
{
|
|
// If either is missing, we can't do anything
|
|
if (original == null || string.IsNullOrEmpty(pathToStrip))
|
|
return;
|
|
|
|
// Get a list of all of the keys
|
|
var keys = original.Keys.ToList();
|
|
|
|
// Iterate and reset keys
|
|
for (int i = 0; i < keys.Count; i++)
|
|
{
|
|
// Get the current key
|
|
string currentKey = keys[i];
|
|
|
|
// If the key doesn't start with the path, don't touch it
|
|
if (!currentKey.StartsWith(pathToStrip, StringComparison.OrdinalIgnoreCase))
|
|
continue;
|
|
|
|
// Otherwise, get the new key name and transfer over
|
|
string newKey = currentKey.Substring(pathToStrip.Length);
|
|
original[newKey] = original[currentKey];
|
|
original.TryRemove(currentKey, out _);
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Concurrent Manipulation
|
|
|
|
/// <summary>
|
|
/// Add a range of values from one queue to another
|
|
/// </summary>
|
|
/// <param name="original">Queue to add data to</param>
|
|
/// <param name="values">Queue to get data from</param>
|
|
public static void AddRange(this ConcurrentQueue<string> original, ConcurrentQueue<string> values)
|
|
{
|
|
while (!values.IsEmpty)
|
|
{
|
|
if (!values.TryDequeue(out string value))
|
|
return;
|
|
|
|
original.Enqueue(value);
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region File Types
|
|
|
|
/// <summary>
|
|
/// Get the supported file type for a magic string
|
|
/// </summary>
|
|
/// <remarks>Recommend sending in 16 bytes to check</remarks>
|
|
public static SupportedFileType GetFileType(byte[] magic)
|
|
{
|
|
// If we have an invalid magic byte array
|
|
if (magic == null || magic.Length == 0)
|
|
return SupportedFileType.UNKNOWN;
|
|
|
|
#region BFPK
|
|
|
|
if (magic.StartsWith(new byte?[] { 0x42, 0x46, 0x50, 0x4b }))
|
|
return SupportedFileType.BFPK;
|
|
|
|
#endregion
|
|
|
|
#region BZip2
|
|
|
|
if (magic.StartsWith(new byte?[] { 0x42, 0x52, 0x68 }))
|
|
return SupportedFileType.BZip2;
|
|
|
|
#endregion
|
|
|
|
#region Executable
|
|
|
|
// DOS MZ executable file format (and descendants)
|
|
if (magic.StartsWith(new byte?[] { 0x4d, 0x5a }))
|
|
return SupportedFileType.Executable;
|
|
|
|
/*
|
|
// None of the following are supported in scans yet
|
|
|
|
// Executable and Linkable Format
|
|
if (magic.StartsWith(new byte?[] { 0x7f, 0x45, 0x4c, 0x46 }))
|
|
return FileTypes.Executable;
|
|
|
|
// Mach-O binary (32-bit)
|
|
if (magic.StartsWith(new byte?[] { 0xfe, 0xed, 0xfa, 0xce }))
|
|
return FileTypes.Executable;
|
|
|
|
// Mach-O binary (32-bit, reverse byte ordering scheme)
|
|
if (magic.StartsWith(new byte?[] { 0xce, 0xfa, 0xed, 0xfe }))
|
|
return FileTypes.Executable;
|
|
|
|
// Mach-O binary (64-bit)
|
|
if (magic.StartsWith(new byte?[] { 0xfe, 0xed, 0xfa, 0xcf }))
|
|
return FileTypes.Executable;
|
|
|
|
// Mach-O binary (64-bit, reverse byte ordering scheme)
|
|
if (magic.StartsWith(new byte?[] { 0xcf, 0xfa, 0xed, 0xfe }))
|
|
return FileTypes.Executable;
|
|
|
|
// Prefrred Executable File Format
|
|
if (magic.StartsWith(new byte?[] { 0x4a, 0x6f, 0x79, 0x21, 0x70, 0x65, 0x66, 0x66 }))
|
|
return FileTypes.Executable;
|
|
*/
|
|
|
|
#endregion
|
|
|
|
#region GZIP
|
|
|
|
if (magic.StartsWith(new byte?[] { 0x1f, 0x8b }))
|
|
return SupportedFileType.GZIP;
|
|
|
|
#endregion
|
|
|
|
#region IniFile
|
|
|
|
// No magic checks for IniFile
|
|
|
|
#endregion
|
|
|
|
#region InstallShieldArchiveV3
|
|
|
|
if (magic.StartsWith(new byte?[] { 0x13, 0x5D, 0x65, 0x8C }))
|
|
return SupportedFileType.InstallShieldArchiveV3;
|
|
|
|
#endregion
|
|
|
|
#region InstallShieldCAB
|
|
|
|
if (magic.StartsWith(new byte?[] { 0x49, 0x53, 0x63 }))
|
|
return SupportedFileType.InstallShieldCAB;
|
|
|
|
#endregion
|
|
|
|
#region MicrosoftCAB
|
|
|
|
if (magic.StartsWith(new byte?[] { 0x4d, 0x53, 0x43, 0x46 }))
|
|
return SupportedFileType.MicrosoftCAB;
|
|
|
|
#endregion
|
|
|
|
#region MPQ
|
|
|
|
if (magic.StartsWith(new byte?[] { 0x4d, 0x50, 0x51, 0x1a }))
|
|
return SupportedFileType.MPQ;
|
|
|
|
#endregion
|
|
|
|
#region MSI
|
|
|
|
if (magic.StartsWith(new byte?[] { 0xD0, 0xCF, 0x11, 0xE0, 0xA1, 0xB1, 0x1A, 0xE1 }))
|
|
return SupportedFileType.MSI;
|
|
|
|
#endregion
|
|
|
|
#region PKZIP
|
|
|
|
// PKZIP (Unknown)
|
|
if (magic.StartsWith(new byte?[] { 0x50, 0x4b, 0x00, 0x00 }))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// PKZIP
|
|
if (magic.StartsWith(new byte?[] { 0x50, 0x4b, 0x03, 0x04 }))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// PKZIP (Empty Archive)
|
|
if (magic.StartsWith(new byte?[] { 0x50, 0x4b, 0x05, 0x06 }))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// PKZIP (Spanned Archive)
|
|
if (magic.StartsWith(new byte?[] { 0x50, 0x4b, 0x07, 0x08 }))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
#endregion
|
|
|
|
#region PLJ
|
|
|
|
// https://www.iana.org/assignments/media-types/audio/vnd.everad.plj
|
|
if (magic.StartsWith(new byte?[] { 0xFF, 0x9D, 0x53, 0x4B }))
|
|
return SupportedFileType.PLJ;
|
|
|
|
#endregion
|
|
|
|
#region RAR
|
|
|
|
// RAR archive version 1.50 onwards
|
|
if (magic.StartsWith(new byte?[] { 0x52, 0x61, 0x72, 0x21, 0x1a, 0x07, 0x00 }))
|
|
return SupportedFileType.RAR;
|
|
|
|
// RAR archive version 5.0 onwards
|
|
if (magic.StartsWith(new byte?[] { 0x52, 0x61, 0x72, 0x21, 0x1a, 0x07, 0x01, 0x00 }))
|
|
return SupportedFileType.RAR;
|
|
|
|
#endregion
|
|
|
|
#region SevenZip
|
|
|
|
if (magic.StartsWith(new byte?[] { 0x37, 0x7a, 0xbc, 0xaf, 0x27, 0x1c }))
|
|
return SupportedFileType.SevenZip;
|
|
|
|
#endregion
|
|
|
|
#region TapeArchive
|
|
|
|
if (magic.StartsWith(new byte?[] { 0x75, 0x73, 0x74, 0x61, 0x72, 0x00, 0x30, 0x30 }))
|
|
return SupportedFileType.TapeArchive;
|
|
|
|
if (magic.StartsWith(new byte?[] { 0x75, 0x73, 0x74, 0x61, 0x72, 0x20, 0x20, 0x00 }))
|
|
return SupportedFileType.TapeArchive;
|
|
|
|
#endregion
|
|
|
|
#region Textfile
|
|
|
|
// Not all textfiles can be determined through magic number
|
|
|
|
// HTML
|
|
if (magic.StartsWith(new byte?[] { 0x3c, 0x68, 0x74, 0x6d, 0x6c }))
|
|
return SupportedFileType.Textfile;
|
|
|
|
// HTML and XML
|
|
if (magic.StartsWith(new byte?[] { 0x3c, 0x21, 0x44, 0x4f, 0x43, 0x54, 0x59, 0x50, 0x45 }))
|
|
return SupportedFileType.Textfile;
|
|
|
|
// InstallShield Compiled Rules
|
|
if (magic.StartsWith(new byte?[] { 0x61, 0x4C, 0x75, 0x5A }))
|
|
return SupportedFileType.Textfile;
|
|
|
|
// Microsoft Office File (old)
|
|
if (magic.StartsWith(new byte?[] { 0xd0, 0xcf, 0x11, 0xe0, 0xa1, 0xb1, 0x1a, 0xe1 }))
|
|
return SupportedFileType.Textfile;
|
|
|
|
// Rich Text File
|
|
if (magic.StartsWith(new byte?[] { 0x7b, 0x5c, 0x72, 0x74, 0x66, 0x31 }))
|
|
return SupportedFileType.Textfile;
|
|
|
|
// Windows Help File
|
|
if (magic.StartsWith(new byte?[] { 0x3F, 0x5F, 0x03, 0x00 }))
|
|
return SupportedFileType.Textfile;
|
|
|
|
#endregion
|
|
|
|
#region Valve
|
|
|
|
if (HLLib.Packages.Package.GetPackageType(magic) != HLLib.Packages.PackageType.HL_PACKAGE_NONE)
|
|
return SupportedFileType.Valve;
|
|
|
|
#endregion
|
|
|
|
#region XZ
|
|
|
|
if (magic.StartsWith(new byte?[] { 0xfd, 0x37, 0x7a, 0x58, 0x5a, 0x00 }))
|
|
return SupportedFileType.XZ;
|
|
|
|
#endregion
|
|
|
|
// We couldn't find a supported match
|
|
return SupportedFileType.UNKNOWN;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the supported file type for an extension
|
|
/// </summary>
|
|
/// <remarks>This is less accurate than a magic string match</remarks>
|
|
public static SupportedFileType GetFileType(string extension)
|
|
{
|
|
// If we have an invalid extension
|
|
if (string.IsNullOrWhiteSpace(extension))
|
|
return SupportedFileType.UNKNOWN;
|
|
|
|
// Normalize the extension
|
|
extension = extension.TrimStart('.').Trim();
|
|
|
|
#region BFPK
|
|
|
|
// No extensions registered for BFPK
|
|
|
|
#endregion
|
|
|
|
#region BZip2
|
|
|
|
if (extension.Equals("bz2", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.BZip2;
|
|
|
|
#endregion
|
|
|
|
#region Executable
|
|
|
|
// DOS MZ executable file format (and descendants)
|
|
if (extension.Equals("exe", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.Executable;
|
|
|
|
// DOS MZ library file format (and descendants)
|
|
if (extension.Equals("dll", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.Executable;
|
|
|
|
#endregion
|
|
|
|
#region GZIP
|
|
|
|
if (extension.Equals("gz", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.GZIP;
|
|
|
|
#endregion
|
|
|
|
#region IniFile
|
|
|
|
if (extension.Equals("ini", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.IniFile;
|
|
|
|
#endregion
|
|
|
|
#region InstallShieldArchiveV3
|
|
|
|
if (extension.Equals("z", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.InstallShieldArchiveV3;
|
|
|
|
#endregion
|
|
|
|
#region InstallShieldCAB
|
|
|
|
// No extensions registered for InstallShieldCAB
|
|
// Both InstallShieldCAB and MicrosoftCAB share the same extension
|
|
|
|
#endregion
|
|
|
|
#region MicrosoftCAB
|
|
|
|
// No extensions registered for InstallShieldCAB
|
|
// Both InstallShieldCAB and MicrosoftCAB share the same extension
|
|
|
|
#endregion
|
|
|
|
#region MPQ
|
|
|
|
if (extension.Equals("mpq", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.MPQ;
|
|
|
|
#endregion
|
|
|
|
#region MSI
|
|
|
|
if (extension.Equals("msi", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.MSI;
|
|
|
|
#endregion
|
|
|
|
#region PKZIP
|
|
|
|
// PKZIP
|
|
if (extension.Equals("zip", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// Android package
|
|
if (extension.Equals("apk", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// Java archive
|
|
if (extension.Equals("jar", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// Google Earth saved working session file
|
|
if (extension.Equals("kmz", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// KWord document
|
|
if (extension.Equals("kwd", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// Microsoft Office Open XML Format (OOXML) Document
|
|
if (extension.Equals("docx", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// Microsoft Office Open XML Format (OOXML) Presentation
|
|
if (extension.Equals("pptx", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// Microsoft Office Open XML Format (OOXML) Spreadsheet
|
|
if (extension.Equals("xlsx", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// OpenDocument text document
|
|
if (extension.Equals("odt", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// OpenDocument presentation
|
|
if (extension.Equals("odp", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// OpenDocument text document template
|
|
if (extension.Equals("ott", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// Microsoft Open XML paper specification file
|
|
if (extension.Equals("oxps", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// OpenOffice spreadsheet
|
|
if (extension.Equals("sxc", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// OpenOffice drawing
|
|
if (extension.Equals("sxd", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// OpenOffice presentation
|
|
if (extension.Equals("sxi", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// OpenOffice word processing
|
|
if (extension.Equals("sxw", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// StarOffice spreadsheet
|
|
if (extension.Equals("sxc", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// Windows Media compressed skin file
|
|
if (extension.Equals("wmz", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// Mozilla Browser Archive
|
|
if (extension.Equals("xpi", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// XML paper specification file
|
|
if (extension.Equals("xps", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
// eXact Packager Models
|
|
if (extension.Equals("xpt", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PKZIP;
|
|
|
|
#endregion
|
|
|
|
#region PLJ
|
|
|
|
// https://www.iana.org/assignments/media-types/audio/vnd.everad.plj
|
|
if (extension.Equals("plj", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.PLJ;
|
|
|
|
#endregion
|
|
|
|
#region RAR
|
|
|
|
if (extension.Equals("rar", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.RAR;
|
|
|
|
#endregion
|
|
|
|
#region SevenZip
|
|
|
|
if (extension.Equals("7z", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.SevenZip;
|
|
|
|
#endregion
|
|
|
|
#region TapeArchive
|
|
|
|
if (extension.Equals("tar", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.SevenZip;
|
|
|
|
#endregion
|
|
|
|
#region Textfile
|
|
|
|
// "Description in Zip"
|
|
if (extension.Equals("diz", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.Textfile;
|
|
|
|
// Generic textfile (no header)
|
|
if (extension.Equals("txt", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.Textfile;
|
|
|
|
// HTML
|
|
if (extension.Equals("htm", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.Textfile;
|
|
if (extension.Equals("html", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.Textfile;
|
|
|
|
// InstallShield Script
|
|
if (extension.Equals("ins", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.Textfile;
|
|
|
|
// Microsoft Office File (old)
|
|
if (extension.Equals("doc", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.Textfile;
|
|
|
|
// Rich Text File
|
|
if (extension.Equals("rtf", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.Textfile;
|
|
|
|
// Setup information
|
|
if (extension.Equals("inf", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.Textfile;
|
|
|
|
// Windows Help File
|
|
if (extension.Equals("hlp", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.Textfile;
|
|
|
|
// XML
|
|
if (extension.Equals("xml", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.Textfile;
|
|
|
|
#endregion
|
|
|
|
#region Valve
|
|
|
|
// No extensions registered for Valve
|
|
|
|
#endregion
|
|
|
|
#region XZ
|
|
|
|
if (extension.Equals("xz", StringComparison.OrdinalIgnoreCase))
|
|
return SupportedFileType.XZ;
|
|
|
|
#endregion
|
|
|
|
// We couldn't find a supported match
|
|
return SupportedFileType.UNKNOWN;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Create an instance of a scannable based on file type
|
|
/// </summary>
|
|
public static IScannable CreateScannable(SupportedFileType fileType)
|
|
{
|
|
switch (fileType)
|
|
{
|
|
case SupportedFileType.BFPK: return new FileType.BFPK();
|
|
case SupportedFileType.BZip2: return new FileType.BZip2();
|
|
case SupportedFileType.Executable: return new FileType.Executable();
|
|
case SupportedFileType.GZIP: return new FileType.GZIP();
|
|
//case FileTypes.IniFile: return new FileType.IniFile();
|
|
case SupportedFileType.InstallShieldArchiveV3: return new FileType.InstallShieldArchiveV3();
|
|
case SupportedFileType.InstallShieldCAB: return new FileType.InstallShieldCAB();
|
|
case SupportedFileType.MicrosoftCAB: return new FileType.MicrosoftCAB();
|
|
case SupportedFileType.MPQ: return new FileType.MPQ();
|
|
case SupportedFileType.MSI: return new FileType.MSI();
|
|
case SupportedFileType.PKZIP: return new FileType.PKZIP();
|
|
case SupportedFileType.PLJ: return new FileType.PLJ();
|
|
case SupportedFileType.RAR: return new FileType.RAR();
|
|
case SupportedFileType.SevenZip: return new FileType.SevenZip();
|
|
case SupportedFileType.TapeArchive: return new FileType.TapeArchive();
|
|
case SupportedFileType.Textfile: return new FileType.Textfile();
|
|
case SupportedFileType.Valve: return new FileType.Valve();
|
|
case SupportedFileType.XZ: return new FileType.XZ();
|
|
default: return null;
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Processed Executable Information
|
|
|
|
/// <summary>
|
|
/// Get the internal version as reported by the filesystem
|
|
/// </summary>
|
|
/// <param name="file">File to check for version</param>
|
|
/// <returns>Version string, null on error</returns>
|
|
public static string GetInternalVersion(string file)
|
|
{
|
|
try
|
|
{
|
|
using (Stream fileStream = File.OpenRead(file))
|
|
{
|
|
var pex = PortableExecutable.Create(fileStream);
|
|
return GetInternalVersion(pex);
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
return string.Empty;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the internal version as reported by the resources
|
|
/// </summary>
|
|
/// <param name="pex">PortableExecutable representing the file contents</param>
|
|
/// <returns>Version string, null on error</returns>
|
|
public static string GetInternalVersion(PortableExecutable pex)
|
|
{
|
|
string version = pex.FileVersion;
|
|
if (!string.IsNullOrWhiteSpace(version))
|
|
return version.Replace(", ", ".");
|
|
|
|
version = pex.ProductVersion;
|
|
if (!string.IsNullOrWhiteSpace(version))
|
|
return version.Replace(", ", ".");
|
|
|
|
version = pex.AssemblyVersion;
|
|
if (!string.IsNullOrWhiteSpace(version))
|
|
return version;
|
|
|
|
return null;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Executable Information
|
|
|
|
/// <summary>
|
|
/// Get the SHA1 hash of a file, if possible
|
|
/// </summary>
|
|
/// <param name="path">Path to the file to be hashed</param>
|
|
/// <returns>SHA1 hash as a string on success, null on error</returns>
|
|
public static string GetFileSHA1(string path)
|
|
{
|
|
if (string.IsNullOrWhiteSpace(path))
|
|
return null;
|
|
|
|
try
|
|
{
|
|
SHA1 sha1 = SHA1.Create();
|
|
using (Stream fileStream = File.OpenRead(path))
|
|
{
|
|
byte[] buffer = new byte[32768];
|
|
while (true)
|
|
{
|
|
int bytesRead = fileStream.Read(buffer, 0, 32768);
|
|
if (bytesRead == 32768)
|
|
{
|
|
sha1.TransformBlock(buffer, 0, bytesRead, null, 0);
|
|
}
|
|
else
|
|
{
|
|
sha1.TransformFinalBlock(buffer, 0, bytesRead);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
string hash = BitConverter.ToString(sha1.Hash);
|
|
hash = hash.Replace("-", string.Empty);
|
|
return hash;
|
|
}
|
|
catch
|
|
{
|
|
return null;
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Wrappers for Matchers
|
|
|
|
/// <summary>
|
|
/// Wrapper for GetInternalVersion for use in path matching
|
|
/// </summary>
|
|
/// <param name="firstMatchedString">File to check for version</param>
|
|
/// <param name="files">Full list of input paths</param>
|
|
/// <returns>Version string, null on error</returns>
|
|
public static string GetInternalVersion(string firstMatchedString, IEnumerable<string> files) => GetInternalVersion(firstMatchedString);
|
|
|
|
#endregion
|
|
}
|
|
}
|