7

我正在为 .NET 4.0 ClickOnce WPF 应用程序自动化安装程序,该应用程序需要在app.config文件中设置一些项目。我已经经历了寻找使用Mage.exe必须遵循的特定步骤的棘手过程(即更新和重新签署应用程序和部署清单),现在正尝试将其自动化以进行安装。

我选择使用.deploy扩展来最大程度地减少IIS / Internet Explorer 安全机制的问题,所以基本上算法如下(基于ClickOnce 中的签名和重新签名清单(Saurabh Bhatia)使用 Mage 更新 ClickOnce WPF 应用程序的配置或 MageUI,作为主要来源):

  1. 转到\Application Files\App_%HighestVersion%\文件夹
  2. 删除具有它的文件的.deploy扩展名
  3. 跑  mage -u %app%.exe.manifest -cf cert.pfx
  4. 恢复.deploy扩展
  5. 跑  mage -u %app%.application -appm %app%.exe.manifest -cf cert.pfx
  6. 向上复制%app%.application2 个级别(到..\..- 部署根目录)

如果手动完成,那效果很好。我可以运行一个.cmd文件,针对环境细节(路径等)进行定制,但随后我需要将其包含mage.exe在部署中,微软是否允许我们这样做对我来说是一个悬而未决的问题。因此,我试图在Installer课堂上执行类似的操作:

X509Certificate2 ct = new X509Certificate2(sPathCert);

//  .. Remove .deploy extension (for files in the sPathApp folder).

sPathMft = Directory.GetFiles(sPathApp, "*.exe.manifest")[0];
ApplicationManifest am = ManifestReader.ReadManifest( "ApplicationManifest", sPathMft, false ) as ApplicationManifest;
if (am == null)
    throw new ArgumentNullException("AppManifest");
am.ResolveFiles();
am.UpdateFileInfo( );
ManifestWriter.WriteManifest(am, sPathMft);
SecurityUtilities.SignFile(ct, null, sPathMft);

//    .. Restore .deploy extensions to files touched above.

sPathMft = Directory.GetFiles(sPathApp, "*.application")[0];
DeployManifest dm = ManifestReader.ReadManifest("DeployManifest", sPathMft, false) as DeployManifest;
if (dm == null)
    throw new ArgumentNullException( "DplManifest" );
dm.ResolveFiles();
dm.UpdateFileInfo();
ManifestWriter.WriteManifest(dm, sPathMft);
SecurityUtilities.SignFile(ct, null, sPathMft);

File.Copy(sPathMft, sPathBin + "\\" + dm.AssemblyIdentity.Name, true);

现在,这是踢球者。除第 5 步外,一切正常。当应用程序下载到用户计算机时,部署清单出现问题:

  • 部署清单在语义上无效。
  • 部署清单缺少 <compatibleFrameworks>。

实际上,这部分不再存在(但是,它在原始的 %app%.application 中!)。ClickOnce - .NET 4.0 错误中描述了类似的结果:“部署清单在语义上无效”和“部署清单缺少 <compatibleFrameworks>”,但它是不同过程 (msbuild) 的结果。这部分是 4.0 清单的新部分(并且是必需的),所以我唯一的猜测是,当 ManifestWriter 以某种方式将更改持久保存到磁盘时,它会以 3.5 的方式进行吗?我三次检查是否使用了正确的库(C:\Program Files (x86)\Reference Assemblies\Microsoft\Framework.NETFramework\v4.0\Microsoft.Build.Tasks.v4.0.dll)。 是什么赋予了?

到目前为止,我尝试手动添加缺少的部分来代替答案:

dm.CompatibleFrameworks.Clear(); // Unnecessary as dm.CompatibleFrameworks.Count == 0 indeed!
CompatibleFramework cf = new CompatibleFramework();
cf.Version= "4.0";
cf.SupportedRuntime = "4.0.30319";
cf.Profile= "Client";
dm.CompatibleFrameworks.Add(cf);
cf = new CompatibleFramework();
cf.Version = "4.0";
cf.SupportedRuntime = "4.0.30319";
cf.Profile = "Full";
dm.CompatibleFrameworks.Add(cf);

但这没有任何效果,无论我将这段代码放在 dm.ResolveFiles( )dm.UpdateFileInfo( )还是ManifestWriter.WriteManifest(..)之前的位置!

我的结果类似于堆栈溢出问题MageUI.exe 删除 compatibleFrameworks 元素为什么 Mage.exe 不生成 compatibleFrameworks 属性?MageUI.exe 不包含 compatibleFrameworks 元素,但我没有使用 mageuimage甚至根本没有使用msbuild

这是怎么回事?

4

2 回答 2

9

我自己想通了。罪魁祸首是ManifestReader.ReadManifest("DeployManifest", sPathMft, true )

MSDN说,[preserveStream 参数] “指定是否在结果清单对象的 InputStream 属性中保留输入流。ManifestWriter 使用它来重构未在对象表示中表示的输入。”

话说回来,单独设置true是不够的: dm.CompatibleFrameworks.Count还是0,但是现在添加CompatibleFrameworkitem会有效果!

对于同一条船上的其他人,我以前这样做过dm.ResolveFiles( )

if(  dm.CompatibleFrameworks.Count <= 0  )
{
    CompatibleFramework cf= new CompatibleFramework( );
    cf.Profile= "Client";       cf.Version= "4.0";      cf.SupportedRuntime=    "4.0.30319";
    dm.CompatibleFrameworks.Add( cf );              //  cf= new CompatibleFramework( );
    cf.Profile= "Full";     //  cf.Version= "4.0";      cf.SupportedRuntime=    "4.0.30319";
    dm.CompatibleFrameworks.Add( cf );              /// no need for separate object
}

@davidair,感谢您的建议!同意,尽管我更喜欢使用 API 对象(与 XML 相比)。
另一种选择是调用mage(直接调用或从 .cmd 文件调用),因为我们似乎可以重新分发它。


我还添加了以下部分,这对问题本身没有影响,但对于遵循相同路径的任何人来说可能非常重要(/client是部署根,并且可以自定义):

dm.DeploymentUrl=   string.Format( "http://{0}/{1}/client/{1}.application",
                        Dns.GetHostName( ), Context.Parameters[ scTokVirtDir ] );
dm.UpdateMode=      UpdateMode.Background;
dm.UpdateUnit=      UpdateUnit.Weeks;
dm.UpdateInterval=  1;
dm.UpdateEnabled=   true;

2019 年 10 月 8 日
刚刚偶然发现一个问题app.manifest:在部署期间删除了
compatibility带有元素的部分。supportedOS

相同的根本原因;读取它的行应设置preserveStreamtrue

ApplicationManifest am = ManifestReader.ReadManifest( "ApplicationManifest", sPathMft, true ) as ApplicationManifest;
于 2012-06-21T19:29:17.367 回答
1

我还需要添加 CompatibleFrameworks。我也尝试过像这样添加 CompatibleFrameworks(这不起作用)

dm.CompatibleFrameworks.Add(...);

我的解决方案是设置:

dm.TargetFrameworkMoniker = ".NETFramework,Version=v4.0";                 

在此之后,Manifest 生成是正确的。

小心 如果您在 WriteManifest 之前设置 TargetFrameworkMoniker,您有两次 <compatibleFrameworks> 并且您的应用程序文件已损坏。这是我的解决方案:

DeployManifest dm = ManifestReader.ReadManifest("DeployManifest", applicationFileName, false) as DeployManifest;
dm.ResolveFiles();
//doing stuff..
dm.UpdateFileInfo();
ManifestWriter.WriteManifest(dm, applicationFileName);
dm.TargetFrameworkMoniker = ".NETFramework,Version=v4.0";
ManifestWriter.WriteManifest(dm, applicationFileName);
于 2015-03-23T14:07:50.340 回答