aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSean Hall <r.sean.hall@gmail.com>2020-03-27 15:49:39 +1000
committerSean Hall <r.sean.hall@gmail.com>2020-03-30 21:30:04 +1000
commitbf435c69fd70f5140eddd99fe02d3dcdae75473a (patch)
tree6fb7d7aad37b9f01f78aef27c5025ca45ce4c4f4
parentc455d2290ef903ff36d540903e27d76d473cb67c (diff)
downloadwix-bf435c69fd70f5140eddd99fe02d3dcdae75473a.tar.gz
wix-bf435c69fd70f5140eddd99fe02d3dcdae75473a.tar.bz2
wix-bf435c69fd70f5140eddd99fe02d3dcdae75473a.zip
Order WixSearches in Core.
-rw-r--r--src/WixToolset.Core.Burn/Bundles/OrderSearchesCommand.cs303
-rw-r--r--src/test/Example.Extension/Data/example.wirbin535 -> 0 bytes
-rw-r--r--src/test/WixToolsetTest.CoreIntegration/TestData/SetVariable/Simple.wxs2
3 files changed, 295 insertions, 10 deletions
diff --git a/src/WixToolset.Core.Burn/Bundles/OrderSearchesCommand.cs b/src/WixToolset.Core.Burn/Bundles/OrderSearchesCommand.cs
index 3f720115..874258bf 100644
--- a/src/WixToolset.Core.Burn/Bundles/OrderSearchesCommand.cs
+++ b/src/WixToolset.Core.Burn/Bundles/OrderSearchesCommand.cs
@@ -2,7 +2,9 @@
2 2
3namespace WixToolset.Core.Burn.Bundles 3namespace WixToolset.Core.Burn.Bundles
4{ 4{
5 using System;
5 using System.Collections.Generic; 6 using System.Collections.Generic;
7 using System.Globalization;
6 using System.Linq; 8 using System.Linq;
7 using WixToolset.Data; 9 using WixToolset.Data;
8 using WixToolset.Data.Burn; 10 using WixToolset.Data.Burn;
@@ -27,6 +29,292 @@ namespace WixToolset.Core.Burn.Bundles
27 29
28 public void Execute() 30 public void Execute()
29 { 31 {
32 this.ExtensionSearchTuplesByExtensionId = new Dictionary<string, IList<IntermediateTuple>>();
33 this.OrderedSearchFacades = new List<ISearchFacade>();
34
35 var searchRelationTuples = this.Section.Tuples.OfType<WixSearchRelationTuple>().ToList();
36 var searchTuples = this.Section.Tuples.OfType<WixSearchTuple>().ToList();
37 if (searchTuples.Count == 0)
38 {
39 // nothing to do!
40 return;
41 }
42
43 var tupleDictionary = searchTuples.ToDictionary(t => t.Id.Id);
44
45 var constraints = new Constraints();
46 if (searchRelationTuples.Count > 0)
47 {
48 // add relational info to our data...
49 foreach (var searchRelationTuple in searchRelationTuples)
50 {
51 constraints.AddConstraint(searchRelationTuple.Id.Id, searchRelationTuple.ParentSearchRef);
52 }
53 }
54
55 this.FindCircularReference(constraints);
56
57 if (this.Messaging.EncounteredError)
58 {
59 return;
60 }
61
62 this.FlattenDependentReferences(constraints);
63
64 // Reorder by topographical sort (http://en.wikipedia.org/wiki/Topological_sorting)
65 // We use a variation of Kahn (1962) algorithm as described in
66 // Wikipedia, with the additional criteria that start nodes are sorted
67 // lexicographically at each step to ensure a deterministic ordering
68 // based on 'after' dependencies and ID.
69 var sorter = new TopologicalSort();
70 var sortedIds = sorter.Sort(tupleDictionary.Keys, constraints);
71
72 // Now, create the search facades with the searches in order...
73 this.OrderSearches(sortedIds, tupleDictionary);
74 }
75
76 /// <summary>
77 /// A dictionary of constraints, mapping an id to a list of ids.
78 /// </summary>
79 private class Constraints : Dictionary<string, List<string>>
80 {
81 public void AddConstraint(string id, string afterId)
82 {
83 if (!this.ContainsKey(id))
84 {
85 this.Add(id, new List<string>());
86 }
87
88 // TODO: Show warning if a constraint is seen twice?
89 if (!this[id].Contains(afterId))
90 {
91 this[id].Add(afterId);
92 }
93 }
94
95 // TODO: Hide other Add methods?
96 }
97
98 /// <summary>
99 /// Finds circular references in the constraints.
100 /// </summary>
101 /// <param name="constraints">Constraints to check.</param>
102 /// <remarks>This is not particularly performant, but it works.</remarks>
103 private void FindCircularReference(Constraints constraints)
104 {
105 foreach (string id in constraints.Keys)
106 {
107 var seenIds = new List<string>();
108 string chain = null;
109 if (this.FindCircularReference(constraints, id, id, seenIds, out chain))
110 {
111 // We will show a separate message for every ID that's in
112 // the loop. We could bail after the first one, but then
113 // we wouldn't catch disjoint loops in a single run.
114 this.Messaging.Write(ErrorMessages.CircularSearchReference(chain));
115 }
116 }
117 }
118
119 /// <summary>
120 /// Recursive function that finds circular references in the constraints.
121 /// </summary>
122 /// <param name="constraints">Constraints to check.</param>
123 /// <param name="checkId">The identifier currently being looking for. (Fixed across a given run.)</param>
124 /// <param name="currentId">The idenifier curently being tested.</param>
125 /// <param name="seenIds">A list of identifiers seen, to ensure each identifier is only expanded once.</param>
126 /// <param name="chain">If a circular reference is found, will contain the chain of references.</param>
127 /// <returns>True if a circular reference is found, false otherwise.</returns>
128 private bool FindCircularReference(Constraints constraints, string checkId, string currentId, List<string> seenIds, out string chain)
129 {
130 chain = null;
131 if (constraints.TryGetValue(currentId, out var afterList))
132 {
133 foreach (string afterId in afterList)
134 {
135 if (afterId == checkId)
136 {
137 chain = String.Format(CultureInfo.InvariantCulture, "{0} -> {1}", currentId, afterId);
138 return true;
139 }
140
141 if (!seenIds.Contains(afterId))
142 {
143 seenIds.Add(afterId);
144 if (this.FindCircularReference(constraints, checkId, afterId, seenIds, out chain))
145 {
146 chain = String.Format(CultureInfo.InvariantCulture, "{0} -> {1}", currentId, chain);
147 return true;
148 }
149 }
150 }
151 }
152
153 return false;
154 }
155
156 /// <summary>
157 /// Flattens any dependency chains to simplify reordering.
158 /// </summary>
159 /// <param name="constraints"></param>
160 private void FlattenDependentReferences(Constraints constraints)
161 {
162 foreach (string id in constraints.Keys)
163 {
164 var flattenedIds = new List<string>();
165 this.AddDependentReferences(constraints, id, flattenedIds);
166 var constraintList = constraints[id];
167 foreach (var flattenedId in flattenedIds)
168 {
169 if (!constraintList.Contains(flattenedId))
170 {
171 constraintList.Add(flattenedId);
172 }
173 }
174 }
175 }
176
177 /// <summary>
178 /// Adds dependent references to a list.
179 /// </summary>
180 /// <param name="constraints"></param>
181 /// <param name="currentId"></param>
182 /// <param name="seenIds"></param>
183 private void AddDependentReferences(Constraints constraints, string currentId, List<string> seenIds)
184 {
185 if (constraints.TryGetValue(currentId, out var afterList))
186 {
187 foreach (var afterId in afterList)
188 {
189 if (!seenIds.Contains(afterId))
190 {
191 seenIds.Add(afterId);
192 this.AddDependentReferences(constraints, afterId, seenIds);
193 }
194 }
195 }
196 }
197
198 /// <summary>
199 /// Reorder by topological sort
200 /// </summary>
201 /// <remarks>
202 /// We use a variation of Kahn (1962) algorithm as described in
203 /// Wikipedia (http://en.wikipedia.org/wiki/Topological_sorting), with
204 /// the additional criteria that start nodes are sorted lexicographically
205 /// at each step to ensure a deterministic ordering based on 'after'
206 /// dependencies and ID.
207 /// </remarks>
208 private class TopologicalSort
209 {
210 private readonly List<string> startIds = new List<string>();
211 private Constraints constraints;
212
213 /// <summary>
214 /// Reorder by topological sort
215 /// </summary>
216 /// <param name="allIds">The complete list of IDs.</param>
217 /// <param name="constraints">Constraints to use.</param>
218 /// <returns>The topologically sorted list of IDs.</returns>
219 internal List<string> Sort(IEnumerable<string> allIds, Constraints constraints)
220 {
221 this.startIds.Clear();
222 this.CopyConstraints(constraints);
223
224 this.FindInitialStartIds(allIds);
225
226 // We always create a new sortedId list, because we return it
227 // to the caller and don't know what its lifetime may be.
228 var sortedIds = new List<string>();
229
230 while (this.startIds.Count > 0)
231 {
232 this.SortStartIds();
233
234 var currentId = this.startIds[0];
235 sortedIds.Add(currentId);
236 this.startIds.RemoveAt(0);
237
238 this.ResolveConstraint(currentId);
239 }
240
241 return sortedIds;
242 }
243
244 /// <summary>
245 /// Copies a Constraints set (to prevent modifying the incoming data).
246 /// </summary>
247 /// <param name="constraints">Constraints to copy.</param>
248 private void CopyConstraints(Constraints constraints)
249 {
250 this.constraints = new Constraints();
251 foreach (var id in constraints.Keys)
252 {
253 foreach (var afterId in constraints[id])
254 {
255 this.constraints.AddConstraint(id, afterId);
256 }
257 }
258 }
259
260 /// <summary>
261 /// Finds initial start IDs. (Those with no constraints.)
262 /// </summary>
263 /// <param name="allIds">The complete list of IDs.</param>
264 private void FindInitialStartIds(IEnumerable<string> allIds)
265 {
266 foreach (var id in allIds)
267 {
268 if (!this.constraints.ContainsKey(id))
269 {
270 this.startIds.Add(id);
271 }
272 }
273 }
274
275 /// <summary>
276 /// Sorts start IDs.
277 /// </summary>
278 private void SortStartIds()
279 {
280 this.startIds.Sort();
281 }
282
283 /// <summary>
284 /// Removes the resolved constraint and updates the list of startIds
285 /// with any now-valid (all constraints resolved) IDs.
286 /// </summary>
287 /// <param name="resolvedId">The ID to resolve from the set of constraints.</param>
288 private void ResolveConstraint(string resolvedId)
289 {
290 var newStartIds = new List<string>();
291
292 foreach (var id in this.constraints.Keys)
293 {
294 if (this.constraints[id].Contains(resolvedId))
295 {
296 this.constraints[id].Remove(resolvedId);
297
298 // If we just removed the last constraint for this
299 // ID, it is now a valid start ID.
300 if (this.constraints[id].Count == 0)
301 {
302 newStartIds.Add(id);
303 }
304 }
305 }
306
307 foreach (var id in newStartIds)
308 {
309 this.constraints.Remove(id);
310 }
311
312 this.startIds.AddRange(newStartIds);
313 }
314 }
315
316 private void OrderSearches(List<string> sortedIds, Dictionary<string, WixSearchTuple> searchTupleDictionary)
317 {
30 // TODO: Although the WixSearch tables are defined in the Util extension, 318 // TODO: Although the WixSearch tables are defined in the Util extension,
31 // the Bundle Binder has to know all about them. We hope to revisit all 319 // the Bundle Binder has to know all about them. We hope to revisit all
32 // of this in the 4.0 timeframe. 320 // of this in the 4.0 timeframe.
@@ -42,22 +330,19 @@ namespace WixToolset.Core.Burn.Bundles
42 var extensionSearchesById = this.Section.Tuples 330 var extensionSearchesById = this.Section.Tuples
43 .Where(t => t.Definition.HasTag(BurnConstants.BundleExtensionSearchTupleDefinitionTag)) 331 .Where(t => t.Definition.HasTag(BurnConstants.BundleExtensionSearchTupleDefinitionTag))
44 .ToDictionary(t => t.Id.Id); 332 .ToDictionary(t => t.Id.Id);
45 var searchTuples = this.Section.Tuples.OfType<WixSearchTuple>().ToList();
46
47 this.ExtensionSearchTuplesByExtensionId = new Dictionary<string, IList<IntermediateTuple>>();
48 this.OrderedSearchFacades = new List<ISearchFacade>(legacySearchesById.Keys.Count + setVariablesById.Keys.Count + extensionSearchesById.Keys.Count);
49 333
50 foreach (var searchTuple in searchTuples) 334 foreach (var searchId in sortedIds)
51 { 335 {
52 if (legacySearchesById.TryGetValue(searchTuple.Id.Id, out var specificSearchTuple)) 336 var searchTuple = searchTupleDictionary[searchId];
337 if (legacySearchesById.TryGetValue(searchId, out var specificSearchTuple))
53 { 338 {
54 this.OrderedSearchFacades.Add(new LegacySearchFacade(searchTuple, specificSearchTuple)); 339 this.OrderedSearchFacades.Add(new LegacySearchFacade(searchTuple, specificSearchTuple));
55 } 340 }
56 else if (setVariablesById.TryGetValue(searchTuple.Id.Id, out var setVariableTuple)) 341 else if (setVariablesById.TryGetValue(searchId, out var setVariableTuple))
57 { 342 {
58 this.OrderedSearchFacades.Add(new SetVariableSearchFacade(searchTuple, setVariableTuple)); 343 this.OrderedSearchFacades.Add(new SetVariableSearchFacade(searchTuple, setVariableTuple));
59 } 344 }
60 else if (extensionSearchesById.TryGetValue(searchTuple.Id.Id, out var extensionSearchTuple)) 345 else if (extensionSearchesById.TryGetValue(searchId, out var extensionSearchTuple))
61 { 346 {
62 this.OrderedSearchFacades.Add(new ExtensionSearchFacade(searchTuple)); 347 this.OrderedSearchFacades.Add(new ExtensionSearchFacade(searchTuple));
63 348
@@ -70,7 +355,7 @@ namespace WixToolset.Core.Burn.Bundles
70 } 355 }
71 else 356 else
72 { 357 {
73 this.Messaging.Write(ErrorMessages.MissingBundleSearch(searchTuple.SourceLineNumbers, searchTuple.Id.Id)); 358 this.Messaging.Write(ErrorMessages.MissingBundleSearch(searchTuple.SourceLineNumbers, searchId));
74 } 359 }
75 } 360 }
76 } 361 }
diff --git a/src/test/Example.Extension/Data/example.wir b/src/test/Example.Extension/Data/example.wir
deleted file mode 100644
index d1ee8b90..00000000
--- a/src/test/Example.Extension/Data/example.wir
+++ /dev/null
Binary files differ
diff --git a/src/test/WixToolsetTest.CoreIntegration/TestData/SetVariable/Simple.wxs b/src/test/WixToolsetTest.CoreIntegration/TestData/SetVariable/Simple.wxs
index 96c92e54..7e8f2e99 100644
--- a/src/test/WixToolsetTest.CoreIntegration/TestData/SetVariable/Simple.wxs
+++ b/src/test/WixToolsetTest.CoreIntegration/TestData/SetVariable/Simple.wxs
@@ -9,7 +9,7 @@
9 <SetVariable Id="SetCoercedString" Variable="CoercedString" Value="Bar" /> 9 <SetVariable Id="SetCoercedString" Variable="CoercedString" Value="Bar" />
10 <SetVariable Id="SetCoercedVersion" Variable="CoercedVersion" Value="v2.0" /> 10 <SetVariable Id="SetCoercedVersion" Variable="CoercedVersion" Value="v2.0" />
11 <SetVariable Id="SetNeedsFormatting" Variable="NeedsFormatting" Value="[One] [Two] [Three]" /> 11 <SetVariable Id="SetNeedsFormatting" Variable="NeedsFormatting" Value="[One] [Two] [Three]" />
12 <SetVariable Id="SetUnset" Variable="Unset" Condition="VersionString = v2.0" After="SetVersionString" />
12 <SetVariable Id="SetVersionString" Variable="VersionString" Value="v1.0" Type="string" /> 13 <SetVariable Id="SetVersionString" Variable="VersionString" Value="v1.0" Type="string" />
13 <SetVariable Id="SetUnset" Variable="Unset" Condition="VersionString = v2.0" />
14 </Fragment> 14 </Fragment>
15</Wix> 15</Wix>