16

我正在编写一个 Asp.Net Core 应用程序,它应该能够自我更新(在运行时替换它自己的二进制文件)。

这篇 MSDN 文章描述了使用经典 .Net 框架的卷影复制,这正是我所需要的。但是 .Net Core 中缺少整个 AppDomain 的东西。

所以我的问题是:

  • .Net Core 中是否有另一种方法可以启用影子复制程序集?
  • .Net Core 中是否还有其他机制允许构建自更新应用程序?
4

4 回答 4

21

由于 .NET Core 中没有内置机制来执行此操作,因此我最终实现了自己的自定义解决方案。它的工作原理大致如下:

  1. 正在运行的应用程序下载新的二进制文件并将其解压缩到新文件夹中。
  2. 正在运行的应用程序启动一个小的更新程序进程。以下参数通过命令行传递给更新程序进程:
    • 正在运行的应用程序的进程 ID
    • 正在运行的应用程序的二进制路径
    • 下载的二进制文件的路径
  3. 正在运行的应用程序自行退出。
  4. 更新程序会一直等待,直到正在运行的应用程序退出(使用进程 ID),或者如果它在给定的超时时间内没有自行退出,则强制终止正在运行的应用程序。
  5. 更新程序进程删除现有的二进制文件并复制新下载的二进制文件。
  6. 更新进程启动主应用程序的新版本。

确保您在主应用程序中尽可能多地进行操作(下载、解包、验证...),并使更新程序过程尽可能简单(将失败的风险降至最低)。

这种方法已被证明是相当稳定的。

于 2018-01-30T07:19:13.613 回答
5

.NET Core 中没有内置影子复制工具

于 2017-03-16T09:37:11.043 回答
2

.Net API Browser指示在 .Net Core 中设置此属性所需的属性是,但 AppDomainSetup 不是。

需要明确的是,AppDomain 是在 .Net Standard 2.0 中添加的,但目前不支持创建域

于 2018-07-19T17:24:23.217 回答
1

为了节省必须做我刚刚做的事情的人并做到这一点 - 这只会复制具有不同日期修改时间的文件。我检查并重建您的应用程序只会在几个文件上更改此内容。这使得自加载程序非常快速,然后在新位置启动 exe,并退出执行从旧位置运行的加载的 exe。这可能取决于一些事情,例如运行代码的 DLL 必须与启动它的 EXE 命名相同。

在 .Net 5 中工作:

using System;
using System.Diagnostics;
using System.IO;

namespace NetworkHelper
{
    public static class LocalCopier
    {
        public static void EnsureRunningLocally(string callingAssemblyDotLocation)
        {
            var assemblyFileFriendlyName = Path.GetFileName(callingAssemblyDotLocation.Replace(".", "-"));
            var assemblyDirToCheck = Path.GetDirectoryName(callingAssemblyDotLocation);
            var localLocation = Configuration.Tools.AppsLocation + assemblyFileFriendlyName + "\\";
            var assemblyFinalExePath = localLocation + assemblyFileFriendlyName.Replace("-dll", ".exe"); 
            
            // Check what assembly passed in path starts with
            var runningFromNetwork = callingAssemblyDotLocation.ToLower().StartsWith(@"\\w2k3nas1\");
            if (callingAssemblyDotLocation.ToLower().StartsWith(@"i:\"))  runningFromNetwork = true;

            if (!runningFromNetwork) return;
            
            // Check if copied to local already
            Directory.CreateDirectory(localLocation);

            // Foreach file in source dir, recursively
            CopyOnlyDifferentFiles(assemblyDirToCheck, localLocation);

            Process.Start(assemblyFinalExePath);
            
            Environment.Exit(0);
        }

        private static void CopyOnlyDifferentFiles(string sourceFolderPath, string destinationFolderPath)
        {
            string[] originalFiles = Directory.GetFiles(sourceFolderPath, "*", SearchOption.AllDirectories);

            Array.ForEach(originalFiles, (originalFileLocation) =>
            {
                FileInfo originalFile = new FileInfo(originalFileLocation);
                FileInfo destFile = new FileInfo(originalFileLocation.Replace(sourceFolderPath, destinationFolderPath));

                if (destFile.Exists)
                {
                    if (originalFile.LastWriteTime != destFile.LastWriteTime)
                    {
                        originalFile.CopyTo(destFile.FullName, true);
                    }
                }
                else
                {
                    Directory.CreateDirectory(destFile.DirectoryName);
                    originalFile.CopyTo(destFile.FullName, false);
                }
            });
        }
    }
}

请注意,“\w2k3nas1”和“i:”是网络位置的示例,如果从这些位置运行,它应该将自身复制到本地目录,我使用应用程序数据/漫游/localApps,然后从新目录重新启动自身。

这都可以放入参考库并从任何客户端应用程序调用: NetworkHelpers.LocalCopier.EnsureRunningLocally(Assembly.GetExecutingAssembly().Location);

(这里,Assembly.GetExecutingAssembly().Location 是从调用应用程序传入的,因为如果您要在参考项目中运行它,您将获得该库的 dll。)

于 2021-03-31T22:17:18.627 回答