Skip to content

Commit 434a872

Browse files
author
Daniel Marbach
authored
AssemblyScanner improvements (#6824)
1 parent a5bcece commit 434a872

File tree

2 files changed

+53
-60
lines changed

2 files changed

+53
-60
lines changed

src/NServiceBus.Core.Tests/AssemblyScanner/AssemblyScannerTests.cs

+12-6
Original file line numberDiff line numberDiff line change
@@ -264,10 +264,13 @@ public void Skipped_dlls_should_be_excluded()
264264

265265
var scanner = new AssemblyScanner(DynamicAssembly.TestAssemblyDirectory)
266266
{
267-
CoreAssemblyName = busAssembly.DynamicName
267+
CoreAssemblyName = busAssembly.DynamicName,
268+
AssembliesToSkip = new[]
269+
{
270+
excludedAssembly1.DynamicName, // without file extension
271+
excludedAssembly2.FileName // with file extension
272+
}
268273
};
269-
scanner.AssembliesToSkip.Add(excludedAssembly1.DynamicName); // without file extension
270-
scanner.AssembliesToSkip.Add(excludedAssembly2.FileName); // with file extension
271274

272275
var result = scanner.GetScannableAssemblies();
273276
Assert.That(result.SkippedFiles.Any(s => s.FilePath == excludedAssembly1.FilePath));
@@ -294,10 +297,13 @@ public void Skipped_exes_should_be_excluded()
294297

295298
var scanner = new AssemblyScanner(DynamicAssembly.TestAssemblyDirectory)
296299
{
297-
CoreAssemblyName = busAssembly.DynamicName
300+
CoreAssemblyName = busAssembly.DynamicName,
301+
AssembliesToSkip = new[]
302+
{
303+
excludedAssembly1.DynamicName, // without file extension
304+
excludedAssembly2.FileName // with file extension
305+
}
298306
};
299-
scanner.AssembliesToSkip.Add(excludedAssembly1.DynamicName); // without file extension
300-
scanner.AssembliesToSkip.Add(excludedAssembly2.FileName); // with file extension
301307

302308
var result = scanner.GetScannableAssemblies();
303309
Assert.That(result.SkippedFiles.Any(s => s.FilePath == excludedAssembly1.FilePath));

src/NServiceBus.Core/Hosting/Helpers/AssemblyScanner.cs

+41-54
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,17 @@ internal AssemblyScanner(Assembly assemblyToScan)
5050
/// </summary>
5151
public bool ScanFileSystemAssemblies { get; set; } = true;
5252

53-
internal string CoreAssemblyName { get; set; } = NServicebusCoreAssemblyName;
53+
internal string CoreAssemblyName { get; set; } = NServiceBusCoreAssemblyName;
54+
55+
internal IReadOnlyCollection<string> AssembliesToSkip
56+
{
57+
set => assembliesToSkip = new HashSet<string>(value.Select(Path.GetFileNameWithoutExtension), StringComparer.OrdinalIgnoreCase);
58+
}
59+
60+
internal IReadOnlyCollection<Type> TypesToSkip
61+
{
62+
set => typesToSkip = new HashSet<Type>(value);
63+
}
5464

5565
internal string AdditionalAssemblyScanningPath { get; set; }
5666

@@ -246,7 +256,7 @@ internal static string FormatReflectionTypeLoadException(string fileName, Reflec
246256
{
247257
var sb = new StringBuilder($"Could not enumerate all types for '{fileName}'.");
248258

249-
if (!e.LoaderExceptions.Any())
259+
if (e.LoaderExceptions.Length == 0)
250260
{
251261
sb.NewLine($"Exception message: {e}");
252262
return sb.ToString();
@@ -327,65 +337,43 @@ static List<FileInfo> ScanDirectoryForAssemblyFiles(string directoryToScan, bool
327337
var searchOption = scanNestedDirectories ? SearchOption.AllDirectories : SearchOption.TopDirectoryOnly;
328338
foreach (var searchPattern in FileSearchPatternsToUse)
329339
{
330-
foreach (var info in baseDir.GetFiles(searchPattern, searchOption))
331-
{
332-
fileInfo.Add(info);
333-
}
340+
fileInfo.AddRange(baseDir.GetFiles(searchPattern, searchOption));
334341
}
335342
return fileInfo;
336343
}
337344

338-
bool IsExcluded(string assemblyNameOrFileName)
339-
{
340-
var isExplicitlyExcluded = AssembliesToSkip.Any(excluded => IsMatch(excluded, assemblyNameOrFileName));
341-
if (isExplicitlyExcluded)
342-
{
343-
return true;
344-
}
345-
346-
var isExcludedByDefault = DefaultAssemblyExclusions.Any(exclusion => IsMatch(exclusion, assemblyNameOrFileName));
347-
if (isExcludedByDefault)
348-
{
349-
return true;
350-
}
351-
352-
return false;
353-
}
354-
355-
static bool IsMatch(string expression1, string expression2)
356-
{
357-
return DistillLowerAssemblyName(expression1) == DistillLowerAssemblyName(expression2);
358-
}
345+
bool IsExcluded(string assemblyNameOrFileNameWithoutExtension) =>
346+
assembliesToSkip.Contains(assemblyNameOrFileNameWithoutExtension) ||
347+
DefaultAssemblyExclusions.Contains(assemblyNameOrFileNameWithoutExtension);
359348

360-
bool IsAllowedType(Type type)
349+
// The parameter and return types of this method are deliberately using the most concrete types
350+
// to avoid unnecessary allocations
351+
List<Type> FilterAllowedTypes(Type[] types)
361352
{
362-
return type != null &&
363-
!type.IsValueType &&
364-
!IsCompilerGenerated(type) &&
365-
!TypesToSkip.Contains(type);
366-
}
367-
368-
static bool IsCompilerGenerated(Type type)
369-
{
370-
return type.GetCustomAttribute<CompilerGeneratedAttribute>(false) != null;
371-
}
372-
373-
static string DistillLowerAssemblyName(string assemblyOrFileName)
374-
{
375-
var lowerAssemblyName = assemblyOrFileName.ToLowerInvariant();
376-
if (lowerAssemblyName.EndsWith(".dll") || lowerAssemblyName.EndsWith(".exe"))
353+
// assume the majority of types will be allowed to preallocate the list
354+
var allowedTypes = new List<Type>(types.Length);
355+
foreach (var typeToAdd in types)
377356
{
378-
lowerAssemblyName = lowerAssemblyName.Substring(0, lowerAssemblyName.Length - 4);
357+
if (IsAllowedType(typeToAdd))
358+
{
359+
allowedTypes.Add(typeToAdd);
360+
}
379361
}
380-
return lowerAssemblyName;
362+
return allowedTypes;
381363
}
382364

365+
bool IsAllowedType(Type type) =>
366+
type != null &&
367+
!type.IsValueType &&
368+
Attribute.GetCustomAttribute(type, typeof(CompilerGeneratedAttribute), false) == null &&
369+
!typesToSkip.Contains(type);
370+
383371
void AddTypesToResult(Assembly assembly, AssemblyScannerResults results)
384372
{
385373
try
386374
{
387-
//will throw if assembly cannot be loaded
388-
results.Types.AddRange(assembly.GetTypes().Where(IsAllowedType));
375+
var types = assembly.GetTypes();
376+
results.Types.AddRange(FilterAllowedTypes(types));
389377
}
390378
catch (ReflectionTypeLoadException e)
391379
{
@@ -398,7 +386,7 @@ void AddTypesToResult(Assembly assembly, AssemblyScannerResults results)
398386
}
399387

400388
LogManager.GetLogger<AssemblyScanner>().Warn(errorMessage);
401-
results.Types.AddRange(e.Types.Where(IsAllowedType));
389+
results.Types.AddRange(FilterAllowedTypes(e.Types));
402390
}
403391
results.Assemblies.Add(assembly);
404392
}
@@ -431,21 +419,20 @@ bool ShouldScanDependencies(Assembly assembly)
431419
}
432420

433421
AssemblyValidator assemblyValidator = new AssemblyValidator();
434-
internal List<string> AssembliesToSkip = new List<string>();
435422
internal bool ScanNestedDirectories;
436-
internal List<Type> TypesToSkip = new List<Type>();
437423
Assembly assemblyToScan;
438424
string baseDirectoryToScan;
439-
const string NServicebusCoreAssemblyName = "NServiceBus.Core";
425+
HashSet<Type> typesToSkip = new HashSet<Type>();
426+
HashSet<string> assembliesToSkip = new HashSet<string>(StringComparer.OrdinalIgnoreCase);
427+
const string NServiceBusCoreAssemblyName = "NServiceBus.Core";
440428

441-
static string[] FileSearchPatternsToUse =
429+
static readonly string[] FileSearchPatternsToUse =
442430
{
443431
"*.dll",
444432
"*.exe"
445433
};
446434

447-
//TODO: delete when we make message scanning lazy #1617
448-
static string[] DefaultAssemblyExclusions =
435+
static readonly HashSet<string> DefaultAssemblyExclusions = new HashSet<string>(StringComparer.OrdinalIgnoreCase)
449436
{
450437
// NSB Build-Dependencies
451438
"nunit",

0 commit comments

Comments
 (0)