@@ -50,7 +50,17 @@ internal AssemblyScanner(Assembly assemblyToScan)
50
50
/// </summary>
51
51
public bool ScanFileSystemAssemblies { get ; set ; } = true ;
52
52
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
+ }
54
64
55
65
internal string AdditionalAssemblyScanningPath { get ; set ; }
56
66
@@ -246,7 +256,7 @@ internal static string FormatReflectionTypeLoadException(string fileName, Reflec
246
256
{
247
257
var sb = new StringBuilder ( $ "Could not enumerate all types for '{ fileName } '.") ;
248
258
249
- if ( ! e . LoaderExceptions . Any ( ) )
259
+ if ( e . LoaderExceptions . Length == 0 )
250
260
{
251
261
sb . NewLine ( $ "Exception message: { e } ") ;
252
262
return sb . ToString ( ) ;
@@ -327,65 +337,43 @@ static List<FileInfo> ScanDirectoryForAssemblyFiles(string directoryToScan, bool
327
337
var searchOption = scanNestedDirectories ? SearchOption . AllDirectories : SearchOption . TopDirectoryOnly ;
328
338
foreach ( var searchPattern in FileSearchPatternsToUse )
329
339
{
330
- foreach ( var info in baseDir . GetFiles ( searchPattern , searchOption ) )
331
- {
332
- fileInfo . Add ( info ) ;
333
- }
340
+ fileInfo . AddRange ( baseDir . GetFiles ( searchPattern , searchOption ) ) ;
334
341
}
335
342
return fileInfo ;
336
343
}
337
344
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 ) ;
359
348
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 )
361
352
{
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 )
377
356
{
378
- lowerAssemblyName = lowerAssemblyName . Substring ( 0 , lowerAssemblyName . Length - 4 ) ;
357
+ if ( IsAllowedType ( typeToAdd ) )
358
+ {
359
+ allowedTypes . Add ( typeToAdd ) ;
360
+ }
379
361
}
380
- return lowerAssemblyName ;
362
+ return allowedTypes ;
381
363
}
382
364
365
+ bool IsAllowedType ( Type type ) =>
366
+ type != null &&
367
+ ! type . IsValueType &&
368
+ Attribute . GetCustomAttribute ( type , typeof ( CompilerGeneratedAttribute ) , false ) == null &&
369
+ ! typesToSkip . Contains ( type ) ;
370
+
383
371
void AddTypesToResult ( Assembly assembly , AssemblyScannerResults results )
384
372
{
385
373
try
386
374
{
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 ) ) ;
389
377
}
390
378
catch ( ReflectionTypeLoadException e )
391
379
{
@@ -398,7 +386,7 @@ void AddTypesToResult(Assembly assembly, AssemblyScannerResults results)
398
386
}
399
387
400
388
LogManager . GetLogger < AssemblyScanner > ( ) . Warn ( errorMessage ) ;
401
- results . Types . AddRange ( e . Types . Where ( IsAllowedType ) ) ;
389
+ results . Types . AddRange ( FilterAllowedTypes ( e . Types ) ) ;
402
390
}
403
391
results . Assemblies . Add ( assembly ) ;
404
392
}
@@ -431,21 +419,20 @@ bool ShouldScanDependencies(Assembly assembly)
431
419
}
432
420
433
421
AssemblyValidator assemblyValidator = new AssemblyValidator ( ) ;
434
- internal List < string > AssembliesToSkip = new List < string > ( ) ;
435
422
internal bool ScanNestedDirectories ;
436
- internal List < Type > TypesToSkip = new List < Type > ( ) ;
437
423
Assembly assemblyToScan ;
438
424
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" ;
440
428
441
- static string [ ] FileSearchPatternsToUse =
429
+ static readonly string [ ] FileSearchPatternsToUse =
442
430
{
443
431
"*.dll" ,
444
432
"*.exe"
445
433
} ;
446
434
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 )
449
436
{
450
437
// NSB Build-Dependencies
451
438
"nunit" ,
0 commit comments