-
Notifications
You must be signed in to change notification settings - Fork 1.9k
ComponentCatalog refactor #970
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Changes from all commits
b7097a2
8d3ff2c
bc35e8c
9f5814d
7cd4e42
f494e3b
a19a08e
56dba0c
22217a9
b435610
34d663c
022504a
430c4a7
a09321c
92971a4
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
@@ -0,0 +1,290 @@ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// Licensed to the .NET Foundation under one or more agreements. | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// The .NET Foundation licenses this file to you under the MIT license. | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// See the LICENSE file in the project root for more information. | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
using Microsoft.ML.Runtime.Internal.Utilities; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
using System; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
using System.IO; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
using System.IO.Compression; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
using System.Reflection; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
namespace Microsoft.ML.Runtime | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
internal static class AssemblyLoadingUtils | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
/// <summary> | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
/// Make sure the given assemblies are loaded and that their loadable classes have been catalogued. | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
/// </summary> | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
public static void LoadAndRegister(IHostEnvironment env, string[] assemblies) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Contracts.AssertValue(env); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if (Utils.Size(assemblies) > 0) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
foreach (string path in assemblies) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Exception ex = null; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
try | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// REVIEW: Will LoadFrom ever return null? | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Contracts.CheckNonEmpty(path, nameof(path)); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
var assem = LoadAssembly(env, path); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if (assem != null) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
continue; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
catch (Exception e) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
ex = e; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// If it is a zip file, load it that way. | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
ZipArchive zip; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
try | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
zip = ZipFile.OpenRead(path); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
catch (Exception e) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// Couldn't load as an assembly and not a zip, so warn the user. | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
ex = ex ?? e; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Console.Error.WriteLine("Warning: Could not load '{0}': {1}", path, ex.Message); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We should not be writing to the Console from libraries. And more generally, why do we swallow this error? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This code was copied out of the ComponentCatalog and moved to an internal class that is only called from the command line. (see src/Microsoft.ML.Maml/HelpCommand.cs and src/Microsoft.ML.ResultProcessor/ResultProcessor.cs) I wasn't going to change the behavior, because I'm sure someone is depending on it. |
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
continue; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
string dir; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
try | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
dir = CreateTempDirectory(); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
catch (Exception e) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
throw Contracts.ExceptIO(e, "Creating temp directory for extra assembly zip extraction failed: '{0}'", path); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
try | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
zip.ExtractToDirectory(dir); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
catch (Exception e) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
throw Contracts.ExceptIO(e, "Extracting extra assembly zip failed: '{0}'", path); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
LoadAssembliesInDir(env, dir, false); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why do we use zip files and just blast them into a folder instead using Nuget? It only works if the extensions don't use other dependencies. The moment they use other dependencies, we need to worry about versions. Nuget (theoretically) handles all of this. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This is existing behavior that I am refactoring out of ComponentCatalog so it isn't part of the public API anymore. It is only called from 2 places in the command-line (HelpCommand and ResultProcessor).
machinelearning/src/Microsoft.ML.ResultProcessor/ResultProcessor.cs Lines 334 to 336 in 160b0df
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Who deletes the temp path and when? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I guess nobody. This is an internal API that is only called from 2 places: HelpCommand and ResultProcessor, so it isn't part of our public API after this change. |
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
public static IDisposable CreateAssemblyRegistrar(IHostEnvironment env, string loadAssembliesPath = null) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. So the only thing that can be done with the registrar is to dispose it? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Correct. This is an internal method that is only invoked on that command-line. It creates an object that listens for new assemblies to be loaded, and registers them automatically. |
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Contracts.CheckValue(env, nameof(env)); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
env.CheckValueOrNull(loadAssembliesPath); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return new AssemblyRegistrar(env, loadAssembliesPath); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
public static void RegisterCurrentLoadedAssemblies(IHostEnvironment env) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Contracts.CheckValue(env, nameof(env)); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
foreach (Assembly a in AppDomain.CurrentDomain.GetAssemblies()) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This will attempt to register lots of assemblies (e.g. all framework assemblies). Isn't it wasteful? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Good point. I brought back the code that checks if the assembly references the assembly containing the machinelearning/src/Microsoft.ML.Core/ComponentModel/ComponentCatalog.cs Lines 410 to 421 in 6812cb5
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
TryRegisterAssembly(env.ComponentCatalog, a); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
private static string CreateTempDirectory() | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
string dir = GetTempPath(); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Directory.CreateDirectory(dir); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return dir; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
private static string GetTempPath() | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Guid guid = Guid.NewGuid(); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return Path.GetFullPath(Path.Combine(Path.GetTempPath(), "MLNET_" + guid.ToString())); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
private static readonly string[] _filePrefixesToAvoid = new string[] { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"api-ms-win", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"clr", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"coreclr", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"dbgshim", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"ext-ms-win", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"microsoft.bond.", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"microsoft.cosmos.", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"microsoft.csharp", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"microsoft.data.", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"microsoft.hpc.", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"microsoft.live.", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"microsoft.platformbuilder.", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"microsoft.visualbasic", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"microsoft.visualstudio.", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"microsoft.win32", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"microsoft.windowsapicodepack.", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"microsoft.windowsazure.", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"mscor", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"msvc", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"petzold.", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"roslyn.", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"sho", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"sni", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"sqm", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"system.", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
"zlib", | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
}; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
private static bool ShouldSkipPath(string path) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
string name = Path.GetFileName(path).ToLowerInvariant(); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
switch (name) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "cqo.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. How did we come up with the list? Are we should the list does not have items that are not needed anymore? Might be at least worth to add a comment. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This is the list in the current code. I just moved the list out of the ComponentCatalog class. machinelearning/src/Microsoft.ML.Core/ComponentModel/ComponentCatalog.cs Lines 307 to 333 in 6812cb5
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "fasttreenative.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "libiomp5md.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "libvw.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "matrixinterf.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "microsoft.ml.neuralnetworks.gpucuda.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "mklimports.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "microsoft.research.controls.decisiontrees.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "microsoft.ml.neuralnetworks.sse.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "neuraltreeevaluator.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "optimizationbuilderdotnet.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "parallelcommunicator.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "microsoft.ml.runtime.runtests.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "scopecompiler.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "tbb.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "internallearnscope.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "unmanagedlib.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "vcclient.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "libxgboost.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "zedgraph.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "__scopecodegen__.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
case "cosmosClientApi.dll": | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return true; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
foreach (var s in _filePrefixesToAvoid) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if (name.StartsWith(s, StringComparison.OrdinalIgnoreCase)) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return true; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return false; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
private static void LoadAssembliesInDir(IHostEnvironment env, string dir, bool filter) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if (!Directory.Exists(dir)) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why do you silently return? Why not throw or at least debug assert. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Because there are cases where this is called on potentially non-existing directories, for example here where we try calling it on the |
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
using (var ch = env.Start("LoadAssembliesInDir")) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// Load all dlls in the given directory. | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
var paths = Directory.EnumerateFiles(dir, "*.dll"); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
foreach (string path in paths) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if (filter && ShouldSkipPath(path)) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
ch.Info($"Skipping assembly '{path}' because its name was filtered."); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
continue; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
LoadAssembly(env, path); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
/// <summary> | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
/// Given an assembly path, load the assembly and register it with the ComponentCatalog. | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
/// </summary> | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
private static Assembly LoadAssembly(IHostEnvironment env, string path) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Assembly assembly = null; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
try | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
assembly = Assembly.LoadFrom(path); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
catch (Exception e) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
using (var ch = env.Start("LoadAssembly")) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
ch.Error("Could not load assembly {0}:\n{1}", path, e.ToString()); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return null; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if (assembly != null) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
TryRegisterAssembly(env.ComponentCatalog, assembly); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return assembly; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
/// <summary> | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
/// Checks whether <paramref name="assembly"/> references the assembly containing LoadableClassAttributeBase, | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
/// and therefore can contain components. | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
/// </summary> | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
private static bool CanContainComponents(Assembly assembly) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
var targetFullName = typeof(LoadableClassAttributeBase).Assembly.GetName().FullName; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
bool found = false; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
foreach (var name in assembly.GetReferencedAssemblies()) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if (name.FullName == targetFullName) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
found = true; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
break; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return found; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
private static void TryRegisterAssembly(ComponentCatalog catalog, Assembly assembly) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// Don't try to index dynamic generated assembly | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if (assembly.IsDynamic) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if (!CanContainComponents(assembly)) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
catalog.RegisterAssembly(assembly); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
private sealed class AssemblyRegistrar : IDisposable | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
private readonly IHostEnvironment _env; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
public AssemblyRegistrar(IHostEnvironment env, string path) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
_env = env; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
RegisterCurrentLoadedAssemblies(_env); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if (!string.IsNullOrEmpty(path)) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
LoadAssembliesInDir(_env, path, true); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
path = Path.Combine(path, "AutoLoad"); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
LoadAssembliesInDir(_env, path, true); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
AppDomain.CurrentDomain.AssemblyLoad += CurrentDomainAssemblyLoad; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
public void Dispose() | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
AppDomain.CurrentDomain.AssemblyLoad -= CurrentDomainAssemblyLoad; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
private void CurrentDomainAssemblyLoad(object sender, AssemblyLoadEventArgs args) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
TryRegisterAssembly(_env.ComponentCatalog, args.LoadedAssembly); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why do we try to load zip files? Why not either load or unzip depending on the extension?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Same as above - existing behavior from the command line code that was removed from ComponentCatalog.