Allow DownloadUrl on embedded payloads.

#5253
This commit is contained in:
Sean Hall 2021-04-25 15:36:31 -05:00
Родитель 8b7545c2b6
Коммит b4210b23ea
8 изменённых файлов: 110 добавлений и 58 удалений

Просмотреть файл

@ -300,7 +300,7 @@ namespace WixToolset.Core.Burn
if (PackagingType.Embedded == payload.Packaging && String.IsNullOrEmpty(payload.EmbeddedId)) if (PackagingType.Embedded == payload.Packaging && String.IsNullOrEmpty(payload.EmbeddedId))
{ {
payload.EmbeddedId = String.Format(CultureInfo.InvariantCulture, BurnCommon.BurnAttachedContainerEmbeddedIdFormat, payloadIndex); payload.EmbeddedId = String.Format(CultureInfo.InvariantCulture, BurnCommon.BurnAuthoredContainerEmbeddedIdFormat, payloadIndex);
++payloadIndex; ++payloadIndex;
} }
} }

Просмотреть файл

@ -5,6 +5,7 @@ namespace WixToolset.Core.Burn.Bind
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using WixToolset.Data; using WixToolset.Data;
using WixToolset.Data.Burn;
using WixToolset.Data.Symbols; using WixToolset.Data.Symbols;
using WixToolset.Extensibility; using WixToolset.Extensibility;
using WixToolset.Extensibility.Services; using WixToolset.Extensibility.Services;
@ -60,7 +61,14 @@ namespace WixToolset.Core.Burn.Bind
{ {
foreach (var payload in this.PayloadsById.Values) foreach (var payload in this.PayloadsById.Values)
{ {
if (payload.Packaging == PackagingType.External) if (payload.Packaging == PackagingType.Embedded && payload.ContainerRef == BurnConstants.BurnUXContainerName)
{
if (!String.IsNullOrEmpty(payload.DownloadUrl))
{
this.Messaging.Write(WarningMessages.DownloadUrlNotSupportedForBAPayloads(payload.SourceLineNumbers, payload.Id.Id));
}
}
else
{ {
var packageId = payload.ParentPackagePayloadRef; var packageId = payload.ParentPackagePayloadRef;
var parentUrl = payload.ParentPackagePayloadRef == null ? null : this.PayloadsById[payload.ParentPackagePayloadRef].DownloadUrl; var parentUrl = payload.ParentPackagePayloadRef == null ? null : this.PayloadsById[payload.ParentPackagePayloadRef].DownloadUrl;
@ -70,13 +78,6 @@ namespace WixToolset.Core.Burn.Bind
payload.DownloadUrl = resolvedUrl; payload.DownloadUrl = resolvedUrl;
} }
} }
else if (payload.Packaging == PackagingType.Embedded)
{
if (!String.IsNullOrEmpty(payload.DownloadUrl))
{
this.Messaging.Write(WarningMessages.DownloadUrlNotSupportedForEmbeddedPayloads(payload.SourceLineNumbers, payload.Id.Id));
}
}
} }
} }

Просмотреть файл

@ -19,8 +19,7 @@ namespace WixToolset.Core.Burn.Bundles
{ {
public const string BurnNamespace = "http://wixtoolset.org/schemas/v4/2008/Burn"; public const string BurnNamespace = "http://wixtoolset.org/schemas/v4/2008/Burn";
public const string BurnUXContainerEmbeddedIdFormat = "u{0}"; public const string BurnUXContainerEmbeddedIdFormat = "u{0}";
public const string BurnUXContainerPayloadIdFormat = "p{0}"; public const string BurnAuthoredContainerEmbeddedIdFormat = "a{0}";
public const string BurnAttachedContainerEmbeddedIdFormat = "a{0}";
public const string BADataFileName = "BootstrapperApplicationData.xml"; public const string BADataFileName = "BootstrapperApplicationData.xml";
public const string BADataNamespace = "http://wixtoolset.org/schemas/v4/BootstrapperApplicationData"; public const string BADataNamespace = "http://wixtoolset.org/schemas/v4/BootstrapperApplicationData";

Просмотреть файл

@ -166,9 +166,7 @@ namespace WixToolset.Core.Burn.Bundles
// write the UX allPayloads... // write the UX allPayloads...
foreach (var payload in this.UXContainerPayloads) foreach (var payload in this.UXContainerPayloads)
{ {
writer.WriteStartElement("Payload"); this.WriteBurnManifestUXPayload(writer, payload);
this.WriteBurnManifestPayloadAttributes(writer, payload, true, this.Payloads);
writer.WriteEndElement();
} }
writer.WriteEndElement(); // </UX> writer.WriteEndElement(); // </UX>
@ -183,20 +181,9 @@ namespace WixToolset.Core.Burn.Bundles
} }
} }
foreach (var payload in this.Payloads.Values) foreach (var payload in this.Payloads.Values.Where(p => p.ContainerRef != BurnConstants.BurnUXContainerName))
{ {
if (PackagingType.Embedded == payload.Packaging && BurnConstants.BurnUXContainerName != payload.ContainerRef) this.WriteBurnManifestPayload(writer, payload);
{
writer.WriteStartElement("Payload");
this.WriteBurnManifestPayloadAttributes(writer, payload, true, this.Payloads);
writer.WriteEndElement();
}
else if (PackagingType.External == payload.Packaging)
{
writer.WriteStartElement("Payload");
this.WriteBurnManifestPayloadAttributes(writer, payload, false, this.Payloads);
writer.WriteEndElement();
}
} }
foreach (var rollbackBoundary in this.RollbackBoundaries) foreach (var rollbackBoundary in this.RollbackBoundaries)
@ -654,9 +641,9 @@ namespace WixToolset.Core.Burn.Bundles
} }
} }
private void WriteBurnManifestPayloadAttributes(XmlTextWriter writer, WixBundlePayloadSymbol payload, bool embeddedOnly, Dictionary<string, WixBundlePayloadSymbol> allPayloads) private void WriteBurnManifestPayload(XmlTextWriter writer, WixBundlePayloadSymbol payload)
{ {
Debug.Assert(!embeddedOnly || PackagingType.Embedded == payload.Packaging); writer.WriteStartElement("Payload");
writer.WriteAttributeString("Id", payload.Id.Id); writer.WriteAttributeString("Id", payload.Id.Id);
writer.WriteAttributeString("FilePath", payload.Name); writer.WriteAttributeString("FilePath", payload.Name);
@ -668,28 +655,46 @@ namespace WixToolset.Core.Burn.Bundles
writer.WriteAttributeString("LayoutOnly", "yes"); writer.WriteAttributeString("LayoutOnly", "yes");
} }
if (!String.IsNullOrEmpty(payload.DownloadUrl))
{
writer.WriteAttributeString("DownloadUrl", payload.DownloadUrl);
}
switch (payload.Packaging) switch (payload.Packaging)
{ {
case PackagingType.Embedded: // this means it's in a container. case PackagingType.Embedded: // this means it's in a container.
Debug.Assert(BurnConstants.BurnUXContainerName != payload.ContainerRef);
writer.WriteAttributeString("Packaging", "embedded"); writer.WriteAttributeString("Packaging", "embedded");
writer.WriteAttributeString("SourcePath", payload.EmbeddedId); writer.WriteAttributeString("SourcePath", payload.EmbeddedId);
writer.WriteAttributeString("Container", payload.ContainerRef);
if (BurnConstants.BurnUXContainerName != payload.ContainerRef)
{
writer.WriteAttributeString("Container", payload.ContainerRef);
}
break; break;
case PackagingType.External: case PackagingType.External:
if (!String.IsNullOrEmpty(payload.DownloadUrl))
{
writer.WriteAttributeString("DownloadUrl", payload.DownloadUrl);
}
writer.WriteAttributeString("Packaging", "external"); writer.WriteAttributeString("Packaging", "external");
writer.WriteAttributeString("SourcePath", payload.Name); writer.WriteAttributeString("SourcePath", payload.Name);
break; break;
} }
writer.WriteEndElement();
}
private void WriteBurnManifestUXPayload(XmlTextWriter writer, WixBundlePayloadSymbol payload)
{
Debug.Assert(PackagingType.Embedded == payload.Packaging);
Debug.Assert(BurnConstants.BurnUXContainerName == payload.ContainerRef);
writer.WriteStartElement("Payload");
// TODO: The engine should be updated to not require FileSize, Hash, or Packaging for UX payloads since the values are never used.
writer.WriteAttributeString("Id", payload.Id.Id);
writer.WriteAttributeString("FilePath", payload.Name);
writer.WriteAttributeString("FileSize", payload.FileSize.Value.ToString(CultureInfo.InvariantCulture));
writer.WriteAttributeString("Hash", payload.Hash);
writer.WriteAttributeString("Packaging", "embedded");
writer.WriteAttributeString("SourcePath", payload.EmbeddedId);
writer.WriteEndElement();
} }
} }
} }

Просмотреть файл

@ -44,13 +44,6 @@ namespace WixToolset.Core.Burn.Bundles
public void Execute() public void Execute()
{ {
var payloadCount = this.Payloads.Count(); // The number of embedded payloads
if (!String.IsNullOrEmpty(this.ManifestFile))
{
++payloadCount;
}
var cabinetPath = Path.GetFullPath(this.OutputPath); var cabinetPath = Path.GetFullPath(this.OutputPath);
var files = new List<CabinetCompressFile>(); var files = new List<CabinetCompressFile>();

Просмотреть файл

@ -15,8 +15,6 @@ namespace WixToolset.Core
public string Description { get; set; } public string Description { get; set; }
public string DisplayName { get; set; }
public string DownloadUrl { get; set; } public string DownloadUrl { get; set; }
public string Hash { get; set; } public string Hash { get; set; }
@ -158,7 +156,7 @@ namespace WixToolset.Core
if (!String.IsNullOrEmpty(this.DownloadUrl)) if (!String.IsNullOrEmpty(this.DownloadUrl))
{ {
this.Core.Write(WarningMessages.DownloadUrlNotSupportedForEmbeddedPayloads(this.SourceLineNumbers, this.Id.Id)); this.Core.Write(WarningMessages.DownloadUrlNotSupportedForBAPayloads(this.SourceLineNumbers, this.Id.Id));
} }
this.Compressed = YesNoDefaultType.Yes; this.Compressed = YesNoDefaultType.Yes;
@ -174,7 +172,7 @@ namespace WixToolset.Core
DownloadUrl = this.DownloadUrl, DownloadUrl = this.DownloadUrl,
Compressed = (this.Compressed == YesNoDefaultType.Yes) ? true : (this.Compressed == YesNoDefaultType.No) ? (bool?)false : null, Compressed = (this.Compressed == YesNoDefaultType.Yes) ? true : (this.Compressed == YesNoDefaultType.No) ? (bool?)false : null,
UnresolvedSourceFile = this.SourceFile, // duplicate of sourceFile but in a string column so it won't get resolved to a full path during binding. UnresolvedSourceFile = this.SourceFile, // duplicate of sourceFile but in a string column so it won't get resolved to a full path during binding.
DisplayName = this.DisplayName ?? this.ProductName, DisplayName = this.ProductName,
Description = this.Description, Description = this.Description,
Hash = this.Hash, Hash = this.Hash,
FileSize = this.Size, FileSize = this.Size,
@ -245,11 +243,6 @@ namespace WixToolset.Core
this.Description = this.Core.GetAttributeValue(this.SourceLineNumbers, attrib); this.Description = this.Core.GetAttributeValue(this.SourceLineNumbers, attrib);
} }
public void ParseDisplayName(XAttribute attrib)
{
this.DisplayName = this.Core.GetAttributeValue(this.SourceLineNumbers, attrib);
}
public void ParseDownloadUrl(XAttribute attrib) public void ParseDownloadUrl(XAttribute attrib)
{ {
this.DownloadUrl = this.Core.GetAttributeValue(this.SourceLineNumbers, attrib); this.DownloadUrl = this.Core.GetAttributeValue(this.SourceLineNumbers, attrib);

Просмотреть файл

@ -15,6 +15,50 @@ namespace WixToolsetTest.CoreIntegration
public class ContainerFixture public class ContainerFixture
{ {
[Fact(Skip = "Test demonstrates failure")]
public void CanBuildWithCustomAttachedContainer()
{
var folder = TestData.Get(@"TestData");
using (var fs = new DisposableFileSystem())
{
var baseFolder = fs.GetFolder();
var intermediateFolder = Path.Combine(baseFolder, "obj");
var binFolder = Path.Combine(baseFolder, "bin");
var bundlePath = Path.Combine(binFolder, "test.exe");
var baFolderPath = Path.Combine(baseFolder, "ba");
var extractFolderPath = Path.Combine(baseFolder, "extract");
this.BuildMsis(folder, intermediateFolder, binFolder, buildToSubfolder: true);
var result = WixRunner.Execute(new[]
{
"build",
Path.Combine(folder, "Container", "HarvestIntoAttachedContainer.wxs"),
Path.Combine(folder, "BundleWithPackageGroupRef", "Bundle.wxs"),
"-bindpath", Path.Combine(folder, "SimpleBundle", "data"),
"-bindpath", binFolder,
"-intermediateFolder", intermediateFolder,
"-o", bundlePath
});
result.AssertSuccess();
Assert.True(File.Exists(bundlePath));
var extractResult = BundleExtractor.ExtractBAContainer(null, bundlePath, baFolderPath, extractFolderPath);
extractResult.AssertSuccess();
var payloads = extractResult.SelectManifestNodes("/burn:BurnManifest/burn:Payload");
Assert.Equal(4, payloads.Count);
var ignoreAttributes = new Dictionary<string, List<string>> { { "Payload", new List<string> { "FileSize", "Hash" } } };
Assert.Equal(@"<Payload Id='FirstX64' FilePath='FirstX64\FirstX64.msi' FileSize='*' Hash='*' DownloadUrl='http://example.com//FirstX64/FirstX64/FirstX64.msi' Packaging='embedded' SourcePath='a0' Container='BundlePackages' />", payloads[0].GetTestXml(ignoreAttributes));
Assert.Equal(@"<Payload Id='FirstX86.msi' FilePath='FirstX86\FirstX86.msi' FileSize='*' Hash='*' DownloadUrl='http://example.com//FirstX86.msi/FirstX86/FirstX86.msi' Packaging='embedded' SourcePath='a1' Container='BundlePackages' />", payloads[1].GetTestXml(ignoreAttributes));
Assert.Equal(@"<Payload Id='fk1m38Cf9RZ2Bx_ipinRY6BftelU' FilePath='FirstX86\PFiles\MsiPackage\test.txt' FileSize='*' Hash='*' DownloadUrl='http://example.com/FirstX86/fk1m38Cf9RZ2Bx_ipinRY6BftelU/FirstX86/PFiles/MsiPackage/test.txt' Packaging='embedded' SourcePath='a2' Container='BundlePackages' />", payloads[2].GetTestXml(ignoreAttributes));
Assert.Equal(@"<Payload Id='ff2L_N_DLQ.nSUi.l8LxG14gd2V4' FilePath='FirstX64\PFiles\MsiPackage\test.txt' FileSize='*' Hash='*' DownloadUrl='http://example.com/FirstX64/ff2L_N_DLQ.nSUi.l8LxG14gd2V4/FirstX64/PFiles/MsiPackage/test.txt' Packaging='embedded' SourcePath='a3' Container='BundlePackages' />", payloads[3].GetTestXml(ignoreAttributes));
}
}
[Fact] [Fact]
public void HarvestedPayloadsArePutInCorrectContainer() public void HarvestedPayloadsArePutInCorrectContainer()
{ {
@ -309,7 +353,7 @@ namespace WixToolsetTest.CoreIntegration
} }
} }
private void BuildMsis(string folder, string intermediateFolder, string binFolder) private void BuildMsis(string folder, string intermediateFolder, string binFolder, bool buildToSubfolder = false)
{ {
var result = WixRunner.Execute(new[] var result = WixRunner.Execute(new[]
{ {
@ -319,7 +363,7 @@ namespace WixToolsetTest.CoreIntegration
Path.Combine(folder, "ProductWithComponentGroupRef", "Product.wxs"), Path.Combine(folder, "ProductWithComponentGroupRef", "Product.wxs"),
"-bindpath", Path.Combine(folder, "SingleFile", "data"), "-bindpath", Path.Combine(folder, "SingleFile", "data"),
"-intermediateFolder", intermediateFolder, "-intermediateFolder", intermediateFolder,
"-o", Path.Combine(binFolder, "FirstX86.msi"), "-o", Path.Combine(binFolder, buildToSubfolder ? "FirstX86" : ".", "FirstX86.msi"),
}); });
result.AssertSuccess(); result.AssertSuccess();
@ -332,7 +376,7 @@ namespace WixToolsetTest.CoreIntegration
Path.Combine(folder, "ProductWithComponentGroupRef", "Product.wxs"), Path.Combine(folder, "ProductWithComponentGroupRef", "Product.wxs"),
"-bindpath", Path.Combine(folder, "SingleFile", "data"), "-bindpath", Path.Combine(folder, "SingleFile", "data"),
"-intermediateFolder", intermediateFolder, "-intermediateFolder", intermediateFolder,
"-o", Path.Combine(binFolder, "FirstX64.msi"), "-o", Path.Combine(binFolder, buildToSubfolder ? "FirstX64" : ".", "FirstX64.msi"),
}); });
result.AssertSuccess(); result.AssertSuccess();

Просмотреть файл

@ -0,0 +1,17 @@
<?xml version="1.0" encoding="utf-8"?>
<Wix xmlns="http://wixtoolset.org/schemas/v4/wxs">
<Fragment>
<PackageGroup Id="BundlePackages">
<MsiPackage Id="FirstX86">
<PayloadGroupRef Id="FirstX86Payloads" />
</MsiPackage>
<MsiPackage Id="FirstX64" Name="FirstX64\FirstX64.msi" SourceFile="FirstX64\" DownloadUrl="http://example.com/{0}/{1}/{2}" />
</PackageGroup>
<Container Id="BundlePackages" Type="attached">
<PackageGroupRef Id="BundlePackages" />
</Container>
<PayloadGroup Id="FirstX86Payloads">
<MsiPackagePayload Name="FirstX86\FirstX86.msi" SourceFile="FirstX86\" DownloadUrl="http://example.com/{0}/{1}/{2}" />
</PayloadGroup>
</Fragment>
</Wix>