blob: 9aab7b989f89da140a879161a5e0e64f664b6ab7 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
|
// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information.
namespace WixToolset.Core.WindowsInstaller.Bind
{
using System.Collections.Generic;
using System.Linq;
using WixToolset.Core.Bind;
using WixToolset.Data;
using WixToolset.Data.Tuples;
internal class UpdateMediaSequencesCommand
{
public UpdateMediaSequencesCommand(IntermediateSection section, List<FileFacade> fileFacades)
{
this.Section = section;
this.FileFacades = fileFacades;
}
private IntermediateSection Section { get; }
private List<FileFacade> FileFacades { get; }
public void Execute()
{
var mediaRows = this.Section.Tuples.OfType<MediaTuple>().ToDictionary(t => t.DiskId);
// Calculate sequence numbers and media disk id layout for all file media information objects.
if (SectionType.Module == this.Section.Type)
{
var lastSequence = 0;
// Order by Component to group the files by directory.
var optimized = this.OptimizedFileFacades();
foreach (var facade in optimized)
{
facade.Sequence = ++lastSequence;
}
}
else
{
var lastSequence = 0;
MediaTuple mediaTuple = null;
var patchGroups = new Dictionary<int, List<FileFacade>>();
// sequence the non-patch-added files
var optimized = this.OptimizedFileFacades();
foreach (var facade in optimized)
{
if (null == mediaTuple)
{
mediaTuple = mediaRows[facade.DiskId];
if (SectionType.Patch == this.Section.Type)
{
// patch Media cannot start at zero
lastSequence = mediaTuple.LastSequence ?? 1;
}
}
else if (mediaTuple.DiskId != facade.DiskId)
{
mediaTuple.LastSequence = lastSequence;
mediaTuple = mediaRows[facade.DiskId];
}
if (facade.PatchGroup.HasValue)
{
if (patchGroups.TryGetValue(facade.PatchGroup.Value, out var patchGroup))
{
patchGroup = new List<FileFacade>();
patchGroups.Add(facade.PatchGroup.Value, patchGroup);
}
patchGroup.Add(facade);
}
else if (!facade.FromModule)
{
facade.Sequence = ++lastSequence;
}
}
if (null != mediaTuple)
{
mediaTuple.LastSequence = lastSequence;
mediaTuple = null;
}
// sequence the patch-added files
foreach (var patchGroup in patchGroups.Values)
{
foreach (var facade in patchGroup)
{
if (null == mediaTuple)
{
mediaTuple = mediaRows[facade.DiskId];
}
else if (mediaTuple.DiskId != facade.DiskId)
{
mediaTuple.LastSequence = lastSequence;
mediaTuple = mediaRows[facade.DiskId];
}
facade.Sequence = ++lastSequence;
}
}
if (null != mediaTuple)
{
mediaTuple.LastSequence = lastSequence;
}
}
}
private IEnumerable<FileFacade> OptimizedFileFacades()
{
// TODO: Sort these facades even smarter by directory path and component id
// and maybe file size or file extension and other creative ideas to
// get optimal install speed out of MSI.
return this.FileFacades.OrderBy(f => f.ComponentRef);
}
}
}
|