From aa0bd9f66dabc6460f93cf9a029e06b079f10db8 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Thu, 22 Apr 2021 20:00:30 -0700 Subject: Move Core.Native into wix --- src/.editorconfig | 37 + src/WixToolset.Core.Native/AssemblyExtensions.cs | 70 -- src/WixToolset.Core.Native/Cabinet.cs | 120 --- src/WixToolset.Core.Native/CabinetCompressFile.cs | 65 -- src/WixToolset.Core.Native/CabinetFileInfo.cs | 63 -- src/WixToolset.Core.Native/DateTimeInterop.cs | 48 - src/WixToolset.Core.Native/FileSystem.cs | 78 -- .../IWindowsInstallerValidatorCallback.cs | 27 - src/WixToolset.Core.Native/Msi/Database.cs | 262 ------ src/WixToolset.Core.Native/Msi/InstallLogModes.cs | 111 --- src/WixToolset.Core.Native/Msi/InstallMessage.cs | 88 -- src/WixToolset.Core.Native/Msi/InstallUILevels.cs | 72 -- src/WixToolset.Core.Native/Msi/Installer.cs | 138 --- src/WixToolset.Core.Native/Msi/MSIFILEHASHINFO.cs | 34 - src/WixToolset.Core.Native/Msi/ModifyView.cs | 75 -- src/WixToolset.Core.Native/Msi/MsiException.cs | 77 -- src/WixToolset.Core.Native/Msi/MsiHandle.cs | 117 --- src/WixToolset.Core.Native/Msi/MsiInterop.cs | 408 --------- src/WixToolset.Core.Native/Msi/OpenDatabase.cs | 40 - src/WixToolset.Core.Native/Msi/Record.cs | 181 ---- src/WixToolset.Core.Native/Msi/Session.cs | 42 - .../Msi/SummaryInformation.cs | 376 -------- .../Msi/TransformErrorConditions.cs | 58 -- .../Msi/TransformValidations.cs | 73 -- src/WixToolset.Core.Native/Msi/View.cs | 206 ----- .../Msi/WixInvalidIdtException.cs | 49 - .../Msm/ConfigurationCallback.cs | 90 -- .../Msm/IMsmConfigureModule.cs | 32 - src/WixToolset.Core.Native/Msm/IMsmError.cs | 77 -- src/WixToolset.Core.Native/Msm/IMsmErrors.cs | 32 - src/WixToolset.Core.Native/Msm/IMsmMerge2.cs | 174 ---- src/WixToolset.Core.Native/Msm/IMsmStrings.cs | 32 - src/WixToolset.Core.Native/Msm/MsmErrorType.cs | 154 ---- src/WixToolset.Core.Native/Msm/MsmInterop.cs | 49 - src/WixToolset.Core.Native/Ole32/Storage.cs | 377 -------- src/WixToolset.Core.Native/Ole32/StorageMode.cs | 55 -- .../PatchAPI/PatchInterop.cs | 990 --------------------- src/WixToolset.Core.Native/ValidationMessage.cs | 47 - .../ValidationMessageType.cs | 31 - .../WindowsInstallerValidator.cs | 427 --------- src/WixToolset.Core.Native/WixNativeExe.cs | 125 --- .../WixToolset.Core.Native.csproj | 49 - .../WixToolset.Core.Native.nuspec | 41 - src/WixToolset.Core.Native/cubes/darice.cub | Bin 684032 -> 0 bytes src/WixToolset.Core.Native/cubes/mergemod.cub | Bin 483328 -> 0 bytes .../targets/WixToolset.Core.Native.targets | 9 - src/signing.json | 13 + .../WixToolsetTest.Core.Native/CabinetFixture.cs | 96 -- src/test/WixToolsetTest.Core.Native/MsmFixture.cs | 17 - .../WixToolsetTest.Core.Native/TestData/test.cab | Bin 115 -> 0 bytes .../WixToolsetTest.Core.Native/TestData/test.txt | 1 - .../Utility/DisposableFileSystem.cs | 86 -- .../WixToolsetTest.Core.Native/Utility/Pushd.cs | 46 - .../WixToolsetTest.Core.Native/Utility/TestData.cs | 17 - .../WixToolsetTest.Core.Native.csproj | 26 - src/test/version.txt | 1 - src/version.txt | 1 + src/wix/README-CoreNative.md | 2 + src/wix/WixToolset.Core.Native.sln | 63 ++ src/wix/WixToolset.Core.Native.v3.ncrunchsolution | 6 + .../WixToolset.Core.Native/AssemblyExtensions.cs | 70 ++ src/wix/WixToolset.Core.Native/Cabinet.cs | 120 +++ .../WixToolset.Core.Native/CabinetCompressFile.cs | 65 ++ src/wix/WixToolset.Core.Native/CabinetFileInfo.cs | 63 ++ src/wix/WixToolset.Core.Native/DateTimeInterop.cs | 48 + src/wix/WixToolset.Core.Native/FileSystem.cs | 78 ++ .../IWindowsInstallerValidatorCallback.cs | 27 + src/wix/WixToolset.Core.Native/Msi/Database.cs | 262 ++++++ .../WixToolset.Core.Native/Msi/InstallLogModes.cs | 111 +++ .../WixToolset.Core.Native/Msi/InstallMessage.cs | 88 ++ .../WixToolset.Core.Native/Msi/InstallUILevels.cs | 72 ++ src/wix/WixToolset.Core.Native/Msi/Installer.cs | 138 +++ .../WixToolset.Core.Native/Msi/MSIFILEHASHINFO.cs | 34 + src/wix/WixToolset.Core.Native/Msi/ModifyView.cs | 75 ++ src/wix/WixToolset.Core.Native/Msi/MsiException.cs | 77 ++ src/wix/WixToolset.Core.Native/Msi/MsiHandle.cs | 117 +++ src/wix/WixToolset.Core.Native/Msi/MsiInterop.cs | 408 +++++++++ src/wix/WixToolset.Core.Native/Msi/OpenDatabase.cs | 40 + src/wix/WixToolset.Core.Native/Msi/Record.cs | 181 ++++ src/wix/WixToolset.Core.Native/Msi/Session.cs | 42 + .../Msi/SummaryInformation.cs | 376 ++++++++ .../Msi/TransformErrorConditions.cs | 58 ++ .../Msi/TransformValidations.cs | 73 ++ src/wix/WixToolset.Core.Native/Msi/View.cs | 206 +++++ .../Msi/WixInvalidIdtException.cs | 49 + .../Msm/ConfigurationCallback.cs | 90 ++ .../Msm/IMsmConfigureModule.cs | 32 + src/wix/WixToolset.Core.Native/Msm/IMsmError.cs | 77 ++ src/wix/WixToolset.Core.Native/Msm/IMsmErrors.cs | 32 + src/wix/WixToolset.Core.Native/Msm/IMsmMerge2.cs | 174 ++++ src/wix/WixToolset.Core.Native/Msm/IMsmStrings.cs | 32 + src/wix/WixToolset.Core.Native/Msm/MsmErrorType.cs | 154 ++++ src/wix/WixToolset.Core.Native/Msm/MsmInterop.cs | 49 + src/wix/WixToolset.Core.Native/Ole32/Storage.cs | 377 ++++++++ .../WixToolset.Core.Native/Ole32/StorageMode.cs | 55 ++ .../PatchAPI/PatchInterop.cs | 990 +++++++++++++++++++++ .../WixToolset.Core.Native/ValidationMessage.cs | 47 + .../ValidationMessageType.cs | 31 + .../WindowsInstallerValidator.cs | 427 +++++++++ src/wix/WixToolset.Core.Native/WixNativeExe.cs | 125 +++ .../WixToolset.Core.Native.csproj | 49 + .../WixToolset.Core.Native.nuspec | 41 + src/wix/WixToolset.Core.Native/cubes/darice.cub | Bin 0 -> 684032 bytes src/wix/WixToolset.Core.Native/cubes/mergemod.cub | Bin 0 -> 483328 bytes .../targets/WixToolset.Core.Native.targets | 9 + src/wix/appveyor-CoreNative.cmd | 19 + src/wix/appveyor-CoreNative.yml | 44 + src/wix/nuget-CoreNative.config | 9 + .../WixToolsetTest.Core.Native/CabinetFixture.cs | 96 ++ .../test/WixToolsetTest.Core.Native/MsmFixture.cs | 17 + .../WixToolsetTest.Core.Native/TestData/test.cab | Bin 0 -> 115 bytes .../WixToolsetTest.Core.Native/TestData/test.txt | 1 + .../Utility/DisposableFileSystem.cs | 86 ++ .../WixToolsetTest.Core.Native/Utility/Pushd.cs | 46 + .../WixToolsetTest.Core.Native/Utility/TestData.cs | 17 + .../WixToolsetTest.Core.Native.csproj | 26 + src/wix/test/version.txt | 1 + src/wix/wixnative/ARM/mergemod.dll | Bin 0 -> 172704 bytes src/wix/wixnative/ARM64/mergemod.dll | Bin 0 -> 194512 bytes src/wix/wixnative/Win32/mergemod.dll | Bin 0 -> 159176 bytes src/wix/wixnative/enumcab.cpp | 47 + src/wix/wixnative/extractcab.cpp | 50 ++ src/wix/wixnative/packages.config | 8 + src/wix/wixnative/precomp.cpp | 3 + src/wix/wixnative/precomp.h | 19 + src/wix/wixnative/resetacls.cpp | 51 ++ src/wix/wixnative/smartcab.cpp | 160 ++++ src/wix/wixnative/wixnative.cpp | 38 + src/wix/wixnative/wixnative.v3.ncrunchproject | 5 + src/wix/wixnative/wixnative.vcxproj | 76 ++ src/wix/wixnative/x64/mergemod.dll | Bin 0 -> 180376 bytes src/wixnative/ARM/mergemod.dll | Bin 172704 -> 0 bytes src/wixnative/ARM64/mergemod.dll | Bin 194512 -> 0 bytes src/wixnative/Win32/mergemod.dll | Bin 159176 -> 0 bytes src/wixnative/enumcab.cpp | 47 - src/wixnative/extractcab.cpp | 50 -- src/wixnative/packages.config | 8 - src/wixnative/precomp.cpp | 3 - src/wixnative/precomp.h | 19 - src/wixnative/resetacls.cpp | 51 -- src/wixnative/smartcab.cpp | 160 ---- src/wixnative/wixnative.cpp | 38 - src/wixnative/wixnative.v3.ncrunchproject | 5 - src/wixnative/wixnative.vcxproj | 76 -- src/wixnative/x64/mergemod.dll | Bin 180376 -> 0 bytes 145 files changed, 6610 insertions(+), 6416 deletions(-) create mode 100644 src/.editorconfig delete mode 100644 src/WixToolset.Core.Native/AssemblyExtensions.cs delete mode 100644 src/WixToolset.Core.Native/Cabinet.cs delete mode 100644 src/WixToolset.Core.Native/CabinetCompressFile.cs delete mode 100644 src/WixToolset.Core.Native/CabinetFileInfo.cs delete mode 100644 src/WixToolset.Core.Native/DateTimeInterop.cs delete mode 100644 src/WixToolset.Core.Native/FileSystem.cs delete mode 100644 src/WixToolset.Core.Native/IWindowsInstallerValidatorCallback.cs delete mode 100644 src/WixToolset.Core.Native/Msi/Database.cs delete mode 100644 src/WixToolset.Core.Native/Msi/InstallLogModes.cs delete mode 100644 src/WixToolset.Core.Native/Msi/InstallMessage.cs delete mode 100644 src/WixToolset.Core.Native/Msi/InstallUILevels.cs delete mode 100644 src/WixToolset.Core.Native/Msi/Installer.cs delete mode 100644 src/WixToolset.Core.Native/Msi/MSIFILEHASHINFO.cs delete mode 100644 src/WixToolset.Core.Native/Msi/ModifyView.cs delete mode 100644 src/WixToolset.Core.Native/Msi/MsiException.cs delete mode 100644 src/WixToolset.Core.Native/Msi/MsiHandle.cs delete mode 100644 src/WixToolset.Core.Native/Msi/MsiInterop.cs delete mode 100644 src/WixToolset.Core.Native/Msi/OpenDatabase.cs delete mode 100644 src/WixToolset.Core.Native/Msi/Record.cs delete mode 100644 src/WixToolset.Core.Native/Msi/Session.cs delete mode 100644 src/WixToolset.Core.Native/Msi/SummaryInformation.cs delete mode 100644 src/WixToolset.Core.Native/Msi/TransformErrorConditions.cs delete mode 100644 src/WixToolset.Core.Native/Msi/TransformValidations.cs delete mode 100644 src/WixToolset.Core.Native/Msi/View.cs delete mode 100644 src/WixToolset.Core.Native/Msi/WixInvalidIdtException.cs delete mode 100644 src/WixToolset.Core.Native/Msm/ConfigurationCallback.cs delete mode 100644 src/WixToolset.Core.Native/Msm/IMsmConfigureModule.cs delete mode 100644 src/WixToolset.Core.Native/Msm/IMsmError.cs delete mode 100644 src/WixToolset.Core.Native/Msm/IMsmErrors.cs delete mode 100644 src/WixToolset.Core.Native/Msm/IMsmMerge2.cs delete mode 100644 src/WixToolset.Core.Native/Msm/IMsmStrings.cs delete mode 100644 src/WixToolset.Core.Native/Msm/MsmErrorType.cs delete mode 100644 src/WixToolset.Core.Native/Msm/MsmInterop.cs delete mode 100644 src/WixToolset.Core.Native/Ole32/Storage.cs delete mode 100644 src/WixToolset.Core.Native/Ole32/StorageMode.cs delete mode 100644 src/WixToolset.Core.Native/PatchAPI/PatchInterop.cs delete mode 100644 src/WixToolset.Core.Native/ValidationMessage.cs delete mode 100644 src/WixToolset.Core.Native/ValidationMessageType.cs delete mode 100644 src/WixToolset.Core.Native/WindowsInstallerValidator.cs delete mode 100644 src/WixToolset.Core.Native/WixNativeExe.cs delete mode 100644 src/WixToolset.Core.Native/WixToolset.Core.Native.csproj delete mode 100644 src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec delete mode 100644 src/WixToolset.Core.Native/cubes/darice.cub delete mode 100644 src/WixToolset.Core.Native/cubes/mergemod.cub delete mode 100644 src/WixToolset.Core.Native/targets/WixToolset.Core.Native.targets create mode 100644 src/signing.json delete mode 100644 src/test/WixToolsetTest.Core.Native/CabinetFixture.cs delete mode 100644 src/test/WixToolsetTest.Core.Native/MsmFixture.cs delete mode 100644 src/test/WixToolsetTest.Core.Native/TestData/test.cab delete mode 100644 src/test/WixToolsetTest.Core.Native/TestData/test.txt delete mode 100644 src/test/WixToolsetTest.Core.Native/Utility/DisposableFileSystem.cs delete mode 100644 src/test/WixToolsetTest.Core.Native/Utility/Pushd.cs delete mode 100644 src/test/WixToolsetTest.Core.Native/Utility/TestData.cs delete mode 100644 src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj delete mode 100644 src/test/version.txt create mode 100644 src/version.txt create mode 100644 src/wix/README-CoreNative.md create mode 100644 src/wix/WixToolset.Core.Native.sln create mode 100644 src/wix/WixToolset.Core.Native.v3.ncrunchsolution create mode 100644 src/wix/WixToolset.Core.Native/AssemblyExtensions.cs create mode 100644 src/wix/WixToolset.Core.Native/Cabinet.cs create mode 100644 src/wix/WixToolset.Core.Native/CabinetCompressFile.cs create mode 100644 src/wix/WixToolset.Core.Native/CabinetFileInfo.cs create mode 100644 src/wix/WixToolset.Core.Native/DateTimeInterop.cs create mode 100644 src/wix/WixToolset.Core.Native/FileSystem.cs create mode 100644 src/wix/WixToolset.Core.Native/IWindowsInstallerValidatorCallback.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/Database.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/InstallLogModes.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/InstallMessage.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/InstallUILevels.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/Installer.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/MSIFILEHASHINFO.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/ModifyView.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/MsiException.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/MsiHandle.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/MsiInterop.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/OpenDatabase.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/Record.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/Session.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/SummaryInformation.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/TransformErrorConditions.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/TransformValidations.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/View.cs create mode 100644 src/wix/WixToolset.Core.Native/Msi/WixInvalidIdtException.cs create mode 100644 src/wix/WixToolset.Core.Native/Msm/ConfigurationCallback.cs create mode 100644 src/wix/WixToolset.Core.Native/Msm/IMsmConfigureModule.cs create mode 100644 src/wix/WixToolset.Core.Native/Msm/IMsmError.cs create mode 100644 src/wix/WixToolset.Core.Native/Msm/IMsmErrors.cs create mode 100644 src/wix/WixToolset.Core.Native/Msm/IMsmMerge2.cs create mode 100644 src/wix/WixToolset.Core.Native/Msm/IMsmStrings.cs create mode 100644 src/wix/WixToolset.Core.Native/Msm/MsmErrorType.cs create mode 100644 src/wix/WixToolset.Core.Native/Msm/MsmInterop.cs create mode 100644 src/wix/WixToolset.Core.Native/Ole32/Storage.cs create mode 100644 src/wix/WixToolset.Core.Native/Ole32/StorageMode.cs create mode 100644 src/wix/WixToolset.Core.Native/PatchAPI/PatchInterop.cs create mode 100644 src/wix/WixToolset.Core.Native/ValidationMessage.cs create mode 100644 src/wix/WixToolset.Core.Native/ValidationMessageType.cs create mode 100644 src/wix/WixToolset.Core.Native/WindowsInstallerValidator.cs create mode 100644 src/wix/WixToolset.Core.Native/WixNativeExe.cs create mode 100644 src/wix/WixToolset.Core.Native/WixToolset.Core.Native.csproj create mode 100644 src/wix/WixToolset.Core.Native/WixToolset.Core.Native.nuspec create mode 100644 src/wix/WixToolset.Core.Native/cubes/darice.cub create mode 100644 src/wix/WixToolset.Core.Native/cubes/mergemod.cub create mode 100644 src/wix/WixToolset.Core.Native/targets/WixToolset.Core.Native.targets create mode 100644 src/wix/appveyor-CoreNative.cmd create mode 100644 src/wix/appveyor-CoreNative.yml create mode 100644 src/wix/nuget-CoreNative.config create mode 100644 src/wix/test/WixToolsetTest.Core.Native/CabinetFixture.cs create mode 100644 src/wix/test/WixToolsetTest.Core.Native/MsmFixture.cs create mode 100644 src/wix/test/WixToolsetTest.Core.Native/TestData/test.cab create mode 100644 src/wix/test/WixToolsetTest.Core.Native/TestData/test.txt create mode 100644 src/wix/test/WixToolsetTest.Core.Native/Utility/DisposableFileSystem.cs create mode 100644 src/wix/test/WixToolsetTest.Core.Native/Utility/Pushd.cs create mode 100644 src/wix/test/WixToolsetTest.Core.Native/Utility/TestData.cs create mode 100644 src/wix/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj create mode 100644 src/wix/test/version.txt create mode 100644 src/wix/wixnative/ARM/mergemod.dll create mode 100644 src/wix/wixnative/ARM64/mergemod.dll create mode 100644 src/wix/wixnative/Win32/mergemod.dll create mode 100644 src/wix/wixnative/enumcab.cpp create mode 100644 src/wix/wixnative/extractcab.cpp create mode 100644 src/wix/wixnative/packages.config create mode 100644 src/wix/wixnative/precomp.cpp create mode 100644 src/wix/wixnative/precomp.h create mode 100644 src/wix/wixnative/resetacls.cpp create mode 100644 src/wix/wixnative/smartcab.cpp create mode 100644 src/wix/wixnative/wixnative.cpp create mode 100644 src/wix/wixnative/wixnative.v3.ncrunchproject create mode 100644 src/wix/wixnative/wixnative.vcxproj create mode 100644 src/wix/wixnative/x64/mergemod.dll delete mode 100644 src/wixnative/ARM/mergemod.dll delete mode 100644 src/wixnative/ARM64/mergemod.dll delete mode 100644 src/wixnative/Win32/mergemod.dll delete mode 100644 src/wixnative/enumcab.cpp delete mode 100644 src/wixnative/extractcab.cpp delete mode 100644 src/wixnative/packages.config delete mode 100644 src/wixnative/precomp.cpp delete mode 100644 src/wixnative/precomp.h delete mode 100644 src/wixnative/resetacls.cpp delete mode 100644 src/wixnative/smartcab.cpp delete mode 100644 src/wixnative/wixnative.cpp delete mode 100644 src/wixnative/wixnative.v3.ncrunchproject delete mode 100644 src/wixnative/wixnative.vcxproj delete mode 100644 src/wixnative/x64/mergemod.dll (limited to 'src') diff --git a/src/.editorconfig b/src/.editorconfig new file mode 100644 index 00000000..1d72e683 --- /dev/null +++ b/src/.editorconfig @@ -0,0 +1,37 @@ +# 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. +# +# Do NOT modify this file. Update the canonical version in Home\repo-template\src\.editorconfig +# then update all of the repos. + +root = true + +[*] +charset = utf-8 +indent_style = space +indent_size = 4 +trim_trailing_whitespace = true + +[*.{cs,vb}] +dotnet_sort_system_directives_first = true + +[*.cs] +csharp_indent_case_contents = true : error +csharp_indent_switch_labels = true : error +csharp_new_line_before_open_brace = all +csharp_prefer_braces = true : error +csharp_style_expression_bodied_methods = when_on_single_line : suggestion +csharp_style_expression_bodied_constructors = when_on_single_line : suggestion +csharp_style_expression_bodied_operators = when_on_single_line : suggestion +csharp_style_expression_bodied_properties = when_on_single_line : suggestion +csharp_style_expression_bodied_indexers = when_on_single_line : suggestion +csharp_style_expression_bodied_accessors = when_on_single_line : suggestion +csharp_style_var_elsewhere = true : suggestion +csharp_style_var_for_built_in_types = true : suggestion +csharp_style_var_when_type_is_apparent = true : suggestion +dotnet_style_qualification_for_event = true : error +dotnet_style_qualification_for_field = true : error +dotnet_style_qualification_for_method = true : error +dotnet_style_qualification_for_property = true : error + +[*.targets] +indent_size = 2 diff --git a/src/WixToolset.Core.Native/AssemblyExtensions.cs b/src/WixToolset.Core.Native/AssemblyExtensions.cs deleted file mode 100644 index 590a6887..00000000 --- a/src/WixToolset.Core.Native/AssemblyExtensions.cs +++ /dev/null @@ -1,70 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native -{ - using System; - using System.IO; - using System.Reflection; - using System.Text; - - internal static class AssemblyExtensions - { - internal static FindAssemblyRelativeFileResult FindFileRelativeToAssembly(this Assembly assembly, string relativePath, bool searchNativeDllDirectories) - { - // First try using the Assembly.Location. This works in almost all cases with - // no side-effects. - var path = Path.Combine(Path.GetDirectoryName(assembly.Location), relativePath); - var possiblePaths = new StringBuilder(path); - - var found = File.Exists(path); - if (!found) - { - // Fallback to the Assembly.CodeBase to handle "shadow copy" scenarios (like unit tests) but - // only check codebase if it is different from the Assembly.Location path. - var codebase = Path.Combine(Path.GetDirectoryName(new Uri(assembly.CodeBase).LocalPath), relativePath); - - if (!codebase.Equals(path, StringComparison.OrdinalIgnoreCase)) - { - path = codebase; - possiblePaths.Append(Path.PathSeparator + path); - - found = File.Exists(path); - } - - if (!found && searchNativeDllDirectories && AppContext.GetData("NATIVE_DLL_SEARCH_DIRECTORIES") is string searchDirectoriesString) - { - // If instructed to search native DLL search directories, try to find our file there. - possiblePaths.Append(Path.PathSeparator + searchDirectoriesString); - - var searchDirectories = searchDirectoriesString?.Split(Path.PathSeparator); - foreach (var directoryPath in searchDirectories) - { - var possiblePath = Path.Combine(directoryPath, relativePath); - if (File.Exists(possiblePath)) - { - path = possiblePath; - found = true; - break; - } - } - } - } - - return new FindAssemblyRelativeFileResult - { - Found = found, - Path = found ? path : null, - PossiblePaths = possiblePaths.ToString() - }; - } - - internal class FindAssemblyRelativeFileResult - { - public bool Found { get; set; } - - public string Path { get; set; } - - public string PossiblePaths { get; set; } - } - } -} diff --git a/src/WixToolset.Core.Native/Cabinet.cs b/src/WixToolset.Core.Native/Cabinet.cs deleted file mode 100644 index 9b77bd37..00000000 --- a/src/WixToolset.Core.Native/Cabinet.cs +++ /dev/null @@ -1,120 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native -{ - using System; - using System.Collections.Generic; - using System.Linq; - using WixToolset.Data; - - /// - /// Cabinet create, enumerate and extract mechanism. - /// - public sealed class Cabinet - { - private const string CompressionLevelVariable = "WIX_COMPRESSION_LEVEL"; - private static readonly char[] TextLineSplitter = new[] { '\t' }; - - /// - /// Creates a cabinet creation, enumeration, extraction mechanism. - /// - /// Path of cabinet - public Cabinet(string path) - { - this.Path = path; - } - - /// - /// Cabinet path. - /// - public string Path { get; } - - /// - /// Creates a cabinet. - /// - /// Files to compress. - /// Level of compression to apply. - /// Maximum size of cabinet. - /// Maximum threshold for each cabinet. - public void Compress(IEnumerable files, CompressionLevel compressionLevel, int maxSize = 0, int maxThresh = 0) - { - var compressionLevelVariable = Environment.GetEnvironmentVariable(CompressionLevelVariable); - - // Override authored compression level if environment variable is present. - if (!String.IsNullOrEmpty(compressionLevelVariable)) - { - if (!Enum.TryParse(compressionLevelVariable, true, out compressionLevel)) - { - throw new WixException(ErrorMessages.IllegalEnvironmentVariable(CompressionLevelVariable, compressionLevelVariable)); - } - } - - var wixnative = new WixNativeExe("smartcab", this.Path, Convert.ToInt32(compressionLevel), files.Count(), maxSize, maxThresh); - - foreach (var file in files) - { - wixnative.AddStdinLine(file.ToWixNativeStdinLine()); - } - - wixnative.Run(); - -#if TOOD_ERROR_HANDLING - catch (COMException ce) - { - // If we get a "the file exists" error, we must have a full temp directory - so report the issue - if (0x80070050 == unchecked((uint)ce.ErrorCode)) - { - throw new WixException(WixErrors.FullTempDirectory("WSC", Path.GetTempPath())); - } - - throw; - } -#endif - } - - /// - /// Enumerates all files in a cabinet. - /// - /// >List of CabinetFileInfo - public List Enumerate() - { - var wixnative = new WixNativeExe("enumcab", this.Path); - var lines = wixnative.Run(); - - var fileInfoList = new List(); - - foreach (var line in lines) - { - if (String.IsNullOrEmpty(line)) - { - continue; - } - - var data = line.Split(TextLineSplitter, StringSplitOptions.None); - - var size = Convert.ToInt32(data[1]); - var date = Convert.ToInt32(data[2]); - var time = Convert.ToInt32(data[3]); - - fileInfoList.Add(new CabinetFileInfo(data[0], size, date, time)); - } - - return fileInfoList; - } - - /// - /// Extracts all the files from a cabinet to a directory. - /// - /// Directory to extract files to. - public IEnumerable Extract(string outputFolder) - { - if (!outputFolder.EndsWith("\\", StringComparison.Ordinal)) - { - outputFolder += "\\"; - } - - var wixnative = new WixNativeExe("extractcab", this.Path, outputFolder); - return wixnative.Run().Where(output => !String.IsNullOrWhiteSpace(output)); - } - } -} diff --git a/src/WixToolset.Core.Native/CabinetCompressFile.cs b/src/WixToolset.Core.Native/CabinetCompressFile.cs deleted file mode 100644 index 6778f4a1..00000000 --- a/src/WixToolset.Core.Native/CabinetCompressFile.cs +++ /dev/null @@ -1,65 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native -{ - /// - /// Information to compress file into a cabinet. - /// - public sealed class CabinetCompressFile - { - /// - /// Cabinet compress file. - /// - /// Path to file to add. - /// The token for the file. - public CabinetCompressFile(string path, string token) - { - this.Path = path; - this.Token = token; - this.Hash = null; - } - - /// - /// Cabinet compress file. - /// - /// Path to file to add. - /// The token for the file. - /// Hash 1 - /// Hash 2 - /// Hash 3 - /// Hash 4 - public CabinetCompressFile(string path, string token, int hash1, int hash2, int hash3, int hash4) - { - this.Path = path; - this.Token = token; - this.Hash = new[] { hash1, hash2, hash3, hash4 }; - } - - /// - /// Gets the path to the file to compress. - /// - public string Path { get; } - - /// - /// Gets the token for the file to compress. - /// - public string Token { get; } - - /// - /// Gets the hash of the file to compress. - /// - public int[] Hash { get; } - - internal string ToWixNativeStdinLine() - { - if (this.Hash == null) - { - return $"{this.Path}\t{this.Token}"; - } - else - { - return $"{this.Path}\t{this.Token}\t{this.Hash[0]}\t{this.Hash[1]}\t{this.Hash[2]}\t{this.Hash[3]}"; - } - } - } -} diff --git a/src/WixToolset.Core.Native/CabinetFileInfo.cs b/src/WixToolset.Core.Native/CabinetFileInfo.cs deleted file mode 100644 index 07387191..00000000 --- a/src/WixToolset.Core.Native/CabinetFileInfo.cs +++ /dev/null @@ -1,63 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native -{ - using System; - - /// - /// Properties of a file in a cabinet. - /// - public sealed class CabinetFileInfo - { - /// - /// Constructs CabinetFileInfo - /// - /// File Id - /// Size of file - /// Last modified date - /// Last modified time - public CabinetFileInfo(string fileId, int size, int date, int time) - { - this.FileId = fileId; - this.Size = size; - this.Date = date; - this.Time = time; - } - - /// - /// Gets the file Id of the file. - /// - /// file Id - public string FileId { get; } - - /// - /// Gets modified date (DOS format). - /// - public int Date { get; } - - /// - /// Gets modified time (DOS format). - /// - public int Time { get; } - - /// - /// Gets the size of the file in bytes. - /// - public int Size { get; } - - /// - /// Compares this file info's date and time with another datetime. - /// - /// Date and time to compare with/ - /// - /// For some reason DateTime.ToLocalTime() does not match kernel32.dll FileTimeToLocalFileTime(). - /// Since cabinets store date and time with the kernel32.dll functions, we need to convert DateTime - /// to local file time using the kernel32.dll functions. - /// - public bool SameAsDateTime(DateTime dateTime) - { - DateTimeInterop.DateTimeToCabDateAndTime(dateTime, out var cabDate, out var cabTime); - return this.Date == cabDate && this.Time == cabTime; - } - } -} diff --git a/src/WixToolset.Core.Native/DateTimeInterop.cs b/src/WixToolset.Core.Native/DateTimeInterop.cs deleted file mode 100644 index d2a0ba2b..00000000 --- a/src/WixToolset.Core.Native/DateTimeInterop.cs +++ /dev/null @@ -1,48 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native -{ - using System; - using System.Runtime.InteropServices; - - /// - /// Interop class for the date/time handling. - /// - internal static class DateTimeInterop - { - /// - /// Converts DateTime to MS-DOS date and time which cabinet uses. - /// - /// DateTime - /// MS-DOS date - /// MS-DOS time - public static void DateTimeToCabDateAndTime(DateTime dateTime, out ushort cabDate, out ushort cabTime) - { - // dateTime.ToLocalTime() does not match FileTimeToLocalFileTime() for some reason. - // so we need to call FileTimeToLocalFileTime() from kernel32.dll. - long filetime = dateTime.ToFileTime(); - long localTime = 0; - FileTimeToLocalFileTime(ref filetime, ref localTime); - FileTimeToDosDateTime(ref localTime, out cabDate, out cabTime); - } - - /// - /// Converts file time to a local file time. - /// - /// file time - /// local file time - /// true if successful, false otherwise - [DllImport("kernel32.dll", SetLastError = true)] - private static extern bool FileTimeToLocalFileTime(ref long fileTime, ref long localTime); - - /// - /// Converts file time to a MS-DOS time. - /// - /// file time - /// MS-DOS date - /// MS-DOS time - /// true if successful, false otherwise - [DllImport("kernel32.dll", SetLastError = true)] - private static extern bool FileTimeToDosDateTime(ref long fileTime, out ushort wFatDate, out ushort wFatTime); - } -} diff --git a/src/WixToolset.Core.Native/FileSystem.cs b/src/WixToolset.Core.Native/FileSystem.cs deleted file mode 100644 index d843a9e8..00000000 --- a/src/WixToolset.Core.Native/FileSystem.cs +++ /dev/null @@ -1,78 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native -{ - using System; - using System.Collections.Generic; - using System.IO; - using System.Runtime.InteropServices; - using System.Security.AccessControl; - - /// - /// File system helpers. - /// - public static class FileSystem - { - /// - /// Copies a file. - /// - /// The file to copy. - /// The destination file. - /// Allow hardlinks. - public static void CopyFile(string source, string destination, bool allowHardlink) - { - EnsureDirectoryWithoutFile(destination); - - if (!allowHardlink || !CreateHardLink(destination, source, IntPtr.Zero)) - { -#if DEBUG - var er = Marshal.GetLastWin32Error(); -#endif - - File.Copy(source, destination, overwrite: true); - } - } - - /// - /// Moves a file. - /// - /// The file to move. - /// The destination file. - public static void MoveFile(string source, string destination) - { - EnsureDirectoryWithoutFile(destination); - - File.Move(source, destination); - } - - /// - /// Reset the ACLs on a set of files. - /// - /// The list of file paths to set ACLs. - public static void ResetAcls(IEnumerable files) - { - var aclReset = new FileSecurity(); - aclReset.SetAccessRuleProtection(false, false); - - foreach (var file in files) - { - new FileInfo(file).SetAccessControl(aclReset); - } - } - - private static void EnsureDirectoryWithoutFile(string path) - { - File.Delete(path); - - var directory = Path.GetDirectoryName(path); - - if (!String.IsNullOrEmpty(directory)) - { - Directory.CreateDirectory(directory); - } - } - - [DllImport("Kernel32.dll", CharSet = CharSet.Unicode, SetLastError = true)] - private static extern bool CreateHardLink(string lpFileName, string lpExistingFileName, IntPtr lpSecurityAttributes); - } -} diff --git a/src/WixToolset.Core.Native/IWindowsInstallerValidatorCallback.cs b/src/WixToolset.Core.Native/IWindowsInstallerValidatorCallback.cs deleted file mode 100644 index f4aff134..00000000 --- a/src/WixToolset.Core.Native/IWindowsInstallerValidatorCallback.cs +++ /dev/null @@ -1,27 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native -{ - /// - /// Callbacks during validation. - /// - public interface IWindowsInstallerValidatorCallback - { - /// - /// Indicates if the validator callback encountered an error. - /// - bool EncounteredError { get; } - - /// - /// Validation blocked by another Windows Installer operation. - /// - void ValidationBlocked(); - - /// - /// Validation message from an ICE. - /// - /// The validation message. - /// True if validation should continue; otherwise cancel the validation. - bool ValidationMessage(ValidationMessage message); - } -} diff --git a/src/WixToolset.Core.Native/Msi/Database.cs b/src/WixToolset.Core.Native/Msi/Database.cs deleted file mode 100644 index b9c5c35b..00000000 --- a/src/WixToolset.Core.Native/Msi/Database.cs +++ /dev/null @@ -1,262 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - using System.IO; - using System.Threading; - - /// - /// Wrapper class for managing MSI API database handles. - /// - public sealed class Database : MsiHandle - { - private const int STG_E_LOCKVIOLATION = unchecked((int)0x80030021); - - /// - /// Constructor that opens an MSI database. - /// - /// Path to the database to be opened. - /// Persist mode to use when opening the database. - public Database(string path, OpenDatabase type) - { - var error = MsiInterop.MsiOpenDatabase(path, new IntPtr((int)type), out var handle); - if (0 != error) - { - throw new MsiException(error); - } - this.Handle = handle; - } - - /// - /// Maximum length of stream in an MSI database. - /// - public static int MsiMaxStreamNameLength => MsiInterop.MsiMaxStreamNameLength; - - /// - /// Apply a transform to the MSI. - /// - /// Path to transform to apply. - public void ApplyTransform(string transformFile) - { - // get the curret validation bits - var conditions = TransformErrorConditions.None; - using (var summaryInfo = new SummaryInformation(transformFile)) - { - try - { - var validationFlags = summaryInfo.GetNumericProperty(SummaryInformation.Transform.ValidationFlags); - conditions = (TransformErrorConditions)(validationFlags & 0xffff); - } - catch (FormatException) - { - // fallback to default of None - } - } - - this.ApplyTransform(transformFile, conditions); - } - - /// - /// Applies a transform to this database. - /// - /// Path to the transform file being applied. - /// Specifies the error conditions that are to be suppressed. - public void ApplyTransform(string transformFile, TransformErrorConditions errorConditions) - { - var error = MsiInterop.MsiDatabaseApplyTransform(this.Handle, transformFile, errorConditions); - if (0 != error) - { - throw new MsiException(error); - } - } - - /// - /// Commits changes made to the database. - /// - public void Commit() - { - // Retry this call 3 times to deal with an MSI internal locking problem. - const int retryWait = 300; - const int retryLimit = 3; - var error = 0; - - for (var i = 1; i <= retryLimit; ++i) - { - error = MsiInterop.MsiDatabaseCommit(this.Handle); - - if (0 == error) - { - return; - } - else - { - var exception = new MsiException(error); - - // We need to see if the error code is contained in any of the strings in ErrorInfo. - // Join the array together and search for the error code to cover the string array. - if (!String.Join(", ", exception.ErrorInfo).Contains(STG_E_LOCKVIOLATION.ToString())) - { - break; - } - - Console.Error.WriteLine(String.Format("Failed to create the database. Info: {0}. Retrying ({1} of {2})", String.Join(", ", exception.ErrorInfo), i, retryLimit)); - Thread.Sleep(retryWait); - } - } - - throw new MsiException(error); - } - - /// - /// Creates and populates the summary information stream of an existing transform file. - /// - /// Required database that does not include the changes. - /// The name of the generated transform file. - /// Required error conditions that should be suppressed when the transform is applied. - /// Required when the transform is applied to a database; - /// shows which properties should be validated to verify that this transform can be applied to the database. - public void CreateTransformSummaryInfo(Database referenceDatabase, string transformFile, TransformErrorConditions errorConditions, TransformValidations validations) - { - var error = MsiInterop.MsiCreateTransformSummaryInfo(this.Handle, referenceDatabase.Handle, transformFile, errorConditions, validations); - if (0 != error) - { - throw new MsiException(error); - } - } - - /// - /// Imports an installer text archive table (idt file) into an open database. - /// - /// Specifies the path to the file to import. - /// Attempted to import an IDT file with an invalid format or unsupported data. - /// Another error occured while importing the IDT file. - public void Import(string idtPath) - { - var folderPath = Path.GetFullPath(Path.GetDirectoryName(idtPath)); - var fileName = Path.GetFileName(idtPath); - - var error = MsiInterop.MsiDatabaseImport(this.Handle, folderPath, fileName); - if (1627 == error) // ERROR_FUNCTION_FAILED - { - throw new WixInvalidIdtException(idtPath); - } - else if (0 != error) - { - throw new MsiException(error); - } - } - - /// - /// Exports an installer table from an open database to a text archive file (idt file). - /// - /// Specifies the name of the table to export. - /// Specifies the name of the folder that contains archive files. If null or empty string, uses current directory. - /// Specifies the name of the exported table archive file. - public void Export(string tableName, string folderPath, string fileName) - { - if (String.IsNullOrEmpty(folderPath)) - { - folderPath = Environment.CurrentDirectory; - } - - var error = MsiInterop.MsiDatabaseExport(this.Handle, tableName, folderPath, fileName); - if (0 != error) - { - throw new MsiException(error); - } - } - - /// - /// Creates a transform that, when applied to the reference database, results in this database. - /// - /// Required database that does not include the changes. - /// The name of the generated transform file. This is optional. - /// true if a transform is generated; false if a transform is not generated because - /// there are no differences between the two databases. - public bool GenerateTransform(Database referenceDatabase, string transformFile) - { - var error = MsiInterop.MsiDatabaseGenerateTransform(this.Handle, referenceDatabase.Handle, transformFile, 0, 0); - if (0 != error && 0xE8 != error) // ERROR_NO_DATA(0xE8) means no differences were found - { - throw new MsiException(error); - } - - return (0xE8 != error); - } - - /// - /// Merges two databases together. - /// - /// The database to merge into the base database. - /// The name of the table to receive merge conflict information. - /// True if there were merge conflicts, otherwise false. - public bool Merge(Database mergeDatabase, string tableName) - { - var error = MsiInterop.MsiDatabaseMerge(this.Handle, mergeDatabase.Handle, tableName); - if (error == 1627) - { - return true; - } - else if (error != 0) - { - throw new MsiException(error); - } - - return false; - } - - /// - /// Prepares a database query and creates a View object. - /// - /// Specifies a SQL query string for querying the database. - /// A view object is returned if the query was successful. - public View OpenView(string query) - { - return new View(this, query); - } - - /// - /// Prepares and executes a database query and creates a View object. - /// - /// Specifies a SQL query string for querying the database. - /// A view object is returned if the query was successful. - public View OpenExecuteView(string query) - { - var view = new View(this, query); - - view.Execute(); - return view; - } - - /// - /// Verifies the existence or absence of a table. - /// - /// Table name to to verify the existence of. - /// Returns true if the table exists, false if it does not. - public bool TableExists(string tableName) - { - var result = MsiInterop.MsiDatabaseIsTablePersistent(this.Handle, tableName); - return MsiInterop.MSICONDITIONTRUE == result; - } - - /// - /// Returns a Record containing the names of all the primary - /// key columns for a specified table. - /// - /// Specifies the name of the table from which to obtain - /// primary key names. - /// Returns a Record containing the names of all the - /// primary key columns for a specified table. - public Record PrimaryKeys(string tableName) - { - var error = MsiInterop.MsiDatabaseGetPrimaryKeys(this.Handle, tableName, out var recordHandle); - if (error != 0) - { - throw new MsiException(error); - } - - return new Record(recordHandle); - } - } -} diff --git a/src/WixToolset.Core.Native/Msi/InstallLogModes.cs b/src/WixToolset.Core.Native/Msi/InstallLogModes.cs deleted file mode 100644 index f7012b35..00000000 --- a/src/WixToolset.Core.Native/Msi/InstallLogModes.cs +++ /dev/null @@ -1,111 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - - /// - /// Windows Installer log modes. - /// - [Flags] - public enum InstallLogModes - { - /// - /// Premature termination of installation. - /// - FatalExit = (1 << ((int)InstallMessage.FatalExit >> 24)), - - /// - /// The error messages are logged. - /// - Error = (1 << ((int)InstallMessage.Error >> 24)), - - /// - /// The warning messages are logged. - /// - Warning = (1 << ((int)InstallMessage.Warning >> 24)), - - /// - /// The user requests are logged. - /// - User = (1 << ((int)InstallMessage.User >> 24)), - - /// - /// The status messages that are not displayed are logged. - /// - Info = (1 << ((int)InstallMessage.Info >> 24)), - - /// - /// Request to determine a valid source location. - /// - ResolveSource = (1 << ((int)InstallMessage.ResolveSource >> 24)), - - /// - /// The was insufficient disk space. - /// - OutOfDiskSpace = (1 << ((int)InstallMessage.OutOfDiskSpace >> 24)), - - /// - /// The start of new installation actions are logged. - /// - ActionStart = (1 << ((int)InstallMessage.ActionStart >> 24)), - - /// - /// The data record with the installation action is logged. - /// - ActionData = (1 << ((int)InstallMessage.ActionData >> 24)), - - /// - /// The parameters for user-interface initialization are logged. - /// - CommonData = (1 << ((int)InstallMessage.CommonData >> 24)), - - /// - /// Logs the property values at termination. - /// - PropertyDump = (1 << ((int)InstallMessage.Progress >> 24)), - - /// - /// Sends large amounts of information to a log file not generally useful to users. - /// May be used for technical support. - /// - Verbose = (1 << ((int)InstallMessage.Initilize >> 24)), - - /// - /// Sends extra debugging information, such as handle creation information, to the log file. - /// - ExtraDebug = (1 << ((int)InstallMessage.Terminate >> 24)), - - /// - /// Progress bar information. This message includes information on units so far and total number of units. - /// See MsiProcessMessage for an explanation of the message format. - /// This message is only sent to an external user interface and is not logged. - /// - Progress = (1 << ((int)InstallMessage.Progress >> 24)), - - /// - /// If this is not a quiet installation, then the basic UI has been initialized. - /// If this is a full UI installation, the full UI is not yet initialized. - /// This message is only sent to an external user interface and is not logged. - /// - Initialize = (1 << ((int)InstallMessage.Initilize >> 24)), - - /// - /// If a full UI is being used, the full UI has ended. - /// If this is not a quiet installation, the basic UI has not yet ended. - /// This message is only sent to an external user interface and is not logged. - /// - Terminate = (1 << ((int)InstallMessage.Terminate >> 24)), - - /// - /// Sent prior to display of the full UI dialog. - /// This message is only sent to an external user interface and is not logged. - /// - ShowDialog = (1 << ((int)InstallMessage.ShowDialog >> 24)), - - /// - /// Files in use information. When this message is received, a FilesInUse Dialog should be displayed. - /// - FilesInUse = (1 << ((int)InstallMessage.FilesInUse >> 24)) - } -} diff --git a/src/WixToolset.Core.Native/Msi/InstallMessage.cs b/src/WixToolset.Core.Native/Msi/InstallMessage.cs deleted file mode 100644 index 35773e13..00000000 --- a/src/WixToolset.Core.Native/Msi/InstallMessage.cs +++ /dev/null @@ -1,88 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - - /// - /// Windows Installer message types. - /// - [Flags] - public enum InstallMessage - { - /// - /// Premature termination, possibly fatal out of memory. - /// - FatalExit = 0x00000000, - - /// - /// Formatted error message, [1] is message number in Error table. - /// - Error = 0x01000000, - - /// - /// Formatted warning message, [1] is message number in Error table. - /// - Warning = 0x02000000, - - /// - /// User request message, [1] is message number in Error table. - /// - User = 0x03000000, - - /// - /// Informative message for log, not to be displayed. - /// - Info = 0x04000000, - - /// - /// List of files in use that need to be replaced. - /// - FilesInUse = 0x05000000, - - /// - /// Request to determine a valid source location. - /// - ResolveSource = 0x06000000, - - /// - /// Insufficient disk space message. - /// - OutOfDiskSpace = 0x07000000, - - /// - /// Progress: start of action, [1] action name, [2] description, [3] template for ACTIONDATA messages. - /// - ActionStart = 0x08000000, - - /// - /// Action data. Record fields correspond to the template of ACTIONSTART message. - /// - ActionData = 0x09000000, - - /// - /// Progress bar information. See the description of record fields below. - /// - Progress = 0x0A000000, - - /// - /// To enable the Cancel button set [1] to 2 and [2] to 1. To disable the Cancel button set [1] to 2 and [2] to 0. - /// - CommonData = 0x0B000000, - - /// - /// Sent prior to UI initialization, no string data. - /// - Initilize = 0x0C000000, - - /// - /// Sent after UI termination, no string data. - /// - Terminate = 0x0D000000, - - /// - /// Sent prior to display or authored dialog or wizard. - /// - ShowDialog = 0x0E000000 - } -} diff --git a/src/WixToolset.Core.Native/Msi/InstallUILevels.cs b/src/WixToolset.Core.Native/Msi/InstallUILevels.cs deleted file mode 100644 index e84b5215..00000000 --- a/src/WixToolset.Core.Native/Msi/InstallUILevels.cs +++ /dev/null @@ -1,72 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - - /// - /// Windows Installer UI levels. - /// - [Flags] - public enum InstallUILevels - { - /// - /// No change in the UI level. However, if phWnd is not Null, the parent window can change. - /// - NoChange = 0, - - /// - /// The installer chooses an appropriate user interface level. - /// - Default = 1, - - /// - /// Completely silent installation. - /// - None = 2, - - /// - /// Simple progress and error handling. - /// - Basic = 3, - - /// - /// Authored user interface with wizard dialog boxes suppressed. - /// - Reduced = 4, - - /// - /// Authored user interface with wizards, progress, and errors. - /// - Full = 5, - - /// - /// If combined with the Basic value, the installer shows simple progress dialog boxes but - /// does not display a Cancel button on the dialog. This prevents users from canceling the install. - /// Available with Windows Installer version 2.0. - /// - HideCancel = 0x20, - - /// - /// If combined with the Basic value, the installer shows simple progress - /// dialog boxes but does not display any modal dialog boxes or error dialog boxes. - /// - ProgressOnly = 0x40, - - /// - /// If combined with any above value, the installer displays a modal dialog - /// box at the end of a successful installation or if there has been an error. - /// No dialog box is displayed if the user cancels. - /// - EndDialog = 0x80, - - /// - /// If this value is combined with the None value, the installer displays only the dialog - /// boxes used for source resolution. No other dialog boxes are shown. This value has no - /// effect if the UI level is not INSTALLUILEVEL_NONE. It is used with an external user - /// interface designed to handle all of the UI except for source resolution. In this case, - /// the installer handles source resolution. This value is only available with Windows Installer 2.0 and later. - /// - SourceResOnly = 0x100 - } -} diff --git a/src/WixToolset.Core.Native/Msi/Installer.cs b/src/WixToolset.Core.Native/Msi/Installer.cs deleted file mode 100644 index 8a45aaa8..00000000 --- a/src/WixToolset.Core.Native/Msi/Installer.cs +++ /dev/null @@ -1,138 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - using System.Diagnostics; - using System.Text; - - /// - /// A callback function that the installer calls for progress notification and error messages. - /// - /// Pointer to an application context. - /// This parameter can be used for error checking. - /// Specifies a combination of one message box style, - /// one message box icon type, one default button, and one installation message type. - /// Specifies the message text. - /// -1 for an error, 0 if no action was taken, 1 if OK, 3 to abort. - public delegate int InstallUIHandler(IntPtr context, uint messageType, string message); - - /// - /// Represents the Windows Installer, provides wrappers to - /// create the top-level objects and access their methods. - /// - public static class Installer - { - /// - /// Extacts the patch metadata as XML. - /// - /// Path to patch. - /// String XML. - public static string ExtractPatchXml(string path) - { - var buffer = new StringBuilder(65535); - var size = buffer.Capacity; - - var error = MsiInterop.MsiExtractPatchXMLData(path, 0, buffer, ref size); - if (234 == error) - { - buffer.EnsureCapacity(++size); - error = MsiInterop.MsiExtractPatchXMLData(path, 0, buffer, ref size); - } - - if (error != 0) - { - throw new MsiException(error); - } - - return buffer.ToString(); - } - - /// - /// Takes the path to a file and returns a 128-bit hash of that file. - /// - /// Path to file that is to be hashed. - /// The value in this column must be 0. This parameter is reserved for future use. - /// Int array that receives the returned file hash information. - public static void GetFileHash(string filePath, int options, out int[] hash) - { - var hashInterop = new MSIFILEHASHINFO(); - hashInterop.FileHashInfoSize = 20; - - var error = MsiInterop.MsiGetFileHash(filePath, Convert.ToUInt32(options), hashInterop); - if (0 != error) - { - throw new MsiException(error); - } - - Debug.Assert(20 == hashInterop.FileHashInfoSize); - - hash = new int[4]; - hash[0] = hashInterop.Data0; - hash[1] = hashInterop.Data1; - hash[2] = hashInterop.Data2; - hash[3] = hashInterop.Data3; - } - - /// - /// Returns the version string and language string in the format that the installer - /// expects to find them in the database. If you just want version information, set - /// lpLangBuf and pcchLangBuf to zero. If you just want language information, set - /// lpVersionBuf and pcchVersionBuf to zero. - /// - /// Specifies the path to the file. - /// Returns the file version. Set to 0 for language information only. - /// Returns the file language. Set to 0 for version information only. - public static void GetFileVersion(string filePath, out string version, out string language) - { - var versionLength = 20; - var languageLength = 20; - var versionBuffer = new StringBuilder(versionLength); - var languageBuffer = new StringBuilder(languageLength); - - var error = MsiInterop.MsiGetFileVersion(filePath, versionBuffer, ref versionLength, languageBuffer, ref languageLength); - if (234 == error) - { - versionBuffer.EnsureCapacity(++versionLength); - languageBuffer.EnsureCapacity(++languageLength); - error = MsiInterop.MsiGetFileVersion(filePath, versionBuffer, ref versionLength, languageBuffer, ref languageLength); - } - else if (1006 == error) - { - // file has no version or language, so no error - error = 0; - } - - if (0 != error) - { - throw new MsiException(error); - } - - version = versionBuffer.ToString(); - language = languageBuffer.ToString(); - } - - /// - /// Enables an external user-interface handler. - /// - /// Specifies a callback function. - /// Specifies which messages to handle using the external message handler. - /// Pointer to an application context that is passed to the callback function. - /// The return value is the previously set external handler, or null if there was no previously set handler. - public static InstallUIHandler SetExternalUI(InstallUIHandler installUIHandler, int messageFilter, IntPtr context) - { - return MsiInterop.MsiSetExternalUI(installUIHandler, messageFilter, context); - } - - /// - /// Enables the installer's internal user interface. - /// - /// Specifies the level of complexity of the user interface. - /// Pointer to a window. This window becomes the owner of any user interface created. - /// The previous user interface level is returned. If an invalid dwUILevel is passed, then INSTALLUILEVEL_NOCHANGE is returned. - public static int SetInternalUI(int uiLevel, ref IntPtr hwnd) - { - return MsiInterop.MsiSetInternalUI(uiLevel, ref hwnd); - } - } -} diff --git a/src/WixToolset.Core.Native/Msi/MSIFILEHASHINFO.cs b/src/WixToolset.Core.Native/Msi/MSIFILEHASHINFO.cs deleted file mode 100644 index ae88ec7e..00000000 --- a/src/WixToolset.Core.Native/Msi/MSIFILEHASHINFO.cs +++ /dev/null @@ -1,34 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System.Runtime.InteropServices; - - /// - /// contains the file hash information returned by MsiGetFileHash and used in the MsiFileHash table. - /// - [StructLayout(LayoutKind.Explicit)] - public class MSIFILEHASHINFO - { - /// - /// - /// - [FieldOffset(0)] public uint FileHashInfoSize; - /// - /// - /// - [FieldOffset(4)] public int Data0; - /// - /// - /// - [FieldOffset(8)] public int Data1; - /// - /// - /// - [FieldOffset(12)] public int Data2; - /// - /// - /// - [FieldOffset(16)] public int Data3; - } -} diff --git a/src/WixToolset.Core.Native/Msi/ModifyView.cs b/src/WixToolset.Core.Native/Msi/ModifyView.cs deleted file mode 100644 index 989de174..00000000 --- a/src/WixToolset.Core.Native/Msi/ModifyView.cs +++ /dev/null @@ -1,75 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - /// - /// Enumeration of different modify modes. - /// - public enum ModifyView - { - /// - /// 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. - /// - Assign = 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. - - /// - /// Remove a row from the table. You must first call the Fetch 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. - /// - Delete = 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. - - /// - /// 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. - /// - Insert = 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. - - /// - /// 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. - /// - InsertTemporary = 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. - - /// - /// 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. - /// - Merge = 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. - - /// - /// Refreshes the information in the record. Must first call Fetch with the - /// same record. Fails for a deleted row. Works with read-write and read-only records. - /// - Refresh = 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. - - /// - /// Updates or deletes and inserts a record into a table. Must first call Fetch 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. - /// - Replace = 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. - - /// - /// 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. - /// - Seek = -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. - - /// - /// Updates an existing record. Non-primary keys only. Must first call Fetch. Fails with a - /// deleted record. Works only with read-write records. - /// - Update = 2, // Updates an existing record. Nonprimary keys only. Must first call MsiViewFetch. Fails with a deleted record. Works only with read-write records. - } -} diff --git a/src/WixToolset.Core.Native/Msi/MsiException.cs b/src/WixToolset.Core.Native/Msi/MsiException.cs deleted file mode 100644 index 07c83d81..00000000 --- a/src/WixToolset.Core.Native/Msi/MsiException.cs +++ /dev/null @@ -1,77 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - using System.ComponentModel; - - /// - /// Exception that wraps MsiGetLastError(). - /// - [Serializable] - public sealed class MsiException : Win32Exception - { - /// - /// Instantiate a new MsiException with a given error. - /// - /// The error code from the MsiXxx() function call. - public MsiException(int error) : base(error) - { - uint handle = MsiInterop.MsiGetLastErrorRecord(); - if (0 != handle) - { - using (Record record = new Record(handle)) - { - this.MsiError = record.GetInteger(1); - - int errorInfoCount = record.GetFieldCount() - 1; - this.ErrorInfo = new string[errorInfoCount]; - for (int i = 0; i < errorInfoCount; ++i) - { - this.ErrorInfo[i] = record.GetString(i + 2); - } - } - } - else - { - this.MsiError = 0; - this.ErrorInfo = new string[0]; - } - - this.Error = error; - } - - /// - /// Gets the error number. - /// - public int Error { get; private set; } - - /// - /// Gets the internal MSI error number. - /// - public int MsiError { get; private set; } - - /// - /// Gets any additional the error information. - /// - public string[] ErrorInfo { get; private set; } - - /// - /// Overrides Message property to return useful error message. - /// - public override string Message - { - get - { - if (0 == this.MsiError) - { - return base.Message; - } - else - { - return String.Format("Internal MSI failure. Win32 error: {0}, MSI error: {1}, detail: {2}", this.Error, this.MsiError, String.Join(", ", this.ErrorInfo)); - } - } - } - } -} diff --git a/src/WixToolset.Core.Native/Msi/MsiHandle.cs b/src/WixToolset.Core.Native/Msi/MsiHandle.cs deleted file mode 100644 index dc2ce605..00000000 --- a/src/WixToolset.Core.Native/Msi/MsiHandle.cs +++ /dev/null @@ -1,117 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - using System.ComponentModel; -#if !DEBUG - using System.Diagnostics; -#endif - using System.Threading; - - /// - /// Wrapper class for MSI handle. - /// - public abstract class MsiHandle : IDisposable - { - private bool disposed; - private uint handle; - private int owningThread; -#if DEBUG - private string creationStack; -#endif - - /// - /// MSI handle destructor. - /// - ~MsiHandle() - { - this.Dispose(false); - } - - /// - /// Gets or sets the MSI handle. - /// - /// The MSI handle. - internal uint Handle - { - get - { - if (this.disposed) - { - throw new ObjectDisposedException("MsiHandle"); - } - - return this.handle; - } - - set - { - if (this.disposed) - { - throw new ObjectDisposedException("MsiHandle"); - } - - this.handle = value; - this.owningThread = Thread.CurrentThread.ManagedThreadId; -#if DEBUG - this.creationStack = Environment.StackTrace; -#endif - } - } - - /// - /// Close the MSI handle. - /// - public void Close() - { - this.Dispose(); - } - - /// - /// Disposes the managed and unmanaged objects in this object. - /// - public void Dispose() - { - this.Dispose(true); - GC.SuppressFinalize(this); - } - - /// - /// Disposes the managed and unmanaged objects in this object. - /// - /// true to dispose the managed objects. - protected virtual void Dispose(bool disposing) - { - if (!this.disposed) - { - if (0 != this.handle) - { - if (Thread.CurrentThread.ManagedThreadId == this.owningThread) - { - int error = MsiInterop.MsiCloseHandle(this.handle); - if (0 != error) - { - throw new Win32Exception(error); - } - this.handle = 0; - } - else - { - // Don't try to close the handle on a different thread than it was opened. - // This will occasionally cause MSI to AV. - string message = String.Format("Leaked msi handle {0} created on thread {1} by type {2}. This handle cannot be closed on thread {3}", - this.handle, this.owningThread, this.GetType(), Thread.CurrentThread.ManagedThreadId); -#if DEBUG - throw new InvalidOperationException(String.Format("{0}. Created {1}", message, this.creationStack)); -#else - Debug.WriteLine(message); -#endif - } - } - - this.disposed = true; - } - } - } -} diff --git a/src/WixToolset.Core.Native/Msi/MsiInterop.cs b/src/WixToolset.Core.Native/Msi/MsiInterop.cs deleted file mode 100644 index 11ac4094..00000000 --- a/src/WixToolset.Core.Native/Msi/MsiInterop.cs +++ /dev/null @@ -1,408 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - using System.Text; - using System.Runtime.InteropServices; - - /// - /// Class exposing static functions and structs from MSI API. - /// - internal static class MsiInterop - { - // Patching constants - internal const int MsiMaxStreamNameLength = 62; // http://msdn2.microsoft.com/library/aa370551.aspx - - 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 - - /// - /// PInvoke of MsiCloseHandle. - /// - /// Handle to a database. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiCloseHandle", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiCloseHandle(uint database); - - /// - /// PInvoke of MsiCreateRecord - /// - /// Count of columns in the record. - /// Handle referencing the record. - [DllImport("msi.dll", EntryPoint = "MsiCreateRecord", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern uint MsiCreateRecord(int parameters); - - /// - /// Creates summary information of an existing transform to include validation and error conditions. - /// - /// The handle to the database that contains the new database summary information. - /// The handle to the database that contains the original summary information. - /// The name of the transform to which the summary information is added. - /// The error conditions that should be suppressed when the transform is applied. - /// Specifies the properties to be validated to verify that the transform can be applied to the database. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiCreateTransformSummaryInfoW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiCreateTransformSummaryInfo(uint database, uint referenceDatabase, string transformFile, TransformErrorConditions errorConditions, TransformValidations validations); - - /// - /// Applies a transform to a database. - /// - /// Handle to the database obtained from MsiOpenDatabase to transform. - /// Specifies the name of the transform file to apply. - /// Error conditions that should be suppressed. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiDatabaseApplyTransformW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiDatabaseApplyTransform(uint database, string transformFile, TransformErrorConditions errorConditions); - - /// - /// PInvoke of MsiDatabaseCommit. - /// - /// Handle to a databse. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiDatabaseCommit", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiDatabaseCommit(uint database); - - /// - /// PInvoke of MsiDatabaseExportW. - /// - /// Handle to a database. - /// Table name. - /// Folder path. - /// File name. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiDatabaseExportW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiDatabaseExport(uint database, string tableName, string folderPath, string fileName); - - /// - /// Generates a transform file of differences between two databases. - /// - /// Handle to the database obtained from MsiOpenDatabase that includes the changes. - /// Handle to the database obtained from MsiOpenDatabase that does not include the changes. - /// A null-terminated string that specifies the name of the transform file being generated. - /// This parameter can be null. If szTransformFile is null, you can use MsiDatabaseGenerateTransform to test whether two - /// databases are identical without creating a transform. If the databases are identical, the function returns ERROR_NO_DATA. - /// If the databases are different the function returns NOERROR. - /// This is a reserved argument and must be set to 0. - /// This is a reserved argument and must be set to 0. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiDatabaseGenerateTransformW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiDatabaseGenerateTransform(uint database, uint databaseReference, string transformFile, int reserved1, int reserved2); - - /// - /// PInvoke of MsiDatabaseImportW. - /// - /// Handle to a database. - /// Folder path. - /// File name. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiDatabaseImportW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiDatabaseImport(uint database, string folderPath, string fileName); - - /// - /// PInvoke of MsiDatabaseMergeW. - /// - /// The handle to the database obtained from MsiOpenDatabase. - /// The handle to the database obtained from MsiOpenDatabase to merge into the base database. - /// The name of the table to receive merge conflict information. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiDatabaseMergeW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiDatabaseMerge(uint database, uint databaseMerge, string tableName); - - /// - /// PInvoke of MsiDatabaseOpenViewW. - /// - /// Handle to a database. - /// SQL query. - /// View handle. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiDatabaseOpenViewW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiDatabaseOpenView(uint database, string query, out uint view); - - /// - /// PInvoke of MsiExtractPatchXMLDataW. - /// - /// Path to patch. - /// Reserved for future use. - /// Output XML data. - /// Count of characters in XML. - /// - [DllImport("msi.dll", EntryPoint = "MsiExtractPatchXMLDataW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiExtractPatchXMLData(string szPatchPath, int dwReserved, StringBuilder szXMLData, ref int pcchXMLData); - - /// - /// PInvoke of MsiGetFileHashW. - /// - /// File path. - /// Hash options (must be 0). - /// Buffer to recieve hash. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiGetFileHashW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiGetFileHash(string filePath, uint options, MSIFILEHASHINFO hash); - - /// - /// PInvoke of MsiGetFileVersionW. - /// - /// File path. - /// Buffer to receive version info. - /// Size of version buffer. - /// Buffer to recieve lang info. - /// Size of lang buffer. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiGetFileVersionW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiGetFileVersion(string filePath, StringBuilder versionBuf, ref int versionBufSize, StringBuilder langBuf, ref int langBufSize); - - /// - /// PInvoke of MsiGetLastErrorRecord. - /// - /// Handle to error record if one exists. - [DllImport("msi.dll", EntryPoint = "MsiGetLastErrorRecord", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern uint MsiGetLastErrorRecord(); - - /// - /// PInvoke of MsiDatabaseGetPrimaryKeysW. - /// - /// Handle to a database. - /// Table name. - /// Handle to receive resulting record. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiDatabaseGetPrimaryKeysW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiDatabaseGetPrimaryKeys(uint database, string tableName, out uint record); - - /// - /// PInvoke of MsiDoActionW. - /// - /// Handle to the installation provided to a DLL custom action or - /// obtained through MsiOpenPackage, MsiOpenPackageEx, or MsiOpenProduct. - /// Specifies the action to execute. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiDoActionW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiDoAction(uint product, string action); - - /// - /// PInvoke of MsiGetSummaryInformationW. Can use either database handle or database path as input. - /// - /// Handle to a database. - /// Path to a database. - /// Max number of updated values. - /// Handle to summary information. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiGetSummaryInformationW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiGetSummaryInformation(uint database, string databasePath, uint updateCount, ref uint summaryInfo); - - /// - /// PInvoke of MsiDatabaseIsTablePersitentW. - /// - /// Handle to a database. - /// Table name. - /// MSICONDITION - [DllImport("msi.dll", EntryPoint = "MsiDatabaseIsTablePersistentW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiDatabaseIsTablePersistent(uint database, string tableName); - - /// - /// PInvoke of MsiOpenDatabaseW. - /// - /// Path to database. - /// Persist mode. - /// Handle to database. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiOpenDatabaseW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiOpenDatabase(string databasePath, IntPtr persist, out uint database); - - /// - /// PInvoke of MsiOpenPackageW. - /// - /// The path to the package. - /// A pointer to a variable that receives the product handle. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiOpenPackageW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiOpenPackage(string packagePath, out uint product); - - /// - /// PInvoke of MsiRecordIsNull. - /// - /// MSI Record handle. - /// Index of field to check for null value. - /// true if the field is null, false if not, and an error code for any error. - [DllImport("msi.dll", EntryPoint = "MsiRecordIsNull", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiRecordIsNull(uint record, int field); - - /// - /// PInvoke of MsiRecordGetInteger. - /// - /// MSI Record handle. - /// Index of field to retrieve integer from. - /// Integer value. - [DllImport("msi.dll", EntryPoint = "MsiRecordGetInteger", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiRecordGetInteger(uint record, int field); - - /// - /// PInvoke of MsiRectordSetInteger. - /// - /// MSI Record handle. - /// Index of field to set integer value in. - /// Value to set field to. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiRecordSetInteger", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiRecordSetInteger(uint record, int field, int value); - - /// - /// PInvoke of MsiRecordGetStringW. - /// - /// MSI Record handle. - /// Index of field to get string value from. - /// Buffer to recieve value. - /// Size of buffer. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiRecordGetStringW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiRecordGetString(uint record, int field, StringBuilder valueBuf, ref int valueBufSize); - - /// - /// PInvoke of MsiRecordSetStringW. - /// - /// MSI Record handle. - /// Index of field to set string value in. - /// String value. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiRecordSetStringW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiRecordSetString(uint record, int field, string value); - - /// - /// PInvoke of MsiRecordSetStreamW. - /// - /// MSI Record handle. - /// Index of field to set stream value in. - /// Path to file to set stream value to. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiRecordSetStreamW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiRecordSetStream(uint record, int field, string filePath); - - /// - /// PInvoke of MsiRecordReadStreamW. - /// - /// MSI Record handle. - /// Index of field to read stream from. - /// Data buffer to recieve stream value. - /// Size of data buffer. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiRecordReadStream", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiRecordReadStream(uint record, int field, byte[] dataBuf, ref int dataBufSize); - - /// - /// PInvoke of MsiRecordGetFieldCount. - /// - /// MSI Record handle. - /// Count of fields in the record. - [DllImport("msi.dll", EntryPoint = "MsiRecordGetFieldCount", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiRecordGetFieldCount(uint record); - - /// - /// PInvoke of MsiSetExternalUIW. - /// - /// Specifies a callback function that conforms to the INSTALLUI_HANDLER specification. - /// Specifies which messages to handle using the external message handler. If the external - /// handler returns a non-zero result, then that message will not be sent to the UI, instead the message will be logged - /// if logging has been enabled. - /// Pointer to an application context that is passed to the callback function. - /// This parameter can be used for error checking. - /// The return value is the previously set external handler, or zero (0) if there was no previously set handler. - [DllImport("msi.dll", EntryPoint = "MsiSetExternalUIW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern InstallUIHandler MsiSetExternalUI(InstallUIHandler installUIHandler, int installLogMode, IntPtr context); - - /// - /// PInvoke of MsiSetInternalUI. - /// - /// Specifies the level of complexity of the user interface. - /// Pointer to a window. This window becomes the owner of any user interface created. - /// A pointer to the previous owner of the user interface is returned. - /// If this parameter is null, the owner of the user interface does not change. - /// The previous user interface level is returned. If an invalid dwUILevel is passed, then INSTALLUILEVEL_NOCHANGE is returned. - [DllImport("msi.dll", EntryPoint = "MsiSetInternalUI", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiSetInternalUI(int uiLevel, ref IntPtr hwnd); - - /// - /// PInvoke of MsiSummaryInfoGetPropertyW. - /// - /// Handle to summary info. - /// Property to get value from. - /// Data type of property. - /// Integer to receive integer value. - /// File time to receive file time value. - /// String buffer to receive string value. - /// Size of string buffer. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiSummaryInfoGetPropertyW", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiSummaryInfoGetProperty(uint summaryInfo, int property, out uint dataType, out int integerValue, ref System.Runtime.InteropServices.ComTypes.FILETIME fileTimeValue, StringBuilder stringValueBuf, ref int stringValueBufSize); - - /// - /// PInvoke of MsiViewGetColumnInfo. - /// - /// Handle to view. - /// Column info. - /// Handle for returned record. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiViewGetColumnInfo", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiViewGetColumnInfo(uint view, int columnInfo, out uint record); - - /// - /// PInvoke of MsiViewExecute. - /// - /// Handle of view to execute. - /// Handle to a record that supplies the parameters for the view. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiViewExecute", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiViewExecute(uint view, uint record); - - /// - /// PInvoke of MsiViewFetch. - /// - /// Handle of view to fetch a row from. - /// Handle to receive record info. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiViewFetch", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiViewFetch(uint view, out uint record); - - /// - /// PInvoke of MsiViewModify. - /// - /// Handle of view to modify. - /// Modify mode. - /// Handle of record. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiViewModify", CharSet = CharSet.Unicode, ExactSpelling = true)] - internal static extern int MsiViewModify(uint view, int modifyMode, uint record); - } -} diff --git a/src/WixToolset.Core.Native/Msi/OpenDatabase.cs b/src/WixToolset.Core.Native/Msi/OpenDatabase.cs deleted file mode 100644 index 18a78f77..00000000 --- a/src/WixToolset.Core.Native/Msi/OpenDatabase.cs +++ /dev/null @@ -1,40 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - /// - /// Enum of predefined persist modes used when opening a database. - /// - public enum OpenDatabase - { - /// - /// Open a database read-only, no persistent changes. - /// - ReadOnly = 0, - - /// - /// Open a database read/write in transaction mode. - /// - Transact = 1, - - /// - /// Open a database direct read/write without transaction. - /// - Direct = 2, - - /// - /// Create a new database, transact mode read/write. - /// - Create = 3, - - /// - /// Create a new database, direct mode read/write. - /// - CreateDirect = 4, - - /// - /// Indicates a patch file is being opened. - /// - OpenPatchFile = 32 - } -} diff --git a/src/WixToolset.Core.Native/Msi/Record.cs b/src/WixToolset.Core.Native/Msi/Record.cs deleted file mode 100644 index c25e76e2..00000000 --- a/src/WixToolset.Core.Native/Msi/Record.cs +++ /dev/null @@ -1,181 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - using System.ComponentModel; - using System.Text; - - /// - /// Wrapper class around msi.dll interop for a record. - /// - public sealed class Record : MsiHandle - { - /// - /// Creates a record with the specified number of fields. - /// - /// Number of fields in record. - public Record(int fieldCount) - { - this.Handle = MsiInterop.MsiCreateRecord(fieldCount); - if (0 == this.Handle) - { - throw new OutOfMemoryException(); - } - } - - /// - /// Creates a record from a handle. - /// - /// Handle to create record from. - internal Record(uint handle) - { - this.Handle = handle; - } - - /// - /// Gets a string value at specified location. - /// - /// Index into record to get string. - public string this[int field] - { - get => this.GetString(field); - set => this.SetString(field, value); - } - - /// - /// Determines if the value is null at the specified location. - /// - /// Index into record of the field to query. - /// true if the value is null, false otherwise. - public bool IsNull(int field) - { - var error = MsiInterop.MsiRecordIsNull(this.Handle, field); - - switch (error) - { - case 0: - return false; - case 1: - return true; - default: - throw new Win32Exception(error); - } - } - - /// - /// Gets integer value at specified location. - /// - /// Index into record to get integer - /// Integer value - public int GetInteger(int field) - { - return MsiInterop.MsiRecordGetInteger(this.Handle, field); - } - - /// - /// Sets integer value at specified location. - /// - /// Index into record to set integer. - /// Value to set into record. - public void SetInteger(int field, int value) - { - var error = MsiInterop.MsiRecordSetInteger(this.Handle, field, value); - if (0 != error) - { - throw new Win32Exception(error); - } - } - - /// - /// Gets string value at specified location. - /// - /// Index into record to get string. - /// String value - public string GetString(int field) - { - var bufferSize = 256; - var buffer = new StringBuilder(bufferSize); - var error = MsiInterop.MsiRecordGetString(this.Handle, field, buffer, ref bufferSize); - if (234 == error) - { - buffer.EnsureCapacity(++bufferSize); - error = MsiInterop.MsiRecordGetString(this.Handle, field, buffer, ref bufferSize); - } - - if (0 != error) - { - throw new Win32Exception(error); - } - - return (0 < buffer.Length ? buffer.ToString() : null); - } - - /// - /// Set string value at specified location - /// - /// Index into record to set string. - /// Value to set into record - public void SetString(int field, string value) - { - var error = MsiInterop.MsiRecordSetString(this.Handle, field, value); - if (0 != error) - { - throw new Win32Exception(error); - } - } - - /// - /// Get stream at specified location. - /// - /// Index into record to get stream. - /// buffer to receive bytes from stream. - /// Buffer size to read. - /// Stream read into string. - public int GetStream(int field, byte[] buffer, int requestedBufferSize) - { - var bufferSize = buffer.Length; - if (requestedBufferSize > 0) - { - bufferSize = requestedBufferSize; - } - - var error = MsiInterop.MsiRecordReadStream(this.Handle, field, buffer, ref bufferSize); - if (0 != error) - { - throw new Win32Exception(error); - } - - return bufferSize; - } - - /// - /// Sets a stream at a specified location. - /// - /// Index into record to set stream. - /// Path to file to read into stream. - public void SetStream(int field, string path) - { - var error = MsiInterop.MsiRecordSetStream(this.Handle, field, path); - if (0 != error) - { - throw new Win32Exception(error); - } - } - - /// - /// Gets the number of fields in record. - /// - /// Count of fields in record. - public int GetFieldCount() - { - var size = MsiInterop.MsiRecordGetFieldCount(this.Handle); - if (0 > size) - { - throw new Win32Exception(); - } - - return size; - } - } -} diff --git a/src/WixToolset.Core.Native/Msi/Session.cs b/src/WixToolset.Core.Native/Msi/Session.cs deleted file mode 100644 index 743fb2be..00000000 --- a/src/WixToolset.Core.Native/Msi/Session.cs +++ /dev/null @@ -1,42 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System.Globalization; - - /// - /// Controls the installation process. - /// - public sealed class Session : MsiHandle - { - /// - /// Instantiate a new Session. - /// - /// The database to open. - public Session(Database database) - { - var packagePath = "#" + database.Handle.ToString(CultureInfo.InvariantCulture); - - var error = MsiInterop.MsiOpenPackage(packagePath, out var handle); - if (0 != error) - { - throw new MsiException(error); - } - - this.Handle = handle; - } - - /// - /// Executes a built-in action, custom action, or user-interface wizard action. - /// - /// Specifies the action to execute. - public void DoAction(string action) - { - var error = MsiInterop.MsiDoAction(this.Handle, action); - if (0 != error) - { - throw new MsiException(error); - } - } - } -} diff --git a/src/WixToolset.Core.Native/Msi/SummaryInformation.cs b/src/WixToolset.Core.Native/Msi/SummaryInformation.cs deleted file mode 100644 index da629df2..00000000 --- a/src/WixToolset.Core.Native/Msi/SummaryInformation.cs +++ /dev/null @@ -1,376 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - using System.Globalization; - using System.Text; - using FILETIME = System.Runtime.InteropServices.ComTypes.FILETIME; - - /// - /// Summary information for the MSI files. - /// - public sealed class SummaryInformation : MsiHandle - { - /// - /// Summary information properties for products. - /// - public enum Package - { - /// PID_CODEPAGE = code page of the summary information stream - CodePage = 1, - - /// PID_TITLE = a brief description of the package type - Title = 2, - - /// PID_SUBJECT = package name - PackageName = 3, - - /// PID_AUTHOR = manufacturer of the patch package - Manufacturer = 4, - - /// PID_KEYWORDS = list of keywords used by file browser - Keywords = 5, - - /// PID_COMMENTS = general purpose of the package - Comments = 6, - - /// PID_TEMPLATE = supported platforms and languages - PlatformsAndLanguages = 7, - - /// PID_LASTAUTHOR should be null for packages - Reserved8 = 8, - - /// PID_REVNUMBER = GUID package code - PackageCode = 9, - - /// PID_LASTPRINTED should be null for packages - Reserved11 = 11, - - /// PID_CREATED datetime when package was created - Created = 12, - - /// PID_SAVED datetime when package was last modified - LastModified = 13, - - /// PID_PAGECOUNT minimum required Windows Installer - InstallerRequirement = 14, - - /// PID_WORDCOUNT elevation privileges of package - FileAndElevatedFlags = 15, - - /// PID_CHARCOUNT should be null for patches - Reserved16 = 16, - - /// PID_APPLICATION tool used to create package - BuildTool = 18, - - /// PID_SECURITY = read-only attribute of the package - Security = 19, - } - - /// - /// Summary information properties for transforms. - /// - public enum Transform - { - /// PID_CODEPAGE = code page for the summary information stream - CodePage = 1, - - /// PID_TITLE = typically just "Transform" - Title = 2, - - /// PID_SUBJECT = original subject of target - TargetSubject = 3, - - /// PID_AUTHOR = original manufacturer of target - TargetManufacturer = 4, - - /// PID_KEYWORDS = keywords for the transform, typically including at least "Installer" - Keywords = 5, - - /// PID_COMMENTS = describes what this package does - Comments = 6, - - /// PID_TEMPLATE = target platform;language - TargetPlatformAndLanguage = 7, - - /// PID_LASTAUTHOR = updated platform;language - UpdatedPlatformAndLanguage = 8, - - /// PID_REVNUMBER = {productcode}version;{newproductcode}newversion;upgradecode - ProductCodes = 9, - - /// PID_LASTPRINTED should be null for transforms - Reserved11 = 11, - - ///.PID_CREATE_DTM = the timestamp when the transform was created - CreationTime = 12, - - /// PID_PAGECOUNT = minimum installer version - InstallerRequirement = 14, - - /// PID_CHARCOUNT = validation and error flags - ValidationFlags = 16, - - /// PID_APPNAME = the application that created the transform - CreatingApplication = 18, - - /// PID_SECURITY = whether read-only is enforced; should always be 4 for transforms - Security = 19, - } - - /// - /// Summary information properties for patches. - /// - public enum Patch - { - /// PID_CODEPAGE = code page of the summary information stream - CodePage = 1, - - /// PID_TITLE = a brief description of the package type - Title = 2, - - /// PID_SUBJECT = package name - PackageName = 3, - - /// PID_AUTHOR = manufacturer of the patch package - Manufacturer = 4, - - /// PID_KEYWORDS = alternate sources for the patch package - Sources = 5, - - /// PID_COMMENTS = general purpose of the patch package - Comments = 6, - - /// PID_TEMPLATE = semicolon delimited list of ProductCodes - ProductCodes = 7, - - /// PID_LASTAUTHOR = semicolon delimited list of transform names - TransformNames = 8, - - /// PID_REVNUMBER = GUID patch code - PatchCode = 9, - - /// PID_LASTPRINTED should be null for patches - Reserved11 = 11, - - /// PID_PAGECOUNT should be null for patches - Reserved14 = 14, - - /// PID_WORDCOUNT = minimum installer version - InstallerRequirement = 15, - - /// PID_CHARCOUNT should be null for patches - Reserved16 = 16, - - /// PID_SECURITY = read-only attribute of the patch package - Security = 19, - } - - /// - /// Summary information values for the InstallerRequirement property. - /// - public enum InstallerRequirement - { - /// Any version of the installer will do - Version10 = 1, - - /// At least 1.2 - Version12 = 2, - - /// At least 2.0 - Version20 = 3, - - /// At least 3.0 - Version30 = 4, - - /// At least 3.1 - Version31 = 5, - } - - /// - /// Instantiate a new SummaryInformation class from an open database. - /// - /// Database to retrieve summary information from. - public SummaryInformation(Database db) - { - if (null == db) - { - throw new ArgumentNullException(nameof(db)); - } - - uint handle = 0; - var error = MsiInterop.MsiGetSummaryInformation(db.Handle, null, 0, ref handle); - if (0 != error) - { - throw new MsiException(error); - } - this.Handle = handle; - } - - /// - /// Instantiate a new SummaryInformation class from a database file. - /// - /// The database file. - public SummaryInformation(string databaseFile) - { - if (null == databaseFile) - { - throw new ArgumentNullException(nameof(databaseFile)); - } - - uint handle = 0; - var error = MsiInterop.MsiGetSummaryInformation(0, databaseFile, 0, ref handle); - if (0 != error) - { - throw new MsiException(error); - } - this.Handle = handle; - } - - /// - /// Gets a summary information package property. - /// - /// The summary information package property. - /// The summary information property. - public string GetProperty(Package property) - { - return this.GetProperty((int)property); - } - - /// - /// Gets a summary information package property as a number. - /// - /// The summary information package property. - /// The summary information property. - public long GetNumericProperty(Package property) - { - return this.GetNumericProperty((int)property); - } - - /// - /// Gets a summary information patch property. - /// - /// The summary information patch property. - /// The summary information property. - public string GetProperty(Patch property) - { - return this.GetProperty((int)property); - } - - /// - /// Gets a summary information transform property. - /// - /// The summary information transform property. - /// The summary information property. - public long GetNumericProperty(Transform property) - { - return this.GetNumericProperty((int)property); - } - - /// - /// Gets a summary information property. - /// - /// Index of the summary information property. - /// The summary information property. - public string GetProperty(int index) - { - this.GetSummaryInformationValue(index, out var dataType, out var intValue, out var stringValue, out var timeValue); - - switch ((VT)dataType) - { - case VT.EMPTY: - return String.Empty; - - case VT.LPSTR: - return stringValue.ToString(); - - case VT.I2: - case VT.I4: - return Convert.ToString(intValue, CultureInfo.InvariantCulture); - - case VT.FILETIME: - var longFileTime = (((long)timeValue.dwHighDateTime) << 32) | unchecked((uint)timeValue.dwLowDateTime); - var dateTime = DateTime.FromFileTime(longFileTime); - return dateTime.ToString("yyyy/MM/dd HH:mm:ss", CultureInfo.InvariantCulture); - - default: - throw new InvalidOperationException(); - } - } - - /// - /// Gets a summary information property as a number. - /// - /// Index of the summary information property. - /// The summary information property. - public long GetNumericProperty(int index) - { - this.GetSummaryInformationValue(index, out var dataType, out var intValue, out var stringValue, out var timeValue); - - switch ((VT)dataType) - { - case VT.EMPTY: - return 0; - - case VT.LPSTR: - return Int64.Parse(stringValue.ToString(), CultureInfo.InvariantCulture); - - case VT.I2: - case VT.I4: - return intValue; - - case VT.FILETIME: - return (((long)timeValue.dwHighDateTime) << 32) | unchecked((uint)timeValue.dwLowDateTime); - - default: - throw new InvalidOperationException(); - } - } - - private void GetSummaryInformationValue(int index, out uint dataType, out int intValue, out StringBuilder stringValue, out FILETIME timeValue) - { - var bufSize = 64; - stringValue = new StringBuilder(bufSize); - timeValue.dwHighDateTime = 0; - timeValue.dwLowDateTime = 0; - - var error = MsiInterop.MsiSummaryInfoGetProperty(this.Handle, index, out dataType, out intValue, ref timeValue, stringValue, ref bufSize); - if (234 == error) - { - stringValue.EnsureCapacity(++bufSize); - error = MsiInterop.MsiSummaryInfoGetProperty(this.Handle, index, out dataType, out intValue, ref timeValue, stringValue, ref bufSize); - } - - if (0 != error) - { - throw new MsiException(error); - } - } - - /// - /// Variant types in the summary information table. - /// - private enum VT : uint - { - /// Variant has not been assigned. - EMPTY = 0, - - /// Null variant type. - NULL = 1, - - /// 16-bit integer variant type. - I2 = 2, - - /// 32-bit integer variant type. - I4 = 3, - - /// String variant type. - LPSTR = 30, - - /// Date time (FILETIME, converted to Variant time) variant type. - FILETIME = 64, - } - } -} diff --git a/src/WixToolset.Core.Native/Msi/TransformErrorConditions.cs b/src/WixToolset.Core.Native/Msi/TransformErrorConditions.cs deleted file mode 100644 index 313dceeb..00000000 --- a/src/WixToolset.Core.Native/Msi/TransformErrorConditions.cs +++ /dev/null @@ -1,58 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - - /// - /// The errors to suppress when applying a transform. - /// - [Flags] - public enum TransformErrorConditions - { - /// - /// None of the following conditions. - /// - None = 0x0, - - /// - /// Suppress error when adding a row that exists. - /// - AddExistingRow = 0x1, - - /// - /// Suppress error when deleting a row that does not exist. - /// - DeleteMissingRow = 0x2, - - /// - /// Suppress error when adding a table that exists. - /// - AddExistingTable = 0x4, - - /// - /// Suppress error when deleting a table that does not exist. - /// - DeleteMissingTable = 0x8, - - /// - /// Suppress error when updating a row that does not exist. - /// - UpdateMissingRow = 0x10, - - /// - /// Suppress error when transform and database code pages do not match, and their code pages are neutral. - /// - ChangeCodepage = 0x20, - - /// - /// Create the temporary _TransformView table when applying a transform. - /// - ViewTransform = 0x100, - - /// - /// Suppress all errors but the option to create the temporary _TransformView table. - /// - All = 0x3F - } -} diff --git a/src/WixToolset.Core.Native/Msi/TransformValidations.cs b/src/WixToolset.Core.Native/Msi/TransformValidations.cs deleted file mode 100644 index 52bddeaf..00000000 --- a/src/WixToolset.Core.Native/Msi/TransformValidations.cs +++ /dev/null @@ -1,73 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - - /// - /// The validation to run while applying a transform. - /// - [Flags] - public enum TransformValidations - { - /// - /// Do not validate properties. - /// - None = 0x0, - - /// - /// Default language must match base database. - /// - Language = 0x1, - - /// - /// Product must match base database. - /// - Product = 0x2, - - /// - /// Check major version only. - /// - MajorVersion = 0x8, - - /// - /// Check major and minor versions only. - /// - MinorVersion = 0x10, - - /// - /// Check major, minor, and update versions. - /// - UpdateVersion = 0x20, - - /// - /// Installed version < base version. - /// - NewLessBaseVersion = 0x40, - - /// - /// Installed version <= base version. - /// - NewLessEqualBaseVersion = 0x80, - - /// - /// Installed version = base version. - /// - NewEqualBaseVersion = 0x100, - - /// - /// Installed version >= base version. - /// - NewGreaterEqualBaseVersion = 0x200, - - /// - /// Installed version > base version. - /// - NewGreaterBaseVersion = 0x400, - - /// - /// UpgradeCode must match base database. - /// - UpgradeCode = 0x800 - } -} diff --git a/src/WixToolset.Core.Native/Msi/View.cs b/src/WixToolset.Core.Native/Msi/View.cs deleted file mode 100644 index 6305a9de..00000000 --- a/src/WixToolset.Core.Native/Msi/View.cs +++ /dev/null @@ -1,206 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - using System.Collections; - using System.Collections.Generic; - using System.Globalization; - - /// - /// Wrapper class for MSI API views. - /// - public sealed class View : MsiHandle - { - /// - /// Constructor that creates a view given a database handle and a query. - /// - /// Handle to the database to run the query on. - /// Query to be executed. - public View(Database db, string query) - { - if (null == db) - { - throw new ArgumentNullException(nameof(db)); - } - - if (null == query) - { - throw new ArgumentNullException(nameof(query)); - } - - var error = MsiInterop.MsiDatabaseOpenView(db.Handle, query, out var handle); - if (0 != error) - { - throw new MsiException(error); - } - - this.Handle = handle; - } - - /// - /// Enumerator that automatically disposes of the retrieved Records. - /// - public IEnumerable Records => new ViewEnumerable(this); - - /// - /// Executes a view with no customizable parameters. - /// - public void Execute() - { - this.Execute(null); - } - - /// - /// Executes a query substituing the values from the records into the customizable parameters - /// in the view. - /// - /// Record containing parameters to be substituded into the view. - public void Execute(Record record) - { - var error = MsiInterop.MsiViewExecute(this.Handle, null == record ? 0 : record.Handle); - if (0 != error) - { - throw new MsiException(error); - } - } - - /// - /// Fetches the next row in the view. - /// - /// Returns the fetched record; otherwise null. - public Record Fetch() - { - var error = MsiInterop.MsiViewFetch(this.Handle, out var recordHandle); - if (259 == error) - { - return null; - } - else if (0 != error) - { - throw new MsiException(error); - } - - return new Record(recordHandle); - } - - /// - /// Updates a fetched record. - /// - /// Type of modification mode. - /// Record to be modified. - public void Modify(ModifyView type, Record record) - { - var error = MsiInterop.MsiViewModify(this.Handle, Convert.ToInt32(type, CultureInfo.InvariantCulture), record.Handle); - if (0 != error) - { - throw new MsiException(error); - } - } - - /// - /// Get the column names in a record. - /// - /// - public Record GetColumnNames() - { - return this.GetColumnInfo(MsiInterop.MSICOLINFONAMES); - } - - /// - /// Get the column types in a record. - /// - /// - public Record GetColumnTypes() - { - return this.GetColumnInfo(MsiInterop.MSICOLINFOTYPES); - } - - /// - /// Returns a record containing column names or definitions. - /// - /// Specifies a flag indicating what type of information is needed. Either MSICOLINFO_NAMES or MSICOLINFO_TYPES. - /// The record containing information about the column. - public Record GetColumnInfo(int columnType) - { - - var error = MsiInterop.MsiViewGetColumnInfo(this.Handle, columnType, out var recordHandle); - if (0 != error) - { - throw new MsiException(error); - } - - return new Record(recordHandle); - } - - private class ViewEnumerable : IEnumerable - { - private readonly View view; - - public ViewEnumerable(View view) => this.view = view; - - public IEnumerator GetEnumerator() => new ViewEnumerator(this.view); - - IEnumerator IEnumerable.GetEnumerator() => new ViewEnumerator(this.view); - } - - private class ViewEnumerator : IEnumerator - { - private readonly View view; - private readonly List records = new List(); - private int position = -1; - private bool disposed; - - public ViewEnumerator(View view) => this.view = view; - - public Record Current => this.records[this.position]; - - object IEnumerator.Current => this.records[this.position]; - - public bool MoveNext() - { - if (this.position + 1 >= this.records.Count) - { - var record = this.view.Fetch(); - - if (record == null) - { - return false; - } - - this.records.Add(record); - this.position = this.records.Count - 1; - } - else - { - ++this.position; - } - - return true; - } - - public void Reset() => this.position = -1; - - public void Dispose() - { - this.Dispose(true); - } - - protected virtual void Dispose(bool disposing) - { - if (!this.disposed) - { - if (disposing) - { - foreach (var record in this.records) - { - record.Dispose(); - } - } - - this.disposed = true; - } - } - } - } -} diff --git a/src/WixToolset.Core.Native/Msi/WixInvalidIdtException.cs b/src/WixToolset.Core.Native/Msi/WixInvalidIdtException.cs deleted file mode 100644 index 268ddc11..00000000 --- a/src/WixToolset.Core.Native/Msi/WixInvalidIdtException.cs +++ /dev/null @@ -1,49 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msi -{ - using System; - using WixToolset.Data; - - /// - /// WiX invalid idt exception. - /// - [Serializable] - public sealed class WixInvalidIdtException : WixException - { - /// - /// Instantiate a new WixInvalidIdtException. - /// - public WixInvalidIdtException() - { - } - - /// - /// Instantiate a new WixInvalidIdtException. - /// - /// - /// - public WixInvalidIdtException(string message, Exception innerException) : base(message, innerException) - { - } - - /// - /// Instantiate a new WixInvalidIdtException. - /// - /// The invalid idt file. - public WixInvalidIdtException(string idtFile) : - base(ErrorMessages.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(ErrorMessages.InvalidIdt(new SourceLineNumber(idtFile), idtFile, tableName)) - { - } - } -} diff --git a/src/WixToolset.Core.Native/Msm/ConfigurationCallback.cs b/src/WixToolset.Core.Native/Msm/ConfigurationCallback.cs deleted file mode 100644 index 31b06d02..00000000 --- a/src/WixToolset.Core.Native/Msm/ConfigurationCallback.cs +++ /dev/null @@ -1,90 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msm -{ - using System; - using System.Collections; - using System.Globalization; - - /// - /// Callback object for configurable merge modules. - /// - public sealed class ConfigurationCallback : IMsmConfigureModule - { - private const int SOk = 0x0; - private const int SFalse = 0x1; - private readonly Hashtable configurationData; - - /// - /// Creates a ConfigurationCallback object. - /// - /// String to break up into name/value pairs. - public ConfigurationCallback(string configData) - { - if (String.IsNullOrEmpty(configData)) - { - throw new ArgumentNullException(nameof(configData)); - } - - var pairs = configData.Split(','); - this.configurationData = new Hashtable(pairs.Length); - for (var i = 0; i < pairs.Length; ++i) - { - var nameVal = pairs[i].Split('='); - var name = nameVal[0]; - var value = nameVal[1]; - - name = name.Replace("%2C", ","); - name = name.Replace("%3D", "="); - name = name.Replace("%25", "%"); - - value = value.Replace("%2C", ","); - value = value.Replace("%3D", "="); - value = value.Replace("%25", "%"); - - this.configurationData[name] = value; - } - } - - /// - /// Returns text data based on name. - /// - /// Name of value to return. - /// Out param to put configuration data into. - /// S_OK if value provided, S_FALSE if not. - public int ProvideTextData(string name, out string configData) - { - if (this.configurationData.Contains(name)) - { - configData = (string)this.configurationData[name]; - return SOk; - } - else - { - configData = null; - return SFalse; - } - } - - /// - /// Returns integer data based on name. - /// - /// Name of value to return. - /// Out param to put configuration data into. - /// S_OK if value provided, S_FALSE if not. - public int ProvideIntegerData(string name, out int configData) - { - if (this.configurationData.Contains(name)) - { - var val = (string)this.configurationData[name]; - configData = Convert.ToInt32(val, CultureInfo.InvariantCulture); - return SOk; - } - else - { - configData = 0; - return SFalse; - } - } - } -} diff --git a/src/WixToolset.Core.Native/Msm/IMsmConfigureModule.cs b/src/WixToolset.Core.Native/Msm/IMsmConfigureModule.cs deleted file mode 100644 index 468fb1fc..00000000 --- a/src/WixToolset.Core.Native/Msm/IMsmConfigureModule.cs +++ /dev/null @@ -1,32 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msm -{ - using System; - using System.Runtime.InteropServices; - - /// - /// Callback for configurable merge modules. - /// - [ComImport, Guid("AC013209-18A7-4851-8A21-2353443D70A0"), InterfaceType(ComInterfaceType.InterfaceIsIDispatch)] - public interface IMsmConfigureModule - { - /// - /// Callback to retrieve text data for configurable merge modules. - /// - /// Name of the data to be retrieved. - /// The data corresponding to the name. - /// The error code (HRESULT). - [PreserveSig] - int ProvideTextData([In, MarshalAs(UnmanagedType.BStr)] string name, [MarshalAs(UnmanagedType.BStr)] out string configData); - - /// - /// Callback to retrieve integer data for configurable merge modules. - /// - /// Name of the data to be retrieved. - /// The data corresponding to the name. - /// The error code (HRESULT). - [PreserveSig] - int ProvideIntegerData([In, MarshalAs(UnmanagedType.BStr)] string name, out int configData); - } -} diff --git a/src/WixToolset.Core.Native/Msm/IMsmError.cs b/src/WixToolset.Core.Native/Msm/IMsmError.cs deleted file mode 100644 index 4f1325a6..00000000 --- a/src/WixToolset.Core.Native/Msm/IMsmError.cs +++ /dev/null @@ -1,77 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msm -{ - using System; - using System.Runtime.InteropServices; - - /// - /// A merge error. - /// - [ComImport, Guid("0ADDA828-2C26-11D2-AD65-00A0C9AF11A6")] - public interface IMsmError - { - /// - /// Gets the type of merge error. - /// - /// The type of merge error. - MsmErrorType Type - { - get; - } - - /// - /// Gets the path information from the merge error. - /// - /// The path information from the merge error. - string Path - { - get; - } - - /// - /// Gets the language information from the merge error. - /// - /// The language information from the merge error. - short Language - { - get; - } - - /// - /// Gets the database table from the merge error. - /// - /// The database table from the merge error. - string DatabaseTable - { - get; - } - - /// - /// Gets the collection of database keys from the merge error. - /// - /// The collection of database keys from the merge error. - IMsmStrings DatabaseKeys - { - get; - } - - /// - /// Gets the module table from the merge error. - /// - /// The module table from the merge error. - string ModuleTable - { - get; - } - - /// - /// Gets the collection of module keys from the merge error. - /// - /// The collection of module keys from the merge error. - IMsmStrings ModuleKeys - { - get; - } - } -} diff --git a/src/WixToolset.Core.Native/Msm/IMsmErrors.cs b/src/WixToolset.Core.Native/Msm/IMsmErrors.cs deleted file mode 100644 index e1472376..00000000 --- a/src/WixToolset.Core.Native/Msm/IMsmErrors.cs +++ /dev/null @@ -1,32 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msm -{ - using System; - using System.Runtime.InteropServices; - - /// - /// Collection of merge errors. - /// - [ComImport, Guid("0ADDA82A-2C26-11D2-AD65-00A0C9AF11A6")] - public interface IMsmErrors - { - /// - /// Gets the IMsmError at the specified index. - /// - /// The one-based index of the IMsmError to get. - IMsmError this[int index] - { - get; - } - - /// - /// Gets the count of IMsmErrors in this collection. - /// - /// The count of IMsmErrors in this collection. - int Count - { - get; - } - } -} diff --git a/src/WixToolset.Core.Native/Msm/IMsmMerge2.cs b/src/WixToolset.Core.Native/Msm/IMsmMerge2.cs deleted file mode 100644 index 400249e7..00000000 --- a/src/WixToolset.Core.Native/Msm/IMsmMerge2.cs +++ /dev/null @@ -1,174 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msm -{ - using System; - using System.Runtime.InteropServices; - - /// - /// IMsmMerge2 interface. - /// - [ComImport, Guid("351A72AB-21CB-47ab-B7AA-C4D7B02EA305")] - public interface IMsmMerge2 - { - /// - /// The OpenDatabase method of the Merge object opens a Windows Installer installation - /// database, located at a specified path, that is to be merged with a module. - /// - /// Path to the database being opened. - void OpenDatabase(string path); - - /// - /// The OpenModule method of the Merge object opens a Windows Installer merge module - /// in read-only mode. A module must be opened before it can be merged with an installation database. - /// - /// Fully qualified file name pointing to a merge module. - /// A valid language identifier (LANGID). - void OpenModule(string fileName, short language); - - /// - /// The CloseDatabase method of the Merge object closes the currently open Windows Installer database. - /// - /// true if changes should be saved, false otherwise. - void CloseDatabase(bool commit); - - /// - /// The CloseModule method of the Merge object closes the currently open Windows Installer merge module. - /// - void CloseModule(); - - /// - /// The OpenLog method of the Merge object opens a log file that receives progress and error messages. - /// If the log file already exists, the installer appends new messages. If the log file does not exist, - /// the installer creates a log file. - /// - /// Fully qualified filename pointing to a file to open or create. - void OpenLog(string fileName); - - /// - /// The CloseLog method of the Merge object closes the current log file. - /// - void CloseLog(); - - /// - /// The Log method of the Merge object writes a text string to the currently open log file. - /// - /// The text string to display. - void Log(string message); - - /// - /// Gets the errors from the last merge operation. - /// - /// The errors from the last merge operation. - IMsmErrors Errors - { - get; - } - - /// - /// Gets a collection of Dependency objects that enumerates a set of unsatisfied dependencies for the current database. - /// - /// A collection of Dependency objects that enumerates a set of unsatisfied dependencies for the current database. - object Dependencies - { - get; - } - - /// - /// The Merge method of the Merge object executes a merge of the current database and current - /// module. The merge attaches the components in the module to the feature identified by Feature. - /// The root of the module's directory tree is redirected to the location given by RedirectDir. - /// - /// The name of a feature in the database. - /// The key of an entry in the Directory table of the database. - /// This parameter may be NULL or an empty string. - void Merge(string feature, string redirectDir); - - /// - /// The Connect method of the Merge object connects a module to an additional feature. - /// The module must have already been merged into the database or will be merged into the database. - /// The feature must exist before calling this function. - /// - /// The name of a feature already existing in the database. - void Connect(string feature); - - /// - /// The ExtractCAB method of the Merge object extracts the embedded .cab file from a module and - /// saves it as the specified file. The installer creates this file if it does not already exist - /// and overwritten if it does exist. - /// - /// The fully qualified destination file. - void ExtractCAB(string fileName); - - /// - /// The ExtractFiles method of the Merge object extracts the embedded .cab file from a module - /// and then writes those files to the destination directory. - /// - /// The fully qualified destination directory. - void ExtractFiles(string path); - - /// - /// The MergeEx method of the Merge object is equivalent to the Merge function, except that it - /// takes an extra argument. The Merge method executes a merge of the current database and - /// current module. The merge attaches the components in the module to the feature identified - /// by Feature. The root of the module's directory tree is redirected to the location given by RedirectDir. - /// - /// The name of a feature in the database. - /// The key of an entry in the Directory table of the database. This parameter may - /// be NULL or an empty string. - /// The pConfiguration argument is an interface implemented by the client. The argument may - /// be NULL. The presence of this argument indicates that the client is capable of supporting the configuration - /// functionality, but does not obligate the client to provide configuration data for any specific configurable item. - void MergeEx(string feature, string redirectDir, IMsmConfigureModule configuration); - - /// - /// The ExtractFilesEx method of the Merge object extracts the embedded .cab file from a module and - /// then writes those files to the destination directory. - /// - /// The fully qualified destination directory. - /// Set to specify using long file names for path segments and final file names. - /// This is a list of fully-qualified paths for the files that were successfully extracted. - /// The list is empty if no files can be extracted. This argument may be null. No list is provided if pFilePaths is null. - void ExtractFilesEx(string path, bool longFileNames, ref IntPtr filePaths); - - /// - /// Gets a collection ConfigurableItem objects, each of which represents a single row from the ModuleConfiguration table. - /// - /// A collection ConfigurableItem objects, each of which represents a single row from the ModuleConfiguration table. - /// Semantically, each interface in the enumerator represents an item that can be configured by the module consumer. - /// The collection is a read-only collection and implements the standard read-only collection interfaces of Item(), Count() and _NewEnum(). - /// The IEnumMsmConfigItems enumerator implements Next(), Skip(), Reset(), and Clone() with the standard semantics. - object ConfigurableItems - { - get; - } - - /// - /// The CreateSourceImage method of the Merge object allows the client to extract the files from a module to - /// a source image on disk after a merge, taking into account changes to the module that might have been made - /// during module configuration. The list of files to be extracted is taken from the file table of the module - /// during the merge process. The list of files consists of every file successfully copied from the file table - /// of the module to the target database. File table entries that were not copied due to primary key conflicts - /// with existing rows in the database are not a part of this list. At image creation time, the directory for - /// each of these files comes from the open (post-merge) database. The path specified in the Path parameter is - /// the root of the source image for the install. fLongFileNames determines whether or not long file names are - /// used for both path segments and final file names. The function fails if no database is open, no module is - /// open, or no merge has been performed. - /// - /// The path of the root of the source image for the install. - /// Determines whether or not long file names are used for both path segments and final file names. - /// This is a list of fully-qualified paths for the files that were successfully extracted. - /// The list is empty if no files can be extracted. This argument may be null. No list is provided if pFilePaths is null. - void CreateSourceImage(string path, bool longFileNames, ref IntPtr filePaths); - - /// - /// The get_ModuleFiles function implements the ModuleFiles property of the GetFiles object. This function - /// returns the primary keys in the File table of the currently open module. The primary keys are returned - /// as a collection of strings. The module must be opened by a call to the OpenModule function before calling get_ModuleFiles. - /// - IMsmStrings ModuleFiles - { - get; - } - } -} diff --git a/src/WixToolset.Core.Native/Msm/IMsmStrings.cs b/src/WixToolset.Core.Native/Msm/IMsmStrings.cs deleted file mode 100644 index 41063bfa..00000000 --- a/src/WixToolset.Core.Native/Msm/IMsmStrings.cs +++ /dev/null @@ -1,32 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msm -{ - using System; - using System.Runtime.InteropServices; - - /// - /// A collection of strings. - /// - [ComImport, Guid("0ADDA827-2C26-11D2-AD65-00A0C9AF11A6")] - public interface IMsmStrings - { - /// - /// Gets the string at the specified index. - /// - /// The one-based index of the string to get. - string this[int index] - { - get; - } - - /// - /// Gets the count of strings in this collection. - /// - /// The count of strings in this collection. - int Count - { - get; - } - } -} diff --git a/src/WixToolset.Core.Native/Msm/MsmErrorType.cs b/src/WixToolset.Core.Native/Msm/MsmErrorType.cs deleted file mode 100644 index c67d37b4..00000000 --- a/src/WixToolset.Core.Native/Msm/MsmErrorType.cs +++ /dev/null @@ -1,154 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msm -{ - using System; - using System.Runtime.InteropServices; - - /// - /// Errors returned by merge operations. - /// - [Guid("0ADDA825-2C26-11D2-AD65-00A0C9AF11A6")] - public enum MsmErrorType - { - /// - /// A request was made to open a module with a language not supported by the module. - /// No more general language is supported by the module. - /// Adds msmErrorLanguageUnsupported to the Type property and the requested language - /// to the Language Property (Error Object). All Error object properties are empty. - /// The OpenModule function returns ERROR_INSTALL_LANGUAGE_UNSUPPORTED (as HRESULT). - /// - msmErrorLanguageUnsupported = 1, - - /// - /// A request was made to open a module with a supported language but the module has - /// an invalid language transform. Adds msmErrorLanguageFailed to the Type property - /// and the applied transform's language to the Language Property of the Error object. - /// This may not be the requested language if a more general language was used. - /// All other properties of the Error object are empty. The OpenModule function - /// returns ERROR_INSTALL_LANGUAGE_UNSUPPORTED (as HRESULT). - /// - msmErrorLanguageFailed = 2, - - /// - /// The module cannot be merged because it excludes, or is excluded by, another module - /// in the database. Adds msmErrorExclusion to the Type property of the Error object. - /// The ModuleKeys property or DatabaseKeys property contains the primary keys of the - /// excluded module's row in the ModuleExclusion table. If an existing module excludes - /// the module being merged, the excluded module's ModuleSignature information is added - /// to ModuleKeys. If the module being merged excludes an existing module, DatabaseKeys - /// contains the excluded module's ModuleSignature information. All other properties - /// are empty (or -1). - /// - msmErrorExclusion = 3, - - /// - /// Merge conflict during merge. The value of the Type property is set to - /// msmErrorTableMerge. The DatabaseTable property and DatabaseKeys property contain - /// the table name and primary keys of the conflicting row in the database. The - /// ModuleTable property and ModuleKeys property contain the table name and primary keys - /// of the conflicting row in the module. The ModuleTable and ModuleKeys entries may be - /// null if the row does not exist in the database. For example, if the conflict is in a - /// generated FeatureComponents table entry. On Windows Installer version 2.0, when - /// merging a configurable merge module, configuration may cause these properties to - /// refer to rows that do not exist in the module. - /// - msmErrorTableMerge = 4, - - /// - /// There was a problem resequencing a sequence table to contain the necessary merged - /// actions. The Type property is set to msmErrorResequenceMerge. The DatabaseTable - /// and DatabaseKeys properties contain the sequence table name and primary keys - /// (action name) of the conflicting row. The ModuleTable and ModuleKeys properties - /// contain the sequence table name and primary key (action name) of the conflicting row. - /// On Windows Installer version 2.0, when merging a configurable merge module, - /// configuration may cause these properties to refer to rows that do not exist in the module. - /// - msmErrorResequenceMerge = 5, - - /// - /// Not used. - /// - msmErrorFileCreate = 6, - - /// - /// There was a problem creating a directory to extract a file to disk. The Path property - /// contains the directory that could not be created. All other properties are empty or -1. - /// Not available with Windows Installer version 1.0. - /// - msmErrorDirCreate = 7, - - /// - /// A feature name is required to complete the merge, but no feature name was provided. - /// The Type property is set to msmErrorFeatureRequired. The DatabaseTable and DatabaseKeys - /// contain the table name and primary keys of the conflicting row. The ModuleTable and - /// ModuleKeys properties contain the table name and primary keys of the row cannot be merged. - /// On Windows Installer version 2.0, when merging a configurable merge module, configuration - /// may cause these properties to refer to rows that do not exist in the module. - /// If the failure is in a generated FeatureComponents table, the DatabaseTable and - /// DatabaseKeys properties are empty and the ModuleTable and ModuleKeys properties refer to - /// the row in the Component table causing the failure. - /// - msmErrorFeatureRequired = 8, - - /// - /// Available with Window Installer version 2.0. Substitution of a Null value into a - /// non-nullable column. This enters msmErrorBadNullSubstitution in the Type property and - /// enters "ModuleSubstitution" and the keys from the ModuleSubstitution table for this row - /// into the ModuleTable property and ModuleKeys property. All other properties of the Error - /// object are set to an empty string or -1. This error causes the immediate failure of the - /// merge and the MergeEx function to return E_FAIL. - /// - msmErrorBadNullSubstitution = 9, - - /// - /// Available with Window Installer version 2.0. Substitution of Text Format Type or Integer - /// Format Type into a Binary Type data column. This type of error returns - /// msmErrorBadSubstitutionType in the Type property and enters "ModuleSubstitution" and the - /// keys from the ModuleSubstitution table for this row into the ModuleTable property. - /// All other properties of the Error object are set to an empty string or -1. This error - /// causes the immediate failure of the merge and the MergeEx function to return E_FAIL. - /// - msmErrorBadSubstitutionType = 10, - - /// - /// Available with Window Installer Version 2.0. A row in the ModuleSubstitution table - /// references a configuration item not defined in the ModuleConfiguration table. - /// This type of error returns msmErrorMissingConfigItem in the Type property and enters - /// "ModuleSubstitution" and the keys from the ModuleSubstitution table for this row into - /// the ModuleTable property. All other properties of the Error object are set to an empty - /// string or -1. This error causes the immediate failure of the merge and the MergeEx - /// function to return E_FAIL. - /// - msmErrorMissingConfigItem = 11, - - /// - /// Available with Window Installer version 2.0. The authoring tool has returned a Null - /// value for an item marked with the msmConfigItemNonNullable attribute. An error of this - /// type returns msmErrorBadNullResponse in the Type property and enters "ModuleSubstitution" - /// and the keys from the ModuleSubstitution table for for the item into the ModuleTable property. - /// All other properties of the Error object are set to an empty string or -1. This error - /// causes the immediate failure of the merge and the MergeEx function to return E_FAIL. - /// - msmErrorBadNullResponse = 12, - - /// - /// Available with Window Installer version 2.0. The authoring tool returned a failure code - /// (not S_OK or S_FALSE) when asked for data. An error of this type will return - /// msmErrorDataRequestFailed in the Type property and enters "ModuleSubstitution" - /// and the keys from the ModuleSubstitution table for the item into the ModuleTable property. - /// All other properties of the Error object are set to an empty string or -1. This error - /// causes the immediate failure of the merge and the MergeEx function to return E_FAIL. - /// - msmErrorDataRequestFailed = 13, - - /// - /// Available with Windows Installer 2.0 and later versions. Indicates that an attempt was - /// made to merge a 64-bit module into a package that was not a 64-bit package. An error of - /// this type returns msmErrorPlatformMismatch in the Type property. All other properties of - /// the error object are set to an empty string or -1. This error causes the immediate failure - /// of the merge and causes the Merge function or MergeEx function to return E_FAIL. - /// - msmErrorPlatformMismatch = 14, - } -} diff --git a/src/WixToolset.Core.Native/Msm/MsmInterop.cs b/src/WixToolset.Core.Native/Msm/MsmInterop.cs deleted file mode 100644 index d2627904..00000000 --- a/src/WixToolset.Core.Native/Msm/MsmInterop.cs +++ /dev/null @@ -1,49 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Msm -{ - using System; - using System.Runtime.InteropServices; - - /// - /// Merge merge modules into an MSI file. - /// - [ComImport, Guid("F94985D5-29F9-4743-9805-99BC3F35B678")] - public class MsmMerge2 - { - } - - /// - /// Defines the standard COM IClassFactory interface. - /// - [ComImport, Guid("00000001-0000-0000-C000-000000000046")] - [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] - public interface IClassFactory - { - /// - /// - /// - [return: MarshalAs(UnmanagedType.IUnknown)] - object CreateInstance(IntPtr unkOuter, [MarshalAs(UnmanagedType.LPStruct)] Guid iid); - } - - /// - /// Contains native methods for merge operations. - /// - public static class MsmInterop - { - [DllImport("mergemod.dll", EntryPoint = "DllGetClassObject", PreserveSig = false)] - [return: MarshalAs(UnmanagedType.IUnknown)] - private static extern object MergeModGetClassObject([MarshalAs(UnmanagedType.LPStruct)] Guid clsid, [MarshalAs(UnmanagedType.LPStruct)] Guid iid); - - /// - /// Load the merge object directly from a local mergemod.dll without going through COM registration. - /// - /// Merge interface. - public static IMsmMerge2 GetMsmMerge() - { - var classFactory = (IClassFactory)MergeModGetClassObject(typeof(MsmMerge2).GUID, typeof(IClassFactory).GUID); - return (IMsmMerge2)classFactory.CreateInstance(IntPtr.Zero, typeof(IMsmMerge2).GUID); - } - } -} diff --git a/src/WixToolset.Core.Native/Ole32/Storage.cs b/src/WixToolset.Core.Native/Ole32/Storage.cs deleted file mode 100644 index 3e4c6af2..00000000 --- a/src/WixToolset.Core.Native/Ole32/Storage.cs +++ /dev/null @@ -1,377 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Ole32 -{ - using System; - using System.Runtime.InteropServices; - using FILETIME = System.Runtime.InteropServices.ComTypes.FILETIME; - using STATSTG = System.Runtime.InteropServices.ComTypes.STATSTG; - - /// - /// Wrapper for the compound storage file APIs. - /// - internal class Storage : IDisposable - { - private readonly IStorage storage; - private bool disposed; - - /// - /// Instantiate a new Storage. - /// - /// The native storage interface. - private Storage(IStorage storage) - { - this.storage = storage; - } - - /// - /// Storage destructor. - /// - ~Storage() - { - this.Dispose(); - } - - /// - /// The IEnumSTATSTG interface enumerates an array of STATSTG structures. - /// - [ComImport, Guid("0000000d-0000-0000-C000-000000000046"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] - private interface IEnumSTATSTG - { - /// - /// Gets a specified number of STATSTG structures. - /// - /// The number of STATSTG structures requested. - /// An array of STATSTG structures returned. - /// The number of STATSTG structures retrieved in the rgelt parameter. - /// The error code. - [PreserveSig] - uint Next(uint celt, [MarshalAs(UnmanagedType.LPArray), Out] STATSTG[] rgelt, out uint pceltFetched); - - /// - /// Skips a specified number of STATSTG structures in the enumeration sequence. - /// - /// The number of STATSTG structures to skip. - void Skip(uint celt); - - /// - /// Resets the enumeration sequence to the beginning of the STATSTG structure array. - /// - void Reset(); - - /// - /// Creates a new enumerator that contains the same enumeration state as the current STATSTG structure enumerator. - /// - /// The cloned IEnumSTATSTG interface. - [return: MarshalAs(UnmanagedType.Interface)] - IEnumSTATSTG Clone(); - } - - /// - /// The IStorage interface supports the creation and management of structured storage objects. - /// - [ComImport, Guid("0000000b-0000-0000-C000-000000000046"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] - private interface IStorage - { - /// - /// Creates and opens a stream object with the specified name contained in this storage object. - /// - /// The name of the newly created stream. - /// Specifies the access mode to use when opening the newly created stream. - /// Reserved for future use; must be zero. - /// Reserved for future use; must be zero. - /// On return, pointer to the location of the new IStream interface pointer. - void CreateStream(string pwcsName, uint grfMode, uint reserved1, uint reserved2, out IStream ppstm); - - /// - /// Opens an existing stream object within this storage object using the specified access permissions in grfMode. - /// - /// The name of the stream to open. - /// Reserved for future use; must be NULL. - /// Specifies the access mode to be assigned to the open stream. - /// Reserved for future use; must be zero. - /// A pointer to IStream pointer variable that receives the interface pointer to the newly opened stream object. - void OpenStream(string pwcsName, IntPtr reserved1, uint grfMode, uint reserved2, out IStream ppstm); - - /// - /// Creates and opens a new storage object nested within this storage object with the specified name in the specified access mode. - /// - /// The name of the newly created storage object. - /// A value that specifies the access mode to use when opening the newly created storage object. - /// Reserved for future use; must be zero. - /// Reserved for future use; must be zero. - /// A pointer, when successful, to the location of the IStorage pointer to the newly created storage object. - void CreateStorage(string pwcsName, uint grfMode, uint reserved1, uint reserved2, out IStorage ppstg); - - /// - /// Opens an existing storage object with the specified name in the specified access mode. - /// - /// The name of the storage object to open. - /// Must be NULL. - /// Specifies the access mode to use when opening the storage object. - /// Must be NULL. - /// Reserved for future use; must be zero. - /// When successful, pointer to the location of an IStorage pointer to the opened storage object. - void OpenStorage(string pwcsName, IStorage pstgPriority, uint grfMode, IntPtr snbExclude, uint reserved, out IStorage ppstg); - - /// - /// Copies the entire contents of an open storage object to another storage object. - /// - /// The number of elements in the array pointed to by rgiidExclude. - /// An array of interface identifiers (IIDs) that either the caller knows about and does not want - /// copied or that the storage object does not support, but whose state the caller will later explicitly copy. - /// A string name block (refer to SNB) that specifies a block of storage or stream objects that are not to be copied to the destination. - /// A pointer to the open storage object into which this storage object is to be copied. - void CopyTo(uint ciidExclude, IntPtr rgiidExclude, IntPtr snbExclude, IStorage pstgDest); - - /// - /// Copies or moves a substorage or stream from this storage object to another storage object. - /// - /// The name of the element in this storage object to be moved or copied. - /// IStorage pointer to the destination storage object. - /// The new name for the element in its new storage object. - /// Specifies whether the operation should be a move (STGMOVE_MOVE) or a copy (STGMOVE_COPY). - void MoveElementTo(string pwcsName, IStorage pstgDest, string pwcsNewName, uint grfFlags); - - /// - /// Reflects changes for a transacted storage object to the parent level. - /// - /// Controls how the changes are committed to the storage object. - void Commit(uint grfCommitFlags); - - /// - /// Discards all changes that have been made to the storage object since the last commit operation. - /// - void Revert(); - - /// - /// Returns an enumerator object that can be used to enumerate the storage and stream objects contained within this storage object. - /// - /// Reserved for future use; must be zero. - /// Reserved for future use; must be NULL. - /// Reserved for future use; must be zero. - /// Pointer to IEnumSTATSTG* pointer variable that receives the interface pointer to the new enumerator object. - void EnumElements(uint reserved1, IntPtr reserved2, uint reserved3, out IEnumSTATSTG ppenum); - - /// - /// Removes the specified storage or stream from this storage object. - /// - /// The name of the storage or stream to be removed. - void DestroyElement(string pwcsName); - - /// - /// Renames the specified storage or stream in this storage object. - /// - /// The name of the substorage or stream to be changed. - /// The new name for the specified substorage or stream. - void RenameElement(string pwcsOldName, string pwcsNewName); - - /// - /// Sets the modification, access, and creation times of the indicated storage element, if supported by the underlying file system. - /// - /// The name of the storage object element whose times are to be modified. - /// Either the new creation time for the element or NULL if the creation time is not to be modified. - /// Either the new access time for the element or NULL if the access time is not to be modified. - /// Either the new modification time for the element or NULL if the modification time is not to be modified. - void SetElementTimes(string pwcsName, FILETIME pctime, FILETIME patime, FILETIME pmtime); - - /// - /// Assigns the specified CLSID to this storage object. - /// - /// The CLSID that is to be associated with the storage object. - void SetClass(Guid clsid); - - /// - /// Stores up to 32 bits of state information in this storage object. - /// - /// Specifies the new values of the bits to set. - /// A binary mask indicating which bits in grfStateBits are significant in this call. - void SetStateBits(uint grfStateBits, uint grfMask); - - /// - /// Returns the STATSTG structure for this open storage object. - /// - /// On return, pointer to a STATSTG structure where this method places information about the open storage object. - /// Specifies that some of the members in the STATSTG structure are not returned, thus saving a memory allocation operation. - void Stat(out STATSTG pstatstg, uint grfStatFlag); - } - - /// - /// The IStream interface lets you read and write data to stream objects. - /// - [ComImport, Guid("0000000c-0000-0000-C000-000000000046"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] - private interface IStream - { - /// - /// Reads a specified number of bytes from the stream object into memory starting at the current seek pointer. - /// - /// A pointer to the buffer which the stream data is read into. - /// The number of bytes of data to read from the stream object. - /// A pointer to a ULONG variable that receives the actual number of bytes read from the stream object. - void Read([Out, MarshalAs(UnmanagedType.LPArray, SizeParamIndex = 1)] byte[] pv, int cb, IntPtr pcbRead); - - /// - /// Writes a specified number of bytes into the stream object starting at the current seek pointer. - /// - /// A pointer to the buffer that contains the data that is to be written to the stream. - /// The number of bytes of data to attempt to write into the stream. - /// A pointer to a ULONG variable where this method writes the actual number of bytes written to the stream object. - void Write([MarshalAs(UnmanagedType.LPArray, SizeParamIndex = 1)] byte[] pv, int cb, IntPtr pcbWritten); - - /// - /// Changes the seek pointer to a new location relative to the beginning of the stream, the end of the stream, or the current seek pointer. - /// - /// The displacement to be added to the location indicated by the dwOrigin parameter. - /// The origin for the displacement specified in dlibMove. - /// A pointer to the location where this method writes the value of the new seek pointer from the beginning of the stream. - void Seek(long dlibMove, int dwOrigin, IntPtr plibNewPosition); - - /// - /// Changes the size of the stream object. - /// - /// Specifies the new size of the stream as a number of bytes. - void SetSize(long libNewSize); - - /// - /// Copies a specified number of bytes from the current seek pointer in the stream to the current seek pointer in another stream. - /// - /// A pointer to the destination stream. - /// The number of bytes to copy from the source stream. - /// A pointer to the location where this method writes the actual number of bytes read from the source. - /// A pointer to the location where this method writes the actual number of bytes written to the destination. - void CopyTo(IStream pstm, long cb, IntPtr pcbRead, IntPtr pcbWritten); - - /// - /// Ensures that any changes made to a stream object open in transacted mode are reflected in the parent storage object. - /// - /// Controls how the changes for the stream object are committed. - void Commit(int grfCommitFlags); - - /// - /// Discards all changes that have been made to a transacted stream since the last call to IStream::Commit. - /// - void Revert(); - - /// - /// Restricts access to a specified range of bytes in the stream. - /// - /// Integer that specifies the byte offset for the beginning of the range. - /// Integer that specifies the length of the range, in bytes, to be restricted. - /// Specifies the restrictions being requested on accessing the range. - void LockRegion(long libOffset, long cb, int dwLockType); - - /// - /// Removes the access restriction on a range of bytes previously restricted with IStream::LockRegion. - /// - /// Specifies the byte offset for the beginning of the range. - /// Specifies, in bytes, the length of the range to be restricted. - /// Specifies the access restrictions previously placed on the range. - void UnlockRegion(long libOffset, long cb, int dwLockType); - - /// - /// Retrieves the STATSTG structure for this stream. - /// - /// Pointer to a STATSTG structure where this method places information about this stream object. - /// Specifies that this method does not return some of the members in the STATSTG structure, thus saving a memory allocation operation. - void Stat(out STATSTG pstatstg, int grfStatFlag); - - /// - /// Creates a new stream object that references the same bytes as the original stream but provides a separate seek pointer to those bytes. - /// - /// When successful, pointer to the location of an IStream pointer to the new stream object. - void Clone(out IStream ppstm); - } - - /// - /// Creates a new compound file storage object. - /// - /// The compound file being created. - /// Specifies the access mode to use when opening the new storage object. - /// The created Storage object. - public static Storage CreateDocFile(string storageFile, StorageMode mode) - { - var storage = NativeMethods.StgCreateDocfile(storageFile, (uint)mode, 0); - - return new Storage(storage); - } - - /// - /// Opens an existing root storage object in the file system. - /// - /// The file that contains the storage object to open. - /// Specifies the access mode to use to open the storage object. - /// The created Storage object. - public static Storage Open(string storageFile, StorageMode mode) - { - var storage = NativeMethods.StgOpenStorage(storageFile, IntPtr.Zero, (uint)mode, IntPtr.Zero, 0); - - return new Storage(storage); - } - - /// - /// Copies the entire contents of this open storage object into another Storage object. - /// - /// The destination Storage object. - public void CopyTo(Storage destinationStorage) - { - this.storage.CopyTo(0, IntPtr.Zero, IntPtr.Zero, destinationStorage.storage); - } - - /// - /// Opens an existing Storage object with the specified name according to the specified access mode. - /// - /// The name of the Storage object. - /// The opened Storage object. - public Storage OpenStorage(string name) - { - this.storage.OpenStorage(name, null, (uint)(StorageMode.Read | StorageMode.ShareExclusive), IntPtr.Zero, 0, out var subStorage); - - return new Storage(subStorage); - } - - /// - /// Disposes the managed and unmanaged objects in this object. - /// - public void Dispose() - { - if (!this.disposed) - { - Marshal.ReleaseComObject(this.storage); - - this.disposed = true; - } - - GC.SuppressFinalize(this); - } - - /// - /// The native methods. - /// - private static class NativeMethods - { - /// - /// Creates a new compound file storage object. - /// - /// The name for the compound file being created. - /// Specifies the access mode to use when opening the new storage object. - /// Reserved for future use; must be zero. - /// A pointer to the location of the IStorage pointer to the new storage object. - [DllImport("ole32.dll", PreserveSig = false)] - [return: MarshalAs(UnmanagedType.Interface)] - internal static extern IStorage StgCreateDocfile([MarshalAs(UnmanagedType.LPWStr)] string pwcsName, uint grfMode, uint reserved); - - /// - /// Opens an existing root storage object in the file system. - /// - /// The file that contains the storage object to open. - /// Most often NULL. - /// Specifies the access mode to use to open the storage object. - /// If not NULL, pointer to a block of elements in the storage to be excluded as the storage object is opened. - /// Indicates reserved for future use; must be zero. - /// A pointer to a IStorage* pointer variable that receives the interface pointer to the opened storage. - [DllImport("ole32.dll", PreserveSig = false)] - [return: MarshalAs(UnmanagedType.Interface)] - internal static extern IStorage StgOpenStorage([MarshalAs(UnmanagedType.LPWStr)] string pwcsName, IntPtr pstgPriority, uint grfMode, IntPtr snbExclude, uint reserved); - } - } -} diff --git a/src/WixToolset.Core.Native/Ole32/StorageMode.cs b/src/WixToolset.Core.Native/Ole32/StorageMode.cs deleted file mode 100644 index 24b60e4d..00000000 --- a/src/WixToolset.Core.Native/Ole32/StorageMode.cs +++ /dev/null @@ -1,55 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.Ole32 -{ - /// - /// Specifies the access mode to use when opening, creating, or deleting a storage object. - /// - internal enum StorageMode - { - /// - /// Indicates that the object is read-only, meaning that modifications cannot be made. - /// - Read = 0x0, - - /// - /// Enables you to save changes to the object, but does not permit access to its data. - /// - Write = 0x1, - - /// - /// Enables access and modification of object data. - /// - ReadWrite = 0x2, - - /// - /// Specifies that subsequent openings of the object are not denied read or write access. - /// - ShareDenyNone = 0x40, - - /// - /// Prevents others from subsequently opening the object in Read mode. - /// - ShareDenyRead = 0x30, - - /// - /// Prevents others from subsequently opening the object for Write or ReadWrite access. - /// - ShareDenyWrite = 0x20, - - /// - /// Prevents others from subsequently opening the object in any mode. - /// - ShareExclusive = 0x10, - - /// - /// Opens the storage object with exclusive access to the most recently committed version. - /// - Priority = 0x40000, - - /// - /// Indicates that an existing storage object or stream should be removed before the new object replaces it. - /// - Create = 0x1000, - } -} diff --git a/src/WixToolset.Core.Native/PatchAPI/PatchInterop.cs b/src/WixToolset.Core.Native/PatchAPI/PatchInterop.cs deleted file mode 100644 index 04f5a553..00000000 --- a/src/WixToolset.Core.Native/PatchAPI/PatchInterop.cs +++ /dev/null @@ -1,990 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native.PatchAPI -{ - using System; - using System.Collections.Generic; - using System.Diagnostics.CodeAnalysis; - using System.Globalization; - using System.Runtime.InteropServices; - using WixToolset.Data.Symbols; - - /// - /// Interop class for the mspatchc.dll. - /// - internal static class PatchInterop - { - // From WinError.h in the Platform SDK - internal const ushort FACILITY_WIN32 = 7; - - /// - /// Parse a number from text in either hex or decimal. - /// - /// Source value. Treated as hex if it starts 0x (or 0X), decimal otherwise. - /// Numeric value that source represents. - internal static uint ParseHexOrDecimal(string source) - { - var value = source.Trim(); - if (String.Equals(value.Substring(0, 2), "0x", StringComparison.OrdinalIgnoreCase)) - { - return UInt32.Parse(value.Substring(2), NumberStyles.AllowHexSpecifier, CultureInfo.InvariantCulture.NumberFormat); - } - else - { - return UInt32.Parse(value, CultureInfo.InvariantCulture.NumberFormat); - } - } - - /// - /// Create a binary delta file. - /// - /// Name of the delta file to create. - /// Name of updated file. - /// Optional paths to updated file's symbols. - /// Optional offsets to the delta retain sections in the updated file. - /// Optional array of target files. - /// Optional array of target files' symbol paths (must match basisFiles array). - /// Optional array of target files' delta ignore section lengths (must match basisFiles array)(each entry must match basisIgnoreOffsets entries). - /// Optional array of target files' delta ignore section offsets (must match basisFiles array)(each entry must match basisIgnoreLengths entries). - /// Optional array of target files' delta protect section lengths (must match basisFiles array)(each entry must match basisRetainOffsets and targetRetainOffsets entries). - /// Optional array of target files' delta protect section offsets (must match basisFiles array)(each entry must match basisRetainLengths and targetRetainOffsets entries). - /// ApiPatchingSymbolFlags value. - /// OptimizePatchSizeForLargeFiles value. - /// Flag to indicate retain ranges were ignored due to mismatch. - /// true if delta file was created, false if whole file should be used instead. - public static bool CreateDelta( - string deltaFile, - string targetFile, - string targetSymbolPath, - string targetRetainOffsets, - string[] basisFiles, - string[] basisSymbolPaths, - string[] basisIgnoreLengths, - string[] basisIgnoreOffsets, - string[] basisRetainLengths, - string[] basisRetainOffsets, - PatchSymbolFlags apiPatchingSymbolFlags, - bool optimizePatchSizeForLargeFiles, - out bool retainRangesIgnored - ) - { - retainRangesIgnored = false; - if (0 != (apiPatchingSymbolFlags & ~(PatchSymbolFlags.PatchSymbolNoImagehlp | PatchSymbolFlags.PatchSymbolNoFailures | PatchSymbolFlags.PatchSymbolUndecoratedToo))) - { - throw new ArgumentOutOfRangeException("apiPatchingSymbolFlags"); - } - - if (null == deltaFile || 0 == deltaFile.Length) - { - throw new ArgumentNullException("deltaFile"); - } - - if (null == targetFile || 0 == targetFile.Length) - { - throw new ArgumentNullException("targetFile"); - } - - if (null == basisFiles || 0 == basisFiles.Length) - { - return false; - } - var countOldFiles = (uint)basisFiles.Length; - - if (null != basisSymbolPaths) - { - if (0 != basisSymbolPaths.Length) - { - if ((uint)basisSymbolPaths.Length != countOldFiles) - { - throw new ArgumentOutOfRangeException("basisSymbolPaths"); - } - } - } - // a null basisSymbolPaths is allowed. - - if (null != basisIgnoreLengths) - { - if (0 != basisIgnoreLengths.Length) - { - if ((uint)basisIgnoreLengths.Length != countOldFiles) - { - throw new ArgumentOutOfRangeException("basisIgnoreLengths"); - } - } - } - else - { - basisIgnoreLengths = new string[countOldFiles]; - } - - if (null != basisIgnoreOffsets) - { - if (0 != basisIgnoreOffsets.Length) - { - if ((uint)basisIgnoreOffsets.Length != countOldFiles) - { - throw new ArgumentOutOfRangeException("basisIgnoreOffsets"); - } - } - } - else - { - basisIgnoreOffsets = new string[countOldFiles]; - } - - if (null != basisRetainLengths) - { - if (0 != basisRetainLengths.Length) - { - if ((uint)basisRetainLengths.Length != countOldFiles) - { - throw new ArgumentOutOfRangeException("basisRetainLengths"); - } - } - } - else - { - basisRetainLengths = new string[countOldFiles]; - } - - if (null != basisRetainOffsets) - { - if (0 != basisRetainOffsets.Length) - { - if ((uint)basisRetainOffsets.Length != countOldFiles) - { - throw new ArgumentOutOfRangeException("basisRetainOffsets"); - } - } - } - else - { - basisRetainOffsets = new string[countOldFiles]; - } - - var pod = new PatchOptionData(); - pod.symbolOptionFlags = apiPatchingSymbolFlags; - pod.newFileSymbolPath = targetSymbolPath; - pod.oldFileSymbolPathArray = basisSymbolPaths; - pod.extendedOptionFlags = 0; - var oldFileInfoArray = new PatchOldFileInfoW[countOldFiles]; - var newRetainOffsetArray = ((null == targetRetainOffsets) ? new string[0] : targetRetainOffsets.Split(',')); - for (uint i = 0; i < countOldFiles; ++i) - { - var ofi = new PatchOldFileInfoW(); - ofi.oldFileName = basisFiles[i]; - var ignoreLengthArray = ((null == basisIgnoreLengths[i]) ? new string[0] : basisIgnoreLengths[i].Split(',')); - var ignoreOffsetArray = ((null == basisIgnoreOffsets[i]) ? new string[0] : basisIgnoreOffsets[i].Split(',')); - var retainLengthArray = ((null == basisRetainLengths[i]) ? new string[0] : basisRetainLengths[i].Split(',')); - var retainOffsetArray = ((null == basisRetainOffsets[i]) ? new string[0] : basisRetainOffsets[i].Split(',')); - // Validate inputs - if (ignoreLengthArray.Length != ignoreOffsetArray.Length) - { - throw new ArgumentOutOfRangeException("basisIgnoreLengths"); - } - - if (retainLengthArray.Length != retainOffsetArray.Length) - { - throw new ArgumentOutOfRangeException("basisRetainLengths"); - } - - if (newRetainOffsetArray.Length != retainOffsetArray.Length) - { - // remove all retain range information - retainRangesIgnored = true; - for (uint j = 0; j < countOldFiles; ++j) - { - basisRetainLengths[j] = null; - basisRetainOffsets[j] = null; - } - retainLengthArray = new string[0]; - retainOffsetArray = new string[0]; - newRetainOffsetArray = new string[0]; - for (uint j = 0; j < oldFileInfoArray.Length; ++j) - { - oldFileInfoArray[j].retainRange = null; - } - } - - // Populate IgnoreRange structure - PatchIgnoreRange[] ignoreArray = null; - if (0 != ignoreLengthArray.Length) - { - ignoreArray = new PatchIgnoreRange[ignoreLengthArray.Length]; - for (var j = 0; j < ignoreLengthArray.Length; ++j) - { - var ignoreRange = new PatchIgnoreRange(); - ignoreRange.offsetInOldFile = ParseHexOrDecimal(ignoreOffsetArray[j]); - ignoreRange.lengthInBytes = ParseHexOrDecimal(ignoreLengthArray[j]); - ignoreArray[j] = ignoreRange; - } - ofi.ignoreRange = ignoreArray; - } - - PatchRetainRange[] retainArray = null; - if (0 != newRetainOffsetArray.Length) - { - retainArray = new PatchRetainRange[retainLengthArray.Length]; - for (var j = 0; j < newRetainOffsetArray.Length; ++j) - { - var retainRange = new PatchRetainRange(); - retainRange.offsetInOldFile = ParseHexOrDecimal(retainOffsetArray[j]); - retainRange.lengthInBytes = ParseHexOrDecimal(retainLengthArray[j]); - retainRange.offsetInNewFile = ParseHexOrDecimal(newRetainOffsetArray[j]); - retainArray[j] = retainRange; - } - ofi.retainRange = retainArray; - } - oldFileInfoArray[i] = ofi; - } - - if (CreatePatchFileExW( - countOldFiles, - oldFileInfoArray, - targetFile, - deltaFile, - PatchOptionFlags(optimizePatchSizeForLargeFiles), - pod, - null, - IntPtr.Zero)) - { - return true; - } - - // determine if this is an error or a need to use whole file. - var err = Marshal.GetLastWin32Error(); - switch (err) - { - case unchecked((int)ERROR_PATCH_BIGGER_THAN_COMPRESSED): - break; - - // too late to exclude this file -- should have been caught before - case unchecked((int)ERROR_PATCH_SAME_FILE): - default: - throw new System.ComponentModel.Win32Exception(err); - } - return false; - } - - /// - /// Extract the delta header. - /// - /// Name of delta file. - /// Name of file to create with the delta's header. - static public void ExtractDeltaHeader(string delta, string deltaHeader) - { - if (!ExtractPatchHeaderToFileW(delta, deltaHeader)) - { - throw new System.ComponentModel.Win32Exception(Marshal.GetLastWin32Error()); - } - } - - /// - /// Returns the PatchOptionFlags to use. - /// - /// True if optimizing for large files. - /// PATCH_OPTION_FLAG values - static private UInt32 PatchOptionFlags(bool optimizeForLargeFiles) - { - var flags = PATCH_OPTION_FAIL_IF_SAME_FILE | PATCH_OPTION_FAIL_IF_BIGGER | PATCH_OPTION_USE_LZX_BEST; - if (optimizeForLargeFiles) - { - flags |= PATCH_OPTION_USE_LZX_LARGE; - } - return flags; - } - - //--------------------------------------------------------------------- - // From PatchApi.h - //--------------------------------------------------------------------- - - // - // The following contants can be combined and used as the OptionFlags - // parameter in the patch creation apis. - - internal const uint PATCH_OPTION_USE_BEST = 0x00000000; // auto choose best (slower) - - internal const uint PATCH_OPTION_USE_LZX_BEST = 0x00000003; // auto choose best of LXZ A/B (but not large) - internal const uint PATCH_OPTION_USE_LZX_A = 0x00000001; // normal - internal const uint PATCH_OPTION_USE_LXZ_B = 0x00000002; // better on some x86 binaries - internal const uint PATCH_OPTION_USE_LZX_LARGE = 0x00000004; // better support for large files (requires 5.1 or higher applyer) - - internal const uint PATCH_OPTION_NO_BINDFIX = 0x00010000; // PE bound imports - internal const uint PATCH_OPTION_NO_LOCKFIX = 0x00020000; // PE smashed locks - internal const uint PATCH_OPTION_NO_REBASE = 0x00040000; // PE rebased image - internal const uint PATCH_OPTION_FAIL_IF_SAME_FILE = 0x00080000; // don't create if same - internal const uint PATCH_OPTION_FAIL_IF_BIGGER = 0x00100000; // fail if patch is larger than simply compressing new file (slower) - internal const uint PATCH_OPTION_NO_CHECKSUM = 0x00200000; // PE checksum zero - internal const uint PATCH_OPTION_NO_RESTIMEFIX = 0x00400000; // PE resource timestamps - internal const uint PATCH_OPTION_NO_TIMESTAMP = 0x00800000; // don't store new file timestamp in patch - internal const uint PATCH_OPTION_SIGNATURE_MD5 = 0x01000000; // use MD5 instead of CRC (reserved for future support) - internal const uint PATCH_OPTION_INTERLEAVE_FILES = 0x40000000; // better support for large files (requires 5.2 or higher applyer) - internal const uint PATCH_OPTION_RESERVED1 = 0x80000000; // (used internally) - - internal const uint PATCH_OPTION_VALID_FLAGS = 0xC0FF0007; - - // - // The following flags are used with PATCH_OPTION_DATA ExtendedOptionFlags: - // - - internal const uint PATCH_TRANSFORM_PE_RESOURCE_2 = 0x00000100; // better handling of PE resources (requires 5.2 or higher applyer) - internal const uint PATCH_TRANSFORM_PE_IRELOC_2 = 0x00000200; // better handling of PE stripped relocs (requires 5.2 or higher applyer) - - // - // In addition to the standard Win32 error codes, the following error codes may - // be returned via GetLastError() when one of the patch APIs fails. - - internal const uint ERROR_PATCH_ENCODE_FAILURE = 0xC00E3101; // create - internal const uint ERROR_PATCH_INVALID_OPTIONS = 0xC00E3102; // create - internal const uint ERROR_PATCH_SAME_FILE = 0xC00E3103; // create - internal const uint ERROR_PATCH_RETAIN_RANGES_DIFFER = 0xC00E3104; // create - internal const uint ERROR_PATCH_BIGGER_THAN_COMPRESSED = 0xC00E3105; // create - internal const uint ERROR_PATCH_IMAGEHLP_FALURE = 0xC00E3106; // create - - /// - /// Delegate type that the PatchAPI calls for progress notification. - /// - /// . - /// . - /// . - /// True for success - public delegate bool PatchProgressCallback( - IntPtr context, - uint currentPosition, - uint maxPosition - ); - - /// - /// Delegate type that the PatchAPI calls for patch symbol load information. - /// - /// . - /// . - /// . - /// . - /// . - /// . - /// . - /// . - /// ??? - public delegate bool PatchSymloadCallback( - uint whichFile, // 0 for new file, 1 for first old file, etc - [MarshalAs(UnmanagedType.LPStr)] string symbolFileName, - uint symType, // see SYM_TYPE in imagehlp.h - uint symbolFileCheckSum, - uint symbolFileTimeDate, - uint imageFileCheckSum, - uint imageFileTimeDate, - IntPtr context - ); - - /// - /// Wraps PATCH_IGNORE_RANGE - /// - [StructLayout(LayoutKind.Sequential)] - internal class PatchIgnoreRange - { - public uint offsetInOldFile; - public uint lengthInBytes; - } - - /// - /// Wraps PATCH_RETAIN_RANGE - /// - [StructLayout(LayoutKind.Sequential)] - internal class PatchRetainRange - { - public uint offsetInOldFile; - public uint lengthInBytes; - public uint offsetInNewFile; - } - - /// - /// Wraps PATCH_OLD_FILE_INFO (except for the OldFile~ portion) - /// - internal class PatchOldFileInfo - { - public PatchIgnoreRange[] ignoreRange; - public PatchRetainRange[] retainRange; - } - - /// - /// Wraps PATCH_OLD_FILE_INFO_W - /// - internal class PatchOldFileInfoW : PatchOldFileInfo - { - public string oldFileName; - } - - /// - /// Wraps each PATCH_INTERLEAVE_MAP Range - /// - [SuppressMessage("Microsoft.Performance", "CA1812:AvoidUninstantiatedInternalClasses"), StructLayout(LayoutKind.Sequential)] - internal class PatchInterleaveMapRange - { - public uint oldOffset; - public uint oldLength; - public uint newLength; - } - - /// - /// Wraps PATCH_INTERLEAVE_MAP - /// - internal class PatchInterleaveMap - { - public PatchInterleaveMapRange[] ranges = null; - } - - - /// - /// Wraps PATCH_OPTION_DATA - /// - [BestFitMapping(false, ThrowOnUnmappableChar = true)] - internal class PatchOptionData - { - public PatchSymbolFlags symbolOptionFlags; // PATCH_SYMBOL_xxx flags - [MarshalAs(UnmanagedType.LPStr)] public string newFileSymbolPath; // always ANSI, never Unicode - [MarshalAs(UnmanagedType.LPStr)] public string[] oldFileSymbolPathArray; // array[ OldFileCount ] - public uint extendedOptionFlags; - public PatchSymloadCallback symLoadCallback = null; - public IntPtr symLoadContext = IntPtr.Zero; - public PatchInterleaveMap[] interleaveMapArray = null; // array[ OldFileCount ] (requires 5.2 or higher applyer) - public uint maxLzxWindowSize = 0; // limit memory requirements (requires 5.2 or higher applyer) - } - - // - // Note that PATCH_OPTION_DATA contains LPCSTR paths, and no LPCWSTR (Unicode) - // path argument is available, even when used with one of the Unicode APIs - // such as CreatePatchFileW. This is because the unlerlying system services - // for symbol file handling (IMAGEHLP.DLL) only support ANSI file/path names. - // - - // - // A note about PATCH_RETAIN_RANGE specifiers with multiple old files: - // - // Each old version file must have the same RetainRangeCount, and the same - // retain range LengthInBytes and OffsetInNewFile values in the same order. - // Only the OffsetInOldFile values can differ between old foles for retain - // ranges. - // - - // - // The following prototypes are (some of the) interfaces for creating patches from files. - // - - /// - /// Creates a new delta. - /// - /// Size of oldFileInfoArray. - /// Target file information. - /// Name of updated file. - /// Name of delta to create. - /// PATCH_OPTION_xxx. - /// Optional PATCH_OPTION_DATA structure. - /// Delegate for progress callbacks. - /// Context for progress callback delegate. - /// true if successfull, sets Marshal.GetLastWin32Error() if not. - [DllImport("mspatchc.dll", SetLastError = true, CharSet = CharSet.Unicode, ExactSpelling = true)] - [return: MarshalAs(UnmanagedType.Bool)] - internal static extern bool CreatePatchFileExW( - uint oldFileCount, // maximum 255 - [MarshalAs(UnmanagedType.CustomMarshaler, MarshalTypeRef=typeof(PatchAPIMarshaler), MarshalCookie="PATCH_OLD_FILE_INFO_W")] - PatchOldFileInfoW[] oldFileInfoArray, - string newFileName, // input file (required) - string patchFileName, // output file (required) - uint optionFlags, - [MarshalAs(UnmanagedType.CustomMarshaler, MarshalTypeRef=typeof(PatchAPIMarshaler), MarshalCookie="PATCH_OPTION_DATA")] - PatchOptionData optionData, - [MarshalAs (UnmanagedType.FunctionPtr)] - PatchProgressCallback progressCallback, - IntPtr context - ); - - /// - /// Extracts delta header from delta. - /// - /// Name of delta file. - /// Name of file to create with delta header. - /// true if successfull, sets Marshal.GetLastWin32Error() if not. - [DllImport("mspatchc.dll", SetLastError = true, CharSet = CharSet.Unicode, ExactSpelling = true)] - [return: MarshalAs(UnmanagedType.Bool)] - internal static extern bool ExtractPatchHeaderToFileW( - string patchFileName, // input file - string patchHeaderFileName // output file - ); - - // TODO: Add rest of APIs to enable custom binders to perform more exhaustive checks - - /// - /// Marshals arguments for the CreatePatch~ APIs - /// - [SuppressMessage("Microsoft.Performance", "CA1812:AvoidUninstantiatedInternalClasses")] - internal class PatchAPIMarshaler : ICustomMarshaler - { - internal static ICustomMarshaler GetInstance(string cookie) - { - return new PatchAPIMarshaler(cookie); - } - - private enum MarshalType - { - PATCH_OPTION_DATA, - PATCH_OLD_FILE_INFO_W - }; - - private readonly PatchAPIMarshaler.MarshalType marshalType; - - private PatchAPIMarshaler(string cookie) - { - this.marshalType = (PatchAPIMarshaler.MarshalType)Enum.Parse(typeof(PatchAPIMarshaler.MarshalType), cookie); - } - - // - // Summary: - // Returns the size of the native data to be marshaled. - // - // Returns: - // The size in bytes of the native data. - public int GetNativeDataSize() - { - return Marshal.SizeOf(typeof(IntPtr)); - } - - // - // Summary: - // Performs necessary cleanup of the managed data when it is no longer needed. - // - // Parameters: - // ManagedObj: - // The managed object to be destroyed. - public void CleanUpManagedData(object ManagedObj) - { - } - - // - // Summary: - // Performs necessary cleanup of the unmanaged data when it is no longer needed. - // - // Parameters: - // pNativeData: - // A pointer to the unmanaged data to be destroyed. - public void CleanUpNativeData(IntPtr pNativeData) - { - if (IntPtr.Zero == pNativeData) - { - return; - } - - switch (this.marshalType) - { - case PatchAPIMarshaler.MarshalType.PATCH_OPTION_DATA: - this.CleanUpPOD(pNativeData); - break; - default: - this.CleanUpPOFI_A(pNativeData); - break; - } - } - - // - // Summary: - // Converts the managed data to unmanaged data. - // - // Parameters: - // ManagedObj: - // The managed object to be converted. - // - // Returns: - // Returns the COM view of the managed object. - public IntPtr MarshalManagedToNative(object ManagedObj) - { - if (null == ManagedObj) - { - return IntPtr.Zero; - } - - switch (this.marshalType) - { - case PatchAPIMarshaler.MarshalType.PATCH_OPTION_DATA: - return this.MarshalPOD(ManagedObj as PatchOptionData); - case PatchAPIMarshaler.MarshalType.PATCH_OLD_FILE_INFO_W: - return this.MarshalPOFIW_A(ManagedObj as PatchOldFileInfoW[]); - default: - throw new InvalidOperationException(); - } - } - - - // - // Summary: - // Converts the unmanaged data to managed data. - // - // Parameters: - // pNativeData: - // A pointer to the unmanaged data to be wrapped. - // - // Returns: - // Returns the managed view of the COM data. - public object MarshalNativeToManaged(IntPtr pNativeData) - { - return null; - } - - // Implementation ************************************************* - - // PATCH_OPTION_DATA offsets - private static readonly int symbolOptionFlagsOffset = Marshal.SizeOf(typeof(Int32)); - private static readonly int newFileSymbolPathOffset = 2 * Marshal.SizeOf(typeof(Int32)); - private static readonly int oldFileSymbolPathArrayOffset = 2 * Marshal.SizeOf(typeof(Int32)) + Marshal.SizeOf(typeof(IntPtr)); - private static readonly int extendedOptionFlagsOffset = 2 * Marshal.SizeOf(typeof(Int32)) + 2 * Marshal.SizeOf(typeof(IntPtr)); - private static readonly int symLoadCallbackOffset = 3 * Marshal.SizeOf(typeof(Int32)) + 2 * Marshal.SizeOf(typeof(IntPtr)); - private static readonly int symLoadContextOffset = 3 * Marshal.SizeOf(typeof(Int32)) + 3 * Marshal.SizeOf(typeof(IntPtr)); - private static readonly int interleaveMapArrayOffset = 3 * Marshal.SizeOf(typeof(Int32)) + 4 * Marshal.SizeOf(typeof(IntPtr)); - private static readonly int maxLzxWindowSizeOffset = 3 * Marshal.SizeOf(typeof(Int32)) + 5 * Marshal.SizeOf(typeof(IntPtr)); - private static readonly int patchOptionDataSize = 4 * Marshal.SizeOf(typeof(Int32)) + 5 * Marshal.SizeOf(typeof(IntPtr)); - - // PATCH_OLD_FILE_INFO offsets - private static readonly int oldFileOffset = Marshal.SizeOf(typeof(Int32)); - private static readonly int ignoreRangeCountOffset = Marshal.SizeOf(typeof(Int32)) + Marshal.SizeOf(typeof(IntPtr)); - private static readonly int ignoreRangeArrayOffset = 2 * Marshal.SizeOf(typeof(Int32)) + Marshal.SizeOf(typeof(IntPtr)); - private static readonly int retainRangeCountOffset = 2 * Marshal.SizeOf(typeof(Int32)) + 2 * Marshal.SizeOf(typeof(IntPtr)); - private static readonly int retainRangeArrayOffset = 3 * Marshal.SizeOf(typeof(Int32)) + 2 * Marshal.SizeOf(typeof(IntPtr)); - private static readonly int patchOldFileInfoSize = 3 * Marshal.SizeOf(typeof(Int32)) + 3 * Marshal.SizeOf(typeof(IntPtr)); - - // Methods and data used to preserve data needed for cleanup - - // This dictionary holds the quantity of items internal to each native structure that will need to be freed (the OldFileCount) - private static readonly Dictionary OldFileCounts = new Dictionary(); - private static readonly object OldFileCountsLock = new object(); - - private IntPtr CreateMainStruct(int oldFileCount) - { - int nativeSize; - switch (this.marshalType) - { - case PatchAPIMarshaler.MarshalType.PATCH_OPTION_DATA: - nativeSize = patchOptionDataSize; - break; - case PatchAPIMarshaler.MarshalType.PATCH_OLD_FILE_INFO_W: - nativeSize = oldFileCount * patchOldFileInfoSize; - break; - default: - throw new InvalidOperationException(); - } - - var native = Marshal.AllocCoTaskMem(nativeSize); - - lock (PatchAPIMarshaler.OldFileCountsLock) - { - PatchAPIMarshaler.OldFileCounts.Add(native, oldFileCount); - } - - return native; - } - - private static void ReleaseMainStruct(IntPtr native) - { - lock (PatchAPIMarshaler.OldFileCountsLock) - { - PatchAPIMarshaler.OldFileCounts.Remove(native); - } - Marshal.FreeCoTaskMem(native); - } - - private static int GetOldFileCount(IntPtr native) - { - lock (PatchAPIMarshaler.OldFileCountsLock) - { - return PatchAPIMarshaler.OldFileCounts[native]; - } - } - - // Helper methods - - private static IntPtr OptionalAnsiString(string managed) - { - return (null == managed) ? IntPtr.Zero : Marshal.StringToCoTaskMemAnsi(managed); - } - - private static IntPtr OptionalUnicodeString(string managed) - { - return (null == managed) ? IntPtr.Zero : Marshal.StringToCoTaskMemUni(managed); - } - - // string array must be of the same length as the number of old files - private static IntPtr CreateArrayOfStringA(string[] managed) - { - if (null == managed) - { - return IntPtr.Zero; - } - - var size = managed.Length * Marshal.SizeOf(typeof(IntPtr)); - var native = Marshal.AllocCoTaskMem(size); - - for (var i = 0; i < managed.Length; ++i) - { - Marshal.WriteIntPtr(native, i * Marshal.SizeOf(typeof(IntPtr)), OptionalAnsiString(managed[i])); - } - - return native; - } - - // string array must be of the same length as the number of old files - private static IntPtr CreateArrayOfStringW(string[] managed) - { - if (null == managed) - { - return IntPtr.Zero; - } - - var size = managed.Length * Marshal.SizeOf(typeof(IntPtr)); - var native = Marshal.AllocCoTaskMem(size); - - for (var i = 0; i < managed.Length; ++i) - { - Marshal.WriteIntPtr(native, i * Marshal.SizeOf(typeof(IntPtr)), OptionalUnicodeString(managed[i])); - } - - return native; - } - - private static IntPtr CreateInterleaveMapRange(PatchInterleaveMap managed) - { - if (null == managed) - { - return IntPtr.Zero; - } - - if (null == managed.ranges) - { - return IntPtr.Zero; - } - - if (0 == managed.ranges.Length) - { - return IntPtr.Zero; - } - - var native = Marshal.AllocCoTaskMem(Marshal.SizeOf(typeof(UInt32)) - + managed.ranges.Length * (Marshal.SizeOf(typeof(PatchInterleaveMap)))); - WriteUInt32(native, (uint)managed.ranges.Length); - - for (var i = 0; i < managed.ranges.Length; ++i) - { - Marshal.StructureToPtr(managed.ranges[i], (IntPtr)((Int64)native + i * Marshal.SizeOf(typeof(PatchInterleaveMap))), false); - } - return native; - } - - private static IntPtr CreateInterleaveMap(PatchInterleaveMap[] managed) - { - if (null == managed) - { - return IntPtr.Zero; - } - - var native = Marshal.AllocCoTaskMem(managed.Length * Marshal.SizeOf(typeof(IntPtr))); - - for (var i = 0; i < managed.Length; ++i) - { - Marshal.WriteIntPtr(native, i * Marshal.SizeOf(typeof(IntPtr)), CreateInterleaveMapRange(managed[i])); - } - - return native; - } - - private static void WriteUInt32(IntPtr native, uint data) - { - Marshal.WriteInt32(native, unchecked((int)data)); - } - - private static void WriteUInt32(IntPtr native, int offset, uint data) - { - Marshal.WriteInt32(native, offset, unchecked((int)data)); - } - - // Marshal operations - - private IntPtr MarshalPOD(PatchOptionData managed) - { - if (null == managed) - { - throw new ArgumentNullException("managed"); - } - - var native = this.CreateMainStruct(managed.oldFileSymbolPathArray.Length); - Marshal.WriteInt32(native, patchOptionDataSize); // SizeOfThisStruct - WriteUInt32(native, symbolOptionFlagsOffset, (uint)managed.symbolOptionFlags); - Marshal.WriteIntPtr(native, newFileSymbolPathOffset, PatchAPIMarshaler.OptionalAnsiString(managed.newFileSymbolPath)); - Marshal.WriteIntPtr(native, oldFileSymbolPathArrayOffset, PatchAPIMarshaler.CreateArrayOfStringA(managed.oldFileSymbolPathArray)); - WriteUInt32(native, extendedOptionFlagsOffset, managed.extendedOptionFlags); - - // GetFunctionPointerForDelegate() throws an ArgumentNullException if the delegate is null. - if (null == managed.symLoadCallback) - { - Marshal.WriteIntPtr(native, symLoadCallbackOffset, IntPtr.Zero); - } - else - { - Marshal.WriteIntPtr(native, symLoadCallbackOffset, Marshal.GetFunctionPointerForDelegate(managed.symLoadCallback)); - } - - Marshal.WriteIntPtr(native, symLoadContextOffset, managed.symLoadContext); - Marshal.WriteIntPtr(native, interleaveMapArrayOffset, PatchAPIMarshaler.CreateInterleaveMap(managed.interleaveMapArray)); - WriteUInt32(native, maxLzxWindowSizeOffset, managed.maxLzxWindowSize); - return native; - } - - private IntPtr MarshalPOFIW_A(PatchOldFileInfoW[] managed) - { - if (null == managed) - { - throw new ArgumentNullException("managed"); - } - - if (0 == managed.Length) - { - return IntPtr.Zero; - } - - var native = this.CreateMainStruct(managed.Length); - - for (var i = 0; i < managed.Length; ++i) - { - PatchAPIMarshaler.MarshalPOFIW(managed[i], (IntPtr)((Int64)native + i * patchOldFileInfoSize)); - } - - return native; - } - - private static void MarshalPOFIW(PatchOldFileInfoW managed, IntPtr native) - { - PatchAPIMarshaler.MarshalPOFI(managed, native); - Marshal.WriteIntPtr(native, oldFileOffset, PatchAPIMarshaler.OptionalUnicodeString(managed.oldFileName)); // OldFileName - } - - private static void MarshalPOFI(PatchOldFileInfo managed, IntPtr native) - { - Marshal.WriteInt32(native, patchOldFileInfoSize); // SizeOfThisStruct - WriteUInt32(native, ignoreRangeCountOffset, - (null == managed.ignoreRange) ? 0 : (uint)managed.ignoreRange.Length); // IgnoreRangeCount // maximum 255 - Marshal.WriteIntPtr(native, ignoreRangeArrayOffset, MarshalPIRArray(managed.ignoreRange)); // IgnoreRangeArray - WriteUInt32(native, retainRangeCountOffset, - (null == managed.retainRange) ? 0 : (uint)managed.retainRange.Length); // RetainRangeCount // maximum 255 - Marshal.WriteIntPtr(native, retainRangeArrayOffset, MarshalPRRArray(managed.retainRange)); // RetainRangeArray - } - - private static IntPtr MarshalPIRArray(PatchIgnoreRange[] array) - { - if (null == array) - { - return IntPtr.Zero; - } - - if (0 == array.Length) - { - return IntPtr.Zero; - } - - var native = Marshal.AllocCoTaskMem(array.Length * Marshal.SizeOf(typeof(PatchIgnoreRange))); - - for (var i = 0; i < array.Length; ++i) - { - Marshal.StructureToPtr(array[i], (IntPtr)((Int64)native + (i * Marshal.SizeOf(typeof(PatchIgnoreRange)))), false); - } - - return native; - } - - private static IntPtr MarshalPRRArray(PatchRetainRange[] array) - { - if (null == array) - { - return IntPtr.Zero; - } - - if (0 == array.Length) - { - return IntPtr.Zero; - } - - var native = Marshal.AllocCoTaskMem(array.Length * Marshal.SizeOf(typeof(PatchRetainRange))); - - for (var i = 0; i < array.Length; ++i) - { - Marshal.StructureToPtr(array[i], (IntPtr)((Int64)native + (i * Marshal.SizeOf(typeof(PatchRetainRange)))), false); - } - - return native; - } - - // CleanUp operations - - private void CleanUpPOD(IntPtr native) - { - Marshal.FreeCoTaskMem(Marshal.ReadIntPtr(native, newFileSymbolPathOffset)); - - if (IntPtr.Zero != Marshal.ReadIntPtr(native, oldFileSymbolPathArrayOffset)) - { - for (var i = 0; i < GetOldFileCount(native); ++i) - { - Marshal.FreeCoTaskMem( - Marshal.ReadIntPtr( - Marshal.ReadIntPtr(native, oldFileSymbolPathArrayOffset), - i * Marshal.SizeOf(typeof(IntPtr)))); - } - - Marshal.FreeCoTaskMem(Marshal.ReadIntPtr(native, oldFileSymbolPathArrayOffset)); - } - - if (IntPtr.Zero != Marshal.ReadIntPtr(native, interleaveMapArrayOffset)) - { - for (var i = 0; i < GetOldFileCount(native); ++i) - { - Marshal.FreeCoTaskMem( - Marshal.ReadIntPtr( - Marshal.ReadIntPtr(native, interleaveMapArrayOffset), - i * Marshal.SizeOf(typeof(IntPtr)))); - } - - Marshal.FreeCoTaskMem(Marshal.ReadIntPtr(native, interleaveMapArrayOffset)); - } - - PatchAPIMarshaler.ReleaseMainStruct(native); - } - - private void CleanUpPOFI_A(IntPtr native) - { - for (var i = 0; i < GetOldFileCount(native); ++i) - { - PatchAPIMarshaler.CleanUpPOFI((IntPtr)((Int64)native + i * patchOldFileInfoSize)); - } - - PatchAPIMarshaler.ReleaseMainStruct(native); - } - - private static void CleanUpPOFI(IntPtr native) - { - if (IntPtr.Zero != Marshal.ReadIntPtr(native, oldFileOffset)) - { - Marshal.FreeCoTaskMem(Marshal.ReadIntPtr(native, oldFileOffset)); - } - - PatchAPIMarshaler.CleanUpPOFIH(native); - } - - private static void CleanUpPOFIH(IntPtr native) - { - if (IntPtr.Zero != Marshal.ReadIntPtr(native, ignoreRangeArrayOffset)) - { - Marshal.FreeCoTaskMem(Marshal.ReadIntPtr(native, ignoreRangeArrayOffset)); - } - - if (IntPtr.Zero != Marshal.ReadIntPtr(native, retainRangeArrayOffset)) - { - Marshal.FreeCoTaskMem(Marshal.ReadIntPtr(native, retainRangeArrayOffset)); - } - } - } - } -} diff --git a/src/WixToolset.Core.Native/ValidationMessage.cs b/src/WixToolset.Core.Native/ValidationMessage.cs deleted file mode 100644 index d7137326..00000000 --- a/src/WixToolset.Core.Native/ValidationMessage.cs +++ /dev/null @@ -1,47 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native -{ - using System.Collections.Generic; - - /// - /// Message from ICE - /// - public class ValidationMessage - { - /// - /// Name of the ICE providing the message. - /// - public string IceName { get; set; } - - /// - /// Validation type. - /// - public ValidationMessageType Type { get; set; } - - /// - /// Message text. - /// - public string Description { get; set; } - - /// - /// Optional help URL for the message. - /// - public string HelpUrl { get; set; } - - /// - /// Optional table causing the message. - /// - public string Table { get; set; } - - /// - /// Optional column causing the message. - /// - public string Column { get; set; } - - /// - /// Optional primary keys causing the message. - /// - public IEnumerable PrimaryKeys { get; set; } - } -} diff --git a/src/WixToolset.Core.Native/ValidationMessageType.cs b/src/WixToolset.Core.Native/ValidationMessageType.cs deleted file mode 100644 index 98635294..00000000 --- a/src/WixToolset.Core.Native/ValidationMessageType.cs +++ /dev/null @@ -1,31 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native -{ - /// - /// Validation message type. - /// - public enum ValidationMessageType - { - /// - /// Failure message reporting the failure of the ICE custom action. - /// - InternalFailure = 0, - - /// - /// Error message reporting database authoring that case incorrect behavior. - /// - Error = 1, - - /// - /// Warning message reporting database authoring that causes incorrect behavior in certain cases. - /// Warnings can also report unexpected side-effects of database authoring. - /// - Warning = 2, - - /// - /// Informational message. - /// - Info = 3, - }; -} diff --git a/src/WixToolset.Core.Native/WindowsInstallerValidator.cs b/src/WixToolset.Core.Native/WindowsInstallerValidator.cs deleted file mode 100644 index 9f4b26a3..00000000 --- a/src/WixToolset.Core.Native/WindowsInstallerValidator.cs +++ /dev/null @@ -1,427 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native -{ - using System; - using System.Collections.Generic; - using System.ComponentModel; - using System.IO; - using System.Linq; - using System.Threading; - using WixToolset.Core.Native.Msi; - using WixToolset.Data; - - /// - /// Windows installer validation implementation. - /// - public class WindowsInstallerValidator - { - private const string CubesFolder = "cubes"; - - /// - /// Creates a new Windows Installer validator. - /// - /// Callback interface to handle messages. - /// Database to validate. - /// Set of CUBe files to merge. - /// ICEs to execute. - /// Suppressed ICEs. - public WindowsInstallerValidator(IWindowsInstallerValidatorCallback callback, string databasePath, IEnumerable cubeFiles, IEnumerable ices, IEnumerable suppressedIces) - { - this.Callback = callback; - this.DatabasePath = databasePath; - this.CubeFiles = cubeFiles; - this.Ices = new SortedSet(ices); - this.SuppressedIces = new SortedSet(suppressedIces); - } - - private IWindowsInstallerValidatorCallback Callback { get; } - - private string DatabasePath { get; } - - private IEnumerable CubeFiles { get; } - - private SortedSet Ices { get; } - - private SortedSet SuppressedIces { get; } - - private bool ValidationSessionInProgress { get; set; } - - private string CurrentIce { get; set; } - - /// - /// Execute the validations. - /// - public void Execute() - { - using (var mutex = new Mutex(false, "WixValidator")) - { - try - { - if (!mutex.WaitOne(0)) - { - this.Callback.ValidationBlocked(); - mutex.WaitOne(); - } - } - catch (AbandonedMutexException) - { - // Another validation process was probably killed, we own the mutex now. - } - - try - { - this.RunValidations(); - } - finally - { - mutex.ReleaseMutex(); - } - } - } - - private void RunValidations() - { - var previousUILevel = (int)InstallUILevels.Basic; - var previousHwnd = IntPtr.Zero; - InstallUIHandler previousUIHandler = null; - - try - { - using (var database = new Database(this.DatabasePath, OpenDatabase.Direct)) - { - var propertyTableExists = database.TableExists("Property"); - string productCode = null; - - // Remove the product code from the database before opening a session to prevent opening an installed product. - if (propertyTableExists) - { - using (var view = database.OpenExecuteView("SELECT `Value` FROM `Property` WHERE Property = 'ProductCode'")) - { - using (var record = view.Fetch()) - { - if (null != record) - { - productCode = record.GetString(1); - - using (var dropProductCodeView = database.OpenExecuteView("DELETE FROM `Property` WHERE `Property` = 'ProductCode'")) - { - } - } - } - } - } - - // Merge in the cube databases. - foreach (var cubeFile in this.CubeFiles) - { - var findCubeFile = typeof(WindowsInstallerValidator).Assembly.FindFileRelativeToAssembly(Path.Combine(CubesFolder, cubeFile), searchNativeDllDirectories: false); - - if (!findCubeFile.Found) - { - throw new WixException(ErrorMessages.CubeFileNotFound(findCubeFile.Path)); - } - - try - { - using (var cubeDatabase = new Database(findCubeFile.Path, OpenDatabase.ReadOnly)) - { - try - { - database.Merge(cubeDatabase, "MergeConflicts"); - } - catch - { - // ignore merge errors since they are expected in the _Validation table - } - } - } - catch (Win32Exception e) - { - if (0x6E == e.NativeErrorCode) // ERROR_OPEN_FAILED - { - throw new WixException(ErrorMessages.CubeFileNotFound(findCubeFile.Path)); - } - - throw; - } - } - - // Commit the database before proceeding to ensure the streams don't get confused. - database.Commit(); - - // The property table may have been added to the database from a cub database without the proper validation rows. - if (!propertyTableExists) - { - using (var view = database.OpenExecuteView("DROP table `Property`")) - { - } - } - - // Get all the action names for ICEs which have not been suppressed. - var actions = new List(); - using (var view = database.OpenExecuteView("SELECT `Action` FROM `_ICESequence` ORDER BY `Sequence`")) - { - foreach (var record in view.Records) - { - var action = record.GetString(1); - - if (!this.SuppressedIces.Contains(action) && this.Ices.Contains(action)) - { - actions.Add(action); - } - } - } - - // Disable the internal UI handler and set an external UI handler. - previousUILevel = Installer.SetInternalUI((int)InstallUILevels.None, ref previousHwnd); - previousUIHandler = Installer.SetExternalUI(this.ValidationUIHandler, (int)InstallLogModes.Error | (int)InstallLogModes.Warning | (int)InstallLogModes.User, IntPtr.Zero); - - // Create a session for running the ICEs. - this.ValidationSessionInProgress = true; - - using (var session = new Session(database)) - { - // Add the product code back into the database. - if (null != productCode) - { - // Some CUBs erroneously have a ProductCode property, so delete it if we just picked one up. - using (var dropProductCodeView = database.OpenExecuteView("DELETE FROM `Property` WHERE `Property` = 'ProductCode'")) - { - } - - using (var view = database.OpenExecuteView($"INSERT INTO `Property` (`Property`, `Value`) VALUES ('ProductCode', '{productCode}')")) - { - } - } - - foreach (var action in actions) - { - this.CurrentIce = action; - - try - { - session.DoAction(action); - } - catch (Win32Exception e) - { - if (!this.Callback.EncounteredError) - { - throw e; - } - } - - this.CurrentIce = null; - } - - // Mark the validation session complete so we ignore any messages that MSI may fire - // during session clean-up. - this.ValidationSessionInProgress = false; - } - } - } - catch (Win32Exception e) - { - // Avoid displaying errors twice since one may have already occurred in the UI handler. - if (!this.Callback.EncounteredError) - { - if (0x6E == e.NativeErrorCode) // ERROR_OPEN_FAILED - { - // The database path is not passed to this exception since inside wix.exe - // this would be the temporary copy and there would be no final output becasue - // this error occured; and during standalone validation they should know the path - // passed in. - throw new WixException(ErrorMessages.ValidationFailedToOpenDatabase()); - } - else if (0x64D == e.NativeErrorCode) - { - throw new WixException(ErrorMessages.ValidationFailedDueToLowMsiEngine()); - } - else if (0x654 == e.NativeErrorCode) - { - throw new WixException(ErrorMessages.ValidationFailedDueToInvalidPackage()); - } - else if (0x658 == e.NativeErrorCode) - { - throw new WixException(ErrorMessages.ValidationFailedDueToMultilanguageMergeModule()); - } - else if (0x659 == e.NativeErrorCode) - { - throw new WixException(WarningMessages.ValidationFailedDueToSystemPolicy()); - } - else - { - var msg = String.IsNullOrEmpty(this.CurrentIce) ? e.Message : $"Action - '{this.CurrentIce}' {e.Message}"; - - throw new WixException(ErrorMessages.Win32Exception(e.NativeErrorCode, msg)); - } - } - } - finally - { - this.ValidationSessionInProgress = false; - - Installer.SetExternalUI(previousUIHandler, 0, IntPtr.Zero); - Installer.SetInternalUI(previousUILevel, ref previousHwnd); - } - } - - /// - /// The validation external UI handler. - /// - /// Pointer to an application context. - /// This parameter can be used for error checking. - /// Specifies a combination of one message box style, - /// one message box icon type, one default button, and one installation message type. - /// Specifies the message text. - /// -1 for an error, 0 if no action was taken, 1 if OK, 3 to abort. - private int ValidationUIHandler(IntPtr context, uint messageType, string message) - { - var continueValidation = true; - - // If we're getting messges during the validation session, log them. - // Otherwise, ignore the messages. - if (!this.ValidationSessionInProgress) - { - var parsedMessage = ParseValidationMessage(message, this.CurrentIce); - - continueValidation = this.Callback.ValidationMessage(parsedMessage); - } - - return continueValidation ? 1 : 3; - } - - /// - /// Parses a message from the Validator. - /// - /// A of tab-delmited tokens - /// in the validation message. - /// The name of the action to which the message - /// belongs. - /// The message cannot be null. - /// - /// The message does not contain four (4) - /// or more tab-delimited tokens. - /// - /// a tab-delimited set of tokens, - /// formatted according to Windows Installer guidelines for ICE - /// message. The following table lists what each token by index - /// should mean. - /// a name that represents the ICE - /// action that was executed (e.g. 'ICE08'). - /// - /// - /// Index - /// Description - /// - /// - /// 0 - /// Name of the ICE. - /// - /// - /// 1 - /// Message type. See the following list. - /// - /// - /// 2 - /// Detailed description. - /// - /// - /// 3 - /// Help URL or location. - /// - /// - /// 4 - /// Table name. - /// - /// - /// 5 - /// Column name. - /// - /// - /// 6 - /// This and remaining fields are primary keys - /// to identify a row. - /// - /// - /// The message types are one of the following value. - /// - /// - /// Value - /// Message Type - /// - /// - /// 0 - /// Failure message reporting the failure of the - /// ICE custom action. - /// - /// - /// 1 - /// Error message reporting database authoring that - /// case incorrect behavior. - /// - /// - /// 2 - /// Warning message reporting database authoring that - /// causes incorrect behavior in certain cases. Warnings can also - /// report unexpected side-effects of database authoring. - /// - /// - /// - /// 3 - /// Informational message. - /// - /// - /// - private static ValidationMessage ParseValidationMessage(string message, string currentIce) - { - if (message == null) - { - throw new ArgumentNullException(nameof(message)); - } - - var messageParts = message.Split('\t'); - if (messageParts.Length < 3) - { - if (null == currentIce) - { - throw new WixException(ErrorMessages.UnexpectedExternalUIMessage(message)); - } - else - { - throw new WixException(ErrorMessages.UnexpectedExternalUIMessage(message, currentIce)); - } - } - - var type = ParseValidationMessageType(messageParts[1]); - - return new ValidationMessage - { - IceName = messageParts[0], - Type = type, - Description = messageParts[2], - HelpUrl = messageParts.Length > 3 ? messageParts[3] : null, - Table = messageParts.Length > 4 ? messageParts[4] : null, - Column = messageParts.Length > 5 ? messageParts[4] : null, - PrimaryKeys = messageParts.Length > 6 ? messageParts.Skip(6).ToArray() : null - }; - } - - private static ValidationMessageType ParseValidationMessageType(string type) - { - switch (type) - { - case "0": - return ValidationMessageType.InternalFailure; - case "1": - return ValidationMessageType.Error; - case "2": - return ValidationMessageType.Warning; - case "3": - return ValidationMessageType.Info; - default: - throw new WixException(ErrorMessages.InvalidValidatorMessageType(type)); - } - } - } -} diff --git a/src/WixToolset.Core.Native/WixNativeExe.cs b/src/WixToolset.Core.Native/WixNativeExe.cs deleted file mode 100644 index fb41b2f2..00000000 --- a/src/WixToolset.Core.Native/WixNativeExe.cs +++ /dev/null @@ -1,125 +0,0 @@ -// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. - -namespace WixToolset.Core.Native -{ - using System; - using System.Collections.Generic; - using System.ComponentModel; - using System.Diagnostics; - using System.IO; - - internal class WixNativeExe - { - private const string WixNativeExeFileName = "wixnative.exe"; - private static string PathToWixNativeExe; - - private readonly string commandLine; - private readonly List stdinLines = new List(); - - public WixNativeExe(params object[] args) - { - this.commandLine = String.Join(" ", QuoteArgumentsAsNecesary(args)); - } - - public void AddStdinLine(string line) - { - this.stdinLines.Add(line); - } - - public void AddStdinLines(IEnumerable lines) - { - this.stdinLines.AddRange(lines); - } - - public IEnumerable Run() - { - EnsurePathToWixNativeExeSet(); - - var wixNativeInfo = new ProcessStartInfo(PathToWixNativeExe, this.commandLine) - { - RedirectStandardInput = true, - RedirectStandardOutput = true, - CreateNoWindow = true, - ErrorDialog = false, - UseShellExecute = false - }; - - var stdoutLines = new List(); - - using (var process = Process.Start(wixNativeInfo)) - { - process.OutputDataReceived += (s, a) => stdoutLines.Add(a.Data); - process.BeginOutputReadLine(); - - if (this.stdinLines.Count > 0) - { - foreach (var line in this.stdinLines) - { - process.StandardInput.WriteLine(line); - } - - // Trailing blank line indicates stdin complete. - process.StandardInput.WriteLine(); - } - - // If the process successfully exits documentation says we need to wait again - // without a timeout to ensure that all of the redirected output is captured. - // - process.WaitForExit(); - - if (process.ExitCode != 0) - { - throw new Win32Exception(process.ExitCode); - } - } - - return stdoutLines; - } - - private static void EnsurePathToWixNativeExeSet() - { - if (String.IsNullOrEmpty(PathToWixNativeExe)) - { - var result = typeof(WixNativeExe).Assembly.FindFileRelativeToAssembly(WixNativeExeFileName, searchNativeDllDirectories: true); - - if (!result.Found) - { - throw new PlatformNotSupportedException( - $"Could not find platform specific '{WixNativeExeFileName}'", - new FileNotFoundException($"Could not find internal piece of WiX Toolset from: {result.PossiblePaths}", WixNativeExeFileName)); - } - - PathToWixNativeExe = result.Path; - } - } - - private static IEnumerable QuoteArgumentsAsNecesary(object[] args) - { - foreach (var arg in args) - { - if (arg is string str) - { - if (String.IsNullOrEmpty(str)) - { - } - else if (str.Contains(" ") && !str.StartsWith("\"")) - { - yield return $"\"{str}\""; - } - else - { - yield return str; - } - } - else if (arg is int i) - { - yield return i.ToString(); - } - else - { - throw new ArgumentException(nameof(args)); - } - } - } - } -} diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj deleted file mode 100644 index fea15922..00000000 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ /dev/null @@ -1,49 +0,0 @@ - - - - - - - netstandard2.0 - embedded - WiX Toolset Native Processing - true - - NU5128 - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec deleted file mode 100644 index 3091ccd5..00000000 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ /dev/null @@ -1,41 +0,0 @@ - - - - $id$ - $version$ - $title$ - $description$ - $authors$ - MS-RL - false - $copyright$ - $projectUrl$ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/src/WixToolset.Core.Native/cubes/darice.cub b/src/WixToolset.Core.Native/cubes/darice.cub deleted file mode 100644 index 4292fede..00000000 Binary files a/src/WixToolset.Core.Native/cubes/darice.cub and /dev/null differ diff --git a/src/WixToolset.Core.Native/cubes/mergemod.cub b/src/WixToolset.Core.Native/cubes/mergemod.cub deleted file mode 100644 index def6dd1a..00000000 Binary files a/src/WixToolset.Core.Native/cubes/mergemod.cub and /dev/null differ diff --git a/src/WixToolset.Core.Native/targets/WixToolset.Core.Native.targets b/src/WixToolset.Core.Native/targets/WixToolset.Core.Native.targets deleted file mode 100644 index aafbd405..00000000 --- a/src/WixToolset.Core.Native/targets/WixToolset.Core.Native.targets +++ /dev/null @@ -1,9 +0,0 @@ - - - - - PreserveNewest - cubes\%(RecursiveDir)%(Filename)%(Extension) - - - diff --git a/src/signing.json b/src/signing.json new file mode 100644 index 00000000..fe1c8c9b --- /dev/null +++ b/src/signing.json @@ -0,0 +1,13 @@ +{ + "SignClient": { + "AzureAd": { + "AADInstance": "https://login.microsoftonline.com/", + "ClientId": "c248d68a-ba6f-4aa9-8a68-71fe872063f8", + "TenantId": "16076fdc-fcc1-4a15-b1ca-32c9a255900e" + }, + "Service": { + "Url": "https://codesign.dotnetfoundation.org/", + "ResourceId": "https://SignService/3c30251f-36f3-490b-a955-520addb85001" + } + } +} diff --git a/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs b/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs deleted file mode 100644 index 2e43dce4..00000000 --- a/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs +++ /dev/null @@ -1,96 +0,0 @@ -// 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 WixToolsetTest.CoreNative -{ - using System.IO; - using System.Linq; - using WixToolset.Core.Native; - using WixToolsetTest.CoreNative.Utility; - using WixToolset.Data; - using Xunit; - - public class CabinetFixture - { - [Fact] - public void CanCreateSingleFileCabinet() - { - using (var fs = new DisposableFileSystem()) - { - var intermediateFolder = fs.GetFolder(true); - var cabPath = Path.Combine(intermediateFolder, "testout.cab"); - - var files = new[] { new CabinetCompressFile(TestData.Get(@"TestData\test.txt"), "test.txt") }; - - var cabinet = new Cabinet(cabPath); - cabinet.Compress(files, CompressionLevel.Low); - - Assert.True(File.Exists(cabPath)); - } - } - - [Fact] - public void CanEnumerateSingleFileCabinet() - { - var cabinetPath = TestData.Get(@"TestData\test.cab"); - - var cabinet = new Cabinet(cabinetPath); - var files = cabinet.Enumerate(); - - var file = files.Single(); - Assert.Equal("test.txt", file.FileId); - Assert.Equal(17, file.Size); - - Assert.Equal(19259, file.Date); - Assert.Equal(47731, file.Time); - // TODO: This doesn't seem to always pass, not clear why but it'd be good to understand one day. - // Assert.True(file.SameAsDateTime(new DateTime(2017, 9, 28, 0, 19, 38))); - } - - [Fact] - public void IntegrationTest() - { - using (var fs = new DisposableFileSystem()) - { - var intermediateFolder = fs.GetFolder(true); - var cabinetPath = Path.Combine(intermediateFolder, "testout.cab"); - var extractFolder = fs.GetFolder(true); - - // Compress. - { - var files = new[] { - new CabinetCompressFile(TestData.Get(@"TestData\test.txt"), "test1.txt"), - new CabinetCompressFile(TestData.Get(@"TestData\test.txt"), "test2.txt"), - }; - - var cabinet = new Cabinet(cabinetPath); - cabinet.Compress(files, CompressionLevel.Low); - } - - // Extract. - { - var cabinet = new Cabinet(cabinetPath); - var reportedFiles = cabinet.Extract(extractFolder); - Assert.Equal(2, reportedFiles.Count()); - } - - // Enumerate to compare cabinet to extracted files. - { - var cabinet = new Cabinet(cabinetPath); - var enumerated = cabinet.Enumerate().OrderBy(f => f.FileId).ToArray(); - - var files = Directory.EnumerateFiles(extractFolder).OrderBy(f => f).ToArray(); - - for (var i = 0; i < enumerated.Length; ++i) - { - var cabFileInfo = enumerated[i]; - var fileInfo = new FileInfo(files[i]); - - Assert.Equal(cabFileInfo.FileId, fileInfo.Name); - Assert.Equal(cabFileInfo.Size, fileInfo.Length); - Assert.True(cabFileInfo.SameAsDateTime(fileInfo.CreationTime)); - } - } - } - } - } -} diff --git a/src/test/WixToolsetTest.Core.Native/MsmFixture.cs b/src/test/WixToolsetTest.Core.Native/MsmFixture.cs deleted file mode 100644 index 709d4b93..00000000 --- a/src/test/WixToolsetTest.Core.Native/MsmFixture.cs +++ /dev/null @@ -1,17 +0,0 @@ -// 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 WixToolsetTest.CoreNative -{ - using WixToolset.Core.Native.Msm; - using Xunit; - - public class MsmFixture - { - [Fact] - public void CanCreateMsmInterface() - { - var merge = MsmInterop.GetMsmMerge(); - Assert.NotNull(merge); - } - } -} diff --git a/src/test/WixToolsetTest.Core.Native/TestData/test.cab b/src/test/WixToolsetTest.Core.Native/TestData/test.cab deleted file mode 100644 index ca78f632..00000000 Binary files a/src/test/WixToolsetTest.Core.Native/TestData/test.cab and /dev/null differ diff --git a/src/test/WixToolsetTest.Core.Native/TestData/test.txt b/src/test/WixToolsetTest.Core.Native/TestData/test.txt deleted file mode 100644 index cd0db0e1..00000000 --- a/src/test/WixToolsetTest.Core.Native/TestData/test.txt +++ /dev/null @@ -1 +0,0 @@ -This is test.txt. \ No newline at end of file diff --git a/src/test/WixToolsetTest.Core.Native/Utility/DisposableFileSystem.cs b/src/test/WixToolsetTest.Core.Native/Utility/DisposableFileSystem.cs deleted file mode 100644 index c9957247..00000000 --- a/src/test/WixToolsetTest.Core.Native/Utility/DisposableFileSystem.cs +++ /dev/null @@ -1,86 +0,0 @@ -// 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 WixToolsetTest.CoreNative.Utility -{ - using System; - using System.Collections.Generic; - using System.IO; - - public class DisposableFileSystem : IDisposable - { - protected bool Disposed { get; private set; } - - private List CleanupPaths { get; } = new List(); - - public string GetFile(bool create = false) - { - var path = Path.GetTempFileName(); - - if (!create) - { - File.Delete(path); - } - - this.CleanupPaths.Add(path); - - return path; - } - - public string GetFolder(bool create = false) - { - var path = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName()); - - if (create) - { - Directory.CreateDirectory(path); - } - - this.CleanupPaths.Add(path); - - return path; - } - - - #region // IDisposable - - public void Dispose() - { - this.Dispose(true); - GC.SuppressFinalize(this); - } - - protected virtual void Dispose(bool disposing) - { - if (this.Disposed) - { - return; - } - - if (disposing) - { - foreach (var path in this.CleanupPaths) - { - try - { - if (File.Exists(path)) - { - File.Delete(path); - } - else if (Directory.Exists(path)) - { - Directory.Delete(path, true); - } - } - catch - { - // Best effort delete, so ignore any failures. - } - } - } - - this.Disposed = true; - } - - #endregion - } -} diff --git a/src/test/WixToolsetTest.Core.Native/Utility/Pushd.cs b/src/test/WixToolsetTest.Core.Native/Utility/Pushd.cs deleted file mode 100644 index 91700c2f..00000000 --- a/src/test/WixToolsetTest.Core.Native/Utility/Pushd.cs +++ /dev/null @@ -1,46 +0,0 @@ -// 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 WixToolsetTest.CoreNative.Utility -{ - using System; - using System.IO; - - public class Pushd : IDisposable - { - protected bool Disposed { get; private set; } - - public Pushd(string path) - { - this.PreviousDirectory = Directory.GetCurrentDirectory(); - - Directory.SetCurrentDirectory(path); - } - - public string PreviousDirectory { get; } - - #region // IDisposable - - public void Dispose() - { - this.Dispose(true); - GC.SuppressFinalize(this); - } - - protected virtual void Dispose(bool disposing) - { - if (this.Disposed) - { - return; - } - - if (disposing) - { - Directory.SetCurrentDirectory(this.PreviousDirectory); - } - - this.Disposed = true; - } - - #endregion - } -} diff --git a/src/test/WixToolsetTest.Core.Native/Utility/TestData.cs b/src/test/WixToolsetTest.Core.Native/Utility/TestData.cs deleted file mode 100644 index cd9c6318..00000000 --- a/src/test/WixToolsetTest.Core.Native/Utility/TestData.cs +++ /dev/null @@ -1,17 +0,0 @@ -// 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 WixToolsetTest.CoreNative.Utility -{ - using System; - using System.IO; - - public class TestData - { - public static string LocalPath => Path.GetDirectoryName(new Uri(System.Reflection.Assembly.GetExecutingAssembly().CodeBase).LocalPath); - - public static string Get(params string[] paths) - { - return Path.Combine(LocalPath, Path.Combine(paths)); - } - } -} diff --git a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj deleted file mode 100644 index 6068dbea..00000000 --- a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj +++ /dev/null @@ -1,26 +0,0 @@ - - - - - - netcoreapp3.1 - false - win-x64 - - - - - - - - - - - - - - - - - - diff --git a/src/test/version.txt b/src/test/version.txt deleted file mode 100644 index cf138743..00000000 --- a/src/test/version.txt +++ /dev/null @@ -1 +0,0 @@ -v42.42.{height}-preview.0 \ No newline at end of file diff --git a/src/version.txt b/src/version.txt new file mode 100644 index 00000000..dc60d914 --- /dev/null +++ b/src/version.txt @@ -0,0 +1 @@ +v4.{apiversion}-preview.0-build.{height} diff --git a/src/wix/README-CoreNative.md b/src/wix/README-CoreNative.md new file mode 100644 index 00000000..787123cc --- /dev/null +++ b/src/wix/README-CoreNative.md @@ -0,0 +1,2 @@ +# Core.Native +Core.Native - native component of WixToolset.Core diff --git a/src/wix/WixToolset.Core.Native.sln b/src/wix/WixToolset.Core.Native.sln new file mode 100644 index 00000000..0d7a5921 --- /dev/null +++ b/src/wix/WixToolset.Core.Native.sln @@ -0,0 +1,63 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio 15 +VisualStudioVersion = 15.0.27004.2009 +MinimumVisualStudioVersion = 15.0.26124.0 +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "WixToolset.Core.Native", "src\WixToolset.Core.Native\WixToolset.Core.Native.csproj", "{C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "wixnative", "src\wixnative\wixnative.vcxproj", "{8497EC72-B8D0-4272-A9D0-7E9D871CEFBF}" +EndProject +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "WixToolsetTest.Core.Native", "src\test\WixToolsetTest.Core.Native\WixToolsetTest.Core.Native.csproj", "{54F5329A-C113-471A-8EE1-83021E8A4853}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Any CPU = Debug|Any CPU + Debug|x64 = Debug|x64 + Debug|x86 = Debug|x86 + Release|Any CPU = Release|Any CPU + Release|x64 = Release|x64 + Release|x86 = Release|x86 + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}.Debug|Any CPU.Build.0 = Debug|Any CPU + {C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}.Debug|x64.ActiveCfg = Debug|Any CPU + {C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}.Debug|x64.Build.0 = Debug|Any CPU + {C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}.Debug|x86.ActiveCfg = Debug|Any CPU + {C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}.Debug|x86.Build.0 = Debug|Any CPU + {C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}.Release|Any CPU.ActiveCfg = Release|Any CPU + {C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}.Release|Any CPU.Build.0 = Release|Any CPU + {C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}.Release|x64.ActiveCfg = Release|Any CPU + {C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}.Release|x64.Build.0 = Release|Any CPU + {C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}.Release|x86.ActiveCfg = Release|Any CPU + {C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}.Release|x86.Build.0 = Release|Any CPU + {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF}.Debug|Any CPU.ActiveCfg = Debug|Win32 + {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF}.Debug|x64.ActiveCfg = Debug|x64 + {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF}.Debug|x64.Build.0 = Debug|x64 + {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF}.Debug|x86.ActiveCfg = Debug|Win32 + {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF}.Debug|x86.Build.0 = Debug|Win32 + {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF}.Release|Any CPU.ActiveCfg = Release|Win32 + {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF}.Release|x64.ActiveCfg = Release|x64 + {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF}.Release|x64.Build.0 = Release|x64 + {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF}.Release|x86.ActiveCfg = Release|Win32 + {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF}.Release|x86.Build.0 = Release|Win32 + {54F5329A-C113-471A-8EE1-83021E8A4853}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {54F5329A-C113-471A-8EE1-83021E8A4853}.Debug|Any CPU.Build.0 = Debug|Any CPU + {54F5329A-C113-471A-8EE1-83021E8A4853}.Debug|x64.ActiveCfg = Debug|Any CPU + {54F5329A-C113-471A-8EE1-83021E8A4853}.Debug|x64.Build.0 = Debug|Any CPU + {54F5329A-C113-471A-8EE1-83021E8A4853}.Debug|x86.ActiveCfg = Debug|Any CPU + {54F5329A-C113-471A-8EE1-83021E8A4853}.Debug|x86.Build.0 = Debug|Any CPU + {54F5329A-C113-471A-8EE1-83021E8A4853}.Release|Any CPU.ActiveCfg = Release|Any CPU + {54F5329A-C113-471A-8EE1-83021E8A4853}.Release|Any CPU.Build.0 = Release|Any CPU + {54F5329A-C113-471A-8EE1-83021E8A4853}.Release|x64.ActiveCfg = Release|Any CPU + {54F5329A-C113-471A-8EE1-83021E8A4853}.Release|x64.Build.0 = Release|Any CPU + {54F5329A-C113-471A-8EE1-83021E8A4853}.Release|x86.ActiveCfg = Release|Any CPU + {54F5329A-C113-471A-8EE1-83021E8A4853}.Release|x86.Build.0 = Release|Any CPU + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + SolutionGuid = {1E952530-A3ED-4E65-AF39-9025EFB85322} + EndGlobalSection +EndGlobal diff --git a/src/wix/WixToolset.Core.Native.v3.ncrunchsolution b/src/wix/WixToolset.Core.Native.v3.ncrunchsolution new file mode 100644 index 00000000..10420ac9 --- /dev/null +++ b/src/wix/WixToolset.Core.Native.v3.ncrunchsolution @@ -0,0 +1,6 @@ + + + True + True + + \ No newline at end of file diff --git a/src/wix/WixToolset.Core.Native/AssemblyExtensions.cs b/src/wix/WixToolset.Core.Native/AssemblyExtensions.cs new file mode 100644 index 00000000..590a6887 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/AssemblyExtensions.cs @@ -0,0 +1,70 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native +{ + using System; + using System.IO; + using System.Reflection; + using System.Text; + + internal static class AssemblyExtensions + { + internal static FindAssemblyRelativeFileResult FindFileRelativeToAssembly(this Assembly assembly, string relativePath, bool searchNativeDllDirectories) + { + // First try using the Assembly.Location. This works in almost all cases with + // no side-effects. + var path = Path.Combine(Path.GetDirectoryName(assembly.Location), relativePath); + var possiblePaths = new StringBuilder(path); + + var found = File.Exists(path); + if (!found) + { + // Fallback to the Assembly.CodeBase to handle "shadow copy" scenarios (like unit tests) but + // only check codebase if it is different from the Assembly.Location path. + var codebase = Path.Combine(Path.GetDirectoryName(new Uri(assembly.CodeBase).LocalPath), relativePath); + + if (!codebase.Equals(path, StringComparison.OrdinalIgnoreCase)) + { + path = codebase; + possiblePaths.Append(Path.PathSeparator + path); + + found = File.Exists(path); + } + + if (!found && searchNativeDllDirectories && AppContext.GetData("NATIVE_DLL_SEARCH_DIRECTORIES") is string searchDirectoriesString) + { + // If instructed to search native DLL search directories, try to find our file there. + possiblePaths.Append(Path.PathSeparator + searchDirectoriesString); + + var searchDirectories = searchDirectoriesString?.Split(Path.PathSeparator); + foreach (var directoryPath in searchDirectories) + { + var possiblePath = Path.Combine(directoryPath, relativePath); + if (File.Exists(possiblePath)) + { + path = possiblePath; + found = true; + break; + } + } + } + } + + return new FindAssemblyRelativeFileResult + { + Found = found, + Path = found ? path : null, + PossiblePaths = possiblePaths.ToString() + }; + } + + internal class FindAssemblyRelativeFileResult + { + public bool Found { get; set; } + + public string Path { get; set; } + + public string PossiblePaths { get; set; } + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Cabinet.cs b/src/wix/WixToolset.Core.Native/Cabinet.cs new file mode 100644 index 00000000..9b77bd37 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Cabinet.cs @@ -0,0 +1,120 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native +{ + using System; + using System.Collections.Generic; + using System.Linq; + using WixToolset.Data; + + /// + /// Cabinet create, enumerate and extract mechanism. + /// + public sealed class Cabinet + { + private const string CompressionLevelVariable = "WIX_COMPRESSION_LEVEL"; + private static readonly char[] TextLineSplitter = new[] { '\t' }; + + /// + /// Creates a cabinet creation, enumeration, extraction mechanism. + /// + /// Path of cabinet + public Cabinet(string path) + { + this.Path = path; + } + + /// + /// Cabinet path. + /// + public string Path { get; } + + /// + /// Creates a cabinet. + /// + /// Files to compress. + /// Level of compression to apply. + /// Maximum size of cabinet. + /// Maximum threshold for each cabinet. + public void Compress(IEnumerable files, CompressionLevel compressionLevel, int maxSize = 0, int maxThresh = 0) + { + var compressionLevelVariable = Environment.GetEnvironmentVariable(CompressionLevelVariable); + + // Override authored compression level if environment variable is present. + if (!String.IsNullOrEmpty(compressionLevelVariable)) + { + if (!Enum.TryParse(compressionLevelVariable, true, out compressionLevel)) + { + throw new WixException(ErrorMessages.IllegalEnvironmentVariable(CompressionLevelVariable, compressionLevelVariable)); + } + } + + var wixnative = new WixNativeExe("smartcab", this.Path, Convert.ToInt32(compressionLevel), files.Count(), maxSize, maxThresh); + + foreach (var file in files) + { + wixnative.AddStdinLine(file.ToWixNativeStdinLine()); + } + + wixnative.Run(); + +#if TOOD_ERROR_HANDLING + catch (COMException ce) + { + // If we get a "the file exists" error, we must have a full temp directory - so report the issue + if (0x80070050 == unchecked((uint)ce.ErrorCode)) + { + throw new WixException(WixErrors.FullTempDirectory("WSC", Path.GetTempPath())); + } + + throw; + } +#endif + } + + /// + /// Enumerates all files in a cabinet. + /// + /// >List of CabinetFileInfo + public List Enumerate() + { + var wixnative = new WixNativeExe("enumcab", this.Path); + var lines = wixnative.Run(); + + var fileInfoList = new List(); + + foreach (var line in lines) + { + if (String.IsNullOrEmpty(line)) + { + continue; + } + + var data = line.Split(TextLineSplitter, StringSplitOptions.None); + + var size = Convert.ToInt32(data[1]); + var date = Convert.ToInt32(data[2]); + var time = Convert.ToInt32(data[3]); + + fileInfoList.Add(new CabinetFileInfo(data[0], size, date, time)); + } + + return fileInfoList; + } + + /// + /// Extracts all the files from a cabinet to a directory. + /// + /// Directory to extract files to. + public IEnumerable Extract(string outputFolder) + { + if (!outputFolder.EndsWith("\\", StringComparison.Ordinal)) + { + outputFolder += "\\"; + } + + var wixnative = new WixNativeExe("extractcab", this.Path, outputFolder); + return wixnative.Run().Where(output => !String.IsNullOrWhiteSpace(output)); + } + } +} diff --git a/src/wix/WixToolset.Core.Native/CabinetCompressFile.cs b/src/wix/WixToolset.Core.Native/CabinetCompressFile.cs new file mode 100644 index 00000000..6778f4a1 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/CabinetCompressFile.cs @@ -0,0 +1,65 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native +{ + /// + /// Information to compress file into a cabinet. + /// + public sealed class CabinetCompressFile + { + /// + /// Cabinet compress file. + /// + /// Path to file to add. + /// The token for the file. + public CabinetCompressFile(string path, string token) + { + this.Path = path; + this.Token = token; + this.Hash = null; + } + + /// + /// Cabinet compress file. + /// + /// Path to file to add. + /// The token for the file. + /// Hash 1 + /// Hash 2 + /// Hash 3 + /// Hash 4 + public CabinetCompressFile(string path, string token, int hash1, int hash2, int hash3, int hash4) + { + this.Path = path; + this.Token = token; + this.Hash = new[] { hash1, hash2, hash3, hash4 }; + } + + /// + /// Gets the path to the file to compress. + /// + public string Path { get; } + + /// + /// Gets the token for the file to compress. + /// + public string Token { get; } + + /// + /// Gets the hash of the file to compress. + /// + public int[] Hash { get; } + + internal string ToWixNativeStdinLine() + { + if (this.Hash == null) + { + return $"{this.Path}\t{this.Token}"; + } + else + { + return $"{this.Path}\t{this.Token}\t{this.Hash[0]}\t{this.Hash[1]}\t{this.Hash[2]}\t{this.Hash[3]}"; + } + } + } +} diff --git a/src/wix/WixToolset.Core.Native/CabinetFileInfo.cs b/src/wix/WixToolset.Core.Native/CabinetFileInfo.cs new file mode 100644 index 00000000..07387191 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/CabinetFileInfo.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.Core.Native +{ + using System; + + /// + /// Properties of a file in a cabinet. + /// + public sealed class CabinetFileInfo + { + /// + /// Constructs CabinetFileInfo + /// + /// File Id + /// Size of file + /// Last modified date + /// Last modified time + public CabinetFileInfo(string fileId, int size, int date, int time) + { + this.FileId = fileId; + this.Size = size; + this.Date = date; + this.Time = time; + } + + /// + /// Gets the file Id of the file. + /// + /// file Id + public string FileId { get; } + + /// + /// Gets modified date (DOS format). + /// + public int Date { get; } + + /// + /// Gets modified time (DOS format). + /// + public int Time { get; } + + /// + /// Gets the size of the file in bytes. + /// + public int Size { get; } + + /// + /// Compares this file info's date and time with another datetime. + /// + /// Date and time to compare with/ + /// + /// For some reason DateTime.ToLocalTime() does not match kernel32.dll FileTimeToLocalFileTime(). + /// Since cabinets store date and time with the kernel32.dll functions, we need to convert DateTime + /// to local file time using the kernel32.dll functions. + /// + public bool SameAsDateTime(DateTime dateTime) + { + DateTimeInterop.DateTimeToCabDateAndTime(dateTime, out var cabDate, out var cabTime); + return this.Date == cabDate && this.Time == cabTime; + } + } +} diff --git a/src/wix/WixToolset.Core.Native/DateTimeInterop.cs b/src/wix/WixToolset.Core.Native/DateTimeInterop.cs new file mode 100644 index 00000000..d2a0ba2b --- /dev/null +++ b/src/wix/WixToolset.Core.Native/DateTimeInterop.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.Core.Native +{ + using System; + using System.Runtime.InteropServices; + + /// + /// Interop class for the date/time handling. + /// + internal static class DateTimeInterop + { + /// + /// Converts DateTime to MS-DOS date and time which cabinet uses. + /// + /// DateTime + /// MS-DOS date + /// MS-DOS time + public static void DateTimeToCabDateAndTime(DateTime dateTime, out ushort cabDate, out ushort cabTime) + { + // dateTime.ToLocalTime() does not match FileTimeToLocalFileTime() for some reason. + // so we need to call FileTimeToLocalFileTime() from kernel32.dll. + long filetime = dateTime.ToFileTime(); + long localTime = 0; + FileTimeToLocalFileTime(ref filetime, ref localTime); + FileTimeToDosDateTime(ref localTime, out cabDate, out cabTime); + } + + /// + /// Converts file time to a local file time. + /// + /// file time + /// local file time + /// true if successful, false otherwise + [DllImport("kernel32.dll", SetLastError = true)] + private static extern bool FileTimeToLocalFileTime(ref long fileTime, ref long localTime); + + /// + /// Converts file time to a MS-DOS time. + /// + /// file time + /// MS-DOS date + /// MS-DOS time + /// true if successful, false otherwise + [DllImport("kernel32.dll", SetLastError = true)] + private static extern bool FileTimeToDosDateTime(ref long fileTime, out ushort wFatDate, out ushort wFatTime); + } +} diff --git a/src/wix/WixToolset.Core.Native/FileSystem.cs b/src/wix/WixToolset.Core.Native/FileSystem.cs new file mode 100644 index 00000000..d843a9e8 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/FileSystem.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.Core.Native +{ + using System; + using System.Collections.Generic; + using System.IO; + using System.Runtime.InteropServices; + using System.Security.AccessControl; + + /// + /// File system helpers. + /// + public static class FileSystem + { + /// + /// Copies a file. + /// + /// The file to copy. + /// The destination file. + /// Allow hardlinks. + public static void CopyFile(string source, string destination, bool allowHardlink) + { + EnsureDirectoryWithoutFile(destination); + + if (!allowHardlink || !CreateHardLink(destination, source, IntPtr.Zero)) + { +#if DEBUG + var er = Marshal.GetLastWin32Error(); +#endif + + File.Copy(source, destination, overwrite: true); + } + } + + /// + /// Moves a file. + /// + /// The file to move. + /// The destination file. + public static void MoveFile(string source, string destination) + { + EnsureDirectoryWithoutFile(destination); + + File.Move(source, destination); + } + + /// + /// Reset the ACLs on a set of files. + /// + /// The list of file paths to set ACLs. + public static void ResetAcls(IEnumerable files) + { + var aclReset = new FileSecurity(); + aclReset.SetAccessRuleProtection(false, false); + + foreach (var file in files) + { + new FileInfo(file).SetAccessControl(aclReset); + } + } + + private static void EnsureDirectoryWithoutFile(string path) + { + File.Delete(path); + + var directory = Path.GetDirectoryName(path); + + if (!String.IsNullOrEmpty(directory)) + { + Directory.CreateDirectory(directory); + } + } + + [DllImport("Kernel32.dll", CharSet = CharSet.Unicode, SetLastError = true)] + private static extern bool CreateHardLink(string lpFileName, string lpExistingFileName, IntPtr lpSecurityAttributes); + } +} diff --git a/src/wix/WixToolset.Core.Native/IWindowsInstallerValidatorCallback.cs b/src/wix/WixToolset.Core.Native/IWindowsInstallerValidatorCallback.cs new file mode 100644 index 00000000..f4aff134 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/IWindowsInstallerValidatorCallback.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.Core.Native +{ + /// + /// Callbacks during validation. + /// + public interface IWindowsInstallerValidatorCallback + { + /// + /// Indicates if the validator callback encountered an error. + /// + bool EncounteredError { get; } + + /// + /// Validation blocked by another Windows Installer operation. + /// + void ValidationBlocked(); + + /// + /// Validation message from an ICE. + /// + /// The validation message. + /// True if validation should continue; otherwise cancel the validation. + bool ValidationMessage(ValidationMessage message); + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/Database.cs b/src/wix/WixToolset.Core.Native/Msi/Database.cs new file mode 100644 index 00000000..b9c5c35b --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/Database.cs @@ -0,0 +1,262 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + using System; + using System.IO; + using System.Threading; + + /// + /// Wrapper class for managing MSI API database handles. + /// + public sealed class Database : MsiHandle + { + private const int STG_E_LOCKVIOLATION = unchecked((int)0x80030021); + + /// + /// Constructor that opens an MSI database. + /// + /// Path to the database to be opened. + /// Persist mode to use when opening the database. + public Database(string path, OpenDatabase type) + { + var error = MsiInterop.MsiOpenDatabase(path, new IntPtr((int)type), out var handle); + if (0 != error) + { + throw new MsiException(error); + } + this.Handle = handle; + } + + /// + /// Maximum length of stream in an MSI database. + /// + public static int MsiMaxStreamNameLength => MsiInterop.MsiMaxStreamNameLength; + + /// + /// Apply a transform to the MSI. + /// + /// Path to transform to apply. + public void ApplyTransform(string transformFile) + { + // get the curret validation bits + var conditions = TransformErrorConditions.None; + using (var summaryInfo = new SummaryInformation(transformFile)) + { + try + { + var validationFlags = summaryInfo.GetNumericProperty(SummaryInformation.Transform.ValidationFlags); + conditions = (TransformErrorConditions)(validationFlags & 0xffff); + } + catch (FormatException) + { + // fallback to default of None + } + } + + this.ApplyTransform(transformFile, conditions); + } + + /// + /// Applies a transform to this database. + /// + /// Path to the transform file being applied. + /// Specifies the error conditions that are to be suppressed. + public void ApplyTransform(string transformFile, TransformErrorConditions errorConditions) + { + var error = MsiInterop.MsiDatabaseApplyTransform(this.Handle, transformFile, errorConditions); + if (0 != error) + { + throw new MsiException(error); + } + } + + /// + /// Commits changes made to the database. + /// + public void Commit() + { + // Retry this call 3 times to deal with an MSI internal locking problem. + const int retryWait = 300; + const int retryLimit = 3; + var error = 0; + + for (var i = 1; i <= retryLimit; ++i) + { + error = MsiInterop.MsiDatabaseCommit(this.Handle); + + if (0 == error) + { + return; + } + else + { + var exception = new MsiException(error); + + // We need to see if the error code is contained in any of the strings in ErrorInfo. + // Join the array together and search for the error code to cover the string array. + if (!String.Join(", ", exception.ErrorInfo).Contains(STG_E_LOCKVIOLATION.ToString())) + { + break; + } + + Console.Error.WriteLine(String.Format("Failed to create the database. Info: {0}. Retrying ({1} of {2})", String.Join(", ", exception.ErrorInfo), i, retryLimit)); + Thread.Sleep(retryWait); + } + } + + throw new MsiException(error); + } + + /// + /// Creates and populates the summary information stream of an existing transform file. + /// + /// Required database that does not include the changes. + /// The name of the generated transform file. + /// Required error conditions that should be suppressed when the transform is applied. + /// Required when the transform is applied to a database; + /// shows which properties should be validated to verify that this transform can be applied to the database. + public void CreateTransformSummaryInfo(Database referenceDatabase, string transformFile, TransformErrorConditions errorConditions, TransformValidations validations) + { + var error = MsiInterop.MsiCreateTransformSummaryInfo(this.Handle, referenceDatabase.Handle, transformFile, errorConditions, validations); + if (0 != error) + { + throw new MsiException(error); + } + } + + /// + /// Imports an installer text archive table (idt file) into an open database. + /// + /// Specifies the path to the file to import. + /// Attempted to import an IDT file with an invalid format or unsupported data. + /// Another error occured while importing the IDT file. + public void Import(string idtPath) + { + var folderPath = Path.GetFullPath(Path.GetDirectoryName(idtPath)); + var fileName = Path.GetFileName(idtPath); + + var error = MsiInterop.MsiDatabaseImport(this.Handle, folderPath, fileName); + if (1627 == error) // ERROR_FUNCTION_FAILED + { + throw new WixInvalidIdtException(idtPath); + } + else if (0 != error) + { + throw new MsiException(error); + } + } + + /// + /// Exports an installer table from an open database to a text archive file (idt file). + /// + /// Specifies the name of the table to export. + /// Specifies the name of the folder that contains archive files. If null or empty string, uses current directory. + /// Specifies the name of the exported table archive file. + public void Export(string tableName, string folderPath, string fileName) + { + if (String.IsNullOrEmpty(folderPath)) + { + folderPath = Environment.CurrentDirectory; + } + + var error = MsiInterop.MsiDatabaseExport(this.Handle, tableName, folderPath, fileName); + if (0 != error) + { + throw new MsiException(error); + } + } + + /// + /// Creates a transform that, when applied to the reference database, results in this database. + /// + /// Required database that does not include the changes. + /// The name of the generated transform file. This is optional. + /// true if a transform is generated; false if a transform is not generated because + /// there are no differences between the two databases. + public bool GenerateTransform(Database referenceDatabase, string transformFile) + { + var error = MsiInterop.MsiDatabaseGenerateTransform(this.Handle, referenceDatabase.Handle, transformFile, 0, 0); + if (0 != error && 0xE8 != error) // ERROR_NO_DATA(0xE8) means no differences were found + { + throw new MsiException(error); + } + + return (0xE8 != error); + } + + /// + /// Merges two databases together. + /// + /// The database to merge into the base database. + /// The name of the table to receive merge conflict information. + /// True if there were merge conflicts, otherwise false. + public bool Merge(Database mergeDatabase, string tableName) + { + var error = MsiInterop.MsiDatabaseMerge(this.Handle, mergeDatabase.Handle, tableName); + if (error == 1627) + { + return true; + } + else if (error != 0) + { + throw new MsiException(error); + } + + return false; + } + + /// + /// Prepares a database query and creates a View object. + /// + /// Specifies a SQL query string for querying the database. + /// A view object is returned if the query was successful. + public View OpenView(string query) + { + return new View(this, query); + } + + /// + /// Prepares and executes a database query and creates a View object. + /// + /// Specifies a SQL query string for querying the database. + /// A view object is returned if the query was successful. + public View OpenExecuteView(string query) + { + var view = new View(this, query); + + view.Execute(); + return view; + } + + /// + /// Verifies the existence or absence of a table. + /// + /// Table name to to verify the existence of. + /// Returns true if the table exists, false if it does not. + public bool TableExists(string tableName) + { + var result = MsiInterop.MsiDatabaseIsTablePersistent(this.Handle, tableName); + return MsiInterop.MSICONDITIONTRUE == result; + } + + /// + /// Returns a Record containing the names of all the primary + /// key columns for a specified table. + /// + /// Specifies the name of the table from which to obtain + /// primary key names. + /// Returns a Record containing the names of all the + /// primary key columns for a specified table. + public Record PrimaryKeys(string tableName) + { + var error = MsiInterop.MsiDatabaseGetPrimaryKeys(this.Handle, tableName, out var recordHandle); + if (error != 0) + { + throw new MsiException(error); + } + + return new Record(recordHandle); + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/InstallLogModes.cs b/src/wix/WixToolset.Core.Native/Msi/InstallLogModes.cs new file mode 100644 index 00000000..f7012b35 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/InstallLogModes.cs @@ -0,0 +1,111 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + using System; + + /// + /// Windows Installer log modes. + /// + [Flags] + public enum InstallLogModes + { + /// + /// Premature termination of installation. + /// + FatalExit = (1 << ((int)InstallMessage.FatalExit >> 24)), + + /// + /// The error messages are logged. + /// + Error = (1 << ((int)InstallMessage.Error >> 24)), + + /// + /// The warning messages are logged. + /// + Warning = (1 << ((int)InstallMessage.Warning >> 24)), + + /// + /// The user requests are logged. + /// + User = (1 << ((int)InstallMessage.User >> 24)), + + /// + /// The status messages that are not displayed are logged. + /// + Info = (1 << ((int)InstallMessage.Info >> 24)), + + /// + /// Request to determine a valid source location. + /// + ResolveSource = (1 << ((int)InstallMessage.ResolveSource >> 24)), + + /// + /// The was insufficient disk space. + /// + OutOfDiskSpace = (1 << ((int)InstallMessage.OutOfDiskSpace >> 24)), + + /// + /// The start of new installation actions are logged. + /// + ActionStart = (1 << ((int)InstallMessage.ActionStart >> 24)), + + /// + /// The data record with the installation action is logged. + /// + ActionData = (1 << ((int)InstallMessage.ActionData >> 24)), + + /// + /// The parameters for user-interface initialization are logged. + /// + CommonData = (1 << ((int)InstallMessage.CommonData >> 24)), + + /// + /// Logs the property values at termination. + /// + PropertyDump = (1 << ((int)InstallMessage.Progress >> 24)), + + /// + /// Sends large amounts of information to a log file not generally useful to users. + /// May be used for technical support. + /// + Verbose = (1 << ((int)InstallMessage.Initilize >> 24)), + + /// + /// Sends extra debugging information, such as handle creation information, to the log file. + /// + ExtraDebug = (1 << ((int)InstallMessage.Terminate >> 24)), + + /// + /// Progress bar information. This message includes information on units so far and total number of units. + /// See MsiProcessMessage for an explanation of the message format. + /// This message is only sent to an external user interface and is not logged. + /// + Progress = (1 << ((int)InstallMessage.Progress >> 24)), + + /// + /// If this is not a quiet installation, then the basic UI has been initialized. + /// If this is a full UI installation, the full UI is not yet initialized. + /// This message is only sent to an external user interface and is not logged. + /// + Initialize = (1 << ((int)InstallMessage.Initilize >> 24)), + + /// + /// If a full UI is being used, the full UI has ended. + /// If this is not a quiet installation, the basic UI has not yet ended. + /// This message is only sent to an external user interface and is not logged. + /// + Terminate = (1 << ((int)InstallMessage.Terminate >> 24)), + + /// + /// Sent prior to display of the full UI dialog. + /// This message is only sent to an external user interface and is not logged. + /// + ShowDialog = (1 << ((int)InstallMessage.ShowDialog >> 24)), + + /// + /// Files in use information. When this message is received, a FilesInUse Dialog should be displayed. + /// + FilesInUse = (1 << ((int)InstallMessage.FilesInUse >> 24)) + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/InstallMessage.cs b/src/wix/WixToolset.Core.Native/Msi/InstallMessage.cs new file mode 100644 index 00000000..35773e13 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/InstallMessage.cs @@ -0,0 +1,88 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + using System; + + /// + /// Windows Installer message types. + /// + [Flags] + public enum InstallMessage + { + /// + /// Premature termination, possibly fatal out of memory. + /// + FatalExit = 0x00000000, + + /// + /// Formatted error message, [1] is message number in Error table. + /// + Error = 0x01000000, + + /// + /// Formatted warning message, [1] is message number in Error table. + /// + Warning = 0x02000000, + + /// + /// User request message, [1] is message number in Error table. + /// + User = 0x03000000, + + /// + /// Informative message for log, not to be displayed. + /// + Info = 0x04000000, + + /// + /// List of files in use that need to be replaced. + /// + FilesInUse = 0x05000000, + + /// + /// Request to determine a valid source location. + /// + ResolveSource = 0x06000000, + + /// + /// Insufficient disk space message. + /// + OutOfDiskSpace = 0x07000000, + + /// + /// Progress: start of action, [1] action name, [2] description, [3] template for ACTIONDATA messages. + /// + ActionStart = 0x08000000, + + /// + /// Action data. Record fields correspond to the template of ACTIONSTART message. + /// + ActionData = 0x09000000, + + /// + /// Progress bar information. See the description of record fields below. + /// + Progress = 0x0A000000, + + /// + /// To enable the Cancel button set [1] to 2 and [2] to 1. To disable the Cancel button set [1] to 2 and [2] to 0. + /// + CommonData = 0x0B000000, + + /// + /// Sent prior to UI initialization, no string data. + /// + Initilize = 0x0C000000, + + /// + /// Sent after UI termination, no string data. + /// + Terminate = 0x0D000000, + + /// + /// Sent prior to display or authored dialog or wizard. + /// + ShowDialog = 0x0E000000 + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/InstallUILevels.cs b/src/wix/WixToolset.Core.Native/Msi/InstallUILevels.cs new file mode 100644 index 00000000..e84b5215 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/InstallUILevels.cs @@ -0,0 +1,72 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + using System; + + /// + /// Windows Installer UI levels. + /// + [Flags] + public enum InstallUILevels + { + /// + /// No change in the UI level. However, if phWnd is not Null, the parent window can change. + /// + NoChange = 0, + + /// + /// The installer chooses an appropriate user interface level. + /// + Default = 1, + + /// + /// Completely silent installation. + /// + None = 2, + + /// + /// Simple progress and error handling. + /// + Basic = 3, + + /// + /// Authored user interface with wizard dialog boxes suppressed. + /// + Reduced = 4, + + /// + /// Authored user interface with wizards, progress, and errors. + /// + Full = 5, + + /// + /// If combined with the Basic value, the installer shows simple progress dialog boxes but + /// does not display a Cancel button on the dialog. This prevents users from canceling the install. + /// Available with Windows Installer version 2.0. + /// + HideCancel = 0x20, + + /// + /// If combined with the Basic value, the installer shows simple progress + /// dialog boxes but does not display any modal dialog boxes or error dialog boxes. + /// + ProgressOnly = 0x40, + + /// + /// If combined with any above value, the installer displays a modal dialog + /// box at the end of a successful installation or if there has been an error. + /// No dialog box is displayed if the user cancels. + /// + EndDialog = 0x80, + + /// + /// If this value is combined with the None value, the installer displays only the dialog + /// boxes used for source resolution. No other dialog boxes are shown. This value has no + /// effect if the UI level is not INSTALLUILEVEL_NONE. It is used with an external user + /// interface designed to handle all of the UI except for source resolution. In this case, + /// the installer handles source resolution. This value is only available with Windows Installer 2.0 and later. + /// + SourceResOnly = 0x100 + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/Installer.cs b/src/wix/WixToolset.Core.Native/Msi/Installer.cs new file mode 100644 index 00000000..8a45aaa8 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/Installer.cs @@ -0,0 +1,138 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + using System; + using System.Diagnostics; + using System.Text; + + /// + /// A callback function that the installer calls for progress notification and error messages. + /// + /// Pointer to an application context. + /// This parameter can be used for error checking. + /// Specifies a combination of one message box style, + /// one message box icon type, one default button, and one installation message type. + /// Specifies the message text. + /// -1 for an error, 0 if no action was taken, 1 if OK, 3 to abort. + public delegate int InstallUIHandler(IntPtr context, uint messageType, string message); + + /// + /// Represents the Windows Installer, provides wrappers to + /// create the top-level objects and access their methods. + /// + public static class Installer + { + /// + /// Extacts the patch metadata as XML. + /// + /// Path to patch. + /// String XML. + public static string ExtractPatchXml(string path) + { + var buffer = new StringBuilder(65535); + var size = buffer.Capacity; + + var error = MsiInterop.MsiExtractPatchXMLData(path, 0, buffer, ref size); + if (234 == error) + { + buffer.EnsureCapacity(++size); + error = MsiInterop.MsiExtractPatchXMLData(path, 0, buffer, ref size); + } + + if (error != 0) + { + throw new MsiException(error); + } + + return buffer.ToString(); + } + + /// + /// Takes the path to a file and returns a 128-bit hash of that file. + /// + /// Path to file that is to be hashed. + /// The value in this column must be 0. This parameter is reserved for future use. + /// Int array that receives the returned file hash information. + public static void GetFileHash(string filePath, int options, out int[] hash) + { + var hashInterop = new MSIFILEHASHINFO(); + hashInterop.FileHashInfoSize = 20; + + var error = MsiInterop.MsiGetFileHash(filePath, Convert.ToUInt32(options), hashInterop); + if (0 != error) + { + throw new MsiException(error); + } + + Debug.Assert(20 == hashInterop.FileHashInfoSize); + + hash = new int[4]; + hash[0] = hashInterop.Data0; + hash[1] = hashInterop.Data1; + hash[2] = hashInterop.Data2; + hash[3] = hashInterop.Data3; + } + + /// + /// Returns the version string and language string in the format that the installer + /// expects to find them in the database. If you just want version information, set + /// lpLangBuf and pcchLangBuf to zero. If you just want language information, set + /// lpVersionBuf and pcchVersionBuf to zero. + /// + /// Specifies the path to the file. + /// Returns the file version. Set to 0 for language information only. + /// Returns the file language. Set to 0 for version information only. + public static void GetFileVersion(string filePath, out string version, out string language) + { + var versionLength = 20; + var languageLength = 20; + var versionBuffer = new StringBuilder(versionLength); + var languageBuffer = new StringBuilder(languageLength); + + var error = MsiInterop.MsiGetFileVersion(filePath, versionBuffer, ref versionLength, languageBuffer, ref languageLength); + if (234 == error) + { + versionBuffer.EnsureCapacity(++versionLength); + languageBuffer.EnsureCapacity(++languageLength); + error = MsiInterop.MsiGetFileVersion(filePath, versionBuffer, ref versionLength, languageBuffer, ref languageLength); + } + else if (1006 == error) + { + // file has no version or language, so no error + error = 0; + } + + if (0 != error) + { + throw new MsiException(error); + } + + version = versionBuffer.ToString(); + language = languageBuffer.ToString(); + } + + /// + /// Enables an external user-interface handler. + /// + /// Specifies a callback function. + /// Specifies which messages to handle using the external message handler. + /// Pointer to an application context that is passed to the callback function. + /// The return value is the previously set external handler, or null if there was no previously set handler. + public static InstallUIHandler SetExternalUI(InstallUIHandler installUIHandler, int messageFilter, IntPtr context) + { + return MsiInterop.MsiSetExternalUI(installUIHandler, messageFilter, context); + } + + /// + /// Enables the installer's internal user interface. + /// + /// Specifies the level of complexity of the user interface. + /// Pointer to a window. This window becomes the owner of any user interface created. + /// The previous user interface level is returned. If an invalid dwUILevel is passed, then INSTALLUILEVEL_NOCHANGE is returned. + public static int SetInternalUI(int uiLevel, ref IntPtr hwnd) + { + return MsiInterop.MsiSetInternalUI(uiLevel, ref hwnd); + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/MSIFILEHASHINFO.cs b/src/wix/WixToolset.Core.Native/Msi/MSIFILEHASHINFO.cs new file mode 100644 index 00000000..ae88ec7e --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/MSIFILEHASHINFO.cs @@ -0,0 +1,34 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + using System.Runtime.InteropServices; + + /// + /// contains the file hash information returned by MsiGetFileHash and used in the MsiFileHash table. + /// + [StructLayout(LayoutKind.Explicit)] + public class MSIFILEHASHINFO + { + /// + /// + /// + [FieldOffset(0)] public uint FileHashInfoSize; + /// + /// + /// + [FieldOffset(4)] public int Data0; + /// + /// + /// + [FieldOffset(8)] public int Data1; + /// + /// + /// + [FieldOffset(12)] public int Data2; + /// + /// + /// + [FieldOffset(16)] public int Data3; + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/ModifyView.cs b/src/wix/WixToolset.Core.Native/Msi/ModifyView.cs new file mode 100644 index 00000000..989de174 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/ModifyView.cs @@ -0,0 +1,75 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + /// + /// Enumeration of different modify modes. + /// + public enum ModifyView + { + /// + /// 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. + /// + Assign = 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. + + /// + /// Remove a row from the table. You must first call the Fetch 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. + /// + Delete = 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. + + /// + /// 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. + /// + Insert = 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. + + /// + /// 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. + /// + InsertTemporary = 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. + + /// + /// 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. + /// + Merge = 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. + + /// + /// Refreshes the information in the record. Must first call Fetch with the + /// same record. Fails for a deleted row. Works with read-write and read-only records. + /// + Refresh = 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. + + /// + /// Updates or deletes and inserts a record into a table. Must first call Fetch 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. + /// + Replace = 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. + + /// + /// 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. + /// + Seek = -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. + + /// + /// Updates an existing record. Non-primary keys only. Must first call Fetch. Fails with a + /// deleted record. Works only with read-write records. + /// + Update = 2, // Updates an existing record. Nonprimary keys only. Must first call MsiViewFetch. Fails with a deleted record. Works only with read-write records. + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/MsiException.cs b/src/wix/WixToolset.Core.Native/Msi/MsiException.cs new file mode 100644 index 00000000..07c83d81 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/MsiException.cs @@ -0,0 +1,77 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + using System; + using System.ComponentModel; + + /// + /// Exception that wraps MsiGetLastError(). + /// + [Serializable] + public sealed class MsiException : Win32Exception + { + /// + /// Instantiate a new MsiException with a given error. + /// + /// The error code from the MsiXxx() function call. + public MsiException(int error) : base(error) + { + uint handle = MsiInterop.MsiGetLastErrorRecord(); + if (0 != handle) + { + using (Record record = new Record(handle)) + { + this.MsiError = record.GetInteger(1); + + int errorInfoCount = record.GetFieldCount() - 1; + this.ErrorInfo = new string[errorInfoCount]; + for (int i = 0; i < errorInfoCount; ++i) + { + this.ErrorInfo[i] = record.GetString(i + 2); + } + } + } + else + { + this.MsiError = 0; + this.ErrorInfo = new string[0]; + } + + this.Error = error; + } + + /// + /// Gets the error number. + /// + public int Error { get; private set; } + + /// + /// Gets the internal MSI error number. + /// + public int MsiError { get; private set; } + + /// + /// Gets any additional the error information. + /// + public string[] ErrorInfo { get; private set; } + + /// + /// Overrides Message property to return useful error message. + /// + public override string Message + { + get + { + if (0 == this.MsiError) + { + return base.Message; + } + else + { + return String.Format("Internal MSI failure. Win32 error: {0}, MSI error: {1}, detail: {2}", this.Error, this.MsiError, String.Join(", ", this.ErrorInfo)); + } + } + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/MsiHandle.cs b/src/wix/WixToolset.Core.Native/Msi/MsiHandle.cs new file mode 100644 index 00000000..dc2ce605 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/MsiHandle.cs @@ -0,0 +1,117 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + using System; + using System.ComponentModel; +#if !DEBUG + using System.Diagnostics; +#endif + using System.Threading; + + /// + /// Wrapper class for MSI handle. + /// + public abstract class MsiHandle : IDisposable + { + private bool disposed; + private uint handle; + private int owningThread; +#if DEBUG + private string creationStack; +#endif + + /// + /// MSI handle destructor. + /// + ~MsiHandle() + { + this.Dispose(false); + } + + /// + /// Gets or sets the MSI handle. + /// + /// The MSI handle. + internal uint Handle + { + get + { + if (this.disposed) + { + throw new ObjectDisposedException("MsiHandle"); + } + + return this.handle; + } + + set + { + if (this.disposed) + { + throw new ObjectDisposedException("MsiHandle"); + } + + this.handle = value; + this.owningThread = Thread.CurrentThread.ManagedThreadId; +#if DEBUG + this.creationStack = Environment.StackTrace; +#endif + } + } + + /// + /// Close the MSI handle. + /// + public void Close() + { + this.Dispose(); + } + + /// + /// Disposes the managed and unmanaged objects in this object. + /// + public void Dispose() + { + this.Dispose(true); + GC.SuppressFinalize(this); + } + + /// + /// Disposes the managed and unmanaged objects in this object. + /// + /// true to dispose the managed objects. + protected virtual void Dispose(bool disposing) + { + if (!this.disposed) + { + if (0 != this.handle) + { + if (Thread.CurrentThread.ManagedThreadId == this.owningThread) + { + int error = MsiInterop.MsiCloseHandle(this.handle); + if (0 != error) + { + throw new Win32Exception(error); + } + this.handle = 0; + } + else + { + // Don't try to close the handle on a different thread than it was opened. + // This will occasionally cause MSI to AV. + string message = String.Format("Leaked msi handle {0} created on thread {1} by type {2}. This handle cannot be closed on thread {3}", + this.handle, this.owningThread, this.GetType(), Thread.CurrentThread.ManagedThreadId); +#if DEBUG + throw new InvalidOperationException(String.Format("{0}. Created {1}", message, this.creationStack)); +#else + Debug.WriteLine(message); +#endif + } + } + + this.disposed = true; + } + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/MsiInterop.cs b/src/wix/WixToolset.Core.Native/Msi/MsiInterop.cs new file mode 100644 index 00000000..11ac4094 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/MsiInterop.cs @@ -0,0 +1,408 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + using System; + using System.Text; + using System.Runtime.InteropServices; + + /// + /// Class exposing static functions and structs from MSI API. + /// + internal static class MsiInterop + { + // Patching constants + internal const int MsiMaxStreamNameLength = 62; // http://msdn2.microsoft.com/library/aa370551.aspx + + 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 + + /// + /// PInvoke of MsiCloseHandle. + /// + /// Handle to a database. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiCloseHandle", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiCloseHandle(uint database); + + /// + /// PInvoke of MsiCreateRecord + /// + /// Count of columns in the record. + /// Handle referencing the record. + [DllImport("msi.dll", EntryPoint = "MsiCreateRecord", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern uint MsiCreateRecord(int parameters); + + /// + /// Creates summary information of an existing transform to include validation and error conditions. + /// + /// The handle to the database that contains the new database summary information. + /// The handle to the database that contains the original summary information. + /// The name of the transform to which the summary information is added. + /// The error conditions that should be suppressed when the transform is applied. + /// Specifies the properties to be validated to verify that the transform can be applied to the database. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiCreateTransformSummaryInfoW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiCreateTransformSummaryInfo(uint database, uint referenceDatabase, string transformFile, TransformErrorConditions errorConditions, TransformValidations validations); + + /// + /// Applies a transform to a database. + /// + /// Handle to the database obtained from MsiOpenDatabase to transform. + /// Specifies the name of the transform file to apply. + /// Error conditions that should be suppressed. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiDatabaseApplyTransformW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiDatabaseApplyTransform(uint database, string transformFile, TransformErrorConditions errorConditions); + + /// + /// PInvoke of MsiDatabaseCommit. + /// + /// Handle to a databse. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiDatabaseCommit", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiDatabaseCommit(uint database); + + /// + /// PInvoke of MsiDatabaseExportW. + /// + /// Handle to a database. + /// Table name. + /// Folder path. + /// File name. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiDatabaseExportW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiDatabaseExport(uint database, string tableName, string folderPath, string fileName); + + /// + /// Generates a transform file of differences between two databases. + /// + /// Handle to the database obtained from MsiOpenDatabase that includes the changes. + /// Handle to the database obtained from MsiOpenDatabase that does not include the changes. + /// A null-terminated string that specifies the name of the transform file being generated. + /// This parameter can be null. If szTransformFile is null, you can use MsiDatabaseGenerateTransform to test whether two + /// databases are identical without creating a transform. If the databases are identical, the function returns ERROR_NO_DATA. + /// If the databases are different the function returns NOERROR. + /// This is a reserved argument and must be set to 0. + /// This is a reserved argument and must be set to 0. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiDatabaseGenerateTransformW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiDatabaseGenerateTransform(uint database, uint databaseReference, string transformFile, int reserved1, int reserved2); + + /// + /// PInvoke of MsiDatabaseImportW. + /// + /// Handle to a database. + /// Folder path. + /// File name. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiDatabaseImportW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiDatabaseImport(uint database, string folderPath, string fileName); + + /// + /// PInvoke of MsiDatabaseMergeW. + /// + /// The handle to the database obtained from MsiOpenDatabase. + /// The handle to the database obtained from MsiOpenDatabase to merge into the base database. + /// The name of the table to receive merge conflict information. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiDatabaseMergeW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiDatabaseMerge(uint database, uint databaseMerge, string tableName); + + /// + /// PInvoke of MsiDatabaseOpenViewW. + /// + /// Handle to a database. + /// SQL query. + /// View handle. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiDatabaseOpenViewW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiDatabaseOpenView(uint database, string query, out uint view); + + /// + /// PInvoke of MsiExtractPatchXMLDataW. + /// + /// Path to patch. + /// Reserved for future use. + /// Output XML data. + /// Count of characters in XML. + /// + [DllImport("msi.dll", EntryPoint = "MsiExtractPatchXMLDataW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiExtractPatchXMLData(string szPatchPath, int dwReserved, StringBuilder szXMLData, ref int pcchXMLData); + + /// + /// PInvoke of MsiGetFileHashW. + /// + /// File path. + /// Hash options (must be 0). + /// Buffer to recieve hash. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiGetFileHashW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiGetFileHash(string filePath, uint options, MSIFILEHASHINFO hash); + + /// + /// PInvoke of MsiGetFileVersionW. + /// + /// File path. + /// Buffer to receive version info. + /// Size of version buffer. + /// Buffer to recieve lang info. + /// Size of lang buffer. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiGetFileVersionW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiGetFileVersion(string filePath, StringBuilder versionBuf, ref int versionBufSize, StringBuilder langBuf, ref int langBufSize); + + /// + /// PInvoke of MsiGetLastErrorRecord. + /// + /// Handle to error record if one exists. + [DllImport("msi.dll", EntryPoint = "MsiGetLastErrorRecord", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern uint MsiGetLastErrorRecord(); + + /// + /// PInvoke of MsiDatabaseGetPrimaryKeysW. + /// + /// Handle to a database. + /// Table name. + /// Handle to receive resulting record. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiDatabaseGetPrimaryKeysW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiDatabaseGetPrimaryKeys(uint database, string tableName, out uint record); + + /// + /// PInvoke of MsiDoActionW. + /// + /// Handle to the installation provided to a DLL custom action or + /// obtained through MsiOpenPackage, MsiOpenPackageEx, or MsiOpenProduct. + /// Specifies the action to execute. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiDoActionW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiDoAction(uint product, string action); + + /// + /// PInvoke of MsiGetSummaryInformationW. Can use either database handle or database path as input. + /// + /// Handle to a database. + /// Path to a database. + /// Max number of updated values. + /// Handle to summary information. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiGetSummaryInformationW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiGetSummaryInformation(uint database, string databasePath, uint updateCount, ref uint summaryInfo); + + /// + /// PInvoke of MsiDatabaseIsTablePersitentW. + /// + /// Handle to a database. + /// Table name. + /// MSICONDITION + [DllImport("msi.dll", EntryPoint = "MsiDatabaseIsTablePersistentW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiDatabaseIsTablePersistent(uint database, string tableName); + + /// + /// PInvoke of MsiOpenDatabaseW. + /// + /// Path to database. + /// Persist mode. + /// Handle to database. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiOpenDatabaseW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiOpenDatabase(string databasePath, IntPtr persist, out uint database); + + /// + /// PInvoke of MsiOpenPackageW. + /// + /// The path to the package. + /// A pointer to a variable that receives the product handle. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiOpenPackageW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiOpenPackage(string packagePath, out uint product); + + /// + /// PInvoke of MsiRecordIsNull. + /// + /// MSI Record handle. + /// Index of field to check for null value. + /// true if the field is null, false if not, and an error code for any error. + [DllImport("msi.dll", EntryPoint = "MsiRecordIsNull", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiRecordIsNull(uint record, int field); + + /// + /// PInvoke of MsiRecordGetInteger. + /// + /// MSI Record handle. + /// Index of field to retrieve integer from. + /// Integer value. + [DllImport("msi.dll", EntryPoint = "MsiRecordGetInteger", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiRecordGetInteger(uint record, int field); + + /// + /// PInvoke of MsiRectordSetInteger. + /// + /// MSI Record handle. + /// Index of field to set integer value in. + /// Value to set field to. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiRecordSetInteger", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiRecordSetInteger(uint record, int field, int value); + + /// + /// PInvoke of MsiRecordGetStringW. + /// + /// MSI Record handle. + /// Index of field to get string value from. + /// Buffer to recieve value. + /// Size of buffer. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiRecordGetStringW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiRecordGetString(uint record, int field, StringBuilder valueBuf, ref int valueBufSize); + + /// + /// PInvoke of MsiRecordSetStringW. + /// + /// MSI Record handle. + /// Index of field to set string value in. + /// String value. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiRecordSetStringW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiRecordSetString(uint record, int field, string value); + + /// + /// PInvoke of MsiRecordSetStreamW. + /// + /// MSI Record handle. + /// Index of field to set stream value in. + /// Path to file to set stream value to. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiRecordSetStreamW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiRecordSetStream(uint record, int field, string filePath); + + /// + /// PInvoke of MsiRecordReadStreamW. + /// + /// MSI Record handle. + /// Index of field to read stream from. + /// Data buffer to recieve stream value. + /// Size of data buffer. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiRecordReadStream", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiRecordReadStream(uint record, int field, byte[] dataBuf, ref int dataBufSize); + + /// + /// PInvoke of MsiRecordGetFieldCount. + /// + /// MSI Record handle. + /// Count of fields in the record. + [DllImport("msi.dll", EntryPoint = "MsiRecordGetFieldCount", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiRecordGetFieldCount(uint record); + + /// + /// PInvoke of MsiSetExternalUIW. + /// + /// Specifies a callback function that conforms to the INSTALLUI_HANDLER specification. + /// Specifies which messages to handle using the external message handler. If the external + /// handler returns a non-zero result, then that message will not be sent to the UI, instead the message will be logged + /// if logging has been enabled. + /// Pointer to an application context that is passed to the callback function. + /// This parameter can be used for error checking. + /// The return value is the previously set external handler, or zero (0) if there was no previously set handler. + [DllImport("msi.dll", EntryPoint = "MsiSetExternalUIW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern InstallUIHandler MsiSetExternalUI(InstallUIHandler installUIHandler, int installLogMode, IntPtr context); + + /// + /// PInvoke of MsiSetInternalUI. + /// + /// Specifies the level of complexity of the user interface. + /// Pointer to a window. This window becomes the owner of any user interface created. + /// A pointer to the previous owner of the user interface is returned. + /// If this parameter is null, the owner of the user interface does not change. + /// The previous user interface level is returned. If an invalid dwUILevel is passed, then INSTALLUILEVEL_NOCHANGE is returned. + [DllImport("msi.dll", EntryPoint = "MsiSetInternalUI", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiSetInternalUI(int uiLevel, ref IntPtr hwnd); + + /// + /// PInvoke of MsiSummaryInfoGetPropertyW. + /// + /// Handle to summary info. + /// Property to get value from. + /// Data type of property. + /// Integer to receive integer value. + /// File time to receive file time value. + /// String buffer to receive string value. + /// Size of string buffer. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiSummaryInfoGetPropertyW", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiSummaryInfoGetProperty(uint summaryInfo, int property, out uint dataType, out int integerValue, ref System.Runtime.InteropServices.ComTypes.FILETIME fileTimeValue, StringBuilder stringValueBuf, ref int stringValueBufSize); + + /// + /// PInvoke of MsiViewGetColumnInfo. + /// + /// Handle to view. + /// Column info. + /// Handle for returned record. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiViewGetColumnInfo", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiViewGetColumnInfo(uint view, int columnInfo, out uint record); + + /// + /// PInvoke of MsiViewExecute. + /// + /// Handle of view to execute. + /// Handle to a record that supplies the parameters for the view. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiViewExecute", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiViewExecute(uint view, uint record); + + /// + /// PInvoke of MsiViewFetch. + /// + /// Handle of view to fetch a row from. + /// Handle to receive record info. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiViewFetch", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiViewFetch(uint view, out uint record); + + /// + /// PInvoke of MsiViewModify. + /// + /// Handle of view to modify. + /// Modify mode. + /// Handle of record. + /// Error code. + [DllImport("msi.dll", EntryPoint = "MsiViewModify", CharSet = CharSet.Unicode, ExactSpelling = true)] + internal static extern int MsiViewModify(uint view, int modifyMode, uint record); + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/OpenDatabase.cs b/src/wix/WixToolset.Core.Native/Msi/OpenDatabase.cs new file mode 100644 index 00000000..18a78f77 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/OpenDatabase.cs @@ -0,0 +1,40 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + /// + /// Enum of predefined persist modes used when opening a database. + /// + public enum OpenDatabase + { + /// + /// Open a database read-only, no persistent changes. + /// + ReadOnly = 0, + + /// + /// Open a database read/write in transaction mode. + /// + Transact = 1, + + /// + /// Open a database direct read/write without transaction. + /// + Direct = 2, + + /// + /// Create a new database, transact mode read/write. + /// + Create = 3, + + /// + /// Create a new database, direct mode read/write. + /// + CreateDirect = 4, + + /// + /// Indicates a patch file is being opened. + /// + OpenPatchFile = 32 + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/Record.cs b/src/wix/WixToolset.Core.Native/Msi/Record.cs new file mode 100644 index 00000000..c25e76e2 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/Record.cs @@ -0,0 +1,181 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + using System; + using System.ComponentModel; + using System.Text; + + /// + /// Wrapper class around msi.dll interop for a record. + /// + public sealed class Record : MsiHandle + { + /// + /// Creates a record with the specified number of fields. + /// + /// Number of fields in record. + public Record(int fieldCount) + { + this.Handle = MsiInterop.MsiCreateRecord(fieldCount); + if (0 == this.Handle) + { + throw new OutOfMemoryException(); + } + } + + /// + /// Creates a record from a handle. + /// + /// Handle to create record from. + internal Record(uint handle) + { + this.Handle = handle; + } + + /// + /// Gets a string value at specified location. + /// + /// Index into record to get string. + public string this[int field] + { + get => this.GetString(field); + set => this.SetString(field, value); + } + + /// + /// Determines if the value is null at the specified location. + /// + /// Index into record of the field to query. + /// true if the value is null, false otherwise. + public bool IsNull(int field) + { + var error = MsiInterop.MsiRecordIsNull(this.Handle, field); + + switch (error) + { + case 0: + return false; + case 1: + return true; + default: + throw new Win32Exception(error); + } + } + + /// + /// Gets integer value at specified location. + /// + /// Index into record to get integer + /// Integer value + public int GetInteger(int field) + { + return MsiInterop.MsiRecordGetInteger(this.Handle, field); + } + + /// + /// Sets integer value at specified location. + /// + /// Index into record to set integer. + /// Value to set into record. + public void SetInteger(int field, int value) + { + var error = MsiInterop.MsiRecordSetInteger(this.Handle, field, value); + if (0 != error) + { + throw new Win32Exception(error); + } + } + + /// + /// Gets string value at specified location. + /// + /// Index into record to get string. + /// String value + public string GetString(int field) + { + var bufferSize = 256; + var buffer = new StringBuilder(bufferSize); + var error = MsiInterop.MsiRecordGetString(this.Handle, field, buffer, ref bufferSize); + if (234 == error) + { + buffer.EnsureCapacity(++bufferSize); + error = MsiInterop.MsiRecordGetString(this.Handle, field, buffer, ref bufferSize); + } + + if (0 != error) + { + throw new Win32Exception(error); + } + + return (0 < buffer.Length ? buffer.ToString() : null); + } + + /// + /// Set string value at specified location + /// + /// Index into record to set string. + /// Value to set into record + public void SetString(int field, string value) + { + var error = MsiInterop.MsiRecordSetString(this.Handle, field, value); + if (0 != error) + { + throw new Win32Exception(error); + } + } + + /// + /// Get stream at specified location. + /// + /// Index into record to get stream. + /// buffer to receive bytes from stream. + /// Buffer size to read. + /// Stream read into string. + public int GetStream(int field, byte[] buffer, int requestedBufferSize) + { + var bufferSize = buffer.Length; + if (requestedBufferSize > 0) + { + bufferSize = requestedBufferSize; + } + + var error = MsiInterop.MsiRecordReadStream(this.Handle, field, buffer, ref bufferSize); + if (0 != error) + { + throw new Win32Exception(error); + } + + return bufferSize; + } + + /// + /// Sets a stream at a specified location. + /// + /// Index into record to set stream. + /// Path to file to read into stream. + public void SetStream(int field, string path) + { + var error = MsiInterop.MsiRecordSetStream(this.Handle, field, path); + if (0 != error) + { + throw new Win32Exception(error); + } + } + + /// + /// Gets the number of fields in record. + /// + /// Count of fields in record. + public int GetFieldCount() + { + var size = MsiInterop.MsiRecordGetFieldCount(this.Handle); + if (0 > size) + { + throw new Win32Exception(); + } + + return size; + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/Session.cs b/src/wix/WixToolset.Core.Native/Msi/Session.cs new file mode 100644 index 00000000..743fb2be --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/Session.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.Core.Native.Msi +{ + using System.Globalization; + + /// + /// Controls the installation process. + /// + public sealed class Session : MsiHandle + { + /// + /// Instantiate a new Session. + /// + /// The database to open. + public Session(Database database) + { + var packagePath = "#" + database.Handle.ToString(CultureInfo.InvariantCulture); + + var error = MsiInterop.MsiOpenPackage(packagePath, out var handle); + if (0 != error) + { + throw new MsiException(error); + } + + this.Handle = handle; + } + + /// + /// Executes a built-in action, custom action, or user-interface wizard action. + /// + /// Specifies the action to execute. + public void DoAction(string action) + { + var error = MsiInterop.MsiDoAction(this.Handle, action); + if (0 != error) + { + throw new MsiException(error); + } + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/SummaryInformation.cs b/src/wix/WixToolset.Core.Native/Msi/SummaryInformation.cs new file mode 100644 index 00000000..da629df2 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/SummaryInformation.cs @@ -0,0 +1,376 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + using System; + using System.Globalization; + using System.Text; + using FILETIME = System.Runtime.InteropServices.ComTypes.FILETIME; + + /// + /// Summary information for the MSI files. + /// + public sealed class SummaryInformation : MsiHandle + { + /// + /// Summary information properties for products. + /// + public enum Package + { + /// PID_CODEPAGE = code page of the summary information stream + CodePage = 1, + + /// PID_TITLE = a brief description of the package type + Title = 2, + + /// PID_SUBJECT = package name + PackageName = 3, + + /// PID_AUTHOR = manufacturer of the patch package + Manufacturer = 4, + + /// PID_KEYWORDS = list of keywords used by file browser + Keywords = 5, + + /// PID_COMMENTS = general purpose of the package + Comments = 6, + + /// PID_TEMPLATE = supported platforms and languages + PlatformsAndLanguages = 7, + + /// PID_LASTAUTHOR should be null for packages + Reserved8 = 8, + + /// PID_REVNUMBER = GUID package code + PackageCode = 9, + + /// PID_LASTPRINTED should be null for packages + Reserved11 = 11, + + /// PID_CREATED datetime when package was created + Created = 12, + + /// PID_SAVED datetime when package was last modified + LastModified = 13, + + /// PID_PAGECOUNT minimum required Windows Installer + InstallerRequirement = 14, + + /// PID_WORDCOUNT elevation privileges of package + FileAndElevatedFlags = 15, + + /// PID_CHARCOUNT should be null for patches + Reserved16 = 16, + + /// PID_APPLICATION tool used to create package + BuildTool = 18, + + /// PID_SECURITY = read-only attribute of the package + Security = 19, + } + + /// + /// Summary information properties for transforms. + /// + public enum Transform + { + /// PID_CODEPAGE = code page for the summary information stream + CodePage = 1, + + /// PID_TITLE = typically just "Transform" + Title = 2, + + /// PID_SUBJECT = original subject of target + TargetSubject = 3, + + /// PID_AUTHOR = original manufacturer of target + TargetManufacturer = 4, + + /// PID_KEYWORDS = keywords for the transform, typically including at least "Installer" + Keywords = 5, + + /// PID_COMMENTS = describes what this package does + Comments = 6, + + /// PID_TEMPLATE = target platform;language + TargetPlatformAndLanguage = 7, + + /// PID_LASTAUTHOR = updated platform;language + UpdatedPlatformAndLanguage = 8, + + /// PID_REVNUMBER = {productcode}version;{newproductcode}newversion;upgradecode + ProductCodes = 9, + + /// PID_LASTPRINTED should be null for transforms + Reserved11 = 11, + + ///.PID_CREATE_DTM = the timestamp when the transform was created + CreationTime = 12, + + /// PID_PAGECOUNT = minimum installer version + InstallerRequirement = 14, + + /// PID_CHARCOUNT = validation and error flags + ValidationFlags = 16, + + /// PID_APPNAME = the application that created the transform + CreatingApplication = 18, + + /// PID_SECURITY = whether read-only is enforced; should always be 4 for transforms + Security = 19, + } + + /// + /// Summary information properties for patches. + /// + public enum Patch + { + /// PID_CODEPAGE = code page of the summary information stream + CodePage = 1, + + /// PID_TITLE = a brief description of the package type + Title = 2, + + /// PID_SUBJECT = package name + PackageName = 3, + + /// PID_AUTHOR = manufacturer of the patch package + Manufacturer = 4, + + /// PID_KEYWORDS = alternate sources for the patch package + Sources = 5, + + /// PID_COMMENTS = general purpose of the patch package + Comments = 6, + + /// PID_TEMPLATE = semicolon delimited list of ProductCodes + ProductCodes = 7, + + /// PID_LASTAUTHOR = semicolon delimited list of transform names + TransformNames = 8, + + /// PID_REVNUMBER = GUID patch code + PatchCode = 9, + + /// PID_LASTPRINTED should be null for patches + Reserved11 = 11, + + /// PID_PAGECOUNT should be null for patches + Reserved14 = 14, + + /// PID_WORDCOUNT = minimum installer version + InstallerRequirement = 15, + + /// PID_CHARCOUNT should be null for patches + Reserved16 = 16, + + /// PID_SECURITY = read-only attribute of the patch package + Security = 19, + } + + /// + /// Summary information values for the InstallerRequirement property. + /// + public enum InstallerRequirement + { + /// Any version of the installer will do + Version10 = 1, + + /// At least 1.2 + Version12 = 2, + + /// At least 2.0 + Version20 = 3, + + /// At least 3.0 + Version30 = 4, + + /// At least 3.1 + Version31 = 5, + } + + /// + /// Instantiate a new SummaryInformation class from an open database. + /// + /// Database to retrieve summary information from. + public SummaryInformation(Database db) + { + if (null == db) + { + throw new ArgumentNullException(nameof(db)); + } + + uint handle = 0; + var error = MsiInterop.MsiGetSummaryInformation(db.Handle, null, 0, ref handle); + if (0 != error) + { + throw new MsiException(error); + } + this.Handle = handle; + } + + /// + /// Instantiate a new SummaryInformation class from a database file. + /// + /// The database file. + public SummaryInformation(string databaseFile) + { + if (null == databaseFile) + { + throw new ArgumentNullException(nameof(databaseFile)); + } + + uint handle = 0; + var error = MsiInterop.MsiGetSummaryInformation(0, databaseFile, 0, ref handle); + if (0 != error) + { + throw new MsiException(error); + } + this.Handle = handle; + } + + /// + /// Gets a summary information package property. + /// + /// The summary information package property. + /// The summary information property. + public string GetProperty(Package property) + { + return this.GetProperty((int)property); + } + + /// + /// Gets a summary information package property as a number. + /// + /// The summary information package property. + /// The summary information property. + public long GetNumericProperty(Package property) + { + return this.GetNumericProperty((int)property); + } + + /// + /// Gets a summary information patch property. + /// + /// The summary information patch property. + /// The summary information property. + public string GetProperty(Patch property) + { + return this.GetProperty((int)property); + } + + /// + /// Gets a summary information transform property. + /// + /// The summary information transform property. + /// The summary information property. + public long GetNumericProperty(Transform property) + { + return this.GetNumericProperty((int)property); + } + + /// + /// Gets a summary information property. + /// + /// Index of the summary information property. + /// The summary information property. + public string GetProperty(int index) + { + this.GetSummaryInformationValue(index, out var dataType, out var intValue, out var stringValue, out var timeValue); + + switch ((VT)dataType) + { + case VT.EMPTY: + return String.Empty; + + case VT.LPSTR: + return stringValue.ToString(); + + case VT.I2: + case VT.I4: + return Convert.ToString(intValue, CultureInfo.InvariantCulture); + + case VT.FILETIME: + var longFileTime = (((long)timeValue.dwHighDateTime) << 32) | unchecked((uint)timeValue.dwLowDateTime); + var dateTime = DateTime.FromFileTime(longFileTime); + return dateTime.ToString("yyyy/MM/dd HH:mm:ss", CultureInfo.InvariantCulture); + + default: + throw new InvalidOperationException(); + } + } + + /// + /// Gets a summary information property as a number. + /// + /// Index of the summary information property. + /// The summary information property. + public long GetNumericProperty(int index) + { + this.GetSummaryInformationValue(index, out var dataType, out var intValue, out var stringValue, out var timeValue); + + switch ((VT)dataType) + { + case VT.EMPTY: + return 0; + + case VT.LPSTR: + return Int64.Parse(stringValue.ToString(), CultureInfo.InvariantCulture); + + case VT.I2: + case VT.I4: + return intValue; + + case VT.FILETIME: + return (((long)timeValue.dwHighDateTime) << 32) | unchecked((uint)timeValue.dwLowDateTime); + + default: + throw new InvalidOperationException(); + } + } + + private void GetSummaryInformationValue(int index, out uint dataType, out int intValue, out StringBuilder stringValue, out FILETIME timeValue) + { + var bufSize = 64; + stringValue = new StringBuilder(bufSize); + timeValue.dwHighDateTime = 0; + timeValue.dwLowDateTime = 0; + + var error = MsiInterop.MsiSummaryInfoGetProperty(this.Handle, index, out dataType, out intValue, ref timeValue, stringValue, ref bufSize); + if (234 == error) + { + stringValue.EnsureCapacity(++bufSize); + error = MsiInterop.MsiSummaryInfoGetProperty(this.Handle, index, out dataType, out intValue, ref timeValue, stringValue, ref bufSize); + } + + if (0 != error) + { + throw new MsiException(error); + } + } + + /// + /// Variant types in the summary information table. + /// + private enum VT : uint + { + /// Variant has not been assigned. + EMPTY = 0, + + /// Null variant type. + NULL = 1, + + /// 16-bit integer variant type. + I2 = 2, + + /// 32-bit integer variant type. + I4 = 3, + + /// String variant type. + LPSTR = 30, + + /// Date time (FILETIME, converted to Variant time) variant type. + FILETIME = 64, + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/TransformErrorConditions.cs b/src/wix/WixToolset.Core.Native/Msi/TransformErrorConditions.cs new file mode 100644 index 00000000..313dceeb --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/TransformErrorConditions.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.Core.Native.Msi +{ + using System; + + /// + /// The errors to suppress when applying a transform. + /// + [Flags] + public enum TransformErrorConditions + { + /// + /// None of the following conditions. + /// + None = 0x0, + + /// + /// Suppress error when adding a row that exists. + /// + AddExistingRow = 0x1, + + /// + /// Suppress error when deleting a row that does not exist. + /// + DeleteMissingRow = 0x2, + + /// + /// Suppress error when adding a table that exists. + /// + AddExistingTable = 0x4, + + /// + /// Suppress error when deleting a table that does not exist. + /// + DeleteMissingTable = 0x8, + + /// + /// Suppress error when updating a row that does not exist. + /// + UpdateMissingRow = 0x10, + + /// + /// Suppress error when transform and database code pages do not match, and their code pages are neutral. + /// + ChangeCodepage = 0x20, + + /// + /// Create the temporary _TransformView table when applying a transform. + /// + ViewTransform = 0x100, + + /// + /// Suppress all errors but the option to create the temporary _TransformView table. + /// + All = 0x3F + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/TransformValidations.cs b/src/wix/WixToolset.Core.Native/Msi/TransformValidations.cs new file mode 100644 index 00000000..52bddeaf --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/TransformValidations.cs @@ -0,0 +1,73 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + using System; + + /// + /// The validation to run while applying a transform. + /// + [Flags] + public enum TransformValidations + { + /// + /// Do not validate properties. + /// + None = 0x0, + + /// + /// Default language must match base database. + /// + Language = 0x1, + + /// + /// Product must match base database. + /// + Product = 0x2, + + /// + /// Check major version only. + /// + MajorVersion = 0x8, + + /// + /// Check major and minor versions only. + /// + MinorVersion = 0x10, + + /// + /// Check major, minor, and update versions. + /// + UpdateVersion = 0x20, + + /// + /// Installed version < base version. + /// + NewLessBaseVersion = 0x40, + + /// + /// Installed version <= base version. + /// + NewLessEqualBaseVersion = 0x80, + + /// + /// Installed version = base version. + /// + NewEqualBaseVersion = 0x100, + + /// + /// Installed version >= base version. + /// + NewGreaterEqualBaseVersion = 0x200, + + /// + /// Installed version > base version. + /// + NewGreaterBaseVersion = 0x400, + + /// + /// UpgradeCode must match base database. + /// + UpgradeCode = 0x800 + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/View.cs b/src/wix/WixToolset.Core.Native/Msi/View.cs new file mode 100644 index 00000000..6305a9de --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/View.cs @@ -0,0 +1,206 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msi +{ + using System; + using System.Collections; + using System.Collections.Generic; + using System.Globalization; + + /// + /// Wrapper class for MSI API views. + /// + public sealed class View : MsiHandle + { + /// + /// Constructor that creates a view given a database handle and a query. + /// + /// Handle to the database to run the query on. + /// Query to be executed. + public View(Database db, string query) + { + if (null == db) + { + throw new ArgumentNullException(nameof(db)); + } + + if (null == query) + { + throw new ArgumentNullException(nameof(query)); + } + + var error = MsiInterop.MsiDatabaseOpenView(db.Handle, query, out var handle); + if (0 != error) + { + throw new MsiException(error); + } + + this.Handle = handle; + } + + /// + /// Enumerator that automatically disposes of the retrieved Records. + /// + public IEnumerable Records => new ViewEnumerable(this); + + /// + /// Executes a view with no customizable parameters. + /// + public void Execute() + { + this.Execute(null); + } + + /// + /// Executes a query substituing the values from the records into the customizable parameters + /// in the view. + /// + /// Record containing parameters to be substituded into the view. + public void Execute(Record record) + { + var error = MsiInterop.MsiViewExecute(this.Handle, null == record ? 0 : record.Handle); + if (0 != error) + { + throw new MsiException(error); + } + } + + /// + /// Fetches the next row in the view. + /// + /// Returns the fetched record; otherwise null. + public Record Fetch() + { + var error = MsiInterop.MsiViewFetch(this.Handle, out var recordHandle); + if (259 == error) + { + return null; + } + else if (0 != error) + { + throw new MsiException(error); + } + + return new Record(recordHandle); + } + + /// + /// Updates a fetched record. + /// + /// Type of modification mode. + /// Record to be modified. + public void Modify(ModifyView type, Record record) + { + var error = MsiInterop.MsiViewModify(this.Handle, Convert.ToInt32(type, CultureInfo.InvariantCulture), record.Handle); + if (0 != error) + { + throw new MsiException(error); + } + } + + /// + /// Get the column names in a record. + /// + /// + public Record GetColumnNames() + { + return this.GetColumnInfo(MsiInterop.MSICOLINFONAMES); + } + + /// + /// Get the column types in a record. + /// + /// + public Record GetColumnTypes() + { + return this.GetColumnInfo(MsiInterop.MSICOLINFOTYPES); + } + + /// + /// Returns a record containing column names or definitions. + /// + /// Specifies a flag indicating what type of information is needed. Either MSICOLINFO_NAMES or MSICOLINFO_TYPES. + /// The record containing information about the column. + public Record GetColumnInfo(int columnType) + { + + var error = MsiInterop.MsiViewGetColumnInfo(this.Handle, columnType, out var recordHandle); + if (0 != error) + { + throw new MsiException(error); + } + + return new Record(recordHandle); + } + + private class ViewEnumerable : IEnumerable + { + private readonly View view; + + public ViewEnumerable(View view) => this.view = view; + + public IEnumerator GetEnumerator() => new ViewEnumerator(this.view); + + IEnumerator IEnumerable.GetEnumerator() => new ViewEnumerator(this.view); + } + + private class ViewEnumerator : IEnumerator + { + private readonly View view; + private readonly List records = new List(); + private int position = -1; + private bool disposed; + + public ViewEnumerator(View view) => this.view = view; + + public Record Current => this.records[this.position]; + + object IEnumerator.Current => this.records[this.position]; + + public bool MoveNext() + { + if (this.position + 1 >= this.records.Count) + { + var record = this.view.Fetch(); + + if (record == null) + { + return false; + } + + this.records.Add(record); + this.position = this.records.Count - 1; + } + else + { + ++this.position; + } + + return true; + } + + public void Reset() => this.position = -1; + + public void Dispose() + { + this.Dispose(true); + } + + protected virtual void Dispose(bool disposing) + { + if (!this.disposed) + { + if (disposing) + { + foreach (var record in this.records) + { + record.Dispose(); + } + } + + this.disposed = true; + } + } + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msi/WixInvalidIdtException.cs b/src/wix/WixToolset.Core.Native/Msi/WixInvalidIdtException.cs new file mode 100644 index 00000000..268ddc11 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msi/WixInvalidIdtException.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.Core.Native.Msi +{ + using System; + using WixToolset.Data; + + /// + /// WiX invalid idt exception. + /// + [Serializable] + public sealed class WixInvalidIdtException : WixException + { + /// + /// Instantiate a new WixInvalidIdtException. + /// + public WixInvalidIdtException() + { + } + + /// + /// Instantiate a new WixInvalidIdtException. + /// + /// + /// + public WixInvalidIdtException(string message, Exception innerException) : base(message, innerException) + { + } + + /// + /// Instantiate a new WixInvalidIdtException. + /// + /// The invalid idt file. + public WixInvalidIdtException(string idtFile) : + base(ErrorMessages.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(ErrorMessages.InvalidIdt(new SourceLineNumber(idtFile), idtFile, tableName)) + { + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msm/ConfigurationCallback.cs b/src/wix/WixToolset.Core.Native/Msm/ConfigurationCallback.cs new file mode 100644 index 00000000..31b06d02 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msm/ConfigurationCallback.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.Core.Native.Msm +{ + using System; + using System.Collections; + using System.Globalization; + + /// + /// Callback object for configurable merge modules. + /// + public sealed class ConfigurationCallback : IMsmConfigureModule + { + private const int SOk = 0x0; + private const int SFalse = 0x1; + private readonly Hashtable configurationData; + + /// + /// Creates a ConfigurationCallback object. + /// + /// String to break up into name/value pairs. + public ConfigurationCallback(string configData) + { + if (String.IsNullOrEmpty(configData)) + { + throw new ArgumentNullException(nameof(configData)); + } + + var pairs = configData.Split(','); + this.configurationData = new Hashtable(pairs.Length); + for (var i = 0; i < pairs.Length; ++i) + { + var nameVal = pairs[i].Split('='); + var name = nameVal[0]; + var value = nameVal[1]; + + name = name.Replace("%2C", ","); + name = name.Replace("%3D", "="); + name = name.Replace("%25", "%"); + + value = value.Replace("%2C", ","); + value = value.Replace("%3D", "="); + value = value.Replace("%25", "%"); + + this.configurationData[name] = value; + } + } + + /// + /// Returns text data based on name. + /// + /// Name of value to return. + /// Out param to put configuration data into. + /// S_OK if value provided, S_FALSE if not. + public int ProvideTextData(string name, out string configData) + { + if (this.configurationData.Contains(name)) + { + configData = (string)this.configurationData[name]; + return SOk; + } + else + { + configData = null; + return SFalse; + } + } + + /// + /// Returns integer data based on name. + /// + /// Name of value to return. + /// Out param to put configuration data into. + /// S_OK if value provided, S_FALSE if not. + public int ProvideIntegerData(string name, out int configData) + { + if (this.configurationData.Contains(name)) + { + var val = (string)this.configurationData[name]; + configData = Convert.ToInt32(val, CultureInfo.InvariantCulture); + return SOk; + } + else + { + configData = 0; + return SFalse; + } + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msm/IMsmConfigureModule.cs b/src/wix/WixToolset.Core.Native/Msm/IMsmConfigureModule.cs new file mode 100644 index 00000000..468fb1fc --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msm/IMsmConfigureModule.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.Core.Native.Msm +{ + using System; + using System.Runtime.InteropServices; + + /// + /// Callback for configurable merge modules. + /// + [ComImport, Guid("AC013209-18A7-4851-8A21-2353443D70A0"), InterfaceType(ComInterfaceType.InterfaceIsIDispatch)] + public interface IMsmConfigureModule + { + /// + /// Callback to retrieve text data for configurable merge modules. + /// + /// Name of the data to be retrieved. + /// The data corresponding to the name. + /// The error code (HRESULT). + [PreserveSig] + int ProvideTextData([In, MarshalAs(UnmanagedType.BStr)] string name, [MarshalAs(UnmanagedType.BStr)] out string configData); + + /// + /// Callback to retrieve integer data for configurable merge modules. + /// + /// Name of the data to be retrieved. + /// The data corresponding to the name. + /// The error code (HRESULT). + [PreserveSig] + int ProvideIntegerData([In, MarshalAs(UnmanagedType.BStr)] string name, out int configData); + } +} diff --git a/src/wix/WixToolset.Core.Native/Msm/IMsmError.cs b/src/wix/WixToolset.Core.Native/Msm/IMsmError.cs new file mode 100644 index 00000000..4f1325a6 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msm/IMsmError.cs @@ -0,0 +1,77 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msm +{ + using System; + using System.Runtime.InteropServices; + + /// + /// A merge error. + /// + [ComImport, Guid("0ADDA828-2C26-11D2-AD65-00A0C9AF11A6")] + public interface IMsmError + { + /// + /// Gets the type of merge error. + /// + /// The type of merge error. + MsmErrorType Type + { + get; + } + + /// + /// Gets the path information from the merge error. + /// + /// The path information from the merge error. + string Path + { + get; + } + + /// + /// Gets the language information from the merge error. + /// + /// The language information from the merge error. + short Language + { + get; + } + + /// + /// Gets the database table from the merge error. + /// + /// The database table from the merge error. + string DatabaseTable + { + get; + } + + /// + /// Gets the collection of database keys from the merge error. + /// + /// The collection of database keys from the merge error. + IMsmStrings DatabaseKeys + { + get; + } + + /// + /// Gets the module table from the merge error. + /// + /// The module table from the merge error. + string ModuleTable + { + get; + } + + /// + /// Gets the collection of module keys from the merge error. + /// + /// The collection of module keys from the merge error. + IMsmStrings ModuleKeys + { + get; + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msm/IMsmErrors.cs b/src/wix/WixToolset.Core.Native/Msm/IMsmErrors.cs new file mode 100644 index 00000000..e1472376 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msm/IMsmErrors.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.Core.Native.Msm +{ + using System; + using System.Runtime.InteropServices; + + /// + /// Collection of merge errors. + /// + [ComImport, Guid("0ADDA82A-2C26-11D2-AD65-00A0C9AF11A6")] + public interface IMsmErrors + { + /// + /// Gets the IMsmError at the specified index. + /// + /// The one-based index of the IMsmError to get. + IMsmError this[int index] + { + get; + } + + /// + /// Gets the count of IMsmErrors in this collection. + /// + /// The count of IMsmErrors in this collection. + int Count + { + get; + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msm/IMsmMerge2.cs b/src/wix/WixToolset.Core.Native/Msm/IMsmMerge2.cs new file mode 100644 index 00000000..400249e7 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msm/IMsmMerge2.cs @@ -0,0 +1,174 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msm +{ + using System; + using System.Runtime.InteropServices; + + /// + /// IMsmMerge2 interface. + /// + [ComImport, Guid("351A72AB-21CB-47ab-B7AA-C4D7B02EA305")] + public interface IMsmMerge2 + { + /// + /// The OpenDatabase method of the Merge object opens a Windows Installer installation + /// database, located at a specified path, that is to be merged with a module. + /// + /// Path to the database being opened. + void OpenDatabase(string path); + + /// + /// The OpenModule method of the Merge object opens a Windows Installer merge module + /// in read-only mode. A module must be opened before it can be merged with an installation database. + /// + /// Fully qualified file name pointing to a merge module. + /// A valid language identifier (LANGID). + void OpenModule(string fileName, short language); + + /// + /// The CloseDatabase method of the Merge object closes the currently open Windows Installer database. + /// + /// true if changes should be saved, false otherwise. + void CloseDatabase(bool commit); + + /// + /// The CloseModule method of the Merge object closes the currently open Windows Installer merge module. + /// + void CloseModule(); + + /// + /// The OpenLog method of the Merge object opens a log file that receives progress and error messages. + /// If the log file already exists, the installer appends new messages. If the log file does not exist, + /// the installer creates a log file. + /// + /// Fully qualified filename pointing to a file to open or create. + void OpenLog(string fileName); + + /// + /// The CloseLog method of the Merge object closes the current log file. + /// + void CloseLog(); + + /// + /// The Log method of the Merge object writes a text string to the currently open log file. + /// + /// The text string to display. + void Log(string message); + + /// + /// Gets the errors from the last merge operation. + /// + /// The errors from the last merge operation. + IMsmErrors Errors + { + get; + } + + /// + /// Gets a collection of Dependency objects that enumerates a set of unsatisfied dependencies for the current database. + /// + /// A collection of Dependency objects that enumerates a set of unsatisfied dependencies for the current database. + object Dependencies + { + get; + } + + /// + /// The Merge method of the Merge object executes a merge of the current database and current + /// module. The merge attaches the components in the module to the feature identified by Feature. + /// The root of the module's directory tree is redirected to the location given by RedirectDir. + /// + /// The name of a feature in the database. + /// The key of an entry in the Directory table of the database. + /// This parameter may be NULL or an empty string. + void Merge(string feature, string redirectDir); + + /// + /// The Connect method of the Merge object connects a module to an additional feature. + /// The module must have already been merged into the database or will be merged into the database. + /// The feature must exist before calling this function. + /// + /// The name of a feature already existing in the database. + void Connect(string feature); + + /// + /// The ExtractCAB method of the Merge object extracts the embedded .cab file from a module and + /// saves it as the specified file. The installer creates this file if it does not already exist + /// and overwritten if it does exist. + /// + /// The fully qualified destination file. + void ExtractCAB(string fileName); + + /// + /// The ExtractFiles method of the Merge object extracts the embedded .cab file from a module + /// and then writes those files to the destination directory. + /// + /// The fully qualified destination directory. + void ExtractFiles(string path); + + /// + /// The MergeEx method of the Merge object is equivalent to the Merge function, except that it + /// takes an extra argument. The Merge method executes a merge of the current database and + /// current module. The merge attaches the components in the module to the feature identified + /// by Feature. The root of the module's directory tree is redirected to the location given by RedirectDir. + /// + /// The name of a feature in the database. + /// The key of an entry in the Directory table of the database. This parameter may + /// be NULL or an empty string. + /// The pConfiguration argument is an interface implemented by the client. The argument may + /// be NULL. The presence of this argument indicates that the client is capable of supporting the configuration + /// functionality, but does not obligate the client to provide configuration data for any specific configurable item. + void MergeEx(string feature, string redirectDir, IMsmConfigureModule configuration); + + /// + /// The ExtractFilesEx method of the Merge object extracts the embedded .cab file from a module and + /// then writes those files to the destination directory. + /// + /// The fully qualified destination directory. + /// Set to specify using long file names for path segments and final file names. + /// This is a list of fully-qualified paths for the files that were successfully extracted. + /// The list is empty if no files can be extracted. This argument may be null. No list is provided if pFilePaths is null. + void ExtractFilesEx(string path, bool longFileNames, ref IntPtr filePaths); + + /// + /// Gets a collection ConfigurableItem objects, each of which represents a single row from the ModuleConfiguration table. + /// + /// A collection ConfigurableItem objects, each of which represents a single row from the ModuleConfiguration table. + /// Semantically, each interface in the enumerator represents an item that can be configured by the module consumer. + /// The collection is a read-only collection and implements the standard read-only collection interfaces of Item(), Count() and _NewEnum(). + /// The IEnumMsmConfigItems enumerator implements Next(), Skip(), Reset(), and Clone() with the standard semantics. + object ConfigurableItems + { + get; + } + + /// + /// The CreateSourceImage method of the Merge object allows the client to extract the files from a module to + /// a source image on disk after a merge, taking into account changes to the module that might have been made + /// during module configuration. The list of files to be extracted is taken from the file table of the module + /// during the merge process. The list of files consists of every file successfully copied from the file table + /// of the module to the target database. File table entries that were not copied due to primary key conflicts + /// with existing rows in the database are not a part of this list. At image creation time, the directory for + /// each of these files comes from the open (post-merge) database. The path specified in the Path parameter is + /// the root of the source image for the install. fLongFileNames determines whether or not long file names are + /// used for both path segments and final file names. The function fails if no database is open, no module is + /// open, or no merge has been performed. + /// + /// The path of the root of the source image for the install. + /// Determines whether or not long file names are used for both path segments and final file names. + /// This is a list of fully-qualified paths for the files that were successfully extracted. + /// The list is empty if no files can be extracted. This argument may be null. No list is provided if pFilePaths is null. + void CreateSourceImage(string path, bool longFileNames, ref IntPtr filePaths); + + /// + /// The get_ModuleFiles function implements the ModuleFiles property of the GetFiles object. This function + /// returns the primary keys in the File table of the currently open module. The primary keys are returned + /// as a collection of strings. The module must be opened by a call to the OpenModule function before calling get_ModuleFiles. + /// + IMsmStrings ModuleFiles + { + get; + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msm/IMsmStrings.cs b/src/wix/WixToolset.Core.Native/Msm/IMsmStrings.cs new file mode 100644 index 00000000..41063bfa --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msm/IMsmStrings.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.Core.Native.Msm +{ + using System; + using System.Runtime.InteropServices; + + /// + /// A collection of strings. + /// + [ComImport, Guid("0ADDA827-2C26-11D2-AD65-00A0C9AF11A6")] + public interface IMsmStrings + { + /// + /// Gets the string at the specified index. + /// + /// The one-based index of the string to get. + string this[int index] + { + get; + } + + /// + /// Gets the count of strings in this collection. + /// + /// The count of strings in this collection. + int Count + { + get; + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Msm/MsmErrorType.cs b/src/wix/WixToolset.Core.Native/Msm/MsmErrorType.cs new file mode 100644 index 00000000..c67d37b4 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msm/MsmErrorType.cs @@ -0,0 +1,154 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Msm +{ + using System; + using System.Runtime.InteropServices; + + /// + /// Errors returned by merge operations. + /// + [Guid("0ADDA825-2C26-11D2-AD65-00A0C9AF11A6")] + public enum MsmErrorType + { + /// + /// A request was made to open a module with a language not supported by the module. + /// No more general language is supported by the module. + /// Adds msmErrorLanguageUnsupported to the Type property and the requested language + /// to the Language Property (Error Object). All Error object properties are empty. + /// The OpenModule function returns ERROR_INSTALL_LANGUAGE_UNSUPPORTED (as HRESULT). + /// + msmErrorLanguageUnsupported = 1, + + /// + /// A request was made to open a module with a supported language but the module has + /// an invalid language transform. Adds msmErrorLanguageFailed to the Type property + /// and the applied transform's language to the Language Property of the Error object. + /// This may not be the requested language if a more general language was used. + /// All other properties of the Error object are empty. The OpenModule function + /// returns ERROR_INSTALL_LANGUAGE_UNSUPPORTED (as HRESULT). + /// + msmErrorLanguageFailed = 2, + + /// + /// The module cannot be merged because it excludes, or is excluded by, another module + /// in the database. Adds msmErrorExclusion to the Type property of the Error object. + /// The ModuleKeys property or DatabaseKeys property contains the primary keys of the + /// excluded module's row in the ModuleExclusion table. If an existing module excludes + /// the module being merged, the excluded module's ModuleSignature information is added + /// to ModuleKeys. If the module being merged excludes an existing module, DatabaseKeys + /// contains the excluded module's ModuleSignature information. All other properties + /// are empty (or -1). + /// + msmErrorExclusion = 3, + + /// + /// Merge conflict during merge. The value of the Type property is set to + /// msmErrorTableMerge. The DatabaseTable property and DatabaseKeys property contain + /// the table name and primary keys of the conflicting row in the database. The + /// ModuleTable property and ModuleKeys property contain the table name and primary keys + /// of the conflicting row in the module. The ModuleTable and ModuleKeys entries may be + /// null if the row does not exist in the database. For example, if the conflict is in a + /// generated FeatureComponents table entry. On Windows Installer version 2.0, when + /// merging a configurable merge module, configuration may cause these properties to + /// refer to rows that do not exist in the module. + /// + msmErrorTableMerge = 4, + + /// + /// There was a problem resequencing a sequence table to contain the necessary merged + /// actions. The Type property is set to msmErrorResequenceMerge. The DatabaseTable + /// and DatabaseKeys properties contain the sequence table name and primary keys + /// (action name) of the conflicting row. The ModuleTable and ModuleKeys properties + /// contain the sequence table name and primary key (action name) of the conflicting row. + /// On Windows Installer version 2.0, when merging a configurable merge module, + /// configuration may cause these properties to refer to rows that do not exist in the module. + /// + msmErrorResequenceMerge = 5, + + /// + /// Not used. + /// + msmErrorFileCreate = 6, + + /// + /// There was a problem creating a directory to extract a file to disk. The Path property + /// contains the directory that could not be created. All other properties are empty or -1. + /// Not available with Windows Installer version 1.0. + /// + msmErrorDirCreate = 7, + + /// + /// A feature name is required to complete the merge, but no feature name was provided. + /// The Type property is set to msmErrorFeatureRequired. The DatabaseTable and DatabaseKeys + /// contain the table name and primary keys of the conflicting row. The ModuleTable and + /// ModuleKeys properties contain the table name and primary keys of the row cannot be merged. + /// On Windows Installer version 2.0, when merging a configurable merge module, configuration + /// may cause these properties to refer to rows that do not exist in the module. + /// If the failure is in a generated FeatureComponents table, the DatabaseTable and + /// DatabaseKeys properties are empty and the ModuleTable and ModuleKeys properties refer to + /// the row in the Component table causing the failure. + /// + msmErrorFeatureRequired = 8, + + /// + /// Available with Window Installer version 2.0. Substitution of a Null value into a + /// non-nullable column. This enters msmErrorBadNullSubstitution in the Type property and + /// enters "ModuleSubstitution" and the keys from the ModuleSubstitution table for this row + /// into the ModuleTable property and ModuleKeys property. All other properties of the Error + /// object are set to an empty string or -1. This error causes the immediate failure of the + /// merge and the MergeEx function to return E_FAIL. + /// + msmErrorBadNullSubstitution = 9, + + /// + /// Available with Window Installer version 2.0. Substitution of Text Format Type or Integer + /// Format Type into a Binary Type data column. This type of error returns + /// msmErrorBadSubstitutionType in the Type property and enters "ModuleSubstitution" and the + /// keys from the ModuleSubstitution table for this row into the ModuleTable property. + /// All other properties of the Error object are set to an empty string or -1. This error + /// causes the immediate failure of the merge and the MergeEx function to return E_FAIL. + /// + msmErrorBadSubstitutionType = 10, + + /// + /// Available with Window Installer Version 2.0. A row in the ModuleSubstitution table + /// references a configuration item not defined in the ModuleConfiguration table. + /// This type of error returns msmErrorMissingConfigItem in the Type property and enters + /// "ModuleSubstitution" and the keys from the ModuleSubstitution table for this row into + /// the ModuleTable property. All other properties of the Error object are set to an empty + /// string or -1. This error causes the immediate failure of the merge and the MergeEx + /// function to return E_FAIL. + /// + msmErrorMissingConfigItem = 11, + + /// + /// Available with Window Installer version 2.0. The authoring tool has returned a Null + /// value for an item marked with the msmConfigItemNonNullable attribute. An error of this + /// type returns msmErrorBadNullResponse in the Type property and enters "ModuleSubstitution" + /// and the keys from the ModuleSubstitution table for for the item into the ModuleTable property. + /// All other properties of the Error object are set to an empty string or -1. This error + /// causes the immediate failure of the merge and the MergeEx function to return E_FAIL. + /// + msmErrorBadNullResponse = 12, + + /// + /// Available with Window Installer version 2.0. The authoring tool returned a failure code + /// (not S_OK or S_FALSE) when asked for data. An error of this type will return + /// msmErrorDataRequestFailed in the Type property and enters "ModuleSubstitution" + /// and the keys from the ModuleSubstitution table for the item into the ModuleTable property. + /// All other properties of the Error object are set to an empty string or -1. This error + /// causes the immediate failure of the merge and the MergeEx function to return E_FAIL. + /// + msmErrorDataRequestFailed = 13, + + /// + /// Available with Windows Installer 2.0 and later versions. Indicates that an attempt was + /// made to merge a 64-bit module into a package that was not a 64-bit package. An error of + /// this type returns msmErrorPlatformMismatch in the Type property. All other properties of + /// the error object are set to an empty string or -1. This error causes the immediate failure + /// of the merge and causes the Merge function or MergeEx function to return E_FAIL. + /// + msmErrorPlatformMismatch = 14, + } +} diff --git a/src/wix/WixToolset.Core.Native/Msm/MsmInterop.cs b/src/wix/WixToolset.Core.Native/Msm/MsmInterop.cs new file mode 100644 index 00000000..d2627904 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Msm/MsmInterop.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.Core.Native.Msm +{ + using System; + using System.Runtime.InteropServices; + + /// + /// Merge merge modules into an MSI file. + /// + [ComImport, Guid("F94985D5-29F9-4743-9805-99BC3F35B678")] + public class MsmMerge2 + { + } + + /// + /// Defines the standard COM IClassFactory interface. + /// + [ComImport, Guid("00000001-0000-0000-C000-000000000046")] + [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] + public interface IClassFactory + { + /// + /// + /// + [return: MarshalAs(UnmanagedType.IUnknown)] + object CreateInstance(IntPtr unkOuter, [MarshalAs(UnmanagedType.LPStruct)] Guid iid); + } + + /// + /// Contains native methods for merge operations. + /// + public static class MsmInterop + { + [DllImport("mergemod.dll", EntryPoint = "DllGetClassObject", PreserveSig = false)] + [return: MarshalAs(UnmanagedType.IUnknown)] + private static extern object MergeModGetClassObject([MarshalAs(UnmanagedType.LPStruct)] Guid clsid, [MarshalAs(UnmanagedType.LPStruct)] Guid iid); + + /// + /// Load the merge object directly from a local mergemod.dll without going through COM registration. + /// + /// Merge interface. + public static IMsmMerge2 GetMsmMerge() + { + var classFactory = (IClassFactory)MergeModGetClassObject(typeof(MsmMerge2).GUID, typeof(IClassFactory).GUID); + return (IMsmMerge2)classFactory.CreateInstance(IntPtr.Zero, typeof(IMsmMerge2).GUID); + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Ole32/Storage.cs b/src/wix/WixToolset.Core.Native/Ole32/Storage.cs new file mode 100644 index 00000000..3e4c6af2 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Ole32/Storage.cs @@ -0,0 +1,377 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Ole32 +{ + using System; + using System.Runtime.InteropServices; + using FILETIME = System.Runtime.InteropServices.ComTypes.FILETIME; + using STATSTG = System.Runtime.InteropServices.ComTypes.STATSTG; + + /// + /// Wrapper for the compound storage file APIs. + /// + internal class Storage : IDisposable + { + private readonly IStorage storage; + private bool disposed; + + /// + /// Instantiate a new Storage. + /// + /// The native storage interface. + private Storage(IStorage storage) + { + this.storage = storage; + } + + /// + /// Storage destructor. + /// + ~Storage() + { + this.Dispose(); + } + + /// + /// The IEnumSTATSTG interface enumerates an array of STATSTG structures. + /// + [ComImport, Guid("0000000d-0000-0000-C000-000000000046"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] + private interface IEnumSTATSTG + { + /// + /// Gets a specified number of STATSTG structures. + /// + /// The number of STATSTG structures requested. + /// An array of STATSTG structures returned. + /// The number of STATSTG structures retrieved in the rgelt parameter. + /// The error code. + [PreserveSig] + uint Next(uint celt, [MarshalAs(UnmanagedType.LPArray), Out] STATSTG[] rgelt, out uint pceltFetched); + + /// + /// Skips a specified number of STATSTG structures in the enumeration sequence. + /// + /// The number of STATSTG structures to skip. + void Skip(uint celt); + + /// + /// Resets the enumeration sequence to the beginning of the STATSTG structure array. + /// + void Reset(); + + /// + /// Creates a new enumerator that contains the same enumeration state as the current STATSTG structure enumerator. + /// + /// The cloned IEnumSTATSTG interface. + [return: MarshalAs(UnmanagedType.Interface)] + IEnumSTATSTG Clone(); + } + + /// + /// The IStorage interface supports the creation and management of structured storage objects. + /// + [ComImport, Guid("0000000b-0000-0000-C000-000000000046"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] + private interface IStorage + { + /// + /// Creates and opens a stream object with the specified name contained in this storage object. + /// + /// The name of the newly created stream. + /// Specifies the access mode to use when opening the newly created stream. + /// Reserved for future use; must be zero. + /// Reserved for future use; must be zero. + /// On return, pointer to the location of the new IStream interface pointer. + void CreateStream(string pwcsName, uint grfMode, uint reserved1, uint reserved2, out IStream ppstm); + + /// + /// Opens an existing stream object within this storage object using the specified access permissions in grfMode. + /// + /// The name of the stream to open. + /// Reserved for future use; must be NULL. + /// Specifies the access mode to be assigned to the open stream. + /// Reserved for future use; must be zero. + /// A pointer to IStream pointer variable that receives the interface pointer to the newly opened stream object. + void OpenStream(string pwcsName, IntPtr reserved1, uint grfMode, uint reserved2, out IStream ppstm); + + /// + /// Creates and opens a new storage object nested within this storage object with the specified name in the specified access mode. + /// + /// The name of the newly created storage object. + /// A value that specifies the access mode to use when opening the newly created storage object. + /// Reserved for future use; must be zero. + /// Reserved for future use; must be zero. + /// A pointer, when successful, to the location of the IStorage pointer to the newly created storage object. + void CreateStorage(string pwcsName, uint grfMode, uint reserved1, uint reserved2, out IStorage ppstg); + + /// + /// Opens an existing storage object with the specified name in the specified access mode. + /// + /// The name of the storage object to open. + /// Must be NULL. + /// Specifies the access mode to use when opening the storage object. + /// Must be NULL. + /// Reserved for future use; must be zero. + /// When successful, pointer to the location of an IStorage pointer to the opened storage object. + void OpenStorage(string pwcsName, IStorage pstgPriority, uint grfMode, IntPtr snbExclude, uint reserved, out IStorage ppstg); + + /// + /// Copies the entire contents of an open storage object to another storage object. + /// + /// The number of elements in the array pointed to by rgiidExclude. + /// An array of interface identifiers (IIDs) that either the caller knows about and does not want + /// copied or that the storage object does not support, but whose state the caller will later explicitly copy. + /// A string name block (refer to SNB) that specifies a block of storage or stream objects that are not to be copied to the destination. + /// A pointer to the open storage object into which this storage object is to be copied. + void CopyTo(uint ciidExclude, IntPtr rgiidExclude, IntPtr snbExclude, IStorage pstgDest); + + /// + /// Copies or moves a substorage or stream from this storage object to another storage object. + /// + /// The name of the element in this storage object to be moved or copied. + /// IStorage pointer to the destination storage object. + /// The new name for the element in its new storage object. + /// Specifies whether the operation should be a move (STGMOVE_MOVE) or a copy (STGMOVE_COPY). + void MoveElementTo(string pwcsName, IStorage pstgDest, string pwcsNewName, uint grfFlags); + + /// + /// Reflects changes for a transacted storage object to the parent level. + /// + /// Controls how the changes are committed to the storage object. + void Commit(uint grfCommitFlags); + + /// + /// Discards all changes that have been made to the storage object since the last commit operation. + /// + void Revert(); + + /// + /// Returns an enumerator object that can be used to enumerate the storage and stream objects contained within this storage object. + /// + /// Reserved for future use; must be zero. + /// Reserved for future use; must be NULL. + /// Reserved for future use; must be zero. + /// Pointer to IEnumSTATSTG* pointer variable that receives the interface pointer to the new enumerator object. + void EnumElements(uint reserved1, IntPtr reserved2, uint reserved3, out IEnumSTATSTG ppenum); + + /// + /// Removes the specified storage or stream from this storage object. + /// + /// The name of the storage or stream to be removed. + void DestroyElement(string pwcsName); + + /// + /// Renames the specified storage or stream in this storage object. + /// + /// The name of the substorage or stream to be changed. + /// The new name for the specified substorage or stream. + void RenameElement(string pwcsOldName, string pwcsNewName); + + /// + /// Sets the modification, access, and creation times of the indicated storage element, if supported by the underlying file system. + /// + /// The name of the storage object element whose times are to be modified. + /// Either the new creation time for the element or NULL if the creation time is not to be modified. + /// Either the new access time for the element or NULL if the access time is not to be modified. + /// Either the new modification time for the element or NULL if the modification time is not to be modified. + void SetElementTimes(string pwcsName, FILETIME pctime, FILETIME patime, FILETIME pmtime); + + /// + /// Assigns the specified CLSID to this storage object. + /// + /// The CLSID that is to be associated with the storage object. + void SetClass(Guid clsid); + + /// + /// Stores up to 32 bits of state information in this storage object. + /// + /// Specifies the new values of the bits to set. + /// A binary mask indicating which bits in grfStateBits are significant in this call. + void SetStateBits(uint grfStateBits, uint grfMask); + + /// + /// Returns the STATSTG structure for this open storage object. + /// + /// On return, pointer to a STATSTG structure where this method places information about the open storage object. + /// Specifies that some of the members in the STATSTG structure are not returned, thus saving a memory allocation operation. + void Stat(out STATSTG pstatstg, uint grfStatFlag); + } + + /// + /// The IStream interface lets you read and write data to stream objects. + /// + [ComImport, Guid("0000000c-0000-0000-C000-000000000046"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] + private interface IStream + { + /// + /// Reads a specified number of bytes from the stream object into memory starting at the current seek pointer. + /// + /// A pointer to the buffer which the stream data is read into. + /// The number of bytes of data to read from the stream object. + /// A pointer to a ULONG variable that receives the actual number of bytes read from the stream object. + void Read([Out, MarshalAs(UnmanagedType.LPArray, SizeParamIndex = 1)] byte[] pv, int cb, IntPtr pcbRead); + + /// + /// Writes a specified number of bytes into the stream object starting at the current seek pointer. + /// + /// A pointer to the buffer that contains the data that is to be written to the stream. + /// The number of bytes of data to attempt to write into the stream. + /// A pointer to a ULONG variable where this method writes the actual number of bytes written to the stream object. + void Write([MarshalAs(UnmanagedType.LPArray, SizeParamIndex = 1)] byte[] pv, int cb, IntPtr pcbWritten); + + /// + /// Changes the seek pointer to a new location relative to the beginning of the stream, the end of the stream, or the current seek pointer. + /// + /// The displacement to be added to the location indicated by the dwOrigin parameter. + /// The origin for the displacement specified in dlibMove. + /// A pointer to the location where this method writes the value of the new seek pointer from the beginning of the stream. + void Seek(long dlibMove, int dwOrigin, IntPtr plibNewPosition); + + /// + /// Changes the size of the stream object. + /// + /// Specifies the new size of the stream as a number of bytes. + void SetSize(long libNewSize); + + /// + /// Copies a specified number of bytes from the current seek pointer in the stream to the current seek pointer in another stream. + /// + /// A pointer to the destination stream. + /// The number of bytes to copy from the source stream. + /// A pointer to the location where this method writes the actual number of bytes read from the source. + /// A pointer to the location where this method writes the actual number of bytes written to the destination. + void CopyTo(IStream pstm, long cb, IntPtr pcbRead, IntPtr pcbWritten); + + /// + /// Ensures that any changes made to a stream object open in transacted mode are reflected in the parent storage object. + /// + /// Controls how the changes for the stream object are committed. + void Commit(int grfCommitFlags); + + /// + /// Discards all changes that have been made to a transacted stream since the last call to IStream::Commit. + /// + void Revert(); + + /// + /// Restricts access to a specified range of bytes in the stream. + /// + /// Integer that specifies the byte offset for the beginning of the range. + /// Integer that specifies the length of the range, in bytes, to be restricted. + /// Specifies the restrictions being requested on accessing the range. + void LockRegion(long libOffset, long cb, int dwLockType); + + /// + /// Removes the access restriction on a range of bytes previously restricted with IStream::LockRegion. + /// + /// Specifies the byte offset for the beginning of the range. + /// Specifies, in bytes, the length of the range to be restricted. + /// Specifies the access restrictions previously placed on the range. + void UnlockRegion(long libOffset, long cb, int dwLockType); + + /// + /// Retrieves the STATSTG structure for this stream. + /// + /// Pointer to a STATSTG structure where this method places information about this stream object. + /// Specifies that this method does not return some of the members in the STATSTG structure, thus saving a memory allocation operation. + void Stat(out STATSTG pstatstg, int grfStatFlag); + + /// + /// Creates a new stream object that references the same bytes as the original stream but provides a separate seek pointer to those bytes. + /// + /// When successful, pointer to the location of an IStream pointer to the new stream object. + void Clone(out IStream ppstm); + } + + /// + /// Creates a new compound file storage object. + /// + /// The compound file being created. + /// Specifies the access mode to use when opening the new storage object. + /// The created Storage object. + public static Storage CreateDocFile(string storageFile, StorageMode mode) + { + var storage = NativeMethods.StgCreateDocfile(storageFile, (uint)mode, 0); + + return new Storage(storage); + } + + /// + /// Opens an existing root storage object in the file system. + /// + /// The file that contains the storage object to open. + /// Specifies the access mode to use to open the storage object. + /// The created Storage object. + public static Storage Open(string storageFile, StorageMode mode) + { + var storage = NativeMethods.StgOpenStorage(storageFile, IntPtr.Zero, (uint)mode, IntPtr.Zero, 0); + + return new Storage(storage); + } + + /// + /// Copies the entire contents of this open storage object into another Storage object. + /// + /// The destination Storage object. + public void CopyTo(Storage destinationStorage) + { + this.storage.CopyTo(0, IntPtr.Zero, IntPtr.Zero, destinationStorage.storage); + } + + /// + /// Opens an existing Storage object with the specified name according to the specified access mode. + /// + /// The name of the Storage object. + /// The opened Storage object. + public Storage OpenStorage(string name) + { + this.storage.OpenStorage(name, null, (uint)(StorageMode.Read | StorageMode.ShareExclusive), IntPtr.Zero, 0, out var subStorage); + + return new Storage(subStorage); + } + + /// + /// Disposes the managed and unmanaged objects in this object. + /// + public void Dispose() + { + if (!this.disposed) + { + Marshal.ReleaseComObject(this.storage); + + this.disposed = true; + } + + GC.SuppressFinalize(this); + } + + /// + /// The native methods. + /// + private static class NativeMethods + { + /// + /// Creates a new compound file storage object. + /// + /// The name for the compound file being created. + /// Specifies the access mode to use when opening the new storage object. + /// Reserved for future use; must be zero. + /// A pointer to the location of the IStorage pointer to the new storage object. + [DllImport("ole32.dll", PreserveSig = false)] + [return: MarshalAs(UnmanagedType.Interface)] + internal static extern IStorage StgCreateDocfile([MarshalAs(UnmanagedType.LPWStr)] string pwcsName, uint grfMode, uint reserved); + + /// + /// Opens an existing root storage object in the file system. + /// + /// The file that contains the storage object to open. + /// Most often NULL. + /// Specifies the access mode to use to open the storage object. + /// If not NULL, pointer to a block of elements in the storage to be excluded as the storage object is opened. + /// Indicates reserved for future use; must be zero. + /// A pointer to a IStorage* pointer variable that receives the interface pointer to the opened storage. + [DllImport("ole32.dll", PreserveSig = false)] + [return: MarshalAs(UnmanagedType.Interface)] + internal static extern IStorage StgOpenStorage([MarshalAs(UnmanagedType.LPWStr)] string pwcsName, IntPtr pstgPriority, uint grfMode, IntPtr snbExclude, uint reserved); + } + } +} diff --git a/src/wix/WixToolset.Core.Native/Ole32/StorageMode.cs b/src/wix/WixToolset.Core.Native/Ole32/StorageMode.cs new file mode 100644 index 00000000..24b60e4d --- /dev/null +++ b/src/wix/WixToolset.Core.Native/Ole32/StorageMode.cs @@ -0,0 +1,55 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.Ole32 +{ + /// + /// Specifies the access mode to use when opening, creating, or deleting a storage object. + /// + internal enum StorageMode + { + /// + /// Indicates that the object is read-only, meaning that modifications cannot be made. + /// + Read = 0x0, + + /// + /// Enables you to save changes to the object, but does not permit access to its data. + /// + Write = 0x1, + + /// + /// Enables access and modification of object data. + /// + ReadWrite = 0x2, + + /// + /// Specifies that subsequent openings of the object are not denied read or write access. + /// + ShareDenyNone = 0x40, + + /// + /// Prevents others from subsequently opening the object in Read mode. + /// + ShareDenyRead = 0x30, + + /// + /// Prevents others from subsequently opening the object for Write or ReadWrite access. + /// + ShareDenyWrite = 0x20, + + /// + /// Prevents others from subsequently opening the object in any mode. + /// + ShareExclusive = 0x10, + + /// + /// Opens the storage object with exclusive access to the most recently committed version. + /// + Priority = 0x40000, + + /// + /// Indicates that an existing storage object or stream should be removed before the new object replaces it. + /// + Create = 0x1000, + } +} diff --git a/src/wix/WixToolset.Core.Native/PatchAPI/PatchInterop.cs b/src/wix/WixToolset.Core.Native/PatchAPI/PatchInterop.cs new file mode 100644 index 00000000..04f5a553 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/PatchAPI/PatchInterop.cs @@ -0,0 +1,990 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native.PatchAPI +{ + using System; + using System.Collections.Generic; + using System.Diagnostics.CodeAnalysis; + using System.Globalization; + using System.Runtime.InteropServices; + using WixToolset.Data.Symbols; + + /// + /// Interop class for the mspatchc.dll. + /// + internal static class PatchInterop + { + // From WinError.h in the Platform SDK + internal const ushort FACILITY_WIN32 = 7; + + /// + /// Parse a number from text in either hex or decimal. + /// + /// Source value. Treated as hex if it starts 0x (or 0X), decimal otherwise. + /// Numeric value that source represents. + internal static uint ParseHexOrDecimal(string source) + { + var value = source.Trim(); + if (String.Equals(value.Substring(0, 2), "0x", StringComparison.OrdinalIgnoreCase)) + { + return UInt32.Parse(value.Substring(2), NumberStyles.AllowHexSpecifier, CultureInfo.InvariantCulture.NumberFormat); + } + else + { + return UInt32.Parse(value, CultureInfo.InvariantCulture.NumberFormat); + } + } + + /// + /// Create a binary delta file. + /// + /// Name of the delta file to create. + /// Name of updated file. + /// Optional paths to updated file's symbols. + /// Optional offsets to the delta retain sections in the updated file. + /// Optional array of target files. + /// Optional array of target files' symbol paths (must match basisFiles array). + /// Optional array of target files' delta ignore section lengths (must match basisFiles array)(each entry must match basisIgnoreOffsets entries). + /// Optional array of target files' delta ignore section offsets (must match basisFiles array)(each entry must match basisIgnoreLengths entries). + /// Optional array of target files' delta protect section lengths (must match basisFiles array)(each entry must match basisRetainOffsets and targetRetainOffsets entries). + /// Optional array of target files' delta protect section offsets (must match basisFiles array)(each entry must match basisRetainLengths and targetRetainOffsets entries). + /// ApiPatchingSymbolFlags value. + /// OptimizePatchSizeForLargeFiles value. + /// Flag to indicate retain ranges were ignored due to mismatch. + /// true if delta file was created, false if whole file should be used instead. + public static bool CreateDelta( + string deltaFile, + string targetFile, + string targetSymbolPath, + string targetRetainOffsets, + string[] basisFiles, + string[] basisSymbolPaths, + string[] basisIgnoreLengths, + string[] basisIgnoreOffsets, + string[] basisRetainLengths, + string[] basisRetainOffsets, + PatchSymbolFlags apiPatchingSymbolFlags, + bool optimizePatchSizeForLargeFiles, + out bool retainRangesIgnored + ) + { + retainRangesIgnored = false; + if (0 != (apiPatchingSymbolFlags & ~(PatchSymbolFlags.PatchSymbolNoImagehlp | PatchSymbolFlags.PatchSymbolNoFailures | PatchSymbolFlags.PatchSymbolUndecoratedToo))) + { + throw new ArgumentOutOfRangeException("apiPatchingSymbolFlags"); + } + + if (null == deltaFile || 0 == deltaFile.Length) + { + throw new ArgumentNullException("deltaFile"); + } + + if (null == targetFile || 0 == targetFile.Length) + { + throw new ArgumentNullException("targetFile"); + } + + if (null == basisFiles || 0 == basisFiles.Length) + { + return false; + } + var countOldFiles = (uint)basisFiles.Length; + + if (null != basisSymbolPaths) + { + if (0 != basisSymbolPaths.Length) + { + if ((uint)basisSymbolPaths.Length != countOldFiles) + { + throw new ArgumentOutOfRangeException("basisSymbolPaths"); + } + } + } + // a null basisSymbolPaths is allowed. + + if (null != basisIgnoreLengths) + { + if (0 != basisIgnoreLengths.Length) + { + if ((uint)basisIgnoreLengths.Length != countOldFiles) + { + throw new ArgumentOutOfRangeException("basisIgnoreLengths"); + } + } + } + else + { + basisIgnoreLengths = new string[countOldFiles]; + } + + if (null != basisIgnoreOffsets) + { + if (0 != basisIgnoreOffsets.Length) + { + if ((uint)basisIgnoreOffsets.Length != countOldFiles) + { + throw new ArgumentOutOfRangeException("basisIgnoreOffsets"); + } + } + } + else + { + basisIgnoreOffsets = new string[countOldFiles]; + } + + if (null != basisRetainLengths) + { + if (0 != basisRetainLengths.Length) + { + if ((uint)basisRetainLengths.Length != countOldFiles) + { + throw new ArgumentOutOfRangeException("basisRetainLengths"); + } + } + } + else + { + basisRetainLengths = new string[countOldFiles]; + } + + if (null != basisRetainOffsets) + { + if (0 != basisRetainOffsets.Length) + { + if ((uint)basisRetainOffsets.Length != countOldFiles) + { + throw new ArgumentOutOfRangeException("basisRetainOffsets"); + } + } + } + else + { + basisRetainOffsets = new string[countOldFiles]; + } + + var pod = new PatchOptionData(); + pod.symbolOptionFlags = apiPatchingSymbolFlags; + pod.newFileSymbolPath = targetSymbolPath; + pod.oldFileSymbolPathArray = basisSymbolPaths; + pod.extendedOptionFlags = 0; + var oldFileInfoArray = new PatchOldFileInfoW[countOldFiles]; + var newRetainOffsetArray = ((null == targetRetainOffsets) ? new string[0] : targetRetainOffsets.Split(',')); + for (uint i = 0; i < countOldFiles; ++i) + { + var ofi = new PatchOldFileInfoW(); + ofi.oldFileName = basisFiles[i]; + var ignoreLengthArray = ((null == basisIgnoreLengths[i]) ? new string[0] : basisIgnoreLengths[i].Split(',')); + var ignoreOffsetArray = ((null == basisIgnoreOffsets[i]) ? new string[0] : basisIgnoreOffsets[i].Split(',')); + var retainLengthArray = ((null == basisRetainLengths[i]) ? new string[0] : basisRetainLengths[i].Split(',')); + var retainOffsetArray = ((null == basisRetainOffsets[i]) ? new string[0] : basisRetainOffsets[i].Split(',')); + // Validate inputs + if (ignoreLengthArray.Length != ignoreOffsetArray.Length) + { + throw new ArgumentOutOfRangeException("basisIgnoreLengths"); + } + + if (retainLengthArray.Length != retainOffsetArray.Length) + { + throw new ArgumentOutOfRangeException("basisRetainLengths"); + } + + if (newRetainOffsetArray.Length != retainOffsetArray.Length) + { + // remove all retain range information + retainRangesIgnored = true; + for (uint j = 0; j < countOldFiles; ++j) + { + basisRetainLengths[j] = null; + basisRetainOffsets[j] = null; + } + retainLengthArray = new string[0]; + retainOffsetArray = new string[0]; + newRetainOffsetArray = new string[0]; + for (uint j = 0; j < oldFileInfoArray.Length; ++j) + { + oldFileInfoArray[j].retainRange = null; + } + } + + // Populate IgnoreRange structure + PatchIgnoreRange[] ignoreArray = null; + if (0 != ignoreLengthArray.Length) + { + ignoreArray = new PatchIgnoreRange[ignoreLengthArray.Length]; + for (var j = 0; j < ignoreLengthArray.Length; ++j) + { + var ignoreRange = new PatchIgnoreRange(); + ignoreRange.offsetInOldFile = ParseHexOrDecimal(ignoreOffsetArray[j]); + ignoreRange.lengthInBytes = ParseHexOrDecimal(ignoreLengthArray[j]); + ignoreArray[j] = ignoreRange; + } + ofi.ignoreRange = ignoreArray; + } + + PatchRetainRange[] retainArray = null; + if (0 != newRetainOffsetArray.Length) + { + retainArray = new PatchRetainRange[retainLengthArray.Length]; + for (var j = 0; j < newRetainOffsetArray.Length; ++j) + { + var retainRange = new PatchRetainRange(); + retainRange.offsetInOldFile = ParseHexOrDecimal(retainOffsetArray[j]); + retainRange.lengthInBytes = ParseHexOrDecimal(retainLengthArray[j]); + retainRange.offsetInNewFile = ParseHexOrDecimal(newRetainOffsetArray[j]); + retainArray[j] = retainRange; + } + ofi.retainRange = retainArray; + } + oldFileInfoArray[i] = ofi; + } + + if (CreatePatchFileExW( + countOldFiles, + oldFileInfoArray, + targetFile, + deltaFile, + PatchOptionFlags(optimizePatchSizeForLargeFiles), + pod, + null, + IntPtr.Zero)) + { + return true; + } + + // determine if this is an error or a need to use whole file. + var err = Marshal.GetLastWin32Error(); + switch (err) + { + case unchecked((int)ERROR_PATCH_BIGGER_THAN_COMPRESSED): + break; + + // too late to exclude this file -- should have been caught before + case unchecked((int)ERROR_PATCH_SAME_FILE): + default: + throw new System.ComponentModel.Win32Exception(err); + } + return false; + } + + /// + /// Extract the delta header. + /// + /// Name of delta file. + /// Name of file to create with the delta's header. + static public void ExtractDeltaHeader(string delta, string deltaHeader) + { + if (!ExtractPatchHeaderToFileW(delta, deltaHeader)) + { + throw new System.ComponentModel.Win32Exception(Marshal.GetLastWin32Error()); + } + } + + /// + /// Returns the PatchOptionFlags to use. + /// + /// True if optimizing for large files. + /// PATCH_OPTION_FLAG values + static private UInt32 PatchOptionFlags(bool optimizeForLargeFiles) + { + var flags = PATCH_OPTION_FAIL_IF_SAME_FILE | PATCH_OPTION_FAIL_IF_BIGGER | PATCH_OPTION_USE_LZX_BEST; + if (optimizeForLargeFiles) + { + flags |= PATCH_OPTION_USE_LZX_LARGE; + } + return flags; + } + + //--------------------------------------------------------------------- + // From PatchApi.h + //--------------------------------------------------------------------- + + // + // The following contants can be combined and used as the OptionFlags + // parameter in the patch creation apis. + + internal const uint PATCH_OPTION_USE_BEST = 0x00000000; // auto choose best (slower) + + internal const uint PATCH_OPTION_USE_LZX_BEST = 0x00000003; // auto choose best of LXZ A/B (but not large) + internal const uint PATCH_OPTION_USE_LZX_A = 0x00000001; // normal + internal const uint PATCH_OPTION_USE_LXZ_B = 0x00000002; // better on some x86 binaries + internal const uint PATCH_OPTION_USE_LZX_LARGE = 0x00000004; // better support for large files (requires 5.1 or higher applyer) + + internal const uint PATCH_OPTION_NO_BINDFIX = 0x00010000; // PE bound imports + internal const uint PATCH_OPTION_NO_LOCKFIX = 0x00020000; // PE smashed locks + internal const uint PATCH_OPTION_NO_REBASE = 0x00040000; // PE rebased image + internal const uint PATCH_OPTION_FAIL_IF_SAME_FILE = 0x00080000; // don't create if same + internal const uint PATCH_OPTION_FAIL_IF_BIGGER = 0x00100000; // fail if patch is larger than simply compressing new file (slower) + internal const uint PATCH_OPTION_NO_CHECKSUM = 0x00200000; // PE checksum zero + internal const uint PATCH_OPTION_NO_RESTIMEFIX = 0x00400000; // PE resource timestamps + internal const uint PATCH_OPTION_NO_TIMESTAMP = 0x00800000; // don't store new file timestamp in patch + internal const uint PATCH_OPTION_SIGNATURE_MD5 = 0x01000000; // use MD5 instead of CRC (reserved for future support) + internal const uint PATCH_OPTION_INTERLEAVE_FILES = 0x40000000; // better support for large files (requires 5.2 or higher applyer) + internal const uint PATCH_OPTION_RESERVED1 = 0x80000000; // (used internally) + + internal const uint PATCH_OPTION_VALID_FLAGS = 0xC0FF0007; + + // + // The following flags are used with PATCH_OPTION_DATA ExtendedOptionFlags: + // + + internal const uint PATCH_TRANSFORM_PE_RESOURCE_2 = 0x00000100; // better handling of PE resources (requires 5.2 or higher applyer) + internal const uint PATCH_TRANSFORM_PE_IRELOC_2 = 0x00000200; // better handling of PE stripped relocs (requires 5.2 or higher applyer) + + // + // In addition to the standard Win32 error codes, the following error codes may + // be returned via GetLastError() when one of the patch APIs fails. + + internal const uint ERROR_PATCH_ENCODE_FAILURE = 0xC00E3101; // create + internal const uint ERROR_PATCH_INVALID_OPTIONS = 0xC00E3102; // create + internal const uint ERROR_PATCH_SAME_FILE = 0xC00E3103; // create + internal const uint ERROR_PATCH_RETAIN_RANGES_DIFFER = 0xC00E3104; // create + internal const uint ERROR_PATCH_BIGGER_THAN_COMPRESSED = 0xC00E3105; // create + internal const uint ERROR_PATCH_IMAGEHLP_FALURE = 0xC00E3106; // create + + /// + /// Delegate type that the PatchAPI calls for progress notification. + /// + /// . + /// . + /// . + /// True for success + public delegate bool PatchProgressCallback( + IntPtr context, + uint currentPosition, + uint maxPosition + ); + + /// + /// Delegate type that the PatchAPI calls for patch symbol load information. + /// + /// . + /// . + /// . + /// . + /// . + /// . + /// . + /// . + /// ??? + public delegate bool PatchSymloadCallback( + uint whichFile, // 0 for new file, 1 for first old file, etc + [MarshalAs(UnmanagedType.LPStr)] string symbolFileName, + uint symType, // see SYM_TYPE in imagehlp.h + uint symbolFileCheckSum, + uint symbolFileTimeDate, + uint imageFileCheckSum, + uint imageFileTimeDate, + IntPtr context + ); + + /// + /// Wraps PATCH_IGNORE_RANGE + /// + [StructLayout(LayoutKind.Sequential)] + internal class PatchIgnoreRange + { + public uint offsetInOldFile; + public uint lengthInBytes; + } + + /// + /// Wraps PATCH_RETAIN_RANGE + /// + [StructLayout(LayoutKind.Sequential)] + internal class PatchRetainRange + { + public uint offsetInOldFile; + public uint lengthInBytes; + public uint offsetInNewFile; + } + + /// + /// Wraps PATCH_OLD_FILE_INFO (except for the OldFile~ portion) + /// + internal class PatchOldFileInfo + { + public PatchIgnoreRange[] ignoreRange; + public PatchRetainRange[] retainRange; + } + + /// + /// Wraps PATCH_OLD_FILE_INFO_W + /// + internal class PatchOldFileInfoW : PatchOldFileInfo + { + public string oldFileName; + } + + /// + /// Wraps each PATCH_INTERLEAVE_MAP Range + /// + [SuppressMessage("Microsoft.Performance", "CA1812:AvoidUninstantiatedInternalClasses"), StructLayout(LayoutKind.Sequential)] + internal class PatchInterleaveMapRange + { + public uint oldOffset; + public uint oldLength; + public uint newLength; + } + + /// + /// Wraps PATCH_INTERLEAVE_MAP + /// + internal class PatchInterleaveMap + { + public PatchInterleaveMapRange[] ranges = null; + } + + + /// + /// Wraps PATCH_OPTION_DATA + /// + [BestFitMapping(false, ThrowOnUnmappableChar = true)] + internal class PatchOptionData + { + public PatchSymbolFlags symbolOptionFlags; // PATCH_SYMBOL_xxx flags + [MarshalAs(UnmanagedType.LPStr)] public string newFileSymbolPath; // always ANSI, never Unicode + [MarshalAs(UnmanagedType.LPStr)] public string[] oldFileSymbolPathArray; // array[ OldFileCount ] + public uint extendedOptionFlags; + public PatchSymloadCallback symLoadCallback = null; + public IntPtr symLoadContext = IntPtr.Zero; + public PatchInterleaveMap[] interleaveMapArray = null; // array[ OldFileCount ] (requires 5.2 or higher applyer) + public uint maxLzxWindowSize = 0; // limit memory requirements (requires 5.2 or higher applyer) + } + + // + // Note that PATCH_OPTION_DATA contains LPCSTR paths, and no LPCWSTR (Unicode) + // path argument is available, even when used with one of the Unicode APIs + // such as CreatePatchFileW. This is because the unlerlying system services + // for symbol file handling (IMAGEHLP.DLL) only support ANSI file/path names. + // + + // + // A note about PATCH_RETAIN_RANGE specifiers with multiple old files: + // + // Each old version file must have the same RetainRangeCount, and the same + // retain range LengthInBytes and OffsetInNewFile values in the same order. + // Only the OffsetInOldFile values can differ between old foles for retain + // ranges. + // + + // + // The following prototypes are (some of the) interfaces for creating patches from files. + // + + /// + /// Creates a new delta. + /// + /// Size of oldFileInfoArray. + /// Target file information. + /// Name of updated file. + /// Name of delta to create. + /// PATCH_OPTION_xxx. + /// Optional PATCH_OPTION_DATA structure. + /// Delegate for progress callbacks. + /// Context for progress callback delegate. + /// true if successfull, sets Marshal.GetLastWin32Error() if not. + [DllImport("mspatchc.dll", SetLastError = true, CharSet = CharSet.Unicode, ExactSpelling = true)] + [return: MarshalAs(UnmanagedType.Bool)] + internal static extern bool CreatePatchFileExW( + uint oldFileCount, // maximum 255 + [MarshalAs(UnmanagedType.CustomMarshaler, MarshalTypeRef=typeof(PatchAPIMarshaler), MarshalCookie="PATCH_OLD_FILE_INFO_W")] + PatchOldFileInfoW[] oldFileInfoArray, + string newFileName, // input file (required) + string patchFileName, // output file (required) + uint optionFlags, + [MarshalAs(UnmanagedType.CustomMarshaler, MarshalTypeRef=typeof(PatchAPIMarshaler), MarshalCookie="PATCH_OPTION_DATA")] + PatchOptionData optionData, + [MarshalAs (UnmanagedType.FunctionPtr)] + PatchProgressCallback progressCallback, + IntPtr context + ); + + /// + /// Extracts delta header from delta. + /// + /// Name of delta file. + /// Name of file to create with delta header. + /// true if successfull, sets Marshal.GetLastWin32Error() if not. + [DllImport("mspatchc.dll", SetLastError = true, CharSet = CharSet.Unicode, ExactSpelling = true)] + [return: MarshalAs(UnmanagedType.Bool)] + internal static extern bool ExtractPatchHeaderToFileW( + string patchFileName, // input file + string patchHeaderFileName // output file + ); + + // TODO: Add rest of APIs to enable custom binders to perform more exhaustive checks + + /// + /// Marshals arguments for the CreatePatch~ APIs + /// + [SuppressMessage("Microsoft.Performance", "CA1812:AvoidUninstantiatedInternalClasses")] + internal class PatchAPIMarshaler : ICustomMarshaler + { + internal static ICustomMarshaler GetInstance(string cookie) + { + return new PatchAPIMarshaler(cookie); + } + + private enum MarshalType + { + PATCH_OPTION_DATA, + PATCH_OLD_FILE_INFO_W + }; + + private readonly PatchAPIMarshaler.MarshalType marshalType; + + private PatchAPIMarshaler(string cookie) + { + this.marshalType = (PatchAPIMarshaler.MarshalType)Enum.Parse(typeof(PatchAPIMarshaler.MarshalType), cookie); + } + + // + // Summary: + // Returns the size of the native data to be marshaled. + // + // Returns: + // The size in bytes of the native data. + public int GetNativeDataSize() + { + return Marshal.SizeOf(typeof(IntPtr)); + } + + // + // Summary: + // Performs necessary cleanup of the managed data when it is no longer needed. + // + // Parameters: + // ManagedObj: + // The managed object to be destroyed. + public void CleanUpManagedData(object ManagedObj) + { + } + + // + // Summary: + // Performs necessary cleanup of the unmanaged data when it is no longer needed. + // + // Parameters: + // pNativeData: + // A pointer to the unmanaged data to be destroyed. + public void CleanUpNativeData(IntPtr pNativeData) + { + if (IntPtr.Zero == pNativeData) + { + return; + } + + switch (this.marshalType) + { + case PatchAPIMarshaler.MarshalType.PATCH_OPTION_DATA: + this.CleanUpPOD(pNativeData); + break; + default: + this.CleanUpPOFI_A(pNativeData); + break; + } + } + + // + // Summary: + // Converts the managed data to unmanaged data. + // + // Parameters: + // ManagedObj: + // The managed object to be converted. + // + // Returns: + // Returns the COM view of the managed object. + public IntPtr MarshalManagedToNative(object ManagedObj) + { + if (null == ManagedObj) + { + return IntPtr.Zero; + } + + switch (this.marshalType) + { + case PatchAPIMarshaler.MarshalType.PATCH_OPTION_DATA: + return this.MarshalPOD(ManagedObj as PatchOptionData); + case PatchAPIMarshaler.MarshalType.PATCH_OLD_FILE_INFO_W: + return this.MarshalPOFIW_A(ManagedObj as PatchOldFileInfoW[]); + default: + throw new InvalidOperationException(); + } + } + + + // + // Summary: + // Converts the unmanaged data to managed data. + // + // Parameters: + // pNativeData: + // A pointer to the unmanaged data to be wrapped. + // + // Returns: + // Returns the managed view of the COM data. + public object MarshalNativeToManaged(IntPtr pNativeData) + { + return null; + } + + // Implementation ************************************************* + + // PATCH_OPTION_DATA offsets + private static readonly int symbolOptionFlagsOffset = Marshal.SizeOf(typeof(Int32)); + private static readonly int newFileSymbolPathOffset = 2 * Marshal.SizeOf(typeof(Int32)); + private static readonly int oldFileSymbolPathArrayOffset = 2 * Marshal.SizeOf(typeof(Int32)) + Marshal.SizeOf(typeof(IntPtr)); + private static readonly int extendedOptionFlagsOffset = 2 * Marshal.SizeOf(typeof(Int32)) + 2 * Marshal.SizeOf(typeof(IntPtr)); + private static readonly int symLoadCallbackOffset = 3 * Marshal.SizeOf(typeof(Int32)) + 2 * Marshal.SizeOf(typeof(IntPtr)); + private static readonly int symLoadContextOffset = 3 * Marshal.SizeOf(typeof(Int32)) + 3 * Marshal.SizeOf(typeof(IntPtr)); + private static readonly int interleaveMapArrayOffset = 3 * Marshal.SizeOf(typeof(Int32)) + 4 * Marshal.SizeOf(typeof(IntPtr)); + private static readonly int maxLzxWindowSizeOffset = 3 * Marshal.SizeOf(typeof(Int32)) + 5 * Marshal.SizeOf(typeof(IntPtr)); + private static readonly int patchOptionDataSize = 4 * Marshal.SizeOf(typeof(Int32)) + 5 * Marshal.SizeOf(typeof(IntPtr)); + + // PATCH_OLD_FILE_INFO offsets + private static readonly int oldFileOffset = Marshal.SizeOf(typeof(Int32)); + private static readonly int ignoreRangeCountOffset = Marshal.SizeOf(typeof(Int32)) + Marshal.SizeOf(typeof(IntPtr)); + private static readonly int ignoreRangeArrayOffset = 2 * Marshal.SizeOf(typeof(Int32)) + Marshal.SizeOf(typeof(IntPtr)); + private static readonly int retainRangeCountOffset = 2 * Marshal.SizeOf(typeof(Int32)) + 2 * Marshal.SizeOf(typeof(IntPtr)); + private static readonly int retainRangeArrayOffset = 3 * Marshal.SizeOf(typeof(Int32)) + 2 * Marshal.SizeOf(typeof(IntPtr)); + private static readonly int patchOldFileInfoSize = 3 * Marshal.SizeOf(typeof(Int32)) + 3 * Marshal.SizeOf(typeof(IntPtr)); + + // Methods and data used to preserve data needed for cleanup + + // This dictionary holds the quantity of items internal to each native structure that will need to be freed (the OldFileCount) + private static readonly Dictionary OldFileCounts = new Dictionary(); + private static readonly object OldFileCountsLock = new object(); + + private IntPtr CreateMainStruct(int oldFileCount) + { + int nativeSize; + switch (this.marshalType) + { + case PatchAPIMarshaler.MarshalType.PATCH_OPTION_DATA: + nativeSize = patchOptionDataSize; + break; + case PatchAPIMarshaler.MarshalType.PATCH_OLD_FILE_INFO_W: + nativeSize = oldFileCount * patchOldFileInfoSize; + break; + default: + throw new InvalidOperationException(); + } + + var native = Marshal.AllocCoTaskMem(nativeSize); + + lock (PatchAPIMarshaler.OldFileCountsLock) + { + PatchAPIMarshaler.OldFileCounts.Add(native, oldFileCount); + } + + return native; + } + + private static void ReleaseMainStruct(IntPtr native) + { + lock (PatchAPIMarshaler.OldFileCountsLock) + { + PatchAPIMarshaler.OldFileCounts.Remove(native); + } + Marshal.FreeCoTaskMem(native); + } + + private static int GetOldFileCount(IntPtr native) + { + lock (PatchAPIMarshaler.OldFileCountsLock) + { + return PatchAPIMarshaler.OldFileCounts[native]; + } + } + + // Helper methods + + private static IntPtr OptionalAnsiString(string managed) + { + return (null == managed) ? IntPtr.Zero : Marshal.StringToCoTaskMemAnsi(managed); + } + + private static IntPtr OptionalUnicodeString(string managed) + { + return (null == managed) ? IntPtr.Zero : Marshal.StringToCoTaskMemUni(managed); + } + + // string array must be of the same length as the number of old files + private static IntPtr CreateArrayOfStringA(string[] managed) + { + if (null == managed) + { + return IntPtr.Zero; + } + + var size = managed.Length * Marshal.SizeOf(typeof(IntPtr)); + var native = Marshal.AllocCoTaskMem(size); + + for (var i = 0; i < managed.Length; ++i) + { + Marshal.WriteIntPtr(native, i * Marshal.SizeOf(typeof(IntPtr)), OptionalAnsiString(managed[i])); + } + + return native; + } + + // string array must be of the same length as the number of old files + private static IntPtr CreateArrayOfStringW(string[] managed) + { + if (null == managed) + { + return IntPtr.Zero; + } + + var size = managed.Length * Marshal.SizeOf(typeof(IntPtr)); + var native = Marshal.AllocCoTaskMem(size); + + for (var i = 0; i < managed.Length; ++i) + { + Marshal.WriteIntPtr(native, i * Marshal.SizeOf(typeof(IntPtr)), OptionalUnicodeString(managed[i])); + } + + return native; + } + + private static IntPtr CreateInterleaveMapRange(PatchInterleaveMap managed) + { + if (null == managed) + { + return IntPtr.Zero; + } + + if (null == managed.ranges) + { + return IntPtr.Zero; + } + + if (0 == managed.ranges.Length) + { + return IntPtr.Zero; + } + + var native = Marshal.AllocCoTaskMem(Marshal.SizeOf(typeof(UInt32)) + + managed.ranges.Length * (Marshal.SizeOf(typeof(PatchInterleaveMap)))); + WriteUInt32(native, (uint)managed.ranges.Length); + + for (var i = 0; i < managed.ranges.Length; ++i) + { + Marshal.StructureToPtr(managed.ranges[i], (IntPtr)((Int64)native + i * Marshal.SizeOf(typeof(PatchInterleaveMap))), false); + } + return native; + } + + private static IntPtr CreateInterleaveMap(PatchInterleaveMap[] managed) + { + if (null == managed) + { + return IntPtr.Zero; + } + + var native = Marshal.AllocCoTaskMem(managed.Length * Marshal.SizeOf(typeof(IntPtr))); + + for (var i = 0; i < managed.Length; ++i) + { + Marshal.WriteIntPtr(native, i * Marshal.SizeOf(typeof(IntPtr)), CreateInterleaveMapRange(managed[i])); + } + + return native; + } + + private static void WriteUInt32(IntPtr native, uint data) + { + Marshal.WriteInt32(native, unchecked((int)data)); + } + + private static void WriteUInt32(IntPtr native, int offset, uint data) + { + Marshal.WriteInt32(native, offset, unchecked((int)data)); + } + + // Marshal operations + + private IntPtr MarshalPOD(PatchOptionData managed) + { + if (null == managed) + { + throw new ArgumentNullException("managed"); + } + + var native = this.CreateMainStruct(managed.oldFileSymbolPathArray.Length); + Marshal.WriteInt32(native, patchOptionDataSize); // SizeOfThisStruct + WriteUInt32(native, symbolOptionFlagsOffset, (uint)managed.symbolOptionFlags); + Marshal.WriteIntPtr(native, newFileSymbolPathOffset, PatchAPIMarshaler.OptionalAnsiString(managed.newFileSymbolPath)); + Marshal.WriteIntPtr(native, oldFileSymbolPathArrayOffset, PatchAPIMarshaler.CreateArrayOfStringA(managed.oldFileSymbolPathArray)); + WriteUInt32(native, extendedOptionFlagsOffset, managed.extendedOptionFlags); + + // GetFunctionPointerForDelegate() throws an ArgumentNullException if the delegate is null. + if (null == managed.symLoadCallback) + { + Marshal.WriteIntPtr(native, symLoadCallbackOffset, IntPtr.Zero); + } + else + { + Marshal.WriteIntPtr(native, symLoadCallbackOffset, Marshal.GetFunctionPointerForDelegate(managed.symLoadCallback)); + } + + Marshal.WriteIntPtr(native, symLoadContextOffset, managed.symLoadContext); + Marshal.WriteIntPtr(native, interleaveMapArrayOffset, PatchAPIMarshaler.CreateInterleaveMap(managed.interleaveMapArray)); + WriteUInt32(native, maxLzxWindowSizeOffset, managed.maxLzxWindowSize); + return native; + } + + private IntPtr MarshalPOFIW_A(PatchOldFileInfoW[] managed) + { + if (null == managed) + { + throw new ArgumentNullException("managed"); + } + + if (0 == managed.Length) + { + return IntPtr.Zero; + } + + var native = this.CreateMainStruct(managed.Length); + + for (var i = 0; i < managed.Length; ++i) + { + PatchAPIMarshaler.MarshalPOFIW(managed[i], (IntPtr)((Int64)native + i * patchOldFileInfoSize)); + } + + return native; + } + + private static void MarshalPOFIW(PatchOldFileInfoW managed, IntPtr native) + { + PatchAPIMarshaler.MarshalPOFI(managed, native); + Marshal.WriteIntPtr(native, oldFileOffset, PatchAPIMarshaler.OptionalUnicodeString(managed.oldFileName)); // OldFileName + } + + private static void MarshalPOFI(PatchOldFileInfo managed, IntPtr native) + { + Marshal.WriteInt32(native, patchOldFileInfoSize); // SizeOfThisStruct + WriteUInt32(native, ignoreRangeCountOffset, + (null == managed.ignoreRange) ? 0 : (uint)managed.ignoreRange.Length); // IgnoreRangeCount // maximum 255 + Marshal.WriteIntPtr(native, ignoreRangeArrayOffset, MarshalPIRArray(managed.ignoreRange)); // IgnoreRangeArray + WriteUInt32(native, retainRangeCountOffset, + (null == managed.retainRange) ? 0 : (uint)managed.retainRange.Length); // RetainRangeCount // maximum 255 + Marshal.WriteIntPtr(native, retainRangeArrayOffset, MarshalPRRArray(managed.retainRange)); // RetainRangeArray + } + + private static IntPtr MarshalPIRArray(PatchIgnoreRange[] array) + { + if (null == array) + { + return IntPtr.Zero; + } + + if (0 == array.Length) + { + return IntPtr.Zero; + } + + var native = Marshal.AllocCoTaskMem(array.Length * Marshal.SizeOf(typeof(PatchIgnoreRange))); + + for (var i = 0; i < array.Length; ++i) + { + Marshal.StructureToPtr(array[i], (IntPtr)((Int64)native + (i * Marshal.SizeOf(typeof(PatchIgnoreRange)))), false); + } + + return native; + } + + private static IntPtr MarshalPRRArray(PatchRetainRange[] array) + { + if (null == array) + { + return IntPtr.Zero; + } + + if (0 == array.Length) + { + return IntPtr.Zero; + } + + var native = Marshal.AllocCoTaskMem(array.Length * Marshal.SizeOf(typeof(PatchRetainRange))); + + for (var i = 0; i < array.Length; ++i) + { + Marshal.StructureToPtr(array[i], (IntPtr)((Int64)native + (i * Marshal.SizeOf(typeof(PatchRetainRange)))), false); + } + + return native; + } + + // CleanUp operations + + private void CleanUpPOD(IntPtr native) + { + Marshal.FreeCoTaskMem(Marshal.ReadIntPtr(native, newFileSymbolPathOffset)); + + if (IntPtr.Zero != Marshal.ReadIntPtr(native, oldFileSymbolPathArrayOffset)) + { + for (var i = 0; i < GetOldFileCount(native); ++i) + { + Marshal.FreeCoTaskMem( + Marshal.ReadIntPtr( + Marshal.ReadIntPtr(native, oldFileSymbolPathArrayOffset), + i * Marshal.SizeOf(typeof(IntPtr)))); + } + + Marshal.FreeCoTaskMem(Marshal.ReadIntPtr(native, oldFileSymbolPathArrayOffset)); + } + + if (IntPtr.Zero != Marshal.ReadIntPtr(native, interleaveMapArrayOffset)) + { + for (var i = 0; i < GetOldFileCount(native); ++i) + { + Marshal.FreeCoTaskMem( + Marshal.ReadIntPtr( + Marshal.ReadIntPtr(native, interleaveMapArrayOffset), + i * Marshal.SizeOf(typeof(IntPtr)))); + } + + Marshal.FreeCoTaskMem(Marshal.ReadIntPtr(native, interleaveMapArrayOffset)); + } + + PatchAPIMarshaler.ReleaseMainStruct(native); + } + + private void CleanUpPOFI_A(IntPtr native) + { + for (var i = 0; i < GetOldFileCount(native); ++i) + { + PatchAPIMarshaler.CleanUpPOFI((IntPtr)((Int64)native + i * patchOldFileInfoSize)); + } + + PatchAPIMarshaler.ReleaseMainStruct(native); + } + + private static void CleanUpPOFI(IntPtr native) + { + if (IntPtr.Zero != Marshal.ReadIntPtr(native, oldFileOffset)) + { + Marshal.FreeCoTaskMem(Marshal.ReadIntPtr(native, oldFileOffset)); + } + + PatchAPIMarshaler.CleanUpPOFIH(native); + } + + private static void CleanUpPOFIH(IntPtr native) + { + if (IntPtr.Zero != Marshal.ReadIntPtr(native, ignoreRangeArrayOffset)) + { + Marshal.FreeCoTaskMem(Marshal.ReadIntPtr(native, ignoreRangeArrayOffset)); + } + + if (IntPtr.Zero != Marshal.ReadIntPtr(native, retainRangeArrayOffset)) + { + Marshal.FreeCoTaskMem(Marshal.ReadIntPtr(native, retainRangeArrayOffset)); + } + } + } + } +} diff --git a/src/wix/WixToolset.Core.Native/ValidationMessage.cs b/src/wix/WixToolset.Core.Native/ValidationMessage.cs new file mode 100644 index 00000000..d7137326 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/ValidationMessage.cs @@ -0,0 +1,47 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native +{ + using System.Collections.Generic; + + /// + /// Message from ICE + /// + public class ValidationMessage + { + /// + /// Name of the ICE providing the message. + /// + public string IceName { get; set; } + + /// + /// Validation type. + /// + public ValidationMessageType Type { get; set; } + + /// + /// Message text. + /// + public string Description { get; set; } + + /// + /// Optional help URL for the message. + /// + public string HelpUrl { get; set; } + + /// + /// Optional table causing the message. + /// + public string Table { get; set; } + + /// + /// Optional column causing the message. + /// + public string Column { get; set; } + + /// + /// Optional primary keys causing the message. + /// + public IEnumerable PrimaryKeys { get; set; } + } +} diff --git a/src/wix/WixToolset.Core.Native/ValidationMessageType.cs b/src/wix/WixToolset.Core.Native/ValidationMessageType.cs new file mode 100644 index 00000000..98635294 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/ValidationMessageType.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.Core.Native +{ + /// + /// Validation message type. + /// + public enum ValidationMessageType + { + /// + /// Failure message reporting the failure of the ICE custom action. + /// + InternalFailure = 0, + + /// + /// Error message reporting database authoring that case incorrect behavior. + /// + Error = 1, + + /// + /// Warning message reporting database authoring that causes incorrect behavior in certain cases. + /// Warnings can also report unexpected side-effects of database authoring. + /// + Warning = 2, + + /// + /// Informational message. + /// + Info = 3, + }; +} diff --git a/src/wix/WixToolset.Core.Native/WindowsInstallerValidator.cs b/src/wix/WixToolset.Core.Native/WindowsInstallerValidator.cs new file mode 100644 index 00000000..9f4b26a3 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/WindowsInstallerValidator.cs @@ -0,0 +1,427 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native +{ + using System; + using System.Collections.Generic; + using System.ComponentModel; + using System.IO; + using System.Linq; + using System.Threading; + using WixToolset.Core.Native.Msi; + using WixToolset.Data; + + /// + /// Windows installer validation implementation. + /// + public class WindowsInstallerValidator + { + private const string CubesFolder = "cubes"; + + /// + /// Creates a new Windows Installer validator. + /// + /// Callback interface to handle messages. + /// Database to validate. + /// Set of CUBe files to merge. + /// ICEs to execute. + /// Suppressed ICEs. + public WindowsInstallerValidator(IWindowsInstallerValidatorCallback callback, string databasePath, IEnumerable cubeFiles, IEnumerable ices, IEnumerable suppressedIces) + { + this.Callback = callback; + this.DatabasePath = databasePath; + this.CubeFiles = cubeFiles; + this.Ices = new SortedSet(ices); + this.SuppressedIces = new SortedSet(suppressedIces); + } + + private IWindowsInstallerValidatorCallback Callback { get; } + + private string DatabasePath { get; } + + private IEnumerable CubeFiles { get; } + + private SortedSet Ices { get; } + + private SortedSet SuppressedIces { get; } + + private bool ValidationSessionInProgress { get; set; } + + private string CurrentIce { get; set; } + + /// + /// Execute the validations. + /// + public void Execute() + { + using (var mutex = new Mutex(false, "WixValidator")) + { + try + { + if (!mutex.WaitOne(0)) + { + this.Callback.ValidationBlocked(); + mutex.WaitOne(); + } + } + catch (AbandonedMutexException) + { + // Another validation process was probably killed, we own the mutex now. + } + + try + { + this.RunValidations(); + } + finally + { + mutex.ReleaseMutex(); + } + } + } + + private void RunValidations() + { + var previousUILevel = (int)InstallUILevels.Basic; + var previousHwnd = IntPtr.Zero; + InstallUIHandler previousUIHandler = null; + + try + { + using (var database = new Database(this.DatabasePath, OpenDatabase.Direct)) + { + var propertyTableExists = database.TableExists("Property"); + string productCode = null; + + // Remove the product code from the database before opening a session to prevent opening an installed product. + if (propertyTableExists) + { + using (var view = database.OpenExecuteView("SELECT `Value` FROM `Property` WHERE Property = 'ProductCode'")) + { + using (var record = view.Fetch()) + { + if (null != record) + { + productCode = record.GetString(1); + + using (var dropProductCodeView = database.OpenExecuteView("DELETE FROM `Property` WHERE `Property` = 'ProductCode'")) + { + } + } + } + } + } + + // Merge in the cube databases. + foreach (var cubeFile in this.CubeFiles) + { + var findCubeFile = typeof(WindowsInstallerValidator).Assembly.FindFileRelativeToAssembly(Path.Combine(CubesFolder, cubeFile), searchNativeDllDirectories: false); + + if (!findCubeFile.Found) + { + throw new WixException(ErrorMessages.CubeFileNotFound(findCubeFile.Path)); + } + + try + { + using (var cubeDatabase = new Database(findCubeFile.Path, OpenDatabase.ReadOnly)) + { + try + { + database.Merge(cubeDatabase, "MergeConflicts"); + } + catch + { + // ignore merge errors since they are expected in the _Validation table + } + } + } + catch (Win32Exception e) + { + if (0x6E == e.NativeErrorCode) // ERROR_OPEN_FAILED + { + throw new WixException(ErrorMessages.CubeFileNotFound(findCubeFile.Path)); + } + + throw; + } + } + + // Commit the database before proceeding to ensure the streams don't get confused. + database.Commit(); + + // The property table may have been added to the database from a cub database without the proper validation rows. + if (!propertyTableExists) + { + using (var view = database.OpenExecuteView("DROP table `Property`")) + { + } + } + + // Get all the action names for ICEs which have not been suppressed. + var actions = new List(); + using (var view = database.OpenExecuteView("SELECT `Action` FROM `_ICESequence` ORDER BY `Sequence`")) + { + foreach (var record in view.Records) + { + var action = record.GetString(1); + + if (!this.SuppressedIces.Contains(action) && this.Ices.Contains(action)) + { + actions.Add(action); + } + } + } + + // Disable the internal UI handler and set an external UI handler. + previousUILevel = Installer.SetInternalUI((int)InstallUILevels.None, ref previousHwnd); + previousUIHandler = Installer.SetExternalUI(this.ValidationUIHandler, (int)InstallLogModes.Error | (int)InstallLogModes.Warning | (int)InstallLogModes.User, IntPtr.Zero); + + // Create a session for running the ICEs. + this.ValidationSessionInProgress = true; + + using (var session = new Session(database)) + { + // Add the product code back into the database. + if (null != productCode) + { + // Some CUBs erroneously have a ProductCode property, so delete it if we just picked one up. + using (var dropProductCodeView = database.OpenExecuteView("DELETE FROM `Property` WHERE `Property` = 'ProductCode'")) + { + } + + using (var view = database.OpenExecuteView($"INSERT INTO `Property` (`Property`, `Value`) VALUES ('ProductCode', '{productCode}')")) + { + } + } + + foreach (var action in actions) + { + this.CurrentIce = action; + + try + { + session.DoAction(action); + } + catch (Win32Exception e) + { + if (!this.Callback.EncounteredError) + { + throw e; + } + } + + this.CurrentIce = null; + } + + // Mark the validation session complete so we ignore any messages that MSI may fire + // during session clean-up. + this.ValidationSessionInProgress = false; + } + } + } + catch (Win32Exception e) + { + // Avoid displaying errors twice since one may have already occurred in the UI handler. + if (!this.Callback.EncounteredError) + { + if (0x6E == e.NativeErrorCode) // ERROR_OPEN_FAILED + { + // The database path is not passed to this exception since inside wix.exe + // this would be the temporary copy and there would be no final output becasue + // this error occured; and during standalone validation they should know the path + // passed in. + throw new WixException(ErrorMessages.ValidationFailedToOpenDatabase()); + } + else if (0x64D == e.NativeErrorCode) + { + throw new WixException(ErrorMessages.ValidationFailedDueToLowMsiEngine()); + } + else if (0x654 == e.NativeErrorCode) + { + throw new WixException(ErrorMessages.ValidationFailedDueToInvalidPackage()); + } + else if (0x658 == e.NativeErrorCode) + { + throw new WixException(ErrorMessages.ValidationFailedDueToMultilanguageMergeModule()); + } + else if (0x659 == e.NativeErrorCode) + { + throw new WixException(WarningMessages.ValidationFailedDueToSystemPolicy()); + } + else + { + var msg = String.IsNullOrEmpty(this.CurrentIce) ? e.Message : $"Action - '{this.CurrentIce}' {e.Message}"; + + throw new WixException(ErrorMessages.Win32Exception(e.NativeErrorCode, msg)); + } + } + } + finally + { + this.ValidationSessionInProgress = false; + + Installer.SetExternalUI(previousUIHandler, 0, IntPtr.Zero); + Installer.SetInternalUI(previousUILevel, ref previousHwnd); + } + } + + /// + /// The validation external UI handler. + /// + /// Pointer to an application context. + /// This parameter can be used for error checking. + /// Specifies a combination of one message box style, + /// one message box icon type, one default button, and one installation message type. + /// Specifies the message text. + /// -1 for an error, 0 if no action was taken, 1 if OK, 3 to abort. + private int ValidationUIHandler(IntPtr context, uint messageType, string message) + { + var continueValidation = true; + + // If we're getting messges during the validation session, log them. + // Otherwise, ignore the messages. + if (!this.ValidationSessionInProgress) + { + var parsedMessage = ParseValidationMessage(message, this.CurrentIce); + + continueValidation = this.Callback.ValidationMessage(parsedMessage); + } + + return continueValidation ? 1 : 3; + } + + /// + /// Parses a message from the Validator. + /// + /// A of tab-delmited tokens + /// in the validation message. + /// The name of the action to which the message + /// belongs. + /// The message cannot be null. + /// + /// The message does not contain four (4) + /// or more tab-delimited tokens. + /// + /// a tab-delimited set of tokens, + /// formatted according to Windows Installer guidelines for ICE + /// message. The following table lists what each token by index + /// should mean. + /// a name that represents the ICE + /// action that was executed (e.g. 'ICE08'). + /// + /// + /// Index + /// Description + /// + /// + /// 0 + /// Name of the ICE. + /// + /// + /// 1 + /// Message type. See the following list. + /// + /// + /// 2 + /// Detailed description. + /// + /// + /// 3 + /// Help URL or location. + /// + /// + /// 4 + /// Table name. + /// + /// + /// 5 + /// Column name. + /// + /// + /// 6 + /// This and remaining fields are primary keys + /// to identify a row. + /// + /// + /// The message types are one of the following value. + /// + /// + /// Value + /// Message Type + /// + /// + /// 0 + /// Failure message reporting the failure of the + /// ICE custom action. + /// + /// + /// 1 + /// Error message reporting database authoring that + /// case incorrect behavior. + /// + /// + /// 2 + /// Warning message reporting database authoring that + /// causes incorrect behavior in certain cases. Warnings can also + /// report unexpected side-effects of database authoring. + /// + /// + /// + /// 3 + /// Informational message. + /// + /// + /// + private static ValidationMessage ParseValidationMessage(string message, string currentIce) + { + if (message == null) + { + throw new ArgumentNullException(nameof(message)); + } + + var messageParts = message.Split('\t'); + if (messageParts.Length < 3) + { + if (null == currentIce) + { + throw new WixException(ErrorMessages.UnexpectedExternalUIMessage(message)); + } + else + { + throw new WixException(ErrorMessages.UnexpectedExternalUIMessage(message, currentIce)); + } + } + + var type = ParseValidationMessageType(messageParts[1]); + + return new ValidationMessage + { + IceName = messageParts[0], + Type = type, + Description = messageParts[2], + HelpUrl = messageParts.Length > 3 ? messageParts[3] : null, + Table = messageParts.Length > 4 ? messageParts[4] : null, + Column = messageParts.Length > 5 ? messageParts[4] : null, + PrimaryKeys = messageParts.Length > 6 ? messageParts.Skip(6).ToArray() : null + }; + } + + private static ValidationMessageType ParseValidationMessageType(string type) + { + switch (type) + { + case "0": + return ValidationMessageType.InternalFailure; + case "1": + return ValidationMessageType.Error; + case "2": + return ValidationMessageType.Warning; + case "3": + return ValidationMessageType.Info; + default: + throw new WixException(ErrorMessages.InvalidValidatorMessageType(type)); + } + } + } +} diff --git a/src/wix/WixToolset.Core.Native/WixNativeExe.cs b/src/wix/WixToolset.Core.Native/WixNativeExe.cs new file mode 100644 index 00000000..fb41b2f2 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/WixNativeExe.cs @@ -0,0 +1,125 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native +{ + using System; + using System.Collections.Generic; + using System.ComponentModel; + using System.Diagnostics; + using System.IO; + + internal class WixNativeExe + { + private const string WixNativeExeFileName = "wixnative.exe"; + private static string PathToWixNativeExe; + + private readonly string commandLine; + private readonly List stdinLines = new List(); + + public WixNativeExe(params object[] args) + { + this.commandLine = String.Join(" ", QuoteArgumentsAsNecesary(args)); + } + + public void AddStdinLine(string line) + { + this.stdinLines.Add(line); + } + + public void AddStdinLines(IEnumerable lines) + { + this.stdinLines.AddRange(lines); + } + + public IEnumerable Run() + { + EnsurePathToWixNativeExeSet(); + + var wixNativeInfo = new ProcessStartInfo(PathToWixNativeExe, this.commandLine) + { + RedirectStandardInput = true, + RedirectStandardOutput = true, + CreateNoWindow = true, + ErrorDialog = false, + UseShellExecute = false + }; + + var stdoutLines = new List(); + + using (var process = Process.Start(wixNativeInfo)) + { + process.OutputDataReceived += (s, a) => stdoutLines.Add(a.Data); + process.BeginOutputReadLine(); + + if (this.stdinLines.Count > 0) + { + foreach (var line in this.stdinLines) + { + process.StandardInput.WriteLine(line); + } + + // Trailing blank line indicates stdin complete. + process.StandardInput.WriteLine(); + } + + // If the process successfully exits documentation says we need to wait again + // without a timeout to ensure that all of the redirected output is captured. + // + process.WaitForExit(); + + if (process.ExitCode != 0) + { + throw new Win32Exception(process.ExitCode); + } + } + + return stdoutLines; + } + + private static void EnsurePathToWixNativeExeSet() + { + if (String.IsNullOrEmpty(PathToWixNativeExe)) + { + var result = typeof(WixNativeExe).Assembly.FindFileRelativeToAssembly(WixNativeExeFileName, searchNativeDllDirectories: true); + + if (!result.Found) + { + throw new PlatformNotSupportedException( + $"Could not find platform specific '{WixNativeExeFileName}'", + new FileNotFoundException($"Could not find internal piece of WiX Toolset from: {result.PossiblePaths}", WixNativeExeFileName)); + } + + PathToWixNativeExe = result.Path; + } + } + + private static IEnumerable QuoteArgumentsAsNecesary(object[] args) + { + foreach (var arg in args) + { + if (arg is string str) + { + if (String.IsNullOrEmpty(str)) + { + } + else if (str.Contains(" ") && !str.StartsWith("\"")) + { + yield return $"\"{str}\""; + } + else + { + yield return str; + } + } + else if (arg is int i) + { + yield return i.ToString(); + } + else + { + throw new ArgumentException(nameof(args)); + } + } + } + } +} diff --git a/src/wix/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/wix/WixToolset.Core.Native/WixToolset.Core.Native.csproj new file mode 100644 index 00000000..fea15922 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -0,0 +1,49 @@ + + + + + + + netstandard2.0 + embedded + WiX Toolset Native Processing + true + + NU5128 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/wix/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/wix/WixToolset.Core.Native/WixToolset.Core.Native.nuspec new file mode 100644 index 00000000..3091ccd5 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -0,0 +1,41 @@ + + + + $id$ + $version$ + $title$ + $description$ + $authors$ + MS-RL + false + $copyright$ + $projectUrl$ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/wix/WixToolset.Core.Native/cubes/darice.cub b/src/wix/WixToolset.Core.Native/cubes/darice.cub new file mode 100644 index 00000000..4292fede Binary files /dev/null and b/src/wix/WixToolset.Core.Native/cubes/darice.cub differ diff --git a/src/wix/WixToolset.Core.Native/cubes/mergemod.cub b/src/wix/WixToolset.Core.Native/cubes/mergemod.cub new file mode 100644 index 00000000..def6dd1a Binary files /dev/null and b/src/wix/WixToolset.Core.Native/cubes/mergemod.cub differ diff --git a/src/wix/WixToolset.Core.Native/targets/WixToolset.Core.Native.targets b/src/wix/WixToolset.Core.Native/targets/WixToolset.Core.Native.targets new file mode 100644 index 00000000..aafbd405 --- /dev/null +++ b/src/wix/WixToolset.Core.Native/targets/WixToolset.Core.Native.targets @@ -0,0 +1,9 @@ + + + + + PreserveNewest + cubes\%(RecursiveDir)%(Filename)%(Extension) + + + diff --git a/src/wix/appveyor-CoreNative.cmd b/src/wix/appveyor-CoreNative.cmd new file mode 100644 index 00000000..d9691a42 --- /dev/null +++ b/src/wix/appveyor-CoreNative.cmd @@ -0,0 +1,19 @@ +@setlocal +@pushd %~dp0 +@set _C=Release +@if /i "%1"=="debug" set _C=Debug + +:: Restore +msbuild -p:Configuration=%_C% -t:Restore || exit /b + +:: Build +msbuild -p:Configuration=%_C% src\test\WixToolsetTest.Core.Native\WixToolsetTest.Core.Native.csproj || exit /b + +:: Test +dotnet test -c %_C% --no-build src\test\WixToolsetTest.Core.Native\WixToolsetTest.Core.Native.csproj || exit /b + +:: Pack +msbuild -p:Configuration=%_C% -p:NoBuild=true -t:Pack src\WixToolset.Core.Native\WixToolset.Core.Native.csproj || exit /b + +@popd +@endlocal diff --git a/src/wix/appveyor-CoreNative.yml b/src/wix/appveyor-CoreNative.yml new file mode 100644 index 00000000..364569cf --- /dev/null +++ b/src/wix/appveyor-CoreNative.yml @@ -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. +# +# Do NOT modify this file. Update the canonical version in Home\repo-template\src\appveyor.yml +# then update all of the repos. + +branches: + only: + - master + - develop + +image: Visual Studio 2019 + +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: + - appveyor.cmd + +test: off + +pull_requests: + do_not_increment_build_number: true + +nuget: + disable_publish_on_pr: true + +skip_branch_with_pr: true +skip_tags: true + +artifacts: +- path: build\Release\**\*.nupkg + name: nuget +- path: build\Release\**\*.snupkg + name: snupkg + +notifications: +- provider: Slack + incoming_webhook: + secure: p5xuu+4x2JHfwGDMDe5KcG1k7gZxqYc4jWVwvyNZv5cvkubPD2waJs5yXMAXZNN7Z63/3PWHb7q4KoY/99AjauYa1nZ4c5qYqRPFRBKTHfA= diff --git a/src/wix/nuget-CoreNative.config b/src/wix/nuget-CoreNative.config new file mode 100644 index 00000000..18404582 --- /dev/null +++ b/src/wix/nuget-CoreNative.config @@ -0,0 +1,9 @@ + + + + + + + + + \ No newline at end of file diff --git a/src/wix/test/WixToolsetTest.Core.Native/CabinetFixture.cs b/src/wix/test/WixToolsetTest.Core.Native/CabinetFixture.cs new file mode 100644 index 00000000..2e43dce4 --- /dev/null +++ b/src/wix/test/WixToolsetTest.Core.Native/CabinetFixture.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 WixToolsetTest.CoreNative +{ + using System.IO; + using System.Linq; + using WixToolset.Core.Native; + using WixToolsetTest.CoreNative.Utility; + using WixToolset.Data; + using Xunit; + + public class CabinetFixture + { + [Fact] + public void CanCreateSingleFileCabinet() + { + using (var fs = new DisposableFileSystem()) + { + var intermediateFolder = fs.GetFolder(true); + var cabPath = Path.Combine(intermediateFolder, "testout.cab"); + + var files = new[] { new CabinetCompressFile(TestData.Get(@"TestData\test.txt"), "test.txt") }; + + var cabinet = new Cabinet(cabPath); + cabinet.Compress(files, CompressionLevel.Low); + + Assert.True(File.Exists(cabPath)); + } + } + + [Fact] + public void CanEnumerateSingleFileCabinet() + { + var cabinetPath = TestData.Get(@"TestData\test.cab"); + + var cabinet = new Cabinet(cabinetPath); + var files = cabinet.Enumerate(); + + var file = files.Single(); + Assert.Equal("test.txt", file.FileId); + Assert.Equal(17, file.Size); + + Assert.Equal(19259, file.Date); + Assert.Equal(47731, file.Time); + // TODO: This doesn't seem to always pass, not clear why but it'd be good to understand one day. + // Assert.True(file.SameAsDateTime(new DateTime(2017, 9, 28, 0, 19, 38))); + } + + [Fact] + public void IntegrationTest() + { + using (var fs = new DisposableFileSystem()) + { + var intermediateFolder = fs.GetFolder(true); + var cabinetPath = Path.Combine(intermediateFolder, "testout.cab"); + var extractFolder = fs.GetFolder(true); + + // Compress. + { + var files = new[] { + new CabinetCompressFile(TestData.Get(@"TestData\test.txt"), "test1.txt"), + new CabinetCompressFile(TestData.Get(@"TestData\test.txt"), "test2.txt"), + }; + + var cabinet = new Cabinet(cabinetPath); + cabinet.Compress(files, CompressionLevel.Low); + } + + // Extract. + { + var cabinet = new Cabinet(cabinetPath); + var reportedFiles = cabinet.Extract(extractFolder); + Assert.Equal(2, reportedFiles.Count()); + } + + // Enumerate to compare cabinet to extracted files. + { + var cabinet = new Cabinet(cabinetPath); + var enumerated = cabinet.Enumerate().OrderBy(f => f.FileId).ToArray(); + + var files = Directory.EnumerateFiles(extractFolder).OrderBy(f => f).ToArray(); + + for (var i = 0; i < enumerated.Length; ++i) + { + var cabFileInfo = enumerated[i]; + var fileInfo = new FileInfo(files[i]); + + Assert.Equal(cabFileInfo.FileId, fileInfo.Name); + Assert.Equal(cabFileInfo.Size, fileInfo.Length); + Assert.True(cabFileInfo.SameAsDateTime(fileInfo.CreationTime)); + } + } + } + } + } +} diff --git a/src/wix/test/WixToolsetTest.Core.Native/MsmFixture.cs b/src/wix/test/WixToolsetTest.Core.Native/MsmFixture.cs new file mode 100644 index 00000000..709d4b93 --- /dev/null +++ b/src/wix/test/WixToolsetTest.Core.Native/MsmFixture.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 WixToolsetTest.CoreNative +{ + using WixToolset.Core.Native.Msm; + using Xunit; + + public class MsmFixture + { + [Fact] + public void CanCreateMsmInterface() + { + var merge = MsmInterop.GetMsmMerge(); + Assert.NotNull(merge); + } + } +} diff --git a/src/wix/test/WixToolsetTest.Core.Native/TestData/test.cab b/src/wix/test/WixToolsetTest.Core.Native/TestData/test.cab new file mode 100644 index 00000000..ca78f632 Binary files /dev/null and b/src/wix/test/WixToolsetTest.Core.Native/TestData/test.cab differ diff --git a/src/wix/test/WixToolsetTest.Core.Native/TestData/test.txt b/src/wix/test/WixToolsetTest.Core.Native/TestData/test.txt new file mode 100644 index 00000000..cd0db0e1 --- /dev/null +++ b/src/wix/test/WixToolsetTest.Core.Native/TestData/test.txt @@ -0,0 +1 @@ +This is test.txt. \ No newline at end of file diff --git a/src/wix/test/WixToolsetTest.Core.Native/Utility/DisposableFileSystem.cs b/src/wix/test/WixToolsetTest.Core.Native/Utility/DisposableFileSystem.cs new file mode 100644 index 00000000..c9957247 --- /dev/null +++ b/src/wix/test/WixToolsetTest.Core.Native/Utility/DisposableFileSystem.cs @@ -0,0 +1,86 @@ +// 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 WixToolsetTest.CoreNative.Utility +{ + using System; + using System.Collections.Generic; + using System.IO; + + public class DisposableFileSystem : IDisposable + { + protected bool Disposed { get; private set; } + + private List CleanupPaths { get; } = new List(); + + public string GetFile(bool create = false) + { + var path = Path.GetTempFileName(); + + if (!create) + { + File.Delete(path); + } + + this.CleanupPaths.Add(path); + + return path; + } + + public string GetFolder(bool create = false) + { + var path = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName()); + + if (create) + { + Directory.CreateDirectory(path); + } + + this.CleanupPaths.Add(path); + + return path; + } + + + #region // IDisposable + + public void Dispose() + { + this.Dispose(true); + GC.SuppressFinalize(this); + } + + protected virtual void Dispose(bool disposing) + { + if (this.Disposed) + { + return; + } + + if (disposing) + { + foreach (var path in this.CleanupPaths) + { + try + { + if (File.Exists(path)) + { + File.Delete(path); + } + else if (Directory.Exists(path)) + { + Directory.Delete(path, true); + } + } + catch + { + // Best effort delete, so ignore any failures. + } + } + } + + this.Disposed = true; + } + + #endregion + } +} diff --git a/src/wix/test/WixToolsetTest.Core.Native/Utility/Pushd.cs b/src/wix/test/WixToolsetTest.Core.Native/Utility/Pushd.cs new file mode 100644 index 00000000..91700c2f --- /dev/null +++ b/src/wix/test/WixToolsetTest.Core.Native/Utility/Pushd.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 WixToolsetTest.CoreNative.Utility +{ + using System; + using System.IO; + + public class Pushd : IDisposable + { + protected bool Disposed { get; private set; } + + public Pushd(string path) + { + this.PreviousDirectory = Directory.GetCurrentDirectory(); + + Directory.SetCurrentDirectory(path); + } + + public string PreviousDirectory { get; } + + #region // IDisposable + + public void Dispose() + { + this.Dispose(true); + GC.SuppressFinalize(this); + } + + protected virtual void Dispose(bool disposing) + { + if (this.Disposed) + { + return; + } + + if (disposing) + { + Directory.SetCurrentDirectory(this.PreviousDirectory); + } + + this.Disposed = true; + } + + #endregion + } +} diff --git a/src/wix/test/WixToolsetTest.Core.Native/Utility/TestData.cs b/src/wix/test/WixToolsetTest.Core.Native/Utility/TestData.cs new file mode 100644 index 00000000..cd9c6318 --- /dev/null +++ b/src/wix/test/WixToolsetTest.Core.Native/Utility/TestData.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 WixToolsetTest.CoreNative.Utility +{ + using System; + using System.IO; + + public class TestData + { + public static string LocalPath => Path.GetDirectoryName(new Uri(System.Reflection.Assembly.GetExecutingAssembly().CodeBase).LocalPath); + + public static string Get(params string[] paths) + { + return Path.Combine(LocalPath, Path.Combine(paths)); + } + } +} diff --git a/src/wix/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj b/src/wix/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj new file mode 100644 index 00000000..6068dbea --- /dev/null +++ b/src/wix/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj @@ -0,0 +1,26 @@ + + + + + + netcoreapp3.1 + false + win-x64 + + + + + + + + + + + + + + + + + + diff --git a/src/wix/test/version.txt b/src/wix/test/version.txt new file mode 100644 index 00000000..cf138743 --- /dev/null +++ b/src/wix/test/version.txt @@ -0,0 +1 @@ +v42.42.{height}-preview.0 \ No newline at end of file diff --git a/src/wix/wixnative/ARM/mergemod.dll b/src/wix/wixnative/ARM/mergemod.dll new file mode 100644 index 00000000..739af831 Binary files /dev/null and b/src/wix/wixnative/ARM/mergemod.dll differ diff --git a/src/wix/wixnative/ARM64/mergemod.dll b/src/wix/wixnative/ARM64/mergemod.dll new file mode 100644 index 00000000..564a75fc Binary files /dev/null and b/src/wix/wixnative/ARM64/mergemod.dll differ diff --git a/src/wix/wixnative/Win32/mergemod.dll b/src/wix/wixnative/Win32/mergemod.dll new file mode 100644 index 00000000..4286df4d Binary files /dev/null and b/src/wix/wixnative/Win32/mergemod.dll differ diff --git a/src/wix/wixnative/enumcab.cpp b/src/wix/wixnative/enumcab.cpp new file mode 100644 index 00000000..e7717bac --- /dev/null +++ b/src/wix/wixnative/enumcab.cpp @@ -0,0 +1,47 @@ +// 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. + +#include "precomp.h" + +static INT_PTR __stdcall EnumCallback(FDINOTIFICATIONTYPE fdint, PFDINOTIFICATION pfdin); + + +HRESULT EnumCabCommand( + __in int argc, + __in LPWSTR argv[] +) +{ + HRESULT hr = E_INVALIDARG; + LPCWSTR wzCabPath = NULL; + + if (argc < 1) + { + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Must specify: cabPath outputFolder"); + } + + wzCabPath = argv[0]; + + hr = CabInitialize(FALSE); + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to initialize cabinet: %ls", wzCabPath); + + hr = CabEnumerate(wzCabPath, L"*", EnumCallback, 0); + ExitOnFailure(hr, "failed to compress files into cabinet: %ls", wzCabPath); + +LExit: + CabUninitialize(); + + return hr; +} + + +static INT_PTR __stdcall EnumCallback( + __in FDINOTIFICATIONTYPE fdint, + __in PFDINOTIFICATION pfdin +) +{ + if (fdint == fdintCOPY_FILE) + { + ConsoleWriteLine(CONSOLE_COLOR_NORMAL, "%s\t%d\t%u\t%u", pfdin->psz1, pfdin->cb, pfdin->date, pfdin->time); + } + + return 0; +} diff --git a/src/wix/wixnative/extractcab.cpp b/src/wix/wixnative/extractcab.cpp new file mode 100644 index 00000000..53f53266 --- /dev/null +++ b/src/wix/wixnative/extractcab.cpp @@ -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. + +#include "precomp.h" + +static HRESULT ProgressCallback(BOOL fBeginFile, LPCWSTR wzFileId, LPVOID pvContext); + + +HRESULT ExtractCabCommand( + __in int argc, + __in LPWSTR argv[] +) +{ + HRESULT hr = E_INVALIDARG; + LPCWSTR wzCabPath = NULL; + LPCWSTR wzOutputFolder = NULL; + + if (argc < 2) + { + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Must specify: cabPath outputFolder"); + } + + wzCabPath = argv[0]; + wzOutputFolder = argv[1]; + + hr = CabInitialize(FALSE); + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to initialize cabinet: %ls", wzCabPath); + + hr = CabExtract(wzCabPath, L"*", wzOutputFolder, ProgressCallback, NULL, 0); + ExitOnFailure(hr, "failed to compress files into cabinet: %ls", wzCabPath); + +LExit: + CabUninitialize(); + + return hr; +} + + +static HRESULT ProgressCallback( + __in BOOL fBeginFile, + __in LPCWSTR wzFileId, + __in LPVOID /*pvContext*/ +) +{ + if (fBeginFile) + { + ConsoleWriteLine(CONSOLE_COLOR_NORMAL, "%ls", wzFileId); + } + + return S_OK; +} diff --git a/src/wix/wixnative/packages.config b/src/wix/wixnative/packages.config new file mode 100644 index 00000000..a98c0c8e --- /dev/null +++ b/src/wix/wixnative/packages.config @@ -0,0 +1,8 @@ + + + + + + + + \ No newline at end of file diff --git a/src/wix/wixnative/precomp.cpp b/src/wix/wixnative/precomp.cpp new file mode 100644 index 00000000..37664a1c --- /dev/null +++ b/src/wix/wixnative/precomp.cpp @@ -0,0 +1,3 @@ +// 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. + +#include "precomp.h" diff --git a/src/wix/wixnative/precomp.h b/src/wix/wixnative/precomp.h new file mode 100644 index 00000000..5bd617e5 --- /dev/null +++ b/src/wix/wixnative/precomp.h @@ -0,0 +1,19 @@ +#pragma once +// 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. + +#include +#include +#include + +#include "dutil.h" +#include "conutil.h" +#include "memutil.h" +#include "pathutil.h" +#include "strutil.h" +#include "cabcutil.h" +#include "cabutil.h" + +HRESULT SmartCabCommand(int argc, LPWSTR argv[]); +HRESULT ResetAclsCommand(int argc, LPWSTR argv[]); +HRESULT EnumCabCommand(int argc, LPWSTR argv[]); +HRESULT ExtractCabCommand(int argc, LPWSTR argv[]); diff --git a/src/wix/wixnative/resetacls.cpp b/src/wix/wixnative/resetacls.cpp new file mode 100644 index 00000000..8c5bdc56 --- /dev/null +++ b/src/wix/wixnative/resetacls.cpp @@ -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. + +#include "precomp.h" + +HRESULT ResetAclsCommand(int argc, LPWSTR argv[]) +{ + Unused(argc); + Unused(argv); + + HRESULT hr = S_OK; + ACL* pacl = NULL; + DWORD cbAcl = sizeof(ACL); + LPWSTR sczFilePath = NULL; + + // create an empty (not NULL!) ACL to use on all the files + pacl = static_cast(MemAlloc(cbAcl, FALSE)); + ConsoleExitOnNull(pacl, hr, E_OUTOFMEMORY, CONSOLE_COLOR_RED, "failed to allocate ACL"); + +#pragma prefast(push) +#pragma prefast(disable:25029) + if (!::InitializeAcl(pacl, cbAcl, ACL_REVISION)) +#pragma prefast(op) + { + ConsoleExitOnLastError(hr, CONSOLE_COLOR_RED, "failed to initialize ACL"); + } + + // Reset the existing security permissions on each provided file. + for (;;) + { + hr = ConsoleReadW(&sczFilePath); + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to read file path from stdin"); + + if (!*sczFilePath) + { + break; + } + + hr = ::SetNamedSecurityInfoW(sczFilePath, SE_FILE_OBJECT, DACL_SECURITY_INFORMATION | UNPROTECTED_DACL_SECURITY_INFORMATION, NULL, NULL, pacl, NULL); + if (ERROR_FILE_NOT_FOUND != hr && ERROR_PATH_NOT_FOUND != hr) + { + ConsoleExitOnFailure(hr = HRESULT_FROM_WIN32(hr), CONSOLE_COLOR_RED, "failed to set security descriptor for file: %ls", sczFilePath); + } + } + + AssertSz(::IsValidAcl(pacl), "ResetAcls() - created invalid ACL"); + +LExit: + ReleaseStr(sczFilePath); + ReleaseMem(pacl); + return hr; +} diff --git a/src/wix/wixnative/smartcab.cpp b/src/wix/wixnative/smartcab.cpp new file mode 100644 index 00000000..0dff6c94 --- /dev/null +++ b/src/wix/wixnative/smartcab.cpp @@ -0,0 +1,160 @@ +// 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. + +#include "precomp.h" + +static HRESULT CompressFiles(HANDLE hCab); +static void __stdcall CabNamesCallback(LPWSTR wzFirstCabName, LPWSTR wzNewCabName, LPWSTR wzFileToken); + + +HRESULT SmartCabCommand( + __in int argc, + __in LPWSTR argv[] +) +{ + HRESULT hr = E_INVALIDARG; + LPCWSTR wzCabPath = NULL; + LPCWSTR wzCabName = NULL; + LPWSTR sczCabDir = NULL; + UINT uiFileCount = 0; + UINT uiMaxSize = 0; + UINT uiMaxThresh = 0; + COMPRESSION_TYPE ct = COMPRESSION_TYPE_NONE; + HANDLE hCab = NULL; + + if (argc < 1) + { + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Must specify: outCabPath [compressionType] [fileCount] [maxSizePerCabInMB [maxThreshold]]"); + } + else + { + wzCabPath = argv[0]; + wzCabName = PathFile(wzCabPath); + + hr = PathGetDirectory(wzCabPath, &sczCabDir); + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Could not parse directory from path: %ls", wzCabPath); + + if (argc > 1) + { + UINT uiCompressionType; + hr = StrStringToUInt32(argv[1], 0, &uiCompressionType); + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Could not parse compression type as number: %ls", argv[1]); + + ct = (uiCompressionType > 4) ? COMPRESSION_TYPE_HIGH : static_cast(uiCompressionType); + } + + if (argc > 2) + { + hr = StrStringToUInt32(argv[2], 0, &uiFileCount); + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Could not parse file count as number: %ls", argv[2]); + } + + if (argc > 3) + { + hr = StrStringToUInt32(argv[3], 0, &uiMaxSize); + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Could not parse max size as number: %ls", argv[3]); + } + + if (argc > 4) + { + hr = StrStringToUInt32(argv[4], 0, &uiMaxThresh); + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Could not parse max threshold as number: %ls", argv[4]); + } + } + + hr = CabCBegin(wzCabName, sczCabDir, uiFileCount, uiMaxSize, uiMaxThresh, ct, &hCab); + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to initialize cabinet: %ls", wzCabPath); + + if (uiFileCount > 0) + { + hr = CompressFiles(hCab); + ExitOnFailure(hr, "failed to compress files into cabinet: %ls", wzCabPath); + } + + hr = CabCFinish(hCab, CabNamesCallback); + hCab = NULL; // once finish is called, the handle is invalid. + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to compress cabinet: %ls", wzCabPath); + + +LExit: + if (hCab) + { + CabCCancel(hCab); + } + ReleaseStr(sczCabDir); + + return hr; +} + + +static HRESULT CompressFiles( + __in HANDLE hCab +) +{ + HRESULT hr = S_OK; + LPWSTR sczLine = NULL; + LPWSTR* rgsczSplit = NULL; + UINT cSplit = 0; + MSIFILEHASHINFO hashInfo = { sizeof(MSIFILEHASHINFO) }; + + for (;;) + { + hr = ConsoleReadW(&sczLine); + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to read smartcab line from stdin"); + + if (!*sczLine) + { + break; + } + + hr = StrSplitAllocArray(&rgsczSplit, &cSplit, sczLine, L"\t"); + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to split smartcab line from stdin: %ls", sczLine); + + if (cSplit != 2 && cSplit != 6) + { + hr = E_INVALIDARG; + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to split smartcab line into hash x 4, token, source file: %ls", sczLine); + } + + LPCWSTR wzFilePath = rgsczSplit[0]; + LPCWSTR wzToken = rgsczSplit[1]; + PMSIFILEHASHINFO pHashInfo = NULL; + + if (cSplit == 6) + { + for (int i = 0; i < 4; ++i) + { + LPCWSTR wzHash = rgsczSplit[i + 2]; + + hr = StrStringToInt32(wzHash, 0, reinterpret_cast(hashInfo.dwData + i)); + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to parse hash: %ls for file: %ls", wzHash, wzFilePath); + } + + pHashInfo = &hashInfo; + } + + hr = CabCAddFile(wzFilePath, wzToken, pHashInfo, hCab); + ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to add file: %ls", wzFilePath); + + ReleaseNullStrArray(rgsczSplit, cSplit); + } + +LExit: + ReleaseNullStrArray(rgsczSplit, cSplit); + ReleaseStr(sczLine); + + return hr; +} + + +// Callback from PFNFCIGETNEXTCABINET CabCGetNextCabinet method +// First argument is the name of splitting cabinet without extension e.g. "cab1" +// Second argument is name of the new cabinet that would be formed by splitting e.g. "cab1b.cab" +// Third argument is the file token of the first file present in the splitting cabinet +static void __stdcall CabNamesCallback( + __in LPWSTR wzFirstCabName, + __in LPWSTR wzNewCabName, + __in LPWSTR wzFileToken +) +{ + ConsoleWriteLine(CONSOLE_COLOR_NORMAL, "%ls\t%ls\t%ls", wzFirstCabName, wzNewCabName, wzFileToken); +} diff --git a/src/wix/wixnative/wixnative.cpp b/src/wix/wixnative/wixnative.cpp new file mode 100644 index 00000000..7bd8dbca --- /dev/null +++ b/src/wix/wixnative/wixnative.cpp @@ -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. + +#include "precomp.h" + +int __cdecl wmain(int argc, LPWSTR argv[]) +{ + HRESULT hr = E_INVALIDARG; + + ConsoleInitialize(); + + if (argc < 2) + { + ConsoleWriteError(hr, CONSOLE_COLOR_RED, "Must specify a command"); + } + else if (CSTR_EQUAL == ::CompareString(LOCALE_INVARIANT, NORM_IGNORECASE, argv[1], -1, L"smartcab", -1)) + { + hr = SmartCabCommand(argc - 2, argv + 2); + } + else if (CSTR_EQUAL == ::CompareString(LOCALE_INVARIANT, NORM_IGNORECASE, argv[1], -1, L"extractcab", -1)) + { + hr = ExtractCabCommand(argc - 2, argv + 2); + } + else if (CSTR_EQUAL == ::CompareString(LOCALE_INVARIANT, NORM_IGNORECASE, argv[1], -1, L"enumcab", -1)) + { + hr = EnumCabCommand(argc - 2, argv + 2); + } + else if (CSTR_EQUAL == ::CompareString(LOCALE_INVARIANT, NORM_IGNORECASE, argv[1], -1, L"resetacls", -1)) + { + hr = ResetAclsCommand(argc - 2, argv + 2); + } + else + { + ConsoleWriteError(hr, CONSOLE_COLOR_RED, "Unknown command: %ls", argv[1]); + } + + ConsoleUninitialize(); + return HRESULT_CODE(hr); +} diff --git a/src/wix/wixnative/wixnative.v3.ncrunchproject b/src/wix/wixnative/wixnative.v3.ncrunchproject new file mode 100644 index 00000000..319cd523 --- /dev/null +++ b/src/wix/wixnative/wixnative.v3.ncrunchproject @@ -0,0 +1,5 @@ + + + True + + \ No newline at end of file diff --git a/src/wix/wixnative/wixnative.vcxproj b/src/wix/wixnative/wixnative.vcxproj new file mode 100644 index 00000000..20959827 --- /dev/null +++ b/src/wix/wixnative/wixnative.vcxproj @@ -0,0 +1,76 @@ + + + + + + + Debug + ARM64 + + + Debug + Win32 + + + Debug + x64 + + + Release + ARM64 + + + Release + Win32 + + + Release + x64 + + + + + {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF} + Application + v142 + Unicode + Console + wixnative + WiX Native Processing + 10.0 + + + + + + + crypt32.lib;cabinet.lib;msi.lib + + + + + + Create + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/wix/wixnative/x64/mergemod.dll b/src/wix/wixnative/x64/mergemod.dll new file mode 100644 index 00000000..b6422174 Binary files /dev/null and b/src/wix/wixnative/x64/mergemod.dll differ diff --git a/src/wixnative/ARM/mergemod.dll b/src/wixnative/ARM/mergemod.dll deleted file mode 100644 index 739af831..00000000 Binary files a/src/wixnative/ARM/mergemod.dll and /dev/null differ diff --git a/src/wixnative/ARM64/mergemod.dll b/src/wixnative/ARM64/mergemod.dll deleted file mode 100644 index 564a75fc..00000000 Binary files a/src/wixnative/ARM64/mergemod.dll and /dev/null differ diff --git a/src/wixnative/Win32/mergemod.dll b/src/wixnative/Win32/mergemod.dll deleted file mode 100644 index 4286df4d..00000000 Binary files a/src/wixnative/Win32/mergemod.dll and /dev/null differ diff --git a/src/wixnative/enumcab.cpp b/src/wixnative/enumcab.cpp deleted file mode 100644 index e7717bac..00000000 --- a/src/wixnative/enumcab.cpp +++ /dev/null @@ -1,47 +0,0 @@ -// 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. - -#include "precomp.h" - -static INT_PTR __stdcall EnumCallback(FDINOTIFICATIONTYPE fdint, PFDINOTIFICATION pfdin); - - -HRESULT EnumCabCommand( - __in int argc, - __in LPWSTR argv[] -) -{ - HRESULT hr = E_INVALIDARG; - LPCWSTR wzCabPath = NULL; - - if (argc < 1) - { - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Must specify: cabPath outputFolder"); - } - - wzCabPath = argv[0]; - - hr = CabInitialize(FALSE); - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to initialize cabinet: %ls", wzCabPath); - - hr = CabEnumerate(wzCabPath, L"*", EnumCallback, 0); - ExitOnFailure(hr, "failed to compress files into cabinet: %ls", wzCabPath); - -LExit: - CabUninitialize(); - - return hr; -} - - -static INT_PTR __stdcall EnumCallback( - __in FDINOTIFICATIONTYPE fdint, - __in PFDINOTIFICATION pfdin -) -{ - if (fdint == fdintCOPY_FILE) - { - ConsoleWriteLine(CONSOLE_COLOR_NORMAL, "%s\t%d\t%u\t%u", pfdin->psz1, pfdin->cb, pfdin->date, pfdin->time); - } - - return 0; -} diff --git a/src/wixnative/extractcab.cpp b/src/wixnative/extractcab.cpp deleted file mode 100644 index 53f53266..00000000 --- a/src/wixnative/extractcab.cpp +++ /dev/null @@ -1,50 +0,0 @@ -// 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. - -#include "precomp.h" - -static HRESULT ProgressCallback(BOOL fBeginFile, LPCWSTR wzFileId, LPVOID pvContext); - - -HRESULT ExtractCabCommand( - __in int argc, - __in LPWSTR argv[] -) -{ - HRESULT hr = E_INVALIDARG; - LPCWSTR wzCabPath = NULL; - LPCWSTR wzOutputFolder = NULL; - - if (argc < 2) - { - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Must specify: cabPath outputFolder"); - } - - wzCabPath = argv[0]; - wzOutputFolder = argv[1]; - - hr = CabInitialize(FALSE); - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to initialize cabinet: %ls", wzCabPath); - - hr = CabExtract(wzCabPath, L"*", wzOutputFolder, ProgressCallback, NULL, 0); - ExitOnFailure(hr, "failed to compress files into cabinet: %ls", wzCabPath); - -LExit: - CabUninitialize(); - - return hr; -} - - -static HRESULT ProgressCallback( - __in BOOL fBeginFile, - __in LPCWSTR wzFileId, - __in LPVOID /*pvContext*/ -) -{ - if (fBeginFile) - { - ConsoleWriteLine(CONSOLE_COLOR_NORMAL, "%ls", wzFileId); - } - - return S_OK; -} diff --git a/src/wixnative/packages.config b/src/wixnative/packages.config deleted file mode 100644 index a98c0c8e..00000000 --- a/src/wixnative/packages.config +++ /dev/null @@ -1,8 +0,0 @@ - - - - - - - - \ No newline at end of file diff --git a/src/wixnative/precomp.cpp b/src/wixnative/precomp.cpp deleted file mode 100644 index 37664a1c..00000000 --- a/src/wixnative/precomp.cpp +++ /dev/null @@ -1,3 +0,0 @@ -// 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. - -#include "precomp.h" diff --git a/src/wixnative/precomp.h b/src/wixnative/precomp.h deleted file mode 100644 index 5bd617e5..00000000 --- a/src/wixnative/precomp.h +++ /dev/null @@ -1,19 +0,0 @@ -#pragma once -// 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. - -#include -#include -#include - -#include "dutil.h" -#include "conutil.h" -#include "memutil.h" -#include "pathutil.h" -#include "strutil.h" -#include "cabcutil.h" -#include "cabutil.h" - -HRESULT SmartCabCommand(int argc, LPWSTR argv[]); -HRESULT ResetAclsCommand(int argc, LPWSTR argv[]); -HRESULT EnumCabCommand(int argc, LPWSTR argv[]); -HRESULT ExtractCabCommand(int argc, LPWSTR argv[]); diff --git a/src/wixnative/resetacls.cpp b/src/wixnative/resetacls.cpp deleted file mode 100644 index 8c5bdc56..00000000 --- a/src/wixnative/resetacls.cpp +++ /dev/null @@ -1,51 +0,0 @@ -// 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. - -#include "precomp.h" - -HRESULT ResetAclsCommand(int argc, LPWSTR argv[]) -{ - Unused(argc); - Unused(argv); - - HRESULT hr = S_OK; - ACL* pacl = NULL; - DWORD cbAcl = sizeof(ACL); - LPWSTR sczFilePath = NULL; - - // create an empty (not NULL!) ACL to use on all the files - pacl = static_cast(MemAlloc(cbAcl, FALSE)); - ConsoleExitOnNull(pacl, hr, E_OUTOFMEMORY, CONSOLE_COLOR_RED, "failed to allocate ACL"); - -#pragma prefast(push) -#pragma prefast(disable:25029) - if (!::InitializeAcl(pacl, cbAcl, ACL_REVISION)) -#pragma prefast(op) - { - ConsoleExitOnLastError(hr, CONSOLE_COLOR_RED, "failed to initialize ACL"); - } - - // Reset the existing security permissions on each provided file. - for (;;) - { - hr = ConsoleReadW(&sczFilePath); - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to read file path from stdin"); - - if (!*sczFilePath) - { - break; - } - - hr = ::SetNamedSecurityInfoW(sczFilePath, SE_FILE_OBJECT, DACL_SECURITY_INFORMATION | UNPROTECTED_DACL_SECURITY_INFORMATION, NULL, NULL, pacl, NULL); - if (ERROR_FILE_NOT_FOUND != hr && ERROR_PATH_NOT_FOUND != hr) - { - ConsoleExitOnFailure(hr = HRESULT_FROM_WIN32(hr), CONSOLE_COLOR_RED, "failed to set security descriptor for file: %ls", sczFilePath); - } - } - - AssertSz(::IsValidAcl(pacl), "ResetAcls() - created invalid ACL"); - -LExit: - ReleaseStr(sczFilePath); - ReleaseMem(pacl); - return hr; -} diff --git a/src/wixnative/smartcab.cpp b/src/wixnative/smartcab.cpp deleted file mode 100644 index 0dff6c94..00000000 --- a/src/wixnative/smartcab.cpp +++ /dev/null @@ -1,160 +0,0 @@ -// 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. - -#include "precomp.h" - -static HRESULT CompressFiles(HANDLE hCab); -static void __stdcall CabNamesCallback(LPWSTR wzFirstCabName, LPWSTR wzNewCabName, LPWSTR wzFileToken); - - -HRESULT SmartCabCommand( - __in int argc, - __in LPWSTR argv[] -) -{ - HRESULT hr = E_INVALIDARG; - LPCWSTR wzCabPath = NULL; - LPCWSTR wzCabName = NULL; - LPWSTR sczCabDir = NULL; - UINT uiFileCount = 0; - UINT uiMaxSize = 0; - UINT uiMaxThresh = 0; - COMPRESSION_TYPE ct = COMPRESSION_TYPE_NONE; - HANDLE hCab = NULL; - - if (argc < 1) - { - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Must specify: outCabPath [compressionType] [fileCount] [maxSizePerCabInMB [maxThreshold]]"); - } - else - { - wzCabPath = argv[0]; - wzCabName = PathFile(wzCabPath); - - hr = PathGetDirectory(wzCabPath, &sczCabDir); - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Could not parse directory from path: %ls", wzCabPath); - - if (argc > 1) - { - UINT uiCompressionType; - hr = StrStringToUInt32(argv[1], 0, &uiCompressionType); - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Could not parse compression type as number: %ls", argv[1]); - - ct = (uiCompressionType > 4) ? COMPRESSION_TYPE_HIGH : static_cast(uiCompressionType); - } - - if (argc > 2) - { - hr = StrStringToUInt32(argv[2], 0, &uiFileCount); - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Could not parse file count as number: %ls", argv[2]); - } - - if (argc > 3) - { - hr = StrStringToUInt32(argv[3], 0, &uiMaxSize); - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Could not parse max size as number: %ls", argv[3]); - } - - if (argc > 4) - { - hr = StrStringToUInt32(argv[4], 0, &uiMaxThresh); - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "Could not parse max threshold as number: %ls", argv[4]); - } - } - - hr = CabCBegin(wzCabName, sczCabDir, uiFileCount, uiMaxSize, uiMaxThresh, ct, &hCab); - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to initialize cabinet: %ls", wzCabPath); - - if (uiFileCount > 0) - { - hr = CompressFiles(hCab); - ExitOnFailure(hr, "failed to compress files into cabinet: %ls", wzCabPath); - } - - hr = CabCFinish(hCab, CabNamesCallback); - hCab = NULL; // once finish is called, the handle is invalid. - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to compress cabinet: %ls", wzCabPath); - - -LExit: - if (hCab) - { - CabCCancel(hCab); - } - ReleaseStr(sczCabDir); - - return hr; -} - - -static HRESULT CompressFiles( - __in HANDLE hCab -) -{ - HRESULT hr = S_OK; - LPWSTR sczLine = NULL; - LPWSTR* rgsczSplit = NULL; - UINT cSplit = 0; - MSIFILEHASHINFO hashInfo = { sizeof(MSIFILEHASHINFO) }; - - for (;;) - { - hr = ConsoleReadW(&sczLine); - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to read smartcab line from stdin"); - - if (!*sczLine) - { - break; - } - - hr = StrSplitAllocArray(&rgsczSplit, &cSplit, sczLine, L"\t"); - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to split smartcab line from stdin: %ls", sczLine); - - if (cSplit != 2 && cSplit != 6) - { - hr = E_INVALIDARG; - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to split smartcab line into hash x 4, token, source file: %ls", sczLine); - } - - LPCWSTR wzFilePath = rgsczSplit[0]; - LPCWSTR wzToken = rgsczSplit[1]; - PMSIFILEHASHINFO pHashInfo = NULL; - - if (cSplit == 6) - { - for (int i = 0; i < 4; ++i) - { - LPCWSTR wzHash = rgsczSplit[i + 2]; - - hr = StrStringToInt32(wzHash, 0, reinterpret_cast(hashInfo.dwData + i)); - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to parse hash: %ls for file: %ls", wzHash, wzFilePath); - } - - pHashInfo = &hashInfo; - } - - hr = CabCAddFile(wzFilePath, wzToken, pHashInfo, hCab); - ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to add file: %ls", wzFilePath); - - ReleaseNullStrArray(rgsczSplit, cSplit); - } - -LExit: - ReleaseNullStrArray(rgsczSplit, cSplit); - ReleaseStr(sczLine); - - return hr; -} - - -// Callback from PFNFCIGETNEXTCABINET CabCGetNextCabinet method -// First argument is the name of splitting cabinet without extension e.g. "cab1" -// Second argument is name of the new cabinet that would be formed by splitting e.g. "cab1b.cab" -// Third argument is the file token of the first file present in the splitting cabinet -static void __stdcall CabNamesCallback( - __in LPWSTR wzFirstCabName, - __in LPWSTR wzNewCabName, - __in LPWSTR wzFileToken -) -{ - ConsoleWriteLine(CONSOLE_COLOR_NORMAL, "%ls\t%ls\t%ls", wzFirstCabName, wzNewCabName, wzFileToken); -} diff --git a/src/wixnative/wixnative.cpp b/src/wixnative/wixnative.cpp deleted file mode 100644 index 7bd8dbca..00000000 --- a/src/wixnative/wixnative.cpp +++ /dev/null @@ -1,38 +0,0 @@ -// 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. - -#include "precomp.h" - -int __cdecl wmain(int argc, LPWSTR argv[]) -{ - HRESULT hr = E_INVALIDARG; - - ConsoleInitialize(); - - if (argc < 2) - { - ConsoleWriteError(hr, CONSOLE_COLOR_RED, "Must specify a command"); - } - else if (CSTR_EQUAL == ::CompareString(LOCALE_INVARIANT, NORM_IGNORECASE, argv[1], -1, L"smartcab", -1)) - { - hr = SmartCabCommand(argc - 2, argv + 2); - } - else if (CSTR_EQUAL == ::CompareString(LOCALE_INVARIANT, NORM_IGNORECASE, argv[1], -1, L"extractcab", -1)) - { - hr = ExtractCabCommand(argc - 2, argv + 2); - } - else if (CSTR_EQUAL == ::CompareString(LOCALE_INVARIANT, NORM_IGNORECASE, argv[1], -1, L"enumcab", -1)) - { - hr = EnumCabCommand(argc - 2, argv + 2); - } - else if (CSTR_EQUAL == ::CompareString(LOCALE_INVARIANT, NORM_IGNORECASE, argv[1], -1, L"resetacls", -1)) - { - hr = ResetAclsCommand(argc - 2, argv + 2); - } - else - { - ConsoleWriteError(hr, CONSOLE_COLOR_RED, "Unknown command: %ls", argv[1]); - } - - ConsoleUninitialize(); - return HRESULT_CODE(hr); -} diff --git a/src/wixnative/wixnative.v3.ncrunchproject b/src/wixnative/wixnative.v3.ncrunchproject deleted file mode 100644 index 319cd523..00000000 --- a/src/wixnative/wixnative.v3.ncrunchproject +++ /dev/null @@ -1,5 +0,0 @@ - - - True - - \ No newline at end of file diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj deleted file mode 100644 index 20959827..00000000 --- a/src/wixnative/wixnative.vcxproj +++ /dev/null @@ -1,76 +0,0 @@ - - - - - - - Debug - ARM64 - - - Debug - Win32 - - - Debug - x64 - - - Release - ARM64 - - - Release - Win32 - - - Release - x64 - - - - - {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF} - Application - v142 - Unicode - Console - wixnative - WiX Native Processing - 10.0 - - - - - - - crypt32.lib;cabinet.lib;msi.lib - - - - - - Create - - - - - - - - - - - - - - - - - - - - - - - diff --git a/src/wixnative/x64/mergemod.dll b/src/wixnative/x64/mergemod.dll deleted file mode 100644 index b6422174..00000000 Binary files a/src/wixnative/x64/mergemod.dll and /dev/null differ -- cgit v1.2.3-55-g6feb