From 4ebc33174c02e1c9f5693b5ef38ecfe3292c687f Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Fri, 11 Aug 2017 00:39:04 -0700 Subject: Move to .NET Core 2.0 --- WixToolset.Data.sln | 43 + appveyor.yml | 24 + src/Directory.Build.props | 12 + src/WixToolset.Data/AccessModifier.cs | 27 + src/WixToolset.Data/AssemblyInfo.cs | 11 + .../BundleApprovedExeForElevationAttributes.cs | 16 + src/WixToolset.Data/ColumnDefinition.cs | 1032 ++ src/WixToolset.Data/Common.cs | 53 + src/WixToolset.Data/ComplexReferenceChildType.cs | 46 + src/WixToolset.Data/ComplexReferenceParentType.cs | 49 + src/WixToolset.Data/CompressionLevel.cs | 26 + src/WixToolset.Data/Data/Xsd/actions.xsd | 73 + src/WixToolset.Data/Data/Xsd/libraries.xsd | 66 + src/WixToolset.Data/Data/Xsd/objects.xsd | 143 + src/WixToolset.Data/Data/Xsd/outputs.xsd | 66 + src/WixToolset.Data/Data/Xsd/pdbs.xsd | 32 + src/WixToolset.Data/Data/Xsd/tables.xsd | 248 + src/WixToolset.Data/Data/actions.xml | 76 + src/WixToolset.Data/Data/messages.xml | 95 + src/WixToolset.Data/Data/tables.xml | 1962 +++ src/WixToolset.Data/DisplayEventArgs.cs | 15 + src/WixToolset.Data/DuplicateSymbolsException.cs | 35 + src/WixToolset.Data/EmptyRule.cs | 22 + src/WixToolset.Data/Field.cs | 266 + src/WixToolset.Data/FileFormat.cs | 17 + src/WixToolset.Data/FileStructure.cs | 294 + src/WixToolset.Data/ILibraryBinaryFileResolver.cs | 9 + src/WixToolset.Data/IMessageHandler.cs | 18 + src/WixToolset.Data/Intermediate.cs | 189 + src/WixToolset.Data/Library.cs | 297 + src/WixToolset.Data/Localization.cs | 372 + src/WixToolset.Data/LocalizedControl.cs | 57 + src/WixToolset.Data/MessageEventArgs.cs | 176 + src/WixToolset.Data/MessageLevel.cs | 25 + src/WixToolset.Data/Messaging.cs | 173 + src/WixToolset.Data/Msi/MsiInterop.cs | 313 + src/WixToolset.Data/NonClosingStreamWrapper.cs | 105 + src/WixToolset.Data/ObjectField.cs | 183 + src/WixToolset.Data/Output.cs | 394 + src/WixToolset.Data/OutputType.cs | 31 + src/WixToolset.Data/PackagingType.cs | 11 + src/WixToolset.Data/Pdb.cs | 163 + src/WixToolset.Data/Platform.cs | 22 + src/WixToolset.Data/Row.cs | 626 + src/WixToolset.Data/RowDictionary.cs | 84 + src/WixToolset.Data/RowIndexedList.cs | 301 + src/WixToolset.Data/RowOperation.cs | 30 + src/WixToolset.Data/Rows/BBControlRow.cs | 113 + src/WixToolset.Data/Rows/ComponentRow.cs | 245 + src/WixToolset.Data/Rows/ContainerType.cs | 13 + src/WixToolset.Data/Rows/ControlRow.cs | 143 + src/WixToolset.Data/Rows/ExitCodeBehaviorType.cs | 13 + src/WixToolset.Data/Rows/FileAssemblyType.cs | 19 + src/WixToolset.Data/Rows/FileRow.cs | 640 + src/WixToolset.Data/Rows/MediaRow.cs | 80 + src/WixToolset.Data/Rows/PatchAttributeType.cs | 27 + src/WixToolset.Data/Rows/PropertyRow.cs | 42 + .../Rows/SummaryInfoRowCollection.cs | 42 + src/WixToolset.Data/Rows/SymbolPathType.cs | 17 + src/WixToolset.Data/Rows/UpgradeRow.cs | 90 + src/WixToolset.Data/Rows/WixActionRow.cs | 374 + src/WixToolset.Data/Rows/WixActionRowCollection.cs | 222 + .../Rows/WixApprovedExeForElevationRow.cs | 79 + src/WixToolset.Data/Rows/WixBundleCatalogRow.cs | 50 + src/WixToolset.Data/Rows/WixBundleContainerRow.cs | 78 + .../Rows/WixBundleExePackageAttributes.cs | 12 + src/WixToolset.Data/Rows/WixBundleExePackageRow.cs | 101 + src/WixToolset.Data/Rows/WixBundleMsiFeatureRow.cs | 93 + .../Rows/WixBundleMsiPackageAttributes.cs | 15 + src/WixToolset.Data/Rows/WixBundleMsiPackageRow.cs | 137 + .../Rows/WixBundleMsiPropertyRow.cs | 58 + .../Rows/WixBundleMspPackageAttributes.cs | 14 + src/WixToolset.Data/Rows/WixBundleMspPackageRow.cs | 99 + src/WixToolset.Data/Rows/WixBundleMsuPackageRow.cs | 57 + .../Rows/WixBundlePackageAttributes.cs | 13 + .../Rows/WixBundlePackageCommandLineRow.cs | 82 + .../Rows/WixBundlePackageExitCodeRow.cs | 51 + src/WixToolset.Data/Rows/WixBundlePackageRow.cs | 226 + src/WixToolset.Data/Rows/WixBundlePackageType.cs | 15 + .../Rows/WixBundlePatchTargetCodeRow.cs | 81 + src/WixToolset.Data/Rows/WixBundlePayloadRow.cs | 185 + .../Rows/WixBundleRelatedPackageRow.cs | 87 + .../Rows/WixBundleRollbackBoundaryRow.cs | 59 + src/WixToolset.Data/Rows/WixBundleRow.cs | 228 + .../Rows/WixBundleSlipstreamMspRow.cs | 48 + src/WixToolset.Data/Rows/WixBundleUpdateRow.cs | 38 + src/WixToolset.Data/Rows/WixBundleVariableRow.cs | 80 + src/WixToolset.Data/Rows/WixChainAttributes.cs | 15 + src/WixToolset.Data/Rows/WixChainItemRow.cs | 39 + src/WixToolset.Data/Rows/WixChainRow.cs | 63 + src/WixToolset.Data/Rows/WixComplexReferenceRow.cs | 208 + src/WixToolset.Data/Rows/WixDeltaPatchFileRow.cs | 142 + .../Rows/WixDeltaPatchSymbolPathsRow.cs | 58 + src/WixToolset.Data/Rows/WixFileRow.cs | 161 + src/WixToolset.Data/Rows/WixGroupRow.cs | 62 + src/WixToolset.Data/Rows/WixMediaRow.cs | 60 + src/WixToolset.Data/Rows/WixMediaTemplateRow.cs | 81 + src/WixToolset.Data/Rows/WixMergeRow.cs | 149 + .../Rows/WixPayloadPropertiesRow.cs | 81 + src/WixToolset.Data/Rows/WixPropertyRow.cs | 118 + src/WixToolset.Data/Rows/WixRelatedBundleRow.cs | 52 + src/WixToolset.Data/Rows/WixSimpleReferenceRow.cs | 63 + .../Rows/WixUpdateRegistrationRow.cs | 62 + src/WixToolset.Data/Rows/WixVariableRow.cs | 81 + src/WixToolset.Data/Section.cs | 240 + src/WixToolset.Data/SectionType.cs | 31 + src/WixToolset.Data/Serialize/CodeDomInterfaces.cs | 96 + src/WixToolset.Data/Serialize/CodeDomReader.cs | 161 + src/WixToolset.Data/Serialize/ElementCollection.cs | 617 + src/WixToolset.Data/SourceLineNumber.cs | 205 + src/WixToolset.Data/SubStorage.cs | 109 + src/WixToolset.Data/Symbol.cs | 89 + src/WixToolset.Data/Table.cs | 446 + src/WixToolset.Data/TableDefinition.cs | 334 + src/WixToolset.Data/TableDefinitionCollection.cs | 229 + src/WixToolset.Data/TableExtensions.cs | 23 + src/WixToolset.Data/TableIndexedCollection.cs | 153 + src/WixToolset.Data/TableOperation.cs | 25 + src/WixToolset.Data/WindowsInstallerStandard.cs | 444 + src/WixToolset.Data/WixCorruptFileException.cs | 29 + src/WixToolset.Data/WixDataStrings.Designer.cs | 307 + src/WixToolset.Data/WixDataStrings.resx | 204 + src/WixToolset.Data/WixException.cs | 44 + src/WixToolset.Data/WixInvalidIdtException.cs | 32 + .../WixMissingTableDefinitionException.cs | 22 + src/WixToolset.Data/WixToolset.Data.csproj | 195 + .../WixUnexpectedFileFormatException.cs | 35 + src/WixToolset.Data/Xsd/wix.xsd | 13036 +++++++++++++++++++ src/WixToolset.Data/Xsd/wixloc.xsd | 134 + src/WixToolset.Data/YesNoAlwaysType.cs | 25 + src/WixToolset.Data/YesNoDefaultType.cs | 25 + src/WixToolset.Data/YesNoType.cs | 22 + src/nuget.config | 11 + src/version.json | 11 + 134 files changed, 31223 insertions(+) create mode 100644 WixToolset.Data.sln create mode 100644 appveyor.yml create mode 100644 src/Directory.Build.props create mode 100644 src/WixToolset.Data/AccessModifier.cs create mode 100644 src/WixToolset.Data/AssemblyInfo.cs create mode 100644 src/WixToolset.Data/BundleApprovedExeForElevationAttributes.cs create mode 100644 src/WixToolset.Data/ColumnDefinition.cs create mode 100644 src/WixToolset.Data/Common.cs create mode 100644 src/WixToolset.Data/ComplexReferenceChildType.cs create mode 100644 src/WixToolset.Data/ComplexReferenceParentType.cs create mode 100644 src/WixToolset.Data/CompressionLevel.cs create mode 100644 src/WixToolset.Data/Data/Xsd/actions.xsd create mode 100644 src/WixToolset.Data/Data/Xsd/libraries.xsd create mode 100644 src/WixToolset.Data/Data/Xsd/objects.xsd create mode 100644 src/WixToolset.Data/Data/Xsd/outputs.xsd create mode 100644 src/WixToolset.Data/Data/Xsd/pdbs.xsd create mode 100644 src/WixToolset.Data/Data/Xsd/tables.xsd create mode 100644 src/WixToolset.Data/Data/actions.xml create mode 100644 src/WixToolset.Data/Data/messages.xml create mode 100644 src/WixToolset.Data/Data/tables.xml create mode 100644 src/WixToolset.Data/DisplayEventArgs.cs create mode 100644 src/WixToolset.Data/DuplicateSymbolsException.cs create mode 100644 src/WixToolset.Data/EmptyRule.cs create mode 100644 src/WixToolset.Data/Field.cs create mode 100644 src/WixToolset.Data/FileFormat.cs create mode 100644 src/WixToolset.Data/FileStructure.cs create mode 100644 src/WixToolset.Data/ILibraryBinaryFileResolver.cs create mode 100644 src/WixToolset.Data/IMessageHandler.cs create mode 100644 src/WixToolset.Data/Intermediate.cs create mode 100644 src/WixToolset.Data/Library.cs create mode 100644 src/WixToolset.Data/Localization.cs create mode 100644 src/WixToolset.Data/LocalizedControl.cs create mode 100644 src/WixToolset.Data/MessageEventArgs.cs create mode 100644 src/WixToolset.Data/MessageLevel.cs create mode 100644 src/WixToolset.Data/Messaging.cs create mode 100644 src/WixToolset.Data/Msi/MsiInterop.cs create mode 100644 src/WixToolset.Data/NonClosingStreamWrapper.cs create mode 100644 src/WixToolset.Data/ObjectField.cs create mode 100644 src/WixToolset.Data/Output.cs create mode 100644 src/WixToolset.Data/OutputType.cs create mode 100644 src/WixToolset.Data/PackagingType.cs create mode 100644 src/WixToolset.Data/Pdb.cs create mode 100644 src/WixToolset.Data/Platform.cs create mode 100644 src/WixToolset.Data/Row.cs create mode 100644 src/WixToolset.Data/RowDictionary.cs create mode 100644 src/WixToolset.Data/RowIndexedList.cs create mode 100644 src/WixToolset.Data/RowOperation.cs create mode 100644 src/WixToolset.Data/Rows/BBControlRow.cs create mode 100644 src/WixToolset.Data/Rows/ComponentRow.cs create mode 100644 src/WixToolset.Data/Rows/ContainerType.cs create mode 100644 src/WixToolset.Data/Rows/ControlRow.cs create mode 100644 src/WixToolset.Data/Rows/ExitCodeBehaviorType.cs create mode 100644 src/WixToolset.Data/Rows/FileAssemblyType.cs create mode 100644 src/WixToolset.Data/Rows/FileRow.cs create mode 100644 src/WixToolset.Data/Rows/MediaRow.cs create mode 100644 src/WixToolset.Data/Rows/PatchAttributeType.cs create mode 100644 src/WixToolset.Data/Rows/PropertyRow.cs create mode 100644 src/WixToolset.Data/Rows/SummaryInfoRowCollection.cs create mode 100644 src/WixToolset.Data/Rows/SymbolPathType.cs create mode 100644 src/WixToolset.Data/Rows/UpgradeRow.cs create mode 100644 src/WixToolset.Data/Rows/WixActionRow.cs create mode 100644 src/WixToolset.Data/Rows/WixActionRowCollection.cs create mode 100644 src/WixToolset.Data/Rows/WixApprovedExeForElevationRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleCatalogRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleContainerRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleExePackageAttributes.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleExePackageRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleMsiFeatureRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleMsiPackageAttributes.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleMsiPackageRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleMsiPropertyRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleMspPackageAttributes.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleMspPackageRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleMsuPackageRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundlePackageAttributes.cs create mode 100644 src/WixToolset.Data/Rows/WixBundlePackageCommandLineRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundlePackageExitCodeRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundlePackageRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundlePackageType.cs create mode 100644 src/WixToolset.Data/Rows/WixBundlePatchTargetCodeRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundlePayloadRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleRelatedPackageRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleRollbackBoundaryRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleSlipstreamMspRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleUpdateRow.cs create mode 100644 src/WixToolset.Data/Rows/WixBundleVariableRow.cs create mode 100644 src/WixToolset.Data/Rows/WixChainAttributes.cs create mode 100644 src/WixToolset.Data/Rows/WixChainItemRow.cs create mode 100644 src/WixToolset.Data/Rows/WixChainRow.cs create mode 100644 src/WixToolset.Data/Rows/WixComplexReferenceRow.cs create mode 100644 src/WixToolset.Data/Rows/WixDeltaPatchFileRow.cs create mode 100644 src/WixToolset.Data/Rows/WixDeltaPatchSymbolPathsRow.cs create mode 100644 src/WixToolset.Data/Rows/WixFileRow.cs create mode 100644 src/WixToolset.Data/Rows/WixGroupRow.cs create mode 100644 src/WixToolset.Data/Rows/WixMediaRow.cs create mode 100644 src/WixToolset.Data/Rows/WixMediaTemplateRow.cs create mode 100644 src/WixToolset.Data/Rows/WixMergeRow.cs create mode 100644 src/WixToolset.Data/Rows/WixPayloadPropertiesRow.cs create mode 100644 src/WixToolset.Data/Rows/WixPropertyRow.cs create mode 100644 src/WixToolset.Data/Rows/WixRelatedBundleRow.cs create mode 100644 src/WixToolset.Data/Rows/WixSimpleReferenceRow.cs create mode 100644 src/WixToolset.Data/Rows/WixUpdateRegistrationRow.cs create mode 100644 src/WixToolset.Data/Rows/WixVariableRow.cs create mode 100644 src/WixToolset.Data/Section.cs create mode 100644 src/WixToolset.Data/SectionType.cs create mode 100644 src/WixToolset.Data/Serialize/CodeDomInterfaces.cs create mode 100644 src/WixToolset.Data/Serialize/CodeDomReader.cs create mode 100644 src/WixToolset.Data/Serialize/ElementCollection.cs create mode 100644 src/WixToolset.Data/SourceLineNumber.cs create mode 100644 src/WixToolset.Data/SubStorage.cs create mode 100644 src/WixToolset.Data/Symbol.cs create mode 100644 src/WixToolset.Data/Table.cs create mode 100644 src/WixToolset.Data/TableDefinition.cs create mode 100644 src/WixToolset.Data/TableDefinitionCollection.cs create mode 100644 src/WixToolset.Data/TableExtensions.cs create mode 100644 src/WixToolset.Data/TableIndexedCollection.cs create mode 100644 src/WixToolset.Data/TableOperation.cs create mode 100644 src/WixToolset.Data/WindowsInstallerStandard.cs create mode 100644 src/WixToolset.Data/WixCorruptFileException.cs create mode 100644 src/WixToolset.Data/WixDataStrings.Designer.cs create mode 100644 src/WixToolset.Data/WixDataStrings.resx create mode 100644 src/WixToolset.Data/WixException.cs create mode 100644 src/WixToolset.Data/WixInvalidIdtException.cs create mode 100644 src/WixToolset.Data/WixMissingTableDefinitionException.cs create mode 100644 src/WixToolset.Data/WixToolset.Data.csproj create mode 100644 src/WixToolset.Data/WixUnexpectedFileFormatException.cs create mode 100644 src/WixToolset.Data/Xsd/wix.xsd create mode 100644 src/WixToolset.Data/Xsd/wixloc.xsd create mode 100644 src/WixToolset.Data/YesNoAlwaysType.cs create mode 100644 src/WixToolset.Data/YesNoDefaultType.cs create mode 100644 src/WixToolset.Data/YesNoType.cs create mode 100644 src/nuget.config create mode 100644 src/version.json diff --git a/WixToolset.Data.sln b/WixToolset.Data.sln new file mode 100644 index 00000000..b5e4dfc2 --- /dev/null +++ b/WixToolset.Data.sln @@ -0,0 +1,43 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio 15 +VisualStudioVersion = 15.0.26730.0 +MinimumVisualStudioVersion = 10.0.40219.1 +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "WixToolset.Data", "src\WixToolset.Data\WixToolset.Data.csproj", "{73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Any CPU = Debug|Any CPU + Debug|arm = Debug|arm + Debug|x64 = Debug|x64 + Debug|x86 = Debug|x86 + Release|Any CPU = Release|Any CPU + Release|arm = Release|arm + Release|x64 = Release|x64 + Release|x86 = Release|x86 + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Debug|Any CPU.Build.0 = Debug|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Debug|arm.ActiveCfg = Debug|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Debug|arm.Build.0 = Debug|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Debug|x64.ActiveCfg = Debug|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Debug|x64.Build.0 = Debug|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Debug|x86.ActiveCfg = Debug|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Debug|x86.Build.0 = Debug|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Release|Any CPU.ActiveCfg = Release|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Release|Any CPU.Build.0 = Release|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Release|arm.ActiveCfg = Release|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Release|arm.Build.0 = Release|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Release|x64.ActiveCfg = Release|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Release|x64.Build.0 = Release|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Release|x86.ActiveCfg = Release|Any CPU + {73ADBD3A-8FB2-47DB-BC79-9BC61C40F2E0}.Release|x86.Build.0 = Release|Any CPU + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + SolutionGuid = {2C73DAA1-A584-4542-A2C3-951EF6203ED5} + EndGlobalSection +EndGlobal diff --git a/appveyor.yml b/appveyor.yml new file mode 100644 index 00000000..f7476084 --- /dev/null +++ b/appveyor.yml @@ -0,0 +1,24 @@ +image: Visual Studio 2017 Preview + +version: 0.0.0.{build} +configuration: Release + +environment: + DOTNET_SKIP_FIRST_TIME_EXPERIENCE: true + DOTNET_CLI_TELEMETRY_OPTOUT: 1 + NUGET_XMLDOC_MODE: skip + +build_script: +- dotnet pack + +pull_requests: + do_not_increment_build_number: true + +nuget: + disable_publish_on_pr: true + +skip_tags: true + +artifacts: +- path: build\Release\**\*.nupkg + name: nuget diff --git a/src/Directory.Build.props b/src/Directory.Build.props new file mode 100644 index 00000000..0f9c550d --- /dev/null +++ b/src/Directory.Build.props @@ -0,0 +1,12 @@ + + + + Debug + $(MSBuildThisFileDirectory)..\build\obj\$(MSBuildProjectName)\ + $(MSBuildThisFileDirectory)..\build\$(Configuration)\ + + Rob Mensching, Bob Arnson + WiX Toolset + Copyright (c) .NET Foundation and contributors. All rights reserved. + + diff --git a/src/WixToolset.Data/AccessModifier.cs b/src/WixToolset.Data/AccessModifier.cs new file mode 100644 index 00000000..64f29f26 --- /dev/null +++ b/src/WixToolset.Data/AccessModifier.cs @@ -0,0 +1,27 @@ +// 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.Data +{ + public enum AccessModifier + { + /// + /// Indicates the identifier is publicly visible to all other sections. + /// + Public, + + /// + /// Indicates the identifier is visible only to sections in the same library. + /// + Internal, + + /// + /// Indicates the identifier is visible only to sections in the same source file. + /// + Protected, + + /// + /// Indicates the identifiers is visible only to the section where it is defined. + /// + Private, + } +} diff --git a/src/WixToolset.Data/AssemblyInfo.cs b/src/WixToolset.Data/AssemblyInfo.cs new file mode 100644 index 00000000..74fd6f90 --- /dev/null +++ b/src/WixToolset.Data/AssemblyInfo.cs @@ -0,0 +1,11 @@ +// 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. + +using System; +using System.Reflection; +using System.Runtime.InteropServices; + +//[assembly: AssemblyTitle("WiX Toolset Data")] +//[assembly: AssemblyDescription("")] +[assembly: AssemblyCulture("")] +[assembly:CLSCompliant(true)] +[assembly: ComVisible(false)] diff --git a/src/WixToolset.Data/BundleApprovedExeForElevationAttributes.cs b/src/WixToolset.Data/BundleApprovedExeForElevationAttributes.cs new file mode 100644 index 00000000..240c2160 --- /dev/null +++ b/src/WixToolset.Data/BundleApprovedExeForElevationAttributes.cs @@ -0,0 +1,16 @@ +// 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.Data +{ + using System; + + /// + /// Attributes available for an ApprovedExeForElevation. + /// + [Flags] + public enum BundleApprovedExeForElevationAttributes : int + { + None = 0x0, + Win64 = 0x1, + } +} diff --git a/src/WixToolset.Data/ColumnDefinition.cs b/src/WixToolset.Data/ColumnDefinition.cs new file mode 100644 index 00000000..7e5a07c5 --- /dev/null +++ b/src/WixToolset.Data/ColumnDefinition.cs @@ -0,0 +1,1032 @@ +// 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.Data +{ + using System; + using System.Globalization; + using System.Xml; + + /// + /// Defines MSI column types. + /// + public enum ColumnType + { + /// Unknown column type, default and invalid. + Unknown, + + /// Column is a string. + String, + + /// Column is a localizable string. + Localized, + + /// Column is a number. + Number, + + /// Column is a binary stream. + Object, + + /// Column is a string that is preserved in transforms (like Object). + Preserved, + } + + /// + /// Specifies if the column should be modularized. + /// + public enum ColumnModularizeType + { + /// Column should not be modularized. + None, + + /// Column should be modularized. + Column, + + /// When the column is an primary or foreign key to the Icon table it should be modularized special. + Icon, + + /// When the column is a companion file it should be modularized. + CompanionFile, + + /// Column is a condition and should be modularized. + Condition, + + /// Special modularization type for the ControlEvent table's Argument column. + ControlEventArgument, + + /// Special modularization type for the Control table's Text column. + ControlText, + + /// Any Properties in the column should be modularized. + Property, + + /// Semi-colon list of keys, all of which need to be modularized. + SemicolonDelimited, + } + + /// + /// Column validation category type + /// + public enum ColumnCategory + { + /// Unknown category, default and invalid. + Unknown, + + /// Text category. + Text, + + /// UpperCase category. + UpperCase, + + /// LowerCase category. + LowerCase, + + /// Integer category. + Integer, + + /// DoubleInteger category. + DoubleInteger, + + /// TimeDate category. + TimeDate, + + /// Identifier category. + Identifier, + + /// Property category. + Property, + + /// Filename category. + Filename, + + /// WildCardFilename category. + WildCardFilename, + + /// Path category. + Path, + + /// Paths category. + Paths, + + /// AnyPath category. + AnyPath, + + /// DefaultDir category. + DefaultDir, + + /// RegPath category. + RegPath, + + /// Formatted category. + Formatted, + + /// Template category. + Template, + + /// Condition category. + Condition, + + /// Guid category. + Guid, + + /// Version category. + Version, + + /// Language category. + Language, + + /// Binary category. + Binary, + + /// CustomSource category. + CustomSource, + + /// Cabinet category. + Cabinet, + + /// Shortcut category. + Shortcut, + + /// Formatted SDDL category. + FormattedSDDLText, + } + + /// + /// Definition of a table's column. + /// + public sealed class ColumnDefinition : IComparable + { + private string name; + private ColumnType type; + private int length; + private bool primaryKey; + private bool nullable; + private ColumnModularizeType modularize; + private bool localizable; + private bool added; + + private bool minValueSet; + private long minValue; + private bool maxValueSet; + private long maxValue; + private string keyTable; + private bool keyColumnSet; + private int keyColumn; + private ColumnCategory category; + private string possibilities; + private string description; + private bool escapeIdtCharacters; + private bool useCData; + + /// + /// Creates a new column definition. + /// + /// Name of column. + /// Type of column + /// Length of column. + /// If column is primary key. + /// If column is nullable. + /// Type of modularization for column + /// If the column is localizable. + /// If the minimum of the value was set. + /// Minimum value for the column. + /// If the maximum value was set. + /// Maximum value for the colum. + /// Optional name of table for foreign key. + /// If the key column was set. + /// Optional name of column for foreign key. + /// Validation category for column. + /// Set of possible values for column. + /// Description of column in vaidation table. + /// If characters should be escaped in IDT. + /// If whitespace should be preserved in a CDATA node. + public ColumnDefinition(string name, ColumnType type, int length, bool primaryKey, bool nullable, ColumnModularizeType modularizeType, bool localizable, bool minValueSet, long minValue, bool maxValueSet, long maxValue, string keyTable, bool keyColumnSet, int keyColumn, ColumnCategory category, string possibilities, string description, bool escapeIdtCharacters, bool useCData) + { + this.name = name; + this.type = type; + this.length = length; + this.primaryKey = primaryKey; + this.nullable = nullable; + this.modularize = modularizeType; + this.localizable = localizable; + this.minValueSet = minValueSet; + this.minValue = minValue; + this.maxValueSet = maxValueSet; + this.maxValue = maxValue; + this.keyTable = keyTable; + this.keyColumnSet = keyColumnSet; + this.keyColumn = keyColumn; + this.category = category; + this.possibilities = possibilities; + this.description = description; + this.escapeIdtCharacters = escapeIdtCharacters; + this.useCData = useCData; + } + + /// + /// Gets whether this column was added via a transform. + /// + /// Whether this column was added via a transform. + public bool Added + { + get { return this.added; } + set { this.added = value; } + } + + /// + /// Gets the name of the column. + /// + /// Name of column. + public string Name + { + get { return this.name; } + } + + /// + /// Gets the type of the column. + /// + /// Type of column. + public ColumnType Type + { + get { return this.type; } + } + + /// + /// Gets the length of the column. + /// + /// Length of column. + public int Length + { + get { return this.length; } + } + + /// + /// Gets if the column is a primary key. + /// + /// true if column is primary key. + public bool PrimaryKey + { + get { return this.primaryKey; } + } + + /// + /// Gets if the column is nullable. + /// + /// true if column is nullable. + public bool Nullable + { + get { return this.nullable; } + } + + /// + /// Gets the type of modularization for this column. + /// + /// Column's modularization type. + public ColumnModularizeType ModularizeType + { + get { return this.modularize; } + } + + /// + /// Gets if the column is localizable. Can be because the type is localizable, or because the column + /// was explicitly set to be so. + /// + /// true if column is localizable. + public bool IsLocalizable + { + get { return this.localizable || ColumnType.Localized == this.Type; } + } + + /// + /// Gets if the minimum value of the column is set. + /// + /// true if minimum value is set. + public bool IsMinValueSet + { + get { return this.minValueSet; } + } + + /// + /// Gets the minimum value for the column, only valid if IsMinValueSet returns true. + /// + /// Minimum value for the column. + public long MinValue + { + get { return this.minValue; } + } + + /// + /// Gets if the maximum value of the column is set. + /// + /// true if maximum value is set. + public bool IsMaxValueSet + { + get { return this.maxValueSet; } + } + + /// + /// Gets the maximum value for the column, only valid if IsMinValueSet returns true. + /// + /// Maximum value for the column. + public long MaxValue + { + get { return this.maxValue; } + } + + /// + /// Gets the table that has the foreign key for this column + /// + /// Foreign key table name. + public string KeyTable + { + get { return this.keyTable; } + } + + /// + /// Gets if the key column is set. + /// + /// True if the key column is set. + public bool IsKeyColumnSet + { + get { return this.keyColumnSet; } + } + + /// + /// Gets the foreign key column that this column refers to. + /// + /// Foreign key column. + public int KeyColumn + { + get { return this.keyColumn; } + } + + /// + /// Gets the validation category for this column. + /// + /// Validation category. + public ColumnCategory Category + { + get { return this.category; } + } + + /// + /// Gets the set of possibilities for this column. + /// + /// Set of possibilities for this column. + public string Possibilities + { + get { return this.possibilities; } + } + + /// + /// Gets the description for this column. + /// + /// Description of column. + public string Description + { + get { return this.description; } + } + + /// + /// Gets if characters should be escaped to fit into IDT. + /// + /// true if data should be escaped when adding to IDT. + public bool EscapeIdtCharacters + { + get { return this.escapeIdtCharacters; } + } + + /// + /// Gets if whitespace should be preserved in a CDATA node. + /// + /// true if whitespace should be preserved in a CDATA node. + public bool UseCData + { + get { return this.useCData; } + } + + /// + /// Gets the type of the column in IDT format. + /// + /// IDT format for column type. + public string IdtType + { + get + { + char typeCharacter; + switch (this.type) + { + case ColumnType.Number: + typeCharacter = this.nullable ? 'I' : 'i'; + break; + case ColumnType.Preserved: + case ColumnType.String: + typeCharacter = this.nullable ? 'S' : 's'; + break; + case ColumnType.Localized: + typeCharacter = this.nullable ? 'L' : 'l'; + break; + case ColumnType.Object: + typeCharacter = this.nullable ? 'V' : 'v'; + break; + default: + throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, WixDataStrings.EXP_UnknownColumnType, this.type)); + } + + return String.Concat(typeCharacter, this.length); + } + } + + /// + /// Parses a column definition in a table definition. + /// + /// Reader to get data from. + /// The ColumnDefintion represented by the Xml. + internal static ColumnDefinition Read(XmlReader reader) + { + if (!reader.LocalName.Equals("columnDefinition")) + { + throw new XmlException(); + } + + bool added = false; + ColumnCategory category = ColumnCategory.Unknown; + string description = null; + bool empty = reader.IsEmptyElement; + bool escapeIdtCharacters = false; + int keyColumn = -1; + bool keyColumnSet = false; + string keyTable = null; + int length = -1; + bool localizable = false; + long maxValue = 0; + bool maxValueSet = false; + long minValue = 0; + bool minValueSet = false; + ColumnModularizeType modularize = ColumnModularizeType.None; + string name = null; + bool nullable = false; + string possibilities = null; + bool primaryKey = false; + ColumnType type = ColumnType.Unknown; + bool useCData = false; + + // parse the attributes + while (reader.MoveToNextAttribute()) + { + switch (reader.LocalName) + { + case "added": + added = reader.Value.Equals("yes"); + break; + case "category": + switch (reader.Value) + { + case "anyPath": + category = ColumnCategory.AnyPath; + break; + case "binary": + category = ColumnCategory.Binary; + break; + case "cabinet": + category = ColumnCategory.Cabinet; + break; + case "condition": + category = ColumnCategory.Condition; + break; + case "customSource": + category = ColumnCategory.CustomSource; + break; + case "defaultDir": + category = ColumnCategory.DefaultDir; + break; + case "doubleInteger": + category = ColumnCategory.DoubleInteger; + break; + case "filename": + category = ColumnCategory.Filename; + break; + case "formatted": + category = ColumnCategory.Formatted; + break; + case "formattedSddl": + category = ColumnCategory.FormattedSDDLText; + break; + case "guid": + category = ColumnCategory.Guid; + break; + case "identifier": + category = ColumnCategory.Identifier; + break; + case "integer": + category = ColumnCategory.Integer; + break; + case "language": + category = ColumnCategory.Language; + break; + case "lowerCase": + category = ColumnCategory.LowerCase; + break; + case "path": + category = ColumnCategory.Path; + break; + case "paths": + category = ColumnCategory.Paths; + break; + case "property": + category = ColumnCategory.Property; + break; + case "regPath": + category = ColumnCategory.RegPath; + break; + case "shortcut": + category = ColumnCategory.Shortcut; + break; + case "template": + category = ColumnCategory.Template; + break; + case "text": + category = ColumnCategory.Text; + break; + case "timeDate": + category = ColumnCategory.TimeDate; + break; + case "upperCase": + category = ColumnCategory.UpperCase; + break; + case "version": + category = ColumnCategory.Version; + break; + case "wildCardFilename": + category = ColumnCategory.WildCardFilename; + break; + default: + throw new InvalidOperationException(); + } + break; + case "description": + description = reader.Value; + break; + case "escapeIdtCharacters": + escapeIdtCharacters = reader.Value.Equals("yes"); + break; + case "keyColumn": + keyColumnSet = true; + keyColumn = Convert.ToInt32(reader.Value, 10); + break; + case "keyTable": + keyTable = reader.Value; + break; + case "length": + length = Convert.ToInt32(reader.Value, 10); + break; + case "localizable": + localizable = reader.Value.Equals("yes"); + break; + case "maxValue": + maxValueSet = true; + maxValue = Convert.ToInt32(reader.Value, 10); + break; + case "minValue": + minValueSet = true; + minValue = Convert.ToInt32(reader.Value, 10); + break; + case "modularize": + switch (reader.Value) + { + case "column": + modularize = ColumnModularizeType.Column; + break; + case "companionFile": + modularize = ColumnModularizeType.CompanionFile; + break; + case "condition": + modularize = ColumnModularizeType.Condition; + break; + case "controlEventArgument": + modularize = ColumnModularizeType.ControlEventArgument; + break; + case "controlText": + modularize = ColumnModularizeType.ControlText; + break; + case "icon": + modularize = ColumnModularizeType.Icon; + break; + case "none": + modularize = ColumnModularizeType.None; + break; + case "property": + modularize = ColumnModularizeType.Property; + break; + case "semicolonDelimited": + modularize = ColumnModularizeType.SemicolonDelimited; + break; + default: + throw new XmlException(); + } + break; + case "name": + switch (reader.Value) + { + case "CREATE": + case "DELETE": + case "DROP": + case "INSERT": + throw new XmlException(); + default: + name = reader.Value; + break; + } + break; + case "nullable": + nullable = reader.Value.Equals("yes"); + break; + case "primaryKey": + primaryKey = reader.Value.Equals("yes"); + break; + case "set": + possibilities = reader.Value; + break; + case "type": + switch (reader.Value) + { + case "localized": + type = ColumnType.Localized; + break; + case "number": + type = ColumnType.Number; + break; + case "object": + type = ColumnType.Object; + break; + case "string": + type = ColumnType.String; + break; + case "preserved": + type = ColumnType.Preserved; + break; + default: + throw new XmlException(); + } + break; + case "useCData": + useCData = reader.Value.Equals("yes"); + break; + } + } + + // parse the child elements (there should be none) + if (!empty) + { + bool done = false; + + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + throw new XmlException(); + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + + ColumnDefinition columnDefinition = new ColumnDefinition(name, type, length, primaryKey, nullable, modularize, localizable, minValueSet, minValue, maxValueSet, maxValue, keyTable, keyColumnSet, keyColumn, category, possibilities, description, escapeIdtCharacters, useCData); + columnDefinition.Added = added; + + return columnDefinition; + } + + /// + /// Persists a ColumnDefinition in an XML format. + /// + /// XmlWriter where the Output should persist itself as XML. + internal void Write(XmlWriter writer) + { + writer.WriteStartElement("columnDefinition", TableDefinitionCollection.XmlNamespaceUri); + + writer.WriteAttributeString("name", this.name); + + switch (this.type) + { + case ColumnType.Localized: + writer.WriteAttributeString("type", "localized"); + break; + case ColumnType.Number: + writer.WriteAttributeString("type", "number"); + break; + case ColumnType.Object: + writer.WriteAttributeString("type", "object"); + break; + case ColumnType.String: + writer.WriteAttributeString("type", "string"); + break; + case ColumnType.Preserved: + writer.WriteAttributeString("type", "preserved"); + break; + } + + writer.WriteAttributeString("length", this.length.ToString(CultureInfo.InvariantCulture.NumberFormat)); + + if (this.primaryKey) + { + writer.WriteAttributeString("primaryKey", "yes"); + } + + if (this.nullable) + { + writer.WriteAttributeString("nullable", "yes"); + } + + if (this.localizable) + { + writer.WriteAttributeString("localizable", "yes"); + } + + if (this.added) + { + writer.WriteAttributeString("added", "yes"); + } + + switch (this.modularize) + { + case ColumnModularizeType.Column: + writer.WriteAttributeString("modularize", "column"); + break; + case ColumnModularizeType.CompanionFile: + writer.WriteAttributeString("modularize", "companionFile"); + break; + case ColumnModularizeType.Condition: + writer.WriteAttributeString("modularize", "condition"); + break; + case ColumnModularizeType.ControlEventArgument: + writer.WriteAttributeString("modularize", "controlEventArgument"); + break; + case ColumnModularizeType.ControlText: + writer.WriteAttributeString("modularize", "controlText"); + break; + case ColumnModularizeType.Icon: + writer.WriteAttributeString("modularize", "icon"); + break; + case ColumnModularizeType.None: + // this is the default value + break; + case ColumnModularizeType.Property: + writer.WriteAttributeString("modularize", "property"); + break; + case ColumnModularizeType.SemicolonDelimited: + writer.WriteAttributeString("modularize", "semicolonDelimited"); + break; + } + + if (this.minValueSet) + { + writer.WriteAttributeString("minValue", this.minValue.ToString(CultureInfo.InvariantCulture.NumberFormat)); + } + + if (this.maxValueSet) + { + writer.WriteAttributeString("maxValue", this.maxValue.ToString(CultureInfo.InvariantCulture.NumberFormat)); + } + + if (!String.IsNullOrEmpty(this.keyTable)) + { + writer.WriteAttributeString("keyTable", this.keyTable); + } + + if (this.keyColumnSet) + { + writer.WriteAttributeString("keyColumn", this.keyColumn.ToString(CultureInfo.InvariantCulture.NumberFormat)); + } + + switch (this.category) + { + case ColumnCategory.AnyPath: + writer.WriteAttributeString("category", "anyPath"); + break; + case ColumnCategory.Binary: + writer.WriteAttributeString("category", "binary"); + break; + case ColumnCategory.Cabinet: + writer.WriteAttributeString("category", "cabinet"); + break; + case ColumnCategory.Condition: + writer.WriteAttributeString("category", "condition"); + break; + case ColumnCategory.CustomSource: + writer.WriteAttributeString("category", "customSource"); + break; + case ColumnCategory.DefaultDir: + writer.WriteAttributeString("category", "defaultDir"); + break; + case ColumnCategory.DoubleInteger: + writer.WriteAttributeString("category", "doubleInteger"); + break; + case ColumnCategory.Filename: + writer.WriteAttributeString("category", "filename"); + break; + case ColumnCategory.Formatted: + writer.WriteAttributeString("category", "formatted"); + break; + case ColumnCategory.FormattedSDDLText: + writer.WriteAttributeString("category", "formattedSddl"); + break; + case ColumnCategory.Guid: + writer.WriteAttributeString("category", "guid"); + break; + case ColumnCategory.Identifier: + writer.WriteAttributeString("category", "identifier"); + break; + case ColumnCategory.Integer: + writer.WriteAttributeString("category", "integer"); + break; + case ColumnCategory.Language: + writer.WriteAttributeString("category", "language"); + break; + case ColumnCategory.LowerCase: + writer.WriteAttributeString("category", "lowerCase"); + break; + case ColumnCategory.Path: + writer.WriteAttributeString("category", "path"); + break; + case ColumnCategory.Paths: + writer.WriteAttributeString("category", "paths"); + break; + case ColumnCategory.Property: + writer.WriteAttributeString("category", "property"); + break; + case ColumnCategory.RegPath: + writer.WriteAttributeString("category", "regPath"); + break; + case ColumnCategory.Shortcut: + writer.WriteAttributeString("category", "shortcut"); + break; + case ColumnCategory.Template: + writer.WriteAttributeString("category", "template"); + break; + case ColumnCategory.Text: + writer.WriteAttributeString("category", "text"); + break; + case ColumnCategory.TimeDate: + writer.WriteAttributeString("category", "timeDate"); + break; + case ColumnCategory.UpperCase: + writer.WriteAttributeString("category", "upperCase"); + break; + case ColumnCategory.Version: + writer.WriteAttributeString("category", "version"); + break; + case ColumnCategory.WildCardFilename: + writer.WriteAttributeString("category", "wildCardFilename"); + break; + } + + if (!String.IsNullOrEmpty(this.possibilities)) + { + writer.WriteAttributeString("set", this.possibilities); + } + + if (!String.IsNullOrEmpty(this.description)) + { + writer.WriteAttributeString("description", this.description); + } + + if (this.escapeIdtCharacters) + { + writer.WriteAttributeString("escapeIdtCharacters", "yes"); + } + + if (this.useCData) + { + writer.WriteAttributeString("useCData", "yes"); + } + + writer.WriteEndElement(); + } + + /// + /// Validate a value for this column. + /// + /// The value to validate. + /// Validated value. + internal object ValidateValue(object value) + { + if (null == value) + { + if (!this.nullable) + { + throw new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, "Cannot set column '{0}' with a null value because this is a required field.", this.name)); + } + } + else // check numerical values against their specified minimum and maximum values. + { + if (ColumnType.Number == this.type && !this.IsLocalizable) + { + // For now all enums in the tables can be represented by integers. This if statement would need to + // be enhanced if that ever changes. + if (value is int || value.GetType().IsEnum) + { + int intValue = (int)value; + + // validate the value against the minimum allowed value + if (this.minValueSet && this.minValue > intValue) + { + throw new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, "Cannot set column '{0}' with value {1} because it is less than the minimum allowed value for this column, {2}.", this.name, intValue, this.minValue)); + } + + // validate the value against the maximum allowed value + if (this.maxValueSet && this.maxValue < intValue) + { + throw new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, "Cannot set column '{0}' with value {1} because it is greater than the maximum allowed value for this column, {2}.", this.name, intValue, this.maxValue)); + } + + return intValue; + } + else if (value is long) + { + long longValue = (long)value; + + // validate the value against the minimum allowed value + if (this.minValueSet && this.minValue > longValue) + { + throw new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, "Cannot set column '{0}' with value {1} because it is less than the minimum allowed value for this column, {2}.", this.name, longValue, this.minValue)); + } + + // validate the value against the maximum allowed value + if (this.maxValueSet && this.maxValue < longValue) + { + throw new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, "Cannot set column '{0}' with value {1} because it is greater than the maximum allowed value for this column, {2}.", this.name, longValue, this.maxValue)); + } + + return longValue; + } + else + { + throw new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, "Cannot set number column '{0}' with a value of type '{1}'.", this.name, value.GetType().ToString())); + } + } + else + { + if (!(value is string)) + { + throw new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, "Cannot set string column '{0}' with a value of type '{1}'.", this.name, value.GetType().ToString())); + } + } + } + + return value; + } + + /// + /// Compare this column definition to another column definition. + /// + /// + /// Only Windows Installer traits are compared, allowing for updates to WiX-specific table definitions. + /// + /// The to compare with this one. + /// 0 if the columns' core propeties are the same; otherwise, non-0. + public int CompareTo(ColumnDefinition other) + { + // by definition, this object is greater than null + if (null == other) + { + return 1; + } + + // compare column names + int ret = String.Compare(this.Name, other.Name, StringComparison.Ordinal); + + // compare column types + if (0 == ret) + { + ret = this.Type == other.Type ? 0 : -1; + + // compare column lengths + if (0 == ret) + { + ret = this.Length == other.Length ? 0 : -1; + + // compare whether both are primary keys + if (0 == ret) + { + ret = this.PrimaryKey == other.PrimaryKey ? 0 : -1; + + // compare nullability + if (0 == ret) + { + ret = this.Nullable == other.Nullable ? 0 : -1; + } + } + } + } + + return ret; + } + } +} diff --git a/src/WixToolset.Data/Common.cs b/src/WixToolset.Data/Common.cs new file mode 100644 index 00000000..f01b4591 --- /dev/null +++ b/src/WixToolset.Data/Common.cs @@ -0,0 +1,53 @@ +// 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.Data +{ + using System; + using System.IO; + using System.Security.Cryptography; + using System.Text; + using System.Text.RegularExpressions; + using System.Xml.Linq; + + internal static class Common + { + public const int IntegerNotSet = int.MinValue; + + internal static readonly XNamespace W3SchemaPrefix = "http://www.w3.org/"; + + private static readonly Regex LegalIdentifierCharacters = new Regex(@"^[_A-Za-z][0-9A-Za-z_\.]*$", RegexOptions.Compiled); + + internal static string GetFileHash(FileInfo fileInfo) + { + byte[] hashBytes; + using (SHA1Managed managed = new SHA1Managed()) + { + using (FileStream stream = fileInfo.OpenRead()) + { + hashBytes = managed.ComputeHash(stream); + } + } + + StringBuilder sb = new StringBuilder(); + for (int i = 0; i < hashBytes.Length; i++) + { + sb.AppendFormat("{0:X2}", hashBytes[i]); + } + + return sb.ToString(); + } + + public static bool IsIdentifier(string value) + { + if (!String.IsNullOrEmpty(value)) + { + if (LegalIdentifierCharacters.IsMatch(value)) + { + return true; + } + } + + return false; + } + } +} diff --git a/src/WixToolset.Data/ComplexReferenceChildType.cs b/src/WixToolset.Data/ComplexReferenceChildType.cs new file mode 100644 index 00000000..0a355afe --- /dev/null +++ b/src/WixToolset.Data/ComplexReferenceChildType.cs @@ -0,0 +1,46 @@ +// 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.Data +{ + /// + /// Types of children in complex refernece. + /// + public enum ComplexReferenceChildType + { + /// Unknown complex reference type, default and invalid. + Unknown, + + /// Component child of complex reference. + Component, + + /// Feature child of complex reference. + Feature, + + /// ComponentGroup child of complex reference. + ComponentGroup, + + /// FeatureGroup child of complex reference. + FeatureGroup, + + /// Module child of complex reference. + Module, + + /// Payload child of complex reference. + Payload, + + /// PayloadGroup child of complex reference. + PayloadGroup, + + /// Package child of complex reference. + Package, + + /// PackageGroup child of complex reference. + PackageGroup, + + /// PatchFamily child of complex reference. + PatchFamily, + + /// PatchFamilyGroup child of complex reference. + PatchFamilyGroup, + } +} diff --git a/src/WixToolset.Data/ComplexReferenceParentType.cs b/src/WixToolset.Data/ComplexReferenceParentType.cs new file mode 100644 index 00000000..87731b97 --- /dev/null +++ b/src/WixToolset.Data/ComplexReferenceParentType.cs @@ -0,0 +1,49 @@ +// 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.Data +{ + /// + /// Types of parents in complex reference. + /// + public enum ComplexReferenceParentType + { + /// Unknown complex reference type, default and invalid. + Unknown, + + /// Feature parent of complex reference. + Feature, + + /// ComponentGroup parent of complex reference. + ComponentGroup, + + /// FeatureGroup parent of complex reference. + FeatureGroup, + + /// Module parent of complex reference. + Module, + + /// Product parent of complex reference. + Product, + + /// PayloadGroup parent of complex reference. + PayloadGroup, + + /// Package parent of complex reference. + Package, + + /// PackageGroup parent of complex reference. + PackageGroup, + + /// Container parent of complex reference. + Container, + + /// Layout parent of complex reference. + Layout, + + /// Patch parent of complex reference. + Patch, + + /// PatchFamilyGroup parent of complex reference. + PatchFamilyGroup, + } +} diff --git a/src/WixToolset.Data/CompressionLevel.cs b/src/WixToolset.Data/CompressionLevel.cs new file mode 100644 index 00000000..13242051 --- /dev/null +++ b/src/WixToolset.Data/CompressionLevel.cs @@ -0,0 +1,26 @@ +// 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.Data +{ + + /// + /// Compression level to use when creating cabinet. + /// + public enum CompressionLevel + { + /// Use no compression. + None, + + /// Use low compression. + Low, + + /// Use medium compression. + Medium, + + /// Use high compression. + High, + + /// Use ms-zip compression. + Mszip + } +} diff --git a/src/WixToolset.Data/Data/Xsd/actions.xsd b/src/WixToolset.Data/Data/Xsd/actions.xsd new file mode 100644 index 00000000..bf0ccb95 --- /dev/null +++ b/src/WixToolset.Data/Data/Xsd/actions.xsd @@ -0,0 +1,73 @@ + + + + + + + + Schema for describing standard actions in the Windows Installer. + + + + + + + + + + + + + + + + Name of action + + + + + Default condition for action + + + + + Sequence of action + + + + + Specifies if action is allowed in AdminExecuteSequence + + + + + Specifies if action is allowed in AdminUISequence + + + + + Specifies if action is allowed in AdvtExecuteSequence + + + + + Specifies if action is allowed in InstallExecuteSequence + + + + + Specifies if action is allowed in InstallUISequence + + + + + + + + + + + + diff --git a/src/WixToolset.Data/Data/Xsd/libraries.xsd b/src/WixToolset.Data/Data/Xsd/libraries.xsd new file mode 100644 index 00000000..a4504c01 --- /dev/null +++ b/src/WixToolset.Data/Data/Xsd/libraries.xsd @@ -0,0 +1,66 @@ + + + + + + + + Schema for describing WiX Library files (.wixlib). + + + + + + + + + + + + + + + + Version of WiX used to create this library file + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/WixToolset.Data/Data/Xsd/objects.xsd b/src/WixToolset.Data/Data/Xsd/objects.xsd new file mode 100644 index 00000000..5d95a59c --- /dev/null +++ b/src/WixToolset.Data/Data/Xsd/objects.xsd @@ -0,0 +1,143 @@ + + + + + + + + Schema for describing WiX Object files (.wixobj). + + + + + + + + + + + + Version of WiX used to create this object file. + + + + + + + + + + + + + Identifier for section (optional for Fragments) + + + + + Type of section + + + + + Codepage for output file, only valid on entry sections. + + + + + + + + + + + + + Name of table in Windows Installer database + + + + + + + + + + + + + + + + Row in a table + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Data for a particular field in a row. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/WixToolset.Data/Data/Xsd/outputs.xsd b/src/WixToolset.Data/Data/Xsd/outputs.xsd new file mode 100644 index 00000000..00e20f12 --- /dev/null +++ b/src/WixToolset.Data/Data/Xsd/outputs.xsd @@ -0,0 +1,66 @@ + + + + + + + + Schema for describing WiX Output files (.wixout). + + + + + + + + + + + + + + + + Codepage of the output. + + + + + Type of the output. + + + + + + + + + + + + + + + Version of WiX used to create this output file. + + + + + + + + + + + + + Name of the substorage. + + + + + diff --git a/src/WixToolset.Data/Data/Xsd/pdbs.xsd b/src/WixToolset.Data/Data/Xsd/pdbs.xsd new file mode 100644 index 00000000..c1d1756d --- /dev/null +++ b/src/WixToolset.Data/Data/Xsd/pdbs.xsd @@ -0,0 +1,32 @@ + + + + + + + + Schema for describing WiX Pdb files (.wixpdb). + + + + + + + + + + + + + + + + Version of WiX used to create this pdb file + + + + + diff --git a/src/WixToolset.Data/Data/Xsd/tables.xsd b/src/WixToolset.Data/Data/Xsd/tables.xsd new file mode 100644 index 00000000..f87471bb --- /dev/null +++ b/src/WixToolset.Data/Data/Xsd/tables.xsd @@ -0,0 +1,248 @@ + + + + + + + + Schema for describing table definitions in Windows Installer. + + + + + + + + + + + + + + + + + + + Boolean whether rows in this table create symbols + + + + + Name of table in Windows Installer database + + + + + Specifies if table is virtual or not + + + + + Specifies if the table is a part of the Bootstrapper Application Data manifest + + + + + + + + + + Name of column in Windows Installer table + + + + + + Whether this column was added by a transform. + + + + + + Type of column in Windows Installer table + + + + + + Type of column in Windows Installer table + + + + + + + + + + + + Boolean whether column is primary key of Windows Installer table + + + + + + Boolean whether column is nullable in Windows Installer table + + + + + + Boolean whether column is virtual in Windows Installer table + + + + + + Enumeration specifying how column should have the ModuleId appended + + + + + + Set to "yes" in order to allow substitution for localized variables. + + + + + + Minimum value for column in Windows Installer table + + + + + + Maximum value for column in Windows Installer table + + + + + + Foreign key table for column in Windows Installer table + + + + + + Maximum value for column in Windows Installer table + + + + + + + + + + + + Specific column data types for column + + + + + + List of permissible values for the column + + + + + + Description of column + + + + + + Set to "yes" in order to make the idt exporter escape whitespace characters \r, \n, and \t. + + + + + + Set to "yes" in order to make the Intermediate and Output objects wrap their data in a CDATA element to preserve whitespace. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/WixToolset.Data/Data/actions.xml b/src/WixToolset.Data/Data/actions.xml new file mode 100644 index 00000000..f65b792d --- /dev/null +++ b/src/WixToolset.Data/Data/actions.xml @@ -0,0 +1,76 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/WixToolset.Data/Data/messages.xml b/src/WixToolset.Data/Data/messages.xml new file mode 100644 index 00000000..191e0b3e --- /dev/null +++ b/src/WixToolset.Data/Data/messages.xml @@ -0,0 +1,95 @@ + + + + + + + + + Unexpected file format loaded from path: {0}. The file was expected to be a {1} but was actually: {2}. Ensure the correct path was provided. + + + + + + + + Attempted to load corrupt file from path: {0}. The file with format {1} contained unexpected content. Ensure the correct path was provided and that the file has not been incorrectly modified. + + + + + + + The localization identifier '{0}' has been duplicated in multiple locations. Please resolve the conflict. + + + + + + The system cannot find the file '{0}'. + + + + The system cannot find the file '{0}' with type '{1}'. + + + + + + + The primary key '{0}' is duplicated in table '{1}'. Please remove one of the entries or rename a part of the primary key to avoid the collision. + + + + + + + There was an error importing the file '{0}'. + + + + There was an error importing table '{1}' from file '{0}'. + + + + + + + The {0} file format version {1} is not compatible with the expected {0} file format version {2}. + + + + + + + Cannot have both the MsidbFileAttributesCompressed and MsidbFileAttributesNoncompressed options set in a file attributes column. + + + + Cannot find the table definitions for the '{0}' table. This is likely due to a typing error or missing extension. Please ensure all the necessary extensions are supplied on the command line with the -ext parameter. + + + + + + The table '{0}' does not contain any primary key columns. At least one column must be marked as the primary key to ensure this table can be patched. + + + + + + A string was provided with characters that are not available in the specified database code page '{0}'. Either change these characters to ones that exist in the database's code page, or update the database's code page by modifying one of the following attributes: Product/@Codepage, Module/@Codepage, Patch/@Codepage, PatchCreation/@Codepage, or WixLocalization/@Codepage. + + + + + + The table '{0}' contains {1} columns which is not supported by Windows Installer. Windows Installer supports a maximum of {2} columns. + + + + + + + diff --git a/src/WixToolset.Data/Data/tables.xml b/src/WixToolset.Data/Data/tables.xml new file mode 100644 index 00000000..280d87a8 --- /dev/null +++ b/src/WixToolset.Data/Data/tables.xml @@ -0,0 +1,1962 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/WixToolset.Data/DisplayEventArgs.cs b/src/WixToolset.Data/DisplayEventArgs.cs new file mode 100644 index 00000000..baa9b0e2 --- /dev/null +++ b/src/WixToolset.Data/DisplayEventArgs.cs @@ -0,0 +1,15 @@ +// 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.Data +{ + using System; + + public delegate void DisplayEventHandler(object sender, DisplayEventArgs e); + + public class DisplayEventArgs : EventArgs + { + public MessageLevel Level { get; set; } + + public string Message { get; set; } + } +} diff --git a/src/WixToolset.Data/DuplicateSymbolsException.cs b/src/WixToolset.Data/DuplicateSymbolsException.cs new file mode 100644 index 00000000..d0f8536c --- /dev/null +++ b/src/WixToolset.Data/DuplicateSymbolsException.cs @@ -0,0 +1,35 @@ +// 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.Data +{ + using System; + using System.Collections; + + /// + /// Duplicate symbols exception. + /// + [Serializable] + public sealed class DuplicateSymbolsException : Exception + { + [NonSerialized] + private Symbol[] duplicateSymbols; + + /// + /// Instantiate a new DuplicateSymbolException. + /// + /// The duplicated symbols. + public DuplicateSymbolsException(ArrayList symbols) + { + this.duplicateSymbols = (Symbol[])symbols.ToArray(typeof(Symbol)); + } + + /// + /// Gets the duplicate symbols. + /// + /// List of duplicate symbols. + public Symbol[] GetDuplicateSymbols() + { + return this.duplicateSymbols; + } + } +} diff --git a/src/WixToolset.Data/EmptyRule.cs b/src/WixToolset.Data/EmptyRule.cs new file mode 100644 index 00000000..0170e457 --- /dev/null +++ b/src/WixToolset.Data/EmptyRule.cs @@ -0,0 +1,22 @@ +// 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.Data +{ + public enum EmptyRule + { + /// + /// The trimmed value cannot be empty. + /// + MustHaveNonWhitespaceCharacters, + + /// + /// The trimmed value can be empty, but the value itself cannot be empty. + /// + CanBeWhitespaceOnly, + + /// + /// The value can be empty. + /// + CanBeEmpty + } +} diff --git a/src/WixToolset.Data/Field.cs b/src/WixToolset.Data/Field.cs new file mode 100644 index 00000000..74b78229 --- /dev/null +++ b/src/WixToolset.Data/Field.cs @@ -0,0 +1,266 @@ +// 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.Data +{ + using System; + using System.Diagnostics; + using System.Globalization; + using System.Xml; + + /// + /// Field containing data for a column in a row. + /// + public class Field + { + private object data; + + /// + /// Instantiates a new Field. + /// + /// Column definition for this field. + protected Field(ColumnDefinition columnDefinition) + { + this.Column = columnDefinition; + } + + /// + /// Gets or sets the column definition for this field. + /// + /// Column definition. + public ColumnDefinition Column { get; private set; } + + /// + /// Gets or sets the data for this field. + /// + /// Data in the field. + public object Data + { + get + { + return this.data; + } + + set + { + // Validate the value before setting it. + this.data = this.Column.ValidateValue(value); + } + } + + /// + /// Gets or sets whether this field is modified. + /// + /// Whether this field is modified. + public bool Modified { get; set; } + + /// + /// Gets or sets the previous data. + /// + /// The previous data. + public string PreviousData { get; set; } + + /// + /// Instantiate a new Field object of the correct type. + /// + /// The column definition for the field. + /// The new Field object. + public static Field Create(ColumnDefinition columnDefinition) + { + return (ColumnType.Object == columnDefinition.Type) ? new ObjectField(columnDefinition) : new Field(columnDefinition); + } + + /// + /// Sets the value of a particular field in the row without validating. + /// + /// field index. + /// Value of a field in the row. + /// True if successful, false if validation failed. + public bool BestEffortSet(object value) + { + bool success = true; + object bestEffortValue = value; + + try + { + bestEffortValue = this.Column.ValidateValue(value); + } + catch (InvalidOperationException) + { + success = false; + } + + this.data = bestEffortValue; + return success; + } + + /// + /// Determine if this field is identical to another field. + /// + /// The other field to compare to. + /// true if they are equal; false otherwise. + public bool IsIdentical(Field field) + { + return (this.Column.Name == field.Column.Name && + ((null != this.data && this.data.Equals(field.data)) || (null == this.data && null == field.data))); + } + + /// + /// Overrides the built in object implementation to return the field's data as a string. + /// + /// Field's data as a string. + public override string ToString() + { + return this.AsString(); + } + + /// + /// Gets the field as an integer. + /// + /// Field's data as an integer. + public int AsInteger() + { + return (this.data is int) ? (int)this.data : Convert.ToInt32(this.data, CultureInfo.InvariantCulture); + } + + /// + /// Gets the field as an integer that could be null. + /// + /// Field's data as an integer that could be null. + public int? AsNullableInteger() + { + return (null == this.data) ? (int?)null : (this.data is int) ? (int)this.data : Convert.ToInt32(this.data, CultureInfo.InvariantCulture); + } + + /// + /// Gets the field as a string. + /// + /// Field's data as a string. + public string AsString() + { + return (null == this.data) ? null : Convert.ToString(this.data, CultureInfo.InvariantCulture); + } + + /// + /// Parse a field from the xml. + /// + /// XmlReader where the intermediate is persisted. + internal virtual void Read(XmlReader reader) + { + Debug.Assert("field" == reader.LocalName); + + bool empty = reader.IsEmptyElement; + + while (reader.MoveToNextAttribute()) + { + switch (reader.LocalName) + { + case "modified": + this.Modified = reader.Value.Equals("yes"); + break; + case "previousData": + this.PreviousData = reader.Value; + break; + } + } + + if (!empty) + { + bool done = false; + + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + throw new XmlException(); + case XmlNodeType.CDATA: + case XmlNodeType.Text: + case XmlNodeType.SignificantWhitespace: + if (0 < reader.Value.Length) + { + if (ColumnType.Number == this.Column.Type && !this.Column.IsLocalizable) + { + // older wix files could persist data as a long value (which would overflow an int) + // since the Convert class always throws exceptions for overflows, read in integral + // values as a long to avoid the overflow, then cast it to an int (this operation can + // overflow without throwing an exception inside an unchecked block) + this.data = unchecked((int)Convert.ToInt64(reader.Value, CultureInfo.InvariantCulture)); + } + else + { + this.data = reader.Value; + } + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + } + + /// + /// Persists a field in an XML format. + /// + /// XmlWriter where the Field should persist itself as XML. + internal virtual void Write(XmlWriter writer) + { + writer.WriteStartElement("field", Intermediate.XmlNamespaceUri); + + if (this.Modified) + { + writer.WriteAttributeString("modified", "yes"); + } + + if (null != this.PreviousData) + { + writer.WriteAttributeString("previousData", this.PreviousData); + } + + // Convert the data to a string that will persist nicely (nulls as String.Empty). + string text = Convert.ToString(this.data, CultureInfo.InvariantCulture); + if (this.Column.UseCData) + { + writer.WriteCData(text); + } + else + { + writer.WriteString(text); + } + + writer.WriteEndElement(); + } + + /// + /// Returns the field data in a format usable in IDT files. + /// + /// Field data in string IDT format. + internal string ToIdtValue() + { + if (null == this.data) + { + return null; + } + else + { + string fieldData = Convert.ToString(this.data, CultureInfo.InvariantCulture); + + // special idt-specific escaping + if (this.Column.EscapeIdtCharacters) + { + fieldData = fieldData.Replace('\t', '\x10'); + fieldData = fieldData.Replace('\r', '\x11'); + fieldData = fieldData.Replace('\n', '\x19'); + } + + return fieldData; + } + } + } +} diff --git a/src/WixToolset.Data/FileFormat.cs b/src/WixToolset.Data/FileFormat.cs new file mode 100644 index 00000000..9bb54fca --- /dev/null +++ b/src/WixToolset.Data/FileFormat.cs @@ -0,0 +1,17 @@ +// 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.Data +{ + public enum FileFormat + { + Unknown, + + Wixobj, + + Wixlib, + + Wixout, + + Wixpdb, + } +} diff --git a/src/WixToolset.Data/FileStructure.cs b/src/WixToolset.Data/FileStructure.cs new file mode 100644 index 00000000..7265a51d --- /dev/null +++ b/src/WixToolset.Data/FileStructure.cs @@ -0,0 +1,294 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + using System.Diagnostics; + using System.IO; + using System.Linq; + + /// + /// Class that understands the standard file structures in the WiX toolset. + /// + public class FileStructure : IDisposable + { + private long dataStreamOffset; + private long[] embeddedFileSizes; + private Stream stream; + private bool disposed; + + private static readonly Dictionary SupportedFileFormats = new Dictionary() + { + { "wixobj", FileFormat.Wixobj }, + { "wixlib", FileFormat.Wixlib }, + { "wixout", FileFormat.Wixout }, + { "wixpdb", FileFormat.Wixpdb }, + { "wixmst", FileFormat.Wixout }, + { "wixmsp", FileFormat.Wixout }, + }; + + /// + /// Use Create or Read to create a FileStructure. + /// + private FileStructure() { } + + /// + /// Count of embedded files in the file structure. + /// + public int EmbeddedFileCount { get { return this.embeddedFileSizes.Length; } } + + /// + /// File format of the file structure. + /// + public FileFormat FileFormat { get; private set; } + + /// + /// Creates a new file structure. + /// + /// Stream to write the file structure to. + /// File format for the file structure. + /// Paths to files to embedd in the file structure. + /// Newly created file structure. + public static FileStructure Create(Stream stream, FileFormat fileFormat, List embedFilePaths) + { + FileStructure fs = new FileStructure(); + using (NonClosingStreamWrapper wrapper = new NonClosingStreamWrapper(stream)) + using (BinaryWriter writer = new BinaryWriter(wrapper)) + { + fs.WriteType(writer, fileFormat); + + fs.WriteEmbeddedFiles(writer, embedFilePaths ?? new List()); + + // Remember the data stream offset, which is right after the embedded files have been written. + fs.dataStreamOffset = stream.Position; + } + + fs.stream = stream; + + return fs; + } + + /// + /// Reads a file structure from an open stream. + /// + /// Stream to read from. + /// File structure populated from the stream. + public static FileStructure Read(Stream stream) + { + FileStructure fs = new FileStructure(); + using (NonClosingStreamWrapper wrapper = new NonClosingStreamWrapper(stream)) + using (BinaryReader reader = new BinaryReader(wrapper)) + { + fs.FileFormat = FileStructure.ReadFileFormat(reader); + + if (FileFormat.Unknown != fs.FileFormat) + { + fs.embeddedFileSizes = FileStructure.ReadEmbeddedFileSizes(reader); + + // Remember the data stream offset, which is right after the embedded files have been written. + fs.dataStreamOffset = stream.Position; + foreach (long size in fs.embeddedFileSizes) + { + fs.dataStreamOffset += size; + } + } + } + + fs.stream = stream; + + return fs; + } + + /// + /// Guess at the file format based on the file extension. + /// + /// File extension to guess the file format for. + /// Best guess at file format. + public static FileFormat GuessFileFormatFromExtension(string extension) + { + FileFormat format; + return FileStructure.SupportedFileFormats.TryGetValue(extension.TrimStart('.').ToLowerInvariant(), out format) ? format : FileFormat.Unknown; + } + + /// + /// Probes a stream to determine the file format. + /// + /// Stream to test. + /// The file format. + public static FileFormat TestFileFormat(Stream stream) + { + FileFormat format = FileFormat.Unknown; + + long position = stream.Position; + + try + { + using (NonClosingStreamWrapper wrapper = new NonClosingStreamWrapper(stream)) + using (BinaryReader reader = new BinaryReader(wrapper)) + { + format = FileStructure.ReadFileFormat(reader); + } + } + finally + { + stream.Seek(position, SeekOrigin.Begin); + } + + return format; + } + + /// + /// Extracts an embedded file. + /// + /// Index to the file to extract. + /// Path to write the extracted file to. + public void ExtractEmbeddedFile(int embeddedIndex, string outputPath) + { + if (this.EmbeddedFileCount <= embeddedIndex) + { + throw new ArgumentOutOfRangeException("embeddedIndex"); + } + + long header = 6 + 4 + (this.embeddedFileSizes.Length * 8); // skip the type + the count of embedded files + all the sizes of embedded files. + long position = this.embeddedFileSizes.Take(embeddedIndex).Sum(); // skip to the embedded file we want. + long size = this.embeddedFileSizes[embeddedIndex]; + + this.stream.Seek(header + position, SeekOrigin.Begin); + + Directory.CreateDirectory(Path.GetDirectoryName(outputPath)); + + using (FileStream output = File.OpenWrite(outputPath)) + { + int read; + int total = 0; + byte[] buffer = new byte[64 * 1024]; + while (0 < (read = this.stream.Read(buffer, 0, (int)Math.Min(buffer.Length, size - total)))) + { + output.Write(buffer, 0, read); + total += read; + } + } + } + + /// + /// Gets a non-closing stream to the data of the file. + /// + /// Stream to the data of the file. + public Stream GetDataStream() + { + this.stream.Seek(this.dataStreamOffset, SeekOrigin.Begin); + return new NonClosingStreamWrapper(this.stream); + } + + /// + /// Disposes of the internsl state of the file structure. + /// + public void Dispose() + { + Dispose(true); + GC.SuppressFinalize(this); + } + + /// + /// Disposes of the internsl state of the file structure. + /// + /// True if disposing. + protected virtual void Dispose(bool disposing) + { + if (!this.disposed) + { + if (disposing) + { + if (null != this.stream) + { + // We do not own the stream, so we don't close it. We're just resetting our internal state. + this.embeddedFileSizes = null; + this.dataStreamOffset = 0; + this.stream = null; + } + } + } + + this.disposed = true; + } + + private static FileFormat ReadFileFormat(BinaryReader reader) + { + FileFormat format = FileFormat.Unknown; + + string type = new string(reader.ReadChars(6)); + FileStructure.SupportedFileFormats.TryGetValue(type, out format); + + return format; + } + + private static long[] ReadEmbeddedFileSizes(BinaryReader reader) + { + uint count = reader.ReadUInt32(); + + long[] embeddedFileSizes = new long[count]; + + for (int i = 0; i < embeddedFileSizes.Length; ++i) + { + embeddedFileSizes[i] = (long)reader.ReadUInt64(); + } + + return embeddedFileSizes; + } + + private BinaryWriter WriteType(BinaryWriter writer, FileFormat fileFormat) + { + string type = null; + foreach (var supported in FileStructure.SupportedFileFormats) + { + if (supported.Value.Equals(fileFormat)) + { + type = supported.Key; + break; + } + } + + if (String.IsNullOrEmpty(type)) + { + throw new ArgumentException("Unknown file format type", "fileFormat"); + } + + this.FileFormat = fileFormat; + + Debug.Assert(6 == type.ToCharArray().Length); + writer.Write(type.ToCharArray()); + return writer; + } + + private BinaryWriter WriteEmbeddedFiles(BinaryWriter writer, List embedFilePaths) + { + // First write the count of embedded files as a Uint32; + writer.Write((uint)embedFilePaths.Count); + + this.embeddedFileSizes = new long[embedFilePaths.Count]; + + // Next write out the size of each file as a Uint64 in order. + FileInfo[] files = new FileInfo[embedFilePaths.Count]; + for (int i = 0; i < embedFilePaths.Count; ++i) + { + files[i] = new FileInfo(embedFilePaths[i]); + + this.embeddedFileSizes[i] = files[i].Length; + writer.Write((ulong)this.embeddedFileSizes[i]); + } + + // Next write out the content of each file *after* the sizes of + // *all* of the files were written. + foreach (FileInfo file in files) + { + using (FileStream stream = file.OpenRead()) + { + stream.CopyTo(writer.BaseStream); + } + } + + return writer; + } + } +} diff --git a/src/WixToolset.Data/ILibraryBinaryFileResolver.cs b/src/WixToolset.Data/ILibraryBinaryFileResolver.cs new file mode 100644 index 00000000..b438429c --- /dev/null +++ b/src/WixToolset.Data/ILibraryBinaryFileResolver.cs @@ -0,0 +1,9 @@ +// 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.Data +{ + public interface ILibraryBinaryFileResolver + { + string Resolve(SourceLineNumber sourceLineNumber, string table, string path); + } +} diff --git a/src/WixToolset.Data/IMessageHandler.cs b/src/WixToolset.Data/IMessageHandler.cs new file mode 100644 index 00000000..5332fe4a --- /dev/null +++ b/src/WixToolset.Data/IMessageHandler.cs @@ -0,0 +1,18 @@ +// 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.Data +{ + using System; + + /// + /// Interface for handling messages (error/warning/verbose). + /// + public interface IMessageHandler + { + /// + /// Sends a message with the given arguments. + /// + /// Message arguments. + void OnMessage(MessageEventArgs e); + } +} diff --git a/src/WixToolset.Data/Intermediate.cs b/src/WixToolset.Data/Intermediate.cs new file mode 100644 index 00000000..fd14f11b --- /dev/null +++ b/src/WixToolset.Data/Intermediate.cs @@ -0,0 +1,189 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + using System.IO; + using System.Xml; + + /// + /// Container class for an intermediate object. + /// + public sealed class Intermediate + { + public const string XmlNamespaceUri = "http://wixtoolset.org/schemas/v4/wixobj"; + private static readonly Version CurrentVersion = new Version("4.0.0.0"); + + private string id; + private List
sections; + + /// + /// Instantiate a new Intermediate. + /// + public Intermediate() + { + this.id = Convert.ToBase64String(Guid.NewGuid().ToByteArray()).TrimEnd('=').Replace('+', '.').Replace('/', '_'); + this.sections = new List
(); + } + + /// + /// Get the sections contained in this intermediate. + /// + /// Sections contained in this intermediate. + public IEnumerable
Sections { get { return this.sections; } } + + /// + /// Adds a section to the intermediate. + /// + /// Section to add to the intermediate. + public void AddSection(Section section) + { + section.IntermediateId = this.id; + this.sections.Add(section); + } + + /// + /// Loads an intermediate from a path on disk. + /// + /// Path to intermediate file saved on disk. + /// Collection containing TableDefinitions to use when reconstituting the intermediate. + /// Suppress checking for wix.dll version mismatches. + /// Returns the loaded intermediate. + public static Intermediate Load(string path, TableDefinitionCollection tableDefinitions, bool suppressVersionCheck) + { + using (FileStream stream = File.OpenRead(path)) + using (FileStructure fs = FileStructure.Read(stream)) + { + if (FileFormat.Wixobj != fs.FileFormat) + { + throw new WixUnexpectedFileFormatException(path, FileFormat.Wixobj, fs.FileFormat); + } + + Uri uri = new Uri(Path.GetFullPath(path)); + using (XmlReader reader = XmlReader.Create(fs.GetDataStream(), null, uri.AbsoluteUri)) + { + try + { + reader.MoveToContent(); + return Intermediate.Read(reader, tableDefinitions, suppressVersionCheck); + } + catch (XmlException xe) + { + throw new WixCorruptFileException(path, fs.FileFormat, xe); + } + } + } + } + + /// + /// Saves an intermediate to a path on disk. + /// + /// Path to save intermediate file to disk. + public void Save(string path) + { + Directory.CreateDirectory(Path.GetDirectoryName(Path.GetFullPath(path))); + + using (FileStream stream = File.Create(path)) + using (FileStructure fs = FileStructure.Create(stream, FileFormat.Wixobj, null)) + using (XmlWriter writer = XmlWriter.Create(fs.GetDataStream())) + { + writer.WriteStartDocument(); + this.Write(writer); + writer.WriteEndDocument(); + } + } + + /// + /// Parse an intermediate from an XML format. + /// + /// XmlReader where the intermediate is persisted. + /// TableDefinitions to use in the intermediate. + /// Suppress checking for wix.dll version mismatch. + /// The parsed Intermediate. + private static Intermediate Read(XmlReader reader, TableDefinitionCollection tableDefinitions, bool suppressVersionCheck) + { + if ("wixObject" != reader.LocalName) + { + throw new XmlException(); + } + + bool empty = reader.IsEmptyElement; + Version objVersion = null; + string id = null; + + while (reader.MoveToNextAttribute()) + { + switch (reader.LocalName) + { + case "version": + objVersion = new Version(reader.Value); + break; + case "id": + id = reader.Value; + break; + } + } + + if (!suppressVersionCheck && null != objVersion && !Intermediate.CurrentVersion.Equals(objVersion)) + { + throw new WixException(WixDataErrors.VersionMismatch(SourceLineNumber.CreateFromUri(reader.BaseURI), "object", objVersion.ToString(), Intermediate.CurrentVersion.ToString())); + } + + Intermediate intermediate = new Intermediate(); + intermediate.id = id; + + if (!empty) + { + bool done = false; + + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + switch (reader.LocalName) + { + case "section": + intermediate.AddSection(Section.Read(reader, tableDefinitions)); + break; + default: + throw new XmlException(); + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + + return intermediate; + } + + /// + /// Persists an intermediate in an XML format. + /// + /// XmlWriter where the Intermediate should persist itself as XML. + private void Write(XmlWriter writer) + { + writer.WriteStartElement("wixObject", XmlNamespaceUri); + + writer.WriteAttributeString("version", Intermediate.CurrentVersion.ToString()); + + writer.WriteAttributeString("id", this.id); + + foreach (Section section in this.Sections) + { + section.Write(writer); + } + + writer.WriteEndElement(); + } + } +} diff --git a/src/WixToolset.Data/Library.cs b/src/WixToolset.Data/Library.cs new file mode 100644 index 00000000..bb04d216 --- /dev/null +++ b/src/WixToolset.Data/Library.cs @@ -0,0 +1,297 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + using System.IO; + using System.Linq; + using System.Xml; + + /// + /// Object that represents a library file. + /// + public sealed class Library + { + public const string XmlNamespaceUri = "http://wixtoolset.org/schemas/v4/wixlib"; + private static readonly Version CurrentVersion = new Version("4.0.0.0"); + + private string id; + private Dictionary localizations; + private List
sections; + + /// + /// Instantiates a new empty library which is only useful from static creating methods. + /// + private Library() + { + this.localizations = new Dictionary(); + this.sections = new List
(); + } + + /// + /// Instantiate a new library populated with sections. + /// + /// Sections to add to the library. + public Library(IEnumerable
sections) + { + this.localizations = new Dictionary(); + this.sections = new List
(sections); + + this.id = Convert.ToBase64String(Guid.NewGuid().ToByteArray()).TrimEnd('=').Replace('+', '.').Replace('/', '_'); + foreach (Section section in this.sections) + { + section.LibraryId = this.id; + } + } + + /// + /// Get the sections contained in this library. + /// + /// Sections contained in this library. + public IEnumerable
Sections { get { return this.sections; } } + + /// + /// Add a localization file to this library. + /// + /// The localization file to add. + public void AddLocalization(Localization localization) + { + Localization existingCulture; + if (this.localizations.TryGetValue(localization.Culture, out existingCulture)) + { + existingCulture.Merge(localization); + } + else + { + this.localizations.Add(localization.Culture, localization); + } + } + + /// + /// Gets localization files from this library that match the cultures passed in, in the order of the array of cultures. + /// + /// The list of cultures to get localizations for. + /// All localizations contained in this library that match the set of cultures provided, in the same order. + public IEnumerable GetLocalizations(string[] cultures) + { + foreach (string culture in cultures ?? new string[0]) + { + Localization localization; + if (this.localizations.TryGetValue(culture, out localization)) + { + yield return localization; + } + } + } + + /// + /// Loads a library from a path on disk. + /// + /// Path to library file saved on disk. + /// Collection containing TableDefinitions to use when reconstituting the intermediates. + /// Suppresses wix.dll version mismatch check. + /// Returns the loaded library. + public static Library Load(string path, TableDefinitionCollection tableDefinitions, bool suppressVersionCheck) + { + using (FileStream stream = File.OpenRead(path)) + { + return Load(stream, new Uri(Path.GetFullPath(path)), tableDefinitions, suppressVersionCheck); + } + } + + /// + /// Loads a library from a stream. + /// + /// Stream containing the library file. + /// Uri for finding this stream. + /// Collection containing TableDefinitions to use when reconstituting the intermediates. + /// Suppresses wix.dll version mismatch check. + /// Returns the loaded library. + public static Library Load(Stream stream, Uri uri, TableDefinitionCollection tableDefinitions, bool suppressVersionCheck) + { + using (FileStructure fs = FileStructure.Read(stream)) + { + if (FileFormat.Wixlib != fs.FileFormat) + { + throw new WixUnexpectedFileFormatException(uri.LocalPath, FileFormat.Wixlib, fs.FileFormat); + } + + using (XmlReader reader = XmlReader.Create(fs.GetDataStream(), null, uri.AbsoluteUri)) + { + try + { + reader.MoveToContent(); + return Library.Read(reader, tableDefinitions, suppressVersionCheck); + } + catch (XmlException xe) + { + throw new WixCorruptFileException(uri.LocalPath, fs.FileFormat, xe); + } + } + } + } + + /// + /// Saves a library to a path on disk. + /// + /// Path to save library file to on disk. + /// The WiX path resolver. + public void Save(string path, ILibraryBinaryFileResolver resolver) + { + List embedFilePaths = new List(); + + // Resolve paths to files that are to be embedded in the library. + if (null != resolver) + { + foreach (Table table in this.sections.SelectMany(s => s.Tables)) + { + foreach (Row row in table.Rows) + { + foreach (ObjectField objectField in row.Fields.Where(f => f is ObjectField)) + { + if (null != objectField.Data) + { + string file = resolver.Resolve(row.SourceLineNumbers, table.Name, (string)objectField.Data); + if (!String.IsNullOrEmpty(file)) + { + // File was successfully resolved so track the embedded index as the embedded file index. + objectField.EmbeddedFileIndex = embedFilePaths.Count; + embedFilePaths.Add(file); + } + else + { + Messaging.Instance.OnMessage(WixDataErrors.FileNotFound(row.SourceLineNumbers, (string)objectField.Data, table.Name)); + } + } + else // clear out embedded file id in case there was one there before. + { + objectField.EmbeddedFileIndex = null; + } + } + } + } + } + + // Do not save the library if errors were found while resolving object paths. + if (Messaging.Instance.EncounteredError) + { + return; + } + + // Ensure the location to output the library exists and write it out. + Directory.CreateDirectory(Path.GetDirectoryName(Path.GetFullPath(path))); + + using (FileStream stream = File.Create(path)) + using (FileStructure fs = FileStructure.Create(stream, FileFormat.Wixlib, embedFilePaths)) + using (XmlWriter writer = XmlWriter.Create(fs.GetDataStream())) + { + writer.WriteStartDocument(); + + this.Write(writer); + + writer.WriteEndDocument(); + } + } + + /// + /// Parse the root library element. + /// + /// XmlReader with library persisted as Xml. + /// Collection containing TableDefinitions to use when reconstituting the intermediates. + /// Suppresses check for wix.dll version mismatch. + /// The parsed Library. + private static Library Read(XmlReader reader, TableDefinitionCollection tableDefinitions, bool suppressVersionCheck) + { + if (!reader.LocalName.Equals("wixLibrary")) + { + throw new XmlException(); + } + + bool empty = reader.IsEmptyElement; + Library library = new Library(); + Version version = null; + + while (reader.MoveToNextAttribute()) + { + switch (reader.LocalName) + { + case "version": + version = new Version(reader.Value); + break; + case "id": + library.id = reader.Value; + break; + } + } + + if (!suppressVersionCheck && null != version && !Library.CurrentVersion.Equals(version)) + { + throw new WixException(WixDataErrors.VersionMismatch(SourceLineNumber.CreateFromUri(reader.BaseURI), "library", version.ToString(), Library.CurrentVersion.ToString())); + } + + if (!empty) + { + bool done = false; + + while (!done && (XmlNodeType.Element == reader.NodeType || reader.Read())) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + switch (reader.LocalName) + { + case "localization": + Localization localization = Localization.Read(reader, tableDefinitions); + library.localizations.Add(localization.Culture, localization); + break; + case "section": + Section section = Section.Read(reader, tableDefinitions); + section.LibraryId = library.id; + library.sections.Add(section); + break; + default: + throw new XmlException(); + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + + return library; + } + + /// + /// Persists a library in an XML format. + /// + /// XmlWriter where the library should persist itself as XML. + private void Write(XmlWriter writer) + { + writer.WriteStartElement("wixLibrary", XmlNamespaceUri); + + writer.WriteAttributeString("version", CurrentVersion.ToString()); + + writer.WriteAttributeString("id", this.id); + + foreach (Localization localization in this.localizations.Values) + { + localization.Write(writer); + } + + foreach (Section section in this.sections) + { + section.Write(writer); + } + + writer.WriteEndElement(); + } + } +} diff --git a/src/WixToolset.Data/Localization.cs b/src/WixToolset.Data/Localization.cs new file mode 100644 index 00000000..658ce14a --- /dev/null +++ b/src/WixToolset.Data/Localization.cs @@ -0,0 +1,372 @@ +// 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.Data +{ + using System; + using System.Collections; + using System.Collections.Generic; + using System.Diagnostics; + using System.Diagnostics.CodeAnalysis; + using System.Globalization; + using System.IO; + using System.Linq; + using System.Reflection; + using System.Xml; + using System.Xml.Linq; + using System.Xml.Schema; + using WixToolset.Data.Msi; + using WixToolset.Data.Rows; + + /// + /// Object that represents a localization file. + /// + public sealed class Localization + { + private static string XmlElementName = "localization"; + + private Dictionary variables = new Dictionary(); + private Dictionary localizedControls = new Dictionary(); + + /// + /// Instantiates a new localization object. + /// + public Localization(int codepage, string culture, IDictionary variables, IDictionary localizedControls) + { + this.Codepage = codepage; + this.Culture = String.IsNullOrEmpty(culture) ? String.Empty : culture.ToLowerInvariant(); + this.variables = new Dictionary(variables); + this.localizedControls = new Dictionary(localizedControls); + } + + /// + /// Gets the codepage. + /// + /// The codepage. + public int Codepage { get; private set; } + + /// + /// Gets the culture. + /// + /// The culture. + public string Culture { get; private set; } + + /// + /// Gets the variables. + /// + /// The variables. + public ICollection Variables + { + get { return this.variables.Values; } + } + + /// + /// Gets the localized controls. + /// + /// The localized controls. + public ICollection> LocalizedControls + { + get { return this.localizedControls; } + } + + /// + /// Merge the information from another localization object into this one. + /// + /// The localization object to be merged into this one. + public void Merge(Localization localization) + { + foreach (WixVariableRow wixVariableRow in localization.Variables) + { + WixVariableRow existingWixVariableRow; + if (!this.variables.TryGetValue(wixVariableRow.Id, out existingWixVariableRow) || (existingWixVariableRow.Overridable && !wixVariableRow.Overridable)) + { + variables[wixVariableRow.Id] = wixVariableRow; + } + else if (!wixVariableRow.Overridable) + { + throw new WixException(WixDataErrors.DuplicateLocalizationIdentifier(wixVariableRow.SourceLineNumbers, wixVariableRow.Id)); + } + } + } + + /// + /// Loads a localization file from a stream. + /// + /// XmlReader where the intermediate is persisted. + /// Collection containing TableDefinitions to use when loading the localization file. + /// Returns the loaded localization. + internal static Localization Read(XmlReader reader, TableDefinitionCollection tableDefinitions) + { + Debug.Assert("localization" == reader.LocalName); + + int codepage = 0; + string culture = null; + bool empty = reader.IsEmptyElement; + + while (reader.MoveToNextAttribute()) + { + switch (reader.Name) + { + case "codepage": + codepage = Convert.ToInt32(reader.Value, CultureInfo.InvariantCulture); + break; + case "culture": + culture = reader.Value; + break; + } + } + + TableDefinition wixVariableTable = tableDefinitions["WixVariable"]; + Dictionary variables = new Dictionary(); + Dictionary localizedControls = new Dictionary(); + + if (!empty) + { + bool done = false; + + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + switch (reader.LocalName) + { + case "string": + WixVariableRow row = Localization.ReadString(reader, wixVariableTable); + variables.Add(row.Id, row); + break; + + case "ui": + LocalizedControl ui = Localization.ReadUI(reader); + localizedControls.Add(ui.GetKey(), ui); + break; + + default: + throw new XmlException(); + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + + return new Localization(codepage, culture, variables, localizedControls); + } + + /// + /// Writes a localization file into an XML format. + /// + /// XmlWriter where the localization file should persist itself as XML. + internal void Write(XmlWriter writer) + { + writer.WriteStartElement(Localization.XmlElementName, Library.XmlNamespaceUri); + + if (-1 != this.Codepage) + { + writer.WriteAttributeString("codepage", this.Codepage.ToString(CultureInfo.InvariantCulture)); + } + + if (!String.IsNullOrEmpty(this.Culture)) + { + writer.WriteAttributeString("culture", this.Culture); + } + + foreach (WixVariableRow wixVariableRow in this.variables.Values) + { + writer.WriteStartElement("string", Library.XmlNamespaceUri); + + writer.WriteAttributeString("id", wixVariableRow.Id); + + if (wixVariableRow.Overridable) + { + writer.WriteAttributeString("overridable", "yes"); + } + + writer.WriteCData(wixVariableRow.Value); + + writer.WriteEndElement(); + } + + foreach (string controlKey in this.localizedControls.Keys) + { + writer.WriteStartElement("ui", Library.XmlNamespaceUri); + + string[] controlKeys = controlKey.Split('/'); + string dialog = controlKeys[0]; + string control = controlKeys[1]; + + if (!String.IsNullOrEmpty(dialog)) + { + writer.WriteAttributeString("dialog", dialog); + } + + if (!String.IsNullOrEmpty(control)) + { + writer.WriteAttributeString("control", control); + } + + LocalizedControl localizedControl = this.localizedControls[controlKey]; + + if (Common.IntegerNotSet != localizedControl.X) + { + writer.WriteAttributeString("x", localizedControl.X.ToString()); + } + + if (Common.IntegerNotSet != localizedControl.Y) + { + writer.WriteAttributeString("y", localizedControl.Y.ToString()); + } + + if (Common.IntegerNotSet != localizedControl.Width) + { + writer.WriteAttributeString("width", localizedControl.Width.ToString()); + } + + if (Common.IntegerNotSet != localizedControl.Height) + { + writer.WriteAttributeString("height", localizedControl.Height.ToString()); + } + + if (MsiInterop.MsidbControlAttributesRTLRO == (localizedControl.Attributes & MsiInterop.MsidbControlAttributesRTLRO)) + { + writer.WriteAttributeString("rightToLeft", "yes"); + } + + if (MsiInterop.MsidbControlAttributesRightAligned == (localizedControl.Attributes & MsiInterop.MsidbControlAttributesRightAligned)) + { + writer.WriteAttributeString("rightAligned", "yes"); + } + + if (MsiInterop.MsidbControlAttributesLeftScroll == (localizedControl.Attributes & MsiInterop.MsidbControlAttributesLeftScroll)) + { + writer.WriteAttributeString("leftScroll", "yes"); + } + + if (!String.IsNullOrEmpty(localizedControl.Text)) + { + writer.WriteCData(localizedControl.Text); + } + + writer.WriteEndElement(); + } + + writer.WriteEndElement(); + } + + /// + /// Loads a localization file from a stream. + /// + /// XmlReader where the intermediate is persisted. + /// Collection containing TableDefinitions to use when loading the localization file. + /// Returns the loaded localization. + private static WixVariableRow ReadString(XmlReader reader, TableDefinition wixVariableTable) + { + Debug.Assert("string" == reader.LocalName); + + string id = null; + string value = null; + bool overridable = false; + bool empty = reader.IsEmptyElement; + + while (reader.MoveToNextAttribute()) + { + switch (reader.Name) + { + case "id": + id = reader.Value; + break; + case "overridable": + overridable = reader.Value.Equals("yes"); + break; + } + } + + + if (!empty) + { + reader.Read(); + + value = reader.Value; + + reader.Read(); + + if (XmlNodeType.EndElement != reader.NodeType) + { + throw new XmlException(); + } + } + + WixVariableRow wixVariableRow = new WixVariableRow(SourceLineNumber.CreateFromUri(reader.BaseURI), wixVariableTable); + wixVariableRow.Id = id; + wixVariableRow.Overridable = overridable; + wixVariableRow.Value = value; + + return wixVariableRow; + } + + private static LocalizedControl ReadUI(XmlReader reader) + { + Debug.Assert("ui" == reader.LocalName); + + string dialog = null; + string control = null; + int x = Common.IntegerNotSet; + int y = Common.IntegerNotSet; + int width = Common.IntegerNotSet; + int height = Common.IntegerNotSet; + int attributes = Common.IntegerNotSet; + string text = null; + bool empty = reader.IsEmptyElement; + + while (reader.MoveToNextAttribute()) + { + switch (reader.Name) + { + case "dialog": + dialog = reader.Value; + break; + case "control": + control = reader.Value; + break; + case "x": + x = Convert.ToInt32(reader.Value, CultureInfo.InvariantCulture); + break; + case "y": + y = Convert.ToInt32(reader.Value, CultureInfo.InvariantCulture); + break; + case "width": + width = Convert.ToInt32(reader.Value, CultureInfo.InvariantCulture); + break; + case "height": + height = Convert.ToInt32(reader.Value, CultureInfo.InvariantCulture); + break; + case "attributes": + attributes = Convert.ToInt32(reader.Value, CultureInfo.InvariantCulture); + break; + } + } + + if (!empty) + { + reader.Read(); + + text = reader.Value; + + reader.Read(); + + if (XmlNodeType.EndElement != reader.NodeType) + { + throw new XmlException(); + } + } + + return new LocalizedControl(dialog, control, x, y, width, height, attributes, text); + } + } +} diff --git a/src/WixToolset.Data/LocalizedControl.cs b/src/WixToolset.Data/LocalizedControl.cs new file mode 100644 index 00000000..50315b29 --- /dev/null +++ b/src/WixToolset.Data/LocalizedControl.cs @@ -0,0 +1,57 @@ +// 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.Data +{ + using System; + + public class LocalizedControl + { + public LocalizedControl(string dialog, string control, int x, int y, int width, int height, int attribs, string text) + { + this.Dialog = dialog; + this.Control = control; + this.X = x; + this.Y = y; + this.Width = width; + this.Height = height; + this.Attributes = attribs; + this.Text = text; + } + + public string Dialog { get; set; } + + public string Control { get; set; } + + public int X { get; private set; } + + public int Y { get; private set; } + + public int Width { get; private set; } + + public int Height { get; private set; } + + public int Attributes { get; private set; } + + public string Text { get; private set; } + + /// + /// Get key for a localized control. + /// + /// The localized control id. + public string GetKey() + { + return LocalizedControl.GetKey(this.Dialog, this.Control); + } + + /// + /// Get key for a localized control. + /// + /// The optional id of the control's dialog. + /// The id of the control. + /// The localized control id. + public static string GetKey(string dialog, string control) + { + return String.Concat(String.IsNullOrEmpty(dialog) ? String.Empty : dialog, "/", String.IsNullOrEmpty(control) ? String.Empty : control); + } + } +} diff --git a/src/WixToolset.Data/MessageEventArgs.cs b/src/WixToolset.Data/MessageEventArgs.cs new file mode 100644 index 00000000..472b5e95 --- /dev/null +++ b/src/WixToolset.Data/MessageEventArgs.cs @@ -0,0 +1,176 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + using System.Diagnostics.CodeAnalysis; + using System.Globalization; + using System.Resources; + using System.Text; + + /// + /// Event args for message events. + /// + public abstract class MessageEventArgs : EventArgs + { + private SourceLineNumber sourceLineNumbers; + private int id; + private string resourceName; + private object[] messageArgs; + private MessageLevel level; + + /// + /// Creates a new MessageEventArgs. + /// + /// Source line numbers for the message. + /// Id for the message. + /// Name of the resource. + /// Arguments for the format string. + protected MessageEventArgs(SourceLineNumber sourceLineNumbers, int id, string resourceName, params object[] messageArgs) + { + this.sourceLineNumbers = sourceLineNumbers; + this.id = id; + this.resourceName = resourceName; + this.messageArgs = messageArgs; + + // Default to Nothing, since the default MessageEventArgs container + // classes define a level, and only WixErrorEventArgs previously + // determined that an error occured without throwing. + this.level = MessageLevel.Nothing; + } + + /// + /// Gets the resource manager for this event args. + /// + /// The resource manager for this event args. + public ResourceManager ResourceManager { get; protected set; } + + /// + /// Gets the source line numbers. + /// + /// The source line numbers. + public SourceLineNumber SourceLineNumbers + { + get { return this.sourceLineNumbers; } + } + + /// + /// Gets the Id for the message. + /// + /// The Id for the message. + public int Id + { + get { return this.id; } + } + + /// + /// Gets the name of the resource. + /// + /// The name of the resource. + public string ResourceName + { + get { return this.resourceName; } + } + + /// + /// Gets or sets the for the message. + /// + /// The for the message. + /// + /// The may set the level differently + /// depending on suppression and escalation of different message levels. + /// Message handlers should check the level to determine if an error + /// or other message level was raised. + /// + public MessageLevel Level + { + get { return this.level; } + set { this.level = value; } + } + + /// + /// Gets the arguments for the format string. + /// + /// The arguments for the format string. + [SuppressMessage("Microsoft.Performance", "CA1819:PropertiesShouldNotReturnArrays")] + public object[] MessageArgs + { + get { return this.messageArgs; } + } + + /// + /// Creates a properly formatted message string. + /// + /// Optional short form of the application name that generated the message. Defaults to "WIX" if unspecified. + /// Optional long form of the application name that generated the message. Defaults to "WIX" if unspecified. Will be overridden by the processed filename if one was provided. + /// Optional override level of the message, as generated by MessageLevel(MessageEventArgs). + /// String containing the formatted message. + public string GenerateMessageString(string shortAppName = null, string longAppName = null, MessageLevel overrideLevel = MessageLevel.Nothing) + { + MessageLevel messageLevel = MessageLevel.Nothing == overrideLevel ? this.Level : overrideLevel; + + List fileNames = new List(); + string errorFileName = String.IsNullOrEmpty(longAppName) ? "WIX" : longAppName; + for (SourceLineNumber sln = this.SourceLineNumbers; null != sln; sln = sln.Parent) + { + if (String.IsNullOrEmpty(sln.FileName)) + { + continue; + } + else if (sln.LineNumber.HasValue) + { + if (0 == fileNames.Count) + { + errorFileName = String.Format(CultureInfo.CurrentUICulture, WixDataStrings.Format_FirstLineNumber, sln.FileName, sln.LineNumber); + } + + fileNames.Add(String.Format(CultureInfo.CurrentUICulture, WixDataStrings.Format_LineNumber, sln.FileName, sln.LineNumber)); + } + else + { + if (0 == fileNames.Count) + { + errorFileName = sln.FileName; + } + + fileNames.Add(sln.FileName); + } + } + + string messageType = String.Empty; + if (MessageLevel.Warning == messageLevel) + { + messageType = WixDataStrings.MessageType_Warning; + } + else if (MessageLevel.Error == messageLevel) + { + messageType = WixDataStrings.MessageType_Error; + } + + StringBuilder messageBuilder = new StringBuilder(); + string message = String.Format(CultureInfo.InvariantCulture, this.ResourceManager.GetString(this.ResourceName), this.MessageArgs); + if (MessageLevel.Information == messageLevel || MessageLevel.Verbose == messageLevel) + { + messageBuilder.AppendFormat(WixDataStrings.Format_InfoMessage, message); + } + else + { + messageBuilder.AppendFormat(WixDataStrings.Format_NonInfoMessage, errorFileName, messageType, String.IsNullOrEmpty(shortAppName) ? "WIX" : shortAppName, this.Id, message); + } + + if (1 < fileNames.Count) + { + messageBuilder.AppendFormat(WixDataStrings.INF_SourceTrace, Environment.NewLine); + foreach (string fileName in fileNames) + { + messageBuilder.AppendFormat(WixDataStrings.INF_SourceTraceLocation, fileName, Environment.NewLine); + } + + messageBuilder.Append(Environment.NewLine); + } + + return messageBuilder.ToString(); + } + } +} diff --git a/src/WixToolset.Data/MessageLevel.cs b/src/WixToolset.Data/MessageLevel.cs new file mode 100644 index 00000000..d980e371 --- /dev/null +++ b/src/WixToolset.Data/MessageLevel.cs @@ -0,0 +1,25 @@ +// 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.Data +{ + /// + /// Enum for message to display. + /// + public enum MessageLevel + { + /// Display nothing. + Nothing, + + /// Display verbose information. + Verbose, + + /// Display information. + Information, + + /// Display warning. + Warning, + + /// Display error. + Error, + } +} diff --git a/src/WixToolset.Data/Messaging.cs b/src/WixToolset.Data/Messaging.cs new file mode 100644 index 00000000..de2cdcfe --- /dev/null +++ b/src/WixToolset.Data/Messaging.cs @@ -0,0 +1,173 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + + public class Messaging : IMessageHandler + { + private static readonly Messaging instance = new Messaging(); + + private HashSet suppressedWarnings = new HashSet(); + private HashSet warningsAsErrors = new HashSet(); + private string longAppName; + private string shortAppName; + + static Messaging() + { + } + + private Messaging() + { + } + + public static Messaging Instance { get { return Messaging.instance; } } + + /// + /// Event fired when messages are to be displayed. + /// + public event DisplayEventHandler Display; + + /// + /// Gets a bool indicating whether an error has been found. + /// + /// A bool indicating whether an error has been found. + public bool EncounteredError { get; private set; } + + /// + /// Gets the last error code encountered during messaging. + /// + /// The exit code for the process. + public int LastErrorNumber { get; private set; } + + /// + /// Gets or sets the option to show verbose messages. + /// + /// The option to show verbose messages. + public bool ShowVerboseMessages { get; set; } + + /// + /// Gets or sets the option to suppress all warning messages. + /// + /// The option to suppress all warning messages. + public bool SuppressAllWarnings { get; set; } + + /// + /// Gets and sets the option to treat warnings as errors. + /// + /// The option to treat warnings as errors. + public bool WarningsAsError { get; set; } + + /// + /// Implements IMessageHandler to display error messages. + /// + /// Message event arguments. + public void OnMessage(MessageEventArgs mea) + { + MessageLevel messageLevel = this.CalculateMessageLevel(mea); + + if (MessageLevel.Nothing == messageLevel) + { + return; + } + else if (MessageLevel.Error == messageLevel) + { + this.EncounteredError = true; + this.LastErrorNumber = mea.Id; + } + + if (null != this.Display) + { + string message = mea.GenerateMessageString(this.shortAppName, this.longAppName, messageLevel); + if (!String.IsNullOrEmpty(message)) + { + this.Display(this, new DisplayEventArgs() { Level = messageLevel, Message = message }); + } + } + else if (MessageLevel.Error == mea.Level) + { + throw new WixException(mea); + } + } + + /// + /// Sets the app names. + /// + /// Short application name; usually 4 uppercase characters. + /// Long application name; usually the executable name. + public Messaging InitializeAppName(string shortName, string longName) + { + this.EncounteredError = false; + this.LastErrorNumber = 0; + + this.Display = null; + this.ShowVerboseMessages = false; + this.SuppressAllWarnings = false; + this.WarningsAsError = false; + this.suppressedWarnings.Clear(); + this.warningsAsErrors.Clear(); + + this.shortAppName = shortName; + this.longAppName = longName; + + return this; + } + + /// + /// Adds a warning message id to be elevated to an error message. + /// + /// Id of the message to elevate. + /// + /// Suppressed warnings will not be elevated as errors. + /// + public void ElevateWarningMessage(int warningNumber) + { + this.warningsAsErrors.Add(warningNumber); + } + + /// + /// Adds a warning message id to be suppressed in message output. + /// + /// Id of the message to suppress. + /// + /// Suppressed warnings will not be elevated as errors. + /// + public void SuppressWarningMessage(int warningNumber) + { + this.suppressedWarnings.Add(warningNumber); + } + + /// + /// Determines the level of this message, when taking into account warning-as-error, + /// warning level, verbosity level and message suppressed by the caller. + /// + /// Event arguments for the message. + /// MessageLevel representing the level of this message. + private MessageLevel CalculateMessageLevel(MessageEventArgs mea) + { + MessageLevel messageLevel = mea.Level; + + if (MessageLevel.Verbose == messageLevel) + { + if (!this.ShowVerboseMessages) + { + messageLevel = MessageLevel.Nothing; + } + } + else if (MessageLevel.Warning == messageLevel) + { + if (this.SuppressAllWarnings || this.suppressedWarnings.Contains(mea.Id)) + { + messageLevel = MessageLevel.Nothing; + } + else if (this.WarningsAsError || this.warningsAsErrors.Contains(mea.Id)) + { + messageLevel = MessageLevel.Error; + } + } + + return messageLevel; + } + } +} diff --git a/src/WixToolset.Data/Msi/MsiInterop.cs b/src/WixToolset.Data/Msi/MsiInterop.cs new file mode 100644 index 00000000..1a8ebbf0 --- /dev/null +++ b/src/WixToolset.Data/Msi/MsiInterop.cs @@ -0,0 +1,313 @@ +// 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.Data.Msi +{ + class MsiInterop + { + // Patching constants + internal const int MsiMaxStreamNameLength = 62; // http://msdn2.microsoft.com/library/aa370551.aspx + + // Component.Attributes + internal const int MsidbComponentAttributesLocalOnly = 0; + internal const int MsidbComponentAttributesSourceOnly = 1; + internal const int MsidbComponentAttributesOptional = 2; + internal const int MsidbComponentAttributesRegistryKeyPath = 4; + internal const int MsidbComponentAttributesSharedDllRefCount = 8; + internal const int MsidbComponentAttributesPermanent = 16; + internal const int MsidbComponentAttributesODBCDataSource = 32; + internal const int MsidbComponentAttributesTransitive = 64; + internal const int MsidbComponentAttributesNeverOverwrite = 128; + internal const int MsidbComponentAttributes64bit = 256; + internal const int MsidbComponentAttributesDisableRegistryReflection = 512; + internal const int MsidbComponentAttributesUninstallOnSupersedence = 1024; + internal const int MsidbComponentAttributesShared = 2048; + + // BBControl.Attributes & Control.Attributes + internal const int MsidbControlAttributesVisible = 0x00000001; + internal const int MsidbControlAttributesEnabled = 0x00000002; + internal const int MsidbControlAttributesSunken = 0x00000004; + internal const int MsidbControlAttributesIndirect = 0x00000008; + internal const int MsidbControlAttributesInteger = 0x00000010; + internal const int MsidbControlAttributesRTLRO = 0x00000020; + internal const int MsidbControlAttributesRightAligned = 0x00000040; + internal const int MsidbControlAttributesLeftScroll = 0x00000080; + internal const int MsidbControlAttributesBiDi = MsidbControlAttributesRTLRO | MsidbControlAttributesRightAligned | MsidbControlAttributesLeftScroll; + + // Text controls + internal const int MsidbControlAttributesTransparent = 0x00010000; + internal const int MsidbControlAttributesNoPrefix = 0x00020000; + internal const int MsidbControlAttributesNoWrap = 0x00040000; + internal const int MsidbControlAttributesFormatSize = 0x00080000; + internal const int MsidbControlAttributesUsersLanguage = 0x00100000; + + // Edit controls + internal const int MsidbControlAttributesMultiline = 0x00010000; + internal const int MsidbControlAttributesPasswordInput = 0x00200000; + + // ProgressBar controls + internal const int MsidbControlAttributesProgress95 = 0x00010000; + + // VolumeSelectCombo and DirectoryCombo controls + internal const int MsidbControlAttributesRemovableVolume = 0x00010000; + internal const int MsidbControlAttributesFixedVolume = 0x00020000; + internal const int MsidbControlAttributesRemoteVolume = 0x00040000; + internal const int MsidbControlAttributesCDROMVolume = 0x00080000; + internal const int MsidbControlAttributesRAMDiskVolume = 0x00100000; + internal const int MsidbControlAttributesFloppyVolume = 0x00200000; + + // VolumeCostList controls + internal const int MsidbControlShowRollbackCost = 0x00400000; + + // ListBox and ComboBox controls + internal const int MsidbControlAttributesSorted = 0x00010000; + internal const int MsidbControlAttributesComboList = 0x00020000; + + // picture button controls + internal const int MsidbControlAttributesImageHandle = 0x00010000; + internal const int MsidbControlAttributesPushLike = 0x00020000; + internal const int MsidbControlAttributesBitmap = 0x00040000; + internal const int MsidbControlAttributesIcon = 0x00080000; + internal const int MsidbControlAttributesFixedSize = 0x00100000; + internal const int MsidbControlAttributesIconSize16 = 0x00200000; + internal const int MsidbControlAttributesIconSize32 = 0x00400000; + internal const int MsidbControlAttributesIconSize48 = 0x00600000; + internal const int MsidbControlAttributesElevationShield = 0x00800000; + + // RadioButton controls + internal const int MsidbControlAttributesHasBorder = 0x01000000; + + // CustomAction.Type + // executable types + internal const int MsidbCustomActionTypeDll = 0x00000001; // Target = entry point name + internal const int MsidbCustomActionTypeExe = 0x00000002; // Target = command line args + internal const int MsidbCustomActionTypeTextData = 0x00000003; // Target = text string to be formatted and set into property + internal const int MsidbCustomActionTypeJScript = 0x00000005; // Target = entry point name; null if none to call + internal const int MsidbCustomActionTypeVBScript = 0x00000006; // Target = entry point name; null if none to call + internal const int MsidbCustomActionTypeInstall = 0x00000007; // Target = property list for nested engine initialization + internal const int MsidbCustomActionTypeSourceBits = 0x00000030; + internal const int MsidbCustomActionTypeTargetBits = 0x00000007; + internal const int MsidbCustomActionTypeReturnBits = 0x000000C0; + internal const int MsidbCustomActionTypeExecuteBits = 0x00000700; + + // source of code + internal const int MsidbCustomActionTypeBinaryData = 0x00000000; // Source = Binary.Name; data stored in stream + internal const int MsidbCustomActionTypeSourceFile = 0x00000010; // Source = File.File; file part of installation + internal const int MsidbCustomActionTypeDirectory = 0x00000020; // Source = Directory.Directory; folder containing existing file + internal const int MsidbCustomActionTypeProperty = 0x00000030; // Source = Property.Property; full path to executable + + // return processing; default is syncronous execution; process return code + internal const int MsidbCustomActionTypeContinue = 0x00000040; // ignore action return status; continue running + internal const int MsidbCustomActionTypeAsync = 0x00000080; // run asynchronously + + // execution scheduling flags; default is execute whenever sequenced + internal const int MsidbCustomActionTypeFirstSequence = 0x00000100; // skip if UI sequence already run + internal const int MsidbCustomActionTypeOncePerProcess = 0x00000200; // skip if UI sequence already run in same process + internal const int MsidbCustomActionTypeClientRepeat = 0x00000300; // run on client only if UI already run on client + internal const int MsidbCustomActionTypeInScript = 0x00000400; // queue for execution within script + internal const int MsidbCustomActionTypeRollback = 0x00000100; // in conjunction with InScript: queue in Rollback script + internal const int MsidbCustomActionTypeCommit = 0x00000200; // in conjunction with InScript: run Commit ops from script on success + + // security context flag; default to impersonate as user; valid only if InScript + internal const int MsidbCustomActionTypeNoImpersonate = 0x00000800; // no impersonation; run in system context + internal const int MsidbCustomActionTypeTSAware = 0x00004000; // impersonate for per-machine installs on TS machines + internal const int MsidbCustomActionType64BitScript = 0x00001000; // script should run in 64bit process + internal const int MsidbCustomActionTypeHideTarget = 0x00002000; // don't record the contents of the Target field in the log file. + + internal const int MsidbCustomActionTypePatchUninstall = 0x00008000; // run on patch uninstall + + // Dialog.Attributes + internal const int MsidbDialogAttributesVisible = 0x00000001; + internal const int MsidbDialogAttributesModal = 0x00000002; + internal const int MsidbDialogAttributesMinimize = 0x00000004; + internal const int MsidbDialogAttributesSysModal = 0x00000008; + internal const int MsidbDialogAttributesKeepModeless = 0x00000010; + internal const int MsidbDialogAttributesTrackDiskSpace = 0x00000020; + internal const int MsidbDialogAttributesUseCustomPalette = 0x00000040; + internal const int MsidbDialogAttributesRTLRO = 0x00000080; + internal const int MsidbDialogAttributesRightAligned = 0x00000100; + internal const int MsidbDialogAttributesLeftScroll = 0x00000200; + internal const int MsidbDialogAttributesBiDi = MsidbDialogAttributesRTLRO | MsidbDialogAttributesRightAligned | MsidbDialogAttributesLeftScroll; + internal const int MsidbDialogAttributesError = 0x00010000; + internal const int CommonControlAttributesInvert = MsidbControlAttributesVisible + MsidbControlAttributesEnabled; + internal const int DialogAttributesInvert = MsidbDialogAttributesVisible + MsidbDialogAttributesModal + MsidbDialogAttributesMinimize; + + // Feature.Attributes + internal const int MsidbFeatureAttributesFavorLocal = 0; + internal const int MsidbFeatureAttributesFavorSource = 1; + internal const int MsidbFeatureAttributesFollowParent = 2; + internal const int MsidbFeatureAttributesFavorAdvertise = 4; + internal const int MsidbFeatureAttributesDisallowAdvertise = 8; + internal const int MsidbFeatureAttributesUIDisallowAbsent = 16; + internal const int MsidbFeatureAttributesNoUnsupportedAdvertise = 32; + + // File.Attributes + internal const int MsidbFileAttributesReadOnly = 1; + internal const int MsidbFileAttributesHidden = 2; + internal const int MsidbFileAttributesSystem = 4; + internal const int MsidbFileAttributesVital = 512; + internal const int MsidbFileAttributesChecksum = 1024; + internal const int MsidbFileAttributesPatchAdded = 4096; + internal const int MsidbFileAttributesNoncompressed = 8192; + internal const int MsidbFileAttributesCompressed = 16384; + + // IniFile.Action & RemoveIniFile.Action + internal const int MsidbIniFileActionAddLine = 0; + internal const int MsidbIniFileActionCreateLine = 1; + internal const int MsidbIniFileActionRemoveLine = 2; + internal const int MsidbIniFileActionAddTag = 3; + internal const int MsidbIniFileActionRemoveTag = 4; + + // MoveFile.Options + internal const int MsidbMoveFileOptionsMove = 1; + + // ServiceInstall.Attributes + internal const int MsidbServiceInstallOwnProcess = 0x00000010; + internal const int MsidbServiceInstallShareProcess = 0x00000020; + internal const int MsidbServiceInstallInteractive = 0x00000100; + internal const int MsidbServiceInstallAutoStart = 0x00000002; + internal const int MsidbServiceInstallDemandStart = 0x00000003; + internal const int MsidbServiceInstallDisabled = 0x00000004; + internal const int MsidbServiceInstallErrorIgnore = 0x00000000; + internal const int MsidbServiceInstallErrorNormal = 0x00000001; + internal const int MsidbServiceInstallErrorCritical = 0x00000003; + internal const int MsidbServiceInstallErrorControlVital = 0x00008000; + + // ServiceConfig.Event + internal const int MsidbServiceConfigEventInstall = 0x00000001; + internal const int MsidbServiceConfigEventUninstall = 0x00000002; + internal const int MsidbServiceConfigEventReinstall = 0x00000004; + + // ServiceControl.Attributes + internal const int MsidbServiceControlEventStart = 0x00000001; + internal const int MsidbServiceControlEventStop = 0x00000002; + internal const int MsidbServiceControlEventDelete = 0x00000008; + internal const int MsidbServiceControlEventUninstallStart = 0x00000010; + internal const int MsidbServiceControlEventUninstallStop = 0x00000020; + internal const int MsidbServiceControlEventUninstallDelete = 0x00000080; + + // TextStyle.StyleBits + internal const int MsidbTextStyleStyleBitsBold = 1; + internal const int MsidbTextStyleStyleBitsItalic = 2; + internal const int MsidbTextStyleStyleBitsUnderline = 4; + internal const int MsidbTextStyleStyleBitsStrike = 8; + + // Upgrade.Attributes + internal const int MsidbUpgradeAttributesMigrateFeatures = 0x00000001; + internal const int MsidbUpgradeAttributesOnlyDetect = 0x00000002; + internal const int MsidbUpgradeAttributesIgnoreRemoveFailure = 0x00000004; + internal const int MsidbUpgradeAttributesVersionMinInclusive = 0x00000100; + internal const int MsidbUpgradeAttributesVersionMaxInclusive = 0x00000200; + internal const int MsidbUpgradeAttributesLanguagesExclusive = 0x00000400; + + // Registry Hive Roots + internal const int MsidbRegistryRootClassesRoot = 0; + internal const int MsidbRegistryRootCurrentUser = 1; + internal const int MsidbRegistryRootLocalMachine = 2; + internal const int MsidbRegistryRootUsers = 3; + + // Locator Types + internal const int MsidbLocatorTypeDirectory = 0; + internal const int MsidbLocatorTypeFileName = 1; + internal const int MsidbLocatorTypeRawValue = 2; + internal const int MsidbLocatorType64bit = 16; + + internal const int MsidbClassAttributesRelativePath = 1; + + // RemoveFile.InstallMode + internal const int MsidbRemoveFileInstallModeOnInstall = 0x00000001; + internal const int MsidbRemoveFileInstallModeOnRemove = 0x00000002; + internal const int MsidbRemoveFileInstallModeOnBoth = 0x00000003; + + // ODBCDataSource.Registration + internal const int MsidbODBCDataSourceRegistrationPerMachine = 0; + internal const int MsidbODBCDataSourceRegistrationPerUser = 1; + + // ModuleConfiguration.Format + internal const int MsidbModuleConfigurationFormatText = 0; + internal const int MsidbModuleConfigurationFormatKey = 1; + internal const int MsidbModuleConfigurationFormatInteger = 2; + internal const int MsidbModuleConfigurationFormatBitfield = 3; + + // ModuleConfiguration.Attributes + internal const int MsidbMsmConfigurableOptionKeyNoOrphan = 1; + internal const int MsidbMsmConfigurableOptionNonNullable = 2; + + // ' Windows API function ShowWindow constants - used in Shortcut table + internal const int SWSHOWNORMAL = 0x00000001; + internal const int SWSHOWMAXIMIZED = 0x00000003; + internal const int SWSHOWMINNOACTIVE = 0x00000007; + + // NameToBit arrays + // UI elements + internal static readonly string[] CommonControlAttributes = { "Hidden", "Disabled", "Sunken", "Indirect", "Integer", "RightToLeft", "RightAligned", "LeftScroll" }; + internal static readonly string[] TextControlAttributes = { "Transparent", "NoPrefix", "NoWrap", "FormatSize", "UserLanguage" }; + internal static readonly string[] HyperlinkControlAttributes = { "Transparent" }; + internal static readonly string[] EditControlAttributes = { "Multiline", null, null, null, null, "Password" }; + internal static readonly string[] ProgressControlAttributes = { "ProgressBlocks" }; + internal static readonly string[] VolumeControlAttributes = { "Removable", "Fixed", "Remote", "CDROM", "RAMDisk", "Floppy", "ShowRollbackCost" }; + internal static readonly string[] ListboxControlAttributes = { "Sorted", null, null, null, "UserLanguage" }; + internal static readonly string[] ListviewControlAttributes = { "Sorted", null, null, null, "FixedSize", "Icon16", "Icon32" }; + internal static readonly string[] ComboboxControlAttributes = { "Sorted", "ComboList", null, null, "UserLanguage" }; + internal static readonly string[] RadioControlAttributes = { "Image", "PushLike", "Bitmap", "Icon", "FixedSize", "Icon16", "Icon32", null, "HasBorder" }; + internal static readonly string[] ButtonControlAttributes = { "Image", null, "Bitmap", "Icon", "FixedSize", "Icon16", "Icon32", "ElevationShield" }; + internal static readonly string[] IconControlAttributes = { "Image", null, null, null, "FixedSize", "Icon16", "Icon32" }; + internal static readonly string[] BitmapControlAttributes = { "Image", null, null, null, "FixedSize" }; + internal static readonly string[] CheckboxControlAttributes = { null, "PushLike", "Bitmap", "Icon", "FixedSize", "Icon16", "Icon32" }; + + internal const int MsidbEmbeddedUI = 0x01; + internal const int MsidbEmbeddedHandlesBasic = 0x02; + + internal const int INSTALLLOGMODE_FATALEXIT = 0x00001; + internal const int INSTALLLOGMODE_ERROR = 0x00002; + internal const int INSTALLLOGMODE_WARNING = 0x00004; + internal const int INSTALLLOGMODE_USER = 0x00008; + internal const int INSTALLLOGMODE_INFO = 0x00010; + internal const int INSTALLLOGMODE_FILESINUSE = 0x00020; + internal const int INSTALLLOGMODE_RESOLVESOURCE = 0x00040; + internal const int INSTALLLOGMODE_OUTOFDISKSPACE = 0x00080; + internal const int INSTALLLOGMODE_ACTIONSTART = 0x00100; + internal const int INSTALLLOGMODE_ACTIONDATA = 0x00200; + internal const int INSTALLLOGMODE_PROGRESS = 0x00400; + internal const int INSTALLLOGMODE_COMMONDATA = 0x00800; + internal const int INSTALLLOGMODE_INITIALIZE = 0x01000; + internal const int INSTALLLOGMODE_TERMINATE = 0x02000; + internal const int INSTALLLOGMODE_SHOWDIALOG = 0x04000; + internal const int INSTALLLOGMODE_RMFILESINUSE = 0x02000000; + internal const int INSTALLLOGMODE_INSTALLSTART = 0x04000000; + internal const int INSTALLLOGMODE_INSTALLEND = 0x08000000; + + internal const int MSICONDITIONFALSE = 0; // The table is temporary. + internal const int MSICONDITIONTRUE = 1; // The table is persistent. + internal const int MSICONDITIONNONE = 2; // The table is unknown. + internal const int MSICONDITIONERROR = 3; // An invalid handle or invalid parameter was passed to the function. + + internal const int MSIDBOPENREADONLY = 0; + internal const int MSIDBOPENTRANSACT = 1; + internal const int MSIDBOPENDIRECT = 2; + internal const int MSIDBOPENCREATE = 3; + internal const int MSIDBOPENCREATEDIRECT = 4; + internal const int MSIDBOPENPATCHFILE = 32; + + internal const int MSIMODIFYSEEK = -1; // Refreshes the information in the supplied record without changing the position in the result set and without affecting subsequent fetch operations. The record may then be used for subsequent Update, Delete, and Refresh. All primary key columns of the table must be in the query and the record must have at least as many fields as the query. Seek cannot be used with multi-table queries. This mode cannot be used with a view containing joins. See also the remarks. + internal const int MSIMODIFYREFRESH = 0; // Refreshes the information in the record. Must first call MsiViewFetch with the same record. Fails for a deleted row. Works with read-write and read-only records. + internal const int MSIMODIFYINSERT = 1; // Inserts a record. Fails if a row with the same primary keys exists. Fails with a read-only database. This mode cannot be used with a view containing joins. + internal const int MSIMODIFYUPDATE = 2; // Updates an existing record. Nonprimary keys only. Must first call MsiViewFetch. Fails with a deleted record. Works only with read-write records. + internal const int MSIMODIFYASSIGN = 3; // Writes current data in the cursor to a table row. Updates record if the primary keys match an existing row and inserts if they do not match. Fails with a read-only database. This mode cannot be used with a view containing joins. + internal const int MSIMODIFYREPLACE = 4; // Updates or deletes and inserts a record into a table. Must first call MsiViewFetch with the same record. Updates record if the primary keys are unchanged. Deletes old row and inserts new if primary keys have changed. Fails with a read-only database. This mode cannot be used with a view containing joins. + internal const int MSIMODIFYMERGE = 5; // Inserts or validates a record in a table. Inserts if primary keys do not match any row and validates if there is a match. Fails if the record does not match the data in the table. Fails if there is a record with a duplicate key that is not identical. Works only with read-write records. This mode cannot be used with a view containing joins. + internal const int MSIMODIFYDELETE = 6; // Remove a row from the table. You must first call the MsiViewFetch function with the same record. Fails if the row has been deleted. Works only with read-write records. This mode cannot be used with a view containing joins. + internal const int MSIMODIFYINSERTTEMPORARY = 7; // Inserts a temporary record. The information is not persistent. Fails if a row with the same primary key exists. Works only with read-write records. This mode cannot be used with a view containing joins. + internal const int MSIMODIFYVALIDATE = 8; // Validates a record. Does not validate across joins. You must first call the MsiViewFetch function with the same record. Obtain validation errors with MsiViewGetError. Works with read-write and read-only records. This mode cannot be used with a view containing joins. + internal const int MSIMODIFYVALIDATENEW = 9; // Validate a new record. Does not validate across joins. Checks for duplicate keys. Obtain validation errors by calling MsiViewGetError. Works with read-write and read-only records. This mode cannot be used with a view containing joins. + internal const int MSIMODIFYVALIDATEFIELD = 10; // Validates fields of a fetched or new record. Can validate one or more fields of an incomplete record. Obtain validation errors by calling MsiViewGetError. Works with read-write and read-only records. This mode cannot be used with a view containing joins. + internal const int MSIMODIFYVALIDATEDELETE = 11; // Validates a record that will be deleted later. You must first call MsiViewFetch. Fails if another row refers to the primary keys of this row. Validation does not check for the existence of the primary keys of this row in properties or strings. Does not check if a column is a foreign key to multiple tables. Obtain validation errors by calling MsiViewGetError. Works with read-write and read-only records. This mode cannot be used with a view containing joins. + + internal const uint VTI2 = 2; + internal const uint VTI4 = 3; + internal const uint VTLPWSTR = 30; + internal const uint VTFILETIME = 64; + + internal const int MSICOLINFONAMES = 0; // return column names + internal const int MSICOLINFOTYPES = 1; // return column definitions, datatype code followed by width + } +} diff --git a/src/WixToolset.Data/NonClosingStreamWrapper.cs b/src/WixToolset.Data/NonClosingStreamWrapper.cs new file mode 100644 index 00000000..53d17d4d --- /dev/null +++ b/src/WixToolset.Data/NonClosingStreamWrapper.cs @@ -0,0 +1,105 @@ +// 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.Data +{ + using System; + using System.IO; + + /// + /// Wrapper around stream to prevent other streams (like BinaryReader/Writer) from prematurely + /// closing a parent stream. + /// + internal class NonClosingStreamWrapper : Stream + { + private Stream stream; + + public NonClosingStreamWrapper(Stream stream) + { + this.stream = stream; + } + + public override bool CanRead { get { return this.stream.CanRead; } } + + public override bool CanSeek { get { return this.stream.CanSeek; } } + + public override bool CanTimeout { get { return this.stream.CanTimeout; } } + + public override bool CanWrite { get { return this.stream.CanWrite; } } + + public override long Length { get { return this.stream.Length; } } + + public override long Position { get { return this.stream.Position; } set { this.stream.Position = value; } } + + public override int ReadTimeout { get { return this.stream.ReadTimeout; } set { this.stream.ReadTimeout = value; } } + + public override int WriteTimeout { get { return this.stream.WriteTimeout; } set { this.stream.WriteTimeout = value; } } + + public override IAsyncResult BeginRead(byte[] buffer, int offset, int count, AsyncCallback callback, object state) + { + return this.stream.BeginRead(buffer, offset, count, callback, state); + } + + public override IAsyncResult BeginWrite(byte[] buffer, int offset, int count, AsyncCallback callback, object state) + { + return this.stream.BeginWrite(buffer, offset, count, callback, state); + } + + public override void Close() + { + // Do not pass through the call since this is what we are overriding. + } + + protected override void Dispose(bool disposing) + { + if (disposing) + { + this.stream.Flush(); + } + } + + public override int EndRead(IAsyncResult asyncResult) + { + return this.stream.EndRead(asyncResult); + } + + public override void EndWrite(IAsyncResult asyncResult) + { + this.stream.EndWrite(asyncResult); + } + + public override void Flush() + { + this.stream.Flush(); + } + + public override int Read(byte[] buffer, int offset, int count) + { + return this.stream.Read(buffer, offset, count); + } + + public override int ReadByte() + { + return this.stream.ReadByte(); + } + + public override long Seek(long offset, SeekOrigin origin) + { + return this.stream.Seek(offset, origin); + } + + public override void SetLength(long value) + { + this.stream.SetLength(value); + } + + public override void Write(byte[] buffer, int offset, int count) + { + this.stream.Write(buffer, offset, count); + } + + public override void WriteByte(byte value) + { + this.stream.WriteByte(value); + } + } +} diff --git a/src/WixToolset.Data/ObjectField.cs b/src/WixToolset.Data/ObjectField.cs new file mode 100644 index 00000000..42ef111b --- /dev/null +++ b/src/WixToolset.Data/ObjectField.cs @@ -0,0 +1,183 @@ +// 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.Data +{ + using System; + using System.Diagnostics; + using System.Globalization; + using System.Xml; + + /// + /// Field containing data for an object column in a row. + /// + public sealed class ObjectField : Field + { + /// + /// Instantiates a new Field. + /// + /// Column definition for this field. + internal ObjectField(ColumnDefinition columnDefinition) : + base(columnDefinition) + { + } + + /// + /// Gets or sets the index of the embedded file in a library. + /// + /// The index of the embedded file. + public int? EmbeddedFileIndex { get; set; } + + /// + /// Gets or sets the previous index of the embedded file in the library. + /// + /// The previous index of the embedded file. + public int? PreviousEmbeddedFileIndex { get; set; } + + /// + /// Gets or sets the path to the embedded cabinet of the previous file. + /// + /// The path of the cabinet containing the previous file. + public Uri PreviousBaseUri { get; set; } + + /// + /// Gets the base URI of the object field. + /// + /// The base URI of the object field. + public Uri BaseUri { get; private set; } + + /// + /// Gets or sets the unresolved data for this field. + /// + /// Unresolved Data in the field. + public string UnresolvedData { get; set; } + + /// + /// Gets or sets the unresolved previous data. + /// + /// The unresolved previous data. + public string UnresolvedPreviousData { get; set; } + + /// + /// Parse a field from the xml. + /// + /// XmlReader where the intermediate is persisted. + internal override void Read(XmlReader reader) + { + Debug.Assert("field" == reader.LocalName); + + bool empty = reader.IsEmptyElement; + + this.BaseUri = new Uri(reader.BaseURI); + + while (reader.MoveToNextAttribute()) + { + switch (reader.LocalName) + { + case "cabinetFileId": + this.EmbeddedFileIndex = Convert.ToInt32(reader.Value); + break; + case "modified": + this.Modified = reader.Value.Equals("yes"); + break; + case "previousData": + this.PreviousData = reader.Value; + break; + case "unresolvedPreviousData": + this.UnresolvedPreviousData = reader.Value; + break; + case "unresolvedData": + this.UnresolvedData = reader.Value; + break; + case "previousCabinetFileId": + this.PreviousEmbeddedFileIndex = Convert.ToInt32(reader.Value); + break; + } + } + + if (!empty) + { + bool done = false; + + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + throw new XmlException(); + case XmlNodeType.CDATA: + case XmlNodeType.Text: + if (0 < reader.Value.Length) + { + this.Data = reader.Value; + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + } + + /// + /// Persists a field in an XML format. + /// + /// XmlWriter where the Field should persist itself as XML. + internal override void Write(XmlWriter writer) + { + writer.WriteStartElement("field", Intermediate.XmlNamespaceUri); + + if (this.EmbeddedFileIndex.HasValue) + { + writer.WriteStartAttribute("cabinetFileId"); + writer.WriteValue(this.EmbeddedFileIndex); + writer.WriteEndAttribute(); + } + + if (this.Modified) + { + writer.WriteAttributeString("modified", "yes"); + } + + if (null != this.UnresolvedPreviousData) + { + writer.WriteAttributeString("unresolvedPreviousData", this.UnresolvedPreviousData); + } + + if (null != this.PreviousData) + { + writer.WriteAttributeString("previousData", this.PreviousData); + } + + if (null != this.UnresolvedData) + { + writer.WriteAttributeString("unresolvedData", this.UnresolvedData); + } + + if (this.PreviousEmbeddedFileIndex.HasValue) + { + writer.WriteStartAttribute("previousCabinetFileId"); + writer.WriteValue(this.PreviousEmbeddedFileIndex); + writer.WriteEndAttribute(); + } + + // Convert the data to a string that will persist nicely (nulls as String.Empty). + string text = Convert.ToString(this.Data, CultureInfo.InvariantCulture); + if (this.Column.UseCData) + { + writer.WriteCData(text); + } + else + { + writer.WriteString(text); + } + + writer.WriteEndElement(); + } + } +} diff --git a/src/WixToolset.Data/Output.cs b/src/WixToolset.Data/Output.cs new file mode 100644 index 00000000..b2a21c6c --- /dev/null +++ b/src/WixToolset.Data/Output.cs @@ -0,0 +1,394 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + using System.Globalization; + using System.IO; + using System.Linq; + using System.Xml; + + /// + /// Output is generated by the linker. + /// + public sealed class Output + { + public const string XmlNamespaceUri = "http://wixtoolset.org/schemas/v4/wixout"; + private static readonly Version CurrentVersion = new Version("4.0.0.0"); + + private Section entrySection; + + /// + /// Creates a new empty output object. + /// + /// The source line information for the output. + public Output(SourceLineNumber sourceLineNumbers) + { + this.Sections = new List
(); + this.SourceLineNumbers = sourceLineNumbers; + this.SubStorages = new List(); + this.Tables = new TableIndexedCollection(); + } + + /// + /// Gets the entry section for the output + /// + /// Entry section for the output. + public Section EntrySection + { + get + { + return this.entrySection; + } + + set + { + this.entrySection = value; + this.Codepage = value.Codepage; + + switch (this.entrySection.Type) + { + case SectionType.Bundle: + this.Type = OutputType.Bundle; + break; + case SectionType.Product: + this.Type = OutputType.Product; + break; + case SectionType.Module: + this.Type = OutputType.Module; + break; + case SectionType.PatchCreation: + this.Type = OutputType.PatchCreation; + break; + case SectionType.Patch: + this.Type = OutputType.Patch; + break; + default: + throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, WixDataStrings.EXP_UnexpectedEntrySectionType, this.entrySection.Type)); + } + } + } + + /// + /// Gets the type of the output. + /// + /// Type of the output. + public OutputType Type { get; set; } + + /// + /// Gets or sets the codepage for this output. + /// + /// Codepage of the output. + public int Codepage { get; set; } + + /// + /// Gets the sections contained in the output. + /// + /// Sections in the output. + public ICollection
Sections { get; private set; } + + /// + /// Gets the source line information for this output. + /// + /// The source line information for this output. + public SourceLineNumber SourceLineNumbers { get; private set; } + + /// + /// Gets the substorages in this output. + /// + /// The substorages in this output. + public ICollection SubStorages { get; private set; } + + /// + /// Gets the tables contained in this output. + /// + /// Collection of tables. + public TableIndexedCollection Tables { get; private set; } + + /// + /// Gets the output type corresponding to a given output filename extension. + /// + /// Case-insensitive output filename extension. + /// Output type for the extension. + public static OutputType GetOutputType(string extension) + { + if (extension.Equals(".exe", StringComparison.OrdinalIgnoreCase)) + { + return OutputType.Bundle; + } + if (extension.Equals(".msi", StringComparison.OrdinalIgnoreCase)) + { + return OutputType.Product; + } + else if (extension.Equals(".msm", StringComparison.OrdinalIgnoreCase)) + { + return OutputType.Module; + } + else if (extension.Equals(".msp", StringComparison.OrdinalIgnoreCase)) + { + return OutputType.Patch; + } + else if (extension.Equals(".mst", StringComparison.OrdinalIgnoreCase)) + { + return OutputType.Transform; + } + else if (extension.Equals(".pcp", StringComparison.OrdinalIgnoreCase)) + { + return OutputType.PatchCreation; + } + else + { + return OutputType.Unknown; + } + } + + /// + /// Gets the filename extension corresponding to a given output type. + /// + /// One of the WiX output types. + /// Filename extension for the output type, for example ".msi". + public static string GetExtension(OutputType type) + { + switch (type) + { + case OutputType.Bundle: + return ".exe"; + case OutputType.Product: + return ".msi"; + case OutputType.Module: + return ".msm"; + case OutputType.Patch: + return ".msp"; + case OutputType.Transform: + return ".mst"; + case OutputType.PatchCreation: + return ".pcp"; + default: + return ".wix"; + } + } + + /// + /// Loads an output from a path on disk. + /// + /// Path to output file saved on disk. + /// Suppresses wix.dll version mismatch check. + /// Output object. + public static Output Load(string path, bool suppressVersionCheck) + { + using (FileStream stream = File.OpenRead(path)) + using (FileStructure fs = FileStructure.Read(stream)) + { + if (FileFormat.Wixout != fs.FileFormat) + { + throw new WixUnexpectedFileFormatException(path, FileFormat.Wixout, fs.FileFormat); + } + + Uri uri = new Uri(Path.GetFullPath(path)); + using (XmlReader reader = XmlReader.Create(fs.GetDataStream(), null, uri.AbsoluteUri)) + { + try + { + reader.MoveToContent(); + return Output.Read(reader, suppressVersionCheck); + } + catch (XmlException xe) + { + throw new WixCorruptFileException(path, fs.FileFormat, xe); + } + } + } + } + + /// + /// Saves an output to a path on disk. + /// + /// Path to save output file to on disk. + public void Save(string path) + { + Directory.CreateDirectory(Path.GetDirectoryName(Path.GetFullPath(path))); + + using (FileStream stream = File.Create(path)) + using (FileStructure fs = FileStructure.Create(stream, FileFormat.Wixout, null)) + using (XmlWriter writer = XmlWriter.Create(fs.GetDataStream())) + { + writer.WriteStartDocument(); + this.Write(writer); + writer.WriteEndDocument(); + } + } + + /// + /// Processes an XmlReader and builds up the output object. + /// + /// Reader to get data from. + /// Suppresses wix.dll version mismatch check. + /// The Output represented by the Xml. + internal static Output Read(XmlReader reader, bool suppressVersionCheck) + { + if (!reader.LocalName.Equals("wixOutput")) + { + throw new XmlException(); + } + + bool empty = reader.IsEmptyElement; + Output output = new Output(SourceLineNumber.CreateFromUri(reader.BaseURI)); + SectionType sectionType = SectionType.Unknown; + Version version = null; + + while (reader.MoveToNextAttribute()) + { + switch (reader.LocalName) + { + case "codepage": + output.Codepage = Convert.ToInt32(reader.Value, CultureInfo.InvariantCulture.NumberFormat); + break; + case "type": + switch (reader.Value) + { + case "Bundle": + output.Type = OutputType.Bundle; + sectionType = SectionType.Bundle; + break; + case "Module": + output.Type = OutputType.Module; + sectionType = SectionType.Module; + break; + case "Patch": + output.Type = OutputType.Patch; + break; + case "PatchCreation": + output.Type = OutputType.PatchCreation; + sectionType = SectionType.PatchCreation; + break; + case "Product": + output.Type = OutputType.Product; + sectionType = SectionType.Product; + break; + case "Transform": + output.Type = OutputType.Transform; + break; + default: + throw new XmlException(); + } + break; + case "version": + version = new Version(reader.Value); + break; + } + } + + if (!suppressVersionCheck && null != version && !Output.CurrentVersion.Equals(version)) + { + throw new WixException(WixDataErrors.VersionMismatch(SourceLineNumber.CreateFromUri(reader.BaseURI), "wixOutput", version.ToString(), Output.CurrentVersion.ToString())); + } + + // create a section for all the rows to belong to + output.entrySection = new Section(null, sectionType, output.Codepage); + + // loop through the rest of the xml building up the Output object + TableDefinitionCollection tableDefinitions = null; + List tables = new List
(); + if (!empty) + { + bool done = false; + + // loop through all the fields in a row + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + switch (reader.LocalName) + { + case "subStorage": + output.SubStorages.Add(SubStorage.Read(reader)); + break; + case "table": + if (null == tableDefinitions) + { + throw new XmlException(); + } + tables.Add(Table.Read(reader, output.entrySection, tableDefinitions)); + break; + case "tableDefinitions": + tableDefinitions = TableDefinitionCollection.Read(reader); + break; + default: + throw new XmlException(); + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + + output.Tables = new TableIndexedCollection(tables); + return output; + } + + /// + /// Ensure this output contains a particular table. + /// + /// Definition of the table that should exist. + /// Optional section to use for the table. If one is not provided, the entry section will be used. + /// The table in this output. + public Table EnsureTable(TableDefinition tableDefinition, Section section = null) + { + Table table; + if (!this.Tables.TryGetTable(tableDefinition.Name, out table)) + { + table = new Table(section ?? this.entrySection, tableDefinition); + this.Tables.Add(table); + } + + return table; + } + + /// + /// Persists an output in an XML format. + /// + /// XmlWriter where the Output should persist itself as XML. + internal void Write(XmlWriter writer) + { + writer.WriteStartElement("wixOutput", XmlNamespaceUri); + + writer.WriteAttributeString("type", this.Type.ToString()); + + if (0 != this.Codepage) + { + writer.WriteAttributeString("codepage", this.Codepage.ToString(CultureInfo.InvariantCulture)); + } + + writer.WriteAttributeString("version", Output.CurrentVersion.ToString()); + + // Collect all the table definitions and write them. + TableDefinitionCollection tableDefinitions = new TableDefinitionCollection(); + foreach (Table table in this.Tables) + { + tableDefinitions.Add(table.Definition); + } + tableDefinitions.Write(writer); + + foreach (Table table in this.Tables.OrderBy(t => t.Name)) + { + table.Write(writer); + } + + foreach (SubStorage subStorage in this.SubStorages) + { + subStorage.Write(writer); + } + + writer.WriteEndElement(); + } + } +} diff --git a/src/WixToolset.Data/OutputType.cs b/src/WixToolset.Data/OutputType.cs new file mode 100644 index 00000000..74381737 --- /dev/null +++ b/src/WixToolset.Data/OutputType.cs @@ -0,0 +1,31 @@ +// 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.Data +{ + /// + /// Various types of output. + /// + public enum OutputType + { + /// Unknown output type. + Unknown, + + /// Bundle output type. + Bundle, + + /// Module output type. + Module, + + /// Patch output type. + Patch, + + /// Patch Creation output type. + PatchCreation, + + /// Product output type. + Product, + + /// Transform output type. + Transform + } +} diff --git a/src/WixToolset.Data/PackagingType.cs b/src/WixToolset.Data/PackagingType.cs new file mode 100644 index 00000000..a50f1236 --- /dev/null +++ b/src/WixToolset.Data/PackagingType.cs @@ -0,0 +1,11 @@ +// 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.Data +{ + public enum PackagingType + { + Unknown, + Embedded, + External, + } +} diff --git a/src/WixToolset.Data/Pdb.cs b/src/WixToolset.Data/Pdb.cs new file mode 100644 index 00000000..03c3ddbb --- /dev/null +++ b/src/WixToolset.Data/Pdb.cs @@ -0,0 +1,163 @@ +// 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.Data +{ + using System; + using System.IO; + using System.Xml; + + /// + /// Pdb generated by the binder. + /// + public sealed class Pdb + { + public const string XmlNamespaceUri = "http://wixtoolset.org/schemas/v4/wixpdb"; + private static readonly Version CurrentVersion = new Version("4.0.0.0"); + + /// + /// Creates a new empty pdb object. + /// + /// The source line information for the pdb. + public Pdb() + { + } + + /// + /// Gets or sets the output that is a part of this pdb. + /// + /// Type of the output. + public Output Output { get; set; } + + /// + /// Loads a pdb from a path on disk. + /// + /// Path to pdb file saved on disk. + /// Suppresses wix.dll version mismatch check. + /// Pdb pdb. + public static Pdb Load(string path, bool suppressVersionCheck) + { + using (FileStream stream = File.OpenRead(path)) + using (FileStructure fs = FileStructure.Read(stream)) + { + if (FileFormat.Wixpdb != fs.FileFormat) + { + throw new WixUnexpectedFileFormatException(path, FileFormat.Wixpdb, fs.FileFormat); + } + + Uri uri = new Uri(Path.GetFullPath(path)); + using (XmlReader reader = XmlReader.Create(fs.GetDataStream(), null, uri.AbsoluteUri)) + { + try + { + reader.MoveToContent(); + return Pdb.Read(reader, suppressVersionCheck); + } + catch (XmlException xe) + { + throw new WixCorruptFileException(path, fs.FileFormat, xe); + } + } + } + } + + /// + /// Saves a pdb to a path on disk. + /// + /// Path to save pdb file to on disk. + public void Save(string path) + { + Directory.CreateDirectory(Path.GetDirectoryName(Path.GetFullPath(path))); + + using (FileStream stream = File.Create(path)) + using (FileStructure fs = FileStructure.Create(stream, FileFormat.Wixpdb, null)) + using (XmlWriter writer = XmlWriter.Create(fs.GetDataStream())) + { + writer.WriteStartDocument(); + this.Write(writer); + writer.WriteEndDocument(); + } + } + + /// + /// Processes an XmlReader and builds up the pdb object. + /// + /// Reader to get data from. + /// Suppresses wix.dll version mismatch check. + /// The Pdb represented by the Xml. + internal static Pdb Read(XmlReader reader, bool suppressVersionCheck) + { + if ("wixPdb" != reader.LocalName) + { + throw new XmlException(); + } + + bool empty = reader.IsEmptyElement; + Pdb pdb = new Pdb(); + Version version = null; + + while (reader.MoveToNextAttribute()) + { + switch (reader.LocalName) + { + case "version": + version = new Version(reader.Value); + break; + } + } + + if (!suppressVersionCheck && null != version && !Pdb.CurrentVersion.Equals(version)) + { + throw new WixException(WixDataErrors.VersionMismatch(SourceLineNumber.CreateFromUri(reader.BaseURI), "wixPdb", version.ToString(), Pdb.CurrentVersion.ToString())); + } + + // loop through the rest of the pdb building up the Output object + if (!empty) + { + bool done = false; + + // loop through all the fields in a row + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + switch (reader.LocalName) + { + case "wixOutput": + pdb.Output = Output.Read(reader, suppressVersionCheck); + break; + default: + throw new XmlException(); + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + + return pdb; + } + + /// + /// Persists a pdb in an XML format. + /// + /// XmlWriter where the Pdb should persist itself as XML. + internal void Write(XmlWriter writer) + { + writer.WriteStartElement("wixPdb", XmlNamespaceUri); + + writer.WriteAttributeString("version", Pdb.CurrentVersion.ToString()); + + this.Output.Write(writer); + + writer.WriteEndElement(); + } + } +} diff --git a/src/WixToolset.Data/Platform.cs b/src/WixToolset.Data/Platform.cs new file mode 100644 index 00000000..4edc9af0 --- /dev/null +++ b/src/WixToolset.Data/Platform.cs @@ -0,0 +1,22 @@ +// 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.Data +{ + /// + /// Platforms supported by compiler. + /// + public enum Platform + { + /// x86. + X86, + + /// x64. + X64, + + /// ia64. + IA64, + + /// arm. + ARM + } +} diff --git a/src/WixToolset.Data/Row.cs b/src/WixToolset.Data/Row.cs new file mode 100644 index 00000000..4ebddc06 --- /dev/null +++ b/src/WixToolset.Data/Row.cs @@ -0,0 +1,626 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + using System.Diagnostics; + using System.Diagnostics.CodeAnalysis; + using System.Globalization; + using System.Text; + using System.Text.RegularExpressions; + using System.Xml; + + /// + /// Row containing data for a table. + /// + public class Row + { + private static long rowCount; + + private Field[] fields; + + /// + /// Creates a row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this row belongs to and should get its column definitions from. + /// The compiler should use this constructor exclusively. + public Row(SourceLineNumber sourceLineNumbers, Table table) + : this(sourceLineNumbers, table.Definition) + { + this.Table = table; + } + + /// + /// Creates a row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this row should get its column definitions from. + /// This constructor is used in cases where there isn't a clear owner of the row. The linker uses this constructor for the rows it generates. + public Row(SourceLineNumber sourceLineNumbers, TableDefinition tableDefinition) + { + this.Number = rowCount++; + this.SourceLineNumbers = sourceLineNumbers; + this.fields = new Field[tableDefinition.Columns.Count]; + this.TableDefinition = tableDefinition; + + for (int i = 0; i < this.fields.Length; ++i) + { + this.fields[i] = Field.Create(this.TableDefinition.Columns[i]); + } + } + + /// + /// Creates a shallow copy of a row from another row. + /// + /// The row the data is copied from. + protected Row(Row source) + { + this.Table = source.Table; + this.TableDefinition = source.TableDefinition; + this.Number = source.Number; + this.Access = source.Access; + this.Operation = source.Operation; + this.Redundant = source.Redundant; + this.SectionId = source.SectionId; + this.SourceLineNumbers = source.SourceLineNumbers; + this.fields = source.fields; + } + + /// + /// Gets or sets the access to the row's primary key. + /// + /// The row access modifier. + public AccessModifier Access { get; set; } + + /// + /// Gets or sets the row transform operation. + /// + /// The row transform operation. + public RowOperation Operation { get; set; } + + /// + /// Gets or sets wether the row is a duplicate of another row thus redundant. + /// + public bool Redundant { get; set; } + + /// + /// Gets the section for the row. + /// + /// Section for the row. + public Section Section { get { return (null == this.Table) ? null : this.Table.Section; } } + + /// + /// Gets or sets the SectionId property on the row. + /// + /// The SectionId property on the row. + public string SectionId { get; set; } + + /// + /// Gets the source file and line number for the row. + /// + /// Source file and line number. + public SourceLineNumber SourceLineNumbers { get; private set; } + + /// + /// Gets the table this row belongs to. + /// + /// null if Row does not belong to a Table, or owner Table otherwise. + public Table Table { get; private set; } + + /// + /// Gets the table definition for this row. + /// + /// A Row always has a TableDefinition, even if the Row does not belong to a Table. + /// TableDefinition for Row. + public TableDefinition TableDefinition { get; private set; } + + /// + /// Gets the fields contained by this row. + /// + /// Array of field objects + [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Performance", "CA1819:PropertiesShouldNotReturnArrays")] + public Field[] Fields + { + get { return this.fields; } + } + + /// + /// Gets the unique number for the row. + /// + /// Number for row. + public long Number { get; private set; } + + /// + /// Gets or sets the value of a particular field in the row. + /// + /// field index. + /// Value of a field in the row. + public object this[int field] + { + get { return this.fields[field].Data; } + set { this.fields[field].Data = value; } + } + + /// + /// Gets the field as an integer. + /// + /// Field's data as an integer. + public int FieldAsInteger(int field) + { + return this.fields[field].AsInteger(); + } + + /// + /// Gets the field as an integer that could be null. + /// + /// Field's data as an integer that could be null. + public int? FieldAsNullableInteger(int field) + { + return this.fields[field].AsNullableInteger(); + } + + /// + /// Gets the field as a string. + /// + /// Field's data as a string. + public string FieldAsString(int field) + { + return this.fields[field].AsString(); + } + + /// + /// Sets the value of a particular field in the row without validating. + /// + /// field index. + /// Value of a field in the row. + /// True if successful, false if validation failed. + public bool BestEffortSetField(int field, object value) + { + return this.fields[field].BestEffortSet(value); + } + + /// + /// Get the value used to represent the row in a keyed row collection. + /// + /// Primary key or row number if no primary key is available. + public string GetKey() + { + return this.GetPrimaryKey() ?? Convert.ToString(this.Number, CultureInfo.InvariantCulture); + } + + /// + /// Get the primary key of this row. + /// + /// Delimiter character for multiple column primary keys. + /// The primary key or null if the row's table has no primary key columns. + public string GetPrimaryKey(char delimiter = '/') + { + return this.GetPrimaryKey(delimiter, String.Empty); + } + + /// + /// Get the primary key of this row. + /// + /// Delimiter character for multiple column primary keys. + /// String to represent null values in the primary key. + /// The primary key or null if the row's table has no primary key columns. + public string GetPrimaryKey(char delimiter, string nullReplacement) + { + bool foundPrimaryKey = false; + StringBuilder primaryKey = new StringBuilder(); + + foreach (Field field in this.fields) + { + if (field.Column.PrimaryKey) + { + if (foundPrimaryKey) + { + primaryKey.Append(delimiter); + } + + primaryKey.Append((null == field.Data) ? nullReplacement : Convert.ToString(field.Data, CultureInfo.InvariantCulture)); + + foundPrimaryKey = true; + } + else // primary keys must be the first columns of a row so the first non-primary key means we can stop looking. + { + break; + } + } + + return foundPrimaryKey ? primaryKey.ToString() : null; + } + + /// + /// Returns true if the specified field is null or an empty string. + /// + /// Index of the field to check. + /// true if the specified field is null or an empty string, false otherwise. + public bool IsColumnEmpty(int field) + { + if (null == this.fields[field].Data) + { + return true; + } + + string dataString = this.fields[field].Data as string; + if (null != dataString && 0 == dataString.Length) + { + return true; + } + + return false; + } + + /// + /// Tests if the passed in row is identical. + /// + /// Row to compare against. + /// True if two rows are identical. + public bool IsIdentical(Row row) + { + bool identical = (this.TableDefinition.Name == row.TableDefinition.Name && this.fields.Length == row.fields.Length); + + for (int i = 0; identical && i < this.fields.Length; ++i) + { + if (!(this.fields[i].IsIdentical(row.fields[i]))) + { + identical = false; + } + } + + return identical; + } + + /// + /// Returns a string representation of the Row. + /// + /// A string representation of the Row. + public override string ToString() + { + return String.Join("/", (object[])this.fields); + } + + /// + /// Creates a Row from the XmlReader. + /// + /// Reader to get data from. + /// Table for this row. + /// New row object. + internal static Row Read(XmlReader reader, Table table) + { + Debug.Assert("row" == reader.LocalName); + + bool empty = reader.IsEmptyElement; + AccessModifier access = AccessModifier.Public; + RowOperation operation = RowOperation.None; + bool redundant = false; + string sectionId = null; + SourceLineNumber sourceLineNumbers = null; + + while (reader.MoveToNextAttribute()) + { + switch (reader.LocalName) + { + case "access": + access = (AccessModifier)Enum.Parse(typeof(AccessModifier), reader.Value, true); + break; + case "op": + operation = (RowOperation)Enum.Parse(typeof(RowOperation), reader.Value, true); + break; + case "redundant": + redundant = reader.Value.Equals("yes"); + break; + case "sectionId": + sectionId = reader.Value; + break; + case "sourceLineNumber": + sourceLineNumbers = SourceLineNumber.CreateFromEncoded(reader.Value); + break; + } + } + + Row row = table.CreateRow(sourceLineNumbers); + row.Access = access; + row.Operation = operation; + row.Redundant = redundant; + row.SectionId = sectionId; + + // loop through all the fields in a row + if (!empty) + { + bool done = false; + int field = 0; + + // loop through all the fields in a row + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + switch (reader.LocalName) + { + case "field": + if (row.Fields.Length <= field) + { + if (!reader.IsEmptyElement) + { + throw new XmlException(); + } + } + else + { + row.fields[field].Read(reader); + } + ++field; + break; + default: + throw new XmlException(); + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + + return row; + } + + /// + /// Returns the row in a format usable in IDT files. + /// + /// Whether to keep columns added in a transform. + /// String with tab delimited field values. + internal string ToIdtDefinition(bool keepAddedColumns) + { + bool first = true; + StringBuilder sb = new StringBuilder(); + + foreach (Field field in this.fields) + { + // Conditionally keep columns added in a transform; otherwise, + // break because columns can only be added at the end. + if (field.Column.Added && !keepAddedColumns) + { + break; + } + + if (first) + { + first = false; + } + else + { + sb.Append('\t'); + } + + sb.Append(field.ToIdtValue()); + } + sb.Append("\r\n"); + + return sb.ToString(); + } + + /// + /// Gets the modularized version of the field data. + /// + /// The field to modularize. + /// String containing the GUID of the Merge Module to append the the field value, if appropriate. + /// Optional collection of identifiers that should not be modularized. + /// moduleGuid is expected to be null when not being used to compile a Merge Module. + /// The modularized version of the field data. + internal string GetModularizedValue(Field field, string modularizationGuid, ISet suppressModularizationIdentifiers) + { + Debug.Assert(null != field.Data && 0 < ((string)field.Data).Length); + string fieldData = Convert.ToString(field.Data, CultureInfo.InvariantCulture); + + if (null != modularizationGuid && ColumnModularizeType.None != field.Column.ModularizeType && !(WindowsInstallerStandard.IsStandardAction(fieldData) || WindowsInstallerStandard.IsStandardProperty(fieldData))) + { + StringBuilder sb; + int start; + ColumnModularizeType modularizeType = field.Column.ModularizeType; + + // special logic for the ControlEvent table's Argument column + // this column requires different modularization methods depending upon the value of the Event column + if (ColumnModularizeType.ControlEventArgument == field.Column.ModularizeType) + { + switch (this[2].ToString()) + { + case "CheckExistingTargetPath": // redirectable property name + case "CheckTargetPath": + case "DoAction": // custom action name + case "NewDialog": // dialog name + case "SelectionBrowse": + case "SetTargetPath": + case "SpawnDialog": + case "SpawnWaitDialog": + if (Common.IsIdentifier(fieldData)) + { + modularizeType = ColumnModularizeType.Column; + } + else + { + modularizeType = ColumnModularizeType.Property; + } + break; + default: // formatted + modularizeType = ColumnModularizeType.Property; + break; + } + } + else if (ColumnModularizeType.ControlText == field.Column.ModularizeType) + { + // icons are stored in the Binary table, so they get column-type modularization + if (("Bitmap" == this[2].ToString() || "Icon" == this[2].ToString()) && Common.IsIdentifier(fieldData)) + { + modularizeType = ColumnModularizeType.Column; + } + else + { + modularizeType = ColumnModularizeType.Property; + } + } + + switch (modularizeType) + { + case ColumnModularizeType.Column: + // ensure the value is an identifier (otherwise it shouldn't be modularized this way) + if (!Common.IsIdentifier(fieldData)) + { + throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, WixDataStrings.EXP_CannotModularizeIllegalID, fieldData)); + } + + // if we're not supposed to suppress modularization of this identifier + if (null == suppressModularizationIdentifiers || !suppressModularizationIdentifiers.Contains(fieldData)) + { + fieldData = String.Concat(fieldData, ".", modularizationGuid); + } + break; + + case ColumnModularizeType.Property: + case ColumnModularizeType.Condition: + Regex regex; + if (ColumnModularizeType.Property == modularizeType) + { + regex = new Regex(@"\[(?[#$!]?[a-zA-Z_][a-zA-Z0-9_\.]*)]", RegexOptions.Singleline | RegexOptions.ExplicitCapture); + } + else + { + Debug.Assert(ColumnModularizeType.Condition == modularizeType); + + // This heinous looking regular expression is actually quite an elegant way + // to shred the entire condition into the identifiers that need to be + // modularized. Let's break it down piece by piece: + // + // 1. Look for the operators: NOT, EQV, XOR, OR, AND, IMP (plus a space). Note that the + // regular expression is case insensitive so we don't have to worry about + // all the permutations of these strings. + // 2. Look for quoted strings. Quoted strings are just text and are ignored + // outright. + // 3. Look for environment variables. These look like identifiers we might + // otherwise be interested in but start with a percent sign. Like quoted + // strings these enviroment variable references are ignored outright. + // 4. Match all identifiers that are things that need to be modularized. Note + // the special characters (!, $, ?, &) that denote Component and Feature states. + regex = new Regex(@"NOT\s|EQV\s|XOR\s|OR\s|AND\s|IMP\s|"".*?""|%[a-zA-Z_][a-zA-Z0-9_\.]*|(?[!$\?&]?[a-zA-Z_][a-zA-Z0-9_\.]*)", RegexOptions.Singleline | RegexOptions.IgnoreCase | RegexOptions.ExplicitCapture); + + // less performant version of the above with captures showing where everything lives + // regex = new Regex(@"(?NOT|EQV|XOR|OR|AND|IMP)|(?"".*?"")|(?%[a-zA-Z_][a-zA-Z0-9_\.]*)|(?[!$\?&]?[a-zA-Z_][a-zA-Z0-9_\.]*)",RegexOptions.Singleline | RegexOptions.IgnoreCase | RegexOptions.ExplicitCapture); + } + + MatchCollection matches = regex.Matches(fieldData); + + sb = new StringBuilder(fieldData); + + // notice how this code walks backward through the list + // because it modifies the string as we through it + for (int i = matches.Count - 1; 0 <= i; i--) + { + Group group = matches[i].Groups["identifier"]; + if (group.Success) + { + string identifier = group.Value; + if (!WindowsInstallerStandard.IsStandardProperty(identifier) && (null == suppressModularizationIdentifiers || !suppressModularizationIdentifiers.Contains(identifier))) + { + sb.Insert(group.Index + group.Length, '.'); + sb.Insert(group.Index + group.Length + 1, modularizationGuid); + } + } + } + + fieldData = sb.ToString(); + break; + + case ColumnModularizeType.CompanionFile: + // if we're not supposed to ignore this identifier and the value does not start with + // a digit, we must have a companion file so modularize it + if ((null == suppressModularizationIdentifiers || !suppressModularizationIdentifiers.Contains(fieldData)) && + 0 < fieldData.Length && !Char.IsDigit(fieldData, 0)) + { + fieldData = String.Concat(fieldData, ".", modularizationGuid); + } + break; + + case ColumnModularizeType.Icon: + if (null == suppressModularizationIdentifiers || !suppressModularizationIdentifiers.Contains(fieldData)) + { + start = fieldData.LastIndexOf(".", StringComparison.Ordinal); + if (-1 == start) + { + fieldData = String.Concat(fieldData, ".", modularizationGuid); + } + else + { + fieldData = String.Concat(fieldData.Substring(0, start), ".", modularizationGuid, fieldData.Substring(start)); + } + } + break; + + case ColumnModularizeType.SemicolonDelimited: + string[] keys = fieldData.Split(';'); + for (int i = 0; i < keys.Length; ++i) + { + keys[i] = String.Concat(keys[i], ".", modularizationGuid); + } + fieldData = String.Join(";", keys); + break; + } + } + + return fieldData; + } + + /// + /// Persists a row in an XML format. + /// + /// XmlWriter where the Row should persist itself as XML. + [SuppressMessage("Microsoft.Globalization", "CA1308:NormalizeStringsToUppercase", Justification = "Changing the way this string normalizes would result " + + "in a change to the way intermediate files are generated, potentially causing extra churn in patches on an MSI built from an older version of WiX. " + + "Furthermore, there is no security hole here, as the strings won't need to make a round trip")] + internal void Write(XmlWriter writer) + { + writer.WriteStartElement("row", Intermediate.XmlNamespaceUri); + + if (AccessModifier.Public != this.Access) + { + writer.WriteAttributeString("access", this.Access.ToString().ToLowerInvariant()); + } + + if (RowOperation.None != this.Operation) + { + writer.WriteAttributeString("op", this.Operation.ToString().ToLowerInvariant()); + } + + if (this.Redundant) + { + writer.WriteAttributeString("redundant", "yes"); + } + + if (null != this.SectionId) + { + writer.WriteAttributeString("sectionId", this.SectionId); + } + + if (null != this.SourceLineNumbers) + { + writer.WriteAttributeString("sourceLineNumber", this.SourceLineNumbers.GetEncoded()); + } + + for (int i = 0; i < this.fields.Length; ++i) + { + this.fields[i].Write(writer); + } + + writer.WriteEndElement(); + } + } +} diff --git a/src/WixToolset.Data/RowDictionary.cs b/src/WixToolset.Data/RowDictionary.cs new file mode 100644 index 00000000..a0cc5302 --- /dev/null +++ b/src/WixToolset.Data/RowDictionary.cs @@ -0,0 +1,84 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + + /// + /// A dictionary of rows. Unlike the this + /// will throw when multiple rows with the same key are added. + /// + public sealed class RowDictionary : Dictionary where T : Row + { + /// + /// Creates an empty . + /// + public RowDictionary() + : base(StringComparer.InvariantCulture) + { + } + + /// + /// Creates and populates a with the rows from the given enumerator. + /// + /// Rows to add. + public RowDictionary(IEnumerable rows) + : this() + { + foreach (T row in rows) + { + this.Add(row); + } + } + + /// + /// Creates and populates a with the rows from the given . + /// + /// The table to index. + /// + /// Rows added to the index are not automatically added to the given . + /// + public RowDictionary(Table table) + : this() + { + if (null != table) + { + foreach (T row in table.Rows) + { + this.Add(row); + } + } + } + + /// + /// Adds a row to the dictionary using the row key. + /// + /// Row to add to the dictionary. + public void Add(T row) + { + this.Add(row.GetKey(), row); + } + + /// + /// Gets the row by integer key. + /// + /// Integer key to look up. + /// Row or null if key is not found. + public T Get(int key) + { + return this.Get(key.ToString()); + } + + /// + /// Gets the row by string key. + /// + /// String key to look up. + /// Row or null if key is not found. + public T Get(string key) + { + T result; + return this.TryGetValue(key, out result) ? result : null; + } + } +} diff --git a/src/WixToolset.Data/RowIndexedList.cs b/src/WixToolset.Data/RowIndexedList.cs new file mode 100644 index 00000000..27c43a81 --- /dev/null +++ b/src/WixToolset.Data/RowIndexedList.cs @@ -0,0 +1,301 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + + /// + /// A list of rows indexed by their primary key. Unlike a + /// this indexed list will track rows in their added order and will allow rows with + /// duplicate keys to be added to the list, although only the first row will be indexed. + /// + public sealed class RowIndexedList : IList where T : Row + { + private Dictionary index; + private List rows; + private List duplicates; + + /// + /// Creates an empty . + /// + public RowIndexedList() + { + this.index = new Dictionary(StringComparer.InvariantCulture); + this.rows = new List(); + this.duplicates = new List(); + } + + /// + /// Creates and populates a with the rows from the given enumerator. + /// + /// Rows to index. + public RowIndexedList(IEnumerable rows) + : this() + { + foreach (T row in rows) + { + this.Add(row); + } + } + + /// + /// Creates and populates a with the rows from the given . + /// + /// The table to index. + /// + /// Rows added to the index are not automatically added to the given . + /// + public RowIndexedList(Table table) + : this() + { + if (null != table) + { + foreach (T row in table.Rows) + { + this.Add(row); + } + } + } + + /// + /// Gets the duplicates in the list. + /// + public IEnumerable Duplicates { get { return this.duplicates; } } + + /// + /// Gets the row by integer key. + /// + /// Integer key to look up. + /// Row or null if key is not found. + public T Get(int key) + { + return this.Get(key.ToString()); + } + + /// + /// Gets the row by string key. + /// + /// String key to look up. + /// Row or null if key is not found. + public T Get(string key) + { + T result; + return this.TryGet(key, out result) ? result : null; + } + + /// + /// Gets the row by string key if it exists. + /// + /// Key of row to get. + /// Row found. + /// True if key was found otherwise false. + public bool TryGet(string key, out T row) + { + return this.index.TryGetValue(key, out row); + } + + /// + /// Tries to add a row as long as it would not create a duplicate. + /// + /// Row to add. + /// True if the row as added otherwise false. + public bool TryAdd(T row) + { + try + { + this.index.Add(row.GetKey(), row); + } + catch (ArgumentException) // if the key already exists, bail. + { + return false; + } + + this.rows.Add(row); + return true; + } + + /// + /// Adds a row to the list. If a row with the same key is already index, the row is + /// is not in the index but will still be part of the list and added to the duplicates + /// list. + /// + /// + public void Add(T row) + { + this.rows.Add(row); + try + { + this.index.Add(row.GetKey(), row); + } + catch (ArgumentException) // if the key already exists, we have a duplicate. + { + this.duplicates.Add(row); + } + } + + /// + /// Gets the index of a row. + /// + /// Iterates through the list of rows to find the index of a particular row. + /// Index of row or -1 if not found. + public int IndexOf(T row) + { + return this.rows.IndexOf(row); + } + + /// + /// Inserts a row at a particular index of the list. + /// + /// Index to insert the row after. + /// Row to insert. + public void Insert(int index, T row) + { + this.rows.Insert(index, row); + try + { + this.index.Add(row.GetKey(), row); + } + catch (ArgumentException) // if the key already exists, we have a duplicate. + { + this.duplicates.Add(row); + } + } + + /// + /// Removes a row from a particular index. + /// + /// Index to remove the row at. + public void RemoveAt(int index) + { + T row = this.rows[index]; + + this.rows.RemoveAt(index); + + T indexRow; + if (this.index.TryGetValue(row.GetKey(), out indexRow) && indexRow == row) + { + this.index.Remove(row.GetKey()); + } + else // only try to remove from duplicates if the row was not indexed (if it was indexed, it wasn't a dupe). + { + this.duplicates.Remove(row); + } + } + + /// + /// Gets or sets a row at the specified index. + /// + /// Index to get the row. + /// Row at specified index. + public T this[int index] + { + get + { + return this.rows[index]; + } + set + { + this.rows[index] = value; + try + { + this.index.Add(value.GetKey(), value); + } + catch (ArgumentException) // if the key already exists, we have a duplicate. + { + this.duplicates.Add(value); + } + } + } + + /// + /// Empties the list and it's index. + /// + public void Clear() + { + this.index.Clear(); + this.rows.Clear(); + this.duplicates.Clear(); + } + + /// + /// Searches the list for a row without using the index. + /// + /// Row to look for in the list. + /// True if the row is in the list, otherwise false. + public bool Contains(T row) + { + return this.rows.Contains(row); + } + + /// + /// Copies the rows of the list to an array. + /// + /// Array to copy the list into. + /// Index to start copying at. + public void CopyTo(T[] array, int arrayIndex) + { + this.rows.CopyTo(array, arrayIndex); + } + + /// + /// Number of rows in the list. + /// + public int Count + { + get { return this.rows.Count; } + } + + /// + /// Indicates whether the list is read-only. Always false. + /// + public bool IsReadOnly + { + get { return false; } + } + + /// + /// Removes a row from the list. Indexed rows will be removed but the colleciton will NOT + /// promote duplicates to the index automatically. The duplicate would also need to be removed + /// and re-added to be indexed. + /// + /// + /// + public bool Remove(T row) + { + bool removed = this.rows.Remove(row); + if (removed) + { + T indexRow; + if (this.index.TryGetValue(row.GetKey(), out indexRow) && indexRow == row) + { + this.index.Remove(row.GetKey()); + } + else // only try to remove from duplicates if the row was not indexed (if it was indexed, it wasn't a dupe). + { + this.duplicates.Remove(row); + } + } + + return removed; + } + + /// + /// Gets an enumerator over the whole list. + /// + /// List enumerator. + public IEnumerator GetEnumerator() + { + return this.rows.GetEnumerator(); + } + + /// + /// Gets an untyped enumerator over the whole list. + /// + /// Untyped list enumerator. + System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() + { + return this.rows.GetEnumerator(); + } + } +} diff --git a/src/WixToolset.Data/RowOperation.cs b/src/WixToolset.Data/RowOperation.cs new file mode 100644 index 00000000..30dadd4e --- /dev/null +++ b/src/WixToolset.Data/RowOperation.cs @@ -0,0 +1,30 @@ +// 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.Data +{ + /// + /// The row transform operations. + /// + public enum RowOperation + { + /// + /// No operation. + /// + None, + + /// + /// Added row. + /// + Add, + + /// + /// Deleted row. + /// + Delete, + + /// + /// Modified row. + /// + Modify + } +} diff --git a/src/WixToolset.Data/Rows/BBControlRow.cs b/src/WixToolset.Data/Rows/BBControlRow.cs new file mode 100644 index 00000000..d0f08662 --- /dev/null +++ b/src/WixToolset.Data/Rows/BBControlRow.cs @@ -0,0 +1,113 @@ +// 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.Data.Rows +{ + using System.Diagnostics.CodeAnalysis; + + /// + /// Specialization of a row for the Control table. + /// + public sealed class BBControlRow : Row + { + /// + /// Creates a Control row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Control row belongs to and should get its column definitions from. + public BBControlRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the dialog of the Control row. + /// + /// Primary key of the Control row. + public string Billboard + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the identifier for this Control row. + /// + /// Identifier for this Control row. + public string BBControl + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the type of the BBControl. + /// + /// Name of the BBControl. + [SuppressMessage("Microsoft.Naming", "CA1721:PropertyNamesShouldNotMatchGetMethods")] + public string Type + { + get { return this.Fields[2].AsString(); } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets the X location of the BBControl. + /// + /// X location of the BBControl. + public string X + { + get { return this.Fields[3].AsString(); } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets or sets the Y location of the BBControl. + /// + /// Y location of the BBControl. + public string Y + { + get { return this.Fields[4].AsString(); } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets or sets the width of the BBControl. + /// + /// Width of the BBControl. + public string Width + { + get { return this.Fields[5].AsString(); } + set { this.Fields[5].Data = value; } + } + + /// + /// Gets or sets the height of the BBControl. + /// + /// Height of the BBControl. + public string Height + { + get { return this.Fields[6].AsString(); } + set { this.Fields[6].Data = value; } + } + + /// + /// Gets or sets the attributes for the BBControl. + /// + /// Attributes for the BBControl. + public int Attributes + { + get { return (int)this.Fields[7].Data; } + set { this.Fields[7].Data = value; } + } + + /// + /// Gets or sets the text of the BBControl. + /// + /// Text of the BBControl. + public string Text + { + get { return (string)this.Fields[8].Data; } + set { this.Fields[8].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/ComponentRow.cs b/src/WixToolset.Data/Rows/ComponentRow.cs new file mode 100644 index 00000000..3ff10175 --- /dev/null +++ b/src/WixToolset.Data/Rows/ComponentRow.cs @@ -0,0 +1,245 @@ +// 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.Data.Rows +{ + using System; + using WixToolset.Data.Msi; + + /// + /// Specialization of a row for the Component table. + /// + public sealed class ComponentRow : Row + { + private string sourceFile; + + /// + /// Creates a Control row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Component row belongs to and should get its column definitions from. + public ComponentRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the identifier for this Component row. + /// + /// Identifier for this Component row. + public string Component + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the ComponentId for this Component row. + /// + /// guid for this Component row. + public string Guid + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the Directory_ of the Component. + /// + /// Directory of the Component. + public string Directory + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets the local only attribute of the Component. + /// + /// Local only attribute of the component. + public bool IsLocalOnly + { + get { return MsiInterop.MsidbComponentAttributesLocalOnly == ((int)this.Fields[3].Data & MsiInterop.MsidbComponentAttributesLocalOnly); } + set + { + if (value) + { + this.Fields[3].Data = (int)this.Fields[3].Data | MsiInterop.MsidbComponentAttributesLocalOnly; + } + else + { + this.Fields[3].Data = (int)this.Fields[3].Data & ~MsiInterop.MsidbComponentAttributesLocalOnly; + } + } + } + + /// + /// Gets or sets the source only attribute of the Component. + /// + /// Source only attribute of the component. + public bool IsSourceOnly + { + get { return MsiInterop.MsidbComponentAttributesSourceOnly == ((int)this.Fields[3].Data & MsiInterop.MsidbComponentAttributesSourceOnly); } + set + { + if (value) + { + this.Fields[3].Data = (int)this.Fields[3].Data | MsiInterop.MsidbComponentAttributesSourceOnly; + } + else + { + this.Fields[3].Data = (int)this.Fields[3].Data & ~MsiInterop.MsidbComponentAttributesSourceOnly; + } + } + } + + /// + /// Gets or sets the optional attribute of the Component. + /// + /// Optional attribute of the component. + public bool IsOptional + { + get { return MsiInterop.MsidbComponentAttributesOptional == ((int)this.Fields[3].Data & MsiInterop.MsidbComponentAttributesOptional); } + set + { + if (value) + { + this.Fields[3].Data = (int)this.Fields[3].Data | MsiInterop.MsidbComponentAttributesOptional; + } + else + { + this.Fields[3].Data = (int)this.Fields[3].Data & ~MsiInterop.MsidbComponentAttributesOptional; + } + } + } + + /// + /// Gets or sets the registry key path attribute of the Component. + /// + /// Registry key path attribute of the component. + public bool IsRegistryKeyPath + { + get { return MsiInterop.MsidbComponentAttributesRegistryKeyPath == ((int)this.Fields[3].Data & MsiInterop.MsidbComponentAttributesRegistryKeyPath); } + set + { + if (value) + { + this.Fields[3].Data = (int)this.Fields[3].Data | MsiInterop.MsidbComponentAttributesRegistryKeyPath; + } + else + { + this.Fields[3].Data = (int)this.Fields[3].Data & ~MsiInterop.MsidbComponentAttributesRegistryKeyPath; + } + } + } + + /// + /// Gets or sets the shared dll ref count attribute of the Component. + /// + /// Shared dll ref countattribute of the component. + public bool IsSharedDll + { + get { return MsiInterop.MsidbComponentAttributesSharedDllRefCount == ((int)this.Fields[3].Data & MsiInterop.MsidbComponentAttributesSharedDllRefCount); } + set + { + if (value) + { + this.Fields[3].Data = (int)this.Fields[3].Data | MsiInterop.MsidbComponentAttributesSharedDllRefCount; + } + else + { + this.Fields[3].Data = (int)this.Fields[3].Data & ~MsiInterop.MsidbComponentAttributesSharedDllRefCount; + } + } + } + + /// + /// Gets or sets the permanent attribute of the Component. + /// + /// Permanent attribute of the component. + public bool IsPermanent + { + get { return MsiInterop.MsidbComponentAttributesPermanent == ((int)this.Fields[3].Data & MsiInterop.MsidbComponentAttributesPermanent); } + set + { + if (value) + { + this.Fields[3].Data = (int)this.Fields[3].Data | MsiInterop.MsidbComponentAttributesPermanent; + } + else + { + this.Fields[3].Data = (int)this.Fields[3].Data & ~MsiInterop.MsidbComponentAttributesPermanent; + } + } + } + + /// + /// Gets or sets the ODBC data source key path attribute of the Component. + /// + /// ODBC data source key path attribute of the component. + public bool IsOdbcDataSourceKeyPath + { + get { return MsiInterop.MsidbComponentAttributesODBCDataSource == ((int)this.Fields[3].Data & MsiInterop.MsidbComponentAttributesODBCDataSource); } + set + { + if (value) + { + this.Fields[3].Data = (int)this.Fields[3].Data | MsiInterop.MsidbComponentAttributesODBCDataSource; + } + else + { + this.Fields[3].Data = (int)this.Fields[3].Data & ~MsiInterop.MsidbComponentAttributesODBCDataSource; + } + } + } + + /// + /// Gets or sets the 64 bit attribute of the Component. + /// + /// 64-bitness of the component. + public bool Is64Bit + { + get { return MsiInterop.MsidbComponentAttributes64bit == ((int)this.Fields[3].Data & MsiInterop.MsidbComponentAttributes64bit); } + set + { + if (value) + { + this.Fields[3].Data = (int)this.Fields[3].Data | MsiInterop.MsidbComponentAttributes64bit; + } + else + { + this.Fields[3].Data = (int)this.Fields[3].Data & ~MsiInterop.MsidbComponentAttributes64bit; + } + } + } + + /// + /// Gets or sets the condition of the Component. + /// + /// Condition of the Component. + public string Condition + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets or sets the key path of the Component. + /// + /// Key path of the Component. + public string KeyPath + { + get { return (string)this.Fields[5].Data; } + set { this.Fields[5].Data = value; } + } + + /// + /// Gets or sets the source location to the file to fill in the Text of the control. + /// + /// Source location to the file to fill in the Text of the control. + public string SourceFile + { + get { return this.sourceFile; } + set { this.sourceFile = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/ContainerType.cs b/src/WixToolset.Data/Rows/ContainerType.cs new file mode 100644 index 00000000..55a74235 --- /dev/null +++ b/src/WixToolset.Data/Rows/ContainerType.cs @@ -0,0 +1,13 @@ +// 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.Data.Rows +{ + /// + /// Types of bundle packages. + /// + public enum ContainerType + { + Attached, + Detached, + } +} diff --git a/src/WixToolset.Data/Rows/ControlRow.cs b/src/WixToolset.Data/Rows/ControlRow.cs new file mode 100644 index 00000000..8fa3f633 --- /dev/null +++ b/src/WixToolset.Data/Rows/ControlRow.cs @@ -0,0 +1,143 @@ +// 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.Data.Rows +{ + using System.Diagnostics.CodeAnalysis; + + /// + /// Specialization of a row for the Control table. + /// + public sealed class ControlRow : Row + { + /// + /// Creates a Control row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Control row belongs to and should get its column definitions from. + public ControlRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the dialog of the Control row. + /// + /// Primary key of the Control row. + public string Dialog + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the identifier for this Control row. + /// + /// Identifier for this Control row. + public string Control + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the type of the control. + /// + /// Name of the control. + [SuppressMessage("Microsoft.Naming", "CA1721:PropertyNamesShouldNotMatchGetMethods")] + public string Type + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets the X location of the control. + /// + /// X location of the control. + public string X + { + get { return this.Fields[3].AsString(); } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets or sets the Y location of the control. + /// + /// Y location of the control. + public string Y + { + get { return this.Fields[4].AsString(); } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets or sets the width of the control. + /// + /// Width of the control. + public string Width + { + get { return this.Fields[5].AsString(); } + set { this.Fields[5].Data = value; } + } + + /// + /// Gets or sets the height of the control. + /// + /// Height of the control. + public string Height + { + get { return this.Fields[6].AsString(); } + set { this.Fields[6].Data = value; } + } + + /// + /// Gets or sets the attributes for the control. + /// + /// Attributes for the control. + public int Attributes + { + get { return (int)this.Fields[7].Data; } + set { this.Fields[7].Data = value; } + } + + /// + /// Gets or sets the Property associated with the control. + /// + /// Property associated with the control. + public string Property + { + get { return (string)this.Fields[8].Data; } + set { this.Fields[8].Data = value; } + } + + /// + /// Gets or sets the text of the control. + /// + /// Text of the control. + public string Text + { + get { return (string)this.Fields[9].Data; } + set { this.Fields[9].Data = value; } + } + + /// + /// Gets or sets the next control. + /// + /// Next control. + public string Next + { + get { return (string)this.Fields[10].Data; } + set { this.Fields[10].Data = value; } + } + + /// + /// Gets or sets the help for the control. + /// + /// Help for the control. + public string Help + { + get { return (string)this.Fields[11].Data; } + set { this.Fields[11].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/ExitCodeBehaviorType.cs b/src/WixToolset.Data/Rows/ExitCodeBehaviorType.cs new file mode 100644 index 00000000..07e66585 --- /dev/null +++ b/src/WixToolset.Data/Rows/ExitCodeBehaviorType.cs @@ -0,0 +1,13 @@ +// 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.Data.Rows +{ + public enum ExitCodeBehaviorType + { + NotSet = -1, + Success, + Error, + ScheduleReboot, + ForceReboot, + } +} diff --git a/src/WixToolset.Data/Rows/FileAssemblyType.cs b/src/WixToolset.Data/Rows/FileAssemblyType.cs new file mode 100644 index 00000000..6e92a083 --- /dev/null +++ b/src/WixToolset.Data/Rows/FileAssemblyType.cs @@ -0,0 +1,19 @@ +// 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.Data.Rows +{ + /// + /// Every file row has an assembly type. + /// + public enum FileAssemblyType + { + /// File is not an assembly. + NotAnAssembly, + + /// File is a Common Language Runtime Assembly. + DotNetAssembly, + + /// File is Win32 SxS assembly. + Win32Assembly, + } +} diff --git a/src/WixToolset.Data/Rows/FileRow.cs b/src/WixToolset.Data/Rows/FileRow.cs new file mode 100644 index 00000000..de5d5652 --- /dev/null +++ b/src/WixToolset.Data/Rows/FileRow.cs @@ -0,0 +1,640 @@ +// 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.Data.Rows +{ + using System; + using System.Diagnostics; + using System.Globalization; + using WixToolset.Data.Msi; + + /// + /// Specialization of a row for the file table. + /// + public sealed class FileRow : Row //, IComparable + { + //private string assemblyApplication; + //private string assemblyManifest; + //private FileAssemblyType assemblyType; + //private string directory; + //private int diskId; + //private bool fromModule; + //private bool isGeneratedShortFileName; + //private int patchGroup; + //private string processorArchitecture; + //private string source; + //private Row hashRow; + //private List assemblyNameRows; + //private string[] previousSource; + //private string symbols; + //private string[] previousSymbols; + //private PatchAttributeType patchAttributes; + //private string retainOffsets; + //private string retainLengths; + //private string ignoreOffsets; + //private string ignoreLengths; + //private string[] previousRetainOffsets; + //private string[] previousRetainLengths; + //private string[] previousIgnoreOffsets; + //private string[] previousIgnoreLengths; + //private string patch; + + /// + /// Creates a File row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this File row belongs to and should get its column definitions from. + public FileRow(SourceLineNumber sourceLineNumbers, Table table) + : base(sourceLineNumbers, table) + { + //this.assemblyType = FileAssemblyType.NotAnAssembly; + //this.previousSource = new string[1]; + //this.previousSymbols = new string[1]; + //this.previousRetainOffsets = new string[1]; + //this.previousRetainLengths = new string[1]; + //this.previousIgnoreOffsets = new string[1]; + //this.previousIgnoreLengths = new string[1]; + } + + /// + /// Creates a File row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this Media row belongs to and should get its column definitions from. + public FileRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDefinition) + : base(sourceLineNumbers, tableDefinition) + { + //this.assemblyType = FileAssemblyType.NotAnAssembly; + //this.previousSource = new string[1]; + //this.previousSymbols = new string[1]; + //this.previousRetainOffsets = new string[1]; + //this.previousRetainLengths = new string[1]; + //this.previousIgnoreOffsets = new string[1]; + //this.previousIgnoreLengths = new string[1]; + } + + /// + /// Gets or sets the primary key of the file row. + /// + /// Primary key of the file row. + public string File + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the component this file row belongs to. + /// + /// Component this file row belongs to. + public string Component + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the name of the file. + /// + /// Name of the file. + public string FileName + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets the real filesystem name of the file (without a pipe). This is typically the long name of the file. + /// However, if no long name is available, falls back to the short name. + /// + /// Long Name of the file - or if no long name is available, falls back to the short name. + public string LongFileName + { + get + { + string fileName = this.FileName; + int index = fileName.IndexOf('|'); + + // If it doesn't contain a pipe, just return the whole string + if (-1 == index) + { + return fileName; + } + else // otherwise, extract the part of the string after the pipe + { + return fileName.Substring(index + 1); + } + } + } + + /// + /// Gets or sets the size of the file. + /// + /// Size of the file. + public int FileSize + { + get { return (int)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets or sets the version of the file. + /// + /// Version of the file. + public string Version + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets or sets the LCID of the file. + /// + /// LCID of the file. + public string Language + { + get { return (string)this.Fields[5].Data; } + set { this.Fields[5].Data = value; } + } + + /// + /// Gets or sets the attributes on a file. + /// + /// Attributes on a file. + public int Attributes + { + get { return Convert.ToInt32(this.Fields[6].Data, CultureInfo.InvariantCulture); } + set { this.Fields[6].Data = value; } + } + + /// + /// Gets or sets whether this file should be compressed. + /// + /// Whether this file should be compressed. + public YesNoType Compressed + { + get + { + bool compressedFlag = (0 < (this.Attributes & MsiInterop.MsidbFileAttributesCompressed)); + bool noncompressedFlag = (0 < (this.Attributes & MsiInterop.MsidbFileAttributesNoncompressed)); + + if (compressedFlag && noncompressedFlag) + { + throw new WixException(WixDataErrors.IllegalFileCompressionAttributes(this.SourceLineNumbers)); + } + else if (compressedFlag) + { + return YesNoType.Yes; + } + else if (noncompressedFlag) + { + return YesNoType.No; + } + else + { + return YesNoType.NotSet; + } + } + + set + { + if (YesNoType.Yes == value) + { + // these are mutually exclusive + this.Attributes |= MsiInterop.MsidbFileAttributesCompressed; + this.Attributes &= ~MsiInterop.MsidbFileAttributesNoncompressed; + } + else if (YesNoType.No == value) + { + // these are mutually exclusive + this.Attributes |= MsiInterop.MsidbFileAttributesNoncompressed; + this.Attributes &= ~MsiInterop.MsidbFileAttributesCompressed; + } + else // not specified + { + Debug.Assert(YesNoType.NotSet == value); + + // clear any compression bits + this.Attributes &= ~MsiInterop.MsidbFileAttributesCompressed; + this.Attributes &= ~MsiInterop.MsidbFileAttributesNoncompressed; + } + } + } + + /// + /// Gets or sets the sequence of the file row. + /// + /// Sequence of the file row. + public int Sequence + { + get { return (int)this.Fields[7].Data; } + set { this.Fields[7].Data = value; } + } + + /////// + /////// Gets or sets the type of assembly of file row. + /////// + /////// Assembly type for file row. + ////public FileAssemblyType AssemblyType + ////{ + //// get { return this.assemblyType; } + //// set { this.assemblyType = value; } + ////} + + /////// + /////// Gets or sets the identifier for the assembly application. + /////// + /////// Identifier for the assembly application. + ////public string AssemblyApplication + ////{ + //// get { return this.assemblyApplication; } + //// set { this.assemblyApplication = value; } + ////} + + /////// + /////// Gets or sets the identifier for the assembly manifest. + /////// + /////// Identifier for the assembly manifest. + ////public string AssemblyManifest + ////{ + //// get { return this.assemblyManifest; } + //// set { this.assemblyManifest = value; } + ////} + + /////// + /////// Gets or sets the directory of the file. + /////// + /////// Directory of the file. + ////public string Directory + ////{ + //// get { return this.directory; } + //// set { this.directory = value; } + ////} + + /////// + /////// Gets or sets the disk id for this file. + /////// + /////// Disk id for the file. + ////public int DiskId + ////{ + //// get { return this.diskId; } + //// set { this.diskId = value; } + ////} + + /////// + /////// Gets or sets the source location to the file. + /////// + /////// Source location to the file. + ////public string Source + ////{ + //// get { return this.source; } + //// set { this.source = value; } + ////} + + /////// + /////// Gets or sets the source location to the previous file. + /////// + /////// Source location to the previous file. + ////public string PreviousSource + ////{ + //// get { return this.previousSource[0]; } + //// set { this.previousSource[0] = value; } + ////} + + /////// + /////// Gets the source location to the previous files. + /////// + /////// Source location to the previous files. + ////[SuppressMessage("Microsoft.Performance", "CA1819:PropertiesShouldNotReturnArrays")] + ////public string[] PreviousSourceArray + ////{ + //// get { return this.previousSource; } + ////} + + /////// + /////// Gets or sets the architecture the file executes on. + /////// + /////// Architecture the file executes on. + ////public string ProcessorArchitecture + ////{ + //// get { return this.processorArchitecture; } + //// set { this.processorArchitecture = value; } + ////} + + /////// + /////// Gets of sets the patch group of a patch-added file. + /////// + /////// The patch group of a patch-added file. + ////public int PatchGroup + ////{ + //// get { return this.patchGroup; } + //// set { this.patchGroup = value; } + ////} + + /////// + /////// Gets or sets the patch header of the file. + /////// + /////// Patch header of the file. + ////public string Patch + ////{ + //// get { return this.patch; } + //// set { this.patch = value; } + ////} + + /////// + /////// Gets or sets the locations to find the file's symbols. + /////// + /////// Symbol paths for the file. + ////public string Symbols + ////{ + //// get { return this.symbols; } + //// set { this.symbols = value; } + ////} + + /////// + /////// Gets or sets the locations to find the file's previous symbols. + /////// + /////// Symbol paths for the previous file. + ////public string PreviousSymbols + ////{ + //// get { return this.previousSymbols[0]; } + //// set { this.previousSymbols[0] = value; } + ////} + + /////// + /////// Gets the locations to find the files' previous symbols. + /////// + /////// Symbol paths for the previous files. + ////[SuppressMessage("Microsoft.Performance", "CA1819:PropertiesShouldNotReturnArrays")] + ////public string[] PreviousSymbolsArray + ////{ + //// get { return this.previousSymbols; } + ////} + + /////// + /////// Gets or sets the generated short file name attribute. + /////// + /////// The generated short file name attribute. + ////public bool IsGeneratedShortFileName + ////{ + //// get { return this.isGeneratedShortFileName; } + + //// set { this.isGeneratedShortFileName = value; } + ////} + + /////// + /////// Gets or sets whether this row came from a merge module. + /////// + /////// Whether this row came from a merge module. + ////public bool FromModule + ////{ + //// get { return this.fromModule; } + //// set { this.fromModule = value; } + ////} + + /////// + /////// Gets or sets the MsiFileHash row created for this FileRow. + /////// + /////// Row for MsiFileHash table. + ////public Row HashRow + ////{ + //// get { return this.hashRow; } + //// set { this.hashRow = value; } + ////} + + /////// + /////// Gets or sets the set of MsiAssemblyName rows created for this FileRow. + /////// + /////// RowCollection of MsiAssemblyName table. + ////[SuppressMessage("Microsoft.Usage", "CA2227:CollectionPropertiesShouldBeReadOnly")] + ////public List AssemblyNameRows + ////{ + //// get { return this.assemblyNameRows; } + //// set { this.assemblyNameRows = value; } + ////} + + /////// + /////// Gets or sets the patching attributes to the file. + /////// + /////// Patching attributes of the file. + ////public PatchAttributeType PatchAttributes + ////{ + //// get { return this.patchAttributes; } + //// set { this.patchAttributes = value; } + ////} + + /////// + /////// Gets or sets the delta patch retain-length list for the file. + /////// + /////// RetainLength list for the file. + ////public string RetainLengths + ////{ + //// get { return this.retainLengths; } + //// set { this.retainLengths = value; } + ////} + + /////// + /////// Gets or sets the delta patch ignore-offset list for the file. + /////// + /////// IgnoreOffset list for the file. + ////public string IgnoreOffsets + ////{ + //// get { return this.ignoreOffsets; } + //// set { this.ignoreOffsets = value; } + ////} + + /////// + /////// Gets or sets the delta patch ignore-length list for the file. + /////// + /////// IgnoreLength list for the file. + ////public string IgnoreLengths + ////{ + //// get { return this.ignoreLengths; } + //// set { this.ignoreLengths = value; } + ////} + + /////// + /////// Gets or sets the delta patch retain-offset list for the file. + /////// + /////// RetainOffset list for the file. + ////public string RetainOffsets + ////{ + //// get { return this.retainOffsets; } + //// set { this.retainOffsets = value; } + ////} + + /////// + /////// Gets or sets the delta patch retain-length list for the previous file. + /////// + /////// RetainLength list for the previous file. + ////public string PreviousRetainLengths + ////{ + //// get { return this.previousRetainLengths[0]; } + //// set { this.previousRetainLengths[0] = value; } + ////} + + /////// + /////// Gets the delta patch retain-length list for the previous files. + /////// + /////// RetainLength list for the previous files. + ////[SuppressMessage("Microsoft.Performance", "CA1819:PropertiesShouldNotReturnArrays")] + ////public string[] PreviousRetainLengthsArray + ////{ + //// get { return this.previousRetainLengths; } + ////} + + /////// + /////// Gets or sets the delta patch ignore-offset list for the previous file. + /////// + /////// IgnoreOffset list for the previous file. + ////public string PreviousIgnoreOffsets + ////{ + //// get { return this.previousIgnoreOffsets[0]; } + //// set { this.previousIgnoreOffsets[0] = value; } + ////} + + /////// + /////// Gets the delta patch ignore-offset list for the previous files. + /////// + /////// IgnoreOffset list for the previous files. + ////[SuppressMessage("Microsoft.Performance", "CA1819:PropertiesShouldNotReturnArrays")] + ////public string[] PreviousIgnoreOffsetsArray + ////{ + //// get { return this.previousIgnoreOffsets; } + ////} + + /////// + /////// Gets or sets the delta patch ignore-length list for the previous file. + /////// + /////// IgnoreLength list for the previous file. + ////public string PreviousIgnoreLengths + ////{ + //// get { return this.previousIgnoreLengths[0]; } + //// set { this.previousIgnoreLengths[0] = value; } + ////} + + /////// + /////// Gets the delta patch ignore-length list for the previous files. + /////// + /////// IgnoreLength list for the previous files. + ////[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Performance", "CA1819:PropertiesShouldNotReturnArrays")] + ////public string[] PreviousIgnoreLengthsArray + ////{ + //// get { return this.previousIgnoreLengths; } + ////} + + /////// + /////// Gets or sets the delta patch retain-offset list for the previous file. + /////// + /////// RetainOffset list for the previous file. + ////public string PreviousRetainOffsets + ////{ + //// get { return this.previousRetainOffsets[0]; } + //// set { this.previousRetainOffsets[0] = value; } + ////} + + /////// + /////// Gets the delta patch retain-offset list for the previous files. + /////// + /////// RetainOffset list for the previous files. + ////[SuppressMessage("Microsoft.Performance", "CA1819:PropertiesShouldNotReturnArrays")] + ////public string[] PreviousRetainOffsetsArray + ////{ + //// get { return this.previousRetainOffsets; } + ////} + + /////// + /////// Compares the current FileRow with another object of the same type. + /////// + /////// An object to compare with this instance. + /////// An integer that indicates the relative order of the comparands. + ////[SuppressMessage("Microsoft.Globalization", "CA1303:DoNotPassLiteralsAsLocalizedParameters", MessageId = "System.ArgumentException.#ctor(System.String)")] + ////[SuppressMessage("Microsoft.Globalization", "CA1309:UseOrdinalStringComparison")] + ////public int CompareTo(object obj) + ////{ + //// if (this == obj) + //// { + //// return 0; + //// } + + //// FileRow fileRow = obj as FileRow; + //// if (null == fileRow) + //// { + //// throw new ArgumentException(WixDataStrings.EXP_OtherObjectIsNotFileRow); + //// } + + //// int compared = this.DiskId - fileRow.DiskId; + //// if (0 == compared) + //// { + //// compared = this.patchGroup - fileRow.patchGroup; + + //// if (0 == compared) + //// { + //// compared = String.Compare(this.File, fileRow.File, StringComparison.InvariantCulture); + //// } + //// } + + //// return compared; + ////} + + /////// + /////// Copies data from another FileRow object. + /////// + /////// An row to get data from. + ////public void CopyFrom(FileRow src) + ////{ + //// for (int i = 0; i < src.Fields.Length; i++) + //// { + //// this[i] = src[i]; + //// } + //// this.assemblyManifest = src.assemblyManifest; + //// this.assemblyType = src.assemblyType; + //// this.directory = src.directory; + //// this.diskId = src.diskId; + //// this.fromModule = src.fromModule; + //// this.isGeneratedShortFileName = src.isGeneratedShortFileName; + //// this.patchGroup = src.patchGroup; + //// this.processorArchitecture = src.processorArchitecture; + //// this.source = src.source; + //// this.PreviousSource = src.PreviousSource; + //// this.Operation = src.Operation; + //// this.symbols = src.symbols; + //// this.PreviousSymbols = src.PreviousSymbols; + //// this.patchAttributes = src.patchAttributes; + //// this.retainOffsets = src.retainOffsets; + //// this.retainLengths = src.retainLengths; + //// this.ignoreOffsets = src.ignoreOffsets; + //// this.ignoreLengths = src.ignoreLengths; + //// this.PreviousRetainOffsets = src.PreviousRetainOffsets; + //// this.PreviousRetainLengths = src.PreviousRetainLengths; + //// this.PreviousIgnoreOffsets = src.PreviousIgnoreOffsets; + //// this.PreviousIgnoreLengths = src.PreviousIgnoreLengths; + ////} + + /////// + /////// Appends previous data from another FileRow object. + /////// + /////// An row to get data from. + ////public void AppendPreviousDataFrom(FileRow src) + ////{ + //// AppendStringToArray(ref this.previousSource, src.previousSource[0]); + //// AppendStringToArray(ref this.previousSymbols, src.previousSymbols[0]); + //// AppendStringToArray(ref this.previousRetainOffsets, src.previousRetainOffsets[0]); + //// AppendStringToArray(ref this.previousRetainLengths, src.previousRetainLengths[0]); + //// AppendStringToArray(ref this.previousIgnoreOffsets, src.previousIgnoreOffsets[0]); + //// AppendStringToArray(ref this.previousIgnoreLengths, src.previousIgnoreLengths[0]); + ////} + + /////// + /////// Helper method for AppendPreviousDataFrom. + /////// + /////// Destination array. + /////// Source string. + ////private static void AppendStringToArray(ref string[] destination, string source) + ////{ + //// string[] result = new string[destination.Length + 1]; + //// destination.CopyTo(result, 0); + //// result[destination.Length] = source; + //// destination = result; + ////} + } +} diff --git a/src/WixToolset.Data/Rows/MediaRow.cs b/src/WixToolset.Data/Rows/MediaRow.cs new file mode 100644 index 00000000..f387a8d2 --- /dev/null +++ b/src/WixToolset.Data/Rows/MediaRow.cs @@ -0,0 +1,80 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the Media table. + /// + public sealed class MediaRow : Row + { + /// + /// Creates a Media row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Media row belongs to and should get its column definitions from. + public MediaRow(SourceLineNumber sourceLineNumbers, Table table) + : base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the disk id for this media row. + /// + /// Disk id. + public int DiskId + { + get { return (int)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the last sequence number for this media row. + /// + /// Last sequence number. + public int LastSequence + { + get { return (int)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the disk prompt for this media row. + /// + /// Disk prompt. + public string DiskPrompt + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets the cabinet name for this media row. + /// + /// Cabinet name. + public string Cabinet + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets or sets the volume label for this media row. + /// + /// Volume label. + public string VolumeLabel + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets or sets the source for this media row. + /// + /// Source. + public string Source + { + get { return (string)this.Fields[5].Data; } + set { this.Fields[5].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/PatchAttributeType.cs b/src/WixToolset.Data/Rows/PatchAttributeType.cs new file mode 100644 index 00000000..b585e845 --- /dev/null +++ b/src/WixToolset.Data/Rows/PatchAttributeType.cs @@ -0,0 +1,27 @@ +// 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.Data.Rows +{ + using System; + + /// + /// PatchAttribute values + /// + [Flags] + public enum PatchAttributeType + { + None = 0, + + /// Prevents the updating of the file that is in fact changed in the upgraded image relative to the target images. + Ignore = 1, + + /// Set if the entire file should be installed rather than creating a binary patch. + IncludeWholeFile = 2, + + /// Set to indicate that the patch is non-vital. + AllowIgnoreOnError = 4, + + /// Allowed bits. + Defined = Ignore | IncludeWholeFile | AllowIgnoreOnError + } +} diff --git a/src/WixToolset.Data/Rows/PropertyRow.cs b/src/WixToolset.Data/Rows/PropertyRow.cs new file mode 100644 index 00000000..558df760 --- /dev/null +++ b/src/WixToolset.Data/Rows/PropertyRow.cs @@ -0,0 +1,42 @@ +// 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.Data.Rows +{ + using System; + + /// + /// Specialization of a row for the upgrade table. + /// + public sealed class PropertyRow : Row + { + /// + /// Creates an Upgrade row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Upgrade row belongs to and should get its column definitions from. + public PropertyRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets and sets the upgrade code for the row. + /// + /// Property identifier for the row. + public string Property + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets and sets the value for the row. + /// + /// Property value for the row. + public string Value + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/SummaryInfoRowCollection.cs b/src/WixToolset.Data/Rows/SummaryInfoRowCollection.cs new file mode 100644 index 00000000..bc931f15 --- /dev/null +++ b/src/WixToolset.Data/Rows/SummaryInfoRowCollection.cs @@ -0,0 +1,42 @@ +// 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.Data.Rows +{ + using System; + using System.Collections; + using System.Collections.ObjectModel; + + /// + /// Indexed container class for summary information rows. + /// + public sealed class SummaryInfoRowCollection : KeyedCollection + { + /// + /// Creates the keyed collection from existing rows in a table. + /// + /// The summary information table to index. + public SummaryInfoRowCollection(Table table) + { + if (0 != String.CompareOrdinal("_SummaryInformation", table.Name)) + { + string message = string.Format(WixDataStrings.EXP_UnsupportedTable, table.Name); + throw new ArgumentException(message, "table"); + } + + foreach (Row row in table.Rows) + { + this.Add(row); + } + } + + /// + /// Gets the summary property ID for the . + /// + /// The row to index. + /// The summary property ID for the . + protected override int GetKeyForItem(Row row) + { + return (int)row[0]; + } + } +} diff --git a/src/WixToolset.Data/Rows/SymbolPathType.cs b/src/WixToolset.Data/Rows/SymbolPathType.cs new file mode 100644 index 00000000..964e1caa --- /dev/null +++ b/src/WixToolset.Data/Rows/SymbolPathType.cs @@ -0,0 +1,17 @@ +// 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.Data.Rows +{ + /// + /// The types that the WixDeltaPatchSymbolPaths table can hold. + /// + /// The order of these values is important since WixDeltaPatchSymbolPaths are sorted by this type. + public enum SymbolPathType + { + File, + Component, + Directory, + Media, + Product + }; +} diff --git a/src/WixToolset.Data/Rows/UpgradeRow.cs b/src/WixToolset.Data/Rows/UpgradeRow.cs new file mode 100644 index 00000000..807a9f93 --- /dev/null +++ b/src/WixToolset.Data/Rows/UpgradeRow.cs @@ -0,0 +1,90 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the upgrade table. + /// + public sealed class UpgradeRow : Row + { + /// + /// Creates an Upgrade row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Upgrade row belongs to and should get its column definitions from. + public UpgradeRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets and sets the upgrade code for the row. + /// + /// Upgrade code for the row. + public string UpgradeCode + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets and sets the version minimum for the row. + /// + /// Version minimum for the row. + public string VersionMin + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets and sets the version maximum for the row. + /// + /// Version maximum for the row. + public string VersionMax + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets and sets the language for the row. + /// + /// Language for the row. + public string Language + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets and sets the attributes for the row. + /// + /// Attributes for the row. + public int Attributes + { + get { return (int)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets and sets the remove code for the row. + /// + /// Remove code for the row. + public string Remove + { + get { return (string)this.Fields[5].Data; } + set { this.Fields[5].Data = value; } + } + + /// + /// Gets and sets the action property for the row. + /// + /// Action property for the row. + public string ActionProperty + { + get { return (string)this.Fields[6].Data; } + set { this.Fields[6].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixActionRow.cs b/src/WixToolset.Data/Rows/WixActionRow.cs new file mode 100644 index 00000000..3009e59d --- /dev/null +++ b/src/WixToolset.Data/Rows/WixActionRow.cs @@ -0,0 +1,374 @@ +// 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.Data.Rows +{ + using System; + using System.Collections.Generic; + using System.Diagnostics; + using System.Globalization; + using System.Xml; + using System.Xml.Schema; + + /// + /// The Sequence tables that actions may belong to. + /// + public enum SequenceTable + { + /// AdminUISequence + AdminUISequence, + + /// AdminExecuteSequence + AdminExecuteSequence, + + /// AdvtExecuteSequence + AdvtExecuteSequence, + + /// InstallUISequence + InstallUISequence, + + /// InstallExecuteSequence + InstallExecuteSequence + } + + /// + /// Specialization of a row for the sequence tables. + /// + public sealed class WixActionRow : Row, IComparable + { + private WixActionRowCollection previousActionRows; + private WixActionRowCollection nextActionRows; + + /// + /// Instantiates an ActionRow that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Action row belongs to and should get its column definitions from. + public WixActionRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Instantiates a standard ActionRow. + /// + /// The sequence table of the standard action. + /// The name of the standard action. + /// The condition of the standard action. + /// The suggested sequence number of the standard action. + private WixActionRow(SequenceTable sequenceTable, string action, string condition, int sequence) : + base(null, WindowsInstallerStandard.GetTableDefinitions()["WixAction"]) + { + this.SequenceTable = sequenceTable; + this.Action = action; + this.Condition = condition; + this.Sequence = sequence; + this.Overridable = true; // all standard actions are overridable by default + } + + /// + /// Instantiates an ActionRow by copying data from another ActionRow. + /// + /// The row the data is copied from. + /// The previous and next action collections are not copied. + private WixActionRow(WixActionRow source) + : base(source) + { + } + + /// + /// Gets or sets the name of the action. + /// + /// The name of the action. + public string Action + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets the name of the action this action should be scheduled after. + /// + /// The name of the action this action should be scheduled after. + public string After + { + get { return (string)this.Fields[5].Data; } + set { this.Fields[5].Data = value; } + } + + /// + /// Gets the name of the action this action should be scheduled before. + /// + /// The name of the action this action should be scheduled before. + public string Before + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets or sets the condition of the action. + /// + /// The condition of the action. + public string Condition + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets whether this action is overridable. + /// + /// Whether this action is overridable. + public bool Overridable + { + get { return (1 == Convert.ToInt32(this.Fields[6].Data, CultureInfo.InvariantCulture)); } + set { this.Fields[6].Data = (value ? 1 : 0); } + } + + /// + /// Gets or sets the sequence number of this action. + /// + /// The sequence number of this action. + public int Sequence + { + get { return Convert.ToInt32(this.Fields[3].Data, CultureInfo.InvariantCulture); } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets of sets the sequence table of this action. + /// + /// The sequence table of this action. + public SequenceTable SequenceTable + { + get { return (SequenceTable)Enum.Parse(typeof(SequenceTable), (string)this.Fields[0].Data); } + set { this.Fields[0].Data = value.ToString(); } + } + + /// + /// Gets the actions that should be scheduled after this action. + /// + /// The actions that should be scheduled after this action. + public WixActionRowCollection NextActionRows + { + get + { + if (null == this.nextActionRows) + { + this.nextActionRows = new WixActionRowCollection(); + } + + return this.nextActionRows; + } + } + + /// + /// Gets the actions that should be scheduled before this action. + /// + /// The actions that should be scheduled before this action. + public WixActionRowCollection PreviousActionRows + { + get + { + if (null == this.previousActionRows) + { + this.previousActionRows = new WixActionRowCollection(); + } + + return this.previousActionRows; + } + } + + /// + /// Creates a clone of the action row. + /// + /// A shallow copy of the source object. + /// The previous and next action collections are not copied. + public WixActionRow Clone() + { + return new WixActionRow(this); + } + + /// + /// Compares the current instance with another object of the same type. + /// + /// Other reference to compare this one to. + /// Returns less than 0 for less than, 0 for equals, and greater than 0 for greater. + public int CompareTo(object obj) + { + WixActionRow otherActionRow = (WixActionRow)obj; + + return this.Sequence.CompareTo(otherActionRow.Sequence); + } + + /// + /// Parses ActionRows from the Xml reader. + /// + /// Xml reader that contains serialized ActionRows. + /// The parsed ActionRows. + internal static WixActionRow[] Parse(XmlReader reader) + { + Debug.Assert("action" == reader.LocalName); + + string id = null; + string condition = null; + bool empty = reader.IsEmptyElement; + int sequence = int.MinValue; + int sequenceCount = 0; + SequenceTable[] sequenceTables = new SequenceTable[Enum.GetValues(typeof(SequenceTable)).Length]; + + while (reader.MoveToNextAttribute()) + { + switch (reader.Name) + { + case "name": + id = reader.Value; + break; + case "AdminExecuteSequence": + if (reader.Value.Equals("yes")) + { + sequenceTables[sequenceCount] = SequenceTable.AdminExecuteSequence; + ++sequenceCount; + } + break; + case "AdminUISequence": + if (reader.Value.Equals("yes")) + { + sequenceTables[sequenceCount] = SequenceTable.AdminUISequence; + ++sequenceCount; + } + break; + case "AdvtExecuteSequence": + if (reader.Value.Equals("yes")) + { + sequenceTables[sequenceCount] = SequenceTable.AdvtExecuteSequence; + ++sequenceCount; + } + break; + case "condition": + condition = reader.Value; + break; + case "InstallExecuteSequence": + if (reader.Value.Equals("yes")) + { + sequenceTables[sequenceCount] = SequenceTable.InstallExecuteSequence; + ++sequenceCount; + } + break; + case "InstallUISequence": + if (reader.Value.Equals("yes")) + { + sequenceTables[sequenceCount] = SequenceTable.InstallUISequence; + ++sequenceCount; + } + break; + case "sequence": + sequence = Convert.ToInt32(reader.Value, CultureInfo.InvariantCulture); + break; + } + } + + if (null == id) + { + throw new XmlException(); + } + + if (int.MinValue == sequence) + { + throw new XmlException(); + } + else if (1 > sequence) + { + throw new XmlException(); + } + + if (0 == sequenceCount) + { + throw new XmlException(); + } + + if (!empty && reader.Read() && XmlNodeType.EndElement != reader.MoveToContent()) + { + throw new XmlException(); + } + + // create the actions + WixActionRow[] actionRows = new WixActionRow[sequenceCount]; + for (int i = 0; i < sequenceCount; i++) + { + WixActionRow actionRow = new WixActionRow(sequenceTables[i], id, condition, sequence); + actionRows[i] = actionRow; + } + + return actionRows; + } + + /// + /// Determines whether this ActionRow contains the specified ActionRow as a child in its dependency tree. + /// + /// The possible child ActionRow. + /// true if the ActionRow is a child of this ActionRow; false otherwise. + public bool ContainsChildActionRow(WixActionRow actionRow) + { + if (null != this.previousActionRows) + { + if (this.previousActionRows.Contains(actionRow.SequenceTable, actionRow.Action)) + { + return true; + } + } + + if (null != this.nextActionRows) + { + if (this.nextActionRows.Contains(actionRow.SequenceTable, actionRow.Action)) + { + return true; + } + } + + return false; + } + + /// + /// Get all the actions scheduled before this one in a particular sequence table. + /// + /// The sequence table. + /// A RowCollection which will contain all the previous actions. + public void GetAllPreviousActionRows(SequenceTable sequenceTable, IList allPreviousActionRows) + { + if (null != this.previousActionRows) + { + foreach (WixActionRow actionRow in this.previousActionRows) + { + if (sequenceTable == actionRow.SequenceTable) + { + actionRow.GetAllPreviousActionRows(sequenceTable, allPreviousActionRows); + allPreviousActionRows.Add(actionRow); + actionRow.GetAllNextActionRows(sequenceTable, allPreviousActionRows); + } + } + } + } + + /// + /// Get all the actions scheduled after this one in a particular sequence table. + /// + /// The sequence table. + /// A RowCollection which will contain all the next actions. + public void GetAllNextActionRows(SequenceTable sequenceTable, IList allNextActionRows) + { + if (null != this.nextActionRows) + { + foreach (WixActionRow actionRow in this.nextActionRows) + { + if (sequenceTable == actionRow.SequenceTable) + { + actionRow.GetAllPreviousActionRows(sequenceTable, allNextActionRows); + allNextActionRows.Add(actionRow); + actionRow.GetAllNextActionRows(sequenceTable, allNextActionRows); + } + } + } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixActionRowCollection.cs b/src/WixToolset.Data/Rows/WixActionRowCollection.cs new file mode 100644 index 00000000..513a104f --- /dev/null +++ b/src/WixToolset.Data/Rows/WixActionRowCollection.cs @@ -0,0 +1,222 @@ +// 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.Data.Rows +{ + using System; + using System.Collections; + using System.Diagnostics; + using System.Xml; + + /// + /// A collection of action rows sorted by their sequence table and action name. + /// + public sealed class WixActionRowCollection : ICollection + { + private SortedList collection; + + /// + /// Creates a new action table object. + /// + public WixActionRowCollection() + { + this.collection = new SortedList(); + } + + /// + /// Gets the number of items in the collection. + /// + /// Number of items in collection. + public int Count + { + get { return this.collection.Count; } + } + + /// + /// Gets if the collection has been synchronized. + /// + /// True if the collection has been synchronized. + public bool IsSynchronized + { + get { return this.collection.IsSynchronized; } + } + + /// + /// Gets the object used to synchronize the collection. + /// + /// Oject used the synchronize the collection. + public object SyncRoot + { + get { return this; } + } + + /// + /// Get an ActionRow by its sequence table and action name. + /// + /// The sequence table of the ActionRow. + /// The action name of the ActionRow. + public WixActionRow this[SequenceTable sequenceTable, string action] + { + get { return (WixActionRow)this.collection[GetKey(sequenceTable, action)]; } + } + + /// + /// Add an ActionRow to the collection. + /// + /// The ActionRow to add. + /// true to overwrite an existing ActionRow; false otherwise. + public void Add(WixActionRow actionRow, bool overwrite) + { + string key = GetKey(actionRow.SequenceTable, actionRow.Action); + + if (overwrite) + { + this.collection[key] = actionRow; + } + else + { + this.collection.Add(key, actionRow); + } + } + + /// + /// Add an ActionRow to the collection. + /// + /// The ActionRow to add. + public void Add(WixActionRow actionRow) + { + this.Add(actionRow, false); + } + + /// + /// Determines if the collection contains an ActionRow with a specific sequence table and name. + /// + /// The sequence table of the ActionRow. + /// The action name of the ActionRow. + /// true if the ActionRow was found; false otherwise. + public bool Contains(SequenceTable sequenceTable, string action) + { + return this.collection.Contains(GetKey(sequenceTable, action)); + } + + /// + /// Copies the collection into an array. + /// + /// Array to copy the collection into. + /// Index to start copying from. + public void CopyTo(System.Array array, int index) + { + this.collection.Values.CopyTo(array, index); + } + + /// + /// Gets the enumerator for the collection. + /// + /// The enumerator for the collection. + public IEnumerator GetEnumerator() + { + return this.collection.Values.GetEnumerator(); + } + + /// + /// Remove an ActionRow from the collection. + /// + /// The sequence table of the ActionRow. + /// The action name of the ActionRow. + public void Remove(SequenceTable sequenceTable, string action) + { + this.collection.Remove(GetKey(sequenceTable, action)); + } + + /// + /// Load an action table from an XmlReader. + /// + /// Reader to get data from. + /// The ActionRowCollection represented by the xml. + internal static WixActionRowCollection Load(XmlReader reader) + { + reader.MoveToContent(); + + return Parse(reader); + } + + /// + /// Creates a new action table object and populates it from an Xml reader. + /// + /// Reader to get data from. + /// The parsed ActionTable. + private static WixActionRowCollection Parse(XmlReader reader) + { + if (!reader.LocalName.Equals("actions")) + { + throw new XmlException(); + } + + WixActionRowCollection actionRows = new WixActionRowCollection(); + bool empty = reader.IsEmptyElement; + + while (reader.MoveToNextAttribute()) + { + } + + if (!empty) + { + bool done = false; + + // loop through all the fields in a row + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + switch (reader.LocalName) + { + case "action": + WixActionRow[] parsedActionRows = WixActionRow.Parse(reader); + + foreach (WixActionRow actionRow in parsedActionRows) + { + actionRows.Add(actionRow); + } + break; + default: + throw new XmlException(); + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + + return actionRows; + } + + /// + /// Get the key for storing an ActionRow. + /// + /// The sequence table of the ActionRow. + /// The action name of the ActionRow. + /// The string key. + private static string GetKey(SequenceTable sequenceTable, string action) + { + return GetKey(sequenceTable.ToString(), action); + } + + /// + /// Get the key for storing an ActionRow. + /// + /// The sequence table of the ActionRow. + /// The action name of the ActionRow. + /// The string key. + private static string GetKey(string sequenceTable, string action) + { + return String.Concat(sequenceTable, '/', action); + } + } +} diff --git a/src/WixToolset.Data/Rows/WixApprovedExeForElevationRow.cs b/src/WixToolset.Data/Rows/WixApprovedExeForElevationRow.cs new file mode 100644 index 00000000..c10a39ab --- /dev/null +++ b/src/WixToolset.Data/Rows/WixApprovedExeForElevationRow.cs @@ -0,0 +1,79 @@ +// 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.Data.Rows +{ + + /// + /// Specialization of a row for the WixApprovedExeForElevation table. + /// + public class WixApprovedExeForElevationRow : Row + { + /// + /// Creates an ApprovedExeForElevation row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this ApprovedExeForElevation row belongs to and should get its column definitions from. + public WixApprovedExeForElevationRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates an ApprovedExeForElevation row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this ApprovedExeForElevation row belongs to and should get its column definitions from. + public WixApprovedExeForElevationRow(SourceLineNumber sourceLineNumbers, Table table) + : base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the ApprovedExeForElevation identifier. + /// + /// The ApprovedExeForElevation identifier. + public string Id + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the Key path. + /// + /// The Key path. + public string Key + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the Value name. + /// + /// The Value name. + public string ValueName + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets the attibutes. + /// + /// The BundleApprovedExeForElevationAttributes. + public BundleApprovedExeForElevationAttributes Attributes + { + get { return (BundleApprovedExeForElevationAttributes)this.Fields[3].Data; } + set { this.Fields[3].Data = (int)value; } + } + + /// + /// Gets whether this row is 64-bit. + /// + public bool Win64 + { + get { return BundleApprovedExeForElevationAttributes.Win64 == (this.Attributes & BundleApprovedExeForElevationAttributes.Win64); } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleCatalogRow.cs b/src/WixToolset.Data/Rows/WixBundleCatalogRow.cs new file mode 100644 index 00000000..05c1e597 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleCatalogRow.cs @@ -0,0 +1,50 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the WixCatalog table. + /// + public sealed class WixBundleCatalogRow : Row + { + /// + /// Creates a Catalog row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this Catalog row belongs to and should get its column definitions from. + public WixBundleCatalogRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a Catalog row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Catalog row belongs to and should get its column definitions from. + public WixBundleCatalogRow(SourceLineNumber sourceLineNumbers, Table table) + : base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the catalog identifier. + /// + /// The catalog identifier. + public string Id + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the payload identifier. + /// + /// The payload identifier. + public string Payload + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleContainerRow.cs b/src/WixToolset.Data/Rows/WixBundleContainerRow.cs new file mode 100644 index 00000000..7b03dcc5 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleContainerRow.cs @@ -0,0 +1,78 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the Container table. + /// + public class WixBundleContainerRow : Row + { + /// + /// Creates a ContainerRow row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this Media row belongs to and should get its column definitions from. + public WixBundleContainerRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a ContainerRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Media row belongs to and should get its column definitions from. + public WixBundleContainerRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + public string Id + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + public string Name + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + public ContainerType Type + { + get { return (ContainerType)this.Fields[2].Data; } + set { this.Fields[2].Data = (int)value; } + } + + public string DownloadUrl + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + public long Size + { + get { return (long)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + public string Hash + { + get { return (string)this.Fields[5].Data; } + set { this.Fields[5].Data = value; } + } + + public int AttachedContainerIndex + { + get { return (null == this.Fields[6].Data) ? -1 : (int)this.Fields[6].Data; } + set { this.Fields[6].Data = value; } + } + + public string WorkingPath + { + get { return (string)this.Fields[7].Data; } + set { this.Fields[7].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleExePackageAttributes.cs b/src/WixToolset.Data/Rows/WixBundleExePackageAttributes.cs new file mode 100644 index 00000000..1d4d7bbb --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleExePackageAttributes.cs @@ -0,0 +1,12 @@ +// 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.Data.Rows +{ + using System; + + [Flags] + public enum WixBundleExePackageAttributes + { + Repairable = 0x1, + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleExePackageRow.cs b/src/WixToolset.Data/Rows/WixBundleExePackageRow.cs new file mode 100644 index 00000000..5bca6029 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleExePackageRow.cs @@ -0,0 +1,101 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the WixBundleExePackage table. + /// + public sealed class WixBundleExePackageRow : Row + { + /// + /// Creates a WixBundleExePackage row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this row belongs to and should get its column definitions from. + public WixBundleExePackageRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixBundleExePackageRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this row belongs to and should get its column definitions from. + public WixBundleExePackageRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the foreign key identifier to the ChainPackage row. + /// + public string ChainPackageId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the raw Exe attributes of a patch. + /// + public WixBundleExePackageAttributes Attributes + { + get { return (WixBundleExePackageAttributes)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the protcol for the executable package. + /// + public string DetectCondition + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets the install command for the executable package. + /// + public string InstallCommand + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets or sets the repair command for the executable package. + /// + public string RepairCommand + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets or sets the uninstall command for the executable package. + /// + public string UninstallCommand + { + get { return (string)this.Fields[5].Data; } + set { this.Fields[5].Data = value; } + } + + /// + /// Gets or sets the protcol for the executable package. + /// + public string ExeProtocol + { + get { return (string)this.Fields[6].Data; } + set { this.Fields[6].Data = value; } + } + + /// + /// Gets whether the executable package is repairable. + /// + public bool Repairable + { + get { return 0 != (this.Attributes & WixBundleExePackageAttributes.Repairable); } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleMsiFeatureRow.cs b/src/WixToolset.Data/Rows/WixBundleMsiFeatureRow.cs new file mode 100644 index 00000000..551eae20 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleMsiFeatureRow.cs @@ -0,0 +1,93 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the MsiFeature table. + /// + public class WixBundleMsiFeatureRow : Row + { + /// + /// Creates a MsiFeatureRow row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this Media row belongs to and should get its column definitions from. + public WixBundleMsiFeatureRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a MsiFeatureRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Media row belongs to and should get its column definitions from. + public WixBundleMsiFeatureRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the foreign key identifier to the ChainPackage row. + /// + public string ChainPackageId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + public string Name + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + public long Size + { + get { return (long)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + public string Parent + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + public string Title + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + public string Description + { + get { return (string)this.Fields[5].Data; } + set { this.Fields[5].Data = value; } + } + + public int Display + { + get { return (int)this.Fields[6].Data; } + set { this.Fields[6].Data = value; } + } + + public int Level + { + get { return (int)this.Fields[7].Data; } + set { this.Fields[7].Data = value; } + } + + public string Directory + { + get { return (string)this.Fields[8].Data; } + set { this.Fields[8].Data = value; } + } + + public int Attributes + { + get { return (int)this.Fields[9].Data; } + set { this.Fields[9].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleMsiPackageAttributes.cs b/src/WixToolset.Data/Rows/WixBundleMsiPackageAttributes.cs new file mode 100644 index 00000000..fc038f81 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleMsiPackageAttributes.cs @@ -0,0 +1,15 @@ +// 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.Data.Rows +{ + using System; + + [Flags] + public enum WixBundleMsiPackageAttributes + { + DisplayInternalUI = 0x1, + EnableFeatureSelection = 0x4, + ForcePerMachine = 0x2, + SuppressLooseFilePayloadGeneration = 0x8, + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleMsiPackageRow.cs b/src/WixToolset.Data/Rows/WixBundleMsiPackageRow.cs new file mode 100644 index 00000000..22e66a7a --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleMsiPackageRow.cs @@ -0,0 +1,137 @@ +// 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.Data.Rows +{ + using System; + using System.Globalization; + + /// + /// Specialization of a row for the WixBundleMsiPackage table. + /// + public sealed class WixBundleMsiPackageRow : Row + { + /// + /// Creates a WixBundleMsiPackage row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this row belongs to and should get its column definitions from. + public WixBundleMsiPackageRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixBundleMsiPackageRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this row belongs to and should get its column definitions from. + public WixBundleMsiPackageRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the foreign key identifier to the ChainPackage row. + /// + public string ChainPackageId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the raw MSI attributes of a package. + /// + public WixBundleMsiPackageAttributes Attributes + { + get { return (WixBundleMsiPackageAttributes)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the MSI package's product code. + /// + public string ProductCode + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets the MSI package's upgrade code. + /// + public string UpgradeCode + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets or sets the product version of the MSI package. + /// + public string ProductVersion + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets or sets the language of the MSI package. + /// + public int ProductLanguage + { + get { return Convert.ToInt32(this.Fields[5].Data, CultureInfo.InvariantCulture); } + set { this.Fields[5].Data = value; } + } + + /// + /// Gets or sets the product name of the MSI package. + /// + public string ProductName + { + get { return (string)this.Fields[6].Data; } + set { this.Fields[6].Data = value; } + } + + /// + /// Gets or sets the MSI package's manufacturer. + /// + public string Manufacturer + { + get { return (string)this.Fields[7].Data; } + set { this.Fields[7].Data = value; } + } + + /// + /// Gets the display internal UI of a package. + /// + public bool DisplayInternalUI + { + get { return 0 != (this.Attributes & WixBundleMsiPackageAttributes.DisplayInternalUI); } + } + + /// + /// Gets the display internal UI of a package. + /// + public bool EnableFeatureSelection + { + get { return 0 != (this.Attributes & WixBundleMsiPackageAttributes.EnableFeatureSelection); } + } + + /// + /// Gets the display internal UI of a package. + /// + public bool ForcePerMachine + { + get { return 0 != (this.Attributes & WixBundleMsiPackageAttributes.ForcePerMachine); } + } + + /// + /// Gets the suppress loose file payload generation of a package. + /// + public bool SuppressLooseFilePayloadGeneration + { + get { return 0 != (this.Attributes & WixBundleMsiPackageAttributes.SuppressLooseFilePayloadGeneration); } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleMsiPropertyRow.cs b/src/WixToolset.Data/Rows/WixBundleMsiPropertyRow.cs new file mode 100644 index 00000000..524f7929 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleMsiPropertyRow.cs @@ -0,0 +1,58 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the WixBundleMsiProperty table. + /// + public sealed class WixBundleMsiPropertyRow : Row + { + /// + /// Creates an WixBundleMsiProperty row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this WixBundleMsiProperty row belongs to and should get its column definitions from. + public WixBundleMsiPropertyRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the foreign key identifier to the ChainPackage row. + /// + public string ChainPackageId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets and sets the property identity. + /// + public string Name + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets and sets the value for the row. + /// + /// MsiProperty value for the row. + public string Value + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets and sets the condition for the row. + /// + /// MsiProperty condition for the row. + public string Condition + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleMspPackageAttributes.cs b/src/WixToolset.Data/Rows/WixBundleMspPackageAttributes.cs new file mode 100644 index 00000000..8f33e283 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleMspPackageAttributes.cs @@ -0,0 +1,14 @@ +// 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.Data.Rows +{ + using System; + + [Flags] + public enum WixBundleMspPackageAttributes + { + DisplayInternalUI = 0x1, + Slipstream = 0x2, + TargetUnspecified = 0x4, + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleMspPackageRow.cs b/src/WixToolset.Data/Rows/WixBundleMspPackageRow.cs new file mode 100644 index 00000000..eccf0aea --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleMspPackageRow.cs @@ -0,0 +1,99 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the ChainMspPackage table. + /// + public sealed class WixBundleMspPackageRow : Row + { + /// + /// Creates a ChainMspPackage row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this row belongs to and should get its column definitions from. + public WixBundleMspPackageRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixBundleMspPackage row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this row belongs to and should get its column definitions from. + public WixBundleMspPackageRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the foreign key identifier to the ChainPackage row. + /// + public string ChainPackageId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the raw MSP attributes of a patch. + /// + public WixBundleMspPackageAttributes Attributes + { + get { return (WixBundleMspPackageAttributes)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the patch code. + /// + public string PatchCode + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets the patch's manufacturer. + /// + public string Manufacturer + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets or sets the patch's xml. + /// + public string PatchXml + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets the display internal UI of a patch. + /// + public bool DisplayInternalUI + { + get { return 0 != (this.Attributes & WixBundleMspPackageAttributes.DisplayInternalUI); } + } + + /// + /// Gets whether to slipstream the patch. + /// + public bool Slipstream + { + get { return 0 != (this.Attributes & WixBundleMspPackageAttributes.Slipstream); } + } + + /// + /// Gets whether the patch targets an unspecified number of packages. + /// + public bool TargetUnspecified + { + get { return 0 != (this.Attributes & WixBundleMspPackageAttributes.TargetUnspecified); } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleMsuPackageRow.cs b/src/WixToolset.Data/Rows/WixBundleMsuPackageRow.cs new file mode 100644 index 00000000..0df635c2 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleMsuPackageRow.cs @@ -0,0 +1,57 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the WixBundleMsuPackage table. + /// + public sealed class WixBundleMsuPackageRow : Row + { + /// + /// Creates a WixBundleMsuPackage row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this row belongs to and should get its column definitions from. + public WixBundleMsuPackageRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixBundleMsuPackage row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this row belongs to and should get its column definitions from. + public WixBundleMsuPackageRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the foreign key identifier to the ChainPackage row. + /// + public string ChainPackageId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the detection condition the package. + /// + public string DetectCondition + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the KB of the package. + /// + public string MsuKB + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundlePackageAttributes.cs b/src/WixToolset.Data/Rows/WixBundlePackageAttributes.cs new file mode 100644 index 00000000..5b88ea5b --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundlePackageAttributes.cs @@ -0,0 +1,13 @@ +// 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.Data.Rows +{ + using System; + + [Flags] + public enum WixBundlePackageAttributes + { + Permanent = 0x1, + Visible = 0x2, + } +} diff --git a/src/WixToolset.Data/Rows/WixBundlePackageCommandLineRow.cs b/src/WixToolset.Data/Rows/WixBundlePackageCommandLineRow.cs new file mode 100644 index 00000000..eba647d5 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundlePackageCommandLineRow.cs @@ -0,0 +1,82 @@ +// 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.Data.Rows +{ + using System; + + /// + /// Specialization of a row for the WixBundlePackageCommandLine table. + /// + public class WixBundlePackageCommandLineRow : Row + { + /// + /// Creates a WixBundlePackageCommandLineRow row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this WixBundlePackageCommandLineRow row belongs to and should get its column definitions from. + public WixBundlePackageCommandLineRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates an WixBundlePackageCommandLineRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this WixBundlePackageCommandLineRow row belongs to and should get its column definitions from. + public WixBundlePackageCommandLineRow(SourceLineNumber sourceLineNumbers, Table table) + : base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the package identifier. + /// + /// The package identifier. + public string ChainPackageId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the command-line argument for installation. + /// + /// The command-line argument. + public string InstallArgument + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the command-line argument for uninstallation. + /// + /// The command-line argument. + public string UninstallArgument + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets the command-line argument for repair. + /// + /// The command-line argument. + public string RepairArgument + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets or sets the condition. + /// + /// The condition. + public string Condition + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundlePackageExitCodeRow.cs b/src/WixToolset.Data/Rows/WixBundlePackageExitCodeRow.cs new file mode 100644 index 00000000..410c191e --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundlePackageExitCodeRow.cs @@ -0,0 +1,51 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the ExitCode table. + /// + public class WixBundlePackageExitCodeRow : Row + { + /// + /// Creates a ExitCodeRow row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this Media row belongs to and should get its column definitions from. + public WixBundlePackageExitCodeRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a ExitCodeRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Media row belongs to and should get its column definitions from. + public WixBundlePackageExitCodeRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the foreign key identifier to the ChainPackage row. + /// + public string ChainPackageId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + public int? Code + { + get { return (null == this.Fields[1].Data) ? (int?)null : (int?)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + public ExitCodeBehaviorType Behavior + { + get { return (ExitCodeBehaviorType)this.Fields[2].Data; } + set { this.Fields[2].Data = (int)value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundlePackageRow.cs b/src/WixToolset.Data/Rows/WixBundlePackageRow.cs new file mode 100644 index 00000000..e278e9d8 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundlePackageRow.cs @@ -0,0 +1,226 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the WixBundlePackage table. + /// + public sealed class WixBundlePackageRow : Row + { + /// + /// Creates a WixBundlePackage row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this row belongs to and should get its column definitions from. + public WixBundlePackageRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixBundlePackage row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this row belongs to and should get its column definitions from. + public WixBundlePackageRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the foreign key to the WixChainItem. + /// + public string WixChainItemId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the item type. + /// + public WixBundlePackageType Type + { + get { return (WixBundlePackageType)this.Fields[1].Data; } + set { this.Fields[1].Data = (int)value; } + } + + /// + /// Gets or sets the indentifier of the package's payload. + /// + public string PackagePayload + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets the raw attributes of a package. + /// + public WixBundlePackageAttributes Attributes + { + get { return (WixBundlePackageAttributes)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets or sets the install condition of the package. + /// + public string InstallCondition + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets or sets the language of the package. + /// + public YesNoAlwaysType Cache + { + get { return (null == this.Fields[5].Data) ? YesNoAlwaysType.NotSet : (YesNoAlwaysType)this.Fields[5].Data; } + set { this.Fields[5].Data = (int)value; } + } + + /// + /// Gets or sets the indentifier of the package's cache. + /// + public string CacheId + { + get { return (string)this.Fields[6].Data; } + set { this.Fields[6].Data = value; } + } + + /// + /// Gets or sets whether the package is vital. + /// + public YesNoType Vital + { + get { return (null == this.Fields[7].Data) ? YesNoType.NotSet : (YesNoType)this.Fields[7].Data; } + set { this.Fields[7].Data = (int)value; } + } + + /// + /// Gets or sets whether the package is per-machine. + /// + public YesNoDefaultType PerMachine + { + get { return (null == this.Fields[8].Data) ? YesNoDefaultType.NotSet : (YesNoDefaultType)this.Fields[8].Data; } + set { this.Fields[8].Data = (int)value; } + } + + /// + /// Gets or sets the variable that points to the log for the package. + /// + public string LogPathVariable + { + get { return (string)this.Fields[9].Data; } + set { this.Fields[9].Data = value; } + } + + /// + /// Gets or sets the variable that points to the rollback log for the package. + /// + public string RollbackLogPathVariable + { + get { return (string)this.Fields[10].Data; } + set { this.Fields[10].Data = value; } + } + + /// + /// Gets or sets the size of the package. + /// + public long Size + { + get { return (long)this.Fields[11].Data; } + set { this.Fields[11].Data = value; } + } + + /// + /// Gets or sets the install size of the package. + /// + public long? InstallSize + { + get { return (long?)this.Fields[12].Data; } + set { this.Fields[12].Data = value; } + } + + /// + /// Gets or sets the version of the package. + /// + public string Version + { + get { return (string)this.Fields[13].Data; } + set { this.Fields[13].Data = value; } + } + + /// + /// Gets or sets the language of the package. + /// + public int Language + { + get { return (int)this.Fields[14].Data; } + set { this.Fields[14].Data = value; } + } + + /// + /// Gets or sets the display name of the package. + /// + public string DisplayName + { + get { return (string)this.Fields[15].Data; } + set { this.Fields[15].Data = value; } + } + + /// + /// Gets or sets the description of the package. + /// + public string Description + { + get { return (string)this.Fields[16].Data; } + set { this.Fields[16].Data = value; } + } + + /// + /// Gets or sets the rollback boundary identifier for the package. + /// + public string RollbackBoundary + { + get { return (string)this.Fields[17].Data; } + set { this.Fields[17].Data = value; } + } + + /// + /// Gets or sets the backward rollback boundary identifier for the package. + /// + public string RollbackBoundaryBackward + { + get { return (string)this.Fields[18].Data; } + set { this.Fields[18].Data = value; } + } + + /// + /// Gets or sets whether the package is x64. + /// + public YesNoType x64 + { + get { return (null == this.Fields[19].Data) ? YesNoType.NotSet : (YesNoType)this.Fields[19].Data; } + set { this.Fields[19].Data = (int)value; } + } + + /// + /// Gets whether the package is permanent. + /// + public bool Permanent + { + get { return 0 != (this.Attributes & WixBundlePackageAttributes.Permanent); } + } + + /// + /// Gets whether the package is visible. + /// + public bool Visible + { + get { return 0 != (this.Attributes & WixBundlePackageAttributes.Visible); } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundlePackageType.cs b/src/WixToolset.Data/Rows/WixBundlePackageType.cs new file mode 100644 index 00000000..a4277a8a --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundlePackageType.cs @@ -0,0 +1,15 @@ +// 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.Data.Rows +{ + /// + /// Types of bundle packages. + /// + public enum WixBundlePackageType + { + Exe, + Msi, + Msp, + Msu, + } +} diff --git a/src/WixToolset.Data/Rows/WixBundlePatchTargetCodeRow.cs b/src/WixToolset.Data/Rows/WixBundlePatchTargetCodeRow.cs new file mode 100644 index 00000000..e25f4a55 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundlePatchTargetCodeRow.cs @@ -0,0 +1,81 @@ +// 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.Data.Rows +{ + using System; + using System.Collections.Generic; + using System.Text; + + /// + /// Attributes for the PatchTargetCode table. + /// + [Flags] + public enum WixBundlePatchTargetCodeAttributes : int + { + None = 0, + + /// + /// The transform targets a specific ProductCode. + /// + TargetsProductCode = 1, + + /// + /// The transform targets a specific UpgradeCode. + /// + TargetsUpgradeCode = 2, + } + + /// + /// Specialization of a row for the PatchTargetCode table. + /// + public class WixBundlePatchTargetCodeRow : Row + { + /// + /// Creates a PatchTargetCodeRow row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this PatchTargetCode row belongs to and should get its column definitions from. + public WixBundlePatchTargetCodeRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a PatchTargetCodeRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this PatchTargetCode row belongs to and should get its column definitions from. + public WixBundlePatchTargetCodeRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + public string MspPackageId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + public string TargetCode + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + public WixBundlePatchTargetCodeAttributes Attributes + { + get { return (WixBundlePatchTargetCodeAttributes)this.Fields[2].Data; } + set { this.Fields[2].Data = (int)value; } + } + + public bool TargetsProductCode + { + get { return 0 != (WixBundlePatchTargetCodeAttributes.TargetsProductCode & this.Attributes); } + } + + public bool TargetsUpgradeCode + { + get { return 0 != (WixBundlePatchTargetCodeAttributes.TargetsUpgradeCode & this.Attributes); } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundlePayloadRow.cs b/src/WixToolset.Data/Rows/WixBundlePayloadRow.cs new file mode 100644 index 00000000..8aac8aa0 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundlePayloadRow.cs @@ -0,0 +1,185 @@ +// 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.Data.Rows +{ + using System; + using System.IO; + + /// + /// Specialization of a row for the PayloadInfo table. + /// + public class WixBundlePayloadRow : Row + { + /// + /// Creates a PayloadRow row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this Media row belongs to and should get its column definitions from. + public WixBundlePayloadRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a PayloadRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Media row belongs to and should get its column definitions from. + public WixBundlePayloadRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + public string Id + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + public string Name + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + public string SourceFile + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + public string DownloadUrl + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + public YesNoDefaultType Compressed + { + get { return (YesNoDefaultType)this.Fields[4].Data; } + set { this.Fields[4].Data = (int)value; } + } + + public string UnresolvedSourceFile + { + get { return (string)this.Fields[5].Data; } + set { this.Fields[5].Data = value; } + } + + public string DisplayName + { + get { return (string)this.Fields[6].Data; } + set { this.Fields[6].Data = value; } + } + + public string Description + { + get { return (string)this.Fields[7].Data; } + set { this.Fields[7].Data = value; } + } + + public bool EnableSignatureValidation + { + get { return (null != this.Fields[8].Data) && (1 == (int)this.Fields[8].Data); } + set { this.Fields[8].Data = value ? 1 : 0; } + } + + public int FileSize + { + get { return (int)this.Fields[9].Data; } + set { this.Fields[9].Data = value; } + } + + public string Version + { + get { return (string)this.Fields[10].Data; } + set { this.Fields[10].Data = value; } + } + + public string Hash + { + get { return (string)this.Fields[11].Data; } + set { this.Fields[11].Data = value; } + } + + public string PublicKey + { + get { return (string)this.Fields[12].Data; } + set { this.Fields[12].Data = value; } + } + + public string Thumbprint + { + get { return (string)this.Fields[13].Data; } + set { this.Fields[13].Data = value; } + } + + public string Catalog + { + get { return (string)this.Fields[14].Data; } + set { this.Fields[14].Data = value; } + } + + public string Container + { + get { return (string)this.Fields[15].Data; } + set { this.Fields[15].Data = value; } + } + + public string Package + { + get { return (string)this.Fields[16].Data; } + set { this.Fields[16].Data = value; } + } + + public bool ContentFile + { + get { return (null != this.Fields[17].Data) && (1 == (int)this.Fields[17].Data); } + set { this.Fields[17].Data = value ? 1 : 0; } + } + + public string EmbeddedId + { + get { return (string)this.Fields[18].Data; } + set { this.Fields[18].Data = value; } + } + + public bool LayoutOnly + { + get { return (null != this.Fields[19].Data) && (1 == (int)this.Fields[19].Data); } + set { this.Fields[19].Data = value ? 1 : 0; } + } + + public PackagingType Packaging + { + get + { + object data = this.Fields[20].Data; + return (null == data) ? PackagingType.Unknown : (PackagingType)data; + } + + set + { + if (PackagingType.Unknown == value) + { + this.Fields[20].Data = null; + } + else + { + this.Fields[20].Data = (int)value; + } + } + } + + public string ParentPackagePayload + { + get { return (string)this.Fields[21].Data; } + set { this.Fields[21].Data = value; } + } + + public string FullFileName + { + get { return String.IsNullOrEmpty(this.SourceFile) ? String.Empty : Path.GetFullPath(this.SourceFile); } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleRelatedPackageRow.cs b/src/WixToolset.Data/Rows/WixBundleRelatedPackageRow.cs new file mode 100644 index 00000000..ea9ff99e --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleRelatedPackageRow.cs @@ -0,0 +1,87 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the RelatedPackage table. + /// + public class WixBundleRelatedPackageRow : Row + { + /// + /// Creates a RelatedPackageRow row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this Media row belongs to and should get its column definitions from. + public WixBundleRelatedPackageRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a RelatedPackageRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Media row belongs to and should get its column definitions from. + public WixBundleRelatedPackageRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the foreign key identifier to the ChainPackage row. + /// + public string ChainPackageId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + public string Id + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + public string MinVersion + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + public string MaxVersion + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + public string Languages + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + public bool MinInclusive + { + get { return 1 == (int)this.Fields[5].Data; } + set { this.Fields[5].Data = value ? 1 : 0; } + } + + public bool MaxInclusive + { + get { return 1 == (int)this.Fields[6].Data; } + set { this.Fields[6].Data = value ? 1 : 0; } + } + + public bool LangInclusive + { + get { return 1 == (int)this.Fields[7].Data; } + set { this.Fields[7].Data = value ? 1 : 0; } + } + + public bool OnlyDetect + { + get { return 1 == (int)this.Fields[8].Data; } + set { this.Fields[8].Data = value ? 1 : 0; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleRollbackBoundaryRow.cs b/src/WixToolset.Data/Rows/WixBundleRollbackBoundaryRow.cs new file mode 100644 index 00000000..d0a994c0 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleRollbackBoundaryRow.cs @@ -0,0 +1,59 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the WixBundleRollbackBoundary table. + /// + public sealed class WixBundleRollbackBoundaryRow : Row + { + /// + /// Creates a WixBundleRollbackBoundary row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this row belongs to and should get its column definitions from. + public WixBundleRollbackBoundaryRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a RollbackBoundaryRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this row belongs to and should get its column definitions from. + public WixBundleRollbackBoundaryRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the foreign key identifier to the ChainPackage row. + /// + public string ChainPackageId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets whether the package is vital. + /// + /// Vitality of the package. + public YesNoType Vital + { + get { return (null == this.Fields[1].Data) ? YesNoType.NotSet : (YesNoType)this.Fields[1].Data; } + set { this.Fields[1].Data = (int)value; } + } + + /// + /// Gets or sets whether the rollback-boundary should be installed as an MSI transaction. + /// + /// Vitality of the package. + public YesNoType Transaction + { + get { return (null == this.Fields[2].Data) ? YesNoType.NotSet : (YesNoType)this.Fields[2].Data; } + set { this.Fields[2].Data = (int)value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleRow.cs b/src/WixToolset.Data/Rows/WixBundleRow.cs new file mode 100644 index 00000000..4c96d6cc --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleRow.cs @@ -0,0 +1,228 @@ +// 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.Data.Rows +{ + using System; + + /// + /// Bundle info for binding Bundles. + /// + public class WixBundleRow : Row + { + /// + /// Creates a WixBundleRow row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this WixBundleRow row belongs to and should get its column definitions from. + public WixBundleRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixBundleRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this WixBundleRow row belongs to and should get its column definitions from. + public WixBundleRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + public string Version + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + public string Copyright + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + public string Name + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + public string AboutUrl + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + public int DisableModify + { + get { return (null == this.Fields[4].Data) ? 0 : (int)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + public bool DisableRemove + { + get { return (null != this.Fields[5].Data && 0 != (int)this.Fields[5].Data); } + set { this.Fields[5].Data = value ? 1 : 0; } + } + + // There is no 6. It used to be DisableRepair. + + public string HelpTelephone + { + get { return (string)this.Fields[7].Data; } + set { this.Fields[7].Data = value; } + } + + public string HelpLink + { + get { return (string)this.Fields[8].Data; } + set { this.Fields[8].Data = value; } + } + + public string Publisher + { + get { return (string)this.Fields[9].Data; } + set { this.Fields[9].Data = value; } + } + + public string UpdateUrl + { + get { return (string)this.Fields[10].Data; } + set { this.Fields[10].Data = value; } + } + + public YesNoDefaultType Compressed + { + get { return (null == this.Fields[11].Data) ? YesNoDefaultType.Default : (0 == (int)this.Fields[11].Data) ? YesNoDefaultType.No : YesNoDefaultType.Yes; } + set { this.Fields[11].Data = (int)value; } + } + + public PackagingType DefaultPackagingType + { + get { return (YesNoDefaultType.No == this.Compressed) ? PackagingType.External : PackagingType.Embedded; } + } + + public string LogPathPrefixExtension + { + get { return (string)this.Fields[12].Data ?? String.Empty; } + set { this.Fields[12].Data = value; } + } + + public string LogPathVariable + { + get + { + string[] logVariableAndPrefixExtension = this.LogPathPrefixExtension.Split(':'); + return logVariableAndPrefixExtension[0]; + } + } + + public string LogPrefix + { + get + { + string[] logVariableAndPrefixExtension = this.LogPathPrefixExtension.Split(':'); + if (2 > logVariableAndPrefixExtension.Length) + { + return String.Empty; + } + string logPrefixAndExtension = logVariableAndPrefixExtension[1]; + int extensionIndex = logPrefixAndExtension.LastIndexOf('.'); + return logPrefixAndExtension.Substring(0, extensionIndex); + } + } + + public string LogExtension + { + get + { + string[] logVariableAndPrefixExtension = this.LogPathPrefixExtension.Split(':'); + if (2 > logVariableAndPrefixExtension.Length) + { + return String.Empty; + } + string logPrefixAndExtension = logVariableAndPrefixExtension[1]; + int extensionIndex = logPrefixAndExtension.LastIndexOf('.'); + return logPrefixAndExtension.Substring(extensionIndex + 1); + } + } + + public string IconPath + { + get { return (string)this.Fields[13].Data; } + set { this.Fields[13].Data = value; } + } + + public string SplashScreenBitmapPath + { + get { return (string)this.Fields[14].Data; } + set { this.Fields[14].Data = value; } + } + + public string Condition + { + get { return (string)this.Fields[15].Data; } + set { this.Fields[15].Data = value; } + } + + public string Tag + { + get { return (string)this.Fields[16].Data; } + set { this.Fields[16].Data = value; } + } + + public Platform Platform + { + get { return (Platform)Enum.Parse(typeof(Platform), (string)this.Fields[17].Data); } + set { this.Fields[17].Data = value.ToString(); } + } + + public string ParentName + { + get { return (string)this.Fields[18].Data; } + set { this.Fields[18].Data = value; } + } + + public string UpgradeCode + { + get { return (string)this.Fields[19].Data; } + set { this.Fields[19].Data = value; } + } + + public Guid BundleId + { + get + { + if (null == this.Fields[20].Data) + { + this.Fields[20].Data = Guid.NewGuid().ToString("B"); + } + + return new Guid((string)this.Fields[20].Data); + } + + set { this.Fields[20].Data = value.ToString(); } + } + + public string ProviderKey + { + get + { + if (null == this.Fields[21].Data) + { + this.Fields[21].Data = this.BundleId.ToString("B"); + } + + return (string)this.Fields[21].Data; + } + + set { this.Fields[21].Data = value; } + } + + public bool PerMachine + { + get { return (null != this.Fields[22].Data && 0 != (int)this.Fields[22].Data); } + set { this.Fields[22].Data = value ? 1 : 0; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleSlipstreamMspRow.cs b/src/WixToolset.Data/Rows/WixBundleSlipstreamMspRow.cs new file mode 100644 index 00000000..d11b23ef --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleSlipstreamMspRow.cs @@ -0,0 +1,48 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the SlipstreamMsp table. + /// + public class WixBundleSlipstreamMspRow : Row + { + /// + /// Creates a SlipstreamMspRow row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this Media row belongs to and should get its column definitions from. + public WixBundleSlipstreamMspRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a SlipstreamMspRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Media row belongs to and should get its column definitions from. + public WixBundleSlipstreamMspRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the foreign key identifier to the ChainPackage row. + /// + public string ChainPackageId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the foreign key identifier to the ChainPackage row for the MSP package. + /// + public string MspPackageId + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleUpdateRow.cs b/src/WixToolset.Data/Rows/WixBundleUpdateRow.cs new file mode 100644 index 00000000..e0150685 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleUpdateRow.cs @@ -0,0 +1,38 @@ +// 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.Data.Rows +{ + using System; + + /// + /// Bundle update info for binding Bundles. + /// + public class WixBundleUpdateRow : Row + { + /// + /// Creates a WixBundleUpdateRow row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this WixBundleUpdateRow row belongs to and should get its column definitions from. + public WixBundleUpdateRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixBundleUpdateRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this WixBundleUpdateRow row belongs to and should get its column definitions from. + public WixBundleUpdateRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + public string Location + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixBundleVariableRow.cs b/src/WixToolset.Data/Rows/WixBundleVariableRow.cs new file mode 100644 index 00000000..e7ff1a4d --- /dev/null +++ b/src/WixToolset.Data/Rows/WixBundleVariableRow.cs @@ -0,0 +1,80 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the Variable table. + /// + public sealed class WixBundleVariableRow : Row + { + /// + /// Creates a Variable row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this Media row belongs to and should get its column definitions from. + public WixBundleVariableRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a Variable row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Media row belongs to and should get its column definitions from. + public WixBundleVariableRow(SourceLineNumber sourceLineNumbers, Table table) + : base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the variable identifier. + /// + /// The variable identifier. + public string Id + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the variable's value. + /// + /// The variable's value. + public string Value + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the variable's type. + /// + /// The variable's type. + public string Type + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets whether this variable is hidden. + /// + /// Whether this variable is hidden. + public bool Hidden + { + get { return (null == this.Fields[3].Data || 0 == ((int)this.Fields[3].Data)) ? false : true; } + set { this.Fields[3].Data = value ? 1 : 0; } + } + + /// + /// Gets or sets whether this variable is persisted. + /// + /// Whether this variable is persisted. + public bool Persisted + { + get { return (null == this.Fields[4].Data || 0 == ((int)this.Fields[4].Data)) ? false : true; } + set { this.Fields[4].Data = value ? 1 : 0; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixChainAttributes.cs b/src/WixToolset.Data/Rows/WixChainAttributes.cs new file mode 100644 index 00000000..017505bc --- /dev/null +++ b/src/WixToolset.Data/Rows/WixChainAttributes.cs @@ -0,0 +1,15 @@ +// 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.Data.Rows +{ + using System; + + [Flags] + public enum WixChainAttributes + { + None = 0x0, + DisableRollback = 0x1, + DisableSystemRestore = 0x2, + ParallelCache = 0x4, + } +} diff --git a/src/WixToolset.Data/Rows/WixChainItemRow.cs b/src/WixToolset.Data/Rows/WixChainItemRow.cs new file mode 100644 index 00000000..12538d71 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixChainItemRow.cs @@ -0,0 +1,39 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the WixChainItem table. + /// + public sealed class WixChainItemRow : Row + { + /// + /// Creates a WixChainItem row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this ChainItem row belongs to and should get its column definitions from. + public WixChainItemRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixChainItem row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this ChainItem row belongs to and should get its column definitions from. + public WixChainItemRow(SourceLineNumber sourceLineNumbers, Table table) + : base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the WixChainItem identifier. + /// + public string Id + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixChainRow.cs b/src/WixToolset.Data/Rows/WixChainRow.cs new file mode 100644 index 00000000..94600efa --- /dev/null +++ b/src/WixToolset.Data/Rows/WixChainRow.cs @@ -0,0 +1,63 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the WixChain table. + /// + public sealed class WixChainRow : Row + { + /// + /// Creates a WixChain row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this row belongs to and should get its column definitions from. + public WixChainRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixChainRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this row belongs to and should get its column definitions from. + public WixChainRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the raw chain attributes. + /// + public WixChainAttributes Attributes + { + get { return (WixChainAttributes)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets the disable rollback state of a chain. + /// + public bool DisableRollback + { + get { return 0 != (this.Attributes & WixChainAttributes.DisableRollback); } + } + + /// + /// Gets disable system restore state of a chain. + /// + public bool DisableSystemRestore + { + get { return 0 != (this.Attributes & WixChainAttributes.DisableSystemRestore); } + } + + /// + /// Gets parallel cache of a chain. + /// + public bool ParallelCache + { + get { return 0 != (this.Attributes & WixChainAttributes.ParallelCache); } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixComplexReferenceRow.cs b/src/WixToolset.Data/Rows/WixComplexReferenceRow.cs new file mode 100644 index 00000000..b516c5e9 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixComplexReferenceRow.cs @@ -0,0 +1,208 @@ +// 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.Data.Rows +{ + using System; + using System.Diagnostics; + using System.Diagnostics.CodeAnalysis; + using System.Xml; + + /// + /// Specialization of a row for the WixComplexReference table. + /// + public sealed class WixComplexReferenceRow : Row, IComparable + { + /// + /// Creates a WixComplexReferenceRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this row belongs to and should get its column definitions from. + public WixComplexReferenceRow(SourceLineNumber sourceLineNumbers, Table table) + : base(sourceLineNumbers, table) + { + } + + /// + /// Gets the parent type of the complex reference. + /// + /// Parent type of the complex reference. + public ComplexReferenceParentType ParentType + { + get { return (ComplexReferenceParentType)Enum.ToObject(typeof(ComplexReferenceParentType), (int)this.Fields[1].Data); } + set { this.Fields[1].Data = (int)value; } + } + + /// + /// Gets or sets the parent identifier of the complex reference. + /// + /// Parent identifier of the complex reference. + public string ParentId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets the parent language of the complex reference. + /// + /// Parent language of the complex reference. + public string ParentLanguage + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets the child type of the complex reference. + /// + /// Child type of the complex reference. + public ComplexReferenceChildType ChildType + { + get { return (ComplexReferenceChildType)Enum.ToObject(typeof(ComplexReferenceChildType), (int)this.Fields[4].Data); } + set { this.Fields[4].Data = (int)value; } + } + + /// + /// Gets the child identifier of the complex reference. + /// + /// Child identifier of the complex reference. + public string ChildId + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets if this is the primary complex reference. + /// + /// true if primary complex reference. + public bool IsPrimary + { + get + { + return (0x1 == ((int)this.Fields[5].Data & 0x1)); + } + + set + { + if (null == this.Fields[5].Data) + { + this.Fields[5].Data = 0; + } + + if (value) + { + this.Fields[5].Data = (int)this.Fields[5].Data | 0x1; + } + else + { + this.Fields[5].Data = (int)this.Fields[5].Data & ~0x1; + } + } + } + + /// + /// Determines if two complex references are equivalent. + /// + /// Complex reference to compare. + /// True if complex references are equivalent. + public override bool Equals(object obj) + { + return 0 == this.CompareTo(obj); + } + + /// + /// Gets the hash code for the complex reference. + /// + /// Hash code for the complex reference. + public override int GetHashCode() + { + return this.ChildType.GetHashCode() ^ this.ChildId.GetHashCode() ^ this.ParentType.GetHashCode() ^ this.ParentLanguage.GetHashCode() ^ this.ParentId.GetHashCode() ^ this.IsPrimary.GetHashCode(); + } + + /// + /// Compares two complex references. + /// + /// Complex reference to compare to. + /// Zero if the objects are equivalent, negative number if the provided object is less, positive if greater. + public int CompareTo(object obj) + { + int comparison = this.CompareToWithoutConsideringPrimary(obj); + if (0 == comparison) + { + comparison = ((WixComplexReferenceRow)obj).IsPrimary.CompareTo(this.IsPrimary); // Note: the order of these is purposely switched to ensure that "Yes" is lower than "No" and "NotSet" + } + return comparison; + } + + /// + /// Compares two complex references without considering the primary bit. + /// + /// Complex reference to compare to. + /// Zero if the objects are equivalent, negative number if the provided object is less, positive if greater. + [SuppressMessage("Microsoft.Globalization", "CA1303:DoNotPassLiteralsAsLocalizedParameters", MessageId = "System.ArgumentException.#ctor(System.String,System.String)")] + public int CompareToWithoutConsideringPrimary(object obj) + { + WixComplexReferenceRow other = obj as WixComplexReferenceRow; + if (null == other) + { + throw new ArgumentException(WixDataStrings.EXP_ExpectedComplexReferenceType, "obj"); + } + + int comparison = this.ChildType - other.ChildType; + if (0 == comparison) + { + comparison = String.Compare(this.ChildId, other.ChildId, StringComparison.Ordinal); + if (0 == comparison) + { + comparison = this.ParentType - other.ParentType; + if (0 == comparison) + { + string thisParentLanguage = null == this.ParentLanguage ? String.Empty : this.ParentLanguage; + string otherParentLanguage = null == other.ParentLanguage ? String.Empty : other.ParentLanguage; + comparison = String.Compare(thisParentLanguage, otherParentLanguage, StringComparison.Ordinal); + if (0 == comparison) + { + comparison = String.Compare(this.ParentId, other.ParentId, StringComparison.Ordinal); + } + } + } + } + + return comparison; + } + + /// + /// Creates a shallow copy of the ComplexReference. + /// + /// A shallow copy of the ComplexReference. + public WixComplexReferenceRow Clone() + { + WixComplexReferenceRow wixComplexReferenceRow = new WixComplexReferenceRow(this.SourceLineNumbers, this.Table); + wixComplexReferenceRow.ParentType = this.ParentType; + wixComplexReferenceRow.ParentId = this.ParentId; + wixComplexReferenceRow.ParentLanguage = this.ParentLanguage; + wixComplexReferenceRow.ChildType = this.ChildType; + wixComplexReferenceRow.ChildId = this.ChildId; + wixComplexReferenceRow.IsPrimary = this.IsPrimary; + + return wixComplexReferenceRow; + } + + /// + /// Changes all of the parent references to point to the passed in parent reference. + /// + /// New parent complex reference. + public void Reparent(WixComplexReferenceRow parent) + { + this.ParentId = parent.ParentId; + this.ParentLanguage = parent.ParentLanguage; + this.ParentType = parent.ParentType; + + if (!this.IsPrimary) + { + this.IsPrimary = parent.IsPrimary; + } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixDeltaPatchFileRow.cs b/src/WixToolset.Data/Rows/WixDeltaPatchFileRow.cs new file mode 100644 index 00000000..000779d9 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixDeltaPatchFileRow.cs @@ -0,0 +1,142 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the WixDeltaPatchFile table. + /// + public sealed class WixDeltaPatchFileRow : Row + { + /// + /// Creates a WixDeltaPatchFile row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this Media row belongs to and should get its column definitions from. + public WixDeltaPatchFileRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixDeltaPatchFile row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this File row belongs to and should get its column definitions from. + public WixDeltaPatchFileRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the primary key of the file row. + /// + /// Primary key of the file row. + public string File + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the delta patch retain-length list for the file. + /// + /// RetainLength list for the file. + public string RetainLengths + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the previous delta patch retain-length list for the file. + /// + /// Previous RetainLength list for the file. + public string PreviousRetainLengths + { + get { return this.Fields[1].PreviousData; } + set { this.Fields[1].PreviousData = value; } + } + + /// + /// Gets or sets the delta patch ignore-offset list for the file. + /// + /// IgnoreOffset list for the file. + public string IgnoreOffsets + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets the previous delta patch ignore-offset list for the file. + /// + /// Previous IgnoreOffset list for the file. + public string PreviousIgnoreOffsets + { + get { return this.Fields[2].PreviousData; } + set { this.Fields[2].PreviousData = value; } + } + + /// + /// Gets or sets the delta patch ignore-length list for the file. + /// + /// IgnoreLength list for the file. + public string IgnoreLengths + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets or sets the previous delta patch ignore-length list for the file. + /// + /// Previous IgnoreLength list for the file. + public string PreviousIgnoreLengths + { + get { return this.Fields[3].PreviousData; } + set { this.Fields[3].PreviousData = value; } + } + + /// + /// Gets or sets the delta patch retain-offset list for the file. + /// + /// RetainOffset list for the file. + public string RetainOffsets + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets or sets the previous delta patch retain-offset list for the file. + /// + /// PreviousRetainOffset list for the file. + public string PreviousRetainOffsets + { + get { return this.Fields[4].PreviousData; } + set { this.Fields[4].PreviousData = value; } + } + + /// + /// Gets or sets the symbol paths for the file. + /// + /// SymbolPath list for the file. + /// This is set during binding. + public string Symbols + { + get { return (string)this.Fields[5].Data; } + set { this.Fields[5].Data = value; } + } + + /// + /// Gets or sets the previous symbol paths for the file. + /// + /// PreviousSymbolPath list for the file. + /// This is set during binding. + public string PreviousSymbols + { + get { return (string)this.Fields[5].PreviousData; } + set { this.Fields[5].PreviousData = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixDeltaPatchSymbolPathsRow.cs b/src/WixToolset.Data/Rows/WixDeltaPatchSymbolPathsRow.cs new file mode 100644 index 00000000..b6c0b840 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixDeltaPatchSymbolPathsRow.cs @@ -0,0 +1,58 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the WixDeltaPatchSymbolPaths table. + /// + public sealed class WixDeltaPatchSymbolPathsRow : Row + { + /// + /// Creates a WixDeltaPatchSymbolPaths row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this row belongs to and should get its column definitions from. + public WixDeltaPatchSymbolPathsRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixDeltaPatchSymbolPaths row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this row belongs to and should get its column definitions from. + public WixDeltaPatchSymbolPathsRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the identifier the symbol paths apply to. + /// + /// RetainLength list for the file. + public string Id + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the type of the identifier. + /// + public SymbolPathType Type + { + get { return (SymbolPathType)this.Fields[1].AsInteger(); } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the delta patch symbol paths. + /// + public string SymbolPaths + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixFileRow.cs b/src/WixToolset.Data/Rows/WixFileRow.cs new file mode 100644 index 00000000..52723f88 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixFileRow.cs @@ -0,0 +1,161 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the WixFile table. + /// + public sealed class WixFileRow : Row + { + /// + /// Creates a WixFile row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this row belongs to and should get its column definitions from. + public WixFileRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixFile row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this row belongs to and should get its column definitions from. + public WixFileRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the primary key of the file row. + /// + /// Primary key of the file row. + public string File + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the assembly type of the file row. + /// + /// Assembly type of the file row. + public FileAssemblyType AssemblyType + { + get { return (null == this.Fields[1]) ? FileAssemblyType.NotAnAssembly : (FileAssemblyType)this.Fields[1].AsInteger(); } + set { this.Fields[1].Data = (int)value; } + } + + /// + /// Gets or sets the identifier for the assembly manifest. + /// + /// Identifier for the assembly manifest. + public string AssemblyManifest + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets or sets the application for the assembly. + /// + /// Application for the assembly. + public string AssemblyApplication + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets or sets the directory of the file. + /// + /// Directory of the file. + public string Directory + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets or sets the disk id for this file. + /// + /// Disk id for the file. + public int DiskId + { + get { return (int)this.Fields[5].Data; } + set { this.Fields[5].Data = value; } + } + + /// + /// Gets or sets the source location to the file. + /// + /// Source location to the file. + public string Source + { + get { return (string)this.Fields[6].Data; } + set { this.Fields[6].Data = value; } + } + + /// + /// Gets or sets the source location to the file. + /// + /// Source location to the file. + public string PreviousSource + { + get { return (string)this.Fields[6].PreviousData; } + set { this.Fields[6].PreviousData = value; } + } + + /// + /// Gets or sets the architecture the file executes on. + /// + /// Architecture the file executes on. + public string ProcessorArchitecture + { + get { return (string)this.Fields[7].Data; } + set { this.Fields[7].Data = value; } + } + + /// + /// Gets or sets the patch group of a patch-added file. + /// + /// The patch group of a patch-added file. + public int PatchGroup + { + get { return (null == this.Fields[8].Data) ? 0 : (int)this.Fields[8].Data; } + set { this.Fields[8].Data = value; } + } + + /// + /// Gets or sets the attributes on a file. + /// + /// Attributes on a file. + public int Attributes + { + get { return (int)this.Fields[9].Data; } + set { this.Fields[9].Data = value; } + } + + /// + /// Gets or sets the patching attributes to the file. + /// + /// Patching attributes of the file. + public PatchAttributeType PatchAttributes + { + get { return (PatchAttributeType)this.Fields[10].AsInteger(); } + set { this.Fields[10].Data = (int)value; } + } + + /// + /// Gets or sets the path to the delta patch header. + /// + /// Patch header path. + /// Set by the binder only when doing delta patching. + public string DeltaPatchHeaderSource + { + get { return (string)this.Fields[11].Data; } + set { this.Fields[11].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixGroupRow.cs b/src/WixToolset.Data/Rows/WixGroupRow.cs new file mode 100644 index 00000000..d36338d1 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixGroupRow.cs @@ -0,0 +1,62 @@ +// 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.Data.Rows +{ + using System; + + /// + /// Specialization of a row for the WixGroup table. + /// + public sealed class WixGroupRow : Row + { + /// + /// Creates a WixGroupRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this row belongs to and should get its column definitions from. + public WixGroupRow(SourceLineNumber sourceLineNumbers, Table table) + : base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the parent identifier of the complex reference. + /// + /// Parent identifier of the complex reference. + public string ParentId + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets the parent type of the complex reference. + /// + /// Parent type of the complex reference. + public ComplexReferenceParentType ParentType + { + get { return (ComplexReferenceParentType)Enum.Parse(typeof(ComplexReferenceParentType), (string)this.Fields[1].Data); } + set { this.Fields[1].Data = value.ToString(); } + } + + /// + /// Gets the child identifier of the complex reference. + /// + /// Child identifier of the complex reference. + public string ChildId + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets the child type of the complex reference. + /// + /// Child type of the complex reference. + public ComplexReferenceChildType ChildType + { + get { return (ComplexReferenceChildType)Enum.Parse(typeof(ComplexReferenceChildType), (string)this.Fields[3].Data); } + set { this.Fields[3].Data = value.ToString(); } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixMediaRow.cs b/src/WixToolset.Data/Rows/WixMediaRow.cs new file mode 100644 index 00000000..c1b3e155 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixMediaRow.cs @@ -0,0 +1,60 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the WixMedia table. + /// + public sealed class WixMediaRow : Row + { + /// + /// Creates a WixMedia row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this Media row belongs to and should get its column definitions from. + public WixMediaRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixMedia row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this Media row belongs to and should get its column definitions from. + public WixMediaRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the disk id for this media. + /// + /// Disk id for the media. + public int DiskId + { + get { return (int)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the compression level for this media row. + /// + /// Compression level. + public CompressionLevel? CompressionLevel + { + get { return (CompressionLevel?)this.Fields[1].AsNullableInteger(); } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the layout location for this media row. + /// + /// Layout location to the root of the media. + public string Layout + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixMediaTemplateRow.cs b/src/WixToolset.Data/Rows/WixMediaTemplateRow.cs new file mode 100644 index 00000000..27c5ccce --- /dev/null +++ b/src/WixToolset.Data/Rows/WixMediaTemplateRow.cs @@ -0,0 +1,81 @@ +// 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.Data.Rows +{ + /// + /// Specialization of a row for the MediaTemplate table. + /// + public sealed class WixMediaTemplateRow : Row + { + /// + /// Creates a MediaTemplate row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this MediaTeplate row belongs to and should get its column definitions from. + public WixMediaTemplateRow(SourceLineNumber sourceLineNumbers, Table table) + : base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the cabinet template name for this media template row. + /// + /// Cabinet name. + public string CabinetTemplate + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the compression level for this media template row. + /// + /// Compression level. + public CompressionLevel? CompressionLevel + { + get { return (CompressionLevel?)this.Fields[1].AsNullableInteger(); } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets the disk prompt for this media template row. + /// + /// Disk prompt. + public string DiskPrompt + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + + /// + /// Gets or sets the volume label for this media template row. + /// + /// Volume label. + public string VolumeLabel + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets or sets the maximum uncompressed media size for this media template row. + /// + /// Disk id. + public int MaximumUncompressedMediaSize + { + get { return (int)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets or sets the Maximum Cabinet Size For Large File Splitting for this media template row. + /// + /// Disk id. + public int MaximumCabinetSizeForLargeFileSplitting + { + get { return (int)this.Fields[5].Data; } + set { this.Fields[5].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixMergeRow.cs b/src/WixToolset.Data/Rows/WixMergeRow.cs new file mode 100644 index 00000000..54f2125c --- /dev/null +++ b/src/WixToolset.Data/Rows/WixMergeRow.cs @@ -0,0 +1,149 @@ +// 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.Data.Rows +{ + using System; + using System.Globalization; + using System.Text; + using System.Xml; + + /// + /// Specialization of a row for tracking merge statements. + /// + public sealed class WixMergeRow : Row + { + /// + /// Creates a Merge row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this Merge row belongs to and should get its column definitions from. + public WixMergeRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// Creates a Merge row that belongs to a table. + /// Original source lines for this row. + /// Table this Merge row belongs to and should get its column definitions from. + public WixMergeRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets and sets the id for a merge row. + /// + /// Id for the row. + public string Id + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets and sets the language for a merge row. + /// + /// Language for the row. + public string Language + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets and sets the directory for a merge row. + /// + /// Direcotory for the row. + public string Directory + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + /// + /// Gets and sets the path to the merge module for a merge row. + /// + /// Source path for the row. + public string SourceFile + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + /// + /// Gets and sets the disk id the merge module should be placed on for a merge row. + /// + /// Disk identifier for row. + public int DiskId + { + get { return (int)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + /// + /// Gets and sets the compression value for a merge row. + /// + /// Compression for a merge row. + public YesNoType FileCompression + { + get + { + if (null == this.Fields[5].Data) + { + return YesNoType.NotSet; + } + else if (1 == (int)this.Fields[5].Data) + { + return YesNoType.Yes; + } + else if (0 == (int)this.Fields[5].Data) + { + return YesNoType.No; + } + else + { + throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, WixDataStrings.EXP_MergeTableFileCompressionColumnContainsInvalidValue, this.Fields[5].Data)); + } + } + set + { + if (YesNoType.Yes == value) + { + this.Fields[5].Data = 1; + } + else if (YesNoType.No == value) + { + this.Fields[5].Data = 0; + } + else if (YesNoType.NotSet == value) + { + this.Fields[5].Data = null; + } + else + { + throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, WixDataStrings.EXP_CannotSetMergeTableFileCompressionColumnToInvalidValue, value)); + } + } + } + + /// + /// Gets and sets the configuration data for a merge row. + /// + /// Comma delimited string of "name=value" pairs. + public string ConfigurationData + { + get { return (string)this.Fields[6].Data; } + set { this.Fields[6].Data = value; } + } + + /// + /// Gets and sets the primary feature for a merge row. + /// + /// The primary feature for a merge row. + public string Feature + { + get { return (string)this.Fields[7].Data; } + set { this.Fields[7].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixPayloadPropertiesRow.cs b/src/WixToolset.Data/Rows/WixPayloadPropertiesRow.cs new file mode 100644 index 00000000..2e5f53ad --- /dev/null +++ b/src/WixToolset.Data/Rows/WixPayloadPropertiesRow.cs @@ -0,0 +1,81 @@ +//------------------------------------------------------------------------------------------------- +// +// Copyright (c) 2004, Outercurve Foundation. +// This software is released under Microsoft Reciprocal License (MS-RL). +// The license and further copyright text can be found in the file +// LICENSE.TXT at the root directory of the distribution. +// +//------------------------------------------------------------------------------------------------- + +namespace WixToolset.Data.Rows +{ + using System; + + /// + /// Specialization of a row for the WixPayloadProperties table. + /// + public class WixPayloadPropertiesRow : Row + { + /// + /// Creates a WixPayloadProperties row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this WixPayloadProperties row belongs to and should get its column definitions from. + public WixPayloadPropertiesRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixPayloadProperties row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this WixPayloadProperties row belongs to and should get its column definitions from. + public WixPayloadPropertiesRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + public string Id + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + public string Package + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + public string Container + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + public string Name + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + public string Size + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + + public string DownloadUrl + { + get { return (string)this.Fields[5].Data; } + set { this.Fields[5].Data = value; } + } + + public string LayoutOnly + { + get { return (string)this.Fields[6].Data; } + set { this.Fields[6].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixPropertyRow.cs b/src/WixToolset.Data/Rows/WixPropertyRow.cs new file mode 100644 index 00000000..5285195c --- /dev/null +++ b/src/WixToolset.Data/Rows/WixPropertyRow.cs @@ -0,0 +1,118 @@ +// 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.Data.Rows +{ + using System; + using System.Globalization; + + /// + /// Specialization of a row for the WixProperty table. + /// + public sealed class WixPropertyRow : Row + { + /// Creates a WixProperty row that belongs to a table. + /// Original source lines for this row. + /// Table this WixProperty row belongs to and should get its column definitions from. + public WixPropertyRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + /// + /// Gets and sets the id for this property row. + /// + /// Id for the property. + public string Id + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets and sets if this is an admin property row. + /// + /// Flag if this is an admin property. + public bool Admin + { + get + { + return (0x1 == (Convert.ToInt32(this.Fields[1].Data, CultureInfo.InvariantCulture) & 0x1)); + } + + set + { + if (null == this.Fields[1].Data) + { + this.Fields[1].Data = 0; + } + + if (value) + { + this.Fields[1].Data = (int)this.Fields[1].Data | 0x1; + } + else + { + this.Fields[1].Data = (int)this.Fields[1].Data & ~0x1; + } + } + } + + /// + /// Gets and sets if this is a hidden property row. + /// + /// Flag if this is a hidden property. + public bool Hidden + { + get + { + return (0x2 == (Convert.ToInt32(this.Fields[1].Data, CultureInfo.InvariantCulture) & 0x2)); + } + + set + { + if (null == this.Fields[1].Data) + { + this.Fields[1].Data = 0; + } + + if (value) + { + this.Fields[1].Data = (int)this.Fields[1].Data | 0x2; + } + else + { + this.Fields[1].Data = (int)this.Fields[1].Data & ~0x2; + } + } + } + + /// + /// Gets and sets if this is a secure property row. + /// + /// Flag if this is a secure property. + public bool Secure + { + get + { + return (0x4 == (Convert.ToInt32(this.Fields[1].Data, CultureInfo.InvariantCulture) & 0x4)); + } + + set + { + if (null == this.Fields[1].Data) + { + this.Fields[1].Data = 0; + } + + if (value) + { + this.Fields[1].Data = (int)this.Fields[1].Data | 0x4; + } + else + { + this.Fields[1].Data = (int)this.Fields[1].Data & ~0x4; + } + } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixRelatedBundleRow.cs b/src/WixToolset.Data/Rows/WixRelatedBundleRow.cs new file mode 100644 index 00000000..95fffde5 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixRelatedBundleRow.cs @@ -0,0 +1,52 @@ +// 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.Data.Rows +{ + using Serialize = WixToolset.Data.Serialize; + + /// + /// Specialization of a row for the RelatedBundle table. + /// + public sealed class WixRelatedBundleRow : Row + { + /// + /// Creates a RelatedBundle row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this RelatedBundle row belongs to and should get its column definitions from. + public WixRelatedBundleRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a RelatedBundle row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this RelatedBundle row belongs to and should get its column definitions from. + public WixRelatedBundleRow(SourceLineNumber sourceLineNumbers, Table table) + : base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the related bundle identifier. + /// + /// The related bundle identifier. + public string Id + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the related bundle action. + /// + /// The related bundle action. + public Serialize.RelatedBundle.ActionType Action + { + get { return (Serialize.RelatedBundle.ActionType)this.Fields[1].Data; } + set { this.Fields[1].Data = (int)value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixSimpleReferenceRow.cs b/src/WixToolset.Data/Rows/WixSimpleReferenceRow.cs new file mode 100644 index 00000000..3a2cf8f1 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixSimpleReferenceRow.cs @@ -0,0 +1,63 @@ +// 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.Data.Rows +{ + using System; + using System.Diagnostics; + using System.Xml; + + /// + /// Specialization of a row for the WixSimpleReference table. + /// + public sealed class WixSimpleReferenceRow : Row + { + /// + /// Creates a WixSimpleReferenceRow that belongs to a table. + /// + /// Original source lines for this row. + /// Table this row belongs to and should get its column definitions from. + public WixSimpleReferenceRow(SourceLineNumber sourceLineNumbers, Table table) + : base(sourceLineNumbers, table) + { + } + + /// + /// Creates a WixSimpleReferenceRow that belongs to a table. + /// + /// Original source lines for this row. + /// Table definitions for this row. + public WixSimpleReferenceRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDefinitions) + : base(sourceLineNumbers, tableDefinitions) + { + } + + /// + /// Gets or sets the primary keys of the simple reference. + /// + /// The primary keys of the simple reference. + public string PrimaryKeys + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets the symbolic name. + /// + /// Symbolic name. + public string SymbolicName + { + get { return String.Concat(this.TableName, ":", this.PrimaryKeys); } + } + + /// + /// Gets or sets the table name of the simple reference. + /// + /// The table name of the simple reference. + public string TableName + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixUpdateRegistrationRow.cs b/src/WixToolset.Data/Rows/WixUpdateRegistrationRow.cs new file mode 100644 index 00000000..8d86f970 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixUpdateRegistrationRow.cs @@ -0,0 +1,62 @@ +// 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.Data.Rows +{ + using System; + + /// + /// Update registration information for Binding. + /// + public class WixUpdateRegistrationRow : Row + { + /// + /// Creates a WixUpdateRegistrationRow row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this WixUpdateRegistrationRow row belongs to and should get its column definitions from. + public WixUpdateRegistrationRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixUpdateRegistrationRow row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this WixUpdateRegistrationRow row belongs to and should get its column definitions from. + public WixUpdateRegistrationRow(SourceLineNumber sourceLineNumbers, Table table) : + base(sourceLineNumbers, table) + { + } + + public string Manufacturer + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + public string Department + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + public string ProductFamily + { + get { return (string)this.Fields[2].Data; } + set { this.Fields[2].Data = value; } + } + + public string Name + { + get { return (string)this.Fields[3].Data; } + set { this.Fields[3].Data = value; } + } + + public string Classification + { + get { return (string)this.Fields[4].Data; } + set { this.Fields[4].Data = value; } + } + } +} diff --git a/src/WixToolset.Data/Rows/WixVariableRow.cs b/src/WixToolset.Data/Rows/WixVariableRow.cs new file mode 100644 index 00000000..241c9257 --- /dev/null +++ b/src/WixToolset.Data/Rows/WixVariableRow.cs @@ -0,0 +1,81 @@ +// 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.Data.Rows +{ + using System; + using System.Globalization; + + /// + /// Specialization of a row for the WixVariable table. + /// + public sealed class WixVariableRow : Row + { + /// + /// Creates a WixVariable row that does not belong to a table. + /// + /// Original source lines for this row. + /// TableDefinition this WixVariable row belongs to and should get its column definitions from. + public WixVariableRow(SourceLineNumber sourceLineNumbers, TableDefinition tableDef) : + base(sourceLineNumbers, tableDef) + { + } + + /// + /// Creates a WixVariable row that belongs to a table. + /// + /// Original source lines for this row. + /// Table this WixVariable row belongs to and should get its column definitions from. + public WixVariableRow(SourceLineNumber sourceLineNumbers, Table table) : base(sourceLineNumbers, table) + { + } + + /// + /// Gets or sets the variable identifier. + /// + /// The variable identifier. + public string Id + { + get { return (string)this.Fields[0].Data; } + set { this.Fields[0].Data = value; } + } + + /// + /// Gets or sets the variable's value. + /// + /// The variable's value. + public string Value + { + get { return (string)this.Fields[1].Data; } + set { this.Fields[1].Data = value; } + } + + /// + /// Gets or sets whether this variable is overridable. + /// + /// Whether this variable is overridable. + public bool Overridable + { + get + { + return (0x1 == (Convert.ToInt32(this.Fields[2].Data, CultureInfo.InvariantCulture) & 0x1)); + } + + set + { + if (null == this.Fields[2].Data) + { + this.Fields[2].Data = 0; + } + + if (value) + { + this.Fields[2].Data = (int)this.Fields[2].Data | 0x1; + } + else + { + this.Fields[2].Data = (int)this.Fields[2].Data & ~0x1; + } + } + } + } +} diff --git a/src/WixToolset.Data/Section.cs b/src/WixToolset.Data/Section.cs new file mode 100644 index 00000000..6757aa77 --- /dev/null +++ b/src/WixToolset.Data/Section.cs @@ -0,0 +1,240 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + using System.Diagnostics; + using System.Globalization; + using System.Linq; + using System.Xml; + + /// + /// Section in an object file. + /// + public sealed class Section + { + /// + /// Creates a new section as part of an intermediate. + /// + /// Identifier for section. + /// Type of section. + /// Codepage for resulting database. + public Section(string id, SectionType type, int codepage) + { + this.Id = id; + this.Type = type; + this.Codepage = codepage; + + this.Tables = new TableIndexedCollection(); + } + + /// + /// Gets the identifier for the section. + /// + /// Section identifier. + public string Id { get; private set; } + + /// + /// Gets the type of the section. + /// + /// Type of section. + public SectionType Type { get; private set; } + + /// + /// Gets the codepage for the section. + /// + /// Codepage for the section. + public int Codepage { get; private set; } + + /// + /// Gets the tables in the section. + /// + /// Tables in section. + public TableIndexedCollection Tables { get; private set; } + + /// + /// Gets the source line information of the file containing this section. + /// + /// The source line information of the file containing this section. + public SourceLineNumber SourceLineNumbers { get; private set; } + + /// + /// Gets the identity of the intermediate the section is contained within. + /// + public string IntermediateId { get; internal set; } + + /// + /// Gets the identity of the library when the section is contained within one. + /// + public string LibraryId { get; internal set; } + + /// + /// Ensures a table is added to the section's table collection. + /// + /// Table definition for the table. + /// Table in the section. + public Table EnsureTable(TableDefinition tableDefinition) + { + Table table; + if (!this.Tables.TryGetTable(tableDefinition.Name, out table)) + { + table = new Table(this, tableDefinition); + this.Tables.Add(table); + } + + return table; + } + + /// + /// Parse a section from the xml. + /// + /// XmlReader where the intermediate is persisted. + /// TableDefinitions to use in the intermediate. + /// The parsed Section. + internal static Section Read(XmlReader reader, TableDefinitionCollection tableDefinitions) + { + Debug.Assert("section" == reader.LocalName); + + int codepage = 0; + bool empty = reader.IsEmptyElement; + string id = null; + SectionType type = SectionType.Unknown; + + while (reader.MoveToNextAttribute()) + { + switch (reader.Name) + { + case "codepage": + codepage = Convert.ToInt32(reader.Value, CultureInfo.InvariantCulture); + break; + case "id": + id = reader.Value; + break; + case "type": + switch (reader.Value) + { + case "bundle": + type = SectionType.Bundle; + break; + case "fragment": + type = SectionType.Fragment; + break; + case "module": + type = SectionType.Module; + break; + case "patchCreation": + type = SectionType.PatchCreation; + break; + case "product": + type = SectionType.Product; + break; + case "patch": + type = SectionType.Patch; + break; + default: + throw new XmlException(); + } + break; + } + } + + if (null == id && (SectionType.Unknown != type && SectionType.Fragment != type)) + { + throw new XmlException(); + } + + if (SectionType.Unknown == type) + { + throw new XmlException(); + } + + Section section = new Section(id, type, codepage); + section.SourceLineNumbers = SourceLineNumber.CreateFromUri(reader.BaseURI); + + List
tables = new List
(); + if (!empty) + { + bool done = false; + + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + switch (reader.LocalName) + { + case "table": + tables.Add(Table.Read(reader, section, tableDefinitions)); + break; + default: + throw new XmlException(); + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + + section.Tables = new TableIndexedCollection(tables); + + return section; + } + + /// + /// Persist the Section to an XmlWriter. + /// + /// XmlWriter which reference will be persisted to. + internal void Write(XmlWriter writer) + { + writer.WriteStartElement("section", Intermediate.XmlNamespaceUri); + + if (null != this.Id) + { + writer.WriteAttributeString("id", this.Id); + } + + switch (this.Type) + { + case SectionType.Bundle: + writer.WriteAttributeString("type", "bundle"); + break; + case SectionType.Fragment: + writer.WriteAttributeString("type", "fragment"); + break; + case SectionType.Module: + writer.WriteAttributeString("type", "module"); + break; + case SectionType.Product: + writer.WriteAttributeString("type", "product"); + break; + case SectionType.PatchCreation: + writer.WriteAttributeString("type", "patchCreation"); + break; + case SectionType.Patch: + writer.WriteAttributeString("type", "patch"); + break; + } + + if (0 != this.Codepage) + { + writer.WriteAttributeString("codepage", this.Codepage.ToString()); + } + + // save the rows in table order + foreach (Table table in this.Tables.OrderBy(t => t.Name)) + { + table.Write(writer); + } + + writer.WriteEndElement(); + } + } +} diff --git a/src/WixToolset.Data/SectionType.cs b/src/WixToolset.Data/SectionType.cs new file mode 100644 index 00000000..3322ba1a --- /dev/null +++ b/src/WixToolset.Data/SectionType.cs @@ -0,0 +1,31 @@ +// 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.Data +{ + /// + /// Type of section. + /// + public enum SectionType + { + /// Unknown section type, default and invalid. + Unknown, + + /// Bundle section type. + Bundle, + + /// Fragment section type. + Fragment, + + /// Module section type. + Module, + + /// Product section type. + Product, + + /// Patch creation section type. + PatchCreation, + + /// Patch section type. + Patch + } +} diff --git a/src/WixToolset.Data/Serialize/CodeDomInterfaces.cs b/src/WixToolset.Data/Serialize/CodeDomInterfaces.cs new file mode 100644 index 00000000..36a05202 --- /dev/null +++ b/src/WixToolset.Data/Serialize/CodeDomInterfaces.cs @@ -0,0 +1,96 @@ +// 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.Data.Serialize +{ + using System; + using System.Collections; + using System.Xml; + + /// + /// Interface for generated schema elements. + /// + public interface ISchemaElement + { + /// + /// Gets and sets the parent of this element. May be null. + /// + /// An ISchemaElement that has this element as a child. + ISchemaElement ParentElement + { + get; + set; + } + + /// + /// Outputs xml representing this element, including the associated attributes + /// and any nested elements. + /// + /// XmlWriter to be used when outputting the element. + void OutputXml(XmlWriter writer); + } + + /// + /// Interface for generated schema elements. Implemented by elements that have child + /// elements. + /// + public interface IParentElement + { + /// + /// Gets an enumerable collection of the children of this element. + /// + /// An enumerable collection of the children of this element. + IEnumerable Children + { + get; + } + + /// + /// Gets an enumerable collection of the children of this element, filtered + /// by the passed in type. + /// + /// The type of children to retrieve. + IEnumerable this[Type childType] + { + get; + } + + /// + /// Adds a child to this element. + /// + /// Child to add. + void AddChild(ISchemaElement child); + + /// + /// Removes a child from this element. + /// + /// Child to remove. + void RemoveChild(ISchemaElement child); + } + + /// + /// Interface for generated schema elements. Implemented by classes with attributes. + /// + public interface ISetAttributes + { + /// + /// Sets the attribute with the given name to the given value. The value here is + /// a string, and is converted to the strongly-typed version inside this method. + /// + /// The name of the attribute to set. + /// The value to assign to the attribute. + void SetAttribute(string name, string value); + } + + /// + /// Interface for generated schema elements. Implemented by classes with children. + /// + public interface ICreateChildren + { + /// + /// Creates an instance of the child with the passed in name. + /// + /// String matching the element name of the child when represented in XML. + /// An instance of that child. + ISchemaElement CreateChild(string childName); + } +} diff --git a/src/WixToolset.Data/Serialize/CodeDomReader.cs b/src/WixToolset.Data/Serialize/CodeDomReader.cs new file mode 100644 index 00000000..76013936 --- /dev/null +++ b/src/WixToolset.Data/Serialize/CodeDomReader.cs @@ -0,0 +1,161 @@ +// 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.Data.Serialize +{ + using System; + using System.Diagnostics.CodeAnalysis; + using System.Globalization; + using System.Reflection; + using System.Xml; + + /// + /// Class used for reading XML files in to the CodeDom. + /// + public class CodeDomReader + { + private Assembly[] assemblies; + + /// + /// Creates a new CodeDomReader, using the current assembly. + /// + public CodeDomReader() + { + this.assemblies = new Assembly[] { Assembly.GetExecutingAssembly() }; + } + + /// + /// Creates a new CodeDomReader, and takes in a list of assemblies in which to + /// look for elements. + /// + /// Assemblies in which to look for types that correspond + /// to elements. + public CodeDomReader(Assembly[] assemblies) + { + this.assemblies = assemblies; + } + + /// + /// Loads an XML file into a strongly-typed code dom. + /// + /// File to load into the code dom. + /// The strongly-typed object at the root of the tree. + [SuppressMessage("Microsoft.Globalization", "CA1303:DoNotPassLiteralsAsLocalizedParameters", MessageId = "System.InvalidOperationException.#ctor(System.String)")] + public ISchemaElement Load(string filePath) + { + XmlDocument document = new XmlDocument(); + document.Load(filePath); + ISchemaElement schemaElement = null; + + foreach (XmlNode node in document.ChildNodes) + { + XmlElement element = node as XmlElement; + if (element != null) + { + if (schemaElement != null) + { + throw new InvalidOperationException(WixDataStrings.EXP_MultipleRootElementsFoundInFile); + } + + schemaElement = this.CreateObjectFromElement(element); + this.ParseObjectFromElement(schemaElement, element); + } + } + return schemaElement; + } + + /// + /// Sets an attribute on an ISchemaElement. + /// + /// Schema element to set attribute on. + /// Name of the attribute to set. + /// Value to set on the attribute. + [SuppressMessage("Microsoft.Globalization", "CA1303:DoNotPassLiteralsAsLocalizedParameters", MessageId = "System.InvalidOperationException.#ctor(System.String)")] + private static void SetAttributeOnObject(ISchemaElement schemaElement, string name, string value) + { + ISetAttributes setAttributes = schemaElement as ISetAttributes; + if (setAttributes == null) + { + throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, WixDataStrings.EXP_ISchemaElementDoesnotImplementISetAttribute, schemaElement.GetType().FullName)); + } + else + { + setAttributes.SetAttribute(name, value); + } + } + + /// + /// Parses an ISchemaElement from the XmlElement. + /// + /// ISchemaElement to fill in. + /// XmlElement to parse from. + [SuppressMessage("Microsoft.Globalization", "CA1303:DoNotPassLiteralsAsLocalizedParameters", MessageId = "System.InvalidOperationException.#ctor(System.String)")] + private void ParseObjectFromElement(ISchemaElement schemaElement, XmlElement element) + { + foreach (XmlAttribute attribute in element.Attributes) + { + SetAttributeOnObject(schemaElement, attribute.LocalName, attribute.Value); + } + + foreach (XmlNode node in element.ChildNodes) + { + XmlElement childElement = node as XmlElement; + if (childElement != null) + { + ISchemaElement childSchemaElement = null; + ICreateChildren createChildren = schemaElement as ICreateChildren; + if (createChildren == null) + { + throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, WixDataStrings.EXP_ISchemaElementDoesnotImplementICreateChildren, element.LocalName)); + } + else + { + childSchemaElement = createChildren.CreateChild(childElement.LocalName); + } + + if (childSchemaElement == null) + { + childSchemaElement = this.CreateObjectFromElement(childElement); + if (childSchemaElement == null) + { + throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, WixDataStrings.EXP_XmlElementDoesnotHaveISchemaElement, childElement.LocalName)); + } + } + + this.ParseObjectFromElement(childSchemaElement, childElement); + IParentElement parentElement = (IParentElement)schemaElement; + parentElement.AddChild(childSchemaElement); + } + else + { + XmlText childText = node as XmlText; + if (childText != null) + { + SetAttributeOnObject(schemaElement, "Content", childText.Value); + } + } + } + } + + /// + /// Creates an object from an XML element by digging through the assembly list. + /// + /// XML Element to create an ISchemaElement from. + /// A constructed ISchemaElement. + private ISchemaElement CreateObjectFromElement(XmlElement element) + { + ISchemaElement schemaElement = null; + foreach (Assembly assembly in this.assemblies) + { + foreach (Type type in assembly.GetTypes()) + { + if (type.FullName.EndsWith(element.LocalName, StringComparison.Ordinal) + && typeof(ISchemaElement).IsAssignableFrom(type)) + { + schemaElement = (ISchemaElement)Activator.CreateInstance(type); + } + } + } + return schemaElement; + } + } +} diff --git a/src/WixToolset.Data/Serialize/ElementCollection.cs b/src/WixToolset.Data/Serialize/ElementCollection.cs new file mode 100644 index 00000000..1d64c5eb --- /dev/null +++ b/src/WixToolset.Data/Serialize/ElementCollection.cs @@ -0,0 +1,617 @@ +// 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.Data.Serialize +{ + using System; + using System.Collections; + using System.Globalization; + + /// + /// Collection used in the CodeDOM for the children of a given element. Provides type-checking + /// on the allowed children to ensure that only allowed types are added. + /// + public class ElementCollection : ICollection, IEnumerable + { + private CollectionType collectionType; + private int totalContainedItems; + private int containersUsed; + private ArrayList items; + + /// + /// Creates a new element collection. + /// + /// Type of the collection to create. + public ElementCollection(CollectionType collectionType) + { + this.collectionType = collectionType; + this.items = new ArrayList(); + } + + /// + /// Enum representing types of XML collections. + /// + public enum CollectionType + { + /// + /// A choice type, corresponding to the XSD choice element. + /// + Choice, + + /// + /// A sequence type, corresponding to the XSD sequence element. + /// + Sequence + } + + /// + /// Gets the type of collection. + /// + /// The type of collection. + public CollectionType Type + { + get { return this.collectionType; } + } + + /// + /// Gets the count of child elements in this collection (counts ISchemaElements, not nested collections). + /// + /// The count of child elements in this collection (counts ISchemaElements, not nested collections). + public int Count + { + get { return this.totalContainedItems; } + } + + /// + /// Gets the flag specifying whether this collection is synchronized. Always returns false. + /// + /// The flag specifying whether this collection is synchronized. Always returns false. + public bool IsSynchronized + { + get { return false; } + } + + /// + /// Gets an object external callers can synchronize on. + /// + /// An object external callers can synchronize on. + public object SyncRoot + { + get { return this; } + } + + /// + /// Adds a child element to this collection. + /// + /// The element to add. + /// Thrown if the child is not of an allowed type. + public void AddElement(ISchemaElement element) + { + foreach (object obj in this.items) + { + bool containerUsed; + + CollectionItem collectionItem = obj as CollectionItem; + if (collectionItem != null) + { + containerUsed = collectionItem.Elements.Count != 0; + if (collectionItem.ElementType.IsAssignableFrom(element.GetType())) + { + collectionItem.AddElement(element); + + if (!containerUsed) + { + this.containersUsed++; + } + + this.totalContainedItems++; + return; + } + + continue; + } + + ElementCollection collection = obj as ElementCollection; + if (collection != null) + { + containerUsed = collection.Count != 0; + + try + { + collection.AddElement(element); + + if (!containerUsed) + { + this.containersUsed++; + } + + this.totalContainedItems++; + return; + } + catch (ArgumentException) + { + // Eat the exception and keep looking. We'll throw our own if we can't find its home. + } + + continue; + } + } + + throw new ArgumentException(String.Format( + CultureInfo.InvariantCulture, + WixDataStrings.EXP_ElementOfTypeIsNotValidForThisCollection, + element.GetType().Name)); + } + + /// + /// Removes a child element from this collection. + /// + /// The element to remove. + /// Thrown if the element is not of an allowed type. + public void RemoveElement(ISchemaElement element) + { + foreach (object obj in this.items) + { + CollectionItem collectionItem = obj as CollectionItem; + if (collectionItem != null) + { + if (collectionItem.ElementType.IsAssignableFrom(element.GetType())) + { + if (collectionItem.Elements.Count == 0) + { + return; + } + + collectionItem.RemoveElement(element); + + if (collectionItem.Elements.Count == 0) + { + this.containersUsed--; + } + + this.totalContainedItems--; + return; + } + + continue; + } + + ElementCollection collection = obj as ElementCollection; + if (collection != null) + { + if (collection.Count == 0) + { + continue; + } + + try + { + collection.RemoveElement(element); + + if (collection.Count == 0) + { + this.containersUsed--; + } + + this.totalContainedItems--; + return; + } + catch (ArgumentException) + { + // Eat the exception and keep looking. We'll throw our own if we can't find its home. + } + + continue; + } + } + + throw new ArgumentException(String.Format( + CultureInfo.InvariantCulture, + WixDataStrings.EXP_ElementOfTypeIsNotValidForThisCollection, + element.GetType().Name)); + } + + /// + /// Copies this collection to an array. + /// + /// Array to copy to. + /// Offset into the array. + public void CopyTo(Array array, int index) + { + int item = 0; + foreach (ISchemaElement element in this) + { + array.SetValue(element, (long)(item + index)); + item++; + } + } + + /// + /// Creates an enumerator for walking the elements in this collection. + /// + /// A newly created enumerator. + public IEnumerator GetEnumerator() + { + return new ElementCollectionEnumerator(this); + } + + /// + /// Gets an enumerable collection of children of a given type. + /// + /// Type of children to get. + /// A collection of children. + /// Thrown if the type isn't a valid child type. + public IEnumerable Filter(Type childType) + { + foreach (object container in this.items) + { + CollectionItem collectionItem = container as CollectionItem; + if (collectionItem != null) + { + if (collectionItem.ElementType.IsAssignableFrom(childType)) + { + return collectionItem.Elements; + } + + continue; + } + + ElementCollection elementCollection = container as ElementCollection; + if (elementCollection != null) + { + IEnumerable nestedFilter = elementCollection.Filter(childType); + if (nestedFilter != null) + { + return nestedFilter; + } + + continue; + } + } + + throw new ArgumentException(String.Format( + CultureInfo.InvariantCulture, + WixDataStrings.EXP_TypeIsNotValidForThisCollection, + childType.Name)); + } + + /// + /// Adds a type to this collection. + /// + /// CollectionItem representing the type to add. + public void AddItem(CollectionItem collectionItem) + { + this.items.Add(collectionItem); + } + + /// + /// Adds a nested collection to this collection. + /// + /// ElementCollection to add. + public void AddCollection(ElementCollection collection) + { + this.items.Add(collection); + } + + /// + /// Class used to represent a given type in the child collection of an element. Abstract, + /// has subclasses for choice and sequence (which can do cardinality checks). + /// + public abstract class CollectionItem + { + private Type elementType; + private ArrayList elements; + + /// + /// Creates a new CollectionItem for the given element type. + /// + /// Type of the element for this collection item. + protected CollectionItem(Type elementType) + { + this.elementType = elementType; + this.elements = new ArrayList(); + } + + /// + /// Gets the type of this collection's items. + /// + /// The type of this collection's items. + public Type ElementType + { + get { return this.elementType; } + } + + /// + /// Gets the elements of this collection. + /// + /// The elements of this collection. + public ArrayList Elements + { + get { return this.elements; } + } + + /// + /// Adds an element to this collection. Must be of an assignable type to the collection's + /// type. + /// + /// The element to add. + /// Thrown if the type isn't assignable to the collection's type. + public void AddElement(ISchemaElement element) + { + if (!this.elementType.IsAssignableFrom(element.GetType())) + { + throw new ArgumentException( + String.Format( + CultureInfo.InvariantCulture, + WixDataStrings.EXP_ElementIsSubclassOfDifferentType, + this.elementType.Name, + element.GetType().Name), + "element"); + } + + this.elements.Add(element); + } + + /// + /// Removes an element from this collection. + /// + /// The element to remove. + /// Thrown if the element's type isn't assignable to the collection's type. + public void RemoveElement(ISchemaElement element) + { + if (!this.elementType.IsAssignableFrom(element.GetType())) + { + throw new ArgumentException( + String.Format( + CultureInfo.InvariantCulture, + WixDataStrings.EXP_ElementIsSubclassOfDifferentType, + this.elementType.Name, + element.GetType().Name), + "element"); + } + + this.elements.Remove(element); + } + } + + /// + /// Class representing a choice item. Doesn't do cardinality checks. + /// + public class ChoiceItem : CollectionItem + { + /// + /// Creates a new choice item. + /// + /// Type of the created item. + public ChoiceItem(Type elementType) + : base(elementType) + { + } + } + + /// + /// Class representing a sequence item. Can do cardinality checks, if required. + /// + public class SequenceItem : CollectionItem + { + /// + /// Creates a new sequence item. + /// + /// Type of the created item. + public SequenceItem(Type elementType) + : base(elementType) + { + } + } + + /// + /// Enumerator for the ElementCollection. + /// + private class ElementCollectionEnumerator : IEnumerator + { + private ElementCollection collection; + private Stack collectionStack; + + /// + /// Creates a new ElementCollectionEnumerator. + /// + /// The collection to create an enumerator for. + public ElementCollectionEnumerator(ElementCollection collection) + { + this.collection = collection; + } + + /// + /// Gets the current object from the enumerator. + /// + public object Current + { + get + { + if (this.collectionStack != null && this.collectionStack.Count > 0) + { + CollectionTuple tuple = (CollectionTuple)this.collectionStack.Peek(); + object container = tuple.Collection.items[tuple.ContainerIndex]; + + CollectionItem collectionItem = container as CollectionItem; + if (collectionItem != null) + { + return collectionItem.Elements[tuple.ItemIndex]; + } + + throw new InvalidOperationException(String.Format( + CultureInfo.InvariantCulture, + WixDataStrings.EXP_ElementMustBeChoiceItemOrSequenceItem, + container.GetType().Name)); + } + + return null; + } + } + + /// + /// Resets the enumerator to the beginning. + /// + public void Reset() + { + if (this.collectionStack != null) + { + this.collectionStack.Clear(); + this.collectionStack = null; + } + } + + /// + /// Moves the enumerator to the next item. + /// + /// True if there is a next item, false otherwise. + public bool MoveNext() + { + if (this.collectionStack == null) + { + if (this.collection.Count == 0) + { + return false; + } + + this.collectionStack = new Stack(); + this.collectionStack.Push(new CollectionTuple(this.collection)); + } + + CollectionTuple tuple = (CollectionTuple)this.collectionStack.Peek(); + + if (this.FindNext(tuple)) + { + return true; + } + + this.collectionStack.Pop(); + if (this.collectionStack.Count == 0) + { + return false; + } + + return this.MoveNext(); + } + + /// + /// Pushes a collection onto the stack. + /// + /// The collection to push. + private void PushCollection(ElementCollection elementCollection) + { + if (elementCollection.Count <= 0) + { + throw new ArgumentException(String.Format( + CultureInfo.InvariantCulture, + WixDataStrings.EXP_CollectionMustHaveAtLeastOneElement, + elementCollection.Count)); + } + + CollectionTuple tuple = new CollectionTuple(elementCollection); + this.collectionStack.Push(tuple); + this.FindNext(tuple); + } + + /// + /// Finds the next item from a given tuple. + /// + /// The tuple to start looking from. + /// True if a next element is found, false otherwise. + private bool FindNext(CollectionTuple tuple) + { + object container = tuple.Collection.items[tuple.ContainerIndex]; + + CollectionItem collectionItem = container as CollectionItem; + if (collectionItem != null) + { + if (tuple.ItemIndex + 1 < collectionItem.Elements.Count) + { + tuple.ItemIndex++; + return true; + } + } + + ElementCollection elementCollection = container as ElementCollection; + if (elementCollection != null && elementCollection.Count > 0 && tuple.ItemIndex == -1) + { + tuple.ItemIndex++; + this.PushCollection(elementCollection); + return true; + } + + tuple.ItemIndex = 0; + + for (int i = tuple.ContainerIndex + 1; i < tuple.Collection.items.Count; ++i) + { + object nestedContainer = tuple.Collection.items[i]; + + CollectionItem nestedCollectionItem = nestedContainer as CollectionItem; + if (nestedCollectionItem != null) + { + if (nestedCollectionItem.Elements.Count > 0) + { + tuple.ContainerIndex = i; + return true; + } + } + + ElementCollection nestedElementCollection = nestedContainer as ElementCollection; + if (nestedElementCollection != null && nestedElementCollection.Count > 0) + { + tuple.ContainerIndex = i; + this.PushCollection(nestedElementCollection); + return true; + } + } + + return false; + } + + /// + /// Class representing a single point in the collection. Consists of an ElementCollection, + /// a container index, and an index into the container. + /// + private class CollectionTuple + { + private ElementCollection collection; + private int containerIndex; + private int itemIndex = -1; + + /// + /// Creates a new CollectionTuple. + /// + /// The collection for the tuple. + public CollectionTuple(ElementCollection collection) + { + this.collection = collection; + } + + /// + /// Gets the collection for the tuple. + /// + public ElementCollection Collection + { + get { return this.collection; } + } + + /// + /// Gets and sets the index of the container in the collection. + /// + public int ContainerIndex + { + get { return this.containerIndex; } + set { this.containerIndex = value; } + } + + /// + /// Gets and sets the index of the item in the container. + /// + public int ItemIndex + { + get { return this.itemIndex; } + set { this.itemIndex = value; } + } + } + } + } +} diff --git a/src/WixToolset.Data/SourceLineNumber.cs b/src/WixToolset.Data/SourceLineNumber.cs new file mode 100644 index 00000000..87a36132 --- /dev/null +++ b/src/WixToolset.Data/SourceLineNumber.cs @@ -0,0 +1,205 @@ +// 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.Data +{ + using System; + using System.IO; + using System.Text; + using System.Xml; + using System.Xml.Linq; + + /// + /// Represents file name and line number for source file + /// + public sealed class SourceLineNumber + { + /// + /// Constructor for a source with no line information. + /// + /// File name of the source. + public SourceLineNumber(string fileName) + { + this.FileName = fileName; + } + + /// + /// Constructor for a source with line information. + /// + /// File name of the source. + /// Line number of the source. + public SourceLineNumber(string fileName, int lineNumber) + { + this.FileName = fileName; + this.LineNumber = lineNumber; + } + + /// + /// Gets the file name of the source. + /// + /// File name for the source. + public string FileName { get; private set; } + + /// + /// Gets or sets the line number of the source. + /// + /// Line number of the source. + public int? LineNumber { get; set; } + + /// + /// Gets or sets the parent source line number that included this source line number. + /// + public SourceLineNumber Parent { get; set; } + + /// + /// Gets the file name and line information. + /// + /// File name and line information. + public string QualifiedFileName + { + get + { + return this.LineNumber.HasValue ? String.Concat(this.FileName, "*", this.LineNumber) : this.FileName; + } + } + + /// + /// Creates a source line number from an encoded string. + /// + /// Encoded string to parse. + public static SourceLineNumber CreateFromEncoded(string encodedSourceLineNumbers) + { + string[] linesSplit = encodedSourceLineNumbers.Split('|'); + + SourceLineNumber first = null; + SourceLineNumber parent = null; + for (int i = 0; i < linesSplit.Length; ++i) + { + string[] filenameSplit = linesSplit[i].Split('*'); + SourceLineNumber source; + + if (2 == filenameSplit.Length) + { + source = new SourceLineNumber(filenameSplit[0], Convert.ToInt32(filenameSplit[1])); + } + else + { + source = new SourceLineNumber(filenameSplit[0]); + } + + if (null != parent) + { + parent.Parent = source; + } + + parent = source; + if (null == first) + { + first = parent; + } + } + + return first; + } + + /// + /// Creates a source line number from a URI. + /// + /// Uri to convert into source line number + public static SourceLineNumber CreateFromUri(string uri) + { + if (String.IsNullOrEmpty(uri)) + { + return null; + } + + // make the local path look like a normal local path + string localPath = new Uri(uri).LocalPath; + localPath = localPath.TrimStart(Path.AltDirectorySeparatorChar).Replace(Path.AltDirectorySeparatorChar, Path.DirectorySeparatorChar); + + return new SourceLineNumber(localPath); + } + + /// + /// Creates a source line number from an XObject. + /// + /// XML node to create source line number from. + /// Optional line number offset into XML file not already included in the line information. + public static SourceLineNumber CreateFromXObject(XObject node, int offset = 0) + { + string uri = node.BaseUri; + IXmlLineInfo lineInfo = node as IXmlLineInfo; + + SourceLineNumber result = CreateFromUri(uri); + if (null != result && null != lineInfo) + { + result.LineNumber = lineInfo.LineNumber + offset; + } + + return result; + } + + /// + /// Get the source line information for the current element. Typically this information + /// is set by the precompiler for each element that it encounters. + /// + /// Element to get source line information for. + /// + /// The source line number used to author the element being processed or + /// null if the preprocessor did not process the element or the node is + /// not an element. + /// + public static SourceLineNumber GetFromXAnnotation(XObject node) + { + return node.Annotation(); + } + + /// + /// Returns the SourceLineNumber and parents encoded as a string. + /// + public string GetEncoded() + { + StringBuilder sb = new StringBuilder(this.QualifiedFileName); + + for (SourceLineNumber source = this.Parent; null != source; source = source.Parent) + { + sb.Append("|"); + sb.Append(source.QualifiedFileName); + } + + return sb.ToString(); + } + + /// + /// Determines if two SourceLineNumbers are equivalent. + /// + /// Object to compare. + /// True if SourceLineNumbers are equivalent. + public override bool Equals(object obj) + { + SourceLineNumber other = obj as SourceLineNumber; + return null != other && + this.LineNumber.HasValue == other.LineNumber.HasValue && + (!this.LineNumber.HasValue || this.LineNumber == other.LineNumber) && + this.FileName.Equals(other.FileName, StringComparison.OrdinalIgnoreCase) && + (null == this.Parent && null == other.Parent || this.Parent.Equals(other.Parent)); + } + + /// + /// Serves as a hash code for a particular type. + /// + /// The hash code. + public override int GetHashCode() + { + return this.GetEncoded().GetHashCode(); + } + + /// + /// Shows a string representation of a source line number. + /// + /// String representation of a source line number. + public override string ToString() + { + return this.LineNumber.HasValue && !String.IsNullOrEmpty(this.FileName) ? String.Concat(this.FileName, "(", this.LineNumber, ")") : this.FileName ?? String.Empty; + } + } +} diff --git a/src/WixToolset.Data/SubStorage.cs b/src/WixToolset.Data/SubStorage.cs new file mode 100644 index 00000000..e136bfe9 --- /dev/null +++ b/src/WixToolset.Data/SubStorage.cs @@ -0,0 +1,109 @@ +// 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.Data +{ + using System.Xml; + + /// + /// Substorage inside an output. + /// + public sealed class SubStorage + { + /// + /// Instantiate a new substorage. + /// + /// The substorage name. + /// The substorage data. + public SubStorage(string name, Output data) + { + this.Name = name; + this.Data = data; + } + + /// + /// Gets the substorage name. + /// + /// The substorage name. + public string Name { get; private set; } + + /// + /// Gets the substorage data. + /// + /// The substorage data. + public Output Data { get; private set; } + + /// + /// Creates a SubStorage from the XmlReader. + /// + /// Reader to get data from. + /// New SubStorage object. + internal static SubStorage Read(XmlReader reader) + { + if (!reader.LocalName.Equals("subStorage" == reader.LocalName)) + { + throw new XmlException(); + } + + Output data = null; + bool empty = reader.IsEmptyElement; + string name = null; + + while (reader.MoveToNextAttribute()) + { + switch (reader.LocalName) + { + case "name": + name = reader.Value; + break; + } + } + + if (!empty) + { + bool done = false; + + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + switch (reader.LocalName) + { + case "wixOutput": + data = Output.Read(reader, true); + break; + default: + throw new XmlException(); + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + + return new SubStorage(name, data); + } + + /// + /// Persists a SubStorage in an XML format. + /// + /// XmlWriter where the SubStorage should persist itself as XML. + internal void Write(XmlWriter writer) + { + writer.WriteStartElement("subStorage", Output.XmlNamespaceUri); + + writer.WriteAttributeString("name", this.Name); + + this.Data.Write(writer); + + writer.WriteEndElement(); + } + } +} diff --git a/src/WixToolset.Data/Symbol.cs b/src/WixToolset.Data/Symbol.cs new file mode 100644 index 00000000..a96f9382 --- /dev/null +++ b/src/WixToolset.Data/Symbol.cs @@ -0,0 +1,89 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + using System.Linq; + + /// + /// Symbol representing a single row in a database. + /// + public sealed class Symbol + { + private HashSet possibleConflictSymbols; + private HashSet redundantSymbols; + + /// + /// Creates a symbol for a row. + /// + /// Row for the symbol + public Symbol(Row row) + { + this.Row = row; + this.Name = String.Concat(this.Row.TableDefinition.Name, ":", this.Row.GetPrimaryKey()); + } + + /// + /// Gets the accessibility of the symbol which is a direct reflection of the accessibility of the row's accessibility. + /// + /// Accessbility of the symbol. + public AccessModifier Access { get { return this.Row.Access; } } + + /// + /// Gets the name of the symbol. + /// + /// Name of the symbol. + public string Name { get; private set; } + + /// + /// Gets the row for this symbol. + /// + /// Row for this symbol. + public Row Row { get; private set; } + + /// + /// Gets the section for the symbol. + /// + /// Section for the symbol. + public Section Section { get { return this.Row.Section; } } + + /// + /// Gets any duplicates of this symbol that are possible conflicts. + /// + public IEnumerable PossiblyConflictingSymbols { get { return this.possibleConflictSymbols ?? Enumerable.Empty(); } } + + /// + /// Gets any duplicates of this symbol that are redundant. + /// + public IEnumerable RedundantSymbols { get { return this.redundantSymbols ?? Enumerable.Empty(); } } + + /// + /// Adds a duplicate symbol that is a possible conflict. + /// + /// Symbol that is a possible conflict of this symbol. + public void AddPossibleConflict(Symbol symbol) + { + if (null == this.possibleConflictSymbols) + { + this.possibleConflictSymbols = new HashSet(); + } + + this.possibleConflictSymbols.Add(symbol); + } + + /// + /// Adds a duplicate symbol that is redundant. + /// + /// Symbol that is redundant of this symbol. + public void AddRedundant(Symbol symbol) + { + if (null == this.redundantSymbols) + { + this.redundantSymbols = new HashSet(); + } + + this.redundantSymbols.Add(symbol); + } + } +} diff --git a/src/WixToolset.Data/Table.cs b/src/WixToolset.Data/Table.cs new file mode 100644 index 00000000..19a3a67d --- /dev/null +++ b/src/WixToolset.Data/Table.cs @@ -0,0 +1,446 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + using System.Diagnostics; + using System.Diagnostics.CodeAnalysis; + using System.Globalization; + using System.IO; + using System.Text; + using System.Xml; + using WixToolset.Data.Rows; + + /// + /// Object that represents a table in a database. + /// + public sealed class Table + { + /// + /// Creates a table in a section. + /// + /// Section to add table to. + /// Definition of the table. + public Table(Section section, TableDefinition tableDefinition) + { + this.Section = section; + this.Definition = tableDefinition; + this.Rows = new List(); + } + + /// + /// Gets the section for the table. + /// + /// Section for the table. + public Section Section { get; private set; } + + /// + /// Gets the table definition. + /// + /// Definition of the table. + public TableDefinition Definition { get; private set; } + + /// + /// Gets the name of the table. + /// + /// Name of the table. + public string Name + { + get { return this.Definition.Name; } + } + + /// + /// Gets or sets the table transform operation. + /// + /// The table transform operation. + public TableOperation Operation { get; set; } + + /// + /// Gets the rows contained in the table. + /// + /// Rows contained in the table. + public IList Rows { get; private set; } + + /// + /// Creates a new row in the table. + /// + /// Original source lines for this row. + /// Specifies whether to only create the row or add it to the table automatically. + /// Row created in table. + public Row CreateRow(SourceLineNumber sourceLineNumbers, bool add = true) + { + Row row; + + switch (this.Name) + { + case "BBControl": + row = new BBControlRow(sourceLineNumbers, this); + break; + case "WixBundlePackage": + row = new WixBundlePackageRow(sourceLineNumbers, this); + break; + case "WixBundleExePackage": + row = new WixBundleExePackageRow(sourceLineNumbers, this); + break; + case "WixBundleMsiPackage": + row = new WixBundleMsiPackageRow(sourceLineNumbers, this); + break; + case "WixBundleMspPackage": + row = new WixBundleMspPackageRow(sourceLineNumbers, this); + break; + case "WixBundleMsuPackage": + row = new WixBundleMsuPackageRow(sourceLineNumbers, this); + break; + case "Component": + row = new ComponentRow(sourceLineNumbers, this); + break; + case "WixBundleContainer": + row = new WixBundleContainerRow(sourceLineNumbers, this); + break; + case "Control": + row = new ControlRow(sourceLineNumbers, this); + break; + case "File": + row = new FileRow(sourceLineNumbers, this); + break; + case "WixBundleMsiFeature": + row = new WixBundleMsiFeatureRow(sourceLineNumbers, this); + break; + case "WixBundleMsiProperty": + row = new WixBundleMsiPropertyRow(sourceLineNumbers, this); + break; + case "Media": + row = new MediaRow(sourceLineNumbers, this); + break; + case "WixBundlePayload": + row = new WixBundlePayloadRow(sourceLineNumbers, this); + break; + case "Property": + row = new PropertyRow(sourceLineNumbers, this); + break; + case "WixRelatedBundle": + row = new WixRelatedBundleRow(sourceLineNumbers, this); + break; + case "WixBundleRelatedPackage": + row = new WixBundleRelatedPackageRow(sourceLineNumbers, this); + break; + case "WixBundleRollbackBoundary": + row = new WixBundleRollbackBoundaryRow(sourceLineNumbers, this); + break; + case "Upgrade": + row = new UpgradeRow(sourceLineNumbers, this); + break; + case "WixBundleVariable": + row = new WixBundleVariableRow(sourceLineNumbers, this); + break; + case "WixAction": + row = new WixActionRow(sourceLineNumbers, this); + break; + case "WixApprovedExeForElevation": + row = new WixApprovedExeForElevationRow(sourceLineNumbers, this); + break; + case "WixBundle": + row = new WixBundleRow(sourceLineNumbers, this); + break; + case "WixBundlePackageExitCode": + row = new WixBundlePackageExitCodeRow(sourceLineNumbers, this); + break; + case "WixBundlePatchTargetCode": + row = new WixBundlePatchTargetCodeRow(sourceLineNumbers, this); + break; + case "WixBundleSlipstreamMsp": + row = new WixBundleSlipstreamMspRow(sourceLineNumbers, this); + break; + case "WixBundleUpdate": + row = new WixBundleUpdateRow(sourceLineNumbers, this); + break; + case "WixBundleCatalog": + row = new WixBundleCatalogRow(sourceLineNumbers, this); + break; + case "WixChain": + row = new WixChainRow(sourceLineNumbers, this); + break; + case "WixChainItem": + row = new WixChainItemRow(sourceLineNumbers, this); + break; + case "WixBundlePackageCommandLine": + row = new WixBundlePackageCommandLineRow(sourceLineNumbers, this); + break; + case "WixComplexReference": + row = new WixComplexReferenceRow(sourceLineNumbers, this); + break; + case "WixDeltaPatchFile": + row = new WixDeltaPatchFileRow(sourceLineNumbers, this); + break; + case "WixDeltaPatchSymbolPaths": + row = new WixDeltaPatchSymbolPathsRow(sourceLineNumbers, this); + break; + case "WixFile": + row = new WixFileRow(sourceLineNumbers, this); + break; + case "WixGroup": + row = new WixGroupRow(sourceLineNumbers, this); + break; + case "WixMedia": + row = new WixMediaRow(sourceLineNumbers, this); + break; + case "WixMediaTemplate": + row = new WixMediaTemplateRow(sourceLineNumbers, this); + break; + case "WixMerge": + row = new WixMergeRow(sourceLineNumbers, this); + break; + case "WixPayloadProperties": + row = new WixPayloadPropertiesRow(sourceLineNumbers, this); + break; + case "WixProperty": + row = new WixPropertyRow(sourceLineNumbers, this); + break; + case "WixSimpleReference": + row = new WixSimpleReferenceRow(sourceLineNumbers, this); + break; + case "WixUpdateRegistration": + row = new WixUpdateRegistrationRow(sourceLineNumbers, this); + break; + case "WixVariable": + row = new WixVariableRow(sourceLineNumbers, this); + break; + + default: + row = new Row(sourceLineNumbers, this); + break; + } + + if (add) + { + this.Rows.Add(row); + } + + return row; + } + + /// + /// Parse a table from the xml. + /// + /// XmlReader where the intermediate is persisted. + /// Section to populate with persisted data. + /// TableDefinitions to use in the intermediate. + /// The parsed table. + internal static Table Read(XmlReader reader, Section section, TableDefinitionCollection tableDefinitions) + { + Debug.Assert("table" == reader.LocalName); + + bool empty = reader.IsEmptyElement; + TableOperation operation = TableOperation.None; + string name = null; + + while (reader.MoveToNextAttribute()) + { + switch (reader.LocalName) + { + case "name": + name = reader.Value; + break; + case "op": + switch (reader.Value) + { + case "add": + operation = TableOperation.Add; + break; + case "drop": + operation = TableOperation.Drop; + break; + default: + throw new XmlException(); + } + break; + } + } + + if (null == name) + { + throw new XmlException(); + } + + TableDefinition tableDefinition = tableDefinitions[name]; + Table table = new Table(section, tableDefinition); + table.Operation = operation; + + if (!empty) + { + bool done = false; + + // loop through all the rows in a table + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + switch (reader.LocalName) + { + case "row": + Row.Read(reader, table); + break; + default: + throw new XmlException(); + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + + return table; + } + + /// + /// Modularize the table. + /// + /// String containing the GUID of the Merge Module, if appropriate. + /// Optional collection of identifiers that should not be modularized. + public void Modularize(string modularizationGuid, ISet suppressModularizationIdentifiers) + { + List modularizedColumns = new List(); + + // find the modularized columns + for (int i = 0; i < this.Definition.Columns.Count; i++) + { + if (ColumnModularizeType.None != this.Definition.Columns[i].ModularizeType) + { + modularizedColumns.Add(i); + } + } + + if (0 < modularizedColumns.Count) + { + foreach (Row row in this.Rows) + { + foreach (int modularizedColumn in modularizedColumns) + { + Field field = row.Fields[modularizedColumn]; + + if (null != field.Data) + { + field.Data = row.GetModularizedValue(field, modularizationGuid, suppressModularizationIdentifiers); + } + } + } + } + } + + /// + /// Persists a row in an XML format. + /// + /// XmlWriter where the Row should persist itself as XML. + [SuppressMessage("Microsoft.Globalization", "CA1308:NormalizeStringsToUppercase", Justification = "Changing the way this string normalizes would result " + + "in a change to the way the intermediate files are generated, potentially causing extra churn in patches on an MSI built from an older version of WiX. " + + "Furthermore, there is no security hole here, as the strings won't need to make a round trip")] + internal void Write(XmlWriter writer) + { + if (null == writer) + { + throw new ArgumentNullException("writer"); + } + + writer.WriteStartElement("table", Intermediate.XmlNamespaceUri); + writer.WriteAttributeString("name", this.Name); + + if (TableOperation.None != this.Operation) + { + writer.WriteAttributeString("op", this.Operation.ToString().ToLowerInvariant()); + } + + foreach (Row row in this.Rows) + { + row.Write(writer); + } + + writer.WriteEndElement(); + } + + /// + /// Writes the table in IDT format to the provided stream. + /// + /// Stream to write the table to. + /// Whether to keep columns added in a transform. + public void ToIdtDefinition(StreamWriter writer, bool keepAddedColumns) + { + if (this.Definition.Unreal) + { + return; + } + + if (TableDefinition.MaxColumnsInRealTable < this.Definition.Columns.Count) + { + throw new WixException(WixDataErrors.TooManyColumnsInRealTable(this.Definition.Name, this.Definition.Columns.Count, TableDefinition.MaxColumnsInRealTable)); + } + + // Tack on the table header, and flush before we start writing bytes directly to the stream. + writer.Write(this.Definition.ToIdtDefinition(keepAddedColumns)); + writer.Flush(); + + using (NonClosingStreamWrapper wrapper = new NonClosingStreamWrapper(writer.BaseStream)) + using (BufferedStream buffStream = new BufferedStream(wrapper)) + { + // Create an encoding that replaces characters with question marks, and doesn't throw. We'll + // use this in case of errors + Encoding convertEncoding = Encoding.GetEncoding(writer.Encoding.CodePage); + + foreach (Row row in this.Rows) + { + if (row.Redundant) + { + continue; + } + + string rowString = row.ToIdtDefinition(keepAddedColumns); + byte[] rowBytes; + + try + { + // GetBytes will throw an exception if any character doesn't match our current encoding + rowBytes = writer.Encoding.GetBytes(rowString); + } + catch (EncoderFallbackException) + { + Messaging.Instance.OnMessage(WixDataErrors.InvalidStringForCodepage(row.SourceLineNumbers, Convert.ToString(writer.Encoding.WindowsCodePage, CultureInfo.InvariantCulture))); + + rowBytes = convertEncoding.GetBytes(rowString); + } + + buffStream.Write(rowBytes, 0, rowBytes.Length); + } + } + } + + /// + /// Validates the rows of this OutputTable and throws if it collides on + /// primary keys. + /// + public void ValidateRows() + { + Dictionary primaryKeys = new Dictionary(); + + foreach (Row row in this.Rows) + { + string primaryKey = row.GetPrimaryKey(); + + SourceLineNumber collisionSourceLineNumber; + if (primaryKeys.TryGetValue(primaryKey, out collisionSourceLineNumber)) + { + throw new WixException(WixDataErrors.DuplicatePrimaryKey(collisionSourceLineNumber, primaryKey, this.Definition.Name)); + } + + primaryKeys.Add(primaryKey, row.SourceLineNumbers); + } + } + } +} diff --git a/src/WixToolset.Data/TableDefinition.cs b/src/WixToolset.Data/TableDefinition.cs new file mode 100644 index 00000000..40aaac84 --- /dev/null +++ b/src/WixToolset.Data/TableDefinition.cs @@ -0,0 +1,334 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + using System.Collections.ObjectModel; + using System.Text; + using System.Xml; + + /// + /// Definition of a table in a database. + /// + public sealed class TableDefinition : IComparable + { + /// + /// Tracks the maximum number of columns supported in a real table. + /// This is a Windows Installer limitation. + /// + public const int MaxColumnsInRealTable = 32; + + /// + /// Creates a table definition. + /// + /// Name of table to create. + /// Flag if rows in this table create symbols. + /// Flag if table is unreal. + /// Flag if table is part of UX Manifest. + public TableDefinition(string name, IList columns, bool createSymbols, bool unreal, bool bootstrapperApplicationData = false) + { + this.Name = name; + this.CreateSymbols = createSymbols; + this.Unreal = unreal; + this.BootstrapperApplicationData = bootstrapperApplicationData; + + this.Columns = new ReadOnlyCollection(columns); + } + + /// + /// Gets if rows in this table create symbols. + /// + /// Flag if rows in this table create symbols. + public bool CreateSymbols { get; private set; } + + /// + /// Gets the name of the table. + /// + /// Name of the table. + public string Name { get; private set; } + + /// + /// Gets if the table is unreal. + /// + /// Flag if table is unreal. + public bool Unreal { get; private set; } + + /// + /// Gets if the table is a part of the bootstrapper application data manifest. + /// + /// Flag if table is a part of the bootstrapper application data manifest. + public bool BootstrapperApplicationData { get; private set; } + + /// + /// Gets the collection of column definitions for this table. + /// + /// Collection of column definitions for this table. + public IList Columns { get; private set; } + + /// + /// Gets the column definition in the table by index. + /// + /// Index of column to locate. + /// Column definition in the table by index. + public ColumnDefinition this[int columnIndex] + { + get { return this.Columns[columnIndex]; } + } + + /// + /// Gets the table definition in IDT format. + /// + /// Whether to keep columns added in a transform. + /// Table definition in IDT format. + public string ToIdtDefinition(bool keepAddedColumns) + { + bool first = true; + StringBuilder columnString = new StringBuilder(); + StringBuilder dataString = new StringBuilder(); + StringBuilder tableString = new StringBuilder(); + + tableString.Append(this.Name); + foreach (ColumnDefinition column in this.Columns) + { + // conditionally keep columns added in a transform; otherwise, + // break because columns can only be added at the end + if (column.Added && !keepAddedColumns) + { + break; + } + + if (!first) + { + columnString.Append('\t'); + dataString.Append('\t'); + } + + columnString.Append(column.Name); + dataString.Append(column.IdtType); + + if (column.PrimaryKey) + { + tableString.AppendFormat("\t{0}", column.Name); + } + + first = false; + } + columnString.Append("\r\n"); + columnString.Append(dataString); + columnString.Append("\r\n"); + columnString.Append(tableString); + columnString.Append("\r\n"); + + return columnString.ToString(); + } + + /// + /// Adds the validation rows to the _Validation table. + /// + /// The _Validation table. + public void AddValidationRows(Table validationTable) + { + foreach (ColumnDefinition columnDef in this.Columns) + { + Row row = validationTable.CreateRow(null); + + row[0] = this.Name; + + row[1] = columnDef.Name; + + if (columnDef.Nullable) + { + row[2] = "Y"; + } + else + { + row[2] = "N"; + } + + if (columnDef.IsMinValueSet) + { + row[3] = columnDef.MinValue; + } + + if (columnDef.IsMaxValueSet) + { + row[4] = columnDef.MaxValue; + } + + row[5] = columnDef.KeyTable; + + if (columnDef.IsKeyColumnSet) + { + row[6] = columnDef.KeyColumn; + } + + if (ColumnCategory.Unknown != columnDef.Category) + { + row[7] = columnDef.Category.ToString(); + } + + row[8] = columnDef.Possibilities; + + row[9] = columnDef.Description; + } + } + + /// + /// Compares this table definition to another table definition. + /// + /// + /// Only Windows Installer traits are compared, allowing for updates to WiX-specific table definitions. + /// + /// The updated to compare with this target definition. + /// 0 if the tables' core properties are the same; otherwise, non-0. + public int CompareTo(TableDefinition updated) + { + // by definition, this object is greater than null + if (null == updated) + { + return 1; + } + + // compare the table names + int ret = String.Compare(this.Name, updated.Name, StringComparison.Ordinal); + + // compare the column count + if (0 == ret) + { + // transforms can only add columns + ret = Math.Min(0, updated.Columns.Count - this.Columns.Count); + + // compare name, type, and length of each column + for (int i = 0; 0 == ret && this.Columns.Count > i; i++) + { + ColumnDefinition thisColumnDef = this.Columns[i]; + ColumnDefinition updatedColumnDef = updated.Columns[i]; + + ret = thisColumnDef.CompareTo(updatedColumnDef); + } + } + + return ret; + } + + /// + /// Parses table definition from xml reader. + /// + /// Reader to get data from. + /// The TableDefintion represented by the Xml. + internal static TableDefinition Read(XmlReader reader) + { + bool empty = reader.IsEmptyElement; + bool createSymbols = false; + string name = null; + bool unreal = false; + bool bootstrapperApplicationData = false; + + while (reader.MoveToNextAttribute()) + { + switch (reader.LocalName) + { + case "createSymbols": + createSymbols = reader.Value.Equals("yes"); + break; + case "name": + name = reader.Value; + break; + case "unreal": + unreal = reader.Value.Equals("yes"); + break; + case "bootstrapperApplicationData": + bootstrapperApplicationData = reader.Value.Equals("yes"); + break; + } + } + + if (null == name) + { + throw new XmlException(); + } + + List columns = new List(); + bool hasPrimaryKeyColumn = false; + + // parse the child elements + if (!empty) + { + bool done = false; + + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + switch (reader.LocalName) + { + case "columnDefinition": + ColumnDefinition columnDefinition = ColumnDefinition.Read(reader); + columns.Add(columnDefinition); + + if (columnDefinition.PrimaryKey) + { + hasPrimaryKeyColumn = true; + } + break; + default: + throw new XmlException(); + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!unreal && !bootstrapperApplicationData && !hasPrimaryKeyColumn) + { + throw new WixException(WixDataErrors.RealTableMissingPrimaryKeyColumn(SourceLineNumber.CreateFromUri(reader.BaseURI), name)); + } + + if (!done) + { + throw new XmlException(); + } + } + + TableDefinition tableDefinition = new TableDefinition(name, columns, createSymbols, unreal, bootstrapperApplicationData); + return tableDefinition; + } + + /// + /// Persists an output in an XML format. + /// + /// XmlWriter where the Output should persist itself as XML. + internal void Write(XmlWriter writer) + { + writer.WriteStartElement("tableDefinition", TableDefinitionCollection.XmlNamespaceUri); + + writer.WriteAttributeString("name", this.Name); + + if (this.CreateSymbols) + { + writer.WriteAttributeString("createSymbols", "yes"); + } + + if (this.Unreal) + { + writer.WriteAttributeString("unreal", "yes"); + } + + if (this.BootstrapperApplicationData) + { + writer.WriteAttributeString("bootstrapperApplicationData", "yes"); + } + + foreach (ColumnDefinition columnDefinition in this.Columns) + { + columnDefinition.Write(writer); + } + + writer.WriteEndElement(); + } + } +} diff --git a/src/WixToolset.Data/TableDefinitionCollection.cs b/src/WixToolset.Data/TableDefinitionCollection.cs new file mode 100644 index 00000000..553f0eaa --- /dev/null +++ b/src/WixToolset.Data/TableDefinitionCollection.cs @@ -0,0 +1,229 @@ +// 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.Data +{ + using System; + using System.Collections; + using System.Collections.Generic; + using System.Linq; + using System.Xml; + + /// + /// Collection for table definitions indexed by table name. + /// + public sealed class TableDefinitionCollection : ICollection + { + public const string XmlNamespaceUri = "http://wixtoolset.org/schemas/v4/wi/tables"; + + private Dictionary collection; + + /// + /// Instantiate a new TableDefinitionCollection class. + /// + public TableDefinitionCollection() + { + this.collection = new Dictionary(); + } + + /// + /// Creates a shallow copy of the provided table definition collection. + /// + public TableDefinitionCollection(TableDefinitionCollection tableDefinitions) + { + this.collection = new Dictionary(tableDefinitions.collection); + } + + /// + /// Gets the number of items in the collection. + /// + /// Number of items in collection. + public int Count + { + get { return this.collection.Count; } + } + + /// + /// Table definition collections are never read-only. + /// + public bool IsReadOnly + { + get { return false; } + } + + /// + /// Gets a table definition by name. + /// + /// Name of table to locate. + public TableDefinition this[string tableName] + { + get + { + TableDefinition table; + if (!this.collection.TryGetValue(tableName, out table)) + { + throw new WixMissingTableDefinitionException(WixDataErrors.MissingTableDefinition(tableName)); + } + + return table; + } + } + + /// + /// Load a table definition collection from an XmlReader. + /// + /// Reader to get data from. + /// Suppress xml schema validation while loading. + /// The TableDefinitionCollection represented by the xml. + public static TableDefinitionCollection Load(XmlReader reader) + { + reader.MoveToContent(); + + return Read(reader); + } + + /// + /// Adds a table definition to the collection. + /// + /// Table definition to add to the collection. + /// Indexes by table definition name. + public void Add(TableDefinition tableDefinition) + { + this.collection.Add(tableDefinition.Name, tableDefinition); + } + + /// + /// Removes all table definitions from the collection. + /// + public void Clear() + { + this.collection.Clear(); + } + + /// + /// Checks if the collection contains a table name. + /// + /// The table to check in the collection. + /// True if collection contains the table. + public bool Contains(string tableName) + { + return this.collection.ContainsKey(tableName); + } + + /// + /// Checks if the collection contains a table. + /// + /// The table to check in the collection. + /// True if collection contains the table. + public bool Contains(TableDefinition table) + { + return this.collection.ContainsKey(table.Name); + } + + /// + /// Copies table definitions to an arry. + /// + /// Array to copy the table definitions to. + /// Index in the array to start copying at. + public void CopyTo(TableDefinition[] array, int index) + { + this.collection.Values.CopyTo(array, index); + } + + /// + /// Removes a table definition from the collection. + /// + /// Table to remove from the collection. + /// True if the table definition existed in the collection and was removed. + public bool Remove(TableDefinition table) + { + return this.collection.Remove(table.Name); + } + + /// + /// Gets enumerator for the collection. + /// + /// Enumerator for the collection. + public IEnumerator GetEnumerator() + { + return this.collection.Values.GetEnumerator(); + } + + /// + /// Gets the untyped enumerator for the collection. + /// + /// Untyped enumerator for the collection. + IEnumerator IEnumerable.GetEnumerator() + { + return this.collection.Values.GetEnumerator(); + } + + /// + /// Loads a collection of table definitions from a XmlReader in memory. + /// + /// Reader to get data from. + /// The TableDefinitionCollection represented by the xml. + internal static TableDefinitionCollection Read(XmlReader reader) + { + if ("tableDefinitions" != reader.LocalName) + { + throw new XmlException(); + } + + bool empty = reader.IsEmptyElement; + TableDefinitionCollection tableDefinitionCollection = new TableDefinitionCollection(); + + while (reader.MoveToNextAttribute()) + { + } + + // parse the child elements + if (!empty) + { + bool done = false; + + while (!done && reader.Read()) + { + switch (reader.NodeType) + { + case XmlNodeType.Element: + switch (reader.LocalName) + { + case "tableDefinition": + tableDefinitionCollection.Add(TableDefinition.Read(reader)); + break; + default: + throw new XmlException(); + } + break; + case XmlNodeType.EndElement: + done = true; + break; + } + } + + if (!done) + { + throw new XmlException(); + } + } + + return tableDefinitionCollection; + } + + /// + /// Persists a TableDefinitionCollection in an XML format. + /// + /// XmlWriter where the TableDefinitionCollection should persist itself as XML. + internal void Write(XmlWriter writer) + { + writer.WriteStartElement("tableDefinitions", XmlNamespaceUri); + + foreach (TableDefinition tableDefinition in this.collection.Values.OrderBy(t => t.Name)) + { + tableDefinition.Write(writer); + } + + writer.WriteEndElement(); + } + } +} diff --git a/src/WixToolset.Data/TableExtensions.cs b/src/WixToolset.Data/TableExtensions.cs new file mode 100644 index 00000000..1be64ffe --- /dev/null +++ b/src/WixToolset.Data/TableExtensions.cs @@ -0,0 +1,23 @@ +// 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.Data +{ + using System.Collections.Generic; + using System.Linq; + + /// + /// Methods that extend . + /// + public static class TableExtensions + { + /// + /// Gets the rows contained in the table as a particular row type. + /// + /// Table to get rows from. + /// If the is null, an empty enumerable will be returned. + public static IEnumerable RowsAs(this Table table) where T : Row + { + return (null == table) ? Enumerable.Empty() : table.Rows.Cast(); + } + } +} diff --git a/src/WixToolset.Data/TableIndexedCollection.cs b/src/WixToolset.Data/TableIndexedCollection.cs new file mode 100644 index 00000000..9f85efff --- /dev/null +++ b/src/WixToolset.Data/TableIndexedCollection.cs @@ -0,0 +1,153 @@ +// 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.Data +{ + using System; + using System.Collections.Generic; + using System.Linq; + + /// + /// Collection for tables. + /// + public sealed class TableIndexedCollection : ICollection
+ { + private Dictionary collection; + + /// + /// Instantiate a new empty collection. + /// + public TableIndexedCollection() + { + this.collection = new Dictionary(); + } + + /// + /// Instantiate a new collection populated with a set of tables. + /// + /// Set of tables. + public TableIndexedCollection(IEnumerable
tables) + { + this.collection = tables.ToDictionary(t => t.Name); + } + + /// + /// Gets the number of items in the collection. + /// + /// Number of items in collection. + public int Count + { + get { return this.collection.Count; } + } + + /// + /// Table indexed collection is never read only. + /// + public bool IsReadOnly + { + get { return false; } + } + + /// + /// Adds a table to the collection. + /// + /// Table to add to the collection. + /// Indexes the table by name. + public void Add(Table table) + { + this.collection.Add(table.Name, table); + } + + /// + /// Clear the tables from the collection. + /// + public void Clear() + { + this.collection.Clear(); + } + + /// + /// Determines if a table is in the collection. + /// + /// Table to check if it is in the collection. + /// True if the table name is in the collection, otherwise false. + public bool Contains(Table table) + { + return this.collection.ContainsKey(table.Name); + } + + /// + /// Copies the collection into an array. + /// + /// Array to copy the collection into. + /// Index to start copying from. + public void CopyTo(Table[] array, int arrayIndex) + { + this.collection.Values.CopyTo(array, arrayIndex); + } + + /// + /// Remove a table from the collection by name. + /// + /// Table name to remove from the collection. + public void Remove(string tableName) + { + this.collection.Remove(tableName); + } + + /// + /// Remove a table from the collection. + /// + /// Table with matching name to remove from the collection. + public bool Remove(Table table) + { + return this.collection.Remove(table.Name); + } + + /// + /// Gets an enumerator over the whole collection. + /// + /// Collection enumerator. + public IEnumerator
GetEnumerator() + { + return this.collection.Values.GetEnumerator(); + } + + /// + /// Gets an untyped enumerator over the whole collection. + /// + /// Untyped collection enumerator. + System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() + { + return this.collection.Values.GetEnumerator(); + } + + /// + /// Gets a table by name. + /// + /// Name of table to locate. + public Table this[string tableName] + { + get + { + Table table; + return this.collection.TryGetValue(tableName, out table) ? table : null; + } + + set + { + this.collection[tableName] = value; + } + } + + /// + /// Tries to find a table by name. + /// + /// Table name to locate. + /// Found table. + /// True if table with table name was found, otherwise false. + public bool TryGetTable(string tableName, out Table table) + { + return this.collection.TryGetValue(tableName, out table); + } + } +} diff --git a/src/WixToolset.Data/TableOperation.cs b/src/WixToolset.Data/TableOperation.cs new file mode 100644 index 00000000..8df44e73 --- /dev/null +++ b/src/WixToolset.Data/TableOperation.cs @@ -0,0 +1,25 @@ +// 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.Data +{ + /// + /// The table transform operations. + /// + public enum TableOperation + { + /// + /// No operation. + /// + None, + + /// + /// Added table. + /// + Add, + + /// + /// Dropped table. + /// + Drop, + } +} diff --git a/src/WixToolset.Data/WindowsInstallerStandard.cs b/src/WixToolset.Data/WindowsInstallerStandard.cs new file mode 100644 index 00000000..1060d11b --- /dev/null +++ b/src/WixToolset.Data/WindowsInstallerStandard.cs @@ -0,0 +1,444 @@ +// 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.Data +{ + using System.Collections.Generic; + using System.Reflection; + using System.Xml; + using WixToolset.Data.Rows; + + /// + /// Represents the Windows Installer standard objects. + /// + public static class WindowsInstallerStandard + { + private static readonly object lockObject = new object(); + + private static TableDefinitionCollection tableDefinitions; + private static WixActionRowCollection standardActions; + + private static HashSet standardActionNames; + private static HashSet standardDirectories; + private static HashSet standardProperties; + + + /// + /// Gets the table definitions stored in this assembly. + /// + /// Table definition collection for tables stored in this assembly. + public static TableDefinitionCollection GetTableDefinitions() + { + lock (lockObject) + { + if (null == WindowsInstallerStandard.tableDefinitions) + { + using (XmlReader reader = XmlReader.Create(Assembly.GetExecutingAssembly().GetManifestResourceStream("WixToolset.Data.Data.tables.xml"))) + { + tableDefinitions = TableDefinitionCollection.Load(reader); + } + } + } + + return WindowsInstallerStandard.tableDefinitions; + } + + /// + /// Gets the standard actions stored in this assembly. + /// + /// Collection of standard actions in this assembly. + public static WixActionRowCollection GetStandardActions() + { + lock (lockObject) + { + if (null == standardActions) + { + using (XmlReader reader = XmlReader.Create(Assembly.GetExecutingAssembly().GetManifestResourceStream("WixToolset.Data.Data.actions.xml"))) + { + standardActions = WixActionRowCollection.Load(reader); + } + } + } + + return standardActions; + } + + + /// + /// Gets (and loads if not yet loaded) the list of standard MSI directories. + /// + /// The list of standard MSI directories. + public static HashSet GetStandardDirectories() + { + lock (lockObject) + { + if (null == standardDirectories) + { + LoadStandardDirectories(); + } + } + + return standardDirectories; + } + + /// + /// Find out if an action is a standard action. + /// + /// Name of the action. + /// true if the action is standard, false otherwise. + public static bool IsStandardAction(string actionName) + { + lock (lockObject) + { + if (null == standardActionNames) + { + standardActionNames = new HashSet(); + standardActionNames.Add("AllocateRegistrySpace"); + standardActionNames.Add("AppSearch"); + standardActionNames.Add("BindImage"); + standardActionNames.Add("CCPSearch"); + standardActionNames.Add("CostFinalize"); + standardActionNames.Add("CostInitialize"); + standardActionNames.Add("CreateFolders"); + standardActionNames.Add("CreateShortcuts"); + standardActionNames.Add("DeleteServices"); + standardActionNames.Add("DisableRollback"); + standardActionNames.Add("DuplicateFiles"); + standardActionNames.Add("ExecuteAction"); + standardActionNames.Add("FileCost"); + standardActionNames.Add("FindRelatedProducts"); + standardActionNames.Add("ForceReboot"); + standardActionNames.Add("InstallAdminPackage"); + standardActionNames.Add("InstallExecute"); + standardActionNames.Add("InstallExecuteAgain"); + standardActionNames.Add("InstallFiles"); + standardActionNames.Add("InstallFinalize"); + standardActionNames.Add("InstallInitialize"); + standardActionNames.Add("InstallODBC"); + standardActionNames.Add("InstallServices"); + standardActionNames.Add("InstallSFPCatalogFile"); + standardActionNames.Add("InstallValidate"); + standardActionNames.Add("IsolateComponents"); + standardActionNames.Add("LaunchConditions"); + standardActionNames.Add("MigrateFeatureStates"); + standardActionNames.Add("MoveFiles"); + standardActionNames.Add("MsiConfigureServices"); + standardActionNames.Add("MsiPublishAssemblies"); + standardActionNames.Add("MsiUnpublishAssemblies"); + standardActionNames.Add("PatchFiles"); + standardActionNames.Add("ProcessComponents"); + standardActionNames.Add("PublishComponents"); + standardActionNames.Add("PublishFeatures"); + standardActionNames.Add("PublishProduct"); + standardActionNames.Add("RegisterClassInfo"); + standardActionNames.Add("RegisterComPlus"); + standardActionNames.Add("RegisterExtensionInfo"); + standardActionNames.Add("RegisterFonts"); + standardActionNames.Add("RegisterMIMEInfo"); + standardActionNames.Add("RegisterProduct"); + standardActionNames.Add("RegisterProgIdInfo"); + standardActionNames.Add("RegisterTypeLibraries"); + standardActionNames.Add("RegisterUser"); + standardActionNames.Add("RemoveDuplicateFiles"); + standardActionNames.Add("RemoveEnvironmentStrings"); + standardActionNames.Add("RemoveExistingProducts"); + standardActionNames.Add("RemoveFiles"); + standardActionNames.Add("RemoveFolders"); + standardActionNames.Add("RemoveIniValues"); + standardActionNames.Add("RemoveODBC"); + standardActionNames.Add("RemoveRegistryValues"); + standardActionNames.Add("RemoveShortcuts"); + standardActionNames.Add("ResolveSource"); + standardActionNames.Add("RMCCPSearch"); + standardActionNames.Add("ScheduleReboot"); + standardActionNames.Add("SelfRegModules"); + standardActionNames.Add("SelfUnregModules"); + standardActionNames.Add("SetODBCFolders"); + standardActionNames.Add("StartServices"); + standardActionNames.Add("StopServices"); + standardActionNames.Add("UnpublishComponents"); + standardActionNames.Add("UnpublishFeatures"); + standardActionNames.Add("UnregisterClassInfo"); + standardActionNames.Add("UnregisterComPlus"); + standardActionNames.Add("UnregisterExtensionInfo"); + standardActionNames.Add("UnregisterFonts"); + standardActionNames.Add("UnregisterMIMEInfo"); + standardActionNames.Add("UnregisterProgIdInfo"); + standardActionNames.Add("UnregisterTypeLibraries"); + standardActionNames.Add("ValidateProductID"); + standardActionNames.Add("WriteEnvironmentStrings"); + standardActionNames.Add("WriteIniValues"); + standardActionNames.Add("WriteRegistryValues"); + } + } + + return standardActionNames.Contains(actionName); + } + + /// + /// Find out if a directory is a standard directory. + /// + /// Name of the directory. + /// true if the directory is standard, false otherwise. + public static bool IsStandardDirectory(string directoryName) + { + lock (lockObject) + { + if (null == standardDirectories) + { + LoadStandardDirectories(); + } + } + + return standardDirectories.Contains(directoryName); + } + + /// + /// Find out if a property is a standard property. + /// References: + /// Title: Property Reference [Windows Installer]: + /// URL: http://msdn.microsoft.com/library/en-us/msi/setup/property_reference.asp + /// + /// Name of the property. + /// true if a property is standard, false otherwise. + public static bool IsStandardProperty(string propertyName) + { + lock (lockObject) + { + if (null == standardProperties) + { + standardProperties = new HashSet(); + standardProperties.Add("~"); // REG_MULTI_SZ/NULL marker + standardProperties.Add("ACTION"); + standardProperties.Add("ADDDEFAULT"); + standardProperties.Add("ADDLOCAL"); + standardProperties.Add("ADDDSOURCE"); + standardProperties.Add("AdminProperties"); + standardProperties.Add("AdminUser"); + standardProperties.Add("ADVERTISE"); + standardProperties.Add("AFTERREBOOT"); + standardProperties.Add("AllowProductCodeMismatches"); + standardProperties.Add("AllowProductVersionMajorMismatches"); + standardProperties.Add("ALLUSERS"); + standardProperties.Add("Alpha"); + standardProperties.Add("ApiPatchingSymbolFlags"); + standardProperties.Add("ARPAUTHORIZEDCDFPREFIX"); + standardProperties.Add("ARPCOMMENTS"); + standardProperties.Add("ARPCONTACT"); + standardProperties.Add("ARPHELPLINK"); + standardProperties.Add("ARPHELPTELEPHONE"); + standardProperties.Add("ARPINSTALLLOCATION"); + standardProperties.Add("ARPNOMODIFY"); + standardProperties.Add("ARPNOREMOVE"); + standardProperties.Add("ARPNOREPAIR"); + standardProperties.Add("ARPPRODUCTIONICON"); + standardProperties.Add("ARPREADME"); + standardProperties.Add("ARPSIZE"); + standardProperties.Add("ARPSYSTEMCOMPONENT"); + standardProperties.Add("ARPULRINFOABOUT"); + standardProperties.Add("ARPURLUPDATEINFO"); + standardProperties.Add("AVAILABLEFREEREG"); + standardProperties.Add("BorderSize"); + standardProperties.Add("BorderTop"); + standardProperties.Add("CaptionHeight"); + standardProperties.Add("CCP_DRIVE"); + standardProperties.Add("ColorBits"); + standardProperties.Add("COMPADDLOCAL"); + standardProperties.Add("COMPADDSOURCE"); + standardProperties.Add("COMPANYNAME"); + standardProperties.Add("ComputerName"); + standardProperties.Add("CostingComplete"); + standardProperties.Add("Date"); + standardProperties.Add("DefaultUIFont"); + standardProperties.Add("DISABLEADVTSHORTCUTS"); + standardProperties.Add("DISABLEMEDIA"); + standardProperties.Add("DISABLEROLLBACK"); + standardProperties.Add("DiskPrompt"); + standardProperties.Add("DontRemoveTempFolderWhenFinished"); + standardProperties.Add("EnableUserControl"); + standardProperties.Add("EXECUTEACTION"); + standardProperties.Add("EXECUTEMODE"); + standardProperties.Add("FASTOEM"); + standardProperties.Add("FILEADDDEFAULT"); + standardProperties.Add("FILEADDLOCAL"); + standardProperties.Add("FILEADDSOURCE"); + standardProperties.Add("IncludeWholeFilesOnly"); + standardProperties.Add("Installed"); + standardProperties.Add("INSTALLLEVEL"); + standardProperties.Add("Intel"); + standardProperties.Add("Intel64"); + standardProperties.Add("IsAdminPackage"); + standardProperties.Add("LeftUnit"); + standardProperties.Add("LIMITUI"); + standardProperties.Add("ListOfPatchGUIDsToReplace"); + standardProperties.Add("ListOfTargetProductCode"); + standardProperties.Add("LOGACTION"); + standardProperties.Add("LogonUser"); + standardProperties.Add("Manufacturer"); + standardProperties.Add("MEDIAPACKAGEPATH"); + standardProperties.Add("MediaSourceDir"); + standardProperties.Add("MinimumRequiredMsiVersion"); + standardProperties.Add("MsiAMD64"); + standardProperties.Add("MSIAPRSETTINGSIDENTIFIER"); + standardProperties.Add("MSICHECKCRCS"); + standardProperties.Add("MSIDISABLERMRESTART"); + standardProperties.Add("MSIENFORCEUPGRADECOMPONENTRULES"); + standardProperties.Add("MSIFASTINSTALL"); + standardProperties.Add("MsiFileToUseToCreatePatchTables"); + standardProperties.Add("MsiHiddenProperties"); + standardProperties.Add("MSIINSTALLPERUSER"); + standardProperties.Add("MSIINSTANCEGUID"); + standardProperties.Add("MsiLogFileLocation"); + standardProperties.Add("MsiLogging"); + standardProperties.Add("MsiNetAssemblySupport"); + standardProperties.Add("MSINEWINSTANCE"); + standardProperties.Add("MSINODISABLEMEDIA"); + standardProperties.Add("MsiNTProductType"); + standardProperties.Add("MsiNTSuiteBackOffice"); + standardProperties.Add("MsiNTSuiteDataCenter"); + standardProperties.Add("MsiNTSuiteEnterprise"); + standardProperties.Add("MsiNTSuiteSmallBusiness"); + standardProperties.Add("MsiNTSuiteSmallBusinessRestricted"); + standardProperties.Add("MsiNTSuiteWebServer"); + standardProperties.Add("MsiNTSuitePersonal"); + standardProperties.Add("MsiPatchRemovalList"); + standardProperties.Add("MSIPATCHREMOVE"); + standardProperties.Add("MSIRESTARTMANAGERCONTROL"); + standardProperties.Add("MsiRestartManagerSessionKey"); + standardProperties.Add("MSIRMSHUTDOWN"); + standardProperties.Add("MsiRunningElevated"); + standardProperties.Add("MsiUIHideCancel"); + standardProperties.Add("MsiUIProgressOnly"); + standardProperties.Add("MsiUISourceResOnly"); + standardProperties.Add("MsiSystemRebootPending"); + standardProperties.Add("MsiWin32AssemblySupport"); + standardProperties.Add("NOCOMPANYNAME"); + standardProperties.Add("NOUSERNAME"); + standardProperties.Add("OLEAdvtSupport"); + standardProperties.Add("OptimizePatchSizeForLargeFiles"); + standardProperties.Add("OriginalDatabase"); + standardProperties.Add("OutOfDiskSpace"); + standardProperties.Add("OutOfNoRbDiskSpace"); + standardProperties.Add("ParentOriginalDatabase"); + standardProperties.Add("ParentProductCode"); + standardProperties.Add("PATCH"); + standardProperties.Add("PATCH_CACHE_DIR"); + standardProperties.Add("PATCH_CACHE_ENABLED"); + standardProperties.Add("PatchGUID"); + standardProperties.Add("PATCHNEWPACKAGECODE"); + standardProperties.Add("PATCHNEWSUMMARYCOMMENTS"); + standardProperties.Add("PATCHNEWSUMMARYSUBJECT"); + standardProperties.Add("PatchOutputPath"); + standardProperties.Add("PatchSourceList"); + standardProperties.Add("PhysicalMemory"); + standardProperties.Add("PIDKEY"); + standardProperties.Add("PIDTemplate"); + standardProperties.Add("Preselected"); + standardProperties.Add("PRIMARYFOLDER"); + standardProperties.Add("PrimaryVolumePath"); + standardProperties.Add("PrimaryVolumeSpaceAvailable"); + standardProperties.Add("PrimaryVolumeSpaceRemaining"); + standardProperties.Add("PrimaryVolumeSpaceRequired"); + standardProperties.Add("Privileged"); + standardProperties.Add("ProductCode"); + standardProperties.Add("ProductID"); + standardProperties.Add("ProductLanguage"); + standardProperties.Add("ProductName"); + standardProperties.Add("ProductState"); + standardProperties.Add("ProductVersion"); + standardProperties.Add("PROMPTROLLBACKCOST"); + standardProperties.Add("REBOOT"); + standardProperties.Add("REBOOTPROMPT"); + standardProperties.Add("RedirectedDllSupport"); + standardProperties.Add("REINSTALL"); + standardProperties.Add("REINSTALLMODE"); + standardProperties.Add("RemoveAdminTS"); + standardProperties.Add("REMOVE"); + standardProperties.Add("ReplacedInUseFiles"); + standardProperties.Add("RestrictedUserControl"); + standardProperties.Add("RESUME"); + standardProperties.Add("RollbackDisabled"); + standardProperties.Add("ROOTDRIVE"); + standardProperties.Add("ScreenX"); + standardProperties.Add("ScreenY"); + standardProperties.Add("SecureCustomProperties"); + standardProperties.Add("ServicePackLevel"); + standardProperties.Add("ServicePackLevelMinor"); + standardProperties.Add("SEQUENCE"); + standardProperties.Add("SharedWindows"); + standardProperties.Add("ShellAdvtSupport"); + standardProperties.Add("SHORTFILENAMES"); + standardProperties.Add("SourceDir"); + standardProperties.Add("SOURCELIST"); + standardProperties.Add("SystemLanguageID"); + standardProperties.Add("TARGETDIR"); + standardProperties.Add("TerminalServer"); + standardProperties.Add("TextHeight"); + standardProperties.Add("Time"); + standardProperties.Add("TRANSFORMS"); + standardProperties.Add("TRANSFORMSATSOURCE"); + standardProperties.Add("TRANSFORMSSECURE"); + standardProperties.Add("TTCSupport"); + standardProperties.Add("UILevel"); + standardProperties.Add("UpdateStarted"); + standardProperties.Add("UpgradeCode"); + standardProperties.Add("UPGRADINGPRODUCTCODE"); + standardProperties.Add("UserLanguageID"); + standardProperties.Add("USERNAME"); + standardProperties.Add("UserSID"); + standardProperties.Add("Version9X"); + standardProperties.Add("VersionDatabase"); + standardProperties.Add("VersionMsi"); + standardProperties.Add("VersionNT"); + standardProperties.Add("VersionNT64"); + standardProperties.Add("VirtualMemory"); + standardProperties.Add("WindowsBuild"); + standardProperties.Add("WindowsVolume"); + } + } + + return standardProperties.Contains(propertyName); + } + + /// + /// Sets up a hashtable with the set of standard MSI directories + /// + private static void LoadStandardDirectories() + { + lock (lockObject) + { + if (null == standardDirectories) + { + standardDirectories = new HashSet(); + standardDirectories.Add("TARGETDIR"); + standardDirectories.Add("AdminToolsFolder"); + standardDirectories.Add("AppDataFolder"); + standardDirectories.Add("CommonAppDataFolder"); + standardDirectories.Add("CommonFilesFolder"); + standardDirectories.Add("DesktopFolder"); + standardDirectories.Add("FavoritesFolder"); + standardDirectories.Add("FontsFolder"); + standardDirectories.Add("LocalAppDataFolder"); + standardDirectories.Add("MyPicturesFolder"); + standardDirectories.Add("PersonalFolder"); + standardDirectories.Add("ProgramFilesFolder"); + standardDirectories.Add("ProgramMenuFolder"); + standardDirectories.Add("SendToFolder"); + standardDirectories.Add("StartMenuFolder"); + standardDirectories.Add("StartupFolder"); + standardDirectories.Add("System16Folder"); + standardDirectories.Add("SystemFolder"); + standardDirectories.Add("TempFolder"); + standardDirectories.Add("TemplateFolder"); + standardDirectories.Add("WindowsFolder"); + standardDirectories.Add("CommonFiles64Folder"); + standardDirectories.Add("ProgramFiles64Folder"); + standardDirectories.Add("System64Folder"); + standardDirectories.Add("NetHoodFolder"); + standardDirectories.Add("PrintHoodFolder"); + standardDirectories.Add("RecentFolder"); + standardDirectories.Add("WindowsVolume"); + } + } + } + } +} diff --git a/src/WixToolset.Data/WixCorruptFileException.cs b/src/WixToolset.Data/WixCorruptFileException.cs new file mode 100644 index 00000000..f663b92d --- /dev/null +++ b/src/WixToolset.Data/WixCorruptFileException.cs @@ -0,0 +1,29 @@ +// 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.Data +{ + using System; + + /// + /// Exception when file does not match the expected format. + /// + public class WixCorruptFileException : WixException + { + public WixCorruptFileException(string path, FileFormat format, Exception innerException = null) + : base(WixDataErrors.CorruptFileFormat(path, format.ToString().ToLowerInvariant()), innerException) + { + this.Path = path; + this.FileFormat = format; + } + + /// + /// Gets the actual file format found in the file. + /// + public FileFormat FileFormat { get; private set; } + + /// + /// Gets the path to the file with unexpected format. + /// + public string Path { get; set; } + } +} diff --git a/src/WixToolset.Data/WixDataStrings.Designer.cs b/src/WixToolset.Data/WixDataStrings.Designer.cs new file mode 100644 index 00000000..27c2b22c --- /dev/null +++ b/src/WixToolset.Data/WixDataStrings.Designer.cs @@ -0,0 +1,307 @@ +// 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.Data { + using System; + + + /// + /// A strongly-typed resource class, for looking up localized strings, etc. + /// + // This class was auto-generated by the StronglyTypedResourceBuilder + // class via a tool like ResGen or Visual Studio. + // To add or remove a member, edit your .ResX file then rerun ResGen + // with the /str option, or rebuild your VS project. + [global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "4.0.0.0")] + [global::System.Diagnostics.DebuggerNonUserCodeAttribute()] + [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()] + internal class WixDataStrings { + + private static global::System.Resources.ResourceManager resourceMan; + + private static global::System.Globalization.CultureInfo resourceCulture; + + [global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")] + internal WixDataStrings() { + } + + /// + /// Returns the cached ResourceManager instance used by this class. + /// + [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] + internal static global::System.Resources.ResourceManager ResourceManager { + get { + if (object.ReferenceEquals(resourceMan, null)) { + global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("WixToolset.Data.WixDataStrings", typeof(WixDataStrings).Assembly); + resourceMan = temp; + } + return resourceMan; + } + } + + /// + /// Overrides the current thread's CurrentUICulture property for all + /// resource lookups using this strongly typed resource class. + /// + [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] + internal static global::System.Globalization.CultureInfo Culture { + get { + return resourceCulture; + } + set { + resourceCulture = value; + } + } + + /// + /// Looks up a localized string similar to Cannot index into a FileRowCollection that allows duplicate FileIds. + /// + internal static string EXP_CannotIndexIntoFileRowCollection { + get { + return ResourceManager.GetString("EXP_CannotIndexIntoFileRowCollection", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to The value '{0}' is not a legal identifier and therefore cannot be modularized.. + /// + internal static string EXP_CannotModularizeIllegalID { + get { + return ResourceManager.GetString("EXP_CannotModularizeIllegalID", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to A Merge table FileCompression column cannot be set to the invalid value '{0}'.. + /// + internal static string EXP_CannotSetMergeTableFileCompressionColumnToInvalidValue { + get { + return ResourceManager.GetString("EXP_CannotSetMergeTableFileCompressionColumnToInvalidValue", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to Collection has {0} elements. Must have at least one.. + /// + internal static string EXP_CollectionMustHaveAtLeastOneElement { + get { + return ResourceManager.GetString("EXP_CollectionMustHaveAtLeastOneElement", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to Didn't find duplicated symbol.. + /// + internal static string EXP_DidnotFindDuplicateSymbol { + get { + return ResourceManager.GetString("EXP_DidnotFindDuplicateSymbol", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to Element must be a subclass of {0}, but was of type {1}.. + /// + internal static string EXP_ElementIsSubclassOfDifferentType { + get { + return ResourceManager.GetString("EXP_ElementIsSubclassOfDifferentType", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to Element of type {0} found in enumerator. Must be ChoiceItem or SequenceItem.. + /// + internal static string EXP_ElementMustBeChoiceItemOrSequenceItem { + get { + return ResourceManager.GetString("EXP_ElementMustBeChoiceItemOrSequenceItem", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to Element of type {0} is not valid for this collection.. + /// + internal static string EXP_ElementOfTypeIsNotValidForThisCollection { + get { + return ResourceManager.GetString("EXP_ElementOfTypeIsNotValidForThisCollection", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to Expected ComplexReference type.. + /// + internal static string EXP_ExpectedComplexReferenceType { + get { + return ResourceManager.GetString("EXP_ExpectedComplexReferenceType", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to ISchemaElement with name {0} does not implement ICreateChildren.. + /// + internal static string EXP_ISchemaElementDoesnotImplementICreateChildren { + get { + return ResourceManager.GetString("EXP_ISchemaElementDoesnotImplementICreateChildren", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to ISchemaElement with name {0} does not implement ISetAttributes.. + /// + internal static string EXP_ISchemaElementDoesnotImplementISetAttribute { + get { + return ResourceManager.GetString("EXP_ISchemaElementDoesnotImplementISetAttribute", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to A Merge table FileCompression column contains an invalid value '{0}'.. + /// + internal static string EXP_MergeTableFileCompressionColumnContainsInvalidValue { + get { + return ResourceManager.GetString("EXP_MergeTableFileCompressionColumnContainsInvalidValue", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to Multiple root elements found in file.. + /// + internal static string EXP_MultipleRootElementsFoundInFile { + get { + return ResourceManager.GetString("EXP_MultipleRootElementsFoundInFile", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to The other object is not a FileRow.. + /// + internal static string EXP_OtherObjectIsNotFileRow { + get { + return ResourceManager.GetString("EXP_OtherObjectIsNotFileRow", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to Type {0} is not valid for this collection.. + /// + internal static string EXP_TypeIsNotValidForThisCollection { + get { + return ResourceManager.GetString("EXP_TypeIsNotValidForThisCollection", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to Unexpected entry section type: {0}. + /// + internal static string EXP_UnexpectedEntrySectionType { + get { + return ResourceManager.GetString("EXP_UnexpectedEntrySectionType", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to Unknown column type: {0}. + /// + internal static string EXP_UnknownColumnType { + get { + return ResourceManager.GetString("EXP_UnknownColumnType", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to Unknown compression level type: {0}. + /// + internal static string EXP_UnknownCompressionLevelType { + get { + return ResourceManager.GetString("EXP_UnknownCompressionLevelType", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to The table {0} is not supported.. + /// + internal static string EXP_UnsupportedTable { + get { + return ResourceManager.GetString("EXP_UnsupportedTable", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to XmlElement with name {0} does not have a corresponding ISchemaElement.. + /// + internal static string EXP_XmlElementDoesnotHaveISchemaElement { + get { + return ResourceManager.GetString("EXP_XmlElementDoesnotHaveISchemaElement", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to {0}({1}). + /// + internal static string Format_FirstLineNumber { + get { + return ResourceManager.GetString("Format_FirstLineNumber", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to {0}. + /// + internal static string Format_InfoMessage { + get { + return ResourceManager.GetString("Format_InfoMessage", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to {0}: line {1}. + /// + internal static string Format_LineNumber { + get { + return ResourceManager.GetString("Format_LineNumber", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to {0} : {1} {2}{3:0000} : {4}. + /// + internal static string Format_NonInfoMessage { + get { + return ResourceManager.GetString("Format_NonInfoMessage", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to Source trace:{0}. + /// + internal static string INF_SourceTrace { + get { + return ResourceManager.GetString("INF_SourceTrace", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to at {0}{1}. + /// + internal static string INF_SourceTraceLocation { + get { + return ResourceManager.GetString("INF_SourceTraceLocation", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to error. + /// + internal static string MessageType_Error { + get { + return ResourceManager.GetString("MessageType_Error", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to warning. + /// + internal static string MessageType_Warning { + get { + return ResourceManager.GetString("MessageType_Warning", resourceCulture); + } + } + } +} diff --git a/src/WixToolset.Data/WixDataStrings.resx b/src/WixToolset.Data/WixDataStrings.resx new file mode 100644 index 00000000..16b70d83 --- /dev/null +++ b/src/WixToolset.Data/WixDataStrings.resx @@ -0,0 +1,204 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + text/microsoft-resx + + + 2.0 + + + System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + Unexpected entry section type: {0} + + + The table {0} is not supported. + + + A Merge table FileCompression column contains an invalid value '{0}'. + + + The other object is not a FileRow. + + + Cannot index into a FileRowCollection that allows duplicate FileIds + + + A Merge table FileCompression column cannot be set to the invalid value '{0}'. + + + Expected ComplexReference type. + + + The value '{0}' is not a legal identifier and therefore cannot be modularized. + + + Didn't find duplicated symbol. + + + Unknown column type: {0} + + + Unknown compression level type: {0} + + + {0}({1}) + + + {0} + + + {0}: line {1} + + + {0} : {1} {2}{3:0000} : {4} + + + Source trace:{0} + + + at {0}{1} + + + error + + + warning + + + Multiple root elements found in file. + + + ISchemaElement with name {0} does not implement ICreateChildren. + + + ISchemaElement with name {0} does not implement ISetAttributes. + + + XmlElement with name {0} does not have a corresponding ISchemaElement. + + + Collection has {0} elements. Must have at least one. + + + Element must be a subclass of {0}, but was of type {1}. + + + Element of type {0} is not valid for this collection. + + + Type {0} is not valid for this collection. + + + Element of type {0} found in enumerator. Must be ChoiceItem or SequenceItem. + + \ No newline at end of file diff --git a/src/WixToolset.Data/WixException.cs b/src/WixToolset.Data/WixException.cs new file mode 100644 index 00000000..1254e090 --- /dev/null +++ b/src/WixToolset.Data/WixException.cs @@ -0,0 +1,44 @@ +// 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.Data +{ + using System; + + /// + /// Base class for all WiX exceptions. + /// + [Serializable] + public class WixException : Exception + { + private MessageEventArgs error; + + /// + /// Instantiate a new WixException with a given WixError. + /// + /// The localized error information. + public WixException(MessageEventArgs error) + : this(error, null) + { + } + + /// + /// Instantiate a new WixException with a given WixError. + /// + /// The localized error information. + /// Original exception. + public WixException(MessageEventArgs error, Exception exception) : + base(error.GenerateMessageString(), exception) + { + this.error = error; + } + + /// + /// Gets the error message. + /// + /// The error message. + public MessageEventArgs Error + { + get { return this.error; } + } + } +} diff --git a/src/WixToolset.Data/WixInvalidIdtException.cs b/src/WixToolset.Data/WixInvalidIdtException.cs new file mode 100644 index 00000000..33fd0591 --- /dev/null +++ b/src/WixToolset.Data/WixInvalidIdtException.cs @@ -0,0 +1,32 @@ +// 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.Data +{ + using System; + + /// + /// WiX invalid idt exception. + /// + [Serializable] + public sealed class WixInvalidIdtException : WixException + { + /// + /// Instantiate a new WixInvalidIdtException. + /// + /// The invalid idt file. + public WixInvalidIdtException(string idtFile) : + base(WixDataErrors.InvalidIdt(new SourceLineNumber(idtFile), idtFile)) + { + } + + /// + /// Instantiate a new WixInvalidIdtException. + /// + /// The invalid idt file. + /// The table name of the invalid idt file. + public WixInvalidIdtException(string idtFile, string tableName) : + base(WixDataErrors.InvalidIdt(new SourceLineNumber(idtFile), idtFile, tableName)) + { + } + } +} diff --git a/src/WixToolset.Data/WixMissingTableDefinitionException.cs b/src/WixToolset.Data/WixMissingTableDefinitionException.cs new file mode 100644 index 00000000..6295813b --- /dev/null +++ b/src/WixToolset.Data/WixMissingTableDefinitionException.cs @@ -0,0 +1,22 @@ +// 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.Data +{ + using System; + + /// + /// Exception thrown when a table definition is missing. + /// + [Serializable] + public class WixMissingTableDefinitionException : WixException + { + /// + /// Instantiate new WixMissingTableDefinitionException. + /// + /// Localized error information. + public WixMissingTableDefinitionException(MessageEventArgs error) + : base(error) + { + } + } +} diff --git a/src/WixToolset.Data/WixToolset.Data.csproj b/src/WixToolset.Data/WixToolset.Data.csproj new file mode 100644 index 00000000..dd14bfae --- /dev/null +++ b/src/WixToolset.Data/WixToolset.Data.csproj @@ -0,0 +1,195 @@ + + + + + + + + netstandard2.0 + WiX Toolset Data + + + + + + + + $(RootNamespace).Data.messages.resources + + + WixToolset.Data.Serialize + + + + + + + + + + diff --git a/src/WixToolset.Data/WixUnexpectedFileFormatException.cs b/src/WixToolset.Data/WixUnexpectedFileFormatException.cs new file mode 100644 index 00000000..4d1e39e9 --- /dev/null +++ b/src/WixToolset.Data/WixUnexpectedFileFormatException.cs @@ -0,0 +1,35 @@ +// 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.Data +{ + using System; + + /// + /// Exception when file does not match the expected format. + /// + public class WixUnexpectedFileFormatException : WixException + { + public WixUnexpectedFileFormatException(string path, FileFormat expectedFormat, FileFormat format, Exception innerException = null) + : base(WixDataErrors.UnexpectedFileFormat(path, expectedFormat.ToString().ToLowerInvariant(), format.ToString().ToLowerInvariant()), innerException) + { + this.Path = path; + this.ExpectedFileFormat = expectedFormat; + this.FileFormat = format; + } + + /// + /// Gets the expected file format. + /// + public FileFormat ExpectedFileFormat { get; private set; } + + /// + /// Gets the actual file format found in the file. + /// + public FileFormat FileFormat { get; private set; } + + /// + /// Gets the path to the file with unexpected format. + /// + public string Path { get; set; } + } +} diff --git a/src/WixToolset.Data/Xsd/wix.xsd b/src/WixToolset.Data/Xsd/wix.xsd new file mode 100644 index 00000000..cff3c219 --- /dev/null +++ b/src/WixToolset.Data/Xsd/wix.xsd @@ -0,0 +1,13036 @@ + + + + + + + + Schema for describing Windows Installer database files (.msi/.msm/.pcp). + + + + + + + + + + This is the top-level container element for every wxs file. Among the possible children, + the Bundle, Product, Module, Patch, and PatchCreation elements are analogous to the main function in a C program. + There can only be one of these present when linking occurs. Product compiles into an msi file, + Module compiles into an msm file, PatchCreation compiles into a pcp file. The Fragment element + is an atomic unit which ultimately links into either a Product, Module, or PatchCreation. The + Fragment can either be completely included or excluded during linking. + + + + + + + + + + + + + + + + + + Required version of the WiX toolset to compile this input file. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional attributes at this point in the schema. + + + + + + + + + This is the top-level container element for every wxi file. + + + + + + + + + + + The root element for creating bundled packages. + + + + + + + + + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + + A URL for more information about the bundle to display in Programs and Features (also + known as Add/Remove Programs). + + + + + + + The legal copyright found in the version resources of final bundle executable. If + this attribute is not provided the copyright will be set to "Copyright (c) [Bundle/@Manufacturer]. All rights reserved.". + + + + + + Whether Packages and Payloads not assigned to a container should be added to the default attached container or if they should be external. The default is yes. + + + + + + Determines whether the bundle can be modified via the Programs and Features (also known as + Add/Remove Programs). If the value is "button" then Programs and Features will show a single + "Uninstall/Change" button. If the value is "yes" then Programs and Features will only show + the "Uninstall" button". If the value is "no", the default, then a "Change" button is shown. + See the DisableRemove attribute for information how to not display the bundle in Programs + and Features. + + + + + + + Determines whether the bundle can be removed via the Programs and Features (also + known as Add/Remove Programs). If the value is "yes" then the "Uninstall" button will + not be displayed. The default is "no" which ensures there is an "Uninstall" button to + remove the bundle. If the "DisableModify" attribute is also "yes" or "button" then the + bundle will not be displayed in Progams and Features and another mechanism (such as + registering as a related bundle addon) must be used to ensure the bundle can be removed. + + + + + + + + + + + + + + A telephone number for help to display in Programs and Features (also known as + Add/Remove Programs). + + + + + + + A URL to the help for the bundle to display in Programs and Features (also known as + Add/Remove Programs). + + + + + + + Path to an icon that will replace the default icon in the final Bundle executable. + This icon will also be displayed in Programs and Features (also known as Add/Remove + Programs). + + + + + + + The publisher of the bundle to display in Programs and Features (also known as + Add/Remove Programs). + + + + + + + The name of the bundle to display in Programs and Features (also known as Add/Remove + Programs). This name can be accessed and overwritten by a BootstrapperApplication + using the WixBundleName bundle variable. + + + + + + + The name of the parent bundle to display in Installed Updates (also known as Add/Remove + Programs). This name is used to nest or group bundles that will appear as updates. + If the parent name does not actually exist, a virtual parent is created automatically. + + + + + + Path to a bitmap that will be shown as the bootstrapper application is being loaded. If this attribute is not specified, no splash screen will be displayed. + + + + + Set this string to uniquely identify this bundle to its own BA, and to related bundles. The value of this string only matters to the BA, and its value has no direct effect on engine functionality. + + + + + + A URL for updates of the bundle to display in Programs and Features (also + known as Add/Remove Programs). + + + + + + + Unique identifier for a family of bundles. If two bundles have the same UpgradeCode the + bundle with the highest version will be installed. + + + + + + + The version of the bundle. Newer versions upgrade earlier versions of the bundles + with matching UpgradeCodes. If the bundle is registered in Programs and Features + then this attribute will be displayed in the Programs and Features user interface. + + + + + + + The condition of the bundle. If the condition is not met, the bundle will + refuse to run. Conditions are checked before the bootstrapper application is loaded + (before detect), and thus can only reference built-in variables such as + variables which indicate the version of the OS. + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + Provides information about an .exe so that the BA can request the engine to run it elevated from any secure location. + + + + + + + + The identifier of the ApprovedExeForElevation element. + + + + + + The key path. + For security purposes, the root key will be HKLM and Variables are not supported. + + + + + + + The value name. + For security purposes, Variables are not supported. + + + + + + + Instructs the search to look in the 64-bit registry when the value is 'yes'. + When the value is 'no', the search looks in the 32-bit registry. + The default value is 'no'. + + + + + + + + Overrides the default log settings for a bundle. + + + + + + + + + Disables the default logging in the Bundle. The end user can still generate a + log file by specifying the "-l" command-line argument when installing the + Bundle. + + + + + + + Name of a Variable that will hold the path to the log file. An empty value + will cause the variable to not be set. The default is "WixBundleLog". + + + + + + + File name and optionally a relative path to use as the prefix for the log file. The + default is to use the Bundle/@Name or, if Bundle/@Name is not specified, the value + "Setup". + + + + + + The extension to use for the log. The default is ".log". + + + + + + + Specify one or more catalog files that will be used to verify the contents of the bundle. + + + + + + + + The identifier of the catalog element. + + + + + The catalog file + + + + + + + Contains all the relevant information about the setup UI. + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + The identifier of the BootstrapperApplication element. Only required if you want to reference this element using a BootstrapperApplicationRef element. + + + + + The DLL with the bootstrapper application entry function. + + + + + The relative destination path and file name for the bootstrapper application DLL. The default is the source file name. Use this attribute to rename the bootstrapper application DLL or extract it into a subfolder. The use of '..' directories is not allowed. + + + + + + + Used to reference a BootstrapperApplication element and optionally add additional payloads to the bootstrapper application. + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + The identifier of the BootstrapperApplication element to reference. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + This element has been deprecated. Use the BootstrapperApplication element instead. + + + + + + + + + + + + + + See the BootstrapperApplication instead. + + + + + See the BootstrapperApplication instead. + + + + + See the BootstrapperApplication instead. + + + + + + + Writes additional information to the Windows registry that can be used to detect the bundle. + This registration is intended primarily for update to an existing product. + + + The attributes are used to write the following registry values to the key: + SOFTWARE\[Manufacturer]\Updates\[ProductFamily]\[Name] + + ThisVersionInstalled: Y + PackageName: >bundle name< + PackageVersion: >bundle version< + Publisher: [Manufacturer] + PublishingGroup: [Department] + ReleaseType: [Classification] + InstalledBy: [LogonUser] + InstalledDate: [Date] + InstallerName: >installer name< + InstallerVersion: >installer version< + + + + + + + + The name of the manufacturer. The default is the Bundle/@Manufacturer attribute, + but may also be a short form, ex: Acme instead of Acme Corporation. + An error is generated at build time if neither attribute is specified. + + + + + The name of the department or division publishing the update bundle. + The PublishingGroup registry value is not written if this attribute is not specified. + + + + + The name of the family of products being updated. The default is the Bundle/@ParentName attribute. + The corresponding registry key is not created if neither attribute is specified. + + + + + The name of the bundle. The default is the Bundle/@Name attribute, + but may also be a short form, ex: KB12345 instead of Update to Product (KB12345). + An error is generated at build time if neither attribute is specified. + + + + + The release type of the update bundle, such as Update, Security Update, Service Pack, etc. + The default value is Update. + + + + + + + Contains the chain of packages to install. + + + + + + + + + + + + + + + + + Specifies whether the bundle will attempt to rollback packages + executed in the chain. If "yes" is specified then when a vital + package fails to install only that package will rollback and the + chain will stop with the error. The default is "no" which + indicates all packages executed during the chain will be + rolledback to their previous state when a vital package fails. + + + + + + + Specifies whether the bundle will attempt to create a system + restore point when executing the chain. If "yes" is specified then + a system restore point will not be created. The default is "no" which + indicates a system restore point will be created when the bundle is + installed, uninstalled, repaired, modified, etc. If the system restore + point cannot be created, the bundle will log the issue and continue. + + + + + + + Specifies whether the bundle will start installing packages + while other packages are still being cached. If "yes", + packages will start executing when a rollback boundary is + encountered. The default is "no" which dictates all packages + must be cached before any packages will start to be installed. + + + + + + + + Describes a single msi package to install. + + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. The extension's + CompilerExtension.ParseElement() + method will be called with the package identifier as the first value in + contextValues. + + + + + + + + + Specifies whether the bundle will show the UI authored into the msi package. The default is "no" + which means all information is routed to the bootstrapper application to provide a unified installation + experience. If "yes" is specified the UI authored into the msi package will be displayed on top of + any bootstrapper application UI. + + + + + + + Specifies whether the bundle will allow individual control over the installation state of Features inside + the msi package. Managing feature selection requires special care to ensure the install, modify, update and + uninstall behavior of the package is always correct. The default is "no". + + + + + + + Override the automatic per-machine detection of MSI packages and force the package to be per-machine. + The default is "no", which allows the tools to detect the expected value. + + + + + + + This attribute has been deprecated. When the value is "yes", the Binder will not read the MSI package + to detect uncompressed files that would otherwise be automatically included in the Bundle as Payloads. + The resulting Bundle may not be able to install the MSI package correctly. The default is "no". + + + + + + + Specifies whether the MSI will be displayed in Programs and Features (also known as Add/Remove Programs). If "yes" is + specified the MSI package information will be displayed in Programs and Features. The default "no" indicates the MSI + will not be displayed. + + + + + + + + Describes a single msp package to install. + + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. The extension's + CompilerExtension.ParseElement() + method will be called with the package identifier as the first value in + contextValues. + + + + + + + + + Specifies whether the bundle will show the UI authored into the msp package. The default is "no" + which means all information is routed to the bootstrapper application to provide a unified installation + experience. If "yes" is specified the UI authored into the msp package will be displayed on top of + any bootstrapper application UI. + + + + + + Indicates the package must be executed elevated. The default is "no". + + + + + + Specifies whether to automatically slipstream the patch for any target msi packages in the chain. The default is "no". + Even when the value is "no", you can still author the SlipstreamMsp element under MsiPackage elements as desired. + + + + + + + + Describes a single msu package to install. + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. The extension's + CompilerExtension.ParseElement() + method will be called with the package identifier as the first value in + contextValues. + + + + + + + + + A condition that determines if the package is present on the target system. This condition can use built-in + variables and variables returned by searches. This condition is necessary because Windows doesn't provide a + method to detect the presence of an MsuPackage. Burn uses this condition to determine how to treat this + package during a bundle action; for example, if this condition is false or omitted and the bundle is being + installed, Burn will install this package. + + + + + + + The knowledge base identifier for the MSU. The KB attribute must be specified to enable the MSU package to + be uninstalled. Even then MSU uninstallation is only supported on Windows 7 and later. When the KB attribute + is specified, the Permanent attribute will the control whether the package is uninstalled. + + + + + + + + Describes a single exe package to install. + + + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. The extension's + CompilerExtension.ParseElement() + method will be called with the package identifier as the first value in + contextValues. + + + + + + + + + A condition that determines if the package is present on the target system. This condition can use built-in + variables and variables returned by searches. This condition is necessary because Windows doesn't provide a + method to detect the presence of an ExePackage. Burn uses this condition to determine how to treat this + package during a bundle action; for example, if this condition is false or omitted and the bundle is being + installed, Burn will install this package. + + + + + + The command-line arguments provided to the ExePackage during install. If this attribute is absent the executable will be launched with no command-line arguments. + + + + + + The command-line arguments to specify to indicate a repair. If the executable package can be repaired but + does not require any special command-line arguments to do so then set the attribute's value to blank. To + indicate that the package does not support repair, omit this attribute. + + + + + + The command-line arguments provided to the ExePackage during uninstall. If this attribute is absent the executable will be launched with no command-line arguments. To prevent an ExePackage from being uninstalled set the Permanent attribute to "yes". + + + + + Indicates the package must be executed elevated. The default is "no". + + + + + Indicates the communication protocol the package supports for extended progress and error reporting. The default is "none". + + + + + + + Describes a rollback boundary in the chain. + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. The extension's + CompilerExtension.ParseElement() + method will be called with the rollback boundary identifier as the 'RollbackBoundaryId' key in + contextValues. + + + + + + + + Identifier for this rollback boundary, for ordering and cross-referencing. If this attribute is + not provided a stable identifier will be generated. + + + + + + + Specifies whether the rollback boundary aborts the chain. The default "yes" indicates that if + the rollback boundary is encountered then the chain will fail and rollback or stop. If "no" + is specified then the chain should continue successfuly at the next rollback boundary. + + + + + + + Specifies whether the rollback boundary is wrapped in an MSI transaction. The default is "no" + + + + + + + + + + Location of the package to add to the bundle. The default value is the Name attribute, if provided. + At a minimum, the SourceFile or Name attribute must be specified. + + + + + + + The destination path and file name for this chain payload. Use this attribute to rename the + chain entry point or extract it into a subfolder. The default value is the file name from the + SourceFile attribute, if provided. At a minimum, the Name or SourceFile attribute must be specified. + The use of '..' directories is not allowed. + + + + + + + The URL to use to download the package. The following substitutions are supported: + + {0} is replaced by the package Id. + {1} is replaced by the payload Id. + {2} is replaced by the payload file name. + + + + + + + + Identifier for this package, for ordering and cross-referencing. The default is the Name attribute + modified to be suitable as an identifier (i.e. invalid characters are replaced with underscores). + + + + + + + The identifier of another package that this one should be installed after. By default the After + attribute is set to the previous sibling package in the Chain or PackageGroup element. If this + attribute is specified ensure that a cycle is not created explicitly or implicitly. + + + + + + + The size this package will take on disk in bytes after it is installed. By default, the binder will + calculate the install size by scanning the package (File table for MSIs, Payloads for EXEs) + and use the total for the install size of the package. + + + + + + A condition to evaluate before installing the package. The package will only be installed if the condition evaluates to true. If the condition evaluates to false and the bundle is being installed, repaired, or modified, the package will be uninstalled. + + + + + Whether to cache the package. The default is "yes". + + + + + The identifier to use when caching the package. + + + + + + Specifies the display name to place in the bootstrapper application data manifest for the package. By default, ExePackages + use the ProductName field from the version information, MsiPackages use the ProductName property, and MspPackages use + the DisplayName patch metadata property. Other package types must use this attribute to define a display name in the + bootstrapper application data manifest. + + + + + + + Specifies the description to place in the bootstrapper application data manifest for the package. By default, ExePackages + use the FileName field from the version information, MsiPackages use the ARPCOMMENTS property, and MspPackages use + the Description patch metadata property. Other package types must use this attribute to define a description in the + bootstrapper application data manifest. + + + + + + + Name of a Variable that will hold the path to the log file. An empty value will cause the variable to not + be set. The default is "WixBundleLog_[PackageId]" except for MSU packages which default to no logging. + + + + + + + Name of a Variable that will hold the path to the log file used during rollback. An empty value will cause + the variable to not be set. The default is "WixBundleRollbackLog_[PackageId]" except for MSU packages which + default to no logging. + + + + + + + Specifies whether the package can be uninstalled. The default is "no". + + + + + + + Specifies whether the package must succeed for the chain to continue. The default "yes" + indicates that if the package fails then the chain will fail and rollback or stop. If + "no" is specified then the chain will continue even if the package reports failure. + + + + + + Whether the package payload should be embedded in a container or left as an external payload. + + + + + + By default, a Bundle will use the hash of a package to verify its contents. If this attribute is set to "yes" + and the package is signed with an Authenticode signature the Bundle will verify the contents of the package using the + signature instead. Beware that there are many real world issues with Windows verifying Authenticode signatures. + Since the Authenticode signatures are no more secure than hashing the packages directly, the default is "no". + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. The extension's + CompilerExtension.ParseAttribute() + method will be called with the package identifier in + contextValues["PackageId"]. + + + + + + + Describes a package group to a bootstrapper. + + + + + + + + + + + + + + + + Identifier for package group. + + + + + + + Create a reference to PackageGroup element that exists inside a Bundle or Fragment element. + + + + + + + + + + + The identifier of the PackageGroup element to reference. + + + + + The identifier of a package that this group should be installed after. + + + + + + + Allows an MSI property to be set based on the value of a burn engine expression. + + + + + + + + The name of the MSI property to set. Burn controls the follow MSI properties so they cannot be set with MsiProperty: ACTION, ALLUSERS, REBOOT, REINSTALL, REINSTALLMODE + + + + + The value to set the property to. This string is evaluated by the burn engine and can be as simple as a burn engine variable reference or as complex as a full expression. + + + + + + + Specifies a patch included in the same bundle that is installed when the parent MSI package is installed. + + + + + You can also specify that any MspPackage elements in the chain are automatically slipstreamed by setting the Slipstream attribute of an MspPackage to "yes". This will reduce the amount of authoring you need to write and will determine which msi packages can slipstream patches when building a bundle. + + + + + + + The identifier for a MspPackage in the bundle. + + + + + + + Describes a burn engine variable to define. + + + + + + + + + Whether the value of the variable should be hidden. + + + + + The name for the variable. + + + + + Whether the variable should be persisted. + + + + + Starting value for the variable. + + + + + Type of the variable, inferred from the value if not specified. + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + Representation of a file that contains one or more files. + + + + + + + + + + + + The URL to use to download the container. This attribute is only valid when the container is detached. The + following substitutions are supported: + + {0} is always null. + {1} is replaced by the container Id. + {2} is replaced by the container file name. + + + + + + + The unique identifier for the container. If this attribute is not specified the Name attribute will be used. + + + + + The file name for this container. A relative path may be provided to place the container in a sub-folder of the bundle. + + + + + + Indicates whether the container is "attached" to the bundle executable or placed external to the bundle extecutable as "detached". If + this attribute is not specified, the default is to create a detached container. + + + + + + + + Create a reference to an existing Container element. + + + + + + + + + The identifier of Container element to reference. + + + + + + + Describes map of exit code returned from executable package to a bootstrapper behavior. + + + + + + + + Exit code returned from executable package. If no value is provided it means all values not explicitly set default to this behavior. + + + + + Choose one of the supported behaviors error codes: success, error, scheduleReboot, forceReboot. + + + + + + + + + + + + + + + Describes additional, conditional command-line arguments for an ExePackage. + + + + + Additional command-line arguments to apply during package installation if Condition is true. + + + + + Additional command-line arguments to apply during package uninstallation if Condition is true. + + + + + Additional command-line arguments to apply during package repair if Condition is true. + + + + + + The condition that controls whether the command-line arguments specified in the + InstallArgument, UninstallArgument, or RepairArgument attributes are appended to the + command line passed to the ExePackage. Which attribute is used depends on the + action being applied to the ExePackage. For example, when the ExePackage is + being installed, the InstallArgument attribute value is appended to the command + line when the ExePackage is executed. + + + + + + + + Describes a payload to a bootstrapper. + + + + + + + + + + + + + The identifier of Payload element. + + + + + Whether the payload should be embedded in a container or left as an external payload. + + + + + Location of the source file. + + + + + The destination path and file name for this payload. The default is the source file name. The use of '..' directories is not allowed. + + + + + + The URL to use to download the package. The following substitutions are supported: + + {0} is replaced by the package Id. + {1} is replaced by the payload Id. + {2} is replaced by the payload file name. + + + + + + + + By default, a Bundle will use the hash of a package to verify its contents. If this attribute is set to "yes" + and the package is signed with an Authenticode signature the Bundle will verify the contents of the package using the + signature instead. Beware that there are many real world issues with Windows verifying Authenticode signatures. + Since the Authenticode signatures are no more secure than hashing the packages directly, the default is "no". + + + + + + + + Describes a payload group to a bootstrapper. PayloadGroups referenced from within a Bundle are tied to the Bundle. + PayloadGroups referenced from a Fragment are tied to the context of whatever references them such as an ExePackage or MsiPackage. + It is possible to share a PayloadGroup between multiple Packages and/or a Bundle by creating multiple references to it. + + + + + + + + + + + + Identifier for payload group. + + + + + + + Create a reference to PayloadGroup element that exists inside a Bundle or Fragment element. + + + + + + + + + + + The identifier of the PayloadGroup element to reference. + + + + + + + Describes information about a remote file payload that is not available at the time of building the bundle. + The parent must specify DownloadUrl and must not specify SourceFile when using this element. + + + + + + + + + Public key of the authenticode certificate used to sign the RemotePayload. Include this attribute if the remote file is signed. + + + + + Thumbprint of the authenticode certificate used to sign the RemotePayload. Include this attribute if the remote file is signed. + + + + + Description of the file from version resources. + + + + + SHA-1 hash of the RemotePayload. Include this attribute if the remote file is unsigned or SuppressSignatureVerification is set to Yes. + + + + + Product name of the file from version resouces. + + + + + Size of the remote file in bytes. + + + + + Version of the remote file + + + + + + + Create a RelatedBundle element. + + + + + + + + + The identifier of the RelatedBundle group. + + + + + The action to take on bundles related to this one. Detect is the default. + + + + + + + + + + + + + + + Defines the update for a Bundle. + + + + + + + + + + The absolute path or URL to check for an update bundle. Currently the engine provides this value + in the IBootstrapperApplication::OnDetectUpdateBegin() and otherwise ignores the value. In the + future the engine will be able to acquire an update bundle from the location and determine if it + is newer than the current executing bundle. + + + + + + + + + The Product element is analogous to the main function in a C program. When linking, only one Product section + can be given to the linker to produce a successful result. Using this element creates an msi file. + + + + You can specify any valid Windows code page by integer like 1252, or by web name like Windows-1252. See Code Pages for more information. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + + The product code GUID for the product. + + + + + The code page integer value or web name for the resulting MSI. See remarks for more information. + + + + + The decimal language ID (LCID) for the product. + + + + + The manufacturer of the product. + + + + + The descriptive name of the product. + + + + + The upgrade code GUID for the product. + + + + + The product's version string. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + The Module element is analogous to the main function in a C program. When linking, only + one Module section can be given to the linker to produce a successful result. Using this + element creates an msm file. + + + + You can specify any valid Windows code by by integer like 1252, or by web name like Windows-1252. See Code Pages for more information. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + + The name of the merge module (not the file name). + + + + + The code page integer value or web name for the resulting MSM. See remarks for more information. + + + + + This attribute is deprecated. Use the Package/@Id attribute instead. + + + + + The decimal language ID (LCID) of the merge module. + + + + + The major and minor versions of the merge module. + + + + + + + Declares a dependency on another merge module. + + + + + Identifier of the merge module required by the merge module. + + + + + Numeric language ID of the merge module in RequiredID. + + + + + Version of the merge module in RequiredID. + + + + + + + Declares a merge module with which this merge module is incompatible. + + + + + Identifier of the merge module that is incompatible. + + + + + Numeric language ID of the merge module in ExcludedID. All except this language will be excluded. Only one of ExcludeExceptLanguage and ExcludeLanguage may be specified. + + + + + Numeric language ID of the merge module in ExcludedID. The specified language will be excluded. Only one of ExcludeExceptLanguage and ExcludeLanguage may be specified. + + + + + Minimum version excluded from a range. If not set, all versions before max are excluded. If neither max nor min, no exclusion based on version. + + + + + Maximum version excluded from a range. If not set, all versions after min are excluded. If neither max nor min, no exclusion based on version. + + + + + + + Defines the configurable attributes of merge module. + + + + + Defines the name of the configurable item. + + + + + Specifies the format of the data being changed. + + + + + + + + + + + + + Specifies the type of the data being changed. + + + + + Specifies a semantic context for the requested data. + + + + + Specifies a default value for the item in this record if the merge tool declines to provide a value. + + + + + Does not merge rule according to rules in MSI SDK. + + + + + If yes, null is not a valid entry. + + + + + Display name for authoring. + + + + + Description for authoring. + + + + + Location of chm file for authoring. + + + + + Keyword into chm file for authoring. + + + + + + + Specifies the configurable fields of a module database and provides a template for the configuration of each field. + + + + + Specifies the name of the table being modified in the module database. + + + + + Specifies the primary keys of the target row in the table named in the Table column. If multiple keys, separated by semicolons. + + + + + Specifies the target column in the row named in the Row column. + + + + + Provides a formatting template for the data being substituted into the target field specified by Table, Row, and Column. + + + + + + + + Specifies a table from the merge module that is not merged into an .msi file. + If the table already exists in an .msi file, it is not modified by the merge. + The specified table can therefore contain data that is unneeded after the merge. + To minimize the size of the .msm file, it is recommended that developers remove + unused tables from modules intended for redistribution rather than creating + IgnoreTable elements for those tables. + + + + + + + The name of the table in the merge module that is not to be merged into the .msi file. + + + + + + + + + The Fragment element is the building block of creating an installer database in WiX. Once defined, + the Fragment becomes an immutable, atomic unit which can either be completely included or excluded + from a product. The contents of a Fragment element can be linked into a product by utilizing one + of the many *Ref elements. When linking in a Fragment, it will be necessary to link in all of its + individual units. For instance, if a given Fragment contains two Component elements, you must link + both under features using ComponentRef for each linked Component. Otherwise, you will get a linker + warning and have a floating Component that does not appear under any Feature. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + + Optional identifier for a Fragment. Should only be set by advanced users to tag sections. + + + + + + + + + The Patch element is analogous to the main function in a C program. When linking, only one Patch section + can be given to the linker to produce a successful result. Using this element creates an MSP file. + + + + You can specify any valid Windows code by by integer like 1252, or by web name like Windows-1252. See Code Pages for more information. + The ClientPatchId attribute allows you to specify an easily referenced identity that you can use in product authoring. This identity prefixes properties added by WiX to a patch transform, such as ClientPatchId.PatchCode and ClientPatchId.AllowRemoval. If the patch code GUID is auto-generated you could not reference any properties using this auto-generated prefix. + For example, if you were planning to ship a patch referred to as "QFE1" and needed to write your own registry values for Add/Remove Programs in product authoring such as the UninstallString for this patch, you could author a RegistryValue with the name UninstallString and the value [SystemFolder]msiexec.exe /package [ProductCode] /uninstall [QFE1.PatchCode]. In your patch authoring you would then set ClientPatchId to "QFE1" and WiX will add the QFE1.PatchCode property to the patch transform when the patch is created. If the Id attribute specified the patch code to be generated automatically, you could not reference the prefix.PatchCode property as shown above. + The summary information is automatically populated from attribute values of the Patch element including the code page. If you want to override some of these summary information properties or use a different code page for the summary information itself, author the PatchInformation element. + + + + + + + + + Optional element that allows overriding summary information properties. + + + + + + Indicates whether custom actions can be skipped when applying the patch. + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + + Patch code for this patch. + + + + + The code page integer value or web name for the resulting MSP. See remarks for more information. + + + + + Whether this is an uninstallable patch. + + + + + Category of updates. Recommended values are Critical Update, Hotfix, Security Rollup, Security Update, Service Pack, Update, Update Rollup. + + + + + An easily referenced identity unique to a patch that can be used in product authoring. See remarks for more information. + + + + + Flag used when creating a binary file patch. Default is "no". Don't use imagehlp.dll. + + + + + Flag used when creating a binary file patch. Default is "no". Don't fail patch due to imagehlp failures. + + + + + Flag used when creating a binary file patch. Default is "no". After matching decorated symbols, try to match remaining by undecorated names. + + + + + Description of the patch. + + + + + A title for the patch that is suitable for public display. In Add/Remove Programs from XP SP2 on. + + + + + Optional comments for browsing. + + + + + Vendor releasing the package + + + + + + Indicates that the patch targets the RTM version of the product or the most recent major + upgrade patch. Author this optional property in minor update patches that contain sequencing + information to indicate that the patch removes all patches up to the RTM version of the + product, or up to the most recent major upgrade patch. This property is available beginning + with Windows Installer 3.1. + + + + + + A URL that provides information specific to this patch. In Add/Remove Programs from XP SP2 on. + + + + + + If this attribute is set to 'yes' in all the patches to be applied in a transaction, the + application of the patch is optimized if possible. Available beginning with Windows Installer 3.1. + + + + + + Name of the application or target product suite. + + + + + When this attribute is set, patches for files greater than approximately 4 MB in size may be made smaller. + + + + + + + Sets information in the patch transform that determines if the transform applies to an installed product and what errors should be ignored when applying the patch transform. + + + A transform contains the differences between the target product and the upgraded product. When a transform or a patch (which contains transforms) is applied, the following properties of the installed product are validated against the properties of the target product stored in a transform. + + ProductCode + ProductLanguage + ProductVersion + UpgradeCode + + Windows Installer simply validates that the ProductCode, ProductLanguage, and UpgradeCode of an installed product are equivalent to those propeties of the target product used to create the transform; however, the ProductVersion can be validated with a greater range of comparisons. + You can compare up to the first three fields of the ProductVersion. Changes to the fourth field are not validated and are useful for small updates. You can also choose how to compare the target ProductVersion used to create the transform with the installed ProductVersion. For example, while the default value of 'Equals' is recommended, if you wanted a minor upgrade patch to apply to the target ProductVersion and all older products with the same ProductCode, you would use 'LesserOrEqual'. + + + + + + + Requires that the installed ProductCode match the target ProductCode used to create the transform. The default is 'yes'. + + + + + Requires that the installed ProductLanguage match the target ProductLanguage used to create the transform. The default is 'no'. + + + + + Determines how many fields of the installed ProductVersion to compare. See remarks for more information. The default is 'Update'. + + + + + + Checks the major version. + + + + + Checks the major and minor versions. + + + + + Checks the major, minor, and update versions. + + + + + + + + Determines how the installed ProductVersion is compared to the target ProductVersion used to create the transform. See remarks for more information. The default is 'Equal'. + + + + + + Installed ProductVersion < target ProductVersion. + + + + + Installed ProductVersion <= target ProductVersion. + + + + + Installed ProductVersion = target ProductVersion. + + + + + Installed ProductVersion >= target ProductVersion. + + + + + Installed ProductVersion > target ProductVersion. + + + + + + + + Requires that the installed UpgradeCode match the target UpgradeCode used to create the transform. The default is 'yes'. + + + + + Ignore errors when adding existing rows. The default is 'yes'. + + + + + Ignore errors when adding existing tables. The default is 'yes'. + + + + + Ignore errors when deleting missing rows. The default is 'yes'. + + + + + Ignore errors when deleting missing tables. The default is 'yes'. + + + + + Ignore errors when updating missing rows. The default is 'yes'. + + + + + Ignore errors when changing the database code page. The default is 'no'. + + + + + + + Indicates whether custom actions can be skipped when applying the patch. + + + + + + + + Skip property (type 51) and directory (type 35) assignment custom actions. + + + + + Skip immediate custom actions that are not property or directory assignment custom actions. + + + + + Skip custom actions that run within the script. + + + + + + + Identifies a set of product versions. + + + + + + + + Identifier for a set of product versions. + + + + + + + Collection of items that should be kept from the differences between two products. + + + + + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + + Identifier which indicates a sequence family to which this patch belongs. + + + + + + Specifies the ProductCode of the product that this family applies to. + + + + + + Used to populate the sequence column of the MsiPatchSequence table in the final MSP file. Specified in x.x.x.x format. See documentation for Sequence column of MsiPatchSequence table in MSI SDK. + + + + + + Set this value to 'yes' to indicate that this patch will supersede all previous patches in this patch family. + The default value is 'no'. + + + + + + + + + Groups together multiple patch families to be used in other locations. + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + Identifier for the PatchFamilyGroup. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + Create a reference to a PatchFamilyGroup in another Fragment. + + + + + + + + The identifier of the PatchFamilyGroup to reference. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + The PatchCreation element is analogous to the main function in a C program. When linking, only one PatchCreation section + can be given to the linker to produce a successful result. Using this element creates a pcp file. + + + + You can specify any valid Windows code by by integer like 1252, or by web name like Windows-1252. See Code Pages for more information. + + + + + + + + + + + + + + + + + + PatchCreation identifier; this is the primary key for identifying patches. + + + + + Use this to set whether the major versions between the upgrade and target images match. See AllowProductVersionMajorMismatches for more information. + + + + + Use this to set whether the product code between the upgrade and target images match. See AllowProductCodeMismatches for more information. + + + + + Use this to set whether Patchwiz should clean the temp folder when finished. See DontRemoveTempFolderWhenFinished for more information. + + + + + The code page integer value or web name for the resulting PCP. See remarks for more information. + + + + + The full path, including file name, of the patch package file that is to be generated. See PatchOutputPath for more information. + + + + + Used to locate the .msp file for the patch if the cached copy is unavailable. See PatchSourceList for more information. + + + + + An 8-digit hex integer representing the combination of patch symbol usage flags to use when creating a binary file patch. See ApiPatchingSymbolFlags for more information. + + + + + Use this to set whether changing files should be included in their entirety. See IncludeWholeFilesOnly for more information. + + + + + + + Properties about the patch to be placed in the Summary Information Stream. These are visible from COM through the IStream interface, and these properties can be seen on the package in Explorer. + + + You can specify any valid Windows code by by integer like 1252, or by web name like Windows-1252. See Code Pages for more information. + + + + + + + A short description of the patch that includes the name of the product. + + + + + + + + + + + + + + + + + + + The name of the manufacturer of the patch package. + + + + + A semicolon-delimited list of network or URL locations for alternate sources of the patch. The default is "Installer,Patching,PCP,Database". + + + + + General purpose of the patch package. For example, "This patch contains the logic and data required to install <product>." + + + + + + The value of this attribute conveys whether the package should be opened as read-only. + A database editing tool should not modify a read-only enforced database and should + issue a warning at attempts to modify a read-only recommended database. + + + + + + The code page integer value or web name for summary info strings only. The default is 1252. See remarks for more information. + + + + + + + + + + + + + + + + + + + + + + + + + + + + Properties about the patch to be placed in the PatchMetadata table. + + + + + + + + + + A custom property that extends the standard set. + + + + + Indicates whether custom actions can be skipped when applying the patch. + + + + + + + Whether this is an uninstallable patch. + + + + + Category of updates. Recommended values are Critical Update, Hotfix, Security Rollup, Security Update, Service Pack, Update, Update Rollup. + + + + + Creation time of the .msp file in the form mm-dd-yy HH:MM (month-day-year hour:minute). + + + + + Description of the patch. + + + + + A title for the patch that is suitable for public display. In Add/Remove Programs from XP SP2 on. + + + + + Name of the manufacturer. + + + + + + Indicates that the patch targets the RTM version of the product or the most recent major + upgrade patch. Author this optional property in minor update patches that contain sequencing + information to indicate that the patch removes all patches up to the RTM version of the + product, or up to the most recent major upgrade patch. This property is available beginning + with Windows Installer 3.1. + + + + + + A URL that provides information specific to this patch. In Add/Remove Programs from XP SP2 on. + + + + + + If this attribute is set to 'yes' in all the patches to be applied in a transaction, the + application of the patch is optimized if possible. Available beginning with Windows Installer 3.1. + + + + + + Name of the application or target product suite. + + + + + + + A custom property for the PatchMetadata table. + + + + + The name of the company. + + + + + The name of the metadata property. + + + + + Value of the metadata property. + + + + + + + A patch that is deprecated by this patch. + + + + + Patch GUID to be unregistered if it exists on the machine targeted by this patch. + + + + + + + + The product codes for products that can accept the patch. + + + + + + + + + Whether to replace the product codes that can accept the patch from the target packages with the child elements. + + + + + + + + A product code for a product that can accept the patch. + + + + When using the PatchCreation element, if the Id attribute value is '*' or this element is not authored, the product codes of all products referenced by the TargetImages element are used. + When using the Patch element, the Id attribute value must not be '*'. Use the TargetProductCodes/@Replace attribute instead. + + + + + + + + The product code for a product that can accept the patch. This can be '*'. See remarks for more information. + + + + + + + + A property for this patch database. + + + + When authored under the Patch element, the PatchProperty defines entries in the MsiPatchMetadata table. + + + + + + + Name of the company for a custom metadata property. + + + + + Name of the patch property. + + + + + Value of the patch property. + + + + + + + Sequence information for this patch database. Sequence information is generated automatically in most cases, and rarely needs to be set explicitly. + + + + + + + + Identifier which indicates a sequence family to which this patch belongs. + + + + + + Specifies the ProductCode of the product that this family applies to. + This attribute cannot the specified if the TargetImage attribute is specified. + + + + + + Used to populate the sequence column of the MsiPatchSequence table in the final MSP file. Specified in x.x.x.x format. See documentation for Sequence column of MsiPatchSequence table in MSI SDK. + + + + + + Set this value to 'yes' to indicate that this patch will supersede all previous patches in this patch family. + The default value is 'no'. + + + + + + + + + + + + + + Specifies the TargetImage that this family applies to. + This attribute cannot the specified if the ProductCode attribute is specified. + + + + + + + + Group of one or more upgraded images of a product. + + + + + + + + + + + + Entered into the DiskId field of the new Media table record. + + + + + Value to display in the "[1]" of the DiskPrompt Property. Using this attribute will require you to define a DiskPrompt Property. + + + + + Entered into the Source field of the new Media table entry of the upgraded image. + + + + + Identifier for the family. + + + + + Sequence number for the starting file. + + + + + Entered into the VolumeLabel field of the new Media table record. + + + + + + + Contains information about the upgraded images of the product. + + + + + + + + + + + + Identifier to connect target images with upgraded image. + + + + + Full path to location of msi file for upgraded image. + + + + + + + + + + + + Modified copy of the upgraded installation database that contains additional authoring specific to patching. + + + + + + + + + + + + + + Contains information about the target images of the product. + + + + + + + + + Identifier for the target image. + + + + + Full path to the location of the msi file for the target image. + + + + + + + + + + + + Relative order of the target image. + + + + + Product checking to avoid applying irrelevant transforms. + + + + + Files missing from the target image are ignored by the installer. + + + + + + + Information about specific files in a target image. + + + + + + + + + + + + Foreign key into the File table. + + + + + + + Specifies part of a file that is to be ignored during patching. + + + + + Offset of the start of the range. + + + + + Length of the range. + + + + + + + Specifies part of a file that cannot be overwritten during patching. + + + + + Offset of the start of the range. + + + + + Length of the range. + + + + + + + Specifies a file to be protected. + + + + + + + + Foreign key into the File table. + + + + + + + Contains information about specific files that are not part of a regular target image. + + + + + + + + + + + + Foreign key into the File table. + + + + + Full path of the external file. + + + + + + + + + + + + Specifies the order of the external files to use when creating the patch. + + + + + + + Specifies files to either ignore or to specify optional data about a file. + + + + + + + + Foreign key into the File table. + + + + + If yes, the file is ignored during patching, and the next two attributes are ignored. + + + + + Specifies whether patching this file is vital. + + + + + Whether the whole file should be installed, rather than creating a binary patch. + + + + + + + A path to symbols. + + + + + The path. + + + + + + + + Properties about the package to be placed in the Summary Information Stream. These are + visible from COM through the IStream interface, and these properties can be seen on the package in Explorer. + + + + You can specify any valid Windows code by by integer like 1252, or by web name like Windows-1252. See Code Pages for more information. + + + + + + + + The package code GUID for a product or merge module. + When compiling a product, this attribute should not be set in order to allow the package + code to be generated for each build. + When compiling a merge module, this attribute must be set to the modularization guid. + + + + + + Set to 'yes' if the source is an admin image. + + + + + Optional comments for browsing. + + + + + + Set to 'yes' to have compressed files in the source. + This attribute cannot be set for merge modules. + + + + + + The product full name or description. + + + + + Use this attribute to specify the priviliges required to install the package on Windows Vista and above. + + + + + + + Set this value to declare that the package does not require elevated privileges to install. + + + + + + + Set this value to declare that the package requires elevated privileges to install. + This is the default value. + + + + + + + + + Use this attribute to specify the installation scope of this package: per-machine or per-user. + + + + + + + Set this value to declare that the package is a per-machine installation and requires elevated privileges to install. + Sets the ALLUSERS property to 1. + + + + + + + Set this value to declare that the package is a per-user installation and does not require elevated privileges to install. + Sets the package's InstallPrivileges attribute to "limited." + + + + + + + + + + The minimum version of the Windows Installer required to install this package. Take the major version of the required Windows Installer + and multiply by a 100 then add the minor version of the Windows Installer. For example, "200" would represent Windows Installer 2.0 and + "405" would represent Windows Installer 4.5. For 64-bit Windows Installer packages, this property is set to 200 by default as + Windows Installer 2.0 was the first version to support 64-bit packages. + + + + + + Optional keywords for browsing. + + + + + The list of language IDs (LCIDs) supported in the package. + + + + + The vendor releasing the package. + + + + + + The list of platforms supported by the package. This attribute has been deprecated. + Specify the -arch switch at the candle.exe command line or the InstallerPlatform + property in a .wixproj MSBuild project. + + + + + + + The platform supported by the package. Use of this attribute is discouraged; instead, + specify the -arch switch at the candle.exe command line or the InstallerPlatform + property in a .wixproj MSBuild project. + + + + + + + + Set this value to declare that the package is an x86 package. + + + + + + + Set this value to declare that the package is an ia64 package. + This value requires that the InstallerVersion property be set to 200 or greater. + + + + + + + Set this value to declare that the package is an x64 package. + This value requires that the InstallerVersion property be set to 200 or greater. + + + + + + + Set this value to declare that the package is an arm package. + This value requires that the InstallerVersion property be set to 500 or greater. + + + + + + + This value has been deprecated. Use "x86" instead. + + + + + + + This value has been deprecated. Use "ia64" instead. + + + + + + + + + + The value of this attribute conveys whether the package should be opened as read-only. + A database editing tool should not modify a read-only enforced database and should + issue a warning at attempts to modify a read-only recommended database. + + + + + + Set to 'yes' to have short filenames in the source. + + + + + The code page integer value or web name for summary info strings only. See remarks for more information. + + + + + + + + The MsiAssemblyName table specifies the schema for the elements of a strong assembly cache name for a .NET Framework or Win32 assembly. + Consider using the Assembly attribute on File element to have the toolset populate these entries automatically. + + + + + + + + + Name of the attribute associated with the value specified in the Value column. + + + + + Value associated with the name specified in the Name column. + + + + + + + + Identifies the possible signer certificates used to digitally sign patches. + + + + + + + + + + + + + + + Digital signatures that identify installation packages in a multi-product transaction. + + + + + + + + + + + + + + + Adds a digital certificate. + + + + + + + + + Identifier for a certificate file. + + + + + The path to the certificate file. + + + + + + + + Reference to a DigitalCertificate element. This will force the entire referenced Fragment's contents + to be included in the installer database. This is only used for references when patching. + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + Adds a digital signature. + + + + + + + + + + + + The path to signature's optional hash file. + + + + + + + + Adds a system file protection update catalog file + + + + + + + + + + + Primary Key to File Table. + + + + + + Filename for catalog file when installed. + + + + + Used to define dependency outside of the package. + + + + + Path to catalog file in binary. + + + + + + + + Provides a many-to-many mapping from the SFPCatalog table to the File table + + + + + + + + + Primary Key to File Table. + + + + + + + + Adds or removes .ini file entries. + + + + + + + + + + Identifier for ini file. + + + + + The type of modification to be made. + + + + + + Creates or updates an .ini entry. + + + + + Creates a new entry or appends a new comma-separated value to an existing entry. + + + + + Creates an .ini entry only if the entry does no already exist. + + + + + Removes an .ini entry. + + + + + Removes a tag from an .ini entry. + + + + + + + + Name of a property, the value of which is the full path of the folder containing the .ini file. Can be name of a directory in the Directory table, a property set by the AppSearch table, or any other property representing a full path. + + + + + The localizable .ini file key within the section. + + + + + + In prior versions of the WiX toolset, this attribute specified the short name. + This attribute's value may now be either a short or long name. + If a short name is specified, the ShortName attribute may not be specified. + Also, if this value is a long name, the ShortName attribute may be omitted to + allow WiX to attempt to generate a unique short name. + However, if this name collides with another file or you wish to manually specify + the short name, then the ShortName attribute may be specified. + + + + + + The localizable .ini file section. + + + + + + The short name of the in 8.3 format. + This attribute should only be set if there is a conflict between generated short names + or the user wants to manually specify the short name. + + + + + + + The localizable value to be written or deleted. This attribute must be set if + the Action attribute's value is "addLine", "addTag", or "createLine". + + + + + + + + + ODBCDataSource for a Component + + + + + + + + + + Translates into ODBCSourceAttributes + + + + + + Identifier of the data source. + + + + + Name for the data source. + + + + + Required if not found as child of ODBCDriver element + + + + + Scope for which the data source should be registered. + + + + + + + Data source is registered per machine. + + + + + + + Data source is registered per user. + + + + + + + + + Set 'yes' to force this file to be key path for parent Component + + + + + + + + ODBCDriver for a Component + + + + + + + + + + Translates into ODBCSourceAttributes + + + + + + + Identifier for the driver. + + + + + Name for the driver. + + + + + Required if not found as child of File element + + + + + Required if not found as child of File element or different from File attribute above + + + + + + + + ODBCTranslator for a Component + + + + + + + + + Identifier for the translator. + + + + + Name for the translator. + + + + + Required if not found as child of File element + + + + + Required if not found as child of File element or different from File attribute above + + + + + + + + + + + + + + + + How To: Check the version number of a file during installation + + When the parent DirectorySearch/@Depth attribute is greater than 0, the FileSearch/@Id attribute must be absent or the same as the parent DirectorySearch/@Id attribute value, unless the parent DirectorySearch/@AssignToProperty attribute value is 'yes'. + + + Searches for file and assigns to fullpath value of parent Property + + + + + Unique identifier for the file search and external key into the Signature table. If this attribute value is not set then the parent element's @Id attribute is used. + + + + + + In prior versions of the WiX toolset, this attribute specified the short file name. + This attribute's value may now be either a short or long file name. + If a short file name is specified, the ShortName attribute may not be specified. + If you wish to manually specify the short file name, then the ShortName + attribute may be specified. + + + + + + + The short file name of the file in 8.3 format. + There is a Windows Installer bug which prevents the FileSearch functionality from working + if both a short and long file name are specified. Since the Name attribute allows either + a short or long name to be specified, it is the only attribute related to file names which + should be specified. + + + + + + The minimum size of the file. + + + + + The maximum size of the file. + + + + + The minimum version of the file. + + + + + The maximum version of the file. + + + + + The minimum modification date and time of the file. Formatted as YYYY-MM-DDTHH:mm:ss, where YYYY is the year, MM is month, DD is day, 'T' is literal, HH is hour, mm is minute and ss is second. + + + + + The maximum modification date and time of the file. Formatted as YYYY-MM-DDTHH:mm:ss, where YYYY is the year, MM is month, DD is day, 'T' is literal, HH is hour, mm is minute and ss is second. + + + + + The languages supported by the file. + + + + + + + + + + A reference to another FileSearch element must reference the same Id and the same Parent Id. If any of these attribute values are different you must instead use a FileSearch element. + + + References an existing FileSearch element. + + + + + Specify the Id to the FileSearch to reference. + + + + + + + + + + + + + How To: Check the version number of a file during installation + How To: Reference another DirectorySearch element + How To: Get the parent directory of a file search + + Use the AssignToProperty attribute to search for a file but set the outer property to the directory containing the file. When this attribute is set to 'yes', you may only nest a FileSearch element with a unique Id or define no child element. + When the parent DirectorySearch/@Depth attribute is greater than 0, the FileSearch/@Id attribute must be absent or the same as the parent DirectorySearch/@Id attribute value, unless the parent DirectorySearch/@AssignToProperty attribute value is 'yes'. + + + Searches for directory and assigns to value of parent Property. + + + + + + + + + + + Unique identifier for the directory search. + + + + + Path on the user's system. Either absolute, or relative to containing directories. + + + + + + Depth below the path that the installer searches for the file or directory specified by the search. See remarks for more information. + + + + + + Set the value of the outer Property to the result of this search. See remarks for more information. + + + + + + + + + + + How To: Reference another DirectorySearch element + + A reference to another DirectorySearch element must reference the same Id, the same Parent Id, and the same Path. If any of these attribute values are different you must instead use a DirectorySearch element. + + + References an existing DirectorySearch element. + + + + + + + + + + + Id of the search being referred to. + + + + + This attribute is the signature of the parent directory of the file or directory in the Signature_ column. If this field is null, and the Path column does not expand to a full path, then all the fixed drives of the user's system are searched by using the Path. This field is a key into one of the following tables: the RegLocator, the IniLocator, the CompLocator, or the DrLocator tables. + + + + + Path on the user's system. Either absolute, or relative to containing directories. + + + + + + + + + + + + + Searches for file or directory and assigns to value of parent Property. + + + + + + + + + + + + The component ID of the component whose key path is to be used for the search. + + + + + Must be file if last child is FileSearch element and must be directory if last child is DirectorySearch element. + + + + + + + The key path of the component is a directory. + + + + + + + The key path of the component is a file. This is the default value. + + + + + + + + + + + + + + + + + Searches for file, directory or registry key and assigns to value of parent Property + + + + + + + + + + + External key into the Signature table. + + + + + The field in the .ini line. If field is Null or 0, the entire line is read. + + + + + The key value within the section. + + + + + + In prior versions of the WiX toolset, this attribute specified the short name. + This attribute's value may now be either a short or long name. + If a short name is specified, the ShortName attribute may not be specified. + Also, if this value is a long name, the ShortName attribute may be omitted to + allow WiX to attempt to generate a unique short name. + However, if you wish to manually specify the short name, then the ShortName + attribute may be specified. + + + + + + The localizable .ini file section. + + + + + + The short name of the file in 8.3 format. + This attribute should only be set if the user wants to manually specify the short name. + + + + + + Must be file if last child is FileSearch element and must be directory if last child is DirectorySearch element. + + + + + + A directory location. + + + + + A file location. This is the default value. + + + + + A raw .ini value. + + + + + + + + + + + + + + + How To: Read a registry entry during installation + + + When the Type attribute value is 'directory' the registry value must specify the path to a directory excluding the file name. + When the Type attribute value is 'file' the registry value must specify the path to a file including the file name; + however, if there is no child FileSearch element the parent directory of the file is returned. The FileSearch element requires + that you author the name of the file you are searching for. If you do not know the file name + you must set the Type attribute to 'raw' to return the full file path including the file name. + + + + Searches for file, directory or registry key and assigns to value of parent Property + + + + + + + + + + + Signature to be used for the file, directory or registry key being searched for. + + + + + Root key for the registry value. + + + + + + + HKEY_CLASSES_ROOT + + + + + + + HKEY_CURRENT_USER + + + + + + + HKEY_LOCAL_MACHINE + + + + + + + HKEY_USERS + + + + + + + + + Key for the registry value. + + + + + Registry value name. If this value is null, then the value from the key's unnamed or default value, if any, is retrieved. + + + + + + The value must be 'file' if the child is a FileSearch element, and must be 'directory' if child is a DirectorySearch element. + + + + + + + + The registry value contains the path to a directory. + + + + + + + The registry value contains the path to a file. To return the full file path you must add a FileSearch element as a child of this element; otherwise, the parent directory of the file path is returned. + + + + + + + Sets the raw value from the registry value. Please note that this value will contain a prefix as follows:DWORDStarts with '#' optionally followed by '+' or '-'.REG_BINARYStarts with '#x' and the installer converts and saves each hexadecimal digit (nibble) as an ASCII character prefixed by '#x'.REG_EXPAND_SZStarts with '#%'.REG_MULTI_SZStarts with '[~]' and ends with '[~]'.REG_SZNo prefix, but if the first character of the registry value is '#', the installer escapes the character by prefixing it with another '#'. + + + + + + + + Instructs the search to look in the 64-bit registry when the value is 'yes'. When the value is 'no', the search looks in the 32-bit registry. + The default value is based on the platform set by the -arch switch to candle.exe + or the InstallerPlatform property in a .wixproj MSBuild project: + For x86 and ARM, the default value is 'no'. + For x64 and IA64, the default value is 'yes'. + + + + + + + + + + + References an existing RegistrySearch element. + + + + + Specify the Id of the RegistrySearch to reference. + + + + + + + Sets the parent of a nested DirectorySearch element to CCP_DRIVE. + + + + + + + + + + + + + + + + Adds a row to the CCPSearch table. + + + + + + + Starts searches from the CCP_DRIVE. + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + + + How To: Check the version number of a file during installation + + Property value for a Product or Module. + + + + + + + Starts searches from the CCP_DRIVE. + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + Unique identifier for Property. + + + + + Sets a default value for the property. The value will be overwritten if the Property is used for a search. + + + + + Adds a row to the CCPSearch table. This attribute is only valid when this Property contains a search element. + + + + + Denotes that the Property is saved during admininistrative installation. See the AdminProperties Property for more information. + + + + + Denotes that the Property can be passed to the server side when doing a managed installation with elevated privileges. See the SecureCustomProperties Property for more information. + + + + + Denotes that the Property is not logged during installation. See the MsiHiddenProperties Property for more information. + + + + + + Use to suppress modularization of this property identifier in merge modules. + Using this functionality is strongly discouraged; it should only be + necessary as a workaround of last resort in rare scenarios. + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + + How To: Check for .NET Framework versions + + Reference to a Property value. + + + + + Identifier of Property to reference. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + Shortcut, default target is parent File, CreateFolder, or Component's Directory + + + + How To: Create a shortcut on the Start Menu + + + + + + + + + + Unique identifier for the shortcut. This value will serve as the primary key for the row. + + + + + Identifier reference to Directory element where shortcut is to be created. When nested under a Component element, this attribute's value will default to the parent directory. Otherwise, this attribute is required. + + + + + + In prior versions of the WiX toolset, this attribute specified the short name. + This attribute's value may now be either a short or long name. + If a short name is specified, the ShortName attribute may not be specified. + Also, if this value is a long name, the ShortName attribute may be omitted to + allow WiX to attempt to generate a unique short name. + However, if this name collides with another shortcut or you wish to manually specify + the short name, then the ShortName attribute may be specified. + + + + + + + The short name of the shortcut in 8.3 format. + This attribute should only be set if there is a conflict between generated short names + or the user wants to manually specify the short name. + + + + + + + This attribute can only be set if this Shortcut element is nested under a Component element. + When nested under a Component element, this attribute's value will default to the parent directory. + This attribute's value is the target for a non-advertised shortcut. + This attribute is not valid for advertised shortcuts. + If you specify this value, its value should be a property identifier enclosed by square brackets ([ ]), that is expanded into the file or a folder pointed to by the shortcut. + + + + + + The localizable description for the shortcut. + + + + + The command-line arguments for the shortcut. Note that the resolution of properties + in the Arguments field is limited. A property formatted as [Property] in this field can only be resolved if the + property already has the intended value when the component owning the shortcut is installed. For example, for the + argument "[#MyDoc.doc]" to resolve to the correct value, the same process must be installing the file MyDoc.doc and + the component that owns the shortcut. + + + + + The hotkey for the shortcut. The low-order byte contains the virtual-key code for + the key, and the high-order byte contains modifier flags. This must be a non-negative number. Authors of + installation packages are generally recommend not to set this option, because this can add duplicate hotkeys to a + users desktop. In addition, the practice of assigning hotkeys to shortcuts can be problematic for users using hotkeys + for accessibility. + + + + + Identifier reference to Icon element. The Icon identifier should have the same extension + as the file that it points at. For example, a shortcut to an executable (e.g. "my.exe") should reference an Icon with identifier + like "MyIcon.exe" + + + + + Identifier reference to Icon element. + + + + + + + + + The shortcut target will be displayed using the SW_SHOWNORMAL attribute. + + + + + + + The shortcut target will be displayed using the SW_SHOWMINNOACTIVE attribute. + + + + + + + The shortcut target will be displayed using the SW_SHOWMAXIMIZED attribute. + + + + + + + + + Directory identifier (or Property identifier that resolves to a directory) that resolves + to the path of the working directory for the shortcut. + + + + + Specifies if the shortcut should be advertised or not. Note that advertised shortcuts + always point at a particular application, identified by a ProductCode, and should not be shared between applications. + Advertised shortcuts only work for the most recently installed application, and are removed when that application is + removed. The default value is 'no'. + + + + + + The Formatted string providing the full path to the language neutral file containing the MUI Manifest. Generally + authored using [#filekey] form. When this attribute is specified, the DisplayResourceId attribute must also + be provided. + + This attribute is only used on Windows Vista and above. If this attribute is not populated and the install + is running on Vista and above, the value in the Name attribute is used. If this attribute is populated and + the install is running on Vista and above, the value in the Name attribute is ignored. + + + + + + + The display name index for the shortcut. This must be a non-negative number. When this attribute is specified, the + DisplayResourceDll attribute must also be provided. + + This attribute is only used on Windows Vista and above. If this attribute is not specified and the install + is running on Vista and above, the value in the Name attribute is used. If this attribute is specified and + the install is running on Vista and above, the value in the Name attribute is ignored. + + + + + + + The Formatted string providing the full path to the language neutral file containing the MUI Manifest. Generally + authored using [#filekey] form. When this attribute is specified, the DescriptionResourceId attribute must also + be provided. + + This attribute is only used on Windows Vista and above. If this attribute is not specified and the install + is running on Vista and above, the value in the Name attribute is used. If this attribute is provided and + the install is running on Vista and above, the value in the Name attribute is ignored. + + + + + + + The description name index for the shortcut. This must be a non-negative number. When this attribute is specified, + the DescriptionResourceDll attribute must also be populated. + + This attribute is only used on Windows Vista and above. If this attribute is not specified and the install + is running on Vista and above, the value in the Name attribute is used. If this attribute is populated and the + install is running on Vista and above, the value in the Name attribute is ignored. + + + + + + + + + + + + Property values for a shortcut. This element's functionality is available starting with MSI 5.0. + + + + + Unique identifier for MsiShortcutProperty table. If omitted, a stable identifier will be generated from the parent shortcut identifier and Key value. + + + + + A formatted string identifying the property to be set. + + + + + A formatted string supplying the value of the property. + + + + + + + + Sets ACLs on File, Registry, or CreateFolder. When under a Registry element, this cannot be used + if the Action attribute's value is remove or removeKeyOnInstall. This element has no Id attribute. + The table and key are taken from the parent element. + + + + + + + + + + + + + + + + + Bit mask for SPECIFIC_RIGHTS_ALL from WinNT.h (0x0000FFFF). + + + + + + + + + + + + For a directory, the right to create a file in the directory. Only valid under a 'CreateFolder' parent. + + + + + For a directory, the right to create a subdirectory. Only valid under a 'CreateFolder' parent. + + + + + For a directory, the right to delete a directory and all the files it contains, including read-only files. Only valid under a 'CreateFolder' parent. + + + + + For a directory, the right to traverse the directory. By default, users are assigned the BYPASS_TRAVERSE_CHECKING privilege, which ignores the FILE_TRAVERSE access right. Only valid under a 'CreateFolder' parent. + + + + + + + + Bit mask for FILE_ALL_ACCESS from WinNT.h (0x001F01FF). + + + + + + + + + + + + + + + + specifying this will fail to grant read access + + + + + + + + Sets ACLs on File, Registry, or CreateFolder. When under a Registry element, this cannot be used + if the Action attribute's value is remove or removeKeyOnInstall. This element is only available + when installing with MSI 5.0. For downlevel support, see the PermissionEx element from the + WixUtilExtension. + + + + + + + + + + + Optional condition that controls whether the permissions are applied. + + + + + + + + Primary key used to identify this particular entry. If this is not specified the parent element's Id attribute + will be used instead. + + + + + + + Security descriptor to apply to parent object. + + + + + + + + + + + + + + Copy or move an existing file on the target machine, or copy a file that is being installed, to another destination. When + this element is nested under a File element, the parent file will be installed, then copied to the specified destination + if the parent component of the file is selected for installation or removal. When this element is nested under + a Component element and no FileId attribute is specified, the file to copy or move must already be on the target machine. + When this element is nested under a Component element and the FileId attribute is specified, the specified file is installed, + then copied to the specified destination if the parent component is selected for installation or removal (use + this option to control the copy of a file in a different component by the parent component's installation state). If the + specified destination directory is the same as the directory containing the original file and the name for the proposed source + file is the same as the original, then no action takes place. + + + + + + Primary key used to identify this particular entry. + + + + + + This attribute cannot be specified if the element is nested under a File element. Set this attribute's value to the identifier + of a file from a different component to copy it based on the install state of the parent component. + + + + + + + This attribute cannot be specified if the element is nested under a File element or the FileId attribute is specified. Set + this value to the source directory from which to copy or move an existing file on the target machine. This Directory must + exist in the installer database at creation time. This attribute cannot be specified in conjunction with SourceProperty. + + + + + + + This attribute cannot be specified if the element is nested under a File element or the FileId attribute is specified. Set + this value to a property that will have a value that resolves to the full path of the source directory (or full path + including file name if SourceName is not specified). The property does not have to exist in the installer database at + creation time; it could be created at installation time by a custom action, on the command line, etc. This attribute + cannot be specified in conjunction with SourceDirectory. + + + + + + + This attribute cannot be specified if the element is nested under a File element or the FileId attribute is specified. Set + this value to the localizable name of the file(s) to be copied or moved. All of the files that + match the wild card will be removed from the specified directory. The value is a filename that may also + contain the wild card characters "?" for any single character or "*" for zero or more occurrences of any character. If this + attribute is not specified (and this element is not nested under a File element or specify a FileId attribute) then the + SourceProperty attribute should be set to the name of a property that will resolve to the full path of the source filename. + If the value of this attribute contains a "*" wildcard and the DestinationName attribute is specified, all moved or copied + files retain the file names from their sources. + + + + + + + Set this value to the destination directory where an existing file on the target machine should be moved or copied to. This + Directory must exist in the installer database at creation time. This attribute cannot be specified in conjunction with + DestinationProperty. + + + + + + + Set this value to a property that will have a value that resolves to the full path of the destination directory. The property + does not have to exist in the installer database at creation time; it could be created at installation time by a custom + action, on the command line, etc. This attribute cannot be specified in conjunction with DestinationDirectory. + + + + + + + In prior versions of the WiX toolset, this attribute specified the short file name. + Now set this value to the localizable name to be given to the original file after it is moved or copied. + If this attribute is not specified, then the destination file is given the same name as the source file. + If a short file name is specified, the DestinationShortName attribute may not be specified. + Also, if this value is a long file name, the DestinationShortName attribute may be omitted to + allow WiX to attempt to generate a unique short file name. + However, if this name collides with another file or you wish to manually specify + the short file name, then the DestinationShortName attribute may be specified. + + + + + + + The short file name of the file in 8.3 format. + This attribute should only be set if there is a conflict between generated short file names + or you wish to manually specify the short file name. + + + + + + + This attribute cannot be specified if the element is nested under a File element or the FileId attribute is specified. In other + cases, if the attribute is not specified, the default value is "no" and the file is copied, not moved. Set the value to "yes" + in order to move the file (thus deleting the source file) instead of copying it. + + + + + + + + + File specification for File table, must be child node of Component. + + + + How To: Add a file to your installer + + + + + + + + Used to configure the ACLs for this file. + + + + + Can also configure the ACLs for this file. + + + + + Used to create a duplicate of this file elsewhere. + + + + + Target of the shortcut will be set to this file. + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + + The unique identifier for this File element. If you omit Id, it defaults to the file name portion of the Source attribute, if specified. May be referenced as a Property by specifying [#value]. + + + + + + Set this attribute to make this file a companion child of another file. The installation + state of a companion file depends not on its own file versioning information, but on the versioning of its + companion parent. A file that is the key path for its component can not be a companion file (that means + this attribute cannot be set if KeyPath="yes" for this file). The Version attribute cannot be set along + with this attribute since companion files are not installed based on their own version. + + + + + + In prior versions of the WiX toolset, this attribute specified the short file name. + This attribute's value may now be either a short or long file name. + If a short file name is specified, the ShortName attribute may not be specified. + Also, if this value is a long file name, the ShortName attribute may be omitted to + allow WiX to attempt to generate a unique short file name. + However, if this name collides with another file or you wish to manually specify + the short file name, then the ShortName attribute may be specified. + Finally, if this attribute is omitted then its default value is the file name portion + of the Source attribute, if one is specified, or the value of the Id attribute, if + the Source attribute is omitted or doesn't contain a file name. + + + + + + Set to yes in order to force this file to be the key path for the parent component. + + + + + + The short file name of the file in 8.3 format. + This attribute should only be set if there is a conflict between generated short file names + or the user wants to manually specify the short file name. + + + + + + + Set to yes in order to have the file's read-only attribute set when it is installed on the target machine. + + + + + Set to yes in order to have the file's hidden attribute set when it is installed on the target machine. + + + + + Set to yes in order to have the file's system attribute set when it is installed on the target machine. + + + + + If a file is vital, then installation cannot proceed unless the file is successfully installed. The user will have no option to ignore an error installing this file. If an error occurs, they can merely retry to install the file or abort the installation. The default is "yes," unless the -sfdvital switch (candle.exe) or SuppressFileDefaultVital property (.wixproj) is used. + + + + + This attribute should be set to "yes" for every executable file in the installation that has a valid checksum stored in the Portable Executable (PE) file header. Only those files that have this attribute set will be verified for valid checksum during a reinstall. + + + + + Sets the file's source type compression. A setting of "yes" or "no" will override the setting in the Word Count Summary Property. + + + + + A list of paths, separated by semicolons, that represent the paths to be searched to find the imported DLLs. The list is usually a list of properties, with each property enclosed inside square brackets. The value may be set to an empty string. Including this attribute will cause an entry to be generated for the file in the BindImage table. + + + + + The cost of registering the file in bytes. This must be a non-negative number. Including this attribute will cause an entry to be generated for the file in the SelfReg table. + + + + + Causes an entry to be generated for the file in the Font table with no FontTitle specified. This attribute is intended to be used to register the file as a TrueType font. + + + + + Causes an entry to be generated for the file in the Font table with the specified FontTitle. This attribute is intended to be used to register the file as a non-TrueType font. + + + + + This is the default language of this file. The linker will replace this value from the value in the file if the suppress files option is not used. + + + + + This is the default size of this file. The linker will replace this value from the value in the file if the suppress files option is not used. + + + + + This is the default version of this file. The linker will replace this value from the value in the file if the suppress files option is not used. + + + + + + + Specifies if this File is a Win32 Assembly or .NET Assembly that needs to be installed into the + Global Assembly Cache (GAC). If the value is '.net' or 'win32', this file must also be the key path of the Component. + + + + + + + + The file is a .NET Framework assembly. + + + + + + + The file is not a .NET Framework or Win32 assembly. This is the default value. + + + + + + + The file is a Win32 assembly. + + + + + + + + + + Specifies the file identifier of the manifest file that describes this assembly. + The manifest file should be in the same component as the assembly it describes. + This attribute may only be specified if the Assembly attribute is set to '.net' or 'win32'. + + + + + + + Specifies the file identifier of the application file. This assembly will be isolated + to the same directory as the application file. + If this attribute is absent, the assembly will be installed to the Global Assembly Cache (GAC). + This attribute may only be specified if the Assembly attribute is set to '.net' or 'win32'. + + + + + + Specifies the architecture for this assembly. This attribute should only be used on .NET Framework 2.0 or higher assemblies. + + + + + + + The file is a .NET Framework assembly that is processor-neutral. + + + + + + + The file is a .NET Framework assembly for the x86 processor. + + + + + + + The file is a .NET Framework assembly for the x64 processor. + + + + + + + The file is a .NET Framework assembly for the ia64 processor. + + + + + + + + + + The value of this attribute should correspond to the Id attribute of a Media + element authored elsewhere. By creating this connection between a file and + its media, you set the packaging options to the values specified in the Media + element (values such as compression level, cab embedding, etc...). Specifying + the DiskId attribute on the File element overrides the default DiskId attribute + from the parent Component element. If no DiskId attribute is specified, + the default is "1". This DiskId attribute is ignored when creating a merge module + because merge modules do not have media. + + + + + + Specifies the path to the File in the build process. Overrides default source path set by parent directories and Name attribute. This attribute must be set if no source information can be gathered from parent directories. For more information, see Specifying source files. + + + + + + + + + + + + + This attribute must be set for patch-added files. Each patch should be assigned a different patch group number. Patch groups + numbers must be greater 0 and should be assigned consecutively. For example, the first patch should use PatchGroup='1', the + second patch will have PatchGroup='2', etc... + + + + + + Prevents the updating of the file that is in fact changed in the upgraded image relative to the target images. + + + + + Set to indicate that the patch is non-vital. + + + + + Set if the entire file should be installed rather than creating a binary patch. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + Use several of these elements to specify each registry value in a multiString registry value. This element + cannot be used if the Value attribute is specified unless the Type attribute is set to 'multiString'. The + values should go in the text area of the MultiStringValue element. + + + + + + + + + + Used for organization of child RegistryValue elements or to create a registry key + (and optionally remove it during uninstallation). + + + + How To: Read a registry entry during installation + How To: Write a registry entry during installation + + + + + + + + + ACL permission + + + + + Can also configure the ACLs for this registry key. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + + Primary key used to identify this particular entry. If this attribute is not specified, an identifier will be + generated by hashing the parent Component identifier, Root, Key, and Name. + + + + + + + The Action attribute has been deprecated. In most cases, you can simply omit @Action. If you need to force Windows Installer + to create an empty key or recursively delete the key, use the ForceCreateOnInstall or ForceDeleteOnUninstall attributes instead. + + + + + + + + Creates the key, if absent, when the parent component is installed. + + + + + + + Creates the key, if absent, when the parent component is installed then remove the key with all its values and subkeys when the parent component is uninstalled. + Note that this value is useful only if your program creates additional values or subkeys under this key and you want an uninstall to remove them. MSI already + removes all values and subkeys that it creates, so this option just adds additional overhead to uninstall. + + + + + + + Does nothing; this element is used merely in WiX authoring for organization and does nothing to the final output. + This is the default value. + + + + + + + + + + Set this attribute to 'yes' to create an empty key, if absent, when the parent component is installed. + This value is needed only to create an empty key with no subkeys or values. Windows Installer creates + keys as needed to store subkeys and values. The default is "no". + + + + + + + Set this attribute to 'yes' to remove the key with all its values and subkeys when the parent component is uninstalled. + Note that this value is useful only if your program creates additional values or subkeys under this key and you want an uninstall to remove them. MSI already + removes all values and subkeys that it creates, so this option just adds additional overhead to uninstall. + The default is "no". + + + + + + + The localizable key for the registry value. + If the parent element is a RegistryKey, this value may be omitted to use the + path of the parent, or if its specified it will be appended to the path of the parent. + + + + + + + The predefined root key for the registry value. + + + + + + + + + Used to create a registry value. For multi-string values, this can be used to prepend or append values. + + For legacy authoring: Use several of these elements to specify each registry value in a multiString registry value. This element + cannot be used if the Value attribute is specified unless the Type attribute is set to 'multiString'. The + values should go in the text area of the RegistryValue element. + + + + How To: Write a registry entry during installation + + + + + + + + Can also configure the ACLs for this registry value. + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + + Primary key used to identify this particular entry. If this attribute is not specified, an identifier will be + generated by hashing the parent Component identifier, Root, Key, and Name. + + + + + + + The predefined root key for the registry value. + + + + + + + The localizable key for the registry value. + If the parent element is a RegistryKey, this value may be omitted to use the + path of the parent, or if its specified it will be appended to the path of the parent. + + + + + + + The localizable registry value name. If this attribute is not provided the default value for the registry key will + be set instead. The Windows Installer allows several special values to be set for this attribute. You should not + use them in WiX. Instead use appropriate values in the Action attribute to get the desired behavior. + + + + + + + Set this attribute to the localizable registry value. This value is formatted. The Windows Installer allows + several special values to be set for this attribute. You should not use them in WiX. Instead use appropriate + values in the Type attribute to get the desired behavior. + + + + + + + + Set this attribute to the type of the desired registry key. This attribute must be specified whenever the Value + attribute or a child RegistryValue element is specified. This attribute + should only be set when the value of the Action attribute does not include the word 'remove'. + + + + + + + + The value is interpreted and stored as a string (REG_SZ). + + + + + + + The value is interpreted and stored as an integer (REG_DWORD). + + + + + + + The value is interpreted and stored as a hexadecimal value (REG_BINARY). + + + + + + + The value is interpreted and stored as an expandable string (REG_EXPAND_SZ). + + + + + + + The value is interpreted and stored as a multiple strings (REG_MULTI_SZ). + Please note that this value will only result in a multi-string value if there is more than one registry value + or the Action attribute's value is 'append' or 'prepend'. Otherwise a string value will be created. + + + + + + + + + + This is the action that will be taken for this registry value. + + + + + + + + Appends the specified value(s) to a multiString registry value. + + + + + + + Prepends the specified value(s) to a multiString registry value. + + + + + + + Writes a registry value. This is the default value. + + + + + + + + + + Set this attribute to 'yes' to make this registry key the KeyPath of the parent component. + Only one resource (registry, file, etc) can be the KeyPath of a component. + + + + + + + + + Used for removing registry keys and all child keys either during install or uninstall. + + + + + + + + + + + Primary key used to identify this particular entry. If this attribute is not specified, an identifier will be + generated by hashing the parent Component identifier, Root, Key, and Name. + + + + + + + This is the action that will be taken for this registry value. + + + + + + + + Removes a key with all its values and subkeys when the parent component is installed. + + + + + + + Removes a key with all its values and subkeys when the parent component is uninstalled. + + + + + + + + + + The localizable key for the registry value. + + + + + + + The predefined root key for the registry value. + + + + + + + + + Used to remove a registry value during installation. + There is no standard way to remove a single registry value during uninstall (but you can remove an entire key with RemoveRegistryKey). + + + + + + + + + + Primary key used to identify this particular entry. If this attribute is not specified, an identifier will be + generated by hashing the parent Component identifier, Root, Key, and Name. + + + + + + + The localizable key for the registry value. + If the parent element is a RegistryKey, this value may be omitted to use the + path of the parent, or if its specified it will be appended to the path of the parent. + + + + + + + The localizable registry value name. If this attribute is not provided the default value for the registry key will + be set instead. The Windows Installer allows several special values to be set for this attribute. You should not + use them in WiX. Instead use appropriate values in the Action attribute to get the desired behavior. + + + + + + + The predefined root key for the registry value. + + + + + + + + + + + + + + + + + + Can also configure the ACLs for this registry key. + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + + Primary key used to identify this particular entry. If this attribute is not specified, an identifier will be + generated by hashing the parent Component identifier, Root, Key, and Name. + + + + + + + This is the action that will be taken for this registry key. + + + + + + + + Appends the specified value(s) to a multiString registry key. + + + + + + + Creates the key, if absent, when the parent component is installed. + + + + + + + Creates the key, if absent, when the parent component is installed then remove the key with all its values and subkeys when the parent component is uninstalled. + + + + + + + Prepends the specified value(s) to a multiString registry key. + + + + + + + Removes a registry name when the parent component is installed. + + + + + + + Removes a key with all its values and subkeys when the parent component is installed. + + + + + + + Removes a key with all its values and subkeys when the parent component is uninstalled. + + + + + + + Writes a registry value. + + + + + + + + + The localizable key for the registry value. + + + + + + Set this attribute to 'yes' to make this registry key the KeyPath of the parent component. Only one resource (registry, + file, etc) can be the KeyPath of a component. + + + + + + + The localizable registry value name. If this attribute is not provided the default value for the registry key will + be set instead. The Windows Installer allows several special values to be set for this attribute. You should not + use them in WiX. Instead use appropriate values in the Action attribute to get the desired behavior. + + + + + + + The predefined root key for the registry value. + + + + + + + Set this attribute to the type of the desired registry key. This attribute must be specified whenever the Value + attribute or a child RegistryValue element is specified. This attribute + should only be set when the value of the Action attribute does not include the word 'remove'. + + + + + + + + The value is interpreted and stored as a string (REG_SZ). + + + + + + + The value is interpreted and stored as an integer (REG_DWORD). + + + + + + + The value is interpreted and stored as a hexadecimal value (REG_BINARY). + + + + + + + The value is interpreted and stored as an expandable string (REG_EXPAND_SZ). + + + + + + + The value is interpreted and stored as a multiple strings (REG_MULTI_SZ). + Please note that this value will only result in a multi-string value if there is more than one registry value + or the Action attribute's value is 'append' or 'prepend'. Otherwise a string value will be created. + + + + + + + + + + Set this attribute to the localizable registry value. This value is formatted. The Windows Installer allows + several special values to be set for this attribute. You should not use them in WiX. Instead use appropriate + values in the Type attribute to get the desired behavior. This attribute cannot be specified if the Action + attribute's value contains the word 'remove'. + + + + + + + + + + + + + Remove a file(s) if the parent component is selected for installation or removal. Multiple files can be removed + by specifying a wildcard for the value of the Name attribute. By default, the source + directory of the file is the directory of the parent component. This can be overridden by specifying the + Directory attribute with a value corresponding to the Id of the source directory, or by specifying the Property + attribute with a value corresponding to a property that will have a value that resolves to the full path + to the source directory. + + + + + + Primary key used to identify this particular entry. + + + + + + Overrides the directory of the parent component with a specific Directory. This Directory must exist in the + installer database at creation time. This attribute cannot be specified in conjunction with the Property attribute. + + + + + + + Overrides the directory of the parent component with the value of the specified property. The property + should have a value that resolves to the full path of the source directory. The property does not have + to exist in the installer database at creation time; it could be created at installation time by a custom + action, on the command line, etc. This attribute cannot be specified in conjunction with the Directory attribute. + + + + + + + This value should be set to the localizable name of the file(s) to be removed. All of the files that + match the wild card will be removed from the specified directory. The value is a filename that may also + contain the wild card characters "?" for any single character or "*" for zero or more occurrences of any character. + In prior versions of the WiX toolset, this attribute specified the short file name. + This attribute's value may now be either a short or long file name. + If a short file name is specified, the ShortName attribute may not be specified. + Also, if this value is a long file name, the ShortName attribute may be omitted to + allow WiX to attempt to generate a unique short file name. + However, if you wish to manually specify the short file name, then the ShortName attribute may be specified. + + + + + + + The short file name of the file in 8.3 format. + This attribute should only be set if you want to manually specify the short file name. + + + + + + + This value determines the time at which the file(s) may be removed. For 'install', the file will + be removed only when the parent component is being installed (msiInstallStateLocal or + msiInstallStateSource); for 'uninstall', the file will be removed only when the parent component + is being removed (msiInstallStateAbsent); for 'both', the file will be removed in both cases. + + + + + + + + + + + + + Remove an empty folder if the parent component is selected for installation or removal. By default, the folder + is the directory of the parent component. This can be overridden by specifying the Directory attribute + with a value corresponding to the Id of the directory, or by specifying the Property attribute with a value + corresponding to a property that will have a value that resolves to the full path of the folder. + + + + + + Primary key used to identify this particular entry. + + + + + + Overrides the directory of the parent component with a specific Directory. This Directory must exist in the + installer database at creation time. This attribute cannot be specified in conjunction with the Property attribute. + + + + + + + Overrides the directory of the parent component with the value of the specified property. The property + should have a value that resolves to the full path of the source directory. The property does not have + to exist in the installer database at creation time; it could be created at installation time by a custom + action, on the command line, etc. This attribute cannot be specified in conjunction with the Directory attribute. + + + + + + + This value determines the time at which the folder may be removed, based on the install/uninstall of the parent component. + For 'install', the folder will be removed only when the parent component is being installed (msiInstallStateLocal or + msiInstallStateSource); for 'uninstall', the folder will be removed only when the parent component + is being removed (msiInstallStateAbsent); for 'both', the folder will be removed in both cases. + + + + + + + + + + + + Create folder as part of parent Component. + + + + + + Non-advertised shortcut to this folder, Shortcut Target is preset to the folder + + + + + ACL permission + + + + + Can also configure the ACLs for this folder. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + Identifier of Directory to create. Defaults to Directory of parent Component. + + + + + + + Optional way for defining AppData, generally used for complex CDATA. + + + + + + Qualified published component for parent Component + + + + + + + + + + + + A string GUID that represents the category of components being grouped together. + + + + + A text string that qualifies the value in the Id attribute. A qualifier is used to distinguish multiple forms of the same Component, such as a Component that is implemented in multiple languages. + + + + + An optional localizable text describing the category. The string is commonly parsed by the application and can be displayed to the user. It should describe the category. + + + + + Feature that controls the advertisement of the category. Defaults to the primary Feature for the parent Component . + + + + + + + + MIME content-type for an Extension + + + + + + + + + Whether this MIME is to be advertised. The default is to match whatever the parent extension element uses. If the parent element is not advertised, then this element cannot be advertised either. + + + + + This is the identifier for the MIME content. It is commonly written in the form of type/format. + + + + + Class ID for the COM server that is to be associated with the MIME content. + + + + + If 'yes', become the content type for the parent Extension. The default value is 'no'. + + + + + + + + Verb definition for an Extension. When advertised, this element creates a row in the + Verb table. + When not advertised, this element creates the appropriate rows in Registry table. + + + + + + + + + + The verb for the command. + + + + + The localized text displayed on the context menu. + + + + + Value for the command arguments. Note that the resolution of properties in the + Argument field is limited. A property formatted as [Property] in this field can only be resolved if the property + already has the intended value when the component owning the verb is installed. For example, for the argument + "[#MyDoc.doc]" to resolve to the correct value, the same process must be installing the file MyDoc.doc and the + component that owns the verb. + + + + + The sequence of the commands. Only verbs for which the Sequence is specified + are used to prepare an ordered list for the default value of the shell key. The Verb with the lowest value in this + column becomes the default verb. Used only for Advertised verbs. + + + + + + + + + + + + + Either this attribute or the TargetProperty attribute must be specified for a non-advertised verb. + The value should be the identifier of the target file to be executed for the verb. + + + + + + + Either this attribute or the TargetFile attribute must be specified for a non-advertised verb. + The value should be the identifier of the property which will resolve to the path to the target file to be executed for the verb. + + + + + + + + + Extension for a Component + + + + + + + + + + + MIME and Verbs can be associated with Extensions + + + + + + + This is simply the file extension, like "doc" or "xml". Do not include the preceding period. + + + + + The MIME type that is to be written. + + + + + Whether this extension is to be advertised. The default is "no". + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + Register a type library (TypeLib). Please note that in order to properly use this + non-advertised, you will need use this element with Advertise='no' and also author the + appropriate child Interface elements by extracting them from the type library itself. + + + + + + + + + + + + + + + The GUID that identifes the type library. + + + + + + Value of 'yes' will create a row in the TypeLib table. + Value of 'no' will create rows in the Registry table. + The default value is 'no'. + + + + + + + Value of 'yes' means the type library describes controls, and should not be displayed in type browsers intended for nonvisual objects. + This attribute can only be set if Advertise='no'. + + + + + + + The cost associated with the registration of the type library in bytes. This attribute cannot be set if Advertise='no'. + + + + + + The localizable description of the type library. + + + + + + Value of 'yes' means the type library exists in a persisted form on disk. This attribute can only be set if Advertise='no'. + + + + + + The identifier of the Directory element for the help directory. + + + + + + Value of 'yes' means the type library should not be displayed to users, although its use is not restricted. + Should be used by controls. Hosts should create a new type library that wraps the control with extended properties. + This attribute can only be set if Advertise='no'. + + + + + + The language of the type library. This must be a non-negative integer. + + + + + The major version of the type library. The value should be an integer from 0 - 255. + + + + + The minor version of the type library. The value should be an integer from 0 - 255. + + + + + The resource id of a typelib. The value is appended to the end of the typelib path in the registry. + + + + + + Value of 'yes' means the type library is restricted, and should not be displayed to users. This attribute can only be set if Advertise='no'. + + + + + + + + + ProgId registration for parent Component. If ProgId has an associated Class, it must be a child of that element. + + + + + + + + + + + + + The version-independent ProgId must be the first child element of actual ProgId. Nesting other ProgId elements within the Version-independent ProgId will create COM+ aliases, see http://support.microsoft.com/kb/305745 for more information. + + + + + Extensions that refer to this ProgId + + + + + + + + For an advertised ProgId, the Id of an Icon element. For a non-advertised ProgId, this is the Id of a file containing an icon resource. + + + + + + + Specifies that the associated ProgId should not be opened by users. The value is presented as a warning to users. An empty string is also valid for this attribute. + + + + + + + + Application ID containing DCOM information for the associated application GUID. + If this element is nested under a Fragment, Module, or Product element, it must be + advertised. + + + + + + When being used in unadvertised mode, the attributes in the AppId element correspond to registry keys + as follows (values that can be specified in authoring are in bold): + IdIn General + [HKCR\AppID\{Id}]Specific Example + [HKCR\AppID\{01234567-89AB-CDEF-0123-456789ABCDEF}]ActivateAtStorageIn General + [HKCR\AppID\{Id}] + ActivateAtStorage="ActivateAtStorage" + Specific Example + [HKCR\AppID\{01234567-89AB-CDEF-0123-456789ABCDEF}] + ActivateAtStorage="Y" + DescriptionIn General + [HKCR\AppID\{Id}] + @="Description" + Specific Example + [HKCR\AppID\{01234567-89AB-CDEF-0123-456789ABCDEF}] + @="My AppId Description" + DllSurrogateIn General + [HKCR\AppID\{Id}] + DllSurrogate="DllSurrogate" + Specific Example + [HKCR\AppID\{01234567-89AB-CDEF-0123-456789ABCDEF}] + DllSurrogate="C:\surrogate.exe" + LocalServiceIn General + [HKCR\AppID\{Id}] + LocalService="LocalService" + Specific Example + [HKCR\AppID\{01234567-89AB-CDEF-0123-456789ABCDEF}] + LocalService="MyServiceName" + RemoteServerNameIn General + [HKCR\AppID\{Id}] + RemoteServerName="RemoteServerName" + Specific Example + [HKCR\AppID\{01234567-89AB-CDEF-0123-456789ABCDEF}] + RemoteServerName="MyRemoteServer" + RunAsInteractiveUserIn General + [HKCR\AppID\{Id}] + RunAs="RunAsInteractiveUser" + Specific Example + [HKCR\AppID\{01234567-89AB-CDEF-0123-456789ABCDEF}] + RunAs="Interactive User" + ServiceParametersIn General + [HKCR\AppID\{Id}] + ServiceParameters="ServiceParameters" + Specific Example + [HKCR\AppID\{01234567-89AB-CDEF-0123-456789ABCDEF}] + ServiceParameters="-param" + + + + + + + + + + + Set this value to 'yes' to configure the client to activate on the same system as persistent storage. + + + + + + + Set this value to 'yes' in order to create a normal AppId table row. Set this value to 'no' in order to + generate Registry rows that perform similar registration (without the often problematic Windows Installer + advertising behavior). + + + + + + + Set this value to the description of the AppId. It can only be specified when the AppId is not being advertised. + + + + + + + Set this value to specify that the class is a DLL that is to be activated in a surrogate EXE + process, and the surrogate process to be used is the path of a surrogate EXE file specified by the value. + + + + + + + Set this value to the AppID GUID that corresponds to the named executable. + + + + + + + Set this value to the name of a service to allow the object to be installed as a Win32 service. + + + + + + + Set this value to the name of the remote server to configure the client to request the object + be run at a particular machine whenever an activation function is called for which a COSERVERINFO + structure is not specified. + + + + + + + Set this value to 'yes' to configure a class to run under the identity of the user currently + logged on and connected to the interactive desktop when activated by a remote client without + being written as a Win32 service. + + + + + + + Set this value to the parameters to be passed to a LocalService on invocation. + + + + + + + + COM Class registration for parent Component. + + + + + + + + When being used in unadvertised mode, the attributes in the Class element correspond to registry keys + as follows (values that can be specified in authoring are in bold): + Id/Context/ServerIn General + [HKCR\CLSID\{Id}\Context1] + @="[!Server]" + [HKCR\CLSID\{Id}\Context2] + @="[!Server]" + Specific Example + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\LocalServer] + @="[!comserv.dll]" + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\LocalServer32] + @="[!comserv.dll]" + Id/Context/ForeignServerIn General + [HKCR\CLSID\{Id}\Context1] + @="ForeignServer" + [HKCR\CLSID\{Id}\Context2] + @="ForeignServer" + Specific Example + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\LocalServer] + @="mscoree.dll" + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\LocalServer32] + @="mscoree.dll" + AppIdIn General + [HKCR\CLSID\{Id}] + AppId="{AppId}" + Specific Example + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}] + AppId="{00000000-89AB-0000-0123-000000000000}" + ArgumentIn General + [HKCR\CLSID\{Id}\Context] + @="[!Server] Argument" + Specific Example + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\LocalServer32] + @="[!comserv.dll] /arg1 /arg2 /arg3"ControlIn General + Value "yes" specified: + [HKCR\CLSID\{Id}\Control] + Specific Example + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\Control] + DescriptionIn General + [HKCR\CLSID\{Id}] + @="Description" + Specific Example + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}] + @="Description of Example COM Component" + HandlerIn General + Value "1" specified: + [HKCR\CLSID\{Id}\InprocHandler] + @="ole.dll" + Value "2" specified: + [HKCR\CLSID\{Id}\InprocHandler32] + @="ole32.dll" + Value "3" specified: + [HKCR\CLSID\{Id}\InprocHandler] + @="ole.dll" + [HKCR\CLSID\{Id}\InprocHandler32] + @="ole32.dll" + Other value specified: + [HKCR\CLSID\{Id}\InprocHandler32] + @="Handler" + Specific Example (for other value) + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\InprocHandler32] + @="handler.dll" + Icon/IconIndexThis is not currently handled properly.InsertableIn General + Value "no" specified: + [HKCR\CLSID\{Id}\NotInsertable] + Value "yes" specified: + [HKCR\CLSID\{Id}\Insertable] + Specific Example + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\Insertable] + ProgrammableIn General + Value "yes" specified: + [HKCR\CLSID\{Id}\Programmable] + Specific Example + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\Programmable] + RelativePathUnsupported. Please contribute this back to WiX if you know.SafeForInitializingIn General + Value "yes" specified: + [HKCR\CLSID\{Id}\Implemented Categories\{7DD95802-9882-11CF-9FA9-00AA006C42C4}] + Specific Example + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\Implemented Categories\{7DD95802-9882-11CF-9FA9-00AA006C42C4}] + SafeForScriptingIn General + Value "yes" specified: + [HKCR\CLSID\{Id}\Implemented Categories\{7DD95801-9882-11CF-9FA9-00AA006C42C4}] + Specific Example + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\Implemented Categories\{7DD95801-9882-11CF-9FA9-00AA006C42C4}] + ThreadingModelIn General + [HKCR\CLSID\{Id}\Context] + ThreadingModel="ThreadingModel" + Specific Example + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\LocalServer32] + ThreadingModel="Apartment" + TypeLibId (from parent TypeLib/@Id)In General + [HKCR\CLSID\{Id}\TypeLib] + @="{TypeLibId}" + Specific Example + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\TypeLib] + @="{11111111-89AB-1111-0123-111111111111}" + VersionIn General + [HKCR\CLSID\{Id}\Version] + @="Version" + Specific Example + [HKCR\CLSID\{01234567-89AB-CDEF-0123-456789ABCDEF}\Version] + @="1.0.0.0" + + + + + + + + A ProgId associated with Class must be a child element of the Class element + + + + + + These Interfaces will be registered with the parent Class and TypeLib (if present). + + + + + + The Class identifier (CLSID) of a COM server. + + + + + + The server context(s) for this COM server. This attribute is optional for VB6 libraries that are marked "PublicNotCreateable". + Class elements marked Advertised must specify at least one server context. It is most common for there to be a single value + for the Context attribute. + + + + + + + + + + A 16-bit local server application. + + + + + + + A 32-bit local server application. + + + + + + + A 16-bit in-process server DLL. + + + + + + + A 32-bit in-process server DLL. + + + + + + + + + + + Localized description associated with the Class ID and Program ID. + + + + + + This attribute is only allowed when a Class is advertised. Using this attribute will reference an Application ID + containing DCOM information for the associated application GUID. The value must correspond to an AppId/@Id of an + AppId element nested under a Fragment, Module, or Product element. To associate an AppId with a non-advertised + class, nest the class within a parent AppId element. + + + + + + + The file providing the icon associated with this CLSID. Reference to an Icon element + (should match the Id attribute of an Icon element). This is currently not supported if the + value of the Advertise attribute is "no". + + + + + + Icon index into the icon file. + + + + + + The default inproc handler. May be optionally provided only for Context = LocalServer or + LocalServer32. Value of "1" creates a 16-bit InprocHandler (appearing as the InprocHandler + value). Value of "2" creates a 32-bit InprocHandler (appearing as the InprocHandler32 value). + Value of "3" creates 16-bit as well as 32-bit InprocHandlers. A non-numeric value is treated + as a system file that serves as the 32-bit InprocHandler (appearing as the InprocHandler32 value). + + + + + + + This column is optional only when the Context column is set to "LocalServer" + or "LocalServer32" server context. The text is registered as the argument against + the OLE server and is used by OLE for invoking the server. Note that the resolution + of properties in the Argument field is limited. A property formatted as [Property] in + this field can only be resolved if the property already has the intended value when + the component owning the class is installed. For example, for the argument "[#MyDoc.doc]" + to resolve to the correct value, the same process must be installing the file MyDoc.doc and the + component that owns the class. + + + + + + + When the value is "yes", the bare file name can be used for COM servers. The installer + registers the file name only instead of the complete path. This enables the server in + the current directory to take precedence and allows multiple copies of the same component. + + + + + + + Set this value to "yes" in order to create a normal Class table row. Set this value to + "no" in order to generate Registry rows that perform similar registration (without the + often problematic Windows Installer advertising behavior). + + + + + + + + Threading model for the CLSID. + + + + + + + + + + + + + + + + + Version for the CLSID. + + + + + + + Specifies the CLSID may be insertable. + + + + + + + Specifies the CLSID may be programmable. + + + + + + + May only be specified if the value of the Advertise attribute is "no" and Server has not been specified. In addition, it may only + be used when the Class element is directly under the Component element. The value can be + that of an registry type (REG_SZ). This attribute should be used to specify foreign servers, such as mscoree.dll if needed. + + + + + + + May only be specified if the value of the Advertise attribute is "no" and the ForeignServer attribute is not specified. File Id of the + COM server file. If this element is nested under a File element, this value defaults to + the value of the parent File/@Id. + + + + + + + Specifies whether or not to use the short path for the COM server. This can only apply when Advertise is set to 'no'. The default is 'no' meaning that it will use the long file name for the COM server. + + + + + + + May only be specified if the value of the Advertise attribute is "no". + + + + + + + May only be specified if the value of the Advertise attribute is "no". + + + + + + + Set this attribute's value to 'yes' to identify an object as an ActiveX Control. The default value is 'no'. + + + + + + + + COM Interface registration for parent TypeLib. + + + + + + + + GUID identifier for COM Interface. + + + + + Name for COM Interface. + + + + + Identifies the interface from which the current interface is derived. + + + + + GUID CLSID for proxy stub to COM Interface. + + + + + GUID CLSID for 32-bit proxy stub to COM Interface. + + + + + Number of methods implemented on COM Interface. + + + + + Determines whether a Typelib version entry should be created with the other COM Interface registry keys. Default is 'yes'. + + + + + + + FileType data for class Id registration. + + + + + Offset into file. If positive, offset is from the beginning; if negative, offset is from the end. + + + + + Hex value that is AND'd against the bytes in the file at Offset. + + + + + If the result of the AND'ing of Mask with the bytes in the file is Value, the file is a match for this File Type. + + + + + + + + Service or group of services that must start before the parent service. + + + + + + + + + + The value of this attribute should be one of the following: + The name (not the display name) of a previously installed service.The name of a service group (in which case the Group attribute must be set to 'yes'). + + + + + + Set to 'yes' to indicate that the value in the Id attribute is the name of a group of services. + + + + + + + + + Adds services for parent Component. Use the ServiceControl element to remove services. + + + + + The service executable installed will point to the KeyPath for the Component. + Therefore, you must ensure that the correct executable is either the first child + File element under this Component or explicitly mark the appropriate File element + as KeyPath='yes'. + + + + + + + + Configures the ACLs for this service. + + + + + Ordered list of dependencies when installing services. + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + + Unique identifier for this service configuration. This value will default to the Name attribute if not + specified. + + + + + + This column is the string that gives the service name to install. + + + + + This column is the localizable string that user interface programs use to identify the service. + + + + + The Windows Installer does not currently support kernelDriver or systemDriver. + + + + + + + A Win32 service that runs its own process. + + + + + + + A Win32 service that shares a process. + + + + + + + A kernel driver service. This value is not currently supported by the Windows Installer. + + + + + + + A file system driver service. This value is not currently supported by the Windows Installer. + + + + + + + + + Whether or not the service interacts with the desktop. + + + + + Determines when the service should be started. The Windows Installer does not support boot or system. + + + + + + + The service will start during startup of the system. + + + + + + + The service will start when the service control manager calls the StartService function. + + + + + + + The service can no longer be started. + + + + + + + The service is a device driver that will be started by the operating system boot loader. This value is not currently supported by the Windows Installer. + + + + + + + The service is a device driver that will be started by the IoInitSystem function. This value is not currently supported by the Windows Installer. + + + + + + + + + Determines what action should be taken on an error. + + + + + + + Logs the error and continues with the startup operation. + + + + + + + Logs the error, displays a message box and continues the startup operation. + + + + + + + Logs the error if it is possible and the system is restarted with the last configuration known to be good. If the last-known-good configuration is being started, the startup operation fails. + + + + + + + + + The overall install should fail if this service fails to install. + + + + + The load ordering group that this service should be a part of. + + + + + Fully qualified names must be used even for local accounts, e.g.: ".\LOCAL_ACCOUNT". Valid only when ServiceType is ownProcess. + + + + + The password for the account. Valid only when the account has a password. + + + + + Contains any command line arguments or properties required to run the service. + + + + + Sets the description of the service. + + + + + Determines whether the existing service description will be ignored. If 'yes', the service description will be null, even if the Description attribute is set. + + + + + + + + Argument used in ServiceControl parent + + + + + + + + + + Starts, stops, and removes services for parent Component. This element is used to control the state + of a service installed by the MSI or MSM file by using the start, stop and remove attributes. + For example, Start='install' Stop='both' Remove='uninstall' would mean: start the service on install, + remove the service when the product is uninstalled, and stop the service both on install and uninstall. + + + + + + + + + + Ordered list of arguments used when modifying services. + + + + + + + Name of the service. + + + + + + Specifies whether the service should be started by the StartServices action on install, uninstall or both. + For 'install', the service will be started only when the parent component is being installed (msiInstallStateLocal or + msiInstallStateSource); for 'uninstall', the service will be started only when the parent component + is being removed (msiInstallStateAbsent); for 'both', the service will be started in both cases. + + + + + + + Specifies whether the service should be stopped by the StopServices action on install, uninstall or both. + For 'install', the service will be stopped only when the parent component is being installed (msiInstallStateLocal or + msiInstallStateSource); for 'uninstall', the service will be stopped only when the parent component + is being removed (msiInstallStateAbsent); for 'both', the service will be stopped in both cases. + + + + + + + Specifies whether the service should be removed by the DeleteServices action on install, uninstall or both. + For 'install', the service will be removed only when the parent component is being installed (msiInstallStateLocal or + msiInstallStateSource); for 'uninstall', the service will be removed only when the parent component + is being removed (msiInstallStateAbsent); for 'both', the service will be removed in both cases. + + + + + + Specifies whether or not to wait for the service to complete before continuing. The default is 'yes'. + + + + + + + + Privilege required by service configured by ServiceConfig parent. Valid values are a privilege constant or a + Formatted property that resolves to a privilege constant. + + + + + + + + + + Configures a service being installed or one that already exists. This element's functionality is available starting with MSI 5.0. + + + + + + + + + + List of privileges to apply to service. + + + + + + + Unique identifier for this service configuration. This value will default to the ServiceName attribute if not + specified. + + + + + + + This attribute specifies whether an auto-start service should delay its start until after all other auto-start + services. This attribute only affects auto-start services. Allowed values are "yes", "no" or a Formatted property that + resolves to "1" (for "yes") or "0" (for "no"). If this attribute is not present the setting is not configured. + + + + + + + This attribute specifies when failure actions should be applied. Allowed values are "failedToStop", "failedToStopOrReturnedError" + or a Formatted property that resolves to "1" (for "failedToStopOrReturnedError") or "0" (for "failedToStop"). If this attribute + is not present the setting is not configured. + + + + + + + This attribute specifies time in milliseconds that the Service Control Manager (SCM) waits after notifying the service of a system + shutdown. If this attribute is not present the default value, 3 minutes, is used. + + + + + + + Specifies whether to configure the service when the parent Component is installed. This attribute may be combined with OnReinstall + and OnUninstall. + + + + + + + Specifies whether to configure the service when the parent Component is reinstalled. This attribute may be combined with OnInstall + and OnUninstall. + + + + + + + Specifies whether to configure the service when the parent Component is uninstalled. This attribute may be combined with OnInstall + and OnReinstall. + + + + + + + Specifies the name of the service to configure. This value will default to the ServiceInstall/@Name attribute when nested under + a ServiceInstall element. + + + + + + + Specifies the service SID to apply to the service. Valid values are "none", "restricted", "unrestricted" or a Formatted property + that resolves to "0" (for "none"), "3" (for "restricted") or "1" (for "unrestricted"). If this attribute is not present the + setting is not configured. + + + + + + + + Failure action for a ServiceConfigFailureActions element. + + + + + + Specifies the action to take when the service fails. Valid values are "none", "restartComputer", "restartService", "runCommand" or a Formatted property + that resolves to "0" (for "none"), "1" (for "restartService"), "2" (for "restartComputer") or "3" (for "runCommand"). + + + + + + + Specifies the time in milliseconds to wait before performing the value from the Action attribute. + + + + + + + + + Configures the failure actions for a service being installed or one that already exists. This element's functionality is available starting with MSI 5.0. + + + + + + + + + + Ordered list of failure actions to apply to service. + + + + + + + Unique identifier for this service configuration. This value will default to the ServiceName attribute if not + specified. + + + + + + + This attribute specifies command to execute when a "runCommand" failure action hit. If an empty string is provided it clears + the existing command. If this attribute is not present the setting is not changed. + + + + + + + Specifies whether to configure the service when the parent Component is installed. This attribute may be combined with OnReinstall + and OnUninstall. + + + + + + + Specifies whether to configure the service when the parent Component is reinstalled. This attribute may be combined with OnInstall + and OnUninstall. + + + + + + + Specifies whether to configure the service when the parent Component is uninstalled. This attribute may be combined with OnInstall + and OnReinstall. + + + + + + + Specifies the message to show for a reboot failure action. If an empty string is provided it clears any existing reboot message. If this + attribute is not present the setting is not changed. + + + + + + + Specifies the time in seconds to reset the failure count. If this attribute is not present the failure count will not be reset. + + + + + + + Specifies the name of the service to configure. This value will default to the ServiceInstall/@Name attribute when nested under + a ServiceInstall element. + + + + + + + + + Environment variables added or removed for the parent component. + + + + + + + + + Unique identifier for environment entry. + + + + + Name of the environment variable. + + + + + + The value to set into the environment variable. + If this attribute is not set, the environment variable is removed during installation if it exists on the machine. + + + + + + Optional attribute to change the separator used between values. By default a semicolon is used. + + + + + Specfies whether the environmental variable should be created, set or removed when the parent component is installed. + + + + + + Creates the environment variable if it does not exist, then set it during installation. This has no effect on the value of the environment variable if it already exists. + + + + + Creates the environment variable if it does not exist, and then set it during installation. If the environment variable exists, set it during the installation. + + + + + + Removes the environment variable during an installation. + The installer only removes an environment variable during an installation if the name and value + of the variable match the entries in the Name and Value attributes. + If you want to remove an environment variable, regardless of its value, do not set the Value attribute. + + + + + + + + + + + + + This value is the entire environmental variable. This is the default. + + + + + + + This value is prefixed. + + + + + + + This value is appended. + + + + + + + + + Specifies that the environment variable should not be removed on uninstall. + + + + + + Specifies that the environment variable should be added to the system environment space. The default + is 'no' which indicates the environment variable is added to the user environment space. + + + + + + + + + Conditions for components, controls, features, and products. The condition is specified in the inner text of the element. + + + + + + + How To: Block installation based on OS version + How To: Check the version number of a file during installation + + + + + + + + Under a Component element, the condition becomes the condition of the component. Under a Control element, + the condition becomes a ControlCondition entry. Under a Feature element, the condition becomes a Condition + entry. Under a Fragment or Product element, the condition becomes a LaunchCondition entry. + + + + + + Used only under Control elements and is required. Allows specific actions to be applied to a control based + on the result of this condition. + + + + + + + + Set the Control as the default. Only used under Control elements. + + + + + + + Enable the Control. Only used under Control elements. + + + + + + + Disable the Control. Only used under Control elements. + + + + + + + Hide the Control. Only used under Control elements. + + + + + + + Display the Control. Only used under Control elements. + + + + + + + + + + Used only under Feature elements and is required. Allows modifying the level of a Feature based on the + result of this condition. + + + + + + + Used only under Fragment or Product elements and is required. Set the value to the text to display when the + condition fails and the installation must be terminated. + + + + + + + + + + + Shared Component to be privately replicated in folder of parent Component + + + + + + + + + Shared Component for this application Component. + + + + + + + + Disk cost to reserve in a folder for running locally and/or from source. + + + + + + + + + A primary key that uniquely identifies this ReserveCost entry. + + + + + + Adds the amount of disk space specified in RunFromSource or RunLocal to the volume cost of the device containing the directory. + If this attribute is not set, it will default to the directory of parent component. + + + + + + The number of bytes of disk space to reserve if the component is installed to run from source. + + + + + The number of bytes of disk space to reserve if the component is installed to run locally. + + + + + + + Component for parent Directory + + + + + + + How To: Add a file to your installer + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + + Component identifier; this is the primary key for identifying components. If omitted, + the compiler defaults the identifier to the identifier of the resource that is the + explicit keypath of the component (for example, a child File element with KeyPath + attribute with value 'yes'. + + + + + + + Set this attribute to create a ComPlus entry. The value should be the export flags used + during the generation of the .msi file. For more information see the COM+ documentation + in the Platform SDK. + + + + + + + Set this attribute to 'yes' in order to disable registry reflection on all existing and + new registry keys affected by this component. + When set to 'yes', the Windows Installer calls the RegDisableReflectionKey on each key + being accessed by the component. + This bit is available with Windows Installer version 4.0 and is ignored on 32-bit systems. + + + + + + + Sets the Directory of the Component. If this element is nested under a Directory element, + this value defaults to the value of the parent Directory/@Id. + + + + + + + This attribute provides a default DiskId attribute for all child File elements. Specifying + the DiskId on a Component element will override any DiskId attributes set by parent Directory + or DirectoryRef elements. See the File element's DiskId attribute for more information about + the purpose of the DiskId. + + + + + + + Identifies a feature to which this component belongs, as a shorthand for a child + ComponentRef element of the Feature element. The value of this attribute should + correspond to the Id attribute of a Feature element authored elsewhere. Note that + a single component can belong to multiple features but this attribute allows you + to specify only a single feature. + + + + + + + This value should be a guid that uniquely identifies this component's contents, language, platform, and version. + If omitted, the default value is '*' which indicates that the linker should generate a stable guid. + Generatable guids are supported only for components with a single file as the component's keypath + or no files and a registry value as the keypath. + It's also possible to set the value to an empty string to specify an unmanaged component. + Unmanaged components are a security vulnerability because the component cannot be removed or repaired + by Windows Installer (it is essentially an unpatchable, permanent component). Therefore, a guid should + always be specified for any component which contains resources that may need to be patched in the future. + + + + + + + If this attribute's value is set to 'yes', then the Directory of this Component is used + as the KeyPath. To set a Registry value or File as the KeyPath of a component, set the + KeyPath attribute to 'yes' on one of those child elements. If KeyPath is not set to 'yes' for the + Component or for a child Registry value or File, WiX will look at the child elements under the + Component in sequential order and try to automatically select one of them as a key path. Allowing + WiX to automatically select a key path can be dangerous because adding or removing child elements + under the Component can inadvertantly cause the key path to change, which can lead to + installation problems. + + + + + + + Optional value that specifies the location that the component can be run from. + + + + + + + + Prevents the component from running from the source or the network (this is the default behavior if this attribute is not set). + + + + + + + Enforces that the component can only be run from the source (it cannot be run from the user's computer). + + + + + + + Allows the component to run from source or locally. + + + + + + + + + + If this attribute is set to 'yes', a new Component/@Guid will be generated for each + instance transform. Ensure that all of the resources contained in a multi-instance + Component will be installed to different paths based on the instance Property; otherwise, + the Component Rules will be violated. + + + + + + + If this attribute is set to 'yes', the installer does not install or reinstall the + component if a key path file or a key path registry entry for the component already + exists. The application does register itself as a client of the component. Use this + flag only for components that are being registered by the Registry table. Do not use + this flag for components registered by the AppId, Class, Extension, ProgId, MIME, and + Verb tables. + + + + + + + If this attribute is set to 'yes', the installer does not remove the component during + an uninstall. The installer registers an extra system client for the component in + the Windows Installer registry settings (which basically just means that at least one + product is always referencing this component). Note that this option differs from the + behavior of not setting a guid because although the component is permanent, it is still + patchable (because Windows Installer still tracks it), it's just not uninstallable. + + + + + + + If this attribute's value is set to 'yes', enables advanced patching semantics for + Components that are shared across multiple Products. Specifically, the Windows Installer + will cache the shared files to improve patch uninstall. This functionality is available + in Windows Installer 4.5 and later. + + + + + + + If this attribute's value is set to 'yes', the installer increments the reference count + in the shared DLL registry of the component's key file. If this bit is not set, the + installer increments the reference count only if the reference count already exists. + + + + + + + If this attribute is set to 'yes', the installer reevaluates the value of the statement + in the Condition upon a reinstall. If the value was previously False and has changed to + True, the installer installs the component. If the value was previously True and has + changed to False, the installer removes the component even if the component has other + products as clients. + + + + + + + If this attribute is set to 'yes', the installer will uninstall the Component's files + and registry keys when it is superseded by a patch. This functionality is available in + Windows Installer 4.5 and later. + + + + + + + Set this attribute to 'yes' to mark this as a 64-bit component. This attribute facilitates + the installation of packages that include both 32-bit and 64-bit components. If this is a 64-bit + component replacing a 32-bit component, set this attribute to 'yes' and assign a new GUID in the Guid attribute. + The default value is based on the platform set by the -arch switch to candle.exe + or the InstallerPlatform property in a .wixproj MSBuild project: + For x86 and ARM, the default value is 'no'. + For x64 and IA64, the default value is 'yes'. + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + Groups together multiple components to be used in other locations. + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + Identifier for the ComponentGroup. + + + + + + Sets the default directory identifier for child Component elements. + + + + + + + Used to set the default file system source for child Component elements. For more information, see + Specifying source files. + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + Create a reference to a ComponentGroup in another Fragment. + + + + + + + + The identifier of the ComponentGroup to reference. + + + + + + Set this attribute to 'yes' in order to make the parent feature of this component + the primary feature for this component. Components may belong to multiple features. + By designating a feature as the primary feature of a component, you ensure that + whenever a component is selected for install-on-demand (IOD), the primary feature + will be the one to install it. This attribute should only be set if a component + actually nests under multiple features. If a component nests under only one feature, + that feature is the primary feature for the component. You cannot set more than one + feature as the primary feature of a given component. + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + Used only for PatchFamilies to include all changes between the baseline and upgraded packages in a patch. + + + Warning: this is intended for testing purposes only. Shipping a patch with all changes negates the benefits of using patch families for including only specific changes. + Because changing the ProductCode is not supported in a patch, the ProductCode property is automatically removed from the transform. + + + + + + + Used only for PatchFamilies to include only a binary table entry in a patch. + + + + + The identifier of the Binary element to reference. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + Used only for PatchFamilies to include only a icon table entry in a patch. + + + + + The identifier of the Icon element to reference. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + Create a reference to a Feature element in another Fragment. + + + How To: Add a file to your installer + + + + + + The identifier of the Component element to reference. + + + + + + Set this attribute to 'yes' in order to make the parent feature of this component + the primary feature for this component. Components may belong to multiple features. + By designating a feature as the primary feature of a component, you ensure that + whenever a component is selected for install-on-demand (IOD), the primary feature + will be the one to install it. This attribute should only be set if a component + actually nests under multiple features. If a component nests under only one feature, + that feature is the primary feature for the component. You cannot set more than one + feature as the primary feature of a given component. + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + + How To: Install the Visual C++ Redistributable with your installer + + Merge directive to bring in a merge module that will be redirected to the parent directory. + + + + + + Data to use as input to a configurable merge module. + + + + + + The unique identifier for the Merge element in the source code. Referenced by the MergeRef/@Id. + + + + + The value of this attribute should correspond to the Id attribute of a + Media element authored elsewhere. By creating this connection between the merge module and Media + element, you set the packaging options to the values specified in the Media + element (values such as compression level, cab embedding, etc...). + + + + + Specifies if the files in the merge module should be compressed. + + + + + Specifies the decimal LCID or localization token for the language to merge the Module in as. + + + + + Path to the source location of the merge module. + + + + + + + + + + + + + + + + How To: Install the Visual C++ Redistributable with your installer + + Merge reference to connect a Merge Module to parent Feature + + + + + The unique identifier for the Merge element to be referenced. + + + + + Specifies whether the feature containing this MergeRef is the primary feature for advertising the merge module's components. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + Data to use as input to a configurable merge module. + + + + + Name of the item in the ModuleConfiguration table. + + + + + Value to be passed to configurable merge module. + + + + + + + Directory layout for the product. Also specifies the mappings between source and target directories. + + + + How To: Add a file to your installer + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + This value is the unique identifier of the directory entry. + + + + + + The Component Guid Generation Seed is a guid that must be used when a Component with the generate guid directive ("*") + is not rooted in a standard Windows Installer directory (for example, ProgramFilesFolder or CommonFilesFolder). + It is recommended that this attribute be avoided and that developers install their Components under standard + directories with unique names instead (for example, "ProgramFilesFolder\Company Name Product Name Version"). It is + important to note that once a directory is assigned a Component Guid Generation Seed the value must not change until + (and must be changed when) the path to that directory, including itself and all parent directories, changes. + + + + + + + Sets the default disk identifier for the files contained in this directory. + This attribute's value may be overridden by a child Component, Directory, + Merge or File element. See the File or Merge elements' DiskId attribute for + more information. + + + + + + Used to set the file system source for this directory's child elements. For more information, see Specifying source files. + + + + + + The name of the directory. + + Do not specify this attribute if this directory represents + the same directory as the parent (see the Windows Installer SDK's + Directory table + topic for more information about the "." operator). + + This attribute's value may either a short or long directory name. If a short directory + name is specified, the ShortName attribute may not be specified. If this value is a long + directory name, the ShortName attribute may be omitted to + allow WiX to attempt to generate a unique short directory name. + However, if this name collides with another directory or you wish to manually specify + the short directory name, then the ShortName attribute may be specified. + + This Name attribute may also define multiple directories using the inline directory syntax. + For example, "ProgramFilesFolder:\My Company\My Product\bin" would create a reference to a + Directory element with Id="ProgramFilesFolder" then create directories named "My Company" then + "My Product" then "bin" nested beneath each other. This syntax is a shortcut to defining + each directory in an individual Directory element. + + + + + + + The short name of the directory in 8.3 format. + This attribute should only be set if there is a conflict between generated short directory names + or the user wants to manually specify the short directory name. + + + + + + + The short name of the directory on the source media in 8.3 format. + This attribute should only be set if there is a conflict between generated short directory names + or the user wants to manually specify the short source directory name. + + + + + + + The name of the directory on the source media. + If this attribute is not specified, Windows Installer will default to the Name attribute. + + In prior versions of the WiX toolset, this attribute specified the short source directory name. + This attribute's value may now be either a short or long directory name. + If a short directory name is specified, the ShortSourceName attribute may not be specified. + If a long directory name is specified, the LongSource attribute may not be specified. + Also, if this value is a long directory name, the ShortSourceName attribute may be omitted to + allow WiX to attempt to generate a unique short directory name. + However, if this name collides with another directory or you wish to manually specify + the short directory name, then the ShortSourceName attribute may be specified. + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + Create a reference to a Directory element in another Fragment. + + + How To: Add a file to your installer + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + The identifier of the Directory element to reference. + + + + + + Sets the default disk identifier for the files contained in this directory. + This attribute's value may be overridden by a child Component, Directory, + Merge or File element. See the File or Merge elements' DiskId attribute for + more information. + + + + + + Used to set the file system source for this DirectoryRef's child elements. For more information, see Specifying source files. + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + + + + + + + + + Specifies the lower bound on the range of product versions to be detected by FindRelatedProducts. + + + + + Specifies the upper boundary of the range of product versions detected by FindRelatedProducts. + + + + + Specifies the set of languages detected by FindRelatedProducts. Enter a list of numeric language identifiers (LANGID) separated by commas (,). Leave this value null to specify all languages. Set ExcludeLanguages to "yes" in order detect all languages, excluding the languages listed in this value. + + + + + The installer sets the REMOVE property to features specified in this column. The features to be removed can be determined at run time. The Formatted string entered in this field must evaluate to a comma-delimited list of feature names. For example: [Feature1],[Feature2],[Feature3]. No features are removed if the field contains formatted text that evaluates to an empty string. The installer sets REMOVE=ALL only if the Remove field is empty. + + + + + When the FindRelatedProducts action detects a related product installed on the system, it appends the product code to the property specified in this field. Windows Installer documentation for the Upgrade table states that the property specified in this field must be a public property and must be added to the SecureCustomProperties property. WiX automatically appends the property specified in this field to the SecureCustomProperties property when creating an MSI. Each UpgradeVersion must have a unique Property value. After the FindRelatedProducts action is run, the value of this property is a list of product codes, separated by semicolons (;), detected on the system. + + + + + Set to "yes" to migrate feature states from upgraded products by enabling the logic in the MigrateFeatureStates action. + + + + + Set to "yes" to detect products and applications but do not uninstall. + + + + + Set to "yes" to continue installation upon failure to remove a product or application. + + + + + Set to "no" to make the range of versions detected exclude the value specified in Minimum. This attribute is "yes" by default. + + + + + Set to "yes" to make the range of versions detected include the value specified in Maximum. + + + + + Set to "yes" to detect all languages, excluding the languages listed in the Language attribute. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + + + Upgrade info for a particular UpgradeCode + + + + + + + + + + + + Nesting a Property element under an Upgrade element has been deprecated. + Please nest Property elements in any of the other supported locations. + + + + + + + This value specifies the upgrade code for the products that are to be detected by the FindRelatedProducts action. + + + + + + + + A feature for the Feature table. Features are the smallest installable unit. See msi.chm for more + detailed information on the myriad installation options for a feature. + + + + + How To: Add a file to your installer + + + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + Unique identifier of the feature. + + + + + + This attribute determines if a user will have the option to set a feature to absent in the user interface. + + + + + + + + Allows the user interface to display an option to change the feature state to Absent. + + + + + + + Prevents the user interface from displaying an option to change the feature state + to Absent by setting the msidbFeatureAttributesUIDisallowAbsent attribute. This will force the feature + to the installation state, whether or not the feature is visible in the UI. + + + + + + + + + + This attribute determines the possible advertise states for this feature. + + + + + + + + Prevents this feature from being advertised by setting the msidbFeatureAttributesDisallowAdvertise attribute. + + + + + + + Prevents advertising for this feature if the operating system shell does not support Windows Installer + descriptors by setting the msidbFeatureAttributesNoUnsupportedAdvertise attribute. + + + + + + + Allows the feature to be advertised. + + + + + + + + + + Specify the Id of a Directory that can be configured by the user at installation time. This identifier + must be a public property and therefore completely uppercase. + + + + + + + Longer string of text describing the feature. This localizable string is displayed by the + Text Control of the Selection Dialog. + + + + + + + Determines the initial display of this feature in the feature tree. + This attribute's value should be one of the following: + collapseInitially shows the feature collapsed. This is the default value.expandInitially shows the feature expanded.hiddenPrevents the feature from displaying in the user interface.<an explicit integer value> + For advanced users only, it is possible to directly set the integer value + of the display value that will appear in the Feature row. + + + + + + + This attribute determines the default install/run location of a feature. This attribute cannot be specified + if the value of the FollowParent attribute is 'yes' since that would ask the installer to force this feature + to follow the parent installation state and simultaneously favor a particular installation state just for this feature. + + + + + + + + Forces the feature to follow the same installation state as its parent feature. + + + + + + + Favors installing this feature locally by setting the msidbFeatureAttributesFavorLocal attribute. + + + + + + + Favors running this feature from source by setting the msidbFeatureAttributesFavorSource attribute. + + + + + + + + + + Sets the install level of this feature. A value of 0 will disable the feature. Processing the + Condition Table can modify the level value (this is set via the Condition child element). The + default value is "1". + + + + + + + Short string of text identifying the feature. This string is listed as an item by the + SelectionTree control of the Selection Dialog. + + + + + + + This attribute determines the default advertise state of the feature. + + + + + + + + Sets the feature to be advertised by setting the msidbFeatureAttributesFavorAdvertise attribute. + This value cannot be set if the value of the AllowAdvertise attribute is 'no' since that would ask the installer to + disallow the advertised state for this feature while at the same time favoring it. + + + + + + + Sets the feature to the default non-advertised installation option. + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + Groups together multiple components, features, and merges to be used in other locations. + + + + + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + Identifier for the FeatureGroup. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + Create a reference to a FeatureGroup in another Fragment. + + + + + + + + The identifier of the FeatureGroup to reference. + + + + + + Normally feature group references that end up nested under a parent element create a + connection to that parent. This behavior is undesirable when trying to simply reference + to a FeatureGroup in a different Fragment. Specify 'yes' to have this feature group + reference not create a connection to its parent. The default is 'no'. + + + + + + + Set this attribute to 'yes' in order to make the parent feature of this group + the primary feature for any components and merges contained in the group. + Features may belong to multiple features. By designating a feature as the + primary feature of a component or merge, you ensure that whenever a component is + selected for install-on-demand (IOD), the primary feature will be the one to install + it. This attribute should only be set if a component actually nests under multiple + features. If a component nests under only one feature, that feature is the primary + feature for the component. You cannot set more than one feature as the primary + feature of a given component. + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + Create a reference to a Feature element in another Fragment. + + + + + + + + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + The identifier of the Feature element to reference. + + + + + + Normally feature references that are nested under a parent element create a connection to that + parent. This behavior is undesirable when trying to simply reference a Feature in a different + Fragment. Specify 'yes' to have this feature reference not create a connection to its parent. + The default is 'no'. + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + Media element describes a disk that makes up the source media for the installation. + + + + + + + + + + + + + + + Disk identifier for Media table. This number must be equal to or greater than 1. + + + + + The name of the cabinet if some or all of the files stored on the media are in a cabinet file. If no cabinets are used, this attribute must not be set. + + + + + + Indicates the compression level for the Media's cabinet. This attribute can + only be used in conjunction with the Cabinet attribute. The default is 'mszip'. + + + + + + The disk name, which is usually the visible text printed on the disk. This localizable text is used to prompt the user when this disk needs to be inserted. This value will be used in the "[1]" of the DiskPrompt Property. Using this attribute will require you to define a DiskPrompt Property. + + + + + Instructs the binder to embed the cabinet in the product if 'yes'. This attribute can only be specified in conjunction with the Cabinet attribute. + + + + + + This attribute specifies the root directory for the uncompressed files that + are a part of this Media element. By default, the src will be the output + directory for the final image. The default value ensures the binder generates + an installable image. If a relative path is specified in the src attribute, + the value will be appended to the image's output directory. If an absolute + path is provided, that path will be used without modification. The latter two + options are provided to ease the layout of an image onto multiple medias (CDs/DVDs). + + + + + + + + + + + + + + The label attributed to the volume. This is the volume label returned + by the GetVolumeInformation function. If the SourceDir property refers + to a removable (floppy or CD-ROM) volume, then this volume label is + used to verify that the proper disk is in the drive before attempting + to install files. The entry in this column must match the volume label + of the physical media. + + + + + + + Optional property that identifies the source of the embedded cabinet. + If a cabinet is specified for a patch, this property should be defined + and unique to each patch so that the embedded cabinet containing patched + and new files can be located in the patch package. If the cabinet is not + embedded - this is not typical - the cabinet can be found in the directory + referenced in this column. If empty, the external cabinet must be located + in the SourceDir directory. + + + + + + + + + MediaTeplate element describes information to automatically assign files to cabinets. + A maximumum number of cabinets created is 999. + + + + + + + Templated name of the cabinet if some or all of the files stored on the media are in + a cabinet file. This name must begin with either a letter or an underscore, contain + maximum of five characters and {0} in the cabinet name part and must end three character extension. + The default is cab{0}.cab. + + + + + + + Indicates the compression level for the Media's cabinet. This attribute can + only be used in conjunction with the Cabinet attribute. The default is 'mszip'. + + + + + + + + + + + + + + + + The disk name, which is usually the visible text printed on the disk. This localizable text is used + to prompt the user when this disk needs to be inserted. This value will be used in the "[1]" of the + DiskPrompt Property. Using this attribute will require you to define a DiskPrompt Property. + + + + + + Instructs the binder to embed the cabinets in the product if 'yes'. + + + + + + The label attributed to the volume. This is the volume label returned + by the GetVolumeInformation function. If the SourceDir property refers + to a removable (floppy or CD-ROM) volume, then this volume label is + used to verify that the proper disk is in the drive before attempting + to install files. The entry in this column must match the volume label + of the physical media. + + + + + + + Size of uncompressed files in each cabinet, in megabytes. WIX_MUMS environment variable + can be used to override this value. Default value is 200 MB. + + + + + + + Maximum size of cabinet files in megabytes for large files. This attribute is used for packaging + files that are larger than MaximumUncompressedMediaSize into smaller cabinets. If cabinet size + exceed this value, then setting this attribute will cause the file to be split into multiple + cabinets of this maximum size. For simply controlling cabinet size without file splitting use + MaximumUncompressedMediaSize attribute. Setting this attribute will disable smart cabbing feature + for this Fragment / Product. Setting WIX_MCSLFS environment variable can be used to override this + value. Minimum allowed value of this attribute is 20 MB. Maximum allowed value and the Default + value of this attribute is 2048 MB (2 GB). + + + + + + + + + This element has been deprecated. + Use the Binary/@SuppressModularization, CustomAction/@SuppressModularization, or Property/@SuppressModularization attributes instead. + + + + + + + The name of the item to ignore modularization for. + + + + + + + The type of the item to ignore modularization for. + + + + + + + + + + + + + + + + + + + + + Specifies a custom action to be added to the MSI CustomAction table. Various combinations of the attributes for this element + correspond to different custom action types. For more information about custom actions see the + + Custom Action Types topic on MSDN. + + + + + + + + The text node is only valid if the Script attribute is specified. In that case, the text node contains the script to embed. + + + + + + The identifier of the custom action. + + + + + + + + This attribute is a reference to a Binary element with matching Id attribute. That binary stream contains + the custom action for use during install. The custom action will not be installed into a target directory. This attribute is + typically used with the DllEntry attribute to specify the custom action DLL to use for a type 1 custom action, with the ExeCommand + attribute to specify a type 17 custom action that runs an embedded executable, or with the VBScriptCall or JScriptCall attributes + to specify a type 5 or 6 custom action. + + + + + + + This attribute specifies a reference to a File element with matching Id attribute that + will execute the custom action code in the file after the file is installed. This + attribute is typically used with the ExeCommand attribute to specify a type 18 custom action + that runs an installed executable, with the DllEntry attribute to specify an installed custom + action DLL to use for a type 17 custom action, or with the VBScriptCall or JScriptCall + attributes to specify a type 21 or 22 custom action. + + + + + + + This attribute specifies a reference to a Property element with matching Id attribute that specifies the Property + to be used or updated on execution of this custom action. This attribute is + typically used with the Value attribute to create a type 51 custom action that parses + the text in Value and places it into the specified Property. This attribute is also used with + the ExeCommand attribute to create a type 50 custom action that uses the value of the + given property to specify the path to the executable. Type 51 custom actions are often useful to + pass values to a deferred custom action. + See + http://msdn.microsoft.com/library/aa370543.aspx + for more information. + + + + + + + This attribute specifies a reference to a Directory element with matching Id attribute containing a directory path. + This attribute is typically used with the ExeCommand attribute to specify the source executable for a type 34 + custom action, or with the Value attribute to specify a formatted string to place in the specified Directory + table entry in a type 35 custom action. + + + + + + + + This attribute specifies the name of a function in a custom action to execute. + This attribute is used with the BinaryKey attribute to create a type 1 custom + action, or with the FileKey attribute to create a type 17 custom action. + + + + + + + This attribute specifies the command line parameters to supply to an externally + run executable. This attribute is typically used with the BinaryKey attribute for a type 2 custom action, + the FileKey attribute for a type 18 custom action, the Property attribute for a type 50 custom action, + or the Directory attribute for a type 34 custom action that specify the executable to run. + + + + + + + This attribute specifies the name of the JScript function to execute in a script. The script must be + provided in a Binary element identified by the BinaryKey attribute described above. In other words, this + attribute must be specified in conjunction with the BinaryKey attribute. + + + + + + + This attribute specifies the name of the VBScript Subroutine to execute in a script. The script must be + provided in a Binary element identified by the BinaryKey attribute described above. In other words, this + attribute must be specified in conjunction with the BinaryKey attribute. + + + + + + + Creates a type 37 or 38 custom action. The text of the element should contain the script to be embedded in the package. + + + + + + + + + + + + + Use to suppress modularization of this custom action name in merge modules. + This should only be necessary for table-driven custom actions because the + table name which they interact with cannot be modularized, so there can only + be one instance of the table. + + + + + + + This attribute specifies a string value to use in the custom action. This attribute + must be used with the Property attribute to set the property as part of a + type 51 custom action or with the Directory attribute to set a directory path in that + table in a type 35 custom action. The value can be a literal value or derived from a + Property element using the Formatted + syntax. + + + + + + + This attribute specifies an index in the MSI Error table to use as an error message for a + type 19 custom action that displays the error message and aborts a product's installation. + + + + + + + + Set this attribute to set the return behavior of the custom action. + + + + + + + + Indicates that the custom action will run asyncronously and execution may continue after the installer terminates. + + + + + + + Indicates that the custom action will run asynchronously but the installer will wait for the return code at sequence end. + + + + + + + Indicates that the custom action will run synchronously and the return code will be checked for success. This is the default. + + + + + + + Indicates that the custom action will run synchronously and the return code will not be checked. + + + + + + + + + + This attribute indicates the scheduling of the custom action. + + + + + + + + Indicates that the custom action will run after successful completion of the installation script (at the end of the installation). + + + + + + + Indicates that the custom action runs in-script (possibly with elevated privileges). + + + + + + + Indicates that the custom action will only run in the first sequence that runs it. + + + + + + + Indicates that the custom action will run during normal processing time with user privileges. This is the default. + + + + + + + Indicates that the custom action will only run in the first sequence that runs it in the same process. + + + + + + + Indicates that a custom action will run in the rollback sequence when a failure + occurs during installation, usually to undo changes made by a deferred custom action. + + + + + + + Indicates that a custom action should be run a second time if it was previously run in an earlier sequence. + + + + + + + + + + This attribute specifies whether the Windows Installer, which executes as LocalSystem, + should impersonate the user context of the installing user when executing this custom action. + Typically the value should be 'yes', except when the custom action needs elevated privileges + to apply changes to the machine. + + + + + + + This attribute specifies that the Windows Installer, execute the custom action only when + a patch is being uninstalled. These custom actions should also be conditioned using the + MSIPATCHREMOVE property to ensure proper down level (less than Windows Installer 4.5) + behavior. + + + + + + + Specifies that a script custom action targets a 64-bit platform. Valid only when used with + the Script, VBScriptCall, and JScriptCall attributes. + The default value is based on the platform set by the -arch switch to candle.exe + or the InstallerPlatform property in a .wixproj MSBuild project: + For x86 and ARM, the default value is 'no'. + For x64 and IA64, the default value is 'yes'. + + + + + + + This attribute specifies controls whether the custom action will impersonate the + installing user during per-machine installs on Terminal Server machines. + Deferred execution custom actions that do not specify this attribute, or explicitly set it 'no', + will run with no user impersonation on Terminal Server machines during + per-machine installations. This attribute is only applicable when installing on the + Windows Server 2003 family. + + + + + + Ensures the installer does not log the CustomActionData for the deferred custom action. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + + + + + + This will cause the entire contents of the Fragment containing the referenced CustomAction to be + included in the installer database. + + + + + + The identifier of the CustomAction to reference. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + + + + + + + + Sets a Directory to a particular value. This is accomplished by creating a Type 51 custom action that is appropriately scheduled in + the InstallUISequence and InstallExecuteSequence. + + + + + + + + The condition that determines whether the Directory is set. If the condition evaluates to false, the SetDirectory is skipped. + + + + + + By default the action is "Set" + Id attribute's value. This optional attribute can override the action name in the case + where multiple SetDirectory elements target the same Id (probably with mutually exclusive conditions). + + + + + + + This attribute specifies a reference to a Directory element with matching Id attribute. The path of the Directory will be set to + the Value attribute. + + + + + + + Controls which sequences the Directory assignment is sequenced in. + For 'execute', the assignment is scheduled in the InstallExecuteSequence. + For 'ui', the assignment is scheduled in the InstallUISequence. + For 'first', the assignment is scheduled in the InstallUISequence or the InstallExecuteSequence if the InstallUISequence is skipped at install time. + For 'both', the assignment is scheduled in both the InstallUISequence and the InstallExecuteSequence. + The default is 'both'. + + + + + + + This attribute specifies a string value to assign to the Directory. The value can be a literal value or derived from a + Property element using the Formatted + syntax. + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + + + + + + + + + + Sets a Property to a particular value. This is accomplished by creating a Type 51 custom action that is appropriately scheduled in + the InstallUISequence and InstallExecuteSequence. + + + + + + + + The condition that determines whether the Property is set. If the condition evaluates to false, the Set is skipped. + + + + + + By default the action is "Set" + Id attribute's value. This optional attribute can override the action name in the case + where multiple SetProperty elements target the same Id (probably with mutually exclusive conditions). + + + + + + The name of the standard or custom action after which this action should be performed. Mutually exclusive with the Before attribute. A Before or After attribute is required when setting a Property. + + + + + The name of the standard or custom action before which this action should be performed. Mutually exclusive with the After attribute. A Before or After attribute is required when setting a Property. + + + + + + This attribute specifies the Property to set to the Value. + + + + + + + Controls which sequences the Property assignment is sequenced in. + For 'execute', the assignment is scheduled in the InstallExecuteSequence. + For 'ui', the assignment is scheduled in the InstallUISequence. + For 'first', the assignment is scheduled in the InstallUISequence or the InstallExecuteSequence if the InstallUISequence is skipped at install time. + For 'both', the assignment is scheduled in both the InstallUISequence and the InstallExecuteSequence. + The default is 'both'. + + + + + + + This attribute specifies a string value to assign to the Property. The value can be a literal value or derived from a + Property element using the Formatted + syntax. + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + + + + + + This will cause the entire contents of the Fragment containing the referenced PatchFamily to be + used in the process of creating a patch. + + + + + + The identifier of the PatchFamily to reference. + + + + + Specifies the ProductCode of the product that this family applies to. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + + + + Sets the ProductID property to the full product identifier. This action must be sequenced before the user interface wizard in the InstallUISequence table and before the RegisterUser action in the InstallExecuteSequence table. If the product identifier has already been validated successfully, the ValidateProductID action does nothing. The ValidateProductID action always returns a success, whether or not the product identifier is valid, so that the product identifier can be entered on the command line the first time the product is run. The product identifier can be validated without having the user reenter this information by setting the PIDKEY property on the command line or by using a transform. The display of the dialog box requesting the user to enter the product identifier can then be made conditional upon the presence of the ProductID property, which is set when the PIDKEY property is validated. The condition for this action may be specified in the element's inner text. + + + + + + + + + + Initiates the internal installation costing process. Any standard or custom actions that affect costing should be sequenced before the CostInitialize action. Call the FileCost action immediately following the CostInitialize action. Then call the CostFinalize action following the CostInitialize action to make all final cost calculations available to the installer through the Component table. The condition for this action may be specified in the element's inner text. + + + + + + + + + + Initiates dynamic costing of standard installation actions. Any standard or custom actions that affect costing should sequenced before the CostInitialize action. Call the FileCost action immediately following the CostInitialize action. Then call the CostFinalize action following the FileCost action to make all final cost calculations available to the installer through the Component table. The CostInitialize action must be executed before the FileCost action. The installer then determines the disk-space cost of every file in the File table, on a per-component basis, taking both volume clustering and the presence of existing files that may need to be overwritten into account. All actions that consume or release disk space are also considered. If an existing file is found, a file version check is performed to determine whether the new file actually needs to be installed or not. If the existing file is of an equal or greater version number, the existing file is not overwritten and no disk-space cost is incurred. In all cases, the installer uses the results of version number checking to set the installation state of each file. The FileCost action initializes cost calculation with the installer. Actual dynamic costing does not occur until the CostFinalize action is executed. The condition for this action may be specified in the element's inner text. + + + + + + + + + Installs a copy of a component (commonly a shared DLL) into a private location for use by a specific application (typically an .exe). This isolates the application from other copies of the component that may be installed to a shared location on the computer. The action refers to each record of the IsolatedComponent table and associates the files of the component listed in the Component_Shared field with the component listed in the Component_Application field. The installer installs the files of Component_Shared into the same directory as Component_Application. The installer generates a file in this directory, zero bytes in length, having the short filename name of the key file for Component_Application (typically this is the same file name as the .exe) appended with .local. The IsolatedComponent action does not affect the installation of Component_Application. Uninstalling Component_Application also removes the Component_Shared files and the .local file from the directory. The IsolateComponents action can be used only in the InstallUISequence table and the InstallExecuteSequence table. This action must come after the CostInitialize action and before the CostFinalize action. The condition for this action may be specified in the element's inner text. + + + + + + + + + + Ends the internal installation costing process begun by the CostInitialize action. Any standard or custom actions that affect costing should be sequenced before the CostInitialize action. Call the FileCost action immediately following the CostInitialize action and then call the CostFinalize action to make all final cost calculations available to the installer through the Component table. The CostFinalize action must be executed before starting any user interface sequence which allows the user to view or modify Feature table selections or directories. The CostFinalize action queries the Condition table to determine which features are scheduled to be installed. Costing is done for each component in the Component table. The CostFinalize action also verifies that all the target directories are writable before allowing the installation to continue. The condition for this action may be specified in the element's inner text. + + + + + + + + Checks for existing ODBC drivers and sets the target directory for each new driver to the location of an existing driver. The condition for this action may be specified in the element's inner text. + + + + + + + + Used for upgrading or installing over an existing application. Reads feature states from existing application and sets these feature states for the pending installation. The condition for this action may be specified in the element's inner text. + + + + + + + + Initiates the execution sequence. The condition for this action may be specified in the element's inner text. + + + + + + + + Verifies that all costed volumes have enough space for the installation. The condition for this action may be specified in the element's inner text. + + + + + + + + + Marks the beginning of a sequence of actions that change the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Ensures the needed amount of space exists in the registry. The condition for this action may be specified in the element's inner text. + + + + + + + + Registers and unregisters components, their key paths, and the component clients. The condition for this action may be specified in the element's inner text. + + + + + + + + Manages the unadvertisement of components listed in the PublishComponent table. The condition for this action may be specified in the element's inner text. + + + + + + + + Manages the unadvertisement of CLR and Win32 assemblies that are being removed. The condition for this action may be specified in the element's inner text. + + + + + + + + Removes selection-state and feature-component mapping information from the registry. The condition for this action may be specified in the element's inner text. + + + + + + + + Stops system services. The condition for this action may be specified in the element's inner text. + + + + + + + + Stops a service and removes its registration from the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Removes COM+ applications from the registry. The condition for this action may be specified in the element's inner text. + + + + + + + + Unregisters all modules listed in the SelfReg table that are scheduled to be uninstalled. The condition for this action may be specified in the element's inner text. + + + + + + + + Unregisters type libraries from the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Removes the data sources, translators, and drivers listed for removal during the installation. The condition for this action may be specified in the element's inner text. + + + + + + + + Removes registration information about installed fonts from the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Removes a registry value that has been authored into the registry table if the associated component was installed locally or as run from source, and is now set to be uninstalled. The condition for this action may be specified in the element's inner text. + + + + + + + + Manages the removal of COM class information from the system registry. The condition for this action may be specified in the element's inner text. + + + + + + + + Manages the removal of extension-related information from the system registry. The condition for this action may be specified in the element's inner text. + + + + + + + + Manages the unregistration of OLE ProgId information with the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Unregisters MIME-related registry information from the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Removes .ini file information specified for removal in the RemoveIniFile table if the component is set to be installed locally or run from source. The condition for this action may be specified in the element's inner text. + + + + + + + + Manages the removal of an advertised shortcut whose feature is selected for uninstallation or a nonadvertised shortcut whose component is selected for uninstallation. The condition for this action may be specified in the element's inner text. + + + + + + + + Modifies the values of environment variables. The condition for this action may be specified in the element's inner text. + + + + + + + + Deletes files installed by the DuplicateFiles action. The condition for this action may be specified in the element's inner text. + + + + + + + + Removes files previously installed by the InstallFiles action. The condition for this action may be specified in the element's inner text. + + + + + + + + Removes any folders linked to components set to be removed or run from source. The condition for this action may be specified in the element's inner text. + + + + + + + + Creates empty folders for components that are set to be installed. The condition for this action may be specified in the element's inner text. + + + + + + + + + Locates existing files on the system and moves or copies those files to a new location. The condition for this action may be specified in the element's inner text. + + + + + + + + Copies the product database to the administrative installation point. The condition for this action may be specified in the element's inner text. + + + + + + + + Copies files specified in the File table from the source directory to the destination directory. The condition for this action may be specified in the element's inner text. + + + + + + + + Duplicates files installed by the InstallFiles action. The condition for this action may be specified in the element's inner text. + + + + + + + + Queries the Patch table to determine which patches are to be applied. The condition for this action may be specified in the element's inner text. + + + + + + + + + Binds each executable or DLL that must be bound to the DLLs imported by it. The condition for this action may be specified in the element's inner text. + + + + + + + + Manages the creation of shortcuts. The condition for this action may be specified in the element's inner text. + + + + + + + + Manages the registration of COM class information with the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Manages the registration of extension related information with the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Manages the registration of OLE ProgId information with the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Registers MIME-related registry information with the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Sets up an application's registry information. The condition for this action may be specified in the element's inner text. + + + + + + + + Writes the .ini file information that the application needs written to its .ini files. The condition for this action may be specified in the element's inner text. + + + + + + + + Modifies the values of environment variables. The condition for this action may be specified in the element's inner text. + + + + + + + + Registers installed fonts with the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Installs the drivers, translators, and data sources in the ODBCDriver table, ODBCTranslator table, and ODBCDataSource table. The condition for this action may be specified in the element's inner text. + + + + + + + + Registers type libraries with the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Processes all modules listed in the SelfReg table and registers all installed modules with the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Registers COM+ applications. The condition for this action may be specified in the element's inner text. + + + + + + + + Registers a service for the system. The condition for this action may be specified in the element's inner text. + + + + + + + + Starts system services. The condition for this action may be specified in the element's inner text. + + + + + + + + Registers the user information with the installer to identify the user of a product. The condition for this action may be specified in the element's inner text. + + + + + + + + Registers the product information with the installer. The condition for this action may be specified in the element's inner text. + + + + + + + + Manages the advertisement of the components from the PublishComponent table. The condition for this action may be specified in the element's inner text. + + + + + + + + Manages the advertisement of CLR and Win32 assemblies. The condition for this action may be specified in the element's inner text. + + + + + + + + Writes each feature's state into the system registry. The condition for this action may be specified in the element's inner text. + + + + + + + + Manages the advertisement of the product information with the system. The condition for this action may be specified in the element's inner text. + + + + + + + + + Marks the end of a sequence of actions that change the system. The condition for this action may be specified in the element's inner text. + + + + + + + + + + + + + Uses file signatures to search for existing versions of products. The AppSearch action may use this information to determine where upgrades are to be installed. The AppSearch action can also be used to set a property to the existing value of an registry or .ini file entry. AppSearch should be authored into the InstallUISequence table and InstallExecuteSequence table. The installer prevents The AppSearch action from running in the InstallExecuteSequence sequence if the action has already run in InstallUISequence sequence. The AppSearch action searches for file signatures using the CompLocator table first, the RegLocator table next, then the IniLocator table, and finally the DrLocator table. The condition for this action may be specified in the element's inner text. + + + + + + + + + + Uses file signatures to validate that qualifying products are installed on a system before an upgrade installation is performed. The CCPSearch action should be authored into the InstallUISequence table and InstallExecuteSequence table. The installer prevents the CCPSearch action from running in the InstallExecuteSequence sequence if the action has already run in InstallUISequence sequence. The CCPSearch action must come before the RMCCPSearch action. The condition for this action may be specified in the element's inner text. + + + + + + + + + + Uses file signatures to validate that qualifying products are installed on a system before an upgrade installation is performed. The RMCCPSearch action should be authored into the InstallUISequence table and InstallExecuteSequence table. The installer prevents RMCCPSearch from running in the InstallExecuteSequence sequence if the action has already run in InstallUISequence sequence. The RMCCPSearch action requires the CCP_DRIVE property to be set to the root path on the removable volume that has the installation for any of the qualifying products. The condition for this action may be specified in the element's inner text. + + + + + + + + + Queries the LaunchCondition table and evaluates each conditional statement recorded there. If any of these conditional statements fail, an error message is displayed to the user and the installation is terminated. The LaunchConditions action is optional. This action is normally the first in the sequence, but the AppSearch Action may be sequenced before the LaunchConditions action. If there are launch conditions that do not apply to all installation modes, the appropriate installation mode property should be used in a conditional expression in the appropriate sequence table. The condition for this action may be specified in the element's inner text. + + + + + + + + + Runs through each record of the Upgrade table in sequence and compares the upgrade code, product version, and language in each row to products installed on the system. When FindRelatedProducts detects a correspondence between the upgrade information and an installed product, it appends the product code to the property specified in the ActionProperty column of the UpgradeTable. The FindRelatedProducts action only runs the first time the product is installed. The FindRelatedProducts action does not run during maintenance mode or uninstallation. FindRelatedProducts should be authored into the InstallUISequence table and InstallExecuteSequence tables. The installer prevents FindRelatedProducts from running in InstallExecuteSequence if the action has already run in InstallUISequence. The FindRelatedProducts action must come before the MigrateFeatureStates action and the RemoveExistingProducts action. The condition for this action may be specified in the element's inner text. + + + + + + + + Runs a script containing all operations spooled since either the start of the installation or the last InstallExecute action, or InstallExecuteAgain action. Special actions don't have a built-in sequence number and thus must appear relative to another action. The suggested way to do this is by using the Before or After attribute. InstallExecute and InstallExecuteAgain can optionally appear anywhere between InstallInitialize and InstallFinalize. + + + + + + + + Runs a script containing all operations spooled since either the start of the installation or the last InstallExecute action, or InstallExecuteAgain action. Should only be used after InstallExecute. Special actions don't have a built-in sequence number and thus must appear relative to another action. The suggested way to do this is by using the Before or After attribute. InstallExecute and InstallExecuteAgain can optionally appear anywhere between InstallInitialize and InstallFinalize. + + + + + + + + Disables rollback for the remainder of the installation. Special actions don't have a built-in sequence number and thus must appear relative to another action. The suggested way to do this is by using the Before or After attribute. InstallExecute and InstallExecuteAgain can optionally appear anywhere between InstallInitialize and InstallFinalize. + + + + + + + + Goes through the product codes listed in the ActionProperty column of the Upgrade table and removes the products in sequence. Special actions don't have a built-in sequence number and thus must appear relative to another action. The suggested way to do this is by using the Before or After attribute. InstallExecute and InstallExecuteAgain can optionally appear anywhere between InstallInitialize and InstallFinalize. + + + + + + + + Prompts the user to restart the system at the end of installation. Special actions don't have a built-in sequence number and thus must appear relative to another action. The suggested way to do this is by using the Before or After attribute. InstallExecute and InstallExecuteAgain can optionally appear anywhere between InstallInitialize and InstallFinalize. + + + + + + + + Prompts the user for a restart of the system during the installation. Special actions don't have a built-in sequence number and thus must appear relative to another action. The suggested way to do this is by using the Before or After attribute. InstallExecute and InstallExecuteAgain can optionally appear anywhere between InstallInitialize and InstallFinalize. + + + + + + + + Determines the location of the source and sets the SourceDir property if the source has not been resolved yet. Special actions don't have a built-in sequence number and thus must appear relative to another action. The suggested way to do this is by using the Before or After attribute. InstallExecute and InstallExecuteAgain can optionally appear anywhere between InstallInitialize and InstallFinalize. + + + + + + + + + Use to sequence a custom action. + + + + + + Text node specifies the condition of the action. + + + + The action to which the Custom element applies. + + + + + Mutually exclusive with Before, After, and Sequence attributes + + + + + The name of the standard or custom action before which this action should be performed. Mutually exclusive with OnExit, After, and Sequence attributes + + + + + The name of the standard or custom action after which this action should be performed. Mutually exclusive with Before, OnExit, and Sequence attributes + + + + + + If "yes", the sequencing of this action may be overridden by sequencing elsewhere. + + + + + + The sequence number for this action. Mutually exclusive with Before, After, and OnExit attributes + + + + + + + + + + + + + + mutually exclusive with Before, After, and Sequence attributes + + + + + + + + If "yes", the sequencing of this dialog may be overridden by sequencing elsewhere. + + + + + + + + + + + + + + + + + + + + Use to sequence a custom action. + + + + + Displays a Dialog. + + + + + Prompts the user to restart the system at the end of installation. Not fixed sequence. + + + + + Queries the LaunchCondition table and evaluates each conditional statement recorded there. + + + + + Runs through each record of the Upgrade table in sequence and compares the upgrade code, product version, and language in each row to products installed on the system. + + + + + Uses file signatures to search for existing versions of products. + + + + + Uses file signatures to validate that qualifying products are installed on a system before an upgrade installation is performed. + + + + + Uses file signatures to validate that qualifying products are installed on a system before an upgrade installation is performed. + + + + + Sets the ProductID property to the full product identifier. + + + + + Initiates the internal installation costing process. + + + + + Initiates dynamic costing of standard installation actions. + + + + + Installs a copy of a component (commonly a shared DLL) into a private location for use by a specific application (typically an .exe). + + + + + Determines the location of the source and sets the SourceDir property if the source has not been resolved yet. + + + + + Ends the internal installation costing process begun by the CostInitialize action. + + + + + Used for upgrading or installing over an existing application. + + + + + Initiates the execution sequence. + + + + + + + + + + + + + + + + + Use to sequence a custom action. + + + + + Prompts the user to restart the system at the end of installation. Not fixed sequence. + + + + + Prompts the user for a restart of the system during the installation. Not fixed sequence. + + + + + Determines the location of the source and sets the SourceDir property if the source has not been resolved yet. Not fixed sequence. + + + + + Queries the LaunchCondition table and evaluates each conditional statement recorded there. + + + + + Runs through each record of the Upgrade table in sequence and compares the upgrade code, product version, and language in each row to products installed on the system. + + + + + Uses file signatures to search for existing versions of products. + + + + + Uses file signatures to validate that qualifying products are installed on a system before an upgrade installation is performed. + + + + + Uses file signatures to validate that qualifying products are installed on a system before an upgrade installation is performed. + + + + + Sets the ProductID property to the full product identifier. + + + + + Initiates the internal installation costing process. + + + + + Initiates dynamic costing of standard installation actions. + + + + + Installs a copy of a component (commonly a shared DLL) into a private location for use by a specific application (typically an .exe). + + + + + Ends the internal installation costing process begun by the CostInitialize action. + + + + + Checks for existing ODBC drivers and sets the target directory for each new driver to the location of an existing driver. + + + + + Used for upgrading or installing over an existing application. + + + + + Verifies that all costed volumes have enough space for the installation. + + + + + Marks the beginning of a sequence of actions that change the system. + + + + + Ensures the needed amount of space exists in the registry. + + + + + Registers and unregisters components, their key paths, and the component clients. + + + + + Manages the unadvertisement of components listed in the PublishComponent table. + + + + + Removes selection-state and feature-component mapping information from the registry. + + + + + Stops system services. + + + + + Stops a service and removes its registration from the system. + + + + + Removes COM+ applications from the registry. + + + + + Unregisters all modules listed in the SelfReg table that are scheduled to be uninstalled. + + + + + Unregisters type libraries from the system. + + + + + Removes the data sources, translators, and drivers listed for removal during the installation. + + + + + Removes registration information about installed fonts from the system. + + + + + Removes a registry value that has been authored into the registry table if the associated component was installed locally or as run from source, and is now set to be uninstalled. + + + + + Manages the removal of COM class information from the system registry. + + + + + Manages the removal of extension-related information from the system registry. + + + + + Manages the unregistration of OLE ProgId information with the system. + + + + + Unregisters MIME-related registry information from the system. + + + + + Removes .ini file information specified for removal in the RemoveIniFile table if the component is set to be installed locally or run from source. + + + + + Manages the removal of an advertised shortcut whose feature is selected for uninstallation or a nonadvertised shortcut whose component is selected for uninstallation. + + + + + Modifies the values of environment variables. + + + + + Deletes files installed by the DuplicateFiles action. + + + + + Removes files previously installed by the InstallFiles action. + + + + + Removes any folders linked to components set to be removed or run from source. + + + + + Creates empty folders for components that are set to be installed. + + + + + Locates existing files on the system and moves or copies those files to a new location. + + + + + Copies files specified in the File table from the source directory to the destination directory. + + + + + Duplicates files installed by the InstallFiles action. + + + + + Queries the Patch table to determine which patches are to be applied. + + + + + Binds each executable or DLL that must be bound to the DLLs imported by it. + + + + + Manages the creation of shortcuts. + + + + + Manages the registration of COM class information with the system. + + + + + Manages the registration of extension related information with the system. + + + + + Manages the registration of OLE ProgId information with the system. + + + + + Registers MIME-related registry information with the system. + + + + + Sets up an application's registry information. + + + + + Writes the .ini file information that the application needs written to its .ini files. + + + + + Modifies the values of environment variables. + + + + + Registers installed fonts with the system. + + + + + Installs the drivers, translators, and data sources in the ODBCDriver table, ODBCTranslator table, and ODBCDataSource table. + + + + + Registers type libraries with the system. + + + + + Processes all modules listed in the SelfReg table and registers all installed modules with the system. + + + + + Registers COM+ applications. + + + + + Registers a service for the system. + + + + + Starts system services. + + + + + Registers the user information with the installer to identify the user of a product. + + + + + Registers the product information with the installer. + + + + + Manages the advertisement of the components from the PublishComponent table. + + + + + Writes each feature's state into the system registry. + + + + + Manages the advertisement of the product information with the system. + + + + + Marks the end of a sequence of actions that change the system. + + + + + Goes through the product codes listed in the ActionProperty column of the Upgrade table and removes the products in sequence. + + + + + Disables rollback for the remainder of the installation. + + + + + + Runs a script containing all operations spooled since either the start of the installation or the last InstallExecute action, or InstallExecuteAgain action. + + + + + Runs a script containing all operations spooled since either the start of the installation or the last InstallExecute action, or InstallExecuteAgain action. + + + + + + Manages the advertisement of CLR and Win32 assemblies. + + + + + Manages the unadvertisement of CLR and Win32 assemblies that are being removed. + + + + + + + + + + + + + + + + Use to sequence a custom action. + + + + + + Initiates the internal installation costing process. + + + + + Initiates dynamic costing of standard installation actions. + + + + + Ends the internal installation costing process begun by the CostInitialize action. + + + + + Initiates the execution sequence. + + + + + Verifies that all costed volumes have enough space for the installation. + + + + + Marks the beginning of a sequence of actions that change the system. + + + + + Copies the product database to the administrative installation point. + + + + + Copies files specified in the File table from the source directory to the destination directory. + + + + + Marks the end of a sequence of actions that change the system. + + + + + Queries the LaunchCondition table and evaluates each conditional statement recorded there. + + + + + + + + + + + + + + + + Use to sequence a custom action. + + + + + Initiates the internal installation costing process. + + + + + Initiates dynamic costing of standard installation actions. + + + + + Ends the internal installation costing process begun by the CostInitialize action. + + + + + Verifies that all costed volumes have enough space for the installation. + + + + + Marks the beginning of a sequence of actions that change the system. + + + + + Copies the product database to the administrative installation point. + + + + + Copies files specified in the File table from the source directory to the destination directory. + + + + + Queries the Patch table to determine which patches are to be applied. + + + + + Marks the end of a sequence of actions that change the system. + + + + + Queries the LaunchCondition table and evaluates each conditional statement recorded there. + + + + + Determines the location of the source and sets the SourceDir property if the source has not been resolved yet. + + + + + + + + + + + + + + + + Initiates the internal installation costing process. + + + + + Ends the internal installation costing process begun by the CostInitialize action. + + + + + Use to sequence a custom action. The only custom actions that are allowed in the AdvtExecuteSequence are type 19 (0x013) type 35 (0x023) and type 51 (0x033). + + + + + Verifies that all costed volumes have enough space for the installation. + + + + + Marks the beginning of a sequence of actions that change the system. + + + + + Manages the creation of shortcuts. + + + + + Manages the registration of COM class information with the system. + + + + + Manages the registration of extension related information with the system. + + + + + Registers MIME-related registry information with the system. + + + + + Manages the registration of OLE ProgId information with the system. + + + + + Manages the advertisement of the components from the PublishComponent table. + + + + + Writes each feature's state into the system registry. + + + + + Manages the advertisement of the product information with the system. + + + + + Marks the end of a sequence of actions that change the system. + + + + + + Manages the advertisement of CLR and Win32 assemblies. + + + + + + + + + + Binary data used for CustomAction elements and UI controls. + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + elements at this point in the schema. + + + + + + + The Id cannot be longer than 55 characters. In order to prevent errors in cases where the Id is modularized, it should not be longer than 18 characters. + + + + + Path to the binary file. + + + + + + + + + + + + + Use to suppress modularization of this Binary identifier in merge modules. + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + Icon used for Shortcut, ProgId, or Class elements (but not UI controls) + + + + How To: Set your installer's icon in Add/Remove Programs + How To: Create a shortcut on the Start Menu + + + + + + The Id cannot be longer than 55 characters. In order to prevent errors in cases where the Id is modularized, it should not be longer than 18 characters. + + + + + Path to the icon file. + + + + + + + + + + + + + + + + + + + + + + + + + + + Element value is the condition. CDATA may be used to when a condition contains many XML characters + that must be escaped. It is important to note that each EmbeddedChainer element must have a mutually exclusive condition + to ensure that only one embedded chainer will execute at a time. If the conditions are not mutually exclusive the chainer + that executes is undeterministic. + + + + + Unique identifier for embedded chainer. + + + + + Value to append to the transaction handle and passed to the chainer executable. + + + + + + Reference to the Binary element that contains the chainer executable. Mutually exclusive with + the FileSource and PropertySource attributes. + + + + + + + Reference to the File element that is the chainer executable. Mutually exclusive with + the BinarySource and PropertySource attributes. + + + + + + + Reference to a Property that resolves to the full path to the chainer executable. Mutually exclusive with + the BinarySource and FileSource attributes. + + + + + + + + + + + Reference to an EmbeddedChainer element. This will force the entire referenced Fragment's contents + to be included in the installer database. + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + + + + Element value is the condition. Use CDATA if message contains delimiter characters. + + + + + + Specifies extra files to be extracted for use by the embedded UI, such as language resources. + + + + + + + Unique identifier for embedded UI.If this attribute is not specified the Name attribute or the file name + portion of the SourceFile attribute will be used. + + + + + + Embedded UI will not recieve any INSTALLLOGMODE_FATALEXIT messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_ERROR messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_WARNING messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_USER messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_INFO messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_FILESINUSE messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_RESOLVESOURCE messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_OUTOFDISKSPACE messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_ACTIONSTART messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_ACTIONDATA messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_PROGRESS messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_COMMONDATA messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_INITIALIZE messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_TERMINATE messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_SHOWDIALOG messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_RMFILESINUSE messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_INSTALLSTART messages. + + + + + Embedded UI will not recieve any INSTALLLOGMODE_INSTALLEND messages. + + + + + + The name for the embedded UI DLL when it is extracted from the Product and executed. (Windows Installer + does not support the typical short filename and long filename combination for embedded UI files as it + does for other kinds of files.) If this attribute is not specified the file name portion of the SourceFile + attribute will be used. + + + + + + + Path to the binary file that is the embedded UI. This must be a DLL that exports the following + three entry points: InitializeEmbeddedUI, EmbeddedUIHandler and ShutdownEmbeddedUI. + + + + + + Set yes to allow the Windows Installer to display the embedded UI during basic UI level installation. + + + + + + + + Defines a resource for use by the embedded UI. + + + + + + + + + + Identifier for the embedded UI resource. + + + + + + The name for the resource when it is extracted from the Product for use by the embedded UI DLL. (Windows + Installer does not support the typical short filename and long filename combination for embedded UI files + as it does for other kinds of files.) If this attribute is not specified the Id attribute will be used. + + + + + + Path to the binary file that is the embedded UI resource. + + + + + + + + + + + + + + + Element value is Message, use CDATA if message contains delimiter characters + + + + Number of the error for which a message is being provided. See MSI SDK for error definitions. + + + + + + + + + + + + + + + + + The element value is the optional Condition expression. + + + + + The parent Control for this Publish element, should only be specified when this element is a child of the UI element. + + + + + + + The parent Dialog for this Publish element, should only be specified when this element is a child of the UI element. + This attribute will create a reference to the specified Dialog, so an additional DialogRef is not necessary. + + + + + + + Set this attribute's value to one of the standard control events to trigger that event. + Either this attribute or the Property attribute must be set, but not both at the same time. + + + + + + + This attribute should only need to be set if this element is nested under a UI element in order to + control the order in which this publish event will be started. + If this element is nested under a Control element, the default value will be one greater than any + previous Publish element's order (the first element's default value is 1). + If this element is nested under a UI element, the default value is always 1 (it does not get a + default value based on any previous Publish elements). + + + + + + + Set this attribute's value to a property name to set that property. + Either this attribute or the Event attribute must be set, but not both at the same time. + + + + + + + If the Property attribute is specified, set the value of this attribute to the new value for the property. + To set a property to null, do not set this attribute (the ControlEvent Argument column will be set to '{}'). + Otherwise, this attribute's value should be the argument for the event specified in the Event attribute. + If the event doesn't take an attribute, a common value to use is "0". + + + + + + + + + + + Sets attributes for events in the EventMapping table + + + + + + + + + must be one of the standard control events' + + + + + if not present can only handle enable, disable, hide, unhide events + + + + + + + + An alternative to using the Text attribute when the value contains special XML characters like <, >, or &. + + + + + + + + Instructs the text to be imported from a file instead of the element value during the binding process. + + + + + + + + + + + + + + + + + + + + + + + + Contains the controls that appear on each dialog. + + + + + + alternative to Text attribute when CDATA is needed to escape XML delimiters + + + + + ComboBox table with ListItem children + + + + + ListBox table with ListItem children + + + + + ListView table with ListItem children + + + + + RadioButton table with RadioButton children + + + + + Property table entry for the Property table column associated with this control + + + + + Icon referenced in icon column of row + + + + + child elements affecting operation of this control + + + + Condition to specify actions for this control based on the outcome of the condition. + + + + + + + + + Combined with the Dialog Id to make up the primary key of the Control table. + + + + + The type of the control. Could be one of the following: Billboard, Bitmap, CheckBox, ComboBox, DirectoryCombo, DirectoryList, Edit, GroupBox, Hyperlink, Icon, Line, ListBox, ListView, MaskedEdit, PathEdit, ProgressBar, PushButton, RadioButtonGroup, ScrollableText, SelectionTree, Text, VolumeCostList, VolumeSelectCombo + + + + + Horizontal coordinate of the upper-left corner of the rectangular boundary of the control. This must be a non-negative number. + + + + + Vertical coordinate of the upper-left corner of the rectangular boundary of the control. This must be a non-negative number. + + + + + Width of the rectangular boundary of the control. This must be a non-negative number. + + + + + Height of the rectangular boundary of the control. This must be a non-negative number. + + + + + The name of a defined property to be linked to this control. This column is required for active controls. + + + + + A localizable string used to set the initial text contained in a control. This attribute can contain a formatted string that is processed at install time to insert the values of properties using [PropertyName] syntax. Also supported are environment variables, file installation paths, and component installation directories; see Formatted for details. + + + + + This attribute is reserved for future use. There is no need to use this until Windows Installer uses it for something. + + + + + The string used for the Tooltip. + + + + + This attribute is only valid for CheckBox Controls. When set, the linked Property will be set to this value when the check box is checked. + + + + + This attribute is only valid for CheckBox controls. The value is the name of a Property that was already used as the Property for another CheckBox control. The Property attribute cannot be specified. The attribute exists to support multiple checkboxes on different dialogs being tied to the same property. + + + + + + Set this attribute to "yes" to cause this Control to be skipped in the tab sequence. + + + + + Set this attribute to "yes" to cause this Control to be invoked by the return key. + + + + + Set this attribute to "yes" to cause this Control to be invoked by the escape key. + + + + + + Set this attribute to "yes" to cause the Control to be hidden. + + + + + Set this attribute to "yes" to cause the Control to be disabled. + + + + + Set this attribute to "yes" to cause the Control to be sunken. + + + + + Specifies whether the value displayed or changed by this control is referenced indirectly. If this bit is set, the control displays or changes the value of the property that has the identifier listed in the Property column of the Control table. + + + + + Set this attribute to "yes" to cause the linked Property value for the Control to be treated as an integer. Otherwise, the Property will be treated as a string. + + + + + Set this attribute to "yes" to cause the Control to display from right to left. + + + + + Set this attribute to "yes" to cause the Control to be right aligned. + + + + + Set this attribute to "yes" to cause the scroll bar to display on the left side of the Control. + + + + + + This attribute is only valid for Text Controls. + + + + + This attribute is only valid for Text Controls. + + + + + This attribute is only valid for Text Controls. + + + + + This attribute is only valid for Text Controls. + + + + + This attribute is only valid for Text Controls. + + + + + This attribute is only valid for Edit Controls. + + + + + This attribute is only valid for Edit Controls. + + + + + This attribute is only valid for ProgressBar Controls. + + + + + This attribute is only valid for Volume and Directory Controls. + + + + + This attribute is only valid for Volume and Directory Controls. + + + + + This attribute is only valid for Volume and Directory Controls. + + + + + This attribute is only valid for Volume and Directory Controls. + + + + + This attribute is only valid for Volume and Directory Controls. + + + + + This attribute is only valid for Volume and Directory Controls. + + + + + This attribute is only valid for VolumeCostList Controls. + + + + + This attribute is only valid for ListBox, ListView, and ComboBox Controls. Set + the value of this attribute to "yes" to have entries appear in the order specified under the Control. + If the attribute value is "no" or absent the entries in the control will appear in alphabetical order. + + + + + + This attribute is only valid for ComboBox Controls. + + + + + This attribute is only valid for RadioButton, PushButton, and Icon Controls. + + + + + This attribute is only valid for RadioButton, PushButton, and Icon Controls. + + + + + + + + + + + + This attribute is only valid for RadioButton, PushButton, and Icon Controls. + + + + + This attribute is only valid for RadioButton and PushButton Controls. + + + + + This attribute is only valid for RadioButton and PushButton Controls. + + + + + This attribute is only valid for RadioButton and Checkbox Controls. + + + + + This attribute is only valid for RadioButton Controls. + + + + + + This attribute is only valid for PushButton controls. + Set this attribute to "yes" to add the User Account Control (UAC) elevation icon (shield icon) to the PushButton control. + If this attribute's value is "yes" and the installation is not yet running with elevated privileges, + the pushbutton control is created using the User Account Control (UAC) elevation icon (shield icon). + If this attribute's value is "yes" and the installation is already running with elevated privileges, + the pushbutton control is created using the other icon attributes. + Otherwise, the pushbutton control is created using the other icon attributes. + + + + + + + + + Billboard to display during install of a Feature + + + + + + + + + + + Only controls of static type such as: Text, Bitmap, Icon, or custom control can be placed on a billboard. + + + + + + Unique identifier for the Billboard. + + + + + Feature whose state determines if the Billboard is shown. + + + + + + + + Billboard action during which child Billboards are displayed + + + + + + + + + + + Order of Billboard elements determines order of display + + + + + + Action name that determines when the Billboard should be shown. + + + + + + + + + + + + + + + + Defines a dialog box in the Dialog Table. + + + + + + + Control elements belonging to this dialog. + + + + + + Unique identifier for the dialog. + + + + + Horizontal placement of the dialog box as a percentage of screen width. The default value is 50. + + + + + Vertical placement of the dialog box as a percentage of screen height. The default value is 50. + + + + + The width of the dialog box in dialog units. + + + + + The height of the dialog box in dialog units. + + + + + The title of the dialog box. + + + + + + Used to hide the dialog. + + + + + Used to set the dialog as modeless. + + + + + Used to specify if the dialog can be minimized. + + + + + Used to set the dialog as system modal. + + + + + Keep modeless dialogs alive when this dialog is created through DoAction. + + + + + Have the dialog periodically call the installer to check if available disk space has changed. + + + + + Used to specify if pictures in the dialog box are rendered with a custom palette. + + + + + Used to specify if the text in the dialog should be displayed in right to left reading order. + + + + + Align text on the right. + + + + + Used to align the scroll bar on the left. + + + + + Specifies this dialog as an error dialog. + + + + + + + + + + + Reference to a Dialog. This will cause the entire referenced section's contents + to be included in the installer database. + + + + + + The identifier of the Dialog to reference. + + + + + + + + + + + + + + + Element value is progress message text for action + + + + + used to format ActionData messages from action processing + + + + + + + + + + + + + + + + + + + 0 to 255 + + + + + 0 to 255 + + + + + 0 to 255 + + + + + + + + + + + + The value (and optional text) associated with an item in a ComboBox, ListBox, or ListView. + + + + + + + + + + + + The value assigned to the associated ComboBox, ListBox, or ListView property if this item is selected. + + + + + + + The localizable, visible text to be assigned to the item. + If not specified, this will default to the value of the Value attribute. + + + + + + + The identifier of the Binary (not Icon) element containing the icon to associate with this item. + This value is only valid when nested under a ListView element. + + + + + + + + + Set of items for a particular ListBox control tied to an install Property + + + + + + + + + + + + entry for ListBox table + + + + + + Property tied to this group + + + + + + + + Set of items for a particular ComboBox control tied to an install Property + + + + + + + + + + + + entry for ComboBox table + + + + + + Property tied to this group + + + + + + + + Set of items for a particular ListView control tied to an install Property + + + + + + + + + + + + entry for ListView table + + + + + + Property tied to this group + + + + + + + + + + + + + Text or Icon plus Value that is assigned to the Property of the parent Control (RadioButtonGroup). + + + + + + This attribute defines the bitmap displayed with the radio button. The value of the attribute creates a reference + to a Binary element that represents the bitmap. This attribute is mutually exclusive with the Icon and Text + attributes. + + + + + + + + + This attribute defines the icon displayed with the radio button. The value of the attribute creates a reference + to a Binary element that represents the icon. This attribute is mutually exclusive with the Bitmap and Text + attributes. + + + + + + Text displayed with the radio button. This attribute is mutually exclusive with the Bitmap and Icon attributes. + + + + + + Value assigned to the associated control Property when this radio button is selected. + + + + + + + + + + + Set of radio buttons tied to the specified Property + + + + + + + + + + + + + + Property tied to this group. + + + + + + + + Text associated with certain controls + + + + + + + + + + Element value is text, may use CDATA if needed to escape XML delimiters + + + + + + + + + + Reference to a UI element. This will force the entire referenced Fragment's contents + to be included in the installer database. + + + + + + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + Enclosing element to compartmentalize UI specifications. + + + + + + + + + + Embedded UI definition with EmbeddedResource children. + + + + + Error text associated with install error + + + + + ActionText entry associated with an action + + + + + Billboard table item with child Controls + + + + + ComboBox table with ListItem children + + + + + ListBox table with ListItem children + + + + + ListView table with ListItem children + + + + + RadioButton table with RadioButton children + + + + + TextStyle entry for use in control text + + + + + values for UIText property, not installer Property + + + + + Dialog specification, called from Sequence + + + + + Reference to a Dialog specification. + + + + + + + + + + + + + + + + + + + + Defines a custom table for use from a custom action. + + + + + + Column definition for the custom table. + + + + + Row definition for the custom table. + + + + + + Identifier for the custom table. + + + + + Indicates the table data is transformed into the bootstrapper application data manifest. + + + + + + + Column definition for a Custom Table + + + + + Identifier for the column. + + + + + Whether this column is a primary key. + + + + + The type of this column. + + + + + + + Column contains a path to a file that will be inserted into the column as a binary object. + If this value is set, the Category attribute must also be set with a value of 'Binary' to pass ICE validation. + + + + + + + Column contains an integer or datetime value (the MinValue and MaxValue attributes should also be set). + + + + + + + Column contains a non-localizable string value. + + + + + + + + + Width of this column. + + + + + Whether this column can be left null. + + + + + Whether this column can be localized. + + + + + Minimum value for a numeric value, date or version in this column. + + + + + Maximum value for a numeric value, date or version in this column. + + + + + Table in which this column is an external key. Can be semicolon delimited. + + + + + Column in the table in KeyTable attribute. + + + + + + Category of this column. + This attribute must be specified with a value of 'Binary' if the Type attribute's value is 'binary'. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Semicolon delimited list of permissible values. + + + + + Description of this column. + + + + + How this column should be modularized, if at all. + + + + + + + Column should not be modularized. This is the default value. + + + + + + + Column should be modularized. + + + + + + + Column is a condition and should be modularized. + + + + + + + When the column is an primary or foreign key to the Icon table it should be modularized special. + + + + + + + Any Properties in the column should be modularized. + + + + + + + Semi-colon list of keys, all of which need to be modularized. + + + + + + + + + + + Row data for a Custom Table + + + + + + + + + + Used for a Custom Table. Specifies the data for the parent Row and specified Column. + + + + + + A data value + + + + Specifies in which column to insert this data. + + + + + + + + + + + Use this element to ensure that a table appears in the installer database, even if its empty. + + + + This element is particularly useful for two problems that may occur while merging merge modules: + + The first likely problem is that in order to properly merge you need to have certain + tables present prior to merging. Using this element is one way to ensure those tables + are present prior to the merging. + + The other common problem is that a merge module has incorrect validation information + about some tables. By ensuring these tables prior to merging, you can avoid this + problem because the correct validation information will go into the installer database + before the merge module has a chance to set it incorrectly. + + + + + + + The name of the table. + + + + + + + + This element exposes advanced WiX functionality. Use this element to declare WiX variables + from directly within your authoring. WiX variables are not resolved until the final msi/msm/pcp + file is actually generated. WiX variables do not persist into the msi/msm/pcp file, so they cannot + be used when an MSI file is being installed; it's a WiX-only concept. + + + + + + The name of the variable. + + + + + + Set this value to 'yes' in order to make the variable's value overridable either by + another WixVariable entry or via the command-line option -d<name>=<value> + for light.exe. If the same variable is declared overridable in multiple places it + will cause an error (since WiX won't know which value is correct). The default value + is 'no'. + + + + + + + The value of the variable. The value cannot be an empty string because that would + make it possible to accidentally set a column to null. + + + + + + + + + Use this element to contain definitions for instance transforms. + + + + + + + + + The Id of the Property who's value should change for each instance. + + + + + + + + Defines an instance transform for your product. + + + + + + + The identity of the instance transform. This value will define the name by which the instance + should be referred to on the command line. In addition, the value of the this attribute will + determine what the value of the property specified in Property attribute on InstanceTransforms + will change to for each instance. + + + + + + The ProductCode for this instance. + + + + + The ProductName for this instance. + + + + + The UpgradeCode for this instance. + + + + + + + + Simplifies authoring for major upgrades, including support for preventing downgrades. + + The parent Product element must have valid UpgradeCode and Version attributes. + + When the FindRelatedProducts action detects a related product installed on the system, + it appends the product code to the property named WIX_UPGRADE_DETECTED. After the + FindRelatedProducts action is run, the value of the WIX_UPGRADE_DETECTED property is a + list of product codes, separated by semicolons (;), detected on the system. + + + + + + + + + + When set to no (the default), products with lower version numbers are blocked from + installing when a product with a higher version is installed; the DowngradeErrorMessage + attribute must also be specified. + + When set to yes, any version can be installed over any other version. + + + + + + + When set to no (the default), installing a product with the same version and upgrade code + (but different product code) is allowed and treated by MSI as two products. When set to yes, + WiX sets the msidbUpgradeAttributesVersionMaxInclusive attribute, which tells MSI to treat + a product with the same version as a major upgrade. + + This is useful when two product versions differ only in the fourth version field. MSI + specifically ignores that field when comparing product versions, so two products that + differ only in the fourth version field are the same product and need this attribute set to + yes to be detected. + + Note that because MSI ignores the fourth product version field, setting this attribute to + yes also allows downgrades when the first three product version fields are identical. + For example, product version 1.0.0.1 will "upgrade" 1.0.0.2998 because they're seen as the + same version (1.0.0). That could reintroduce serious bugs so the safest choice is to change + the first three version fields and omit this attribute to get the default of no. + + This attribute cannot be "yes" when AllowDowngrades is also "yes" -- AllowDowngrades + already allows two products with the same version number to upgrade each other. + + + + + + + When set to yes, products with higer version numbers are blocked from + installing when a product with a lower version is installed; the UpgradeErrorMessage + attribute must also be specified. + + When set to no (the default), any version can be installed over any lower version. + + + + + + + The message displayed if users try to install a product with a lower version number + when a product with a higher version is installed. Used only when AllowDowngrades + is no (the default). + + + + + + + The message displayed if users try to install a product with a higer version number + when a product with a lower version is installed. Used only when Disallow + is yes. + + + + + + + When set to yes (the default), the MigrateFeatureStates standard action will set the + feature states of the upgrade product to those of the installed product. + + When set to no, the installed features have no effect on the upgrade installation. + + + + + + + When set to yes, the Upgrade table rows will match any product with the same UpgradeCode. + + When set to no (the default), the Upgrade table rows will match only products with the + same UpgradeCode and ProductLanguage. + + + + + + + When set to yes, failures removing the installed product during the upgrade will be + ignored. + + When set to no (the default), failures removing the installed product during the upgrade + will be considered a failure and, depending on the scheduling, roll back the upgrade. + + + + + + + A formatted string that contains the list of features to remove from the installed + product. The default is to remove all features. Note that if you use formatted property + values that evaluate to an empty string, no features will be removed; only omitting + this attribute defaults to removing all features. + + + + + + + Determines the scheduling of the RemoveExistingProducts standard action, which is when + the installed product is removed. The default is "afterInstallValidate" which removes + the installed product entirely before installing the upgrade product. It's slowest but + gives the most flexibility in changing components and features in the upgrade product. + + For more information, see RemoveExistingProducts. + + + + + + + + (Default) Schedules RemoveExistingProducts after the InstallValidate standard + action. This scheduling removes the installed product entirely before installing + the upgrade product. It's slowest but gives the most flexibility in changing + components and features in the upgrade product. Note that if the installation + of the upgrade product fails, the machine will have neither version installed. + + + + + + + Schedules RemoveExistingProducts after the InstallInitialize standard action. + This is similar to the afterInstallValidate scheduling, but if the installation + of the upgrade product fails, Windows Installer also rolls back the removal of + the installed product -- in other words, reinstalls it. + + + + + + + Schedules RemoveExistingProducts between the InstallExecute and InstallFinalize standard actions. + This scheduling installs the upgrade product "on top of" the installed product then lets + RemoveExistingProducts uninstall any components that don't also exist in the upgrade product. + Note that this scheduling requires strict adherence to the component rules because it relies + on component reference counts to be accurate during installation of the upgrade product and + removal of the installed product. For more information, see + + Bob Arnson's blog post "Paying for Upgrades" + for details. If installation of the upgrade product fails, Windows Installer + also rolls back the removal of the installed product -- in other words, reinstalls it. + + + + + + + Schedules RemoveExistingProducts between the InstallExecuteAgain and InstallFinalize standard actions. + This is identical to the afterInstallExecute scheduling but after the InstallExecuteAgain standard + action instead of InstallExecute. + + + + + + + Schedules RemoveExistingProducts after the InstallFinalize standard action. This is similar to the + afterInstallExecute and afterInstallExecuteAgain schedulings but takes place outside the + installation transaction so if installation of the upgrade product fails, Windows Installer does + not roll back the removal of the installed product, so the machine will have both versions + installed. + + + + + + + + + + + + + + + + + + + + Specifies the lower bound on the range of product versions to be detected by FindRelatedProducts. + + + + + Specifies the upper boundary of the range of product versions detected by FindRelatedProducts. + + + + + Specifies the set of languages detected by FindRelatedProducts. Enter a list of numeric language identifiers (LANGID) separated by commas (,). Leave this value null to specify all languages. Set ExcludeLanguages to "yes" in order detect all languages, excluding the languages listed in this value. + + + + + Set to "no" to make the range of versions detected exclude the value specified in Minimum. This attribute is "yes" by default. + + + + + Set to "yes" to make the range of versions detected include the value specified in Maximum. + + + + + Set to "yes" to detect all languages, excluding the languages listed in the Language attribute. + + + + + This value specifies the upgrade code for the products that are to be detected by the FindRelatedProducts action. + + + + + + Extensibility point in the WiX XML Schema. Schema extensions can register additional + attributes at this point in the schema. + + + + + + + + + + + + + Text node specifies the condition of the action. + + + + The name of an action that this action should come after. + + + + + The name of an action that this action should come before. + + + + + + If "yes", the sequencing of this action may be overridden by sequencing elsewhere. + + + + + + A value used to indicate the position of this action in a sequence. + + + + + If yes, this action will not occur. + + + + + + + + + + + A value used to indicate the position of this action in a sequence. + + + + + If yes, this action will not occur. + + + + + + + + + Values of this type will look like: "01234567-89AB-CDEF-0123-456789ABCDEF" or "{01234567-89AB-CDEF-0123-456789ABCDEF}". Also allows "PUT-GUID-HERE" for use in examples. + + + + + + + + Values of this type will look like: "01234567-89AB-CDEF-0123-456789ABCDEF" or "{01234567-89AB-CDEF-0123-456789ABCDEF}". A GUID can be auto-generated by setting the value to "*". Also allows "PUT-GUID-HERE" for use in examples. + + + + + + + + Values of this type will either be "attached" or "detached". + + + + + + + + + The list of communcation protocols with executable packages Burn supports. + + + + + + + The executable package does not support a communication protocol. + + + + + + + The executable package is another Burn bundle and supports the Burn communication protocol. + + + + + + + The executable package implements the .NET Framework v4.0 communication protocol. + + + + + + + + Values of this type will look like: "01234567-89AB-CDEF-0123-456789ABCDEF" or "{01234567-89AB-CDEF-0123-456789ABCDEF}", but also allows "PUT-GUID-HERE" for use in examples. It's also possible to have an empty value "". + + + + + + + + Values of this type must be an integer or the value can be a localization variable with the format !(loc.Variable) where "Variable" is the name of the variable. + + + + + + + + Values of this type will look like: "FileName.ext". Only one period is allowed. The following characters are not allowed: \ ? | > : / * " + , ; = [ ] less-than, or whitespace. The name cannot be longer than 8 characters and the extension cannot exceed 3 characters. The value could also be a localization variable with the format !(loc.VARIABLE). + + + + + + + + Values of this type will look like: "Long File Name.extension". Legal long names contain no more than 260 characters and must contain at least one non-period character. The following characters are not allowed: \ ? | > : / * " or less-than. The name must be shorter than 260 characters. The value could also be a localization variable with the format !(loc.VARIABLE). + + + + + + + + Values of this type will look like: "x.x.x.x" where x is an integer from 0 to 65534. + + + + + + + + Values of this type will look like: "File?.*". Only one period is allowed. The following characters are not allowed: \ | > : / " + , ; = [ ] less-than, or whitespace. The name cannot be longer than 8 characters and the extension cannot exceed 3 characters. The value could also be a localization variable with the format !(loc.VARIABLE). + + + + + + + + Values of this type will look like: "Long File N?me.extension*". Legal long names contain no more than 260 characters and must contain at least one non-period character. The following characters are not allowed: \ | > : / " or less-than. The name must be shorter than 260 characters. The value could also be a localization variable with the format !(loc.VARIABLE). + + + + + + + + This type supports any hexadecimal number. Both upper and lower case is acceptable for letters appearing in the number. This type also includes the empty string: "". + + + + + + + + Values of this type will either be "yes" or "no". + + + + + + Values of this type will either be "yes" or "no". + + + + + + + + + Values of this type will either be "button", "yes" or "no". + + + + + + Values of this type will either be "button", "yes" or "no". + + + + + + + + + + Values of this type will either be "default", "yes", or "no". + + + + + + Values of this type will either be "default", "yes", or "no". + + + + + + + + + + Values of this type will either be "always", "yes", or "no". + + + + + + Values of this type will either be "always", "yes", or "no". + + + + + + + + + + Values of this type represent possible registry roots. + + + + + + A per-user installation will make the operation occur under HKEY_CURRENT_USER. + A per-machine installation will make the operation occur under HKEY_LOCAL_MACHINE. + + + + + + + Operation occurs under HKEY_CLASSES_ROOT. When using Windows 2000 or later, the installer writes or removes the value + from the HKCU\Software\Classes hive during per-user installations. When using Windows 2000 or later operating systems, + the installer writes or removes the value from the HKLM\Software\Classes hive during per-machine installations. + + + + + + + Operation occurs under HKEY_CURRENT_USER. It is recommended to set the KeyPath='yes' attribute when setting this value for writing values + in order to ensure that the installer writes the necessary registry entries when there are multiple users on the same computer. + + + + + + + Operation occurs under HKEY_LOCAL_MACHINE. + + + + + + + Operation occurs under HKEY_USERS. + + + + + + + + Value indicates that this action is executed if the installer returns the associated exit type. Each exit type can be used with no more than one action. + Multiple actions can have exit types assigned, but every action and exit type must be different. Exit types are typically used with dialog boxes. + + + + + + + + + + + Specifies whether an action occur on install, uninstall or both. + + + + + + The action should happen during install (msiInstallStateLocal or msiInstallStateSource). + + + + + + + The action should happen during uninstall (msiInstallStateAbsent). + + + + + + + The action should happen during both install and uninstall. + + + + + + + + + Controls which sequences the item assignment is sequenced in. + + + + + + + Schedules the assignment in the InstallUISequence and the InstallExecuteSequence. + + + + + + + Schedules the assignment to run in the InstallUISequence or the InstallExecuteSequence if the InstallUISequence is skipped. + + + + + + + Schedules the assignment only in the the InstallExecuteSequence. + + + + + + + Schedules the assignment only in the the InstallUISequence. + + + + + + + + + Indicates the compression level for a cabinet. + + + + + + + + Indicates the compression level for a cabinet. + + + + + + + + + + + + + A type that represents that 1 or more preprocessor variables (as they appear in sources on disk, before preprocessor has run). + + + + + + + + Values of this type must be an integer or the value of one or more preprocessor variables with the format $(var.Variable) where "Variable" is the name of the preprocessor variable. + + + + + + diff --git a/src/WixToolset.Data/Xsd/wixloc.xsd b/src/WixToolset.Data/Xsd/wixloc.xsd new file mode 100644 index 00000000..9f1814c4 --- /dev/null +++ b/src/WixToolset.Data/Xsd/wixloc.xsd @@ -0,0 +1,134 @@ + + + + + + + + Schema for describing WiX Localization files (.wxl). + + + + + + + + You can specify any valid Windows code page by integer like 1252, or by web name like Windows-1252 or iso-8859-1. See Code Pages for more information. + + How To: Build a localized version of your installer + How To: Make your installer localizable + + + + + + + + + + The code page integer value or web name for the resulting database. You can also specify -1 which will not reset the database code page. See remarks for more information. + + + + + Culture of the localization strings. + + + + + The decimal language ID (LCID) for the culture. + + + + + + + + + How To: Build a localized version of your installer + How To: Make your installer localizable + + + + + + Identity of the resource. + + + + + Determines if the localized string may be overridden. + + + + + Indicates whether the string is localizable text or a non-localizable string that must be unique per locale. No WiX tools are affected by the value of this attribute; it used as documentation for localizers to ignore things like GUIDs or identifiers that look like text. + + + + + + + + Allows a localization to override the position, size, and text of dialogs and controls. Override the text by specifying the replacement text in the inner text of the UI element. + + + + + Identifies the dialog to localize or the dialog that a control to localize is in. + + + + + Combined with the Dialog attribute, identifies the control to localize. + + + + + For a dialog, overrides the authored horizontal centering. For a control, overrides the authored horizontal coordinate of the upper-left corner of the rectangular boundary. This must be a non-negative number. + + + + + For a dialog, overrides the authored vertical centering. For a control, overrides the authored vertical coordinate of the upper-left corner of the rectangular boundary of the control. This must be a non-negative number. + + + + + For a dialog, overrides the authored width in dialog units. For a control, overrides the authored width of the rectangular boundary of the control. This must be a non-negative number. + + + + + For a dialog, overrides the authored height in dialog units. For a control, overrides the authored height of the rectangular boundary of the control. This must be a non-negative number. + + + + + Set this attribute to "yes" to cause the Control to display from right to left. Not valid for a dialog. + + + + + Set this attribute to "yes" to cause the Control to be right aligned. Not valid for a dialog. + + + + + Set this attribute to "yes" to cause the scroll bar to display on the left side of the Control. Not valid for a dialog. + + + + + + + + + + + + diff --git a/src/WixToolset.Data/YesNoAlwaysType.cs b/src/WixToolset.Data/YesNoAlwaysType.cs new file mode 100644 index 00000000..3b4ca5d7 --- /dev/null +++ b/src/WixToolset.Data/YesNoAlwaysType.cs @@ -0,0 +1,25 @@ +// 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.Data +{ + /// + /// Yes, No, Always xml simple type. + /// + public enum YesNoAlwaysType + { + /// Not a valid yes, no or always value. + IllegalValue = -2, + + /// Value not set; equivalent to null for reference types. + NotSet = -1, + + /// The no value. + No, + + /// The yes value. + Yes, + + /// The always value. + Always, + } +} diff --git a/src/WixToolset.Data/YesNoDefaultType.cs b/src/WixToolset.Data/YesNoDefaultType.cs new file mode 100644 index 00000000..fd782d46 --- /dev/null +++ b/src/WixToolset.Data/YesNoDefaultType.cs @@ -0,0 +1,25 @@ +// 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.Data +{ + /// + /// Yes, No, Default xml simple type. + /// + public enum YesNoDefaultType + { + /// Not a valid yes, no or default value. + IllegalValue = -2, + + /// Value not set; equivalent to null for reference types. + NotSet = -1, + + /// The no value. + No, + + /// The yes value. + Yes, + + /// The default value. + Default, + } +} diff --git a/src/WixToolset.Data/YesNoType.cs b/src/WixToolset.Data/YesNoType.cs new file mode 100644 index 00000000..9c1cc9a7 --- /dev/null +++ b/src/WixToolset.Data/YesNoType.cs @@ -0,0 +1,22 @@ +// 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.Data +{ + /// + /// Yes/no type (kinda like a boolean). + /// + public enum YesNoType + { + /// Not a valid yes or no value. + IllegalValue = -2, + + /// Value not set; equivalent to null for reference types. + NotSet = -1, + + /// The no value. + No, + + /// The yes value. + Yes, + } +} diff --git a/src/nuget.config b/src/nuget.config new file mode 100644 index 00000000..d040e229 --- /dev/null +++ b/src/nuget.config @@ -0,0 +1,11 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/src/version.json b/src/version.json new file mode 100644 index 00000000..fba429f7 --- /dev/null +++ b/src/version.json @@ -0,0 +1,11 @@ +{ + "version": "4.0-preview", + "publicReleaseRefSpec": [ + "^refs/heads/master$" + ], + "cloudBuild": { + "buildNumber": { + "enabled": true + } + } +} -- cgit v1.2.3-55-g6feb