mirror of
https://github.com/SabreTools/BinaryObjectScanner.git
synced 2026-02-14 13:46:03 +00:00
221 lines
10 KiB
C#
221 lines
10 KiB
C#
using System.Collections.Generic;
|
|
using System.Linq;
|
|
|
|
namespace BurnOutSharp.Matching
|
|
{
|
|
/// <summary>
|
|
/// Helper class for matching
|
|
/// </summary>
|
|
internal static class MatchUtil
|
|
{
|
|
#region Content Matching
|
|
|
|
/// <summary>
|
|
/// Get all content matches for a given list of matchers
|
|
/// </summary>
|
|
/// <param name="file">File to check for matches</param>
|
|
/// <param name="fileContent">Byte array representing the file contents</param>
|
|
/// <param name="matchers">Enumerable of matchers to be run on the file</param>
|
|
/// <param name="includePosition">True to include positional data, false otherwise</param>
|
|
/// <returns>List of strings representing the matched protections, null or empty otherwise</returns>
|
|
public static List<string> GetAllContentMatches(
|
|
string file,
|
|
byte[] fileContent,
|
|
IEnumerable<Matcher> matchers,
|
|
bool includePosition = false)
|
|
{
|
|
return FindAllContentMatches(file, fileContent, matchers, includePosition, false);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get first content match for a given list of matchers
|
|
/// </summary>
|
|
/// <param name="file">File to check for matches</param>
|
|
/// <param name="fileContent">Byte array representing the file contents</param>
|
|
/// <param name="matchers">Enumerable of matchers to be run on the file</param>
|
|
/// <param name="includePosition">True to include positional data, false otherwise</param>
|
|
/// <returns>String representing the matched protection, null otherwise</returns>
|
|
public static string GetFirstContentMatch(
|
|
string file,
|
|
byte[] fileContent,
|
|
IEnumerable<Matcher> matchers,
|
|
bool includePosition = false)
|
|
{
|
|
var contentMatches = FindAllContentMatches(file, fileContent, matchers, includePosition, true);
|
|
if (contentMatches == null || !contentMatches.Any())
|
|
return null;
|
|
|
|
return contentMatches.First();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the required set of content matches on a per Matcher basis
|
|
/// </summary>
|
|
/// <param name="file">File to check for matches</param>
|
|
/// <param name="fileContent">Byte array representing the file contents</param>
|
|
/// <param name="matchers">Enumerable of matchers to be run on the file</param>
|
|
/// <param name="includePosition">True to include positional data, false otherwise</param>
|
|
/// <param name="stopAfterFirst">True to stop after the first match, false otherwise</param>
|
|
/// <returns>List of strings representing the matched protections, null or empty otherwise</returns>
|
|
private static List<string> FindAllContentMatches(
|
|
string file,
|
|
byte[] fileContent,
|
|
IEnumerable<Matcher> matchers,
|
|
bool includePosition,
|
|
bool stopAfterFirst)
|
|
{
|
|
// If there's no mappings, we can't match
|
|
if (matchers == null || !matchers.Any())
|
|
return null;
|
|
|
|
// Initialize the list of matched protections
|
|
List<string> matchedProtections = new List<string>();
|
|
|
|
// Loop through and try everything otherwise
|
|
foreach (var matcher in matchers)
|
|
{
|
|
// Determine if the matcher passes
|
|
(bool passes, List<int> positions) = matcher.MatchesAll(fileContent);
|
|
if (!passes)
|
|
continue;
|
|
|
|
// Format the list of all positions found
|
|
string positionsString = string.Join(", ", positions);
|
|
|
|
// If we there is no version method, just return the protection name
|
|
if (matcher.GetContentVersion == null)
|
|
{
|
|
matchedProtections.Add((matcher.ProtectionName ?? "Unknown Protection") + (includePosition ? $" (Index {positionsString})" : string.Empty));
|
|
}
|
|
|
|
// Otherwise, invoke the version method
|
|
// TODO: Pass all positions to the version finding method
|
|
else
|
|
{
|
|
string version = matcher.GetContentVersion(file, fileContent, positions[0]) ?? "Unknown Version";
|
|
matchedProtections.Add($"{matcher.ProtectionName ?? "Unknown Protection"} {version}" + (includePosition ? $" (Index {positionsString})" : string.Empty));
|
|
}
|
|
|
|
// If we're stopping after the first protection, bail out here
|
|
if (stopAfterFirst)
|
|
return matchedProtections;
|
|
}
|
|
|
|
return matchedProtections;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Path Matching
|
|
|
|
/// <summary>
|
|
/// Get all path matches for a given list of matchers
|
|
/// </summary>
|
|
/// <param name="file">File path to check for matches</param>
|
|
/// <param name="matchers">Enumerable of matchers to be run on the file</param>
|
|
/// <param name="any">True if any path match is a success, false if all have to match</param>
|
|
/// <returns>List of strings representing the matched protections, null or empty otherwise</returns>
|
|
public static List<string> GetAllPathMatches(string file, IEnumerable<Matcher> matchers, bool any = false)
|
|
{
|
|
return FindAllPathMatches(new List<string> { file }, matchers, any, false);
|
|
}
|
|
|
|
// <summary>
|
|
/// Get all path matches for a given list of matchers
|
|
/// </summary>
|
|
/// <param name="files">File paths to check for matches</param>
|
|
/// <param name="matchers">Enumerable of matchers to be run on the file</param>
|
|
/// <param name="any">True if any path match is a success, false if all have to match</param>
|
|
/// <returns>List of strings representing the matched protections, null or empty otherwise</returns>
|
|
public static List<string> GetAllPathMatches(List<string> files, IEnumerable<Matcher> matchers, bool any = false)
|
|
{
|
|
return FindAllPathMatches(files, matchers, any, false);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get first path match for a given list of matchers
|
|
/// </summary>
|
|
/// <param name="file">File path to check for matches</param>
|
|
/// <param name="matchers">Enumerable of matchers to be run on the file</param>
|
|
/// <param name="any">True if any path match is a success, false if all have to match</param>
|
|
/// <returns>String representing the matched protection, null otherwise</returns>
|
|
public static string GetFirstPathMatch(string file, IEnumerable<Matcher> matchers, bool any = false)
|
|
{
|
|
var contentMatches = FindAllPathMatches(new List<string> { file }, matchers, any, true);
|
|
if (contentMatches == null || !contentMatches.Any())
|
|
return null;
|
|
|
|
return contentMatches.First();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get first path match for a given list of matchers
|
|
/// </summary>
|
|
/// <param name="files">File paths to check for matches</param>
|
|
/// <param name="matchers">Enumerable of matchers to be run on the file</param>
|
|
/// <param name="any">True if any path match is a success, false if all have to match</param>
|
|
/// <returns>String representing the matched protection, null otherwise</returns>
|
|
public static string GetFirstPathMatch(List<string> files, IEnumerable<Matcher> matchers, bool any = false)
|
|
{
|
|
var contentMatches = FindAllPathMatches(files, matchers, any, true);
|
|
if (contentMatches == null || !contentMatches.Any())
|
|
return null;
|
|
|
|
return contentMatches.First();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the required set of path matches on a per Matcher basis
|
|
/// </summary>
|
|
/// <param name="files">File paths to check for matches</param>
|
|
/// <param name="matchers">Enumerable of matchers to be run on the file</param>
|
|
/// <param name="any">True if any path match is a success, false if all have to match</param>
|
|
/// <param name="stopAfterFirst">True to stop after the first match, false otherwise</param>
|
|
/// <returns>List of strings representing the matched protections, null or empty otherwise</returns>
|
|
private static List<string> FindAllPathMatches(List<string> files, IEnumerable<Matcher> matchers, bool any, bool stopAfterFirst)
|
|
{
|
|
// If there's no mappings, we can't match
|
|
if (matchers == null || !matchers.Any())
|
|
return null;
|
|
|
|
// Initialize the list of matched protections
|
|
List<string> matchedProtections = new List<string>();
|
|
|
|
// Loop through and try everything otherwise
|
|
foreach (var matcher in matchers)
|
|
{
|
|
// Determine if the matcher passes
|
|
bool passes;
|
|
if (any)
|
|
(passes, _) = matcher.MatchesAny(files);
|
|
else
|
|
(passes, _) = matcher.MatchesAll(files);
|
|
|
|
// If we don't have a pass, just continue
|
|
if (!passes)
|
|
continue;
|
|
|
|
// If we there is no version method, just return the protection name
|
|
if (matcher.GetPathVersion == null)
|
|
{
|
|
matchedProtections.Add(matcher.ProtectionName ?? "Unknown Protection");
|
|
}
|
|
|
|
// Otherwise, invoke the version method
|
|
else
|
|
{
|
|
string version = matcher.GetPathVersion(files) ?? "Unknown Version";
|
|
matchedProtections.Add($"{matcher.ProtectionName ?? "Unknown Protection"} {version}");
|
|
}
|
|
|
|
// If we're stopping after the first protection, bail out here
|
|
if (stopAfterFirst)
|
|
return matchedProtections;
|
|
}
|
|
|
|
return matchedProtections;
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
} |