问题 在运行时加载nuget依赖项


我正在寻找一种通过执行以下步骤来运行代码的方法:

  1. 接收NuGet包列表(元组列表(“包名”,“包版本”,“主类路径”)。
  2. 在本地目录中检索它们(参见代码示例#1)
  3. 在运行时将它们加载到我的程序中
  4. 通过内省运行主类(参见代码示例#2)

到现在为止,我正在努力迈出第三步。我无法在运行时找到如何加载我的包。

我的主要问题是:

  • 如何找出存储检索到的包的文件夹?
  • 如何将这些目录的内容加载到我的程序中?

谢谢你的帮助。

代码示例#1:

private static void getPackageByNameAndVersion(string packageID, string version)
{
    IPackageRepository repo = 
            PackageRepositoryFactory.Default                          
                  .CreateRepository("https://packages.nuget.org/api/v2");

   string path = "C:/tmp_repo";
   PackageManager packageManager = new PackageManager(repo, path);
   Console.WriteLine("before dl pkg");
   packageManager.InstallPackage(packageID, SemanticVersion.Parse(version));

}

代码示例#2:

private static void loadByAssemblyNameAndTypeName(string assemblyName, string typeName)
{
   AppDomain isolationAppDomain = AppDomain.CreateDomain("tmp");
   object a = isolationAppDomain.CreateInstanceAndUnwrap(assemblyName, typeName);
   Type x = a.GetType();
   MethodInfo m = x.GetMethod("Main");
   m.Invoke(a, new object[] { });
}

4305
2017-08-06 15:03


起源



答案:


拿一杯咖啡:)

下载nuget包?

Nuget.Core(nuget包)是一个不错的选择,这里有一段代码,我应该可以通过以下方式下载一个nuget包 id 和 version

var repo = PackageRepositoryFactory.Default
                .CreateRepository("https://packages.nuget.org/api/v2");

string path = "c:\\temp";
var packageManager = new PackageManager(repo, path);
packageManager.PackageInstalled += PackageManager_PackageInstalled;

var package = repo.FindPackage("packageName", SemanticVersion.Parse("1.0.0"));
if (package != null)
{
    packageManager.InstallPackage(package, false, true);
}

请注意,我将事件处理程序插入到 PackageInstalled 的事件 PackageManager 类。

我们如何在隔离的应用程序域中加载程序集?

由于反射API不提供在特定域中加载程序集的方法,因此我们将创建一个代理类,在我们的隔离域中充当加载器:

public class TypeProxy : MarshalByRefObject
{
    public Type LoadFromAssembly(string assemblyPath, string typeName)
    {
        try
        {
            var asm = Assembly.LoadFile(assemblyPath);
            return asm.GetType(typeName);
        }
        catch (Exception) { return null; }
    }
}

而现在,是如何把它们放在一起的?

这是复杂的部分:

private static void PackageManager_PackageInstalled(object sender, 
                                                    PackageOperationEventArgs e)
{
    var files = e.FileSystem.GetFiles(e.InstallPath, "*.dll", true);
    foreach (var file in files)
    {
        try
        {
            AppDomain domain = AppDomain.CreateDomain("tmp");
            Type typeProxyType = typeof(TypeProxy);
            var typeProxyInstance = (TypeProxy)domain.CreateInstanceAndUnwrap(
                    typeProxyType.Assembly.FullName,
                    typeProxyType.FullName);

            var type = typeProxyInstance.LoadFromAssembly(file, "<KnownTypeName>");
            object instance = 
                domain.CreateInstanceAndUnwrap(type.Assembly.FullName, type.FullName);
        }
        catch (Exception ex)
        {
            Console.WriteLine("failed to load {0}", file);
            Console.WriteLine(ex.ToString());
        }

    }
}

请注意,此方法是下载nuget包后执行的事件处理程序

请注意,您需要更换 <KnownTypeName> 具有来自程序集的预期类型名称(或者可能在程序集中运行所有公共类型的发现)


值得注意的是,我自己并没有执行此代码,也无法保证它能够开箱即用,仍然可能需要进行一些调整。但希望它是允许您解决问题的概念。


13
2017-08-12 01:56



您的示例的一个问题是PackageManager_PackageInstalled仅在第一次在c:\\ temp中安装了被询问的包时被调用。如果它已经存在于先前的代码执行中,则不会调用PackageManager_PackageInstalled。 - Manuel Leduc


答案:


拿一杯咖啡:)

下载nuget包?

Nuget.Core(nuget包)是一个不错的选择,这里有一段代码,我应该可以通过以下方式下载一个nuget包 id 和 version

var repo = PackageRepositoryFactory.Default
                .CreateRepository("https://packages.nuget.org/api/v2");

string path = "c:\\temp";
var packageManager = new PackageManager(repo, path);
packageManager.PackageInstalled += PackageManager_PackageInstalled;

var package = repo.FindPackage("packageName", SemanticVersion.Parse("1.0.0"));
if (package != null)
{
    packageManager.InstallPackage(package, false, true);
}

请注意,我将事件处理程序插入到 PackageInstalled 的事件 PackageManager 类。

我们如何在隔离的应用程序域中加载程序集?

由于反射API不提供在特定域中加载程序集的方法,因此我们将创建一个代理类,在我们的隔离域中充当加载器:

public class TypeProxy : MarshalByRefObject
{
    public Type LoadFromAssembly(string assemblyPath, string typeName)
    {
        try
        {
            var asm = Assembly.LoadFile(assemblyPath);
            return asm.GetType(typeName);
        }
        catch (Exception) { return null; }
    }
}

而现在,是如何把它们放在一起的?

这是复杂的部分:

private static void PackageManager_PackageInstalled(object sender, 
                                                    PackageOperationEventArgs e)
{
    var files = e.FileSystem.GetFiles(e.InstallPath, "*.dll", true);
    foreach (var file in files)
    {
        try
        {
            AppDomain domain = AppDomain.CreateDomain("tmp");
            Type typeProxyType = typeof(TypeProxy);
            var typeProxyInstance = (TypeProxy)domain.CreateInstanceAndUnwrap(
                    typeProxyType.Assembly.FullName,
                    typeProxyType.FullName);

            var type = typeProxyInstance.LoadFromAssembly(file, "<KnownTypeName>");
            object instance = 
                domain.CreateInstanceAndUnwrap(type.Assembly.FullName, type.FullName);
        }
        catch (Exception ex)
        {
            Console.WriteLine("failed to load {0}", file);
            Console.WriteLine(ex.ToString());
        }

    }
}

请注意,此方法是下载nuget包后执行的事件处理程序

请注意,您需要更换 <KnownTypeName> 具有来自程序集的预期类型名称(或者可能在程序集中运行所有公共类型的发现)


值得注意的是,我自己并没有执行此代码,也无法保证它能够开箱即用,仍然可能需要进行一些调整。但希望它是允许您解决问题的概念。


13
2017-08-12 01:56



您的示例的一个问题是PackageManager_PackageInstalled仅在第一次在c:\\ temp中安装了被询问的包时被调用。如果它已经存在于先前的代码执行中,则不会调用PackageManager_PackageInstalled。 - Manuel Leduc


不要那样做! 您可能正在尝试在客户计算机上加载nugets以节省一些软件分发空间。不是吗?

常见的推荐方法是下载nuget作为自动构建的第二步(下载源代码后),构建软件并使用已下载的nugets运行自动化测试。然后使用您测试过的复杂整体单元分配构建。


-1
2017-08-15 06:29



我理解您的担忧,但我正在编写的软件的主要目标是在运行时加载软件组件。所以加载的代码将是我在构建时无法知道的第三方。但无论如何,谢谢你这个明智的建议:) - Manuel Leduc