// 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 { using System; using System.Collections; using System.Collections.Generic; using System.Diagnostics; using System.Globalization; using System.Linq; using WixToolset.Data; using WixToolset.Link; using WixToolset.Core.Link; using WixToolset.Data.Tuples; using WixToolset.Extensibility.Services; /// /// Linker core of the WiX toolset. /// public sealed class Linker { private static readonly char[] colonCharacter = ":".ToCharArray(); private static readonly string emptyGuid = Guid.Empty.ToString("B"); private bool sectionIdOnRows; /// /// Creates a linker. /// public Linker() { this.sectionIdOnRows = true; // TODO: what is the correct value for this? //this.extensionData = new List(); //this.inspectorExtensions = new List(); } private ILinkContext Context { get; set; } /// /// Gets or sets the path to output unreferenced symbols to. If null or empty, there is no output. /// /// The path to output the xml file. public string UnreferencedSymbolsFile { get; set; } /// /// Gets or sets the option to show pedantic messages. /// /// The option to show pedantic messages. public bool ShowPedanticMessages { get; set; } /// /// Gets or sets the Wix variable resolver. /// /// The Wix variable resolver. //internal IBindVariableResolver WixVariableResolver { get; set; } /// /// Links a collection of sections into an output. /// /// The collection of sections to link together. /// Expected output type, based on output file extension provided to the linker. /// Output object from the linking. public Intermediate Link(ILinkContext context) { this.Context = context ?? throw new ArgumentNullException(nameof(context)); var sections = this.Context.Intermediates.SelectMany(i => i.Sections).ToList(); // Add sections from the extensions with data. foreach (var data in context.ExtensionData) { var library = data.GetLibrary(context.TupleDefinitionCreator); if (library != null) { sections.AddRange(library.Sections); } } #if MOVE_TO_BACKEND bool containsModuleSubstitution = false; bool containsModuleConfiguration = false; #endif //this.activeOutput = null; //TableDefinitionCollection customTableDefinitions = new TableDefinitionCollection(); //IntermediateTuple customRows = new List(); #if MOVE_TO_BACKEND StringCollection generatedShortFileNameIdentifiers = new StringCollection(); Hashtable generatedShortFileNames = new Hashtable(); #endif Hashtable multipleFeatureComponents = new Hashtable(); var wixVariables = new Dictionary(); #if MOVE_TO_BACKEND // verify that modularization types match for foreign key relationships foreach (TableDefinition tableDefinition in this.tableDefinitions) { foreach (ColumnDefinition columnDefinition in tableDefinition.Columns) { if (null != columnDefinition.KeyTable && 0 > columnDefinition.KeyTable.IndexOf(';') && columnDefinition.IsKeyColumnSet) { try { TableDefinition keyTableDefinition = this.tableDefinitions[columnDefinition.KeyTable]; if (0 >= columnDefinition.KeyColumn || keyTableDefinition.Columns.Count < columnDefinition.KeyColumn) { this.OnMessage(WixErrors.InvalidKeyColumn(tableDefinition.Name, columnDefinition.Name, columnDefinition.KeyTable, columnDefinition.KeyColumn)); } else if (keyTableDefinition.Columns[columnDefinition.KeyColumn - 1].ModularizeType != columnDefinition.ModularizeType && ColumnModularizeType.CompanionFile != columnDefinition.ModularizeType) { this.OnMessage(WixErrors.CollidingModularizationTypes(tableDefinition.Name, columnDefinition.Name, columnDefinition.KeyTable, columnDefinition.KeyColumn, columnDefinition.ModularizeType.ToString(), keyTableDefinition.Columns[columnDefinition.KeyColumn - 1].ModularizeType.ToString())); } } catch (WixMissingTableDefinitionException) { // ignore missing table definitions - this error is caught in other places } } } } #endif // First find the entry section and while processing all sections load all the symbols from all of the sections. // sections.FindEntrySectionAndLoadSymbols(false, this, expectedOutputType, out entrySection, out allSymbols); var find = new FindEntrySectionAndLoadSymbolsCommand(this.Context.Messaging, sections); find.ExpectedOutputType = this.Context.ExpectedOutputType; find.Execute(); // Must have found the entry section by now. if (null == find.EntrySection) { throw new WixException(ErrorMessages.MissingEntrySection(this.Context.ExpectedOutputType.ToString())); } // Add the missing standard action symbols. this.LoadStandardActionSymbols(find.EntrySection, find.Symbols); // Resolve the symbol references to find the set of sections we care about for linking. // Of course, we start with the entry section (that's how it got its name after all). var resolve = new ResolveReferencesCommand(this.Context.Messaging, find.EntrySection, find.Symbols); resolve.BuildingMergeModule = (SectionType.Module == find.EntrySection.Type); resolve.Execute(); if (this.Context.Messaging.EncounteredError) { return null; } // Reset the sections to only those that were resolved then flatten the complex // references that particpate in groups. sections = resolve.ResolvedSections.ToList(); this.FlattenSectionsComplexReferences(sections); if (this.Context.Messaging.EncounteredError) { return null; } // The hard part in linking is processing the complex references. var referencedComponents = new HashSet(); var componentsToFeatures = new ConnectToFeatureCollection(); var featuresToFeatures = new ConnectToFeatureCollection(); var modulesToFeatures = new ConnectToFeatureCollection(); this.ProcessComplexReferences(find.EntrySection, sections, referencedComponents, componentsToFeatures, featuresToFeatures, modulesToFeatures); if (this.Context.Messaging.EncounteredError) { return null; } // Display an error message for Components that were not referenced by a Feature. foreach (var symbol in resolve.ReferencedSymbols.Where(s => s.Row.Definition.Type == TupleDefinitionType.Component)) { if (!referencedComponents.Contains(symbol.Name)) { this.OnMessage(ErrorMessages.OrphanedComponent(symbol.Row.SourceLineNumbers, symbol.Row.Id.Id)); } } // Report duplicates that would ultimately end up being primary key collisions. var reportDupes = new ReportConflictingSymbolsCommand(this.Context.Messaging, find.PossiblyConflictingSymbols, resolve.ResolvedSections); reportDupes.Execute(); if (this.Context.Messaging.EncounteredError) { return null; } // resolve the feature to feature connects this.ResolveFeatureToFeatureConnects(featuresToFeatures, find.Symbols); // start generating OutputTables and OutputRows for all the sections in the output var ensureTableRows = new List(); // Create the section to hold the linked content. var resolvedSection = new IntermediateSection(find.EntrySection.Id, find.EntrySection.Type, find.EntrySection.Codepage); var sectionCount = 0; foreach (var section in sections) { sectionCount++; var sectionId = section.Id; if (null == sectionId && this.sectionIdOnRows) { sectionId = "wix.section." + sectionCount.ToString(CultureInfo.InvariantCulture); } foreach (var tuple in section.Tuples) { var copyTuple = true; // by default, copy tuples. // handle special tables switch (tuple.Definition.Type) { #if MOVE_TO_BACKEND case "AppSearch": this.activeOutput.EnsureTable(this.tableDefinitions["Signature"]); break; #endif case TupleDefinitionType.Class: if (SectionType.Product == resolvedSection.Type) { this.ResolveFeatures(tuple, 2, 11, componentsToFeatures, multipleFeatureComponents); } break; #if MOVE_TO_BACKEND case "CustomAction": if (OutputType.Module == this.activeOutput.Type) { this.activeOutput.EnsureTable(this.tableDefinitions["AdminExecuteSequence"]); this.activeOutput.EnsureTable(this.tableDefinitions["AdminUISequence"]); this.activeOutput.EnsureTable(this.tableDefinitions["AdvtExecuteSequence"]); this.activeOutput.EnsureTable(this.tableDefinitions["InstallExecuteSequence"]); this.activeOutput.EnsureTable(this.tableDefinitions["InstallUISequence"]); } break; case "Dialog": this.activeOutput.EnsureTable(this.tableDefinitions["ListBox"]); break; case "Directory": foreach (Row row in table.Rows) { if (OutputType.Module == this.activeOutput.Type) { string directory = row[0].ToString(); if (WindowsInstallerStandard.IsStandardDirectory(directory)) { // if the directory table contains references to standard windows folders // mergemod.dll will add customactions to set the MSM directory to // the same directory as the standard windows folder and will add references to // custom action to all the standard sequence tables. A problem will occur // if the MSI does not have these tables as mergemod.dll does not add these // tables to the MSI if absent. This code adds the tables in case mergemod.dll // needs them. this.activeOutput.EnsureTable(this.tableDefinitions["CustomAction"]); this.activeOutput.EnsureTable(this.tableDefinitions["AdminExecuteSequence"]); this.activeOutput.EnsureTable(this.tableDefinitions["AdminUISequence"]); this.activeOutput.EnsureTable(this.tableDefinitions["AdvtExecuteSequence"]); this.activeOutput.EnsureTable(this.tableDefinitions["InstallExecuteSequence"]); this.activeOutput.EnsureTable(this.tableDefinitions["InstallUISequence"]); } else { foreach (string standardDirectory in WindowsInstallerStandard.GetStandardDirectories()) { if (directory.StartsWith(standardDirectory, StringComparison.Ordinal)) { this.OnMessage(WixWarnings.StandardDirectoryConflictInMergeModule(row.SourceLineNumbers, directory, standardDirectory)); } } } } } break; #endif case TupleDefinitionType.Extension: if (SectionType.Product == resolvedSection.Type) { this.ResolveFeatures(tuple, 1, 4, componentsToFeatures, multipleFeatureComponents); } break; #if MOVE_TO_BACKEND case TupleDefinitionType.ModuleSubstitution: containsModuleSubstitution = true; break; case TupleDefinitionType.ModuleConfiguration: containsModuleConfiguration = true; break; #endif case TupleDefinitionType.MsiAssembly: if (SectionType.Product == resolvedSection.Type) { this.ResolveFeatures(tuple, 0, 1, componentsToFeatures, multipleFeatureComponents); } break; #if MOVE_TO_BACKEND case "ProgId": // the Extension table is required with a ProgId table this.activeOutput.EnsureTable(this.tableDefinitions["Extension"]); break; case "Property": // Remove property rows with no value. These are properties associated with // AppSearch but without a default value. for (int i = 0; i < table.Rows.Count; i++) { if (null == table.Rows[i][1]) { table.Rows.RemoveAt(i); i--; } } break; #endif case TupleDefinitionType.PublishComponent: if (SectionType.Product == resolvedSection.Type) { this.ResolveFeatures(tuple, 2, 4, componentsToFeatures, multipleFeatureComponents); } break; case TupleDefinitionType.Shortcut: if (SectionType.Product == resolvedSection.Type) { this.ResolveFeatures(tuple, 3, 4, componentsToFeatures, multipleFeatureComponents); } break; case TupleDefinitionType.TypeLib: if (SectionType.Product == resolvedSection.Type) { this.ResolveFeatures(tuple, 2, 6, componentsToFeatures, multipleFeatureComponents); } break; #if SOLVE_CUSTOM_TABLE case "WixCustomTable": this.LinkCustomTable(table, customTableDefinitions); copyTuple = false; // we've created table definitions from these rows, no need to process them any longer break; case "WixCustomRow": foreach (Row row in table.Rows) { row.SectionId = (this.sectionIdOnRows ? sectionId : null); customRows.Add(row); } copyTuple = false; break; #endif case TupleDefinitionType.WixEnsureTable: ensureTableRows.Add(tuple); break; #if MOVE_TO_BACKEND case "WixFile": foreach (Row row in table.Rows) { // DiskId is not valid when creating a module, so set it to // 0 for all files to ensure proper sorting in the binder if (SectionType.Module == entrySection.Type) { row[5] = 0; } // if the short file name was generated, check for collisions if (0x1 == (int)row[9]) { generatedShortFileNameIdentifiers.Add((string)row[0]); } } break; #endif case TupleDefinitionType.WixMerge: if (SectionType.Product == resolvedSection.Type) { this.ResolveFeatures(tuple, 0, 7, modulesToFeatures, null); } break; case TupleDefinitionType.WixComplexReference: copyTuple = false; break; case TupleDefinitionType.WixSimpleReference: copyTuple = false; break; case TupleDefinitionType.WixVariable: // check for colliding values and collect the wix variable rows { var row = (WixVariableTuple)tuple; if (wixVariables.TryGetValue(row.WixVariable, out var collidingRow)) { if (collidingRow.Overridable && !row.Overridable) { wixVariables[row.WixVariable] = row; } else if (!row.Overridable || (collidingRow.Overridable && row.Overridable)) { this.OnMessage(ErrorMessages.WixVariableCollision(row.SourceLineNumbers, row.WixVariable)); } } else { wixVariables.Add(row.WixVariable, row); } } copyTuple = false; break; } if (copyTuple) { resolvedSection.Tuples.Add(tuple); } } } // copy the module to feature connections into the output foreach (ConnectToFeature connectToFeature in modulesToFeatures) { foreach (var feature in connectToFeature.ConnectFeatures) { var row = new WixFeatureModulesTuple(); row.Feature_ = feature; row.WixMerge_ = connectToFeature.ChildId; resolvedSection.Tuples.Add(row); } } #if MOVE_TO_BACKEND // ensure the creation of tables that need to exist if (0 < ensureTableRows.Count) { foreach (Row row in ensureTableRows) { string tableId = (string)row[0]; TableDefinition tableDef = null; try { tableDef = this.tableDefinitions[tableId]; } catch (WixMissingTableDefinitionException) { tableDef = customTableDefinitions[tableId]; } this.activeOutput.EnsureTable(tableDef); } } #endif #if MOVE_TO_BACKEND // check for missing table and add them or display an error as appropriate switch (this.activeOutput.Type) { case OutputType.Module: this.activeOutput.EnsureTable(this.tableDefinitions["Component"]); this.activeOutput.EnsureTable(this.tableDefinitions["Directory"]); this.activeOutput.EnsureTable(this.tableDefinitions["FeatureComponents"]); this.activeOutput.EnsureTable(this.tableDefinitions["File"]); this.activeOutput.EnsureTable(this.tableDefinitions["ModuleComponents"]); this.activeOutput.EnsureTable(this.tableDefinitions["ModuleSignature"]); break; case OutputType.PatchCreation: Table imageFamiliesTable = this.activeOutput.Tables["ImageFamilies"]; Table targetImagesTable = this.activeOutput.Tables["TargetImages"]; Table upgradedImagesTable = this.activeOutput.Tables["UpgradedImages"]; if (null == imageFamiliesTable || 1 > imageFamiliesTable.Rows.Count) { this.OnMessage(WixErrors.ExpectedRowInPatchCreationPackage("ImageFamilies")); } if (null == targetImagesTable || 1 > targetImagesTable.Rows.Count) { this.OnMessage(WixErrors.ExpectedRowInPatchCreationPackage("TargetImages")); } if (null == upgradedImagesTable || 1 > upgradedImagesTable.Rows.Count) { this.OnMessage(WixErrors.ExpectedRowInPatchCreationPackage("UpgradedImages")); } this.activeOutput.EnsureTable(this.tableDefinitions["Properties"]); break; case OutputType.Product: this.activeOutput.EnsureTable(this.tableDefinitions["File"]); this.activeOutput.EnsureTable(this.tableDefinitions["Media"]); break; } this.CheckForIllegalTables(this.activeOutput); #endif #if SOLVE_CUSTOM_TABLE // add the custom row data foreach (Row row in customRows) { TableDefinition customTableDefinition = (TableDefinition)customTableDefinitions[row[0].ToString()]; Table customTable = this.activeOutput.EnsureTable(customTableDefinition); Row customRow = customTable.CreateRow(row.SourceLineNumbers); customRow.SectionId = row.SectionId; string[] data = row[1].ToString().Split(Common.CustomRowFieldSeparator); for (int i = 0; i < data.Length; ++i) { bool foundColumn = false; string[] item = data[i].Split(colonCharacter, 2); for (int j = 0; j < customRow.Fields.Length; ++j) { if (customRow.Fields[j].Column.Name == item[0]) { if (0 < item[1].Length) { if (ColumnType.Number == customRow.Fields[j].Column.Type) { try { customRow.Fields[j].Data = Convert.ToInt32(item[1], CultureInfo.InvariantCulture); } catch (FormatException) { this.OnMessage(WixErrors.IllegalIntegerValue(row.SourceLineNumbers, customTableDefinition.Columns[i].Name, customTableDefinition.Name, item[1])); } catch (OverflowException) { this.OnMessage(WixErrors.IllegalIntegerValue(row.SourceLineNumbers, customTableDefinition.Columns[i].Name, customTableDefinition.Name, item[1])); } } else if (ColumnCategory.Identifier == customRow.Fields[j].Column.Category) { if (Common.IsIdentifier(item[1]) || Common.IsValidBinderVariable(item[1]) || ColumnCategory.Formatted == customRow.Fields[j].Column.Category) { customRow.Fields[j].Data = item[1]; } else { this.OnMessage(WixErrors.IllegalIdentifier(row.SourceLineNumbers, "Data", item[1])); } } else { customRow.Fields[j].Data = item[1]; } } foundColumn = true; break; } } if (!foundColumn) { this.OnMessage(WixErrors.UnexpectedCustomTableColumn(row.SourceLineNumbers, item[0])); } } for (int i = 0; i < customTableDefinition.Columns.Count; ++i) { if (!customTableDefinition.Columns[i].Nullable && (null == customRow.Fields[i].Data || 0 == customRow.Fields[i].Data.ToString().Length)) { this.OnMessage(WixErrors.NoDataForColumn(row.SourceLineNumbers, customTableDefinition.Columns[i].Name, customTableDefinition.Name)); } } } #endif //correct the section Id in FeatureComponents table if (this.sectionIdOnRows) { //var componentSectionIds = new Dictionary(); //foreach (var componentTuple in entrySection.Tuples.OfType()) //{ // componentSectionIds.Add(componentTuple.Id.Id, componentTuple.SectionId); //} //foreach (var featureComponentTuple in entrySection.Tuples.OfType()) //{ // if (componentSectionIds.TryGetValue(featureComponentTuple.Component_, out var componentSectionId)) // { // featureComponentTuple.SectionId = componentSectionId; // } //} } #if MOVE_TO_BACKEND // add the ModuleSubstitution table to the ModuleIgnoreTable if (containsModuleSubstitution) { Table moduleIgnoreTableTable = this.activeOutput.EnsureTable(this.tableDefinitions["ModuleIgnoreTable"]); Row moduleIgnoreTableRow = moduleIgnoreTableTable.CreateRow(null); moduleIgnoreTableRow[0] = "ModuleSubstitution"; } // add the ModuleConfiguration table to the ModuleIgnoreTable if (containsModuleConfiguration) { Table moduleIgnoreTableTable = this.activeOutput.EnsureTable(this.tableDefinitions["ModuleIgnoreTable"]); Row moduleIgnoreTableRow = moduleIgnoreTableTable.CreateRow(null); moduleIgnoreTableRow[0] = "ModuleConfiguration"; } #endif #if MOVE_TO_BACKEND // index all the file rows Table fileTable = this.activeOutput.Tables["File"]; RowDictionary indexedFileRows = (null == fileTable) ? new RowDictionary() : new RowDictionary(fileTable); // flag all the generated short file name collisions foreach (string fileId in generatedShortFileNameIdentifiers) { FileRow fileRow = indexedFileRows[fileId]; string[] names = fileRow.FileName.Split('|'); string shortFileName = names[0]; // create lists of conflicting generated short file names if (!generatedShortFileNames.Contains(shortFileName)) { generatedShortFileNames.Add(shortFileName, new ArrayList()); } ((ArrayList)generatedShortFileNames[shortFileName]).Add(fileRow); } // check for generated short file name collisions foreach (DictionaryEntry entry in generatedShortFileNames) { string shortFileName = (string)entry.Key; ArrayList fileRows = (ArrayList)entry.Value; if (1 < fileRows.Count) { // sort the rows by DiskId fileRows.Sort(); this.OnMessage(WixWarnings.GeneratedShortFileNameConflict(((FileRow)fileRows[0]).SourceLineNumbers, shortFileName)); for (int i = 1; i < fileRows.Count; i++) { FileRow fileRow = (FileRow)fileRows[i]; if (null != fileRow.SourceLineNumbers) { this.OnMessage(WixWarnings.GeneratedShortFileNameConflict2(fileRow.SourceLineNumbers)); } } } } #endif // copy the wix variable rows to the output after all overriding has been accounted for. foreach (var row in wixVariables.Values) { resolvedSection.Tuples.Add(row); } // Bundles have groups of data that must be flattened in a way different from other types. this.FlattenBundleTables(resolvedSection); if (this.Context.Messaging.EncounteredError) { return null; } var output = new Intermediate(resolvedSection.Id, new[] { resolvedSection }, null, null); #if MOVE_TO_BACKEND this.CheckOutputConsistency(output); #endif return this.Context.Messaging.EncounteredError ? null : output; } #if SOLVE_CUSTOM_TABLE /// /// Links the definition of a custom table. /// /// The table to link. /// Receives the linked definition of the custom table. private void LinkCustomTable(Table table, TableDefinitionCollection customTableDefinitions) { foreach (Row row in table.Rows) { bool bootstrapperApplicationData = (null != row[13] && 1 == (int)row[13]); if (null == row[4]) { this.OnMessage(WixErrors.ExpectedAttribute(row.SourceLineNumbers, "CustomTable/Column", "PrimaryKey")); } string[] columnNames = row[2].ToString().Split('\t'); string[] columnTypes = row[3].ToString().Split('\t'); string[] primaryKeys = row[4].ToString().Split('\t'); string[] minValues = row[5] == null ? null : row[5].ToString().Split('\t'); string[] maxValues = row[6] == null ? null : row[6].ToString().Split('\t'); string[] keyTables = row[7] == null ? null : row[7].ToString().Split('\t'); string[] keyColumns = row[8] == null ? null : row[8].ToString().Split('\t'); string[] categories = row[9] == null ? null : row[9].ToString().Split('\t'); string[] sets = row[10] == null ? null : row[10].ToString().Split('\t'); string[] descriptions = row[11] == null ? null : row[11].ToString().Split('\t'); string[] modularizations = row[12] == null ? null : row[12].ToString().Split('\t'); int currentPrimaryKey = 0; List columns = new List(columnNames.Length); for (int i = 0; i < columnNames.Length; ++i) { string name = columnNames[i]; ColumnType type = ColumnType.Unknown; if (columnTypes[i].StartsWith("s", StringComparison.OrdinalIgnoreCase)) { type = ColumnType.String; } else if (columnTypes[i].StartsWith("l", StringComparison.OrdinalIgnoreCase)) { type = ColumnType.Localized; } else if (columnTypes[i].StartsWith("i", StringComparison.OrdinalIgnoreCase)) { type = ColumnType.Number; } else if (columnTypes[i].StartsWith("v", StringComparison.OrdinalIgnoreCase)) { type = ColumnType.Object; } else { throw new WixException(WixErrors.UnknownCustomTableColumnType(row.SourceLineNumbers, columnTypes[i])); } bool nullable = columnTypes[i].Substring(0, 1) == columnTypes[i].Substring(0, 1).ToUpper(CultureInfo.InvariantCulture); int length = Convert.ToInt32(columnTypes[i].Substring(1), CultureInfo.InvariantCulture); bool primaryKey = false; if (currentPrimaryKey < primaryKeys.Length && primaryKeys[currentPrimaryKey] == columnNames[i]) { primaryKey = true; currentPrimaryKey++; } bool minValSet = null != minValues && null != minValues[i] && 0 < minValues[i].Length; int minValue = 0; if (minValSet) { minValue = Convert.ToInt32(minValues[i], CultureInfo.InvariantCulture); } bool maxValSet = null != maxValues && null != maxValues[i] && 0 < maxValues[i].Length; int maxValue = 0; if (maxValSet) { maxValue = Convert.ToInt32(maxValues[i], CultureInfo.InvariantCulture); } bool keyColumnSet = null != keyColumns && null != keyColumns[i] && 0 < keyColumns[i].Length; int keyColumn = 0; if (keyColumnSet) { keyColumn = Convert.ToInt32(keyColumns[i], CultureInfo.InvariantCulture); } ColumnCategory category = ColumnCategory.Unknown; if (null != categories && null != categories[i] && 0 < categories[i].Length) { switch (categories[i]) { case "Text": category = ColumnCategory.Text; break; case "UpperCase": category = ColumnCategory.UpperCase; break; case "LowerCase": category = ColumnCategory.LowerCase; break; case "Integer": category = ColumnCategory.Integer; break; case "DoubleInteger": category = ColumnCategory.DoubleInteger; break; case "TimeDate": category = ColumnCategory.TimeDate; break; case "Identifier": category = ColumnCategory.Identifier; break; case "Property": category = ColumnCategory.Property; break; case "Filename": category = ColumnCategory.Filename; break; case "WildCardFilename": category = ColumnCategory.WildCardFilename; break; case "Path": category = ColumnCategory.Path; break; case "Paths": category = ColumnCategory.Paths; break; case "AnyPath": category = ColumnCategory.AnyPath; break; case "DefaultDir": category = ColumnCategory.DefaultDir; break; case "RegPath": category = ColumnCategory.RegPath; break; case "Formatted": category = ColumnCategory.Formatted; break; case "FormattedSddl": category = ColumnCategory.FormattedSDDLText; break; case "Template": category = ColumnCategory.Template; break; case "Condition": category = ColumnCategory.Condition; break; case "Guid": category = ColumnCategory.Guid; break; case "Version": category = ColumnCategory.Version; break; case "Language": category = ColumnCategory.Language; break; case "Binary": category = ColumnCategory.Binary; break; case "CustomSource": category = ColumnCategory.CustomSource; break; case "Cabinet": category = ColumnCategory.Cabinet; break; case "Shortcut": category = ColumnCategory.Shortcut; break; default: break; } } string keyTable = keyTables != null ? keyTables[i] : null; string setValue = sets != null ? sets[i] : null; string description = descriptions != null ? descriptions[i] : null; string modString = modularizations != null ? modularizations[i] : null; ColumnModularizeType modularization = ColumnModularizeType.None; if (modString != null) { switch (modString) { case "None": modularization = ColumnModularizeType.None; break; case "Column": modularization = ColumnModularizeType.Column; break; case "Property": modularization = ColumnModularizeType.Property; break; case "Condition": modularization = ColumnModularizeType.Condition; break; case "CompanionFile": modularization = ColumnModularizeType.CompanionFile; break; case "SemicolonDelimited": modularization = ColumnModularizeType.SemicolonDelimited; break; } } ColumnDefinition columnDefinition = new ColumnDefinition(name, type, length, primaryKey, nullable, modularization, ColumnType.Localized == type, minValSet, minValue, maxValSet, maxValue, keyTable, keyColumnSet, keyColumn, category, setValue, description, true, true); columns.Add(columnDefinition); } TableDefinition customTable = new TableDefinition((string)row[0], columns, false, bootstrapperApplicationData, bootstrapperApplicationData); customTableDefinitions.Add(customTable); } } #endif #if MOVE_TO_BACKEND /// /// Checks for any tables in the output which are not allowed in the output type. /// /// The output to check. private void CheckForIllegalTables(Output output) { foreach (Table table in output.Tables) { switch (output.Type) { case OutputType.Module: if ("BBControl" == table.Name || "Billboard" == table.Name || "CCPSearch" == table.Name || "Feature" == table.Name || "LaunchCondition" == table.Name || "Media" == table.Name || "Patch" == table.Name || "Upgrade" == table.Name || "WixMerge" == table.Name) { foreach (Row row in table.Rows) { this.OnMessage(WixErrors.UnexpectedTableInMergeModule(row.SourceLineNumbers, table.Name)); } } else if ("Error" == table.Name) { foreach (Row row in table.Rows) { this.OnMessage(WixWarnings.DangerousTableInMergeModule(row.SourceLineNumbers, table.Name)); } } break; case OutputType.PatchCreation: if (!table.Definition.Unreal && "_SummaryInformation" != table.Name && "ExternalFiles" != table.Name && "FamilyFileRanges" != table.Name && "ImageFamilies" != table.Name && "PatchMetadata" != table.Name && "PatchSequence" != table.Name && "Properties" != table.Name && "TargetFiles_OptionalData" != table.Name && "TargetImages" != table.Name && "UpgradedFiles_OptionalData" != table.Name && "UpgradedFilesToIgnore" != table.Name && "UpgradedImages" != table.Name) { foreach (Row row in table.Rows) { this.OnMessage(WixErrors.UnexpectedTableInPatchCreationPackage(row.SourceLineNumbers, table.Name)); } } break; case OutputType.Patch: if (!table.Definition.Unreal && "_SummaryInformation" != table.Name && "Media" != table.Name && "MsiPatchMetadata" != table.Name && "MsiPatchSequence" != table.Name) { foreach (Row row in table.Rows) { this.OnMessage(WixErrors.UnexpectedTableInPatch(row.SourceLineNumbers, table.Name)); } } break; case OutputType.Product: if ("ModuleAdminExecuteSequence" == table.Name || "ModuleAdminUISequence" == table.Name || "ModuleAdvtExecuteSequence" == table.Name || "ModuleAdvtUISequence" == table.Name || "ModuleComponents" == table.Name || "ModuleConfiguration" == table.Name || "ModuleDependency" == table.Name || "ModuleExclusion" == table.Name || "ModuleIgnoreTable" == table.Name || "ModuleInstallExecuteSequence" == table.Name || "ModuleInstallUISequence" == table.Name || "ModuleSignature" == table.Name || "ModuleSubstitution" == table.Name) { foreach (Row row in table.Rows) { this.OnMessage(WixWarnings.UnexpectedTableInProduct(row.SourceLineNumbers, table.Name)); } } break; } } } #endif #if MOVE_TO_BACKEND /// /// Performs various consistency checks on the output. /// /// Output containing instance transform definitions. private void CheckOutputConsistency(Output output) { // Get the output's minimum installer version int outputInstallerVersion = int.MinValue; Table summaryInformationTable = output.Tables["_SummaryInformation"]; if (null != summaryInformationTable) { foreach (Row row in summaryInformationTable.Rows) { if (14 == (int)row[0]) { outputInstallerVersion = Convert.ToInt32(row[1], CultureInfo.InvariantCulture); break; } } } // ensure the Error table exists if output is marked for MSI 1.0 or below (see ICE40) if (100 >= outputInstallerVersion && OutputType.Product == output.Type) { output.EnsureTable(this.tableDefinitions["Error"]); } // check for the presence of tables/rows/columns that require MSI 1.1 or later if (110 > outputInstallerVersion) { Table isolatedComponentTable = output.Tables["IsolatedComponent"]; if (null != isolatedComponentTable) { foreach (Row row in isolatedComponentTable.Rows) { this.OnMessage(WixWarnings.TableIncompatibleWithInstallerVersion(row.SourceLineNumbers, "IsolatedComponent", outputInstallerVersion)); } } } // check for the presence of tables/rows/columns that require MSI 4.0 or later if (400 > outputInstallerVersion) { Table shortcutTable = output.Tables["Shortcut"]; if (null != shortcutTable) { foreach (Row row in shortcutTable.Rows) { if (null != row[12] || null != row[13] || null != row[14] || null != row[15]) { this.OnMessage(WixWarnings.ColumnsIncompatibleWithInstallerVersion(row.SourceLineNumbers, "Shortcut", outputInstallerVersion)); } } } } } #endif /// /// Sends a message to the message delegate if there is one. /// /// Message event arguments. public void OnMessage(Message message) { this.Context.Messaging.Write(message); } /// /// Load the standard action symbols. /// /// Collection of symbols. private void LoadStandardActionSymbols(IntermediateSection section, IDictionary symbols) { foreach (var actionRow in WindowsInstallerStandard.StandardActions()) { var symbol = new Symbol(section, actionRow); // If the action's symbol has not already been defined (i.e. overriden by the user), add it now. if (!symbols.ContainsKey(symbol.Name)) { symbols.Add(symbol.Name, symbol); } } } /// /// Process the complex references. /// /// Active section to add tuples to. /// Sections that are referenced during the link process. /// Collection of all components referenced by complex reference. /// Component to feature complex references. /// Feature to feature complex references. /// Module to feature complex references. private void ProcessComplexReferences(IntermediateSection resolvedSection, IEnumerable sections, ISet referencedComponents, ConnectToFeatureCollection componentsToFeatures, ConnectToFeatureCollection featuresToFeatures, ConnectToFeatureCollection modulesToFeatures) { Hashtable componentsToModules = new Hashtable(); foreach (var section in sections) { var featureComponents = new List(); foreach (var wixComplexReferenceRow in section.Tuples.OfType()) { ConnectToFeature connection; switch (wixComplexReferenceRow.ParentType) { case ComplexReferenceParentType.Feature: switch (wixComplexReferenceRow.ChildType) { case ComplexReferenceChildType.Component: connection = componentsToFeatures[wixComplexReferenceRow.Child]; if (null == connection) { componentsToFeatures.Add(new ConnectToFeature(section, wixComplexReferenceRow.Child, wixComplexReferenceRow.Parent, wixComplexReferenceRow.IsPrimary)); } else if (wixComplexReferenceRow.IsPrimary) { if (connection.IsExplicitPrimaryFeature) { this.OnMessage(ErrorMessages.MultiplePrimaryReferences(wixComplexReferenceRow.SourceLineNumbers, wixComplexReferenceRow.ChildType.ToString(), wixComplexReferenceRow.Child, wixComplexReferenceRow.ParentType.ToString(), wixComplexReferenceRow.Parent, (null != connection.PrimaryFeature ? "Feature" : "Product"), connection.PrimaryFeature ?? resolvedSection.Id)); continue; } else { connection.ConnectFeatures.Add(connection.PrimaryFeature); // move the guessed primary feature to the list of connects connection.PrimaryFeature = wixComplexReferenceRow.Parent; // set the new primary feature connection.IsExplicitPrimaryFeature = true; // and make sure we remember that we set it so we can fail if we try to set it again } } else { connection.ConnectFeatures.Add(wixComplexReferenceRow.Parent); } // add a row to the FeatureComponents table var featureComponent = new FeatureComponentsTuple(); featureComponent.Feature_ = wixComplexReferenceRow.Parent; featureComponent.Component_ = wixComplexReferenceRow.Child; featureComponents.Add(featureComponent); // index the component for finding orphaned records var symbolName = String.Concat("Component:", wixComplexReferenceRow.Child); referencedComponents.Add(symbolName); break; case ComplexReferenceChildType.Feature: connection = featuresToFeatures[wixComplexReferenceRow.Child]; if (null != connection) { this.OnMessage(ErrorMessages.MultiplePrimaryReferences(wixComplexReferenceRow.SourceLineNumbers, wixComplexReferenceRow.ChildType.ToString(), wixComplexReferenceRow.Child, wixComplexReferenceRow.ParentType.ToString(), wixComplexReferenceRow.Parent, (null != connection.PrimaryFeature ? "Feature" : "Product"), (null != connection.PrimaryFeature ? connection.PrimaryFeature : resolvedSection.Id))); continue; } featuresToFeatures.Add(new ConnectToFeature(section, wixComplexReferenceRow.Child, wixComplexReferenceRow.Parent, wixComplexReferenceRow.IsPrimary)); break; case ComplexReferenceChildType.Module: connection = modulesToFeatures[wixComplexReferenceRow.Child]; if (null == connection) { modulesToFeatures.Add(new ConnectToFeature(section, wixComplexReferenceRow.Child, wixComplexReferenceRow.Parent, wixComplexReferenceRow.IsPrimary)); } else if (wixComplexReferenceRow.IsPrimary) { if (connection.IsExplicitPrimaryFeature) { this.OnMessage(ErrorMessages.MultiplePrimaryReferences(wixComplexReferenceRow.SourceLineNumbers, wixComplexReferenceRow.ChildType.ToString(), wixComplexReferenceRow.Child, wixComplexReferenceRow.ParentType.ToString(), wixComplexReferenceRow.Parent, (null != connection.PrimaryFeature ? "Feature" : "Product"), (null != connection.PrimaryFeature ? connection.PrimaryFeature : resolvedSection.Id))); continue; } else { connection.ConnectFeatures.Add(connection.PrimaryFeature); // move the guessed primary feature to the list of connects connection.PrimaryFeature = wixComplexReferenceRow.Parent; // set the new primary feature connection.IsExplicitPrimaryFeature = true; // and make sure we remember that we set it so we can fail if we try to set it again } } else { connection.ConnectFeatures.Add(wixComplexReferenceRow.Parent); } break; default: throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, WixStrings.EXP_UnexpectedComplexReferenceChildType, Enum.GetName(typeof(ComplexReferenceChildType), wixComplexReferenceRow.ChildType))); } break; case ComplexReferenceParentType.Module: switch (wixComplexReferenceRow.ChildType) { case ComplexReferenceChildType.Component: if (componentsToModules.ContainsKey(wixComplexReferenceRow.Child)) { this.OnMessage(ErrorMessages.ComponentReferencedTwice(wixComplexReferenceRow.SourceLineNumbers, wixComplexReferenceRow.Child)); continue; } else { componentsToModules.Add(wixComplexReferenceRow.Child, wixComplexReferenceRow); // should always be new // add a row to the ModuleComponents table var moduleComponent = new ModuleComponentsTuple(); moduleComponent.Component = wixComplexReferenceRow.Child; moduleComponent.ModuleID = wixComplexReferenceRow.Parent; moduleComponent.Language = Convert.ToInt32(wixComplexReferenceRow.ParentLanguage); } // index the component for finding orphaned records string componentSymbolName = String.Concat("Component:", wixComplexReferenceRow.Child); referencedComponents.Add(componentSymbolName); break; default: throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, WixStrings.EXP_UnexpectedComplexReferenceChildType, Enum.GetName(typeof(ComplexReferenceChildType), wixComplexReferenceRow.ChildType))); } break; case ComplexReferenceParentType.Patch: switch (wixComplexReferenceRow.ChildType) { case ComplexReferenceChildType.PatchFamily: case ComplexReferenceChildType.PatchFamilyGroup: break; default: throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, WixStrings.EXP_UnexpectedComplexReferenceChildType, Enum.GetName(typeof(ComplexReferenceChildType), wixComplexReferenceRow.ChildType))); } break; case ComplexReferenceParentType.Product: switch (wixComplexReferenceRow.ChildType) { case ComplexReferenceChildType.Feature: connection = featuresToFeatures[wixComplexReferenceRow.Child]; if (null != connection) { this.OnMessage(ErrorMessages.MultiplePrimaryReferences(wixComplexReferenceRow.SourceLineNumbers, wixComplexReferenceRow.ChildType.ToString(), wixComplexReferenceRow.Child, wixComplexReferenceRow.ParentType.ToString(), wixComplexReferenceRow.Parent, (null != connection.PrimaryFeature ? "Feature" : "Product"), (null != connection.PrimaryFeature ? connection.PrimaryFeature : resolvedSection.Id))); continue; } featuresToFeatures.Add(new ConnectToFeature(section, wixComplexReferenceRow.Child, null, wixComplexReferenceRow.IsPrimary)); break; default: throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, WixStrings.EXP_UnexpectedComplexReferenceChildType, Enum.GetName(typeof(ComplexReferenceChildType), wixComplexReferenceRow.ChildType))); } break; default: // Note: Groups have been processed before getting here so they are not handled by any case above. throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, WixStrings.EXP_UnexpectedComplexReferenceChildType, Enum.GetName(typeof(ComplexReferenceParentType), wixComplexReferenceRow.ParentType))); } } foreach (var featureComponent in featureComponents) { section.Tuples.Add(featureComponent); } } } /// /// Flattens all complex references in all sections in the collection. /// /// Sections that are referenced during the link process. private void FlattenSectionsComplexReferences(IEnumerable sections) { var parentGroups = new Dictionary>(); var parentGroupsSections = new Dictionary(); var parentGroupsNeedingProcessing = new Dictionary(); // DisplaySectionComplexReferences("--- section's complex references before flattening ---", sections); // Step 1: Gather all of the complex references that are going participate // in the flatting process. This means complex references that have "grouping // parents" of Features, Modules, and, of course, Groups. These references // that participate in a "grouping parent" will be removed from their section // now and after processing added back in Step 3 below. foreach (var section in sections) { // Count down because we'll sometimes remove items from the list. for (int i = section.Tuples.Count - 1; i >= 0; --i) { // Only process the "grouping parents" such as FeatureGroup, ComponentGroup, Feature, // and Module. Non-grouping complex references are simple and // resolved during normal complex reference resolutions. if (section.Tuples[i] is WixComplexReferenceTuple wixComplexReferenceRow && (ComplexReferenceParentType.FeatureGroup == wixComplexReferenceRow.ParentType || ComplexReferenceParentType.ComponentGroup == wixComplexReferenceRow.ParentType || ComplexReferenceParentType.Feature == wixComplexReferenceRow.ParentType || ComplexReferenceParentType.Module == wixComplexReferenceRow.ParentType || ComplexReferenceParentType.PatchFamilyGroup == wixComplexReferenceRow.ParentType || ComplexReferenceParentType.Product == wixComplexReferenceRow.ParentType)) { var parentTypeAndId = CombineTypeAndId(wixComplexReferenceRow.ParentType, wixComplexReferenceRow.Parent); // Group all complex references with a common parent // together so we can find them quickly while processing in // Step 2. if (!parentGroups.TryGetValue(parentTypeAndId, out var childrenComplexRefs)) { childrenComplexRefs = new List(); parentGroups.Add(parentTypeAndId, childrenComplexRefs); } childrenComplexRefs.Add(wixComplexReferenceRow); section.Tuples.RemoveAt(i); // Remember the mapping from set of complex references with a common // parent to their section. We'll need this to add them back to the // correct section in Step 3. if (!parentGroupsSections.TryGetValue(parentTypeAndId, out var parentSection)) { parentGroupsSections.Add(parentTypeAndId, section); } // If the child of the complex reference is another group, then in Step 2 // we're going to have to process this complex reference again to copy // the child group's references into the parent group. if ((ComplexReferenceChildType.ComponentGroup == wixComplexReferenceRow.ChildType) || (ComplexReferenceChildType.FeatureGroup == wixComplexReferenceRow.ChildType) || (ComplexReferenceChildType.PatchFamilyGroup == wixComplexReferenceRow.ChildType)) { if (!parentGroupsNeedingProcessing.ContainsKey(parentTypeAndId)) { parentGroupsNeedingProcessing.Add(parentTypeAndId, section); } } } } } Debug.Assert(parentGroups.Count == parentGroupsSections.Count); Debug.Assert(parentGroupsNeedingProcessing.Count <= parentGroups.Count); // DisplaySectionComplexReferences("\r\n\r\n--- section's complex references middle of flattening ---", sections); // Step 2: Loop through the parent groups that have nested groups removing // them from the hash table as they are processed. At the end of this the // complex references should all be flattened. var keys = parentGroupsNeedingProcessing.Keys.ToList(); foreach (string key in keys) { if (parentGroupsNeedingProcessing.ContainsKey(key)) { var loopDetector = new Stack(); this.FlattenGroup(key, loopDetector, parentGroups, parentGroupsNeedingProcessing); } else { // the group must have allready been procesed and removed from the hash table } } Debug.Assert(0 == parentGroupsNeedingProcessing.Count); // Step 3: Finally, ensure that all of the groups that were removed // in Step 1 and flattened in Step 2 are added to their appropriate // section. This is where we will toss out the final no-longer-needed // groups. foreach (var parentGroup in parentGroups.Keys) { var section = parentGroupsSections[parentGroup]; foreach (var wixComplexReferenceRow in parentGroups[parentGroup]) { if ((ComplexReferenceParentType.FeatureGroup != wixComplexReferenceRow.ParentType) && (ComplexReferenceParentType.ComponentGroup != wixComplexReferenceRow.ParentType) && (ComplexReferenceParentType.PatchFamilyGroup != wixComplexReferenceRow.ParentType)) { section.Tuples.Add(wixComplexReferenceRow); } } } // DisplaySectionComplexReferences("\r\n\r\n--- section's complex references after flattening ---", sections); } private string CombineTypeAndId(ComplexReferenceParentType type, string id) { return String.Concat(type.ToString(), ":", id); } private string CombineTypeAndId(ComplexReferenceChildType type, string id) { return String.Concat(type.ToString(), ":", id); } /// /// Recursively processes the group. /// /// String combination type and id of group to process next. /// Stack of groups processed thus far. Used to detect loops. /// Hash table of complex references grouped by parent id. /// Hash table of parent groups that still have nested groups that need to be flattened. private void FlattenGroup(string parentTypeAndId, Stack loopDetector, Dictionary> parentGroups, Dictionary parentGroupsNeedingProcessing) { Debug.Assert(parentGroupsNeedingProcessing.ContainsKey(parentTypeAndId)); loopDetector.Push(parentTypeAndId); // push this complex reference parent identfier into the stack for loop verifying var allNewChildComplexReferences = new List(); var referencesToParent = parentGroups[parentTypeAndId]; foreach (var wixComplexReferenceRow in referencesToParent) { Debug.Assert(ComplexReferenceParentType.ComponentGroup == wixComplexReferenceRow.ParentType || ComplexReferenceParentType.FeatureGroup == wixComplexReferenceRow.ParentType || ComplexReferenceParentType.Feature == wixComplexReferenceRow.ParentType || ComplexReferenceParentType.Module == wixComplexReferenceRow.ParentType || ComplexReferenceParentType.Product == wixComplexReferenceRow.ParentType || ComplexReferenceParentType.PatchFamilyGroup == wixComplexReferenceRow.ParentType || ComplexReferenceParentType.Patch == wixComplexReferenceRow.ParentType); Debug.Assert(parentTypeAndId == CombineTypeAndId(wixComplexReferenceRow.ParentType, wixComplexReferenceRow.Parent)); // We are only interested processing when the child is a group. if ((ComplexReferenceChildType.ComponentGroup == wixComplexReferenceRow.ChildType) || (ComplexReferenceChildType.FeatureGroup == wixComplexReferenceRow.ChildType) || (ComplexReferenceChildType.PatchFamilyGroup == wixComplexReferenceRow.ChildType)) { string childTypeAndId = CombineTypeAndId(wixComplexReferenceRow.ChildType, wixComplexReferenceRow.Child); if (loopDetector.Contains(childTypeAndId)) { // Create a comma delimited list of the references that participate in the // loop for the error message. Start at the bottom of the stack and work the // way up to present the loop as a directed graph. var loop = String.Join(" -> ", loopDetector); this.OnMessage(ErrorMessages.ReferenceLoopDetected(wixComplexReferenceRow?.SourceLineNumbers, loop)); // Cleanup the parentGroupsNeedingProcessing and the loopDetector just like the // exit of this method does at the end because we are exiting early. loopDetector.Pop(); parentGroupsNeedingProcessing.Remove(parentTypeAndId); return; // bail } // Check to see if the child group still needs to be processed. If so, // go do that so that we'll get all of that children's (and children's // children) complex references correctly merged into our parent group. if (parentGroupsNeedingProcessing.ContainsKey(childTypeAndId)) { this.FlattenGroup(childTypeAndId, loopDetector, parentGroups, parentGroupsNeedingProcessing); } // If the child is a parent to anything (i.e. the parent has grandchildren) // clone each of the children's complex references, repoint them to the parent // complex reference (because we're moving references up the tree), and finally // add the cloned child's complex reference to the list of complex references // that we'll eventually add to the parent group. if (parentGroups.TryGetValue(childTypeAndId, out var referencesToChild)) { foreach (var crefChild in referencesToChild) { // Only merge up the non-group items since groups are purged // after this part of the processing anyway (cloning them would // be a complete waste of time). if ((ComplexReferenceChildType.FeatureGroup != crefChild.ChildType) || (ComplexReferenceChildType.ComponentGroup != crefChild.ChildType) || (ComplexReferenceChildType.PatchFamilyGroup != crefChild.ChildType)) { var crefChildClone = crefChild.Clone(); Debug.Assert(crefChildClone.Parent == wixComplexReferenceRow.Child); crefChildClone.Reparent(wixComplexReferenceRow); allNewChildComplexReferences.Add(crefChildClone); } } } } } // Add the children group's complex references to the parent // group. Clean out any left over groups and quietly remove any // duplicate complex references that occurred during the merge. referencesToParent.AddRange(allNewChildComplexReferences); referencesToParent.Sort(ComplexReferenceComparision); for (int i = referencesToParent.Count - 1; i >= 0; --i) { var wixComplexReferenceRow = referencesToParent[i]; if ((ComplexReferenceChildType.FeatureGroup == wixComplexReferenceRow.ChildType) || (ComplexReferenceChildType.ComponentGroup == wixComplexReferenceRow.ChildType) || (ComplexReferenceChildType.PatchFamilyGroup == wixComplexReferenceRow.ChildType)) { referencesToParent.RemoveAt(i); } else if (i > 0) { // Since the list is already sorted, we can find duplicates by simply // looking at the next sibling in the list and tossing out one if they // match. var crefCompare = referencesToParent[i - 1]; if (0 == wixComplexReferenceRow.CompareToWithoutConsideringPrimary(crefCompare)) { referencesToParent.RemoveAt(i); } } } int ComplexReferenceComparision(WixComplexReferenceTuple x, WixComplexReferenceTuple y) { var comparison = x.ChildType - y.ChildType; if (0 == comparison) { comparison = String.Compare(x.Child, y.Child, StringComparison.Ordinal); if (0 == comparison) { comparison = x.ParentType - y.ParentType; if (0 == comparison) { comparison = String.Compare(x.ParentLanguage ?? String.Empty, y.ParentLanguage ?? String.Empty, StringComparison.Ordinal); if (0 == comparison) { comparison = String.Compare(x.Parent, y.Parent, StringComparison.Ordinal); } } } } return comparison; } loopDetector.Pop(); // pop this complex reference off the stack since we're done verify the loop here parentGroupsNeedingProcessing.Remove(parentTypeAndId); // remove the newly processed complex reference } /* /// /// Debugging method for displaying the section complex references. /// /// The header. /// The sections to display. private void DisplaySectionComplexReferences(string header, SectionCollection sections) { Console.WriteLine(header); foreach (Section section in sections) { Table wixComplexReferenceTable = section.Tables["WixComplexReference"]; foreach (WixComplexReferenceRow cref in wixComplexReferenceTable.Rows) { Console.WriteLine("Section: {0} Parent: {1} Type: {2} Child: {3} Primary: {4}", section.Id, cref.ParentId, cref.ParentType, cref.ChildId, cref.IsPrimary); } } } */ /// /// Flattens the tables used in a Bundle. /// /// Output containing the tables to process. private void FlattenBundleTables(IntermediateSection entrySection) { if (SectionType.Bundle != entrySection.Type) { return; } // We need to flatten the nested PayloadGroups and PackageGroups under // UX, Chain, and any Containers. When we're done, the WixGroups table // will hold Payloads under UX, ChainPackages (references?) under Chain, // and ChainPackages/Payloads under the attached and any detatched // Containers. var groups = new WixGroupingOrdering(entrySection, this.Context.Messaging); // Create UX payloads and Package payloads groups.UseTypes(new string[] { "Container", "Layout", "PackageGroup", "PayloadGroup", "Package" }, new string[] { "PackageGroup", "Package", "PayloadGroup", "Payload" }); groups.FlattenAndRewriteGroups("Package", false); groups.FlattenAndRewriteGroups("Container", false); groups.FlattenAndRewriteGroups("Layout", false); // Create Chain packages... groups.UseTypes(new string[] { "PackageGroup" }, new string[] { "Package", "PackageGroup" }); groups.FlattenAndRewriteRows("PackageGroup", "WixChain", false); groups.RemoveUsedGroupRows(); } /// /// Resolves the features connected to other features in the active output. /// /// Feature to feature complex references. /// All symbols loaded from the sections. private void ResolveFeatureToFeatureConnects(ConnectToFeatureCollection featuresToFeatures, IDictionary allSymbols) { foreach (ConnectToFeature connection in featuresToFeatures) { var wixSimpleReferenceRow = new WixSimpleReferenceTuple(); wixSimpleReferenceRow.Table = "Feature"; wixSimpleReferenceRow.PrimaryKeys = connection.ChildId; if (!allSymbols.TryGetValue(wixSimpleReferenceRow.SymbolicName, out var symbol)) { continue; } var row = symbol.Row; row.Set(1, connection.PrimaryFeature); } } #if DEAD_CODE /// /// Copies a table's rows to an output table. /// /// Source table to copy rows from. /// Destination table in output to copy rows into. /// Id of the section that the table lives in. private void CopyTableRowsToOutputTable(Table table, Table outputTable, string sectionId) { int[] localizedColumns = new int[table.Definition.Columns.Count]; int localizedColumnCount = 0; // if there are localization strings, figure out which columns can be localized in this table if (null != this.Localizer) { for (int i = 0; i < table.Definition.Columns.Count; i++) { if (table.Definition.Columns[i].IsLocalizable) { localizedColumns[localizedColumnCount++] = i; } } } // process each row in the table doing the string resource substitutions // then add the row to the output foreach (Row row in table.Rows) { for (int j = 0; j < localizedColumnCount; j++) { Field field = row.Fields[localizedColumns[j]]; if (null != field.Data) { field.Data = this.WixVariableResolver.ResolveVariables(row.SourceLineNumbers, (string)field.Data, true); } } row.SectionId = (this.sectionIdOnRows ? sectionId : null); outputTable.Rows.Add(row); } } #endif /// /// Resolve features for columns that have null guid placeholders. /// /// Rows to resolve. /// Number of the column containing the connection identifier. /// Number of the column containing the feature. /// Connect to feature complex references. /// Hashtable of known components under multiple features. private void ResolveFeatures(IntermediateTuple row, int connectionColumn, int featureColumn, ConnectToFeatureCollection connectToFeatures, Hashtable multipleFeatureComponents) { var connectionId = row.AsString(connectionColumn); var featureId = row.AsString(featureColumn); if (emptyGuid == featureId) { ConnectToFeature connection = connectToFeatures[connectionId]; if (null == connection) { // display an error for the component or merge module as approrpriate if (null != multipleFeatureComponents) { this.OnMessage(ErrorMessages.ComponentExpectedFeature(row.SourceLineNumbers, connectionId, row.Definition.Name, row.Id.Id)); } else { this.OnMessage(ErrorMessages.MergeModuleExpectedFeature(row.SourceLineNumbers, connectionId)); } } else { // check for unique, implicit, primary feature parents with multiple possible parent features if (this.ShowPedanticMessages && !connection.IsExplicitPrimaryFeature && 0 < connection.ConnectFeatures.Count) { // display a warning for the component or merge module as approrpriate if (null != multipleFeatureComponents) { if (!multipleFeatureComponents.Contains(connectionId)) { this.OnMessage(WarningMessages.ImplicitComponentPrimaryFeature(connectionId)); // remember this component so only one warning is generated for it multipleFeatureComponents[connectionId] = null; } } else { this.OnMessage(WarningMessages.ImplicitMergeModulePrimaryFeature(connectionId)); } } // set the feature row.Set(featureColumn, connection.PrimaryFeature); } } } } }