From 83d09505bc2e5ccb9840a30badb875d4061dd797 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Sun, 3 Sep 2017 15:16:18 -0700 Subject: Initial commit --- README.md | 2 ++ 1 file changed, 2 insertions(+) create mode 100644 README.md diff --git a/README.md b/README.md new file mode 100644 index 00000000..787123cc --- /dev/null +++ b/README.md @@ -0,0 +1,2 @@ +# Core.Native +Core.Native - native component of WixToolset.Core -- cgit v1.2.3-55-g6feb From 2a72f06449431b326c671cf59811b9cefb73a2c8 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Sun, 3 Sep 2017 15:30:10 -0700 Subject: Initialize repo --- .gitattributes | 2 + .gitignore | 295 ++++++++++++++++++++++++++++++++++++++++++++++ LICENSE.TXT | 28 +++++ appveyor.cmd | 5 + appveyor.yml | 27 +++++ nuget.config | 8 ++ src/Cpp.Build.props | 100 ++++++++++++++++ src/Directory.Build.props | 21 ++++ version.json | 11 ++ 9 files changed, 497 insertions(+) create mode 100644 .gitattributes create mode 100644 .gitignore create mode 100644 LICENSE.TXT create mode 100644 appveyor.cmd create mode 100644 appveyor.yml create mode 100644 nuget.config create mode 100644 src/Cpp.Build.props create mode 100644 src/Directory.Build.props create mode 100644 version.json diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 00000000..dfe07704 --- /dev/null +++ b/.gitattributes @@ -0,0 +1,2 @@ +# Auto detect text files and perform LF normalization +* text=auto diff --git a/.gitignore b/.gitignore new file mode 100644 index 00000000..3c6208a8 --- /dev/null +++ b/.gitignore @@ -0,0 +1,295 @@ +## Ignore Visual Studio temporary files, build results, and +## files generated by popular Visual Studio add-ons. +## +## Get latest from https://github.com/github/gitignore/blob/master/VisualStudio.gitignore + +# User-specific files +*.suo +*.user +*.userosscache +*.sln.docstates + +# User-specific files (MonoDevelop/Xamarin Studio) +*.userprefs + +# Build results +[Dd]ebug/ +[Dd]ebugPublic/ +[Rr]elease/ +[Rr]eleases/ +x64/ +x86/ +bld/ +[Bb]in/ +[Oo]bj/ +[Ll]og/ + +# Visual Studio 2015 cache/options directory +.vs/ +# Uncomment if you have tasks that create the project's static files in wwwroot +#wwwroot/ + +# MSTest test Results +[Tt]est[Rr]esult*/ +[Bb]uild[Ll]og.* + +# NUNIT +*.VisualState.xml +TestResult.xml + +# Build Results of an ATL Project +[Dd]ebugPS/ +[Rr]eleasePS/ +dlldata.c + +# Benchmark Results +BenchmarkDotNet.Artifacts/ + +# .NET Core +project.lock.json +project.fragment.lock.json +artifacts/ +**/Properties/launchSettings.json + +*_i.c +*_p.c +*_i.h +*.ilk +*.meta +*.obj +*.pch +*.pdb +*.pgc +*.pgd +*.rsp +*.sbr +*.tlb +*.tli +*.tlh +*.tmp +*.tmp_proj +*.log +*.vspscc +*.vssscc +.builds +*.pidb +*.svclog +*.scc + +# Chutzpah Test files +_Chutzpah* + +# Visual C++ cache files +ipch/ +*.aps +*.ncb +*.opendb +*.opensdf +*.sdf +*.cachefile +*.VC.db +*.VC.VC.opendb + +# Visual Studio profiler +*.psess +*.vsp +*.vspx +*.sap + +# TFS 2012 Local Workspace +$tf/ + +# Guidance Automation Toolkit +*.gpState + +# ReSharper is a .NET coding add-in +_ReSharper*/ +*.[Rr]e[Ss]harper +*.DotSettings.user + +# JustCode is a .NET coding add-in +.JustCode + +# TeamCity is a build add-in +_TeamCity* + +# DotCover is a Code Coverage Tool +*.dotCover + +# Visual Studio code coverage results +*.coverage +*.coveragexml + +# NCrunch +_NCrunch_* +.*crunch*.local.xml +nCrunchTemp_* + +# MightyMoose +*.mm.* +AutoTest.Net/ + +# Web workbench (sass) +.sass-cache/ + +# Installshield output folder +[Ee]xpress/ + +# DocProject is a documentation generator add-in +DocProject/buildhelp/ +DocProject/Help/*.HxT +DocProject/Help/*.HxC +DocProject/Help/*.hhc +DocProject/Help/*.hhk +DocProject/Help/*.hhp +DocProject/Help/Html2 +DocProject/Help/html + +# Click-Once directory +publish/ + +# Publish Web Output +*.[Pp]ublish.xml +*.azurePubxml +# Note: Comment the next line if you want to checkin your web deploy settings, +# but database connection strings (with potential passwords) will be unencrypted +*.pubxml +*.publishproj + +# Microsoft Azure Web App publish settings. Comment the next line if you want to +# checkin your Azure Web App publish settings, but sensitive information contained +# in these scripts will be unencrypted +PublishScripts/ + +# NuGet Packages +*.nupkg +# The packages folder can be ignored because of Package Restore +**/packages/* +# except build/, which is used as an MSBuild target. +!**/packages/build/ +# Uncomment if necessary however generally it will be regenerated when needed +#!**/packages/repositories.config +# NuGet v3's project.json files produces more ignorable files +*.nuget.props +*.nuget.targets + +# Microsoft Azure Build Output +csx/ +*.build.csdef + +# Microsoft Azure Emulator +ecf/ +rcf/ + +# Windows Store app package directories and files +AppPackages/ +BundleArtifacts/ +Package.StoreAssociation.xml +_pkginfo.txt +*.appx + +# Visual Studio cache files +# files ending in .cache can be ignored +*.[Cc]ache +# but keep track of directories ending in .cache +!*.[Cc]ache/ + +# Others +ClientBin/ +~$* +*~ +*.dbmdl +*.dbproj.schemaview +*.jfm +*.pfx +*.publishsettings +orleans.codegen.cs + +# Since there are multiple workflows, uncomment next line to ignore bower_components +# (https://github.com/github/gitignore/pull/1529#issuecomment-104372622) +#bower_components/ + +# RIA/Silverlight projects +Generated_Code/ + +# Backup & report files from converting an old project file +# to a newer Visual Studio version. Backup files are not needed, +# because we have git ;-) +_UpgradeReport_Files/ +Backup*/ +UpgradeLog*.XML +UpgradeLog*.htm + +# SQL Server files +*.mdf +*.ldf +*.ndf + +# Business Intelligence projects +*.rdl.data +*.bim.layout +*.bim_*.settings + +# Microsoft Fakes +FakesAssemblies/ + +# GhostDoc plugin setting file +*.GhostDoc.xml + +# Node.js Tools for Visual Studio +.ntvs_analysis.dat +node_modules/ + +# Typescript v1 declaration files +typings/ + +# Visual Studio 6 build log +*.plg + +# Visual Studio 6 workspace options file +*.opt + +# Visual Studio 6 auto-generated workspace file (contains which files were open etc.) +*.vbw + +# Visual Studio LightSwitch build output +**/*.HTMLClient/GeneratedArtifacts +**/*.DesktopClient/GeneratedArtifacts +**/*.DesktopClient/ModelManifest.xml +**/*.Server/GeneratedArtifacts +**/*.Server/ModelManifest.xml +_Pvt_Extensions + +# Paket dependency manager +.paket/paket.exe +paket-files/ + +# FAKE - F# Make +.fake/ + +# JetBrains Rider +.idea/ +*.sln.iml + +# CodeRush +.cr/ + +# Python Tools for Visual Studio (PTVS) +__pycache__/ +*.pyc + +# Cake - Uncomment if you are using it +# tools/** +# !tools/packages.config + +# Tabs Studio +*.tss + +# Telerik's JustMock configuration file +*.jmconfig + +# BizTalk build output +*.btp.cs +*.btm.cs +*.odx.cs +*.xsd.cs \ No newline at end of file diff --git a/LICENSE.TXT b/LICENSE.TXT new file mode 100644 index 00000000..d4d316ef --- /dev/null +++ b/LICENSE.TXT @@ -0,0 +1,28 @@ +Copyright (c) .NET Foundation and contributors. +This software is released under the Microsoft Reciprocal License (MS-RL) (the "License"); you may not use the software except in compliance with the License. + +The text of the Microsoft Reciprocal License (MS-RL) can be found online at: + http://opensource.org/licenses/ms-rl + + +Microsoft Reciprocal License (MS-RL) + +This license governs use of the accompanying software. If you use the software, you accept this license. If you do not accept the license, do not use the software. + +1. Definitions + The terms "reproduce," "reproduction," "derivative works," and "distribution" have the same meaning here as under U.S. copyright law. + A "contribution" is the original software, or any additions or changes to the software. + A "contributor" is any person that distributes its contribution under this license. + "Licensed patents" are a contributor's patent claims that read directly on its contribution. + +2. Grant of Rights + (A) Copyright Grant- Subject to the terms of this license, including the license conditions and limitations in section 3, each contributor grants you a non-exclusive, worldwide, royalty-free copyright license to reproduce its contribution, prepare derivative works of its contribution, and distribute its contribution or any derivative works that you create. + (B) Patent Grant- Subject to the terms of this license, including the license conditions and limitations in section 3, each contributor grants you a non-exclusive, worldwide, royalty-free license under its licensed patents to make, have made, use, sell, offer for sale, import, and/or otherwise dispose of its contribution in the software or derivative works of the contribution in the software. + +3. Conditions and Limitations + (A) Reciprocal Grants- For any file you distribute that contains code from the software (in source code or binary format), you must provide recipients the source code to that file along with a copy of this license, which license will govern that file. You may license other files that are entirely your own work and do not contain code from the software under any terms you choose. + (B) No Trademark License- This license does not grant you rights to use any contributors' name, logo, or trademarks. + (C) If you bring a patent claim against any contributor over patents that you claim are infringed by the software, your patent license from such contributor to the software ends automatically. + (D) If you distribute any portion of the software, you must retain all copyright, patent, trademark, and attribution notices that are present in the software. + (E) If you distribute any portion of the software in source code form, you may do so only under this license by including a complete copy of this license with your distribution. If you distribute any portion of the software in compiled or object code form, you may only do so under a license that complies with this license. + (F) The software is licensed "as-is." You bear the risk of using it. The contributors give no express warranties, guarantees or conditions. You may have additional consumer rights under your local laws which this license cannot change. To the extent permitted under your local laws, the contributors exclude the implied warranties of merchantability, fitness for a particular purpose and non-infringement. diff --git a/appveyor.cmd b/appveyor.cmd new file mode 100644 index 00000000..bb3accd1 --- /dev/null +++ b/appveyor.cmd @@ -0,0 +1,5 @@ +@setlocal +@pushd %~dp0 + +@popd +@endlocal \ No newline at end of file diff --git a/appveyor.yml b/appveyor.yml new file mode 100644 index 00000000..432653c6 --- /dev/null +++ b/appveyor.yml @@ -0,0 +1,27 @@ +image: Visual Studio 2017 + +version: 0.0.0.{build} +configuration: Release + +environment: + DOTNET_SKIP_FIRST_TIME_EXPERIENCE: true + DOTNET_CLI_TELEMETRY_OPTOUT: 1 + NUGET_XMLDOC_MODE: skip + +before_build: + - nuget restore + +build_script: + - appveyor.cmd + +pull_requests: + do_not_increment_build_number: true + +nuget: + disable_publish_on_pr: true + +skip_tags: true + +artifacts: +- path: build\Release\**\*.nupkg + name: nuget diff --git a/nuget.config b/nuget.config new file mode 100644 index 00000000..790be2b0 --- /dev/null +++ b/nuget.config @@ -0,0 +1,8 @@ + + + + + + + + \ No newline at end of file diff --git a/src/Cpp.Build.props b/src/Cpp.Build.props new file mode 100644 index 00000000..1e4d4cbc --- /dev/null +++ b/src/Cpp.Build.props @@ -0,0 +1,100 @@ + + + + + + $(OutputPath) + $(BaseIntermediateOutputPath)$(Platform)\ + $(OutputPath)$(Platform)\ + + + + + $(DisableSpecificCompilerWarnings) + Level4 + $(ProjectDir)inc;$(MSBuildProjectDirectory);$(IntDir);$(SqlCESdkIncludePath);$(ProjectAdditionalIncludeDirectories);%(AdditionalIncludeDirectories) + WIN32;_WINDOWS;_WIN32_MSI=500;_WIN32_WINNT=0x0501;$(ArmPreprocessorDefinitions);$(UnicodePreprocessorDefinitions);_CRT_STDIO_LEGACY_WIDE_SPECIFIERS;_WINSOCK_DEPRECATED_NO_WARNINGS;%(PreprocessorDefinitions) + Use + precomp.h + StdCall + true + false + -YlprecompDefine + /Zc:threadSafeInit- %(AdditionalOptions) + true + + + $(ArmPreprocessorDefinitions);%(PreprocessorDefinitions) + $(ProjectAdditionalResourceIncludeDirectories);%(AdditionalIncludeDirectories) + + + $(OutDir);$(AdditionalMultiTargetLibraryPath);$(ProjectAdditionalLibraryDirectories);%(AdditionalLibraryDirectories) + + + $(ProjectSubSystem) + $(ProjectModuleDefinitionFile) + $(ResourceOnlyDll) + true + $(ProjectAdditionalLinkLibraries);advapi32.lib;comdlg32.lib;user32.lib;oleaut32.lib;gdi32.lib;shell32.lib;ole32.lib;version.lib;%(AdditionalDependencies) + $(OutDir);$(AdditionalMultiTargetLibraryPath);$(ArmLibraryDirectories);$(ProjectAdditionalLinkLibraryDirectories);%(AdditionalLibraryDirectories) + /IGNORE:4099 %(AdditionalOptions) + + + + + + NoExtensions + + + + + CDecl + + + + + OldStyle + true + true + + + + + Disabled + EnableFastChecks + _DEBUG;DEBUG;%(PreprocessorDefinitions) + MultiThreadedDebug + + + + + + MultiThreadedDebugDll + + + + + MinSpace + NDEBUG;%(PreprocessorDefinitions) + true + true + MultiThreaded + + + true + true + + + + + + MultiThreadedDll + + + + + $(LinkKeyFile) + $(LinkDelaySign) + + + diff --git a/src/Directory.Build.props b/src/Directory.Build.props new file mode 100644 index 00000000..48ba462d --- /dev/null +++ b/src/Directory.Build.props @@ -0,0 +1,21 @@ + + + + + + Debug + $(MSBuildThisFileDirectory)..\build\obj\$(MSBuildProjectName)\ + $(MSBuildThisFileDirectory)..\build\$(Configuration)\ + + WiX Toolset Team + WiX Toolset + Copyright (c) .NET Foundation and contributors. All rights reserved. + + + + $(MSBuildThisFileDirectory)..\..\ + + + + + diff --git a/version.json b/version.json new file mode 100644 index 00000000..5f857771 --- /dev/null +++ b/version.json @@ -0,0 +1,11 @@ +{ + "version": "4.0", + "publicReleaseRefSpec": [ + "^refs/heads/master$" + ], + "cloudBuild": { + "buildNumber": { + "enabled": true + } + } +} -- cgit v1.2.3-55-g6feb From 088dc648a3478e2cacdbdab1cb1782556642ee69 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Sun, 3 Sep 2017 16:51:39 -0700 Subject: Initial commit --- WixToolset.Core.Native.sln | 42 +++ appveyor.cmd | 5 + src/Cpp.Build.props | 1 + src/Directory.Build.props | 1 + src/WixToolset.Core.Native/CabInterop.cs | 312 +++++++++++++++++++++ .../WixToolset.Core.Native.csproj | 23 ++ .../WixToolset.Core.Native.nuspec | 24 ++ src/winterop/packages.config | 5 + src/winterop/precomp.h | 12 + .../runtime.win-xxx.WixToolset.Core.Native.nuspec | 19 ++ src/winterop/winterop.cpp | 216 ++++++++++++++ src/winterop/winterop.def | 18 ++ src/winterop/winterop.vcxproj | 79 ++++++ 13 files changed, 757 insertions(+) create mode 100644 WixToolset.Core.Native.sln create mode 100644 src/WixToolset.Core.Native/CabInterop.cs create mode 100644 src/WixToolset.Core.Native/WixToolset.Core.Native.csproj create mode 100644 src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec create mode 100644 src/winterop/packages.config create mode 100644 src/winterop/precomp.h create mode 100644 src/winterop/runtime.win-xxx.WixToolset.Core.Native.nuspec create mode 100644 src/winterop/winterop.cpp create mode 100644 src/winterop/winterop.def create mode 100644 src/winterop/winterop.vcxproj diff --git a/WixToolset.Core.Native.sln b/WixToolset.Core.Native.sln new file mode 100644 index 00000000..2252fdf1 --- /dev/null +++ b/WixToolset.Core.Native.sln @@ -0,0 +1,42 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio 15 +VisualStudioVersion = 15.0.26730.12 +MinimumVisualStudioVersion = 15.0.26124.0 +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "winterop", "src\winterop\winterop.vcxproj", "{26D45E58-E703-431D-B67E-493C72C9DA0B}" +EndProject +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "WixToolset.Core.Native", "src\WixToolset.Core.Native\WixToolset.Core.Native.csproj", "{C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}" + ProjectSection(ProjectDependencies) = postProject + {26D45E58-E703-431D-B67E-493C72C9DA0B} = {26D45E58-E703-431D-B67E-493C72C9DA0B} + EndProjectSection +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Any CPU = Debug|Any CPU + Debug|x86 = Debug|x86 + Release|Any CPU = Release|Any CPU + Release|x86 = Release|x86 + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {26D45E58-E703-431D-B67E-493C72C9DA0B}.Debug|Any CPU.ActiveCfg = Debug|Win32 + {26D45E58-E703-431D-B67E-493C72C9DA0B}.Debug|x86.ActiveCfg = Debug|Win32 + {26D45E58-E703-431D-B67E-493C72C9DA0B}.Debug|x86.Build.0 = Debug|Win32 + {26D45E58-E703-431D-B67E-493C72C9DA0B}.Release|Any CPU.ActiveCfg = Release|Win32 + {26D45E58-E703-431D-B67E-493C72C9DA0B}.Release|x86.ActiveCfg = Release|Win32 + {26D45E58-E703-431D-B67E-493C72C9DA0B}.Release|x86.Build.0 = Release|Win32 + {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|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|x86.ActiveCfg = Release|Any CPU + {C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}.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/appveyor.cmd b/appveyor.cmd index bb3accd1..364d2ccd 100644 --- a/appveyor.cmd +++ b/appveyor.cmd @@ -1,5 +1,10 @@ @setlocal @pushd %~dp0 +msbuild -p:Configuration=Release;Platform=Win32 -t:PackNativeNuget src\winterop\winterop.vcxproj +msbuild -p:Configuration=Release;Platform=x64 -t:PackNativeNuget src\winterop\winterop.vcxproj + +dotnet pack -c Release .\src\WixToolset.Core.Native\WixToolset.Core.Native.csproj + @popd @endlocal \ No newline at end of file diff --git a/src/Cpp.Build.props b/src/Cpp.Build.props index 1e4d4cbc..453aa442 100644 --- a/src/Cpp.Build.props +++ b/src/Cpp.Build.props @@ -3,6 +3,7 @@ + Win32 $(OutputPath) $(BaseIntermediateOutputPath)$(Platform)\ $(OutputPath)$(Platform)\ diff --git a/src/Directory.Build.props b/src/Directory.Build.props index 48ba462d..63ad5d6e 100644 --- a/src/Directory.Build.props +++ b/src/Directory.Build.props @@ -4,6 +4,7 @@ Debug + AnyCPU $(MSBuildThisFileDirectory)..\build\obj\$(MSBuildProjectName)\ $(MSBuildThisFileDirectory)..\build\$(Configuration)\ diff --git a/src/WixToolset.Core.Native/CabInterop.cs b/src/WixToolset.Core.Native/CabInterop.cs new file mode 100644 index 00000000..6ad11c67 --- /dev/null +++ b/src/WixToolset.Core.Native/CabInterop.cs @@ -0,0 +1,312 @@ +// 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.Diagnostics.CodeAnalysis; + using System.Text; + using System.Runtime.InteropServices; + // using WixToolset.Msi; + // using WixToolset.Msi.Interop; + + /// + /// The native methods. + /// + public sealed class NativeMethods + { + /// + /// Starts creating a cabinet. + /// + /// Name of cabinet to create. + /// Directory to create cabinet in. + /// Maximum number of files that will be added to cabinet. + /// Maximum size of the cabinet. + /// Maximum threshold in the cabinet. + /// Type of compression to use in the cabinet. + /// Handle to opened cabinet. + [DllImport("winterop.dll", EntryPoint = "CreateCabBegin", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void CreateCabBegin(string cabinetName, string cabinetDirectory, uint maxFiles, uint maxSize, uint maxThreshold, uint compressionType, out IntPtr contextHandle); + + /// + /// Adds a file to an open cabinet. + /// + /// Full path to file to add to cabinet. + /// Name of file in cabinet. + /// Handle to open cabinet. + // [DllImport("winterop.dll", EntryPoint = "CreateCabAddFile", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + // public static extern void CreateCabAddFile(string file, string token, MsiInterop.MSIFILEHASHINFO fileHash, IntPtr contextHandle); + + /// + /// Closes a cabinet. + /// + /// Handle to open cabinet to close. + /// Address of Binder's cabinet split callback + [DllImport("winterop.dll", EntryPoint = "CreateCabFinish", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void CreateCabFinish(IntPtr contextHandle, IntPtr newCabNamesCallBackAddress); + + /// + /// Cancels cabinet creation. + /// + /// Handle to open cabinet to cancel. + [DllImport("winterop.dll", EntryPoint = "CreateCabCancel", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void CreateCabCancel(IntPtr contextHandle); + + /// + /// Initializes cabinet extraction. + /// + [DllImport("winterop.dll", EntryPoint = "ExtractCabBegin", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void ExtractCabBegin(); + + /// + /// Extracts files from cabinet. + /// + /// Path to cabinet to extract files from. + /// Directory to extract files to. + [DllImport("winterop.dll", EntryPoint = "ExtractCab", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true, PreserveSig = false)] + public static extern void ExtractCab(string cabinet, string extractDirectory); + + /// + /// Cleans up after cabinet extraction. + /// + [DllImport("winterop.dll", EntryPoint = "ExtractCabFinish", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true)] + public static extern void ExtractCabFinish(); + + /// + /// Initializes cabinet enumeration. + /// + [DllImport("winterop.dll", EntryPoint = "EnumerateCabBegin", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void EnumerateCabBegin(); + + /// + /// Enumerates files from cabinet. + /// + /// Path to cabinet to enumerate files from. + /// callback that gets each file. + [DllImport("winterop.dll", EntryPoint = "EnumerateCab", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true, PreserveSig = false)] + public static extern void EnumerateCab(string cabinet, CabInterop.PFNNOTIFY notify); + + /// + /// Cleans up after cabinet enumeration. + /// + [DllImport("winterop.dll", EntryPoint = "EnumerateCabFinish", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true)] + public static extern void EnumerateCabFinish(); + + /// + /// Resets the DACL on an array of files to "empty". + /// + /// Array of file reset ACL to "empty". + /// Number of file paths in array. + [DllImport("winterop.dll", EntryPoint = "ResetAcls", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void ResetAcls(string[] files, uint fileCount); + + /// + /// Gets the hash of the pCertContext->pCertInfo->SubjectPublicKeyInfo using ::CryptHashPublicKeyInfo() which does not seem + /// to be exposed by .NET Frameowkr. + /// + /// Pointer to a CERT_CONTEXT struct with public key information to hash. + /// Number of file paths in array. + [DllImport("winterop.dll", EntryPoint = "HashPublicKeyInfo", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void HashPublicKeyInfo(IntPtr certContext, byte[] publicKeyInfoHashed, ref uint sizePublicKeyInfoHashed); + + /// + /// Converts file time to a local file time. + /// + /// file time + /// local file time + /// true if successful, false otherwise + [DllImport("kernel32.dll", SetLastError = true)] + [return: MarshalAs(UnmanagedType.Bool)] + public 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)] + [return: MarshalAs(UnmanagedType.Bool)] + public static extern bool FileTimeToDosDateTime(ref long fileTime, out ushort wFatDate, out ushort wFatTime); + } + + /// + /// Interop class for the winterop.dll. + /// + public static class CabInterop + { + /// + /// Delegate type that's called by cabinet api for every file in cabinet. + /// + /// NOTIFICATIONTYPE + /// NOTIFICATION + /// 0 for success, -1 otherwise + public delegate Int32 PFNNOTIFY(NOTIFICATIONTYPE fdint, NOTIFICATION pfdin); + + /// + /// Wraps FDINOTIFICATIONTYPE. + /// + public enum NOTIFICATIONTYPE : int + { + /// Info about the cabinet. + CABINET_INFO, + /// One or more files are continued. + PARTIAL_FILE, + /// Called for each file in cabinet. + COPY_FILE, + /// Called after all of the data has been written to a target file. + CLOSE_FILE_INFO, + /// A file is continued to the next cabinet. + NEXT_CABINET, + /// Called once after a call to FDICopy() starts scanning a CAB's CFFILE entries, and again when there are no more CFFILE entries. + ENUMERATE, + } + + /// + /// 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; + NativeMethods.FileTimeToLocalFileTime(ref filetime, ref localTime); + NativeMethods.FileTimeToDosDateTime(ref localTime, out cabDate, out cabTime); + } + + /// + /// Wraps FDINOTIFICATION. + /// + [SuppressMessage("Microsoft.Performance", "CA1812:AvoidUninstantiatedInternalClasses")] + [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)] + public class NOTIFICATION + { + private int cb; + [MarshalAs(UnmanagedType.LPStr)] + private string psz1; + [MarshalAs(UnmanagedType.LPStr)] + private string psz2; + [MarshalAs(UnmanagedType.LPStr)] + private string psz3; + private IntPtr pv; + + private IntPtr hf; + + private ushort date; + private ushort time; + private ushort attribs; + private ushort setID; + private ushort cabinet; + private ushort folder; + private int fdie; + + /// + /// Uncompressed size of file. + /// + public int Cb + { + get { return this.cb; } + } + + /// + /// File name in cabinet. + /// + public String Psz1 + { + get { return this.psz1; } + } + + /// + /// Name of next disk. + /// + public string Psz2 + { + get { return this.psz2; } + } + + /// + /// Points to a 256 character buffer. + /// + public string Psz3 + { + get { return this.psz3; } + } + + /// + /// Value for client. + /// + public IntPtr Pv + { + get { return this.pv; } + } + + /// + /// Not used. + /// + public Int32 Hf + { + get { return (Int32)this.hf; } + } + + /// + /// Last modified MS-DOS date. + /// + public ushort Date + { + get { return this.date; } + } + + /// + /// Last modified MS-DOS time. + /// + public ushort Time + { + get { return this.time; } + } + + /// + /// File attributes. + /// + public ushort Attribs + { + get { return this.attribs; } + } + + /// + /// Cabinet set ID (a random 16-bit number). + /// + public ushort SetID + { + get { return this.setID; } + } + + /// + /// Cabinet number within cabinet set (0-based). + /// + public ushort Cabinet + { + get { return this.cabinet; } + } + + /// + /// File's folder index. + /// + public ushort Folder + { + get { return this.folder; } + } + + /// + /// Error code. + /// + public int Fdie + { + get { return this.fdie; } + } + } + } +} diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj new file mode 100644 index 00000000..858b9be2 --- /dev/null +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -0,0 +1,23 @@ + + + + netstandard2.0 + $(MSBuildThisFileName).nuspec + + + + + + + + + + + $(OutputPath) + Configuration=$(Configuration);Id=$(MSBuildThisFileName);Version=$(BuildVersionSimple);Authors=$(Authors);Copyright=$(Copyright);Description=$(Description) + + + diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec new file mode 100644 index 00000000..68d154c8 --- /dev/null +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -0,0 +1,24 @@ + + + + $id$ + $version$ + $authors$ + $authors$ + https://github.com/wixtoolset/Core.Native/blob/master/LICENSE.TXT + https://github.com/wixtoolset/Core.Native + false + $description$ + $copyright$ + + + + + + + + + + + + diff --git a/src/winterop/packages.config b/src/winterop/packages.config new file mode 100644 index 00000000..b11fe210 --- /dev/null +++ b/src/winterop/packages.config @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/src/winterop/precomp.h b/src/winterop/precomp.h new file mode 100644 index 00000000..eba996c7 --- /dev/null +++ b/src/winterop/precomp.h @@ -0,0 +1,12 @@ +#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 "dutil.h" +#include "fileutil.h" +#include "memutil.h" +#include "strutil.h" +#include "cabcutil.h" +#include "cabutil.h" diff --git a/src/winterop/runtime.win-xxx.WixToolset.Core.Native.nuspec b/src/winterop/runtime.win-xxx.WixToolset.Core.Native.nuspec new file mode 100644 index 00000000..87cf0919 --- /dev/null +++ b/src/winterop/runtime.win-xxx.WixToolset.Core.Native.nuspec @@ -0,0 +1,19 @@ + + + + $id$ + $version$ + $authors$ + $authors$ + https://github.com/wixtoolset/Core.Native/blob/master/LICENSE.TXT + https://github.com/wixtoolset/Core.Native + false + $description$ + $copyright$ + + + + + + + diff --git a/src/winterop/winterop.cpp b/src/winterop/winterop.cpp new file mode 100644 index 00000000..12d8ca3f --- /dev/null +++ b/src/winterop/winterop.cpp @@ -0,0 +1,216 @@ +// 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 HashPublicKeyInfo( + __in PCERT_CONTEXT pCertContext, + __in_ecount(*pcbSubjectKeyIndentifier) BYTE* rgbSubjectKeyIdentifier, + __inout DWORD* pcbSubjectKeyIndentifier + ) +{ + HRESULT hr = S_OK; + + if (!::CryptHashPublicKeyInfo(NULL, CALG_SHA1, 0, X509_ASN_ENCODING, &pCertContext->pCertInfo->SubjectPublicKeyInfo, rgbSubjectKeyIdentifier, pcbSubjectKeyIndentifier)) + { + ExitWithLastError(hr, "Failed to hash public key information."); + } + +LExit: + return hr; +} + +HRESULT ResetAcls( + __in LPCWSTR pwzFiles[], + __in DWORD cFiles + ) +{ + HRESULT hr = S_OK; + ACL* pacl = NULL; + DWORD cbAcl = sizeof(ACL); + + OSVERSIONINFO osvi; + + osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO); + if (!::GetVersionExA(&osvi)) + { + ExitOnLastError(hr, "failed to get OS version"); + } + + // If we're running on NT 4 or earlier, or ME or earlier, don't reset ACLs. + if (4 >= osvi.dwMajorVersion) + { + ExitFunction1(hr = S_FALSE); + } + + // create an empty (not NULL!) ACL to use on all the files + pacl = static_cast(MemAlloc(cbAcl, FALSE)); + ExitOnNull(pacl, hr, E_OUTOFMEMORY, "failed to allocate ACL"); + +#pragma prefast(push) +#pragma prefast(disable:25029) + if (!::InitializeAcl(pacl, cbAcl, ACL_REVISION)) +#pragma prefast(op) + { + ExitOnLastError(hr, "failed to initialize ACL"); + } + + // reset the existing security permissions on each file + for (DWORD i = 0; i < cFiles; ++i) + { + hr = ::SetNamedSecurityInfoW(const_cast(pwzFiles[i]), 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) + { + ExitOnFailure(hr = HRESULT_FROM_WIN32(hr), "failed to set security descriptor for file: %S", pwzFiles[i]); + } + } + + // Setting to S_OK because we could end with ERROR_FILE_NOT_FOUND or ERROR_PATH_NOT_FOUND as valid return values. + hr = S_OK; + + AssertSz(::IsValidAcl(pacl), "ResetAcls() - created invalid ACL"); + +LExit: + if (pacl) + { + MemFree(pacl); + } + + return hr; +} + + +HRESULT CreateCabBegin( + __in LPCWSTR wzCab, + __in LPCWSTR wzCabDir, + __in DWORD dwMaxFiles, + __in DWORD dwMaxSize, + __in DWORD dwMaxThresh, + __in COMPRESSION_TYPE ct, + __out HANDLE *phContext + ) +{ + return CabCBegin(wzCab, wzCabDir, dwMaxFiles, dwMaxSize, dwMaxThresh, ct, phContext); +} + + +HRESULT CreateCabAddFile( + __in LPCWSTR wzFile, + __in_opt LPCWSTR wzToken, + __in_opt PMSIFILEHASHINFO pmfHash, + __in HANDLE hContext + ) +{ + return CabCAddFile(wzFile, wzToken, pmfHash, hContext); +} + + +HRESULT CreateCabAddFiles( + __in LPCWSTR pwzFiles[], + __in LPCWSTR pwzTokens[], + __in PMSIFILEHASHINFO pmfHash[], + __in DWORD cFiles, + __in HANDLE hContext + ) +{ + HRESULT hr = S_OK; + DWORD i; + + Assert(pwzFiles); + Assert(hContext); + + for (i = 0; i < cFiles; i++) + { + hr = CreateCabAddFile( + pwzFiles[i], + pwzTokens ? pwzTokens[i] : NULL, + pmfHash[i], + hContext + ); + ExitOnFailure(hr, "Failed to add file %S to cab", pwzFiles[i]); + } + +LExit: + return hr; +} + + +HRESULT CreateCabFinish( + __in HANDLE hContext, + __in_opt FileSplitCabNamesCallback newCabNamesCallBackAddress + ) +{ + // Convert address into Binder callback function + return CabCFinish(hContext, newCabNamesCallBackAddress); +} + + +void CreateCabCancel( + __in HANDLE hContext + ) +{ + CabCCancel(hContext); +} + + +HRESULT ExtractCabBegin() +{ + return CabInitialize(FALSE); +} + + +HRESULT ExtractCab( + __in LPCWSTR wzCabinet, + __in LPCWSTR wzExtractDir + ) +{ + return CabExtract(wzCabinet, L"*", wzExtractDir, NULL, NULL, 0); +} + + +void ExtractCabFinish() +{ + CabUninitialize(); + return; +} + + +HRESULT EnumerateCabBegin() +{ + return CabInitialize(FALSE); +} + + +HRESULT EnumerateCab( + __in LPCWSTR wzCabinet, + __in STDCALL_PFNFDINOTIFY pfnNotify + ) +{ + return CabEnumerate(wzCabinet, L"*", pfnNotify, 0); +} + + +void EnumerateCabFinish() +{ + CabUninitialize(); + return; +} + + +BOOL WINAPI DllMain( + __in HINSTANCE /*hInstance*/, + __in DWORD dwReason, + __in LPVOID /*lpvReserved*/ + ) +{ + switch(dwReason) + { + case DLL_PROCESS_ATTACH: + case DLL_PROCESS_DETACH: + case DLL_THREAD_ATTACH: + case DLL_THREAD_DETACH: + break; + } + + return TRUE; +} diff --git a/src/winterop/winterop.def b/src/winterop/winterop.def new file mode 100644 index 00000000..dffa6268 --- /dev/null +++ b/src/winterop/winterop.def @@ -0,0 +1,18 @@ +; Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +LIBRARY "winterop.dll" + +EXPORTS + CreateCabBegin + CreateCabCancel + CreateCabAddFile + CreateCabAddFiles + CreateCabFinish + EnumerateCabBegin + EnumerateCab + EnumerateCabFinish + ExtractCabBegin + ExtractCab + ExtractCabFinish + ResetAcls + HashPublicKeyInfo diff --git a/src/winterop/winterop.vcxproj b/src/winterop/winterop.vcxproj new file mode 100644 index 00000000..6274a063 --- /dev/null +++ b/src/winterop/winterop.vcxproj @@ -0,0 +1,79 @@ + + + + + + + + + Debug + Win32 + + + Release + Win32 + + + Debug + x64 + + + Release + x64 + + + + + {26D45E58-E703-431D-B67E-493C72C9DA0B} + DynamicLibrary + winterop + v141 + MultiByte + winterop.def + Native component of WixToolset.Core + + + + + + + + + + + + + crypt32.lib;cabinet.lib;msi.lib + + + + + + 4996 + Create + + + + + + + + + + + + + + + + + + + + + This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}. + + + + + -- cgit v1.2.3-55-g6feb From 40d64f03447d8fa187e1d23addee931025bb6f93 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Sun, 17 Sep 2017 15:22:33 -0700 Subject: Update to latest GitVersioning --- src/WixToolset.Core.Native/WixToolset.Core.Native.csproj | 2 +- src/winterop/packages.config | 2 +- src/winterop/winterop.vcxproj | 4 ++-- 3 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index 858b9be2..3e66d84e 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -7,7 +7,7 @@ - + @@ -10,9 +11,26 @@ - + + + + + + + x86\ + + + x64\ + + + + PreserveNewest + %(TargetRelativeFolder)%(Filename)%(Extension) + + + + diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index 68d154c8..6a96167e 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -10,15 +10,20 @@ false $description$ $copyright$ - + + + + + diff --git a/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs b/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs new file mode 100644 index 00000000..baab3bee --- /dev/null +++ b/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs @@ -0,0 +1,115 @@ +// 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; + using System.IO; + using System.Linq; + using WixToolset.Core.Native; + using WixToolsetTest.CoreNative.Utility; + 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, CabinetCompressionLevel.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); + Assert.True(file.SameAsDateTime(new DateTime(2017, 9, 28, 0, 19, 38))); + } + + [Fact] + public void CanExtractSingleFileCabinet() + { + var cabinetPath = TestData.Get(@"TestData\test.cab"); + + using (var fs = new DisposableFileSystem()) + { + var extractFolder = fs.GetFolder(true); + + var cabinet = new Cabinet(cabinetPath); + cabinet.Extract(extractFolder); + + var files = Directory.EnumerateFiles(extractFolder); + + var file = new FileInfo(files.Single()); + CabInterop.DateTimeToCabDateAndTime(file.CreationTime, out var date, out var time); + + Assert.Equal("test.txt", file.Name); + Assert.Equal(17, file.Length); + Assert.Equal(19259, date); + Assert.Equal(47731, time); + } + } + + [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"), "test.txt") }; + + var cabinet = new Cabinet(cabinetPath); + cabinet.Compress(files, CabinetCompressionLevel.Low); + } + + // Extract. + { + var cabinet = new Cabinet(cabinetPath); + cabinet.Extract(extractFolder); + } + + // 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/TestData/test.cab b/src/test/WixToolsetTest.Core.Native/TestData/test.cab new file mode 100644 index 00000000..ca78f632 Binary files /dev/null and b/src/test/WixToolsetTest.Core.Native/TestData/test.cab differ diff --git a/src/test/WixToolsetTest.Core.Native/TestData/test.txt b/src/test/WixToolsetTest.Core.Native/TestData/test.txt new file mode 100644 index 00000000..cd0db0e1 --- /dev/null +++ b/src/test/WixToolsetTest.Core.Native/TestData/test.txt @@ -0,0 +1 @@ +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 new file mode 100644 index 00000000..c9957247 --- /dev/null +++ b/src/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/test/WixToolsetTest.Core.Native/Utility/Pushd.cs b/src/test/WixToolsetTest.Core.Native/Utility/Pushd.cs new file mode 100644 index 00000000..91700c2f --- /dev/null +++ b/src/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/test/WixToolsetTest.Core.Native/Utility/TestData.cs b/src/test/WixToolsetTest.Core.Native/Utility/TestData.cs new file mode 100644 index 00000000..cd9c6318 --- /dev/null +++ b/src/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/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj new file mode 100644 index 00000000..c7fd89ea --- /dev/null +++ b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj @@ -0,0 +1,24 @@ + + + + + + netcoreapp2.0 + false + + + + + + + + + + + + + + + + + diff --git a/src/wixnative/enumcab.cpp b/src/wixnative/enumcab.cpp new file mode 100644 index 00000000..e7717bac --- /dev/null +++ b/src/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/wixnative/extractcab.cpp b/src/wixnative/extractcab.cpp new file mode 100644 index 00000000..53f53266 --- /dev/null +++ b/src/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/wixnative/packages.config b/src/wixnative/packages.config new file mode 100644 index 00000000..02ee2250 --- /dev/null +++ b/src/wixnative/packages.config @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/src/wixnative/precomp.cpp b/src/wixnative/precomp.cpp new file mode 100644 index 00000000..37664a1c --- /dev/null +++ b/src/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/wixnative/precomp.h b/src/wixnative/precomp.h new file mode 100644 index 00000000..5bd617e5 --- /dev/null +++ b/src/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/wixnative/resetacls.cpp b/src/wixnative/resetacls.cpp new file mode 100644 index 00000000..8c5bdc56 --- /dev/null +++ b/src/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/wixnative/smartcab.cpp b/src/wixnative/smartcab.cpp new file mode 100644 index 00000000..da9087a3 --- /dev/null +++ b/src/wixnative/smartcab.cpp @@ -0,0 +1,157 @@ +// 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); + + 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 new file mode 100644 index 00000000..7bd8dbca --- /dev/null +++ b/src/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/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj new file mode 100644 index 00000000..2a4ce3d5 --- /dev/null +++ b/src/wixnative/wixnative.vcxproj @@ -0,0 +1,80 @@ + + + + + + + + + Debug + Win32 + + + Release + Win32 + + + Debug + x64 + + + Release + x64 + + + + + {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF} + Application + Console + wixnative + v141 + Unicode + Native component of WixToolset.Core + + + + + + + + + + + + + crypt32.lib;cabinet.lib;msi.lib + + + + + + 4996 + + + Create + + + + + + + + + + + + + + + + + + + + This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}. + + + + + -- cgit v1.2.3-55-g6feb From 6076d8c50fe5a43f272cc30fbf557fcd844e04e3 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Wed, 27 Dec 2017 14:11:39 -0800 Subject: Send build notifications to Slack --- appveyor.yml | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/appveyor.yml b/appveyor.yml index d9da1df5..0c74d54b 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -22,3 +22,8 @@ skip_tags: true artifacts: - path: build\Release\**\*.nupkg name: nuget + +notifications: +- provider: Slack + incoming_webhook: + secure: p5xuu+4x2JHfwGDMDe5KcG1k7gZxqYc4jWVwvyNZv5cvkubPD2waJs5yXMAXZNN7Z63/3PWHb7q4KoY/99AjauYa1nZ4c5qYqRPFRBKTHfA= -- cgit v1.2.3-55-g6feb From cb5d0691ce447193c9189b621fc2704d188bfd24 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Thu, 12 Jul 2018 22:12:20 -0700 Subject: Disable part of test that fails irregularly --- src/test/WixToolsetTest.Core.Native/CabinetFixture.cs | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs b/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs index baab3bee..bc8f0ff8 100644 --- a/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs +++ b/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs @@ -42,7 +42,8 @@ namespace WixToolsetTest.CoreNative Assert.Equal(19259, file.Date); Assert.Equal(47731, file.Time); - Assert.True(file.SameAsDateTime(new DateTime(2017, 9, 28, 0, 19, 38))); + // 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] -- cgit v1.2.3-55-g6feb From c080bd2aa54b22ae8f1c06803ba9397ff80df696 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Thu, 12 Jul 2018 22:19:02 -0700 Subject: Improve debuggability --- src/WixToolset.Core.Native/WixToolset.Core.Native.csproj | 2 ++ src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec | 1 - 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index aa87186b..80c9e372 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -4,10 +4,12 @@ netstandard2.0 $(MSBuildThisFileName).nuspec Core Native + embedded + diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index 6a96167e..88dd4d34 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -20,7 +20,6 @@ - -- cgit v1.2.3-55-g6feb From 15de67f41ad006b195ad2c3da4a91783ee244cee Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Thu, 12 Jul 2018 22:19:37 -0700 Subject: Update development dependencies --- src/WixToolset.Core.Native/WixToolset.Core.Native.csproj | 2 +- .../WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj | 6 +++--- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index 80c9e372..51238020 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -10,7 +10,7 @@ - + - - - + + + -- cgit v1.2.3-55-g6feb From 1c3d51462cc7574b318cba2f30552de50348f803 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Fri, 13 Jul 2018 14:15:40 -0700 Subject: Update development dependencies --- src/wixnative/packages.config | 4 ++-- src/wixnative/wixnative.vcxproj | 18 +++++++++--------- 2 files changed, 11 insertions(+), 11 deletions(-) diff --git a/src/wixnative/packages.config b/src/wixnative/packages.config index 02ee2250..20d71c08 100644 --- a/src/wixnative/packages.config +++ b/src/wixnative/packages.config @@ -1,5 +1,5 @@  - - + + \ No newline at end of file diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index 2a4ce3d5..cfec54e4 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -2,7 +2,7 @@ - + @@ -40,6 +40,7 @@ + @@ -54,10 +55,10 @@ Create - - - - + + + + @@ -69,12 +70,11 @@ - - This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}. + This project references NuGet package(s) that are missing on this computer. Enable NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105.The missing file is {0}. - - + + -- cgit v1.2.3-55-g6feb From e6897a667c06125beced3921b16cf0e3a1b21bc1 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Fri, 13 Jul 2018 14:19:31 -0700 Subject: Attempt to move wixnative.exe to independent runtime .nupkg --- appveyor.cmd | 2 ++ src/WixToolset.Core.Native/WixNativeExe.cs | 2 +- .../WixToolset.Core.Native.csproj | 4 ++-- .../WixToolset.Core.Native.nuspec | 8 ++++++-- .../runtime.win.WixToolset.Core.Native.nuspec | 21 +++++++++++++++++++++ src/wixnative/wixnative.vcxproj | 11 ++++++++++- 6 files changed, 42 insertions(+), 6 deletions(-) create mode 100644 src/wixnative/runtime.win.WixToolset.Core.Native.nuspec diff --git a/appveyor.cmd b/appveyor.cmd index 872f2c93..a39fa567 100644 --- a/appveyor.cmd +++ b/appveyor.cmd @@ -7,5 +7,7 @@ msbuild -p:Configuration=Release .\src\test\WixToolsetTest.Core.Native\WixToolse msbuild -t:Pack -p:Configuration=Release .\src\WixToolset.Core.Native\WixToolset.Core.Native.csproj +msbuild -t:Pack -p:Configuration=Release .\src\wixnative\wixnative.vcxproj + @popd @endlocal \ No newline at end of file diff --git a/src/WixToolset.Core.Native/WixNativeExe.cs b/src/WixToolset.Core.Native/WixNativeExe.cs index 8626bea3..92dcfdda 100644 --- a/src/WixToolset.Core.Native/WixNativeExe.cs +++ b/src/WixToolset.Core.Native/WixNativeExe.cs @@ -19,7 +19,7 @@ namespace WixToolset.Core.Native static WixNativeExe() { - PathToWixNativeExe = Path.Combine(Path.GetDirectoryName(new Uri(Assembly.GetExecutingAssembly().CodeBase).LocalPath), "x86\\wixnative.exe"); + PathToWixNativeExe = Path.Combine(Path.GetDirectoryName(new Uri(Assembly.GetExecutingAssembly().CodeBase).LocalPath), "wixnative.x86.exe"); } public WixNativeExe(params object[] args) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index 51238020..a95f8852 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -20,10 +20,10 @@ - x86\ + .\ - x64\ + .\ diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index 88dd4d34..3394f8ab 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -10,19 +10,23 @@ false $description$ $copyright$ - - --> + diff --git a/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec b/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec new file mode 100644 index 00000000..d6f122cc --- /dev/null +++ b/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec @@ -0,0 +1,21 @@ + + + + $id$ + $version$ + $authors$ + $authors$ + https://github.com/wixtoolset/Core.Native/blob/master/LICENSE.TXT + https://github.com/wixtoolset/Core.Native + false + $description$ + $copyright$ + + + + + + + + + diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index cfec54e4..d4b31fd8 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -23,11 +23,16 @@ + + x86 + amd64 + + {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF} Application Console - wixnative + wixnative.$(NameSuffix) v141 Unicode Native component of WixToolset.Core @@ -69,6 +74,10 @@ + + + + -- cgit v1.2.3-55-g6feb From 2db44de83b23f8363490e3a94e2d3a31b96fcd0e Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Fri, 13 Jul 2018 22:01:11 -0700 Subject: Attempt to use platform specific runtimes --- src/wixnative/runtime.win.WixToolset.Core.Native.nuspec | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec b/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec index d6f122cc..d2b9c1e7 100644 --- a/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec +++ b/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec @@ -13,9 +13,9 @@ - - - - + + + + -- cgit v1.2.3-55-g6feb From bf2a6a42ec4fb3296a291e996903def23d9d5b31 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Fri, 10 Aug 2018 23:30:15 -0700 Subject: Update test references and simplify BuildWixNative file gathering --- .../WixToolset.Core.Native.csproj | 29 +++++++++------------- .../WixToolsetTest.Core.Native.csproj | 6 ++--- 2 files changed, 15 insertions(+), 20 deletions(-) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index a95f8852..b02c00ad 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -1,4 +1,4 @@ - + netstandard2.0 @@ -9,32 +9,27 @@ - + - - - + + + + + + + - - .\ - - - .\ - - - + PreserveNewest - %(TargetRelativeFolder)%(Filename)%(Extension) + %(Filename)%(Extension) - + $(OutputPath) Configuration=$(Configuration);Id=$(MSBuildThisFileName);Version=$(BuildVersionSimple);Authors=$(Authors);Copyright=$(Copyright);Description=$(Description) diff --git a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj index d6cdead2..4e02073e 100644 --- a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj +++ b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj @@ -17,8 +17,8 @@ - - - + + + -- cgit v1.2.3-55-g6feb From 989686d2fdb3b46ed3c411f03ebdcdeeb9f42e15 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Wed, 3 Oct 2018 14:01:55 -0700 Subject: Add better error message when wixnative.exe cannot be found --- src/WixToolset.Core.Native/WixNativeExe.cs | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/src/WixToolset.Core.Native/WixNativeExe.cs b/src/WixToolset.Core.Native/WixNativeExe.cs index 92dcfdda..90ab0042 100644 --- a/src/WixToolset.Core.Native/WixNativeExe.cs +++ b/src/WixToolset.Core.Native/WixNativeExe.cs @@ -39,6 +39,11 @@ namespace WixToolset.Core.Native public IEnumerable Run() { + if (!File.Exists(PathToWixNativeExe)) + { + throw new FileNotFoundException($"Could not find internal piece of WiX Toolset at: {PathToWixNativeExe}", PathToWixNativeExe); + } + var wixNativeInfo = new ProcessStartInfo(PathToWixNativeExe, this.commandLine) { RedirectStandardInput = true, -- cgit v1.2.3-55-g6feb From 77b1c956e6037717f98ae01c2ff4e989c0252db8 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Fri, 10 Aug 2018 23:32:23 -0700 Subject: Enable use of NCrunch --- .gitignore | 58 ++++++++++++++++++---- src/Cpp.Build.props | 5 +- src/Directory.Build.props | 15 ++++-- .../WixToolset.Core.Native.csproj | 27 +++++++++- .../WixToolset.Core.Native.v3.ncrunchproject | 10 ++++ src/wixnative/wixnative.vcxproj | 2 +- 6 files changed, 98 insertions(+), 19 deletions(-) create mode 100644 src/WixToolset.Core.Native/WixToolset.Core.Native.v3.ncrunchproject diff --git a/.gitignore b/.gitignore index 3c6208a8..efc59600 100644 --- a/.gitignore +++ b/.gitignore @@ -19,16 +19,20 @@ [Rr]eleases/ x64/ x86/ +[Ww]in32/ bld/ [Bb]in/ [Oo]bj/ [Ll]og/ -# Visual Studio 2015 cache/options directory +# Visual Studio 2015/2017 cache/options directory .vs/ # Uncomment if you have tasks that create the project's static files in wwwroot #wwwroot/ +# Visual Studio 2017 auto generated files +Generated\ Files/ + # MSTest test Results [Tt]est[Rr]esult*/ [Bb]uild[Ll]og.* @@ -49,16 +53,21 @@ BenchmarkDotNet.Artifacts/ project.lock.json project.fragment.lock.json artifacts/ -**/Properties/launchSettings.json +# StyleCop +StyleCopReport.xml + +# Files built by Visual Studio *_i.c *_p.c -*_i.h +*_h.h *.ilk *.meta *.obj +*.iobj *.pch *.pdb +*.ipdb *.pgc *.pgd *.rsp @@ -96,6 +105,9 @@ ipch/ *.vspx *.sap +# Visual Studio Trace Files +*.e2e + # TFS 2012 Local Workspace $tf/ @@ -116,6 +128,10 @@ _TeamCity* # DotCover is a Code Coverage Tool *.dotCover +# AxoCover is a Code Coverage Tool +.axoCover/* +!.axoCover/settings.json + # Visual Studio code coverage results *.coverage *.coveragexml @@ -164,11 +180,11 @@ PublishScripts/ # NuGet Packages *.nupkg # The packages folder can be ignored because of Package Restore -**/packages/* +**/[Pp]ackages/* # except build/, which is used as an MSBuild target. -!**/packages/build/ +!**/[Pp]ackages/build/ # Uncomment if necessary however generally it will be regenerated when needed -#!**/packages/repositories.config +#!**/[Pp]ackages/repositories.config # NuGet v3's project.json files produces more ignorable files *.nuget.props *.nuget.targets @@ -205,6 +221,10 @@ ClientBin/ *.publishsettings orleans.codegen.cs +# Including strong name files can present a security risk +# (https://github.com/github/gitignore/pull/2483#issue-259490424) +#*.snk + # Since there are multiple workflows, uncomment next line to ignore bower_components # (https://github.com/github/gitignore/pull/1529#issuecomment-104372622) #bower_components/ @@ -219,6 +239,8 @@ _UpgradeReport_Files/ Backup*/ UpgradeLog*.XML UpgradeLog*.htm +ServiceFabricBackup/ +*.rptproj.bak # SQL Server files *.mdf @@ -229,6 +251,7 @@ UpgradeLog*.htm *.rdl.data *.bim.layout *.bim_*.settings +*.rptproj.rsuser # Microsoft Fakes FakesAssemblies/ @@ -240,9 +263,6 @@ FakesAssemblies/ .ntvs_analysis.dat node_modules/ -# Typescript v1 declaration files -typings/ - # Visual Studio 6 build log *.plg @@ -292,4 +312,22 @@ __pycache__/ *.btp.cs *.btm.cs *.odx.cs -*.xsd.cs \ No newline at end of file +*.xsd.cs + +# OpenCover UI analysis results +OpenCover/ + +# Azure Stream Analytics local run output +ASALocalRun/ + +# MSBuild Binary and Structured Log +*.binlog + +# NVidia Nsight GPU debugger configuration file +*.nvuser + +# MFractors (Xamarin productivity tool) working folder +.mfractor/ + +# Local History for Visual Studio +.localhistory/ diff --git a/src/Cpp.Build.props b/src/Cpp.Build.props index 453aa442..296b36ca 100644 --- a/src/Cpp.Build.props +++ b/src/Cpp.Build.props @@ -3,9 +3,8 @@ - Win32 - $(OutputPath) - $(BaseIntermediateOutputPath)$(Platform)\ + Win32 + $(BaseIntermediateOutputPath)$(Configuration)\$(Platform)\ $(OutputPath)$(Platform)\ diff --git a/src/Directory.Build.props b/src/Directory.Build.props index 63ad5d6e..9eacf3f5 100644 --- a/src/Directory.Build.props +++ b/src/Directory.Build.props @@ -1,16 +1,23 @@ - + Debug - AnyCPU - $(MSBuildThisFileDirectory)..\build\obj\$(MSBuildProjectName)\ - $(MSBuildThisFileDirectory)..\build\$(Configuration)\ + false + + $(MSBuildProjectName) + $(MSBuildThisFileDirectory)..\build\ + $(BaseOutputPath)obj\$(ProjectName)\ + $(BaseOutputPath)$(Configuration)\ WiX Toolset Team WiX Toolset Copyright (c) .NET Foundation and contributors. All rights reserved. + WiX Toolset diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index b02c00ad..e2431fef 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -13,7 +13,12 @@ - + + $(MSBuildProjectDir)..\..\build\obj\$(ProjectName)\$(Configuration)\NativeFileList.txt + $(NCrunchOriginalProjectDir)..\..\build\obj\$(ProjectName)\$(Configuration)\NativeFileList.txt + + + @@ -21,7 +26,11 @@ + + + + PreserveNewest %(Filename)%(Extension) @@ -29,6 +38,22 @@ + + + + + + + + + + + + + + + + $(OutputPath) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.v3.ncrunchproject b/src/WixToolset.Core.Native/WixToolset.Core.Native.v3.ncrunchproject new file mode 100644 index 00000000..0da1f42d --- /dev/null +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.v3.ncrunchproject @@ -0,0 +1,10 @@ + + + + ..\..\build\Debug\Win32\wixnative.x86.exe + ..\..\build\Debug\Win32\wixnative.x86.pdb + ..\..\build\Debug\x64\wixnative.amd64.exe + ..\..\build\Debug\x64\wixnative.amd64.pdb + + + \ No newline at end of file diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index d4b31fd8..420615ed 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -75,7 +75,7 @@ - + -- cgit v1.2.3-55-g6feb From c90ae13f242350786cf9103330c28e660d35eeb8 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Sat, 22 Dec 2018 09:10:56 -0800 Subject: Update to latest Home\repo-template --- .editorconfig | 37 +++++++++++++++++ .gitignore | 16 ++++++-- appveyor.yml | 6 +++ src/Directory.Build.props | 9 ++-- src/Directory.Build.targets | 48 ++++++++++++++++++++++ .../WixToolset.Core.Native.nuspec | 3 +- .../runtime.win-xxx.WixToolset.Core.Native.nuspec | 3 +- .../runtime.win.WixToolset.Core.Native.nuspec | 3 +- 8 files changed, 112 insertions(+), 13 deletions(-) create mode 100644 .editorconfig create mode 100644 src/Directory.Build.targets diff --git a/.editorconfig b/.editorconfig new file mode 100644 index 00000000..1d72e683 --- /dev/null +++ b/.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/.gitignore b/.gitignore index efc59600..3e8a1553 100644 --- a/.gitignore +++ b/.gitignore @@ -4,6 +4,7 @@ ## Get latest from https://github.com/github/gitignore/blob/master/VisualStudio.gitignore # User-specific files +*.rsuser *.suo *.user *.userosscache @@ -19,7 +20,8 @@ [Rr]eleases/ x64/ x86/ -[Ww]in32/ +[Aa][Rr][Mm]/ +[Aa][Rr][Mm]64/ bld/ [Bb]in/ [Oo]bj/ @@ -77,6 +79,7 @@ StyleCopReport.xml *.tlh *.tmp *.tmp_proj +*_wpftmp.csproj *.log *.vspscc *.vssscc @@ -208,7 +211,7 @@ _pkginfo.txt # files ending in .cache can be ignored *.[Cc]ache # but keep track of directories ending in .cache -!*.[Cc]ache/ +!?*.[Cc]ache/ # Others ClientBin/ @@ -228,6 +231,8 @@ orleans.codegen.cs # Since there are multiple workflows, uncomment next line to ignore bower_components # (https://github.com/github/gitignore/pull/1529#issuecomment-104372622) #bower_components/ +# ASP.NET Core default setup: bower directory is configured as wwwroot/lib/ and bower restore is true +**/wwwroot/lib/ # RIA/Silverlight projects Generated_Code/ @@ -291,8 +296,8 @@ paket-files/ .idea/ *.sln.iml -# CodeRush -.cr/ +# CodeRush personal settings +.cr/personal # Python Tools for Visual Studio (PTVS) __pycache__/ @@ -331,3 +336,6 @@ ASALocalRun/ # Local History for Visual Studio .localhistory/ + +# BeatPulse healthcheck temp database +healthchecksdb diff --git a/appveyor.yml b/appveyor.yml index 0c74d54b..d55322da 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -1,3 +1,8 @@ +# 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. + image: Visual Studio 2017 version: 0.0.0.{build} @@ -17,6 +22,7 @@ pull_requests: nuget: disable_publish_on_pr: true +skip_branch_with_pr: true skip_tags: true artifacts: diff --git a/src/Directory.Build.props b/src/Directory.Build.props index 9eacf3f5..e853e22d 100644 --- a/src/Directory.Build.props +++ b/src/Directory.Build.props @@ -1,7 +1,7 @@ @@ -10,20 +10,17 @@ false $(MSBuildProjectName) - $(MSBuildThisFileDirectory)..\build\ + $([System.IO.Path]::GetFullPath($(MSBuildThisFileDirectory)..\build\)) $(BaseOutputPath)obj\$(ProjectName)\ $(BaseOutputPath)$(Configuration)\ WiX Toolset Team WiX Toolset Copyright (c) .NET Foundation and contributors. All rights reserved. + MS-RL WiX Toolset - - $(MSBuildThisFileDirectory)..\..\ - - diff --git a/src/Directory.Build.targets b/src/Directory.Build.targets new file mode 100644 index 00000000..dac7452a --- /dev/null +++ b/src/Directory.Build.targets @@ -0,0 +1,48 @@ + + + + + + + true + $(SolutionPath) + $(NCrunchOriginalSolutionPath) + + + + + + + $([System.IO.File]::ReadAllText($(TheSolutionPath))) + $([System.IO.Path]::GetDirectoryName( $(TheSolutionPath) )) + (?<="[PackageName]", ")(.*)(?=", ") + + + + + + %(Identity) + $(SolutionFileContent.Contains('\%(Identity).csproj')) + + + + + $(RegexPattern.Replace('[PackageName]','%(PackageName)') ) + $([System.Text.RegularExpressions.Regex]::Match('$(SolutionFileContent)', '%(Pattern)')) + + + + + + + + + + + diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index 3394f8ab..25034078 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -5,7 +5,8 @@ $version$ $authors$ $authors$ - https://github.com/wixtoolset/Core.Native/blob/master/LICENSE.TXT + + https://licenses.nuget.org/MS-RL https://github.com/wixtoolset/Core.Native false $description$ diff --git a/src/winterop/runtime.win-xxx.WixToolset.Core.Native.nuspec b/src/winterop/runtime.win-xxx.WixToolset.Core.Native.nuspec index 87cf0919..18676197 100644 --- a/src/winterop/runtime.win-xxx.WixToolset.Core.Native.nuspec +++ b/src/winterop/runtime.win-xxx.WixToolset.Core.Native.nuspec @@ -5,7 +5,8 @@ $version$ $authors$ $authors$ - https://github.com/wixtoolset/Core.Native/blob/master/LICENSE.TXT + + https://licenses.nuget.org/MS-RL https://github.com/wixtoolset/Core.Native false $description$ diff --git a/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec b/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec index d2b9c1e7..f7697694 100644 --- a/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec +++ b/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec @@ -5,7 +5,8 @@ $version$ $authors$ $authors$ - https://github.com/wixtoolset/Core.Native/blob/master/LICENSE.TXT + + https://licenses.nuget.org/MS-RL https://github.com/wixtoolset/Core.Native false $description$ -- cgit v1.2.3-55-g6feb From f8b6183ed434ee9cb07d8adfa5ac660b1ae1cce3 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Sat, 22 Dec 2018 11:05:22 -0800 Subject: Include wixnative.exe in the Core.Native .nupkg This is done to place the wixnative.exe correctly for testing purposes. The dependency on runtime.win.WixToolset.Core.Native.nupkg is used to deliver wixnative.exe when publishing. --- src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec | 15 +++++++++------ 1 file changed, 9 insertions(+), 6 deletions(-) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index 25034078..195ea7b6 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -23,11 +23,14 @@ - + + + + + + -- cgit v1.2.3-55-g6feb From 22672837ce4248778cbe3ad0b0056c3998a33a84 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Mon, 24 Dec 2018 07:32:57 -0800 Subject: Move wixnative.*.exe in Core.Native.nupkg into arch folder Moving the native executable to a subdirectory in the Core.Native to avoid NuGet warning about non-managed code files in "lib" folder. Also, better support NCrunch by ignoring wixnative.exe --- src/WixToolset.Core.Native/WixNativeExe.cs | 36 ++++++++++++++++------ .../WixToolset.Core.Native.nuspec | 8 ++--- src/wixnative/wixnative.v3.ncrunchproject | 5 +++ 3 files changed, 35 insertions(+), 14 deletions(-) create mode 100644 src/wixnative/wixnative.v3.ncrunchproject diff --git a/src/WixToolset.Core.Native/WixNativeExe.cs b/src/WixToolset.Core.Native/WixNativeExe.cs index 90ab0042..3bd340bc 100644 --- a/src/WixToolset.Core.Native/WixNativeExe.cs +++ b/src/WixToolset.Core.Native/WixNativeExe.cs @@ -12,16 +12,12 @@ namespace WixToolset.Core.Native internal class WixNativeExe { private const int FiveMinutesInMilliseconds = 300000; - private static readonly string PathToWixNativeExe; + private static readonly object PathToWixNativeExeLock = new object(); + private static string PathToWixNativeExe; private readonly string commandLine; private readonly List stdinLines = new List(); - static WixNativeExe() - { - PathToWixNativeExe = Path.Combine(Path.GetDirectoryName(new Uri(Assembly.GetExecutingAssembly().CodeBase).LocalPath), "wixnative.x86.exe"); - } - public WixNativeExe(params object[] args) { this.commandLine = String.Join(" ", QuoteArgumentsAsNecesary(args)); @@ -39,10 +35,7 @@ namespace WixToolset.Core.Native public IEnumerable Run() { - if (!File.Exists(PathToWixNativeExe)) - { - throw new FileNotFoundException($"Could not find internal piece of WiX Toolset at: {PathToWixNativeExe}", PathToWixNativeExe); - } + EnsurePathToWixNativeExeSet(); var wixNativeInfo = new ProcessStartInfo(PathToWixNativeExe, this.commandLine) { @@ -88,6 +81,29 @@ namespace WixToolset.Core.Native return stdoutLines; } + private static void EnsurePathToWixNativeExeSet() + { + lock (PathToWixNativeExeLock) + { + if (String.IsNullOrEmpty(PathToWixNativeExe)) + { + var path = Path.Combine(Path.GetDirectoryName(new Uri(Assembly.GetExecutingAssembly().CodeBase).LocalPath), @"x86\wixnative.x86.exe"); + + if (!File.Exists(path)) + { + path = Path.Combine(Path.GetDirectoryName(new Uri(Assembly.GetExecutingAssembly().CodeBase).LocalPath), "wixnative.x86.exe"); + + if (!File.Exists(path)) + { + throw new FileNotFoundException($"Could not find internal piece of WiX Toolset at: {path}", path); + } + } + + PathToWixNativeExe = path; + } + } + } + private static IEnumerable QuoteArgumentsAsNecesary(object[] args) { foreach (var arg in args) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index 195ea7b6..a99c5ec3 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -28,9 +28,9 @@ These native executables are included in this .nupkg to place the .exe correctly for tests to work. That are ignored when published. The dependency above is used when publishing the tools. --> - - - - + + + + diff --git a/src/wixnative/wixnative.v3.ncrunchproject b/src/wixnative/wixnative.v3.ncrunchproject new file mode 100644 index 00000000..319cd523 --- /dev/null +++ b/src/wixnative/wixnative.v3.ncrunchproject @@ -0,0 +1,5 @@ + + + True + + \ No newline at end of file -- cgit v1.2.3-55-g6feb From 6dfba4f5a6528991b57d3939c8f62ba9de7cd7d5 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Wed, 26 Dec 2018 16:11:51 -0800 Subject: Move wixnative.*.exe back to root libs in .nupkg It seems NuGet will not copy the .exes from subfolders so move them back to root folder and live with the warnings that the .exe files are not managed code (because they aren't). --- src/WixToolset.Core.Native/WixNativeExe.cs | 23 +++++++--------------- .../WixToolset.Core.Native.nuspec | 8 ++++---- 2 files changed, 11 insertions(+), 20 deletions(-) diff --git a/src/WixToolset.Core.Native/WixNativeExe.cs b/src/WixToolset.Core.Native/WixNativeExe.cs index 3bd340bc..994fc7ec 100644 --- a/src/WixToolset.Core.Native/WixNativeExe.cs +++ b/src/WixToolset.Core.Native/WixNativeExe.cs @@ -12,7 +12,6 @@ namespace WixToolset.Core.Native internal class WixNativeExe { private const int FiveMinutesInMilliseconds = 300000; - private static readonly object PathToWixNativeExeLock = new object(); private static string PathToWixNativeExe; private readonly string commandLine; @@ -83,24 +82,16 @@ namespace WixToolset.Core.Native private static void EnsurePathToWixNativeExeSet() { - lock (PathToWixNativeExeLock) + if (String.IsNullOrEmpty(PathToWixNativeExe)) { - if (String.IsNullOrEmpty(PathToWixNativeExe)) - { - var path = Path.Combine(Path.GetDirectoryName(new Uri(Assembly.GetExecutingAssembly().CodeBase).LocalPath), @"x86\wixnative.x86.exe"); - - if (!File.Exists(path)) - { - path = Path.Combine(Path.GetDirectoryName(new Uri(Assembly.GetExecutingAssembly().CodeBase).LocalPath), "wixnative.x86.exe"); + var path = Path.Combine(Path.GetDirectoryName(new Uri(Assembly.GetExecutingAssembly().CodeBase).LocalPath), "wixnative.x86.exe"); - if (!File.Exists(path)) - { - throw new FileNotFoundException($"Could not find internal piece of WiX Toolset at: {path}", path); - } - } - - PathToWixNativeExe = path; + if (!File.Exists(path)) + { + throw new FileNotFoundException($"Could not find internal piece of WiX Toolset at: {path}", path); } + + PathToWixNativeExe = path; } } diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index a99c5ec3..195ea7b6 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -28,9 +28,9 @@ These native executables are included in this .nupkg to place the .exe correctly for tests to work. That are ignored when published. The dependency above is used when publishing the tools. --> - - - - + + + + -- cgit v1.2.3-55-g6feb From 2fb6a320c79c74dc4af2053841859c137879c089 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Sun, 30 Dec 2018 09:03:20 -0800 Subject: Reduce extra appveyor builds by targeting only master and develop branches --- appveyor.yml | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/appveyor.yml b/appveyor.yml index d55322da..c1df03cc 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -3,6 +3,11 @@ # 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 2017 version: 0.0.0.{build} -- cgit v1.2.3-55-g6feb From 0ef921687dd07e7c298b1c7533e51982c56f8d28 Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Sun, 13 Jan 2019 19:23:14 -0600 Subject: Update to latest Cpp.Build.props for locating latest Win10 SDK. --- src/Cpp.Build.props | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/src/Cpp.Build.props b/src/Cpp.Build.props index 296b36ca..0e00132b 100644 --- a/src/Cpp.Build.props +++ b/src/Cpp.Build.props @@ -8,6 +8,10 @@ $(OutputPath)$(Platform)\ + + $([Microsoft.Build.Utilities.ToolLocationHelper]::GetLatestSDKTargetPlatformVersion('Windows', '10.0')) + + $(DisableSpecificCompilerWarnings) @@ -16,7 +20,7 @@ WIN32;_WINDOWS;_WIN32_MSI=500;_WIN32_WINNT=0x0501;$(ArmPreprocessorDefinitions);$(UnicodePreprocessorDefinitions);_CRT_STDIO_LEGACY_WIDE_SPECIFIERS;_WINSOCK_DEPRECATED_NO_WARNINGS;%(PreprocessorDefinitions) Use precomp.h - StdCall + StdCall true false -YlprecompDefine -- cgit v1.2.3-55-g6feb From 0fd0f531db8250a121939dc9d453a236b477ae7a Mon Sep 17 00:00:00 2001 From: Bob Arnson Date: Thu, 21 Mar 2019 20:05:15 -0400 Subject: Avoid hang when a .cab is has no files. --- src/wixnative/smartcab.cpp | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/src/wixnative/smartcab.cpp b/src/wixnative/smartcab.cpp index da9087a3..0dff6c94 100644 --- a/src/wixnative/smartcab.cpp +++ b/src/wixnative/smartcab.cpp @@ -64,8 +64,11 @@ HRESULT SmartCabCommand( hr = CabCBegin(wzCabName, sczCabDir, uiFileCount, uiMaxSize, uiMaxThresh, ct, &hCab); ConsoleExitOnFailure(hr, CONSOLE_COLOR_RED, "failed to initialize cabinet: %ls", wzCabPath); - hr = CompressFiles(hCab); - ExitOnFailure(hr, "failed to compress files into 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. -- cgit v1.2.3-55-g6feb From 26b1860dcd8a84b410ab5cd3df7e37104e6ee1b1 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Wed, 8 May 2019 14:04:05 -0700 Subject: Remove MSI interop from Core.Native --- src/WixToolset.Core.Native/CabInterop.cs | 5 +- src/WixToolset.Core.Native/MSIFILEHASHINFO.cs | 19 + src/WixToolset.Core.Native/MsiInterop.cs | 851 --------------------- src/WixToolset.Core.Native/MsmInterop.cs | 507 ------------ .../WixToolset.Core.Native.csproj | 2 +- .../WixToolsetTest.Core.Native.csproj | 4 +- 6 files changed, 23 insertions(+), 1365 deletions(-) create mode 100644 src/WixToolset.Core.Native/MSIFILEHASHINFO.cs delete mode 100644 src/WixToolset.Core.Native/MsiInterop.cs delete mode 100644 src/WixToolset.Core.Native/MsmInterop.cs diff --git a/src/WixToolset.Core.Native/CabInterop.cs b/src/WixToolset.Core.Native/CabInterop.cs index db31a30d..4c1eb93d 100644 --- a/src/WixToolset.Core.Native/CabInterop.cs +++ b/src/WixToolset.Core.Native/CabInterop.cs @@ -4,10 +4,7 @@ namespace WixToolset.Core.Native { using System; using System.Diagnostics.CodeAnalysis; - using System.Text; using System.Runtime.InteropServices; - // using WixToolset.Msi; - // using WixToolset.Msi.Interop; /// /// The native methods. @@ -34,7 +31,7 @@ namespace WixToolset.Core.Native /// Name of file in cabinet. /// Handle to open cabinet. [DllImport("winterop.dll", EntryPoint = "CreateCabAddFile", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void CreateCabAddFile(string file, string token, MsiInterop.MSIFILEHASHINFO fileHash, IntPtr contextHandle); + public static extern void CreateCabAddFile(string file, string token, MSIFILEHASHINFO fileHash, IntPtr contextHandle); /// /// Closes a cabinet. diff --git a/src/WixToolset.Core.Native/MSIFILEHASHINFO.cs b/src/WixToolset.Core.Native/MSIFILEHASHINFO.cs new file mode 100644 index 00000000..c11b44ea --- /dev/null +++ b/src/WixToolset.Core.Native/MSIFILEHASHINFO.cs @@ -0,0 +1,19 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.Core.Native +{ + 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/MsiInterop.cs b/src/WixToolset.Core.Native/MsiInterop.cs deleted file mode 100644 index 8e5200c7..00000000 --- a/src/WixToolset.Core.Native/MsiInterop.cs +++ /dev/null @@ -1,851 +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.Text; - using System.Runtime.InteropServices; - using FILETIME = System.Runtime.InteropServices.ComTypes.FILETIME; - - /// - /// 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, [MarshalAs(UnmanagedType.LPWStr)] string message); - - /// - /// Enum of predefined persist modes used when opening a database. - /// - public enum OpenDatabase - { - /// - /// Open a database read-only, no persistent changes. - /// - ReadOnly = MsiInterop.MSIDBOPENREADONLY, - - /// - /// Open a database read/write in transaction mode. - /// - Transact = MsiInterop.MSIDBOPENTRANSACT, - - /// - /// Open a database direct read/write without transaction. - /// - Direct = MsiInterop.MSIDBOPENDIRECT, - - /// - /// Create a new database, transact mode read/write. - /// - Create = MsiInterop.MSIDBOPENCREATE, - - /// - /// Create a new database, direct mode read/write. - /// - CreateDirect = MsiInterop.MSIDBOPENCREATEDIRECT, - - /// - /// Indicates a patch file is being opened. - /// - OpenPatchFile = MsiInterop.MSIDBOPENPATCHFILE - } - - /// - /// 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 - } - - /// - /// 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 - } - - /// - /// Class exposing static functions and structs from MSI API. - /// - public sealed class MsiInterop - { - // Patching constants - public const int MsiMaxStreamNameLength = 62; // http://msdn2.microsoft.com/library/aa370551.aspx - - // Component.Attributes - public const int MsidbComponentAttributesLocalOnly = 0; - public const int MsidbComponentAttributesSourceOnly = 1; - public const int MsidbComponentAttributesOptional = 2; - public const int MsidbComponentAttributesRegistryKeyPath = 4; - public const int MsidbComponentAttributesSharedDllRefCount = 8; - public const int MsidbComponentAttributesPermanent = 16; - public const int MsidbComponentAttributesODBCDataSource = 32; - public const int MsidbComponentAttributesTransitive = 64; - public const int MsidbComponentAttributesNeverOverwrite = 128; - public const int MsidbComponentAttributes64bit = 256; - public const int MsidbComponentAttributesDisableRegistryReflection = 512; - public const int MsidbComponentAttributesUninstallOnSupersedence = 1024; - public const int MsidbComponentAttributesShared = 2048; - - // BBControl.Attributes & Control.Attributes - public const int MsidbControlAttributesVisible = 0x00000001; - public const int MsidbControlAttributesEnabled = 0x00000002; - public const int MsidbControlAttributesSunken = 0x00000004; - public const int MsidbControlAttributesIndirect = 0x00000008; - public const int MsidbControlAttributesInteger = 0x00000010; - public const int MsidbControlAttributesRTLRO = 0x00000020; - public const int MsidbControlAttributesRightAligned = 0x00000040; - public const int MsidbControlAttributesLeftScroll = 0x00000080; - public const int MsidbControlAttributesBiDi = MsidbControlAttributesRTLRO | MsidbControlAttributesRightAligned | MsidbControlAttributesLeftScroll; - - // Text controls - public const int MsidbControlAttributesTransparent = 0x00010000; - public const int MsidbControlAttributesNoPrefix = 0x00020000; - public const int MsidbControlAttributesNoWrap = 0x00040000; - public const int MsidbControlAttributesFormatSize = 0x00080000; - public const int MsidbControlAttributesUsersLanguage = 0x00100000; - - // Edit controls - public const int MsidbControlAttributesMultiline = 0x00010000; - public const int MsidbControlAttributesPasswordInput = 0x00200000; - - // ProgressBar controls - public const int MsidbControlAttributesProgress95 = 0x00010000; - - // VolumeSelectCombo and DirectoryCombo controls - public const int MsidbControlAttributesRemovableVolume = 0x00010000; - public const int MsidbControlAttributesFixedVolume = 0x00020000; - public const int MsidbControlAttributesRemoteVolume = 0x00040000; - public const int MsidbControlAttributesCDROMVolume = 0x00080000; - public const int MsidbControlAttributesRAMDiskVolume = 0x00100000; - public const int MsidbControlAttributesFloppyVolume = 0x00200000; - - // VolumeCostList controls - public const int MsidbControlShowRollbackCost = 0x00400000; - - // ListBox and ComboBox controls - public const int MsidbControlAttributesSorted = 0x00010000; - public const int MsidbControlAttributesComboList = 0x00020000; - - // picture button controls - public const int MsidbControlAttributesImageHandle = 0x00010000; - public const int MsidbControlAttributesPushLike = 0x00020000; - public const int MsidbControlAttributesBitmap = 0x00040000; - public const int MsidbControlAttributesIcon = 0x00080000; - public const int MsidbControlAttributesFixedSize = 0x00100000; - public const int MsidbControlAttributesIconSize16 = 0x00200000; - public const int MsidbControlAttributesIconSize32 = 0x00400000; - public const int MsidbControlAttributesIconSize48 = 0x00600000; - public const int MsidbControlAttributesElevationShield = 0x00800000; - - // RadioButton controls - public const int MsidbControlAttributesHasBorder = 0x01000000; - - // CustomAction.Type - // executable types - public const int MsidbCustomActionTypeDll = 0x00000001; // Target = entry point name - public const int MsidbCustomActionTypeExe = 0x00000002; // Target = command line args - public const int MsidbCustomActionTypeTextData = 0x00000003; // Target = text string to be formatted and set into property - public const int MsidbCustomActionTypeJScript = 0x00000005; // Target = entry point name; null if none to call - public const int MsidbCustomActionTypeVBScript = 0x00000006; // Target = entry point name; null if none to call - public const int MsidbCustomActionTypeInstall = 0x00000007; // Target = property list for nested engine initialization - public const int MsidbCustomActionTypeSourceBits = 0x00000030; - public const int MsidbCustomActionTypeTargetBits = 0x00000007; - public const int MsidbCustomActionTypeReturnBits = 0x000000C0; - public const int MsidbCustomActionTypeExecuteBits = 0x00000700; - - // source of code - public const int MsidbCustomActionTypeBinaryData = 0x00000000; // Source = Binary.Name; data stored in stream - public const int MsidbCustomActionTypeSourceFile = 0x00000010; // Source = File.File; file part of installation - public const int MsidbCustomActionTypeDirectory = 0x00000020; // Source = Directory.Directory; folder containing existing file - public const int MsidbCustomActionTypeProperty = 0x00000030; // Source = Property.Property; full path to executable - - // return processing; default is syncronous execution; process return code - public const int MsidbCustomActionTypeContinue = 0x00000040; // ignore action return status; continue running - public const int MsidbCustomActionTypeAsync = 0x00000080; // run asynchronously - - // execution scheduling flags; default is execute whenever sequenced - public const int MsidbCustomActionTypeFirstSequence = 0x00000100; // skip if UI sequence already run - public const int MsidbCustomActionTypeOncePerProcess = 0x00000200; // skip if UI sequence already run in same process - public const int MsidbCustomActionTypeClientRepeat = 0x00000300; // run on client only if UI already run on client - public const int MsidbCustomActionTypeInScript = 0x00000400; // queue for execution within script - public const int MsidbCustomActionTypeRollback = 0x00000100; // in conjunction with InScript: queue in Rollback script - public const int MsidbCustomActionTypeCommit = 0x00000200; // in conjunction with InScript: run Commit ops from script on success - - // security context flag; default to impersonate as user; valid only if InScript - public const int MsidbCustomActionTypeNoImpersonate = 0x00000800; // no impersonation; run in system context - public const int MsidbCustomActionTypeTSAware = 0x00004000; // impersonate for per-machine installs on TS machines - public const int MsidbCustomActionType64BitScript = 0x00001000; // script should run in 64bit process - public const int MsidbCustomActionTypeHideTarget = 0x00002000; // don't record the contents of the Target field in the log file. - - public const int MsidbCustomActionTypePatchUninstall = 0x00008000; // run on patch uninstall - - // Dialog.Attributes - public const int MsidbDialogAttributesVisible = 0x00000001; - public const int MsidbDialogAttributesModal = 0x00000002; - public const int MsidbDialogAttributesMinimize = 0x00000004; - public const int MsidbDialogAttributesSysModal = 0x00000008; - public const int MsidbDialogAttributesKeepModeless = 0x00000010; - public const int MsidbDialogAttributesTrackDiskSpace = 0x00000020; - public const int MsidbDialogAttributesUseCustomPalette = 0x00000040; - public const int MsidbDialogAttributesRTLRO = 0x00000080; - public const int MsidbDialogAttributesRightAligned = 0x00000100; - public const int MsidbDialogAttributesLeftScroll = 0x00000200; - public const int MsidbDialogAttributesBiDi = MsidbDialogAttributesRTLRO | MsidbDialogAttributesRightAligned | MsidbDialogAttributesLeftScroll; - public const int MsidbDialogAttributesError = 0x00010000; - public const int CommonControlAttributesInvert = MsidbControlAttributesVisible + MsidbControlAttributesEnabled; - public const int DialogAttributesInvert = MsidbDialogAttributesVisible + MsidbDialogAttributesModal + MsidbDialogAttributesMinimize; - - // Feature.Attributes - public const int MsidbFeatureAttributesFavorLocal = 0; - public const int MsidbFeatureAttributesFavorSource = 1; - public const int MsidbFeatureAttributesFollowParent = 2; - public const int MsidbFeatureAttributesFavorAdvertise = 4; - public const int MsidbFeatureAttributesDisallowAdvertise = 8; - public const int MsidbFeatureAttributesUIDisallowAbsent = 16; - public const int MsidbFeatureAttributesNoUnsupportedAdvertise = 32; - - // File.Attributes - public const int MsidbFileAttributesReadOnly = 1; - public const int MsidbFileAttributesHidden = 2; - public const int MsidbFileAttributesSystem = 4; - public const int MsidbFileAttributesVital = 512; - public const int MsidbFileAttributesChecksum = 1024; - public const int MsidbFileAttributesPatchAdded = 4096; - public const int MsidbFileAttributesNoncompressed = 8192; - public const int MsidbFileAttributesCompressed = 16384; - - // IniFile.Action & RemoveIniFile.Action - public const int MsidbIniFileActionAddLine = 0; - public const int MsidbIniFileActionCreateLine = 1; - public const int MsidbIniFileActionRemoveLine = 2; - public const int MsidbIniFileActionAddTag = 3; - public const int MsidbIniFileActionRemoveTag = 4; - - // MoveFile.Options - public const int MsidbMoveFileOptionsMove = 1; - - // ServiceInstall.Attributes - public const int MsidbServiceInstallOwnProcess = 0x00000010; - public const int MsidbServiceInstallShareProcess = 0x00000020; - public const int MsidbServiceInstallInteractive = 0x00000100; - public const int MsidbServiceInstallAutoStart = 0x00000002; - public const int MsidbServiceInstallDemandStart = 0x00000003; - public const int MsidbServiceInstallDisabled = 0x00000004; - public const int MsidbServiceInstallErrorIgnore = 0x00000000; - public const int MsidbServiceInstallErrorNormal = 0x00000001; - public const int MsidbServiceInstallErrorCritical = 0x00000003; - public const int MsidbServiceInstallErrorControlVital = 0x00008000; - - // ServiceConfig.Event - public const int MsidbServiceConfigEventInstall = 0x00000001; - public const int MsidbServiceConfigEventUninstall = 0x00000002; - public const int MsidbServiceConfigEventReinstall = 0x00000004; - - // ServiceControl.Attributes - public const int MsidbServiceControlEventStart = 0x00000001; - public const int MsidbServiceControlEventStop = 0x00000002; - public const int MsidbServiceControlEventDelete = 0x00000008; - public const int MsidbServiceControlEventUninstallStart = 0x00000010; - public const int MsidbServiceControlEventUninstallStop = 0x00000020; - public const int MsidbServiceControlEventUninstallDelete = 0x00000080; - - // TextStyle.StyleBits - public const int MsidbTextStyleStyleBitsBold = 1; - public const int MsidbTextStyleStyleBitsItalic = 2; - public const int MsidbTextStyleStyleBitsUnderline = 4; - public const int MsidbTextStyleStyleBitsStrike = 8; - - // Upgrade.Attributes - public const int MsidbUpgradeAttributesMigrateFeatures = 0x00000001; - public const int MsidbUpgradeAttributesOnlyDetect = 0x00000002; - public const int MsidbUpgradeAttributesIgnoreRemoveFailure = 0x00000004; - public const int MsidbUpgradeAttributesVersionMinInclusive = 0x00000100; - public const int MsidbUpgradeAttributesVersionMaxInclusive = 0x00000200; - public const int MsidbUpgradeAttributesLanguagesExclusive = 0x00000400; - - // Registry Hive Roots - public const int MsidbRegistryRootClassesRoot = 0; - public const int MsidbRegistryRootCurrentUser = 1; - public const int MsidbRegistryRootLocalMachine = 2; - public const int MsidbRegistryRootUsers = 3; - - // Locator Types - public const int MsidbLocatorTypeDirectory = 0; - public const int MsidbLocatorTypeFileName = 1; - public const int MsidbLocatorTypeRawValue = 2; - public const int MsidbLocatorType64bit = 16; - - public const int MsidbClassAttributesRelativePath = 1; - - // RemoveFile.InstallMode - public const int MsidbRemoveFileInstallModeOnInstall = 0x00000001; - public const int MsidbRemoveFileInstallModeOnRemove = 0x00000002; - public const int MsidbRemoveFileInstallModeOnBoth = 0x00000003; - - // ODBCDataSource.Registration - public const int MsidbODBCDataSourceRegistrationPerMachine = 0; - public const int MsidbODBCDataSourceRegistrationPerUser = 1; - - // ModuleConfiguration.Format - public const int MsidbModuleConfigurationFormatText = 0; - public const int MsidbModuleConfigurationFormatKey = 1; - public const int MsidbModuleConfigurationFormatInteger = 2; - public const int MsidbModuleConfigurationFormatBitfield = 3; - - // ModuleConfiguration.Attributes - public const int MsidbMsmConfigurableOptionKeyNoOrphan = 1; - public const int MsidbMsmConfigurableOptionNonNullable = 2; - - // ' Windows API function ShowWindow constants - used in Shortcut table - public const int SWSHOWNORMAL = 0x00000001; - public const int SWSHOWMAXIMIZED = 0x00000003; - public const int SWSHOWMINNOACTIVE = 0x00000007; - - // NameToBit arrays - // UI elements - public static readonly string[] CommonControlAttributes = { "Hidden", "Disabled", "Sunken", "Indirect", "Integer", "RightToLeft", "RightAligned", "LeftScroll" }; - public static readonly string[] TextControlAttributes = { "Transparent", "NoPrefix", "NoWrap", "FormatSize", "UserLanguage" }; - public static readonly string[] HyperlinkControlAttributes = { "Transparent" }; - public static readonly string[] EditControlAttributes = { "Multiline", null, null, null, null, "Password" }; - public static readonly string[] ProgressControlAttributes = { "ProgressBlocks" }; - public static readonly string[] VolumeControlAttributes = { "Removable", "Fixed", "Remote", "CDROM", "RAMDisk", "Floppy", "ShowRollbackCost" }; - public static readonly string[] ListboxControlAttributes = { "Sorted", null, null, null, "UserLanguage" }; - public static readonly string[] ListviewControlAttributes = { "Sorted", null, null, null, "FixedSize", "Icon16", "Icon32" }; - public static readonly string[] ComboboxControlAttributes = { "Sorted", "ComboList", null, null, "UserLanguage" }; - public static readonly string[] RadioControlAttributes = { "Image", "PushLike", "Bitmap", "Icon", "FixedSize", "Icon16", "Icon32", null, "HasBorder" }; - public static readonly string[] ButtonControlAttributes = { "Image", null, "Bitmap", "Icon", "FixedSize", "Icon16", "Icon32", "ElevationShield" }; - public static readonly string[] IconControlAttributes = { "Image", null, null, null, "FixedSize", "Icon16", "Icon32" }; - public static readonly string[] BitmapControlAttributes = { "Image", null, null, null, "FixedSize" }; - public static readonly string[] CheckboxControlAttributes = { null, "PushLike", "Bitmap", "Icon", "FixedSize", "Icon16", "Icon32" }; - - public const int MsidbEmbeddedUI = 0x01; - public const int MsidbEmbeddedHandlesBasic = 0x02; - - public const int INSTALLLOGMODE_FATALEXIT = 0x00001; - public const int INSTALLLOGMODE_ERROR = 0x00002; - public const int INSTALLLOGMODE_WARNING = 0x00004; - public const int INSTALLLOGMODE_USER = 0x00008; - public const int INSTALLLOGMODE_INFO = 0x00010; - public const int INSTALLLOGMODE_FILESINUSE = 0x00020; - public const int INSTALLLOGMODE_RESOLVESOURCE = 0x00040; - public const int INSTALLLOGMODE_OUTOFDISKSPACE = 0x00080; - public const int INSTALLLOGMODE_ACTIONSTART = 0x00100; - public const int INSTALLLOGMODE_ACTIONDATA = 0x00200; - public const int INSTALLLOGMODE_PROGRESS = 0x00400; - public const int INSTALLLOGMODE_COMMONDATA = 0x00800; - public const int INSTALLLOGMODE_INITIALIZE = 0x01000; - public const int INSTALLLOGMODE_TERMINATE = 0x02000; - public const int INSTALLLOGMODE_SHOWDIALOG = 0x04000; - public const int INSTALLLOGMODE_RMFILESINUSE = 0x02000000; - public const int INSTALLLOGMODE_INSTALLSTART = 0x04000000; - public const int INSTALLLOGMODE_INSTALLEND = 0x08000000; - - public const int MSICONDITIONFALSE = 0; // The table is temporary. - public const int MSICONDITIONTRUE = 1; // The table is persistent. - public const int MSICONDITIONNONE = 2; // The table is unknown. - public const int MSICONDITIONERROR = 3; // An invalid handle or invalid parameter was passed to the function. - - public const int MSIDBOPENREADONLY = 0; - public const int MSIDBOPENTRANSACT = 1; - public const int MSIDBOPENDIRECT = 2; - public const int MSIDBOPENCREATE = 3; - public const int MSIDBOPENCREATEDIRECT = 4; - public const int MSIDBOPENPATCHFILE = 32; - - public 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. - public 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. - public 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. - public 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. - public 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. - public 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. - public 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. - public 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. - public 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. - public 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. - public 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. - public 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. - public 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. - - public const uint VTI2 = 2; - public const uint VTI4 = 3; - public const uint VTLPWSTR = 30; - public const uint VTFILETIME = 64; - - public const int MSICOLINFONAMES = 0; // return column names - public const int MSICOLINFOTYPES = 1; // return column definitions, datatype code followed by width - - /// - /// Protect the constructor. - /// - private MsiInterop() - { - } - - /// - /// PInvoke of MsiCloseHandle. - /// - /// Handle to a database. - /// Error code. - [DllImport("msi.dll", EntryPoint = "MsiCloseHandle", CharSet = CharSet.Unicode, ExactSpelling = true)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public static extern int MsiDatabaseOpenView(uint database, string query, out uint view); - - /// - /// 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public 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)] - public static extern InstallUIHandler MsiSetExternalUI(InstallUIHandler installUIHandler, int installLogMode, IntPtr context); - - /// - /// PInvoke of MsiSetpublicUI. - /// - /// 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 = "MsiSetpublicUI", CharSet = CharSet.Unicode, ExactSpelling = true)] - public 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)] - public static extern int MsiSummaryInfoGetProperty(uint summaryInfo, int property, out uint dataType, out int integerValue, ref 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)] - public 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)] - public 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)] - public 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)] - public static extern int MsiViewModify(uint view, int modifyMode, uint record); - - /// - /// 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/MsmInterop.cs b/src/WixToolset.Core.Native/MsmInterop.cs deleted file mode 100644 index e309e57d..00000000 --- a/src/WixToolset.Core.Native/MsmInterop.cs +++ /dev/null @@ -1,507 +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.MergeMod -{ - using System; - using System.Collections; - using System.Runtime.CompilerServices; - 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, - } - - /// - /// 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; - } - } - - /// - /// 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; - } - } - - /// - /// 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; - } - } - - /// - /// 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; - } - } - - /// - /// 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); - } - - /// - /// 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 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() - { - IClassFactory 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/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index e2431fef..c93ff5bc 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -9,7 +9,7 @@ - + diff --git a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj index 4e02073e..47396ead 100644 --- a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj +++ b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj @@ -18,7 +18,7 @@ - - + + -- cgit v1.2.3-55-g6feb From 36cfc9509cb6a364f58531ff41eae16243091a3c Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Thu, 23 May 2019 16:29:55 -0700 Subject: User shared data types from WixToolset.Data --- nuget.config | 1 + src/WixToolset.Core.Native/Cabinet.cs | 6 +++--- .../CabinetCompressionLevel.cs | 25 ---------------------- .../WixToolset.Core.Native.csproj | 6 +++++- .../WixToolsetTest.Core.Native/CabinetFixture.cs | 6 +++--- .../WixToolsetTest.Core.Native.csproj | 4 ++++ 6 files changed, 16 insertions(+), 32 deletions(-) delete mode 100644 src/WixToolset.Core.Native/CabinetCompressionLevel.cs diff --git a/nuget.config b/nuget.config index 790be2b0..18404582 100644 --- a/nuget.config +++ b/nuget.config @@ -2,6 +2,7 @@ + diff --git a/src/WixToolset.Core.Native/Cabinet.cs b/src/WixToolset.Core.Native/Cabinet.cs index 27b0ec74..3f5fa715 100644 --- a/src/WixToolset.Core.Native/Cabinet.cs +++ b/src/WixToolset.Core.Native/Cabinet.cs @@ -5,6 +5,7 @@ namespace WixToolset.Core.Native using System; using System.Collections.Generic; using System.Linq; + using WixToolset.Data; /// /// Wrapper class around interop with wixcab.dll to compress files into a cabinet. @@ -29,7 +30,7 @@ namespace WixToolset.Core.Native /// Maximum number of files that will be added to cabinet. /// Maximum size of cabinet. /// Maximum threshold for each cabinet. - public void Compress(IEnumerable files, CabinetCompressionLevel compressionLevel, int maxSize = 0, int maxThresh = 0) + public void Compress(IEnumerable files, CompressionLevel compressionLevel, int maxSize = 0, int maxThresh = 0) { var compressionLevelVariable = Environment.GetEnvironmentVariable(CompressionLevelVariable); @@ -38,8 +39,7 @@ namespace WixToolset.Core.Native { if (!Enum.TryParse(compressionLevelVariable, true, out compressionLevel)) { - //throw new WixException(WixErrors.IllegalEnvironmentVariable(CompressionLevelVariable, compressionLevelVariable)); - throw new ArgumentException(); + throw new WixException(ErrorMessages.IllegalEnvironmentVariable(CompressionLevelVariable, compressionLevelVariable)); } } diff --git a/src/WixToolset.Core.Native/CabinetCompressionLevel.cs b/src/WixToolset.Core.Native/CabinetCompressionLevel.cs deleted file mode 100644 index fce1ff41..00000000 --- a/src/WixToolset.Core.Native/CabinetCompressionLevel.cs +++ /dev/null @@ -1,25 +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 -{ - /// - /// Compression level to use when creating cabinet. - /// - public enum CabinetCompressionLevel - { - /// Use no compression. - None, - - /// Use low compression. - Low, - - /// Use medium compression. - Medium, - - /// Use high compression. - High, - - /// Use ms-zip compression. - Mszip - } -} \ No newline at end of file diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index c93ff5bc..b4b1082f 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -1,4 +1,4 @@ - + netstandard2.0 @@ -8,6 +8,10 @@ + + + + diff --git a/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs b/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs index bc8f0ff8..fba0d5c9 100644 --- a/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs +++ b/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs @@ -2,11 +2,11 @@ namespace WixToolsetTest.CoreNative { - using System; using System.IO; using System.Linq; using WixToolset.Core.Native; using WixToolsetTest.CoreNative.Utility; + using WixToolset.Data; using Xunit; public class CabinetFixture @@ -22,7 +22,7 @@ namespace WixToolsetTest.CoreNative var files = new[] { new CabinetCompressFile(TestData.Get(@"TestData\test.txt"), "test.txt") }; var cabinet = new Cabinet(cabPath); - cabinet.Compress(files, CabinetCompressionLevel.Low); + cabinet.Compress(files, CompressionLevel.Low); Assert.True(File.Exists(cabPath)); } @@ -84,7 +84,7 @@ namespace WixToolsetTest.CoreNative var files = new[] { new CabinetCompressFile(TestData.Get(@"TestData\test.txt"), "test.txt") }; var cabinet = new Cabinet(cabinetPath); - cabinet.Compress(files, CabinetCompressionLevel.Low); + cabinet.Compress(files, CompressionLevel.Low); } // Extract. diff --git a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj index 47396ead..263712bb 100644 --- a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj +++ b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj @@ -16,6 +16,10 @@ + + + + -- cgit v1.2.3-55-g6feb From 54ade276db9bd437375a57bddc9eb88a21a6ea7a Mon Sep 17 00:00:00 2001 From: Bob Arnson Date: Wed, 11 Dec 2019 19:17:17 -0500 Subject: Return the set of files extracted from a cabinet. --- src/WixToolset.Core.Native/Cabinet.cs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/WixToolset.Core.Native/Cabinet.cs b/src/WixToolset.Core.Native/Cabinet.cs index 3f5fa715..74ab7809 100644 --- a/src/WixToolset.Core.Native/Cabinet.cs +++ b/src/WixToolset.Core.Native/Cabinet.cs @@ -100,7 +100,7 @@ namespace WixToolset.Core.Native /// Extracts all the files from a cabinet to a directory. /// /// Directory to extract files to. - public void Extract(string outputFolder) + public IEnumerable Extract(string outputFolder) { if (!outputFolder.EndsWith("\\", StringComparison.Ordinal)) { @@ -108,7 +108,7 @@ namespace WixToolset.Core.Native } var wixnative = new WixNativeExe("extractcab", this.Path, outputFolder); - wixnative.Run(); + return wixnative.Run(); } #if TOOD_ERROR_HANDLING -- cgit v1.2.3-55-g6feb From c53d0d1cc57ce3ed1450adb2c469e2b747584159 Mon Sep 17 00:00:00 2001 From: Bob Arnson Date: Wed, 11 Dec 2019 21:52:38 -0500 Subject: Exclude null output from report of extracted files. --- src/WixToolset.Core.Native/Cabinet.cs | 2 +- src/test/WixToolsetTest.Core.Native/CabinetFixture.cs | 14 +++++++++----- 2 files changed, 10 insertions(+), 6 deletions(-) diff --git a/src/WixToolset.Core.Native/Cabinet.cs b/src/WixToolset.Core.Native/Cabinet.cs index 74ab7809..2d624658 100644 --- a/src/WixToolset.Core.Native/Cabinet.cs +++ b/src/WixToolset.Core.Native/Cabinet.cs @@ -108,7 +108,7 @@ namespace WixToolset.Core.Native } var wixnative = new WixNativeExe("extractcab", this.Path, outputFolder); - return wixnative.Run(); + return wixnative.Run().Where(output => !String.IsNullOrWhiteSpace(output)); } #if TOOD_ERROR_HANDLING diff --git a/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs b/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs index fba0d5c9..4f5a3427 100644 --- a/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs +++ b/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs @@ -56,9 +56,9 @@ namespace WixToolsetTest.CoreNative var extractFolder = fs.GetFolder(true); var cabinet = new Cabinet(cabinetPath); - cabinet.Extract(extractFolder); - + var reportedFiles = cabinet.Extract(extractFolder); var files = Directory.EnumerateFiles(extractFolder); + Assert.Equal(reportedFiles, files.Select(f => Path.GetFileName(f))); var file = new FileInfo(files.Single()); CabInterop.DateTimeToCabDateAndTime(file.CreationTime, out var date, out var time); @@ -81,7 +81,10 @@ namespace WixToolsetTest.CoreNative // Compress. { - var files = new[] { new CabinetCompressFile(TestData.Get(@"TestData\test.txt"), "test.txt") }; + 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); @@ -90,7 +93,8 @@ namespace WixToolsetTest.CoreNative // Extract. { var cabinet = new Cabinet(cabinetPath); - cabinet.Extract(extractFolder); + var reportedFiles = cabinet.Extract(extractFolder); + Assert.Equal(2, reportedFiles.Count()); } // Enumerate to compare cabinet to extracted files. @@ -100,7 +104,7 @@ namespace WixToolsetTest.CoreNative var files = Directory.EnumerateFiles(extractFolder).OrderBy(f => f).ToArray(); - for (var i =0; i < enumerated.Length; ++i) + for (var i = 0; i < enumerated.Length; ++i) { var cabFileInfo = enumerated[i]; var fileInfo = new FileInfo(files[i]); -- cgit v1.2.3-55-g6feb From 5eabd27e1af2827228b1ed925664bc15d9035484 Mon Sep 17 00:00:00 2001 From: Bob Arnson Date: Thu, 7 May 2020 21:34:09 -0400 Subject: Fix timeout Increase WixNative*.exe timeout to 10 minutes and properly handle when there's a timeout. --- src/WixToolset.Core.Native/WixNativeExe.cs | 15 ++++++++++----- 1 file changed, 10 insertions(+), 5 deletions(-) diff --git a/src/WixToolset.Core.Native/WixNativeExe.cs b/src/WixToolset.Core.Native/WixNativeExe.cs index 994fc7ec..afa2ed7c 100644 --- a/src/WixToolset.Core.Native/WixNativeExe.cs +++ b/src/WixToolset.Core.Native/WixNativeExe.cs @@ -11,7 +11,7 @@ namespace WixToolset.Core.Native internal class WixNativeExe { - private const int FiveMinutesInMilliseconds = 300000; + private const int TenMinutesInMilliseconds = 600000; private static string PathToWixNativeExe; private readonly string commandLine; @@ -63,17 +63,22 @@ namespace WixToolset.Core.Native process.StandardInput.WriteLine(); } - if (process.WaitForExit(FiveMinutesInMilliseconds)) + if (process.WaitForExit(TenMinutesInMilliseconds)) { // 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) + if (process.ExitCode != 0) + { + throw new Win32Exception(process.ExitCode); + } + } + else { - throw new Win32Exception(process.ExitCode); + process.Kill(); + throw new Win32Exception(1460/*ERROR_TIMEOUT*/); } } -- cgit v1.2.3-55-g6feb From 45a46d9175aa29c526e22bf6193837543610bb37 Mon Sep 17 00:00:00 2001 From: Bob Arnson Date: Thu, 7 May 2020 21:43:35 -0400 Subject: Remove timeout on WixNative*.exe to avoid judging on build time. --- src/WixToolset.Core.Native/WixNativeExe.cs | 21 ++++++--------------- 1 file changed, 6 insertions(+), 15 deletions(-) diff --git a/src/WixToolset.Core.Native/WixNativeExe.cs b/src/WixToolset.Core.Native/WixNativeExe.cs index afa2ed7c..935c2a89 100644 --- a/src/WixToolset.Core.Native/WixNativeExe.cs +++ b/src/WixToolset.Core.Native/WixNativeExe.cs @@ -11,7 +11,6 @@ namespace WixToolset.Core.Native internal class WixNativeExe { - private const int TenMinutesInMilliseconds = 600000; private static string PathToWixNativeExe; private readonly string commandLine; @@ -63,22 +62,14 @@ namespace WixToolset.Core.Native process.StandardInput.WriteLine(); } - if (process.WaitForExit(TenMinutesInMilliseconds)) - { - // 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 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); - } - } - else + if (process.ExitCode != 0) { - process.Kill(); - throw new Win32Exception(1460/*ERROR_TIMEOUT*/); + throw new Win32Exception(process.ExitCode); } } -- cgit v1.2.3-55-g6feb From df6e15e2b026f8a419fb275c0b50c9d3b3298859 Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Sat, 9 May 2020 21:06:30 +1000 Subject: Update dependencies. --- appveyor.yml | 2 +- src/Cpp.Build.props | 20 +------------------- .../WixToolset.Core.Native.csproj | 2 +- src/winterop/packages.config | 4 ++-- src/winterop/winterop.vcxproj | 8 ++++---- src/wixnative/packages.config | 2 +- src/wixnative/wixnative.vcxproj | 4 ++-- 7 files changed, 12 insertions(+), 30 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index c1df03cc..7c686b04 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -8,7 +8,7 @@ branches: - master - develop -image: Visual Studio 2017 +image: Visual Studio 2019 version: 0.0.0.{build} configuration: Release diff --git a/src/Cpp.Build.props b/src/Cpp.Build.props index 0e00132b..9b7a1bb5 100644 --- a/src/Cpp.Build.props +++ b/src/Cpp.Build.props @@ -8,7 +8,7 @@ $(OutputPath)$(Platform)\ - + $([Microsoft.Build.Utilities.ToolLocationHelper]::GetLatestSDKTargetPlatformVersion('Windows', '10.0')) @@ -70,12 +70,6 @@ MultiThreadedDebug - - - - MultiThreadedDebugDll - - MinSpace @@ -89,16 +83,4 @@ true - - - - MultiThreadedDll - - - - - $(LinkKeyFile) - $(LinkDelaySign) - - diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index b4b1082f..ee5cbc8e 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -13,7 +13,7 @@ - + diff --git a/src/winterop/packages.config b/src/winterop/packages.config index 02ee2250..478af2c6 100644 --- a/src/winterop/packages.config +++ b/src/winterop/packages.config @@ -1,5 +1,5 @@  - - + + \ No newline at end of file diff --git a/src/winterop/winterop.vcxproj b/src/winterop/winterop.vcxproj index 4c606656..48dbb69b 100644 --- a/src/winterop/winterop.vcxproj +++ b/src/winterop/winterop.vcxproj @@ -2,7 +2,7 @@ - + @@ -68,12 +68,12 @@ - + This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}. - - + + diff --git a/src/wixnative/packages.config b/src/wixnative/packages.config index 20d71c08..478af2c6 100644 --- a/src/wixnative/packages.config +++ b/src/wixnative/packages.config @@ -1,5 +1,5 @@  - + \ No newline at end of file diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index 420615ed..1c1c9010 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -2,7 +2,7 @@ - + @@ -84,6 +84,6 @@ This project references NuGet package(s) that are missing on this computer. Enable NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105.The missing file is {0}. - + -- cgit v1.2.3-55-g6feb From 670075995cf1b78d0a66e1a7678fd2facc8958ff Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Sat, 9 May 2020 21:08:49 +1000 Subject: Use NATIVE_DLL_SEARCH_DIRECTORIES to support cross-platform wixnative.exe --- src/WixToolset.Core.Native/WixNativeExe.cs | 21 ++++++++++++++++++++- .../WixToolset.Core.Native.nuspec | 4 +--- .../runtime.win.WixToolset.Core.Native.nuspec | 8 ++++---- src/wixnative/wixnative.vcxproj | 7 +------ 4 files changed, 26 insertions(+), 14 deletions(-) diff --git a/src/WixToolset.Core.Native/WixNativeExe.cs b/src/WixToolset.Core.Native/WixNativeExe.cs index 935c2a89..063485d3 100644 --- a/src/WixToolset.Core.Native/WixNativeExe.cs +++ b/src/WixToolset.Core.Native/WixNativeExe.cs @@ -11,6 +11,7 @@ namespace WixToolset.Core.Native internal class WixNativeExe { + private const string WixNativeExeFileName = "wixnative.exe"; private static string PathToWixNativeExe; private readonly string commandLine; @@ -80,7 +81,25 @@ namespace WixToolset.Core.Native { if (String.IsNullOrEmpty(PathToWixNativeExe)) { - var path = Path.Combine(Path.GetDirectoryName(new Uri(Assembly.GetExecutingAssembly().CodeBase).LocalPath), "wixnative.x86.exe"); + var path = Path.Combine(Path.GetDirectoryName(new Uri(Assembly.GetExecutingAssembly().CodeBase).LocalPath), WixNativeExeFileName); + + if (!File.Exists(path)) + { + var searchDirectoriesString = AppContext.GetData("NATIVE_DLL_SEARCH_DIRECTORIES") as string; + var searchDirectories = searchDirectoriesString?.Split(';'); + if (searchDirectories != null) + { + foreach (string directoryPath in searchDirectories) + { + var possiblePath = Path.Combine(directoryPath, WixNativeExeFileName); + if (File.Exists(possiblePath)) + { + path = possiblePath; + break; + } + } + } + } if (!File.Exists(path)) { diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index 195ea7b6..1ff5538b 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -25,12 +25,10 @@ + --> diff --git a/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec b/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec index f7697694..59eff4e6 100644 --- a/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec +++ b/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec @@ -14,9 +14,9 @@ - - - - + + + + diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index 1c1c9010..57afcc0b 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -23,16 +23,11 @@ - - x86 - amd64 - - {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF} Application Console - wixnative.$(NameSuffix) + wixnative v141 Unicode Native component of WixToolset.Core -- cgit v1.2.3-55-g6feb From c7a1b6b6ea12b3f231d3d8f83590bda74b9284e5 Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Sun, 10 May 2020 12:51:16 +1000 Subject: The directory separator on non-Windows platforms is ':' --- src/WixToolset.Core.Native/WixNativeExe.cs | 28 +++++++++++++++------------- 1 file changed, 15 insertions(+), 13 deletions(-) diff --git a/src/WixToolset.Core.Native/WixNativeExe.cs b/src/WixToolset.Core.Native/WixNativeExe.cs index 063485d3..2ea5146e 100644 --- a/src/WixToolset.Core.Native/WixNativeExe.cs +++ b/src/WixToolset.Core.Native/WixNativeExe.cs @@ -8,6 +8,7 @@ namespace WixToolset.Core.Native using System.Diagnostics; using System.IO; using System.Reflection; + using System.Runtime.InteropServices; internal class WixNativeExe { @@ -82,28 +83,29 @@ namespace WixToolset.Core.Native if (String.IsNullOrEmpty(PathToWixNativeExe)) { var path = Path.Combine(Path.GetDirectoryName(new Uri(Assembly.GetExecutingAssembly().CodeBase).LocalPath), WixNativeExeFileName); + var possiblePaths = path; - if (!File.Exists(path)) + var found = File.Exists(path); + if (!found && AppContext.GetData("NATIVE_DLL_SEARCH_DIRECTORIES") is string searchDirectoriesString) { - var searchDirectoriesString = AppContext.GetData("NATIVE_DLL_SEARCH_DIRECTORIES") as string; - var searchDirectories = searchDirectoriesString?.Split(';'); - if (searchDirectories != null) + possiblePaths = searchDirectoriesString; + var separatorChar = RuntimeInformation.IsOSPlatform(OSPlatform.Windows) ? ';' : ':'; + var searchDirectories = searchDirectoriesString?.Split(separatorChar); + foreach (var directoryPath in searchDirectories) { - foreach (string directoryPath in searchDirectories) + var possiblePath = Path.Combine(directoryPath, WixNativeExeFileName); + if (File.Exists(possiblePath)) { - var possiblePath = Path.Combine(directoryPath, WixNativeExeFileName); - if (File.Exists(possiblePath)) - { - path = possiblePath; - break; - } + path = possiblePath; + found = true; + break; } } } - if (!File.Exists(path)) + if (!found) { - throw new FileNotFoundException($"Could not find internal piece of WiX Toolset at: {path}", path); + throw new FileNotFoundException($"Could not find internal piece of WiX Toolset at: {possiblePaths}", WixNativeExeFileName); } PathToWixNativeExe = path; -- cgit v1.2.3-55-g6feb From a4cc6c64cb94279daeef2ab14e6319105806c053 Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Wed, 13 May 2020 13:35:19 +1000 Subject: Wrap wixnative.exe FileNotFoundException in PlatformNotSupportedException --- src/WixToolset.Core.Native/WixNativeExe.cs | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/WixToolset.Core.Native/WixNativeExe.cs b/src/WixToolset.Core.Native/WixNativeExe.cs index 2ea5146e..325fa18f 100644 --- a/src/WixToolset.Core.Native/WixNativeExe.cs +++ b/src/WixToolset.Core.Native/WixNativeExe.cs @@ -105,7 +105,9 @@ namespace WixToolset.Core.Native if (!found) { - throw new FileNotFoundException($"Could not find internal piece of WiX Toolset at: {possiblePaths}", WixNativeExeFileName); + throw new PlatformNotSupportedException( + $"Could not find platform specific '{WixNativeExeFileName}'", + new FileNotFoundException($"Could not find internal piece of WiX Toolset from: {possiblePaths}", WixNativeExeFileName)); } PathToWixNativeExe = path; -- cgit v1.2.3-55-g6feb From e28fcc906046ed12a867b621fc0eabf75fd9db09 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Thu, 21 May 2020 23:08:52 -0700 Subject: Add support using mergmod.dll and its interfaces --- WixToolset.Core.Native.v3.ncrunchsolution | 6 + src/WixToolset.Core.Native/MsmInterop.cs | 507 +++++++++++++++++++++ .../WixToolset.Core.Native.csproj | 6 +- .../WixToolset.Core.Native.v3.ncrunchproject | 10 - src/test/WixToolsetTest.Core.Native/MsmFixture.cs | 18 + .../WixToolsetTest.Core.Native.csproj | 2 +- src/wixnative/Win32/mergemod.dll | Bin 0 -> 169304 bytes .../runtime.win.WixToolset.Core.Native.nuspec | 2 + src/wixnative/wixnative.vcxproj | 8 + src/wixnative/x64/mergemod.dll | Bin 0 -> 183848 bytes 10 files changed, 545 insertions(+), 14 deletions(-) create mode 100644 WixToolset.Core.Native.v3.ncrunchsolution create mode 100644 src/WixToolset.Core.Native/MsmInterop.cs delete mode 100644 src/WixToolset.Core.Native/WixToolset.Core.Native.v3.ncrunchproject create mode 100644 src/test/WixToolsetTest.Core.Native/MsmFixture.cs create mode 100644 src/wixnative/Win32/mergemod.dll create mode 100644 src/wixnative/x64/mergemod.dll diff --git a/WixToolset.Core.Native.v3.ncrunchsolution b/WixToolset.Core.Native.v3.ncrunchsolution new file mode 100644 index 00000000..10420ac9 --- /dev/null +++ b/WixToolset.Core.Native.v3.ncrunchsolution @@ -0,0 +1,6 @@ + + + True + True + + \ No newline at end of file diff --git a/src/WixToolset.Core.Native/MsmInterop.cs b/src/WixToolset.Core.Native/MsmInterop.cs new file mode 100644 index 00000000..6b1ad141 --- /dev/null +++ b/src/WixToolset.Core.Native/MsmInterop.cs @@ -0,0 +1,507 @@ +// 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.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, + } + + /// + /// 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; + } + } + + /// + /// 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; + } + } + + /// + /// 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; + } + } + + /// + /// 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; + } + } + + /// + /// 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); + } + + /// + /// 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 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 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/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index ee5cbc8e..ca8797a9 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -23,13 +23,13 @@ - + - + - + diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.v3.ncrunchproject b/src/WixToolset.Core.Native/WixToolset.Core.Native.v3.ncrunchproject deleted file mode 100644 index 0da1f42d..00000000 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.v3.ncrunchproject +++ /dev/null @@ -1,10 +0,0 @@ - - - - ..\..\build\Debug\Win32\wixnative.x86.exe - ..\..\build\Debug\Win32\wixnative.x86.pdb - ..\..\build\Debug\x64\wixnative.amd64.exe - ..\..\build\Debug\x64\wixnative.amd64.pdb - - - \ No newline at end of file diff --git a/src/test/WixToolsetTest.Core.Native/MsmFixture.cs b/src/test/WixToolsetTest.Core.Native/MsmFixture.cs new file mode 100644 index 00000000..a1e42d94 --- /dev/null +++ b/src/test/WixToolsetTest.Core.Native/MsmFixture.cs @@ -0,0 +1,18 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolsetTest.CoreNative +{ + using WixToolset.Core.Native; + using Xunit; + + public class MsmFixture + { + [Fact] + public void CanCreateMsmInterface() + { + var msm = new MsmInterop(); + var merge = msm.GetMsmMerge(); + Assert.NotNull(merge); + } + } +} diff --git a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj index 263712bb..14928112 100644 --- a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj +++ b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj @@ -3,7 +3,7 @@ - netcoreapp2.0 + netcoreapp2.1 false diff --git a/src/wixnative/Win32/mergemod.dll b/src/wixnative/Win32/mergemod.dll new file mode 100644 index 00000000..2bf647b2 Binary files /dev/null and b/src/wixnative/Win32/mergemod.dll differ diff --git a/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec b/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec index 59eff4e6..a775b570 100644 --- a/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec +++ b/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec @@ -14,8 +14,10 @@ + + diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index 57afcc0b..80e0aae5 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -69,6 +69,14 @@ + + + true + %(Filename)%(Extension) + PreserveNewest + + + diff --git a/src/wixnative/x64/mergemod.dll b/src/wixnative/x64/mergemod.dll new file mode 100644 index 00000000..2f4cdc7c Binary files /dev/null and b/src/wixnative/x64/mergemod.dll differ -- cgit v1.2.3-55-g6feb From 5cd1e6bfadbbe327d02cd7a5510851178ddbbeee Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Mon, 15 Jun 2020 21:57:19 +1000 Subject: Use Path.PathSeparator. --- appveyor.cmd | 8 ++++---- src/WixToolset.Core.Native/WixNativeExe.cs | 2 +- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/appveyor.cmd b/appveyor.cmd index a39fa567..63bf9301 100644 --- a/appveyor.cmd +++ b/appveyor.cmd @@ -1,13 +1,13 @@ @setlocal @pushd %~dp0 -nuget restore +nuget restore || exit /b -msbuild -p:Configuration=Release .\src\test\WixToolsetTest.Core.Native\WixToolsetTest.Core.Native.csproj +msbuild -p:Configuration=Release .\src\test\WixToolsetTest.Core.Native\WixToolsetTest.Core.Native.csproj || exit /b -msbuild -t:Pack -p:Configuration=Release .\src\WixToolset.Core.Native\WixToolset.Core.Native.csproj +msbuild -t:Pack -p:Configuration=Release .\src\WixToolset.Core.Native\WixToolset.Core.Native.csproj || exit /b -msbuild -t:Pack -p:Configuration=Release .\src\wixnative\wixnative.vcxproj +msbuild -t:Pack -p:Configuration=Release .\src\wixnative\wixnative.vcxproj || exit /b @popd @endlocal \ No newline at end of file diff --git a/src/WixToolset.Core.Native/WixNativeExe.cs b/src/WixToolset.Core.Native/WixNativeExe.cs index 325fa18f..eaa2b2e0 100644 --- a/src/WixToolset.Core.Native/WixNativeExe.cs +++ b/src/WixToolset.Core.Native/WixNativeExe.cs @@ -89,7 +89,7 @@ namespace WixToolset.Core.Native if (!found && AppContext.GetData("NATIVE_DLL_SEARCH_DIRECTORIES") is string searchDirectoriesString) { possiblePaths = searchDirectoriesString; - var separatorChar = RuntimeInformation.IsOSPlatform(OSPlatform.Windows) ? ';' : ':'; + var separatorChar = Path.PathSeparator; var searchDirectories = searchDirectoriesString?.Split(separatorChar); foreach (var directoryPath in searchDirectories) { -- cgit v1.2.3-55-g6feb From d7bd6328ea9227f76b5337f703de9460a1738302 Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Tue, 23 Jun 2020 13:48:11 +1000 Subject: Update to latest dutil. --- src/winterop/packages.config | 2 +- src/winterop/winterop.vcxproj | 4 ++-- src/wixnative/packages.config | 2 +- src/wixnative/wixnative.vcxproj | 4 ++-- 4 files changed, 6 insertions(+), 6 deletions(-) diff --git a/src/winterop/packages.config b/src/winterop/packages.config index 478af2c6..38ce9597 100644 --- a/src/winterop/packages.config +++ b/src/winterop/packages.config @@ -1,5 +1,5 @@  - + \ No newline at end of file diff --git a/src/winterop/winterop.vcxproj b/src/winterop/winterop.vcxproj index 48dbb69b..a4603557 100644 --- a/src/winterop/winterop.vcxproj +++ b/src/winterop/winterop.vcxproj @@ -2,7 +2,7 @@ - + @@ -73,7 +73,7 @@ This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}. - + diff --git a/src/wixnative/packages.config b/src/wixnative/packages.config index 478af2c6..38ce9597 100644 --- a/src/wixnative/packages.config +++ b/src/wixnative/packages.config @@ -1,5 +1,5 @@  - + \ No newline at end of file diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index 80e0aae5..5927c14c 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -2,7 +2,7 @@ - + @@ -87,6 +87,6 @@ This project references NuGet package(s) that are missing on this computer. Enable NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105.The missing file is {0}. - + -- cgit v1.2.3-55-g6feb From c50a0e000ab1812ec4781ace349a5d84fa01c8e4 Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Wed, 24 Jun 2020 14:23:51 +1000 Subject: Merge the native nupkg into the Core.Native nupkg. There was no benefit in separating the native assets into its own package, the runtime..PackageName strategy is for binaries to be copied to the root of project and therefore doesn't support the portable scenario where all available assets are required. Manually author the repository element in the nupkg for SourceLink. --- appveyor.cmd | 8 +++--- .../WixToolset.Core.Native.csproj | 30 ++++++++++++++-------- .../WixToolset.Core.Native.nuspec | 25 ++++++++---------- .../WixToolsetTest.Core.Native.csproj | 5 ++-- .../runtime.win.WixToolset.Core.Native.nuspec | 24 ----------------- src/wixnative/wixnative.vcxproj | 13 ---------- 6 files changed, 37 insertions(+), 68 deletions(-) delete mode 100644 src/wixnative/runtime.win.WixToolset.Core.Native.nuspec diff --git a/appveyor.cmd b/appveyor.cmd index 63bf9301..6de09dad 100644 --- a/appveyor.cmd +++ b/appveyor.cmd @@ -1,13 +1,13 @@ @setlocal @pushd %~dp0 +@set _C=Release nuget restore || exit /b -msbuild -p:Configuration=Release .\src\test\WixToolsetTest.Core.Native\WixToolsetTest.Core.Native.csproj || exit /b +msbuild -p:Configuration=%_C% src\test\WixToolsetTest.Core.Native\WixToolsetTest.Core.Native.csproj || exit /b +dotnet test -c %_C% --no-build src\test\WixToolsetTest.Core.Native\WixToolsetTest.Core.Native.csproj || exit /b -msbuild -t:Pack -p:Configuration=Release .\src\WixToolset.Core.Native\WixToolset.Core.Native.csproj || exit /b - -msbuild -t:Pack -p:Configuration=Release .\src\wixnative\wixnative.vcxproj || exit /b +msbuild -t:Pack -p:Configuration=%_C% src\WixToolset.Core.Native\WixToolset.Core.Native.csproj || exit /b @popd @endlocal \ No newline at end of file diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index ca8797a9..d12d0da6 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -5,9 +5,19 @@ $(MSBuildThisFileName).nuspec Core Native embedded - + + + All + Platform=Win32 + + + All + Platform=x64 + + + @@ -20,16 +30,16 @@ $(MSBuildProjectDir)..\..\build\obj\$(ProjectName)\$(Configuration)\NativeFileList.txt $(NCrunchOriginalProjectDir)..\..\build\obj\$(ProjectName)\$(Configuration)\NativeFileList.txt + $(MSBuildThisFileDirectory)..\wixnative\ - - - - - - - + + <_NativeProjectOutput Include="$(MergeModDirectory)x64\mergemod.dll" /> + <_NativeProjectOutput Include="$(OutputPath)..\x64\wixnative.exe" /> + <_NativeProjectOutput Include="$(OutputPath)..\x64\wixnative.pdb" /> + + @@ -60,8 +70,8 @@ - $(OutputPath) - Configuration=$(Configuration);Id=$(MSBuildThisFileName);Version=$(BuildVersionSimple);Authors=$(Authors);Copyright=$(Copyright);Description=$(Description) + $(OutputPath)..\ + Id=$(MSBuildThisFileName);Version=$(BuildVersionSimple);Authors=$(Authors);Copyright=$(Copyright);Description=$(Description);RepositoryCommit=$(SourceRevisionId);RepositoryType=$(RepositoryType);RepositoryUrl=$(PrivateRepositoryUrl);MergeModDir=$(MergeModDirectory) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index 1ff5538b..e862b1ba 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -5,30 +5,25 @@ $version$ $authors$ $authors$ - - https://licenses.nuget.org/MS-RL + MS-RL https://github.com/wixtoolset/Core.Native false $description$ $copyright$ - + - - + - + - + + + + + + diff --git a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj index 14928112..9c493d84 100644 --- a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj +++ b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj @@ -3,8 +3,9 @@ - netcoreapp2.1 + netcoreapp3.1 false + win-x64 @@ -21,7 +22,7 @@ - + diff --git a/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec b/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec deleted file mode 100644 index a775b570..00000000 --- a/src/wixnative/runtime.win.WixToolset.Core.Native.nuspec +++ /dev/null @@ -1,24 +0,0 @@ - - - - $id$ - $version$ - $authors$ - $authors$ - - https://licenses.nuget.org/MS-RL - https://github.com/wixtoolset/Core.Native - false - $description$ - $copyright$ - - - - - - - - - - - diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index 5927c14c..e53af651 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -30,7 +30,6 @@ wixnative v141 Unicode - Native component of WixToolset.Core @@ -69,18 +68,6 @@ - - - true - %(Filename)%(Extension) - PreserveNewest - - - - - - - -- cgit v1.2.3-55-g6feb From c1cc095efa7deec07ffe29034a076035458c260f Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Wed, 24 Jun 2020 14:27:03 +1000 Subject: Move native pdbs into symbols package. --- src/WixToolset.Core.Native/WixToolset.Core.Native.csproj | 1 + 1 file changed, 1 insertion(+) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index d12d0da6..d62de257 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -5,6 +5,7 @@ $(MSBuildThisFileName).nuspec Core Native embedded + true -- cgit v1.2.3-55-g6feb From 84a60ab50339f158715172ef24804c5b9cb5ba9d Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Wed, 24 Jun 2020 14:32:26 +1000 Subject: Add ARM/ARM64 wixnative.exe Update mergemod.dll from SDK 10.0.19041.0. --- .../WixToolset.Core.Native.csproj | 8 +++++++ .../WixToolset.Core.Native.nuspec | 6 +++++ src/winterop/winterop.vcxproj | 26 +++++++++++++++++---- src/wixnative/ARM/mergemod.dll | Bin 0 -> 172704 bytes src/wixnative/ARM64/mergemod.dll | Bin 0 -> 194512 bytes src/wixnative/Win32/mergemod.dll | Bin 169304 -> 159176 bytes src/wixnative/wixnative.vcxproj | 26 +++++++++++++++++---- src/wixnative/x64/mergemod.dll | Bin 183848 -> 180376 bytes 8 files changed, 56 insertions(+), 10 deletions(-) create mode 100644 src/wixnative/ARM/mergemod.dll create mode 100644 src/wixnative/ARM64/mergemod.dll diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index d62de257..2626f749 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -9,6 +9,14 @@ + + All + Platform=ARM + + + All + Platform=ARM64 + All Platform=Win32 diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index e862b1ba..2e4b491c 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -19,6 +19,12 @@ + + + + + + diff --git a/src/winterop/winterop.vcxproj b/src/winterop/winterop.vcxproj index a4603557..be006c38 100644 --- a/src/winterop/winterop.vcxproj +++ b/src/winterop/winterop.vcxproj @@ -5,18 +5,34 @@ - + Debug - Win32 + ARM - - Release + + Debug + ARM64 + + + Debug Win32 Debug x64 + + Release + ARM + + + Release + ARM64 + + + Release + Win32 + Release x64 @@ -27,7 +43,7 @@ {26D45E58-E703-431D-B67E-493C72C9DA0B} DynamicLibrary winterop - v141 + v142 MultiByte winterop.def Native component of WixToolset.Core diff --git a/src/wixnative/ARM/mergemod.dll b/src/wixnative/ARM/mergemod.dll new file mode 100644 index 00000000..739af831 Binary files /dev/null and b/src/wixnative/ARM/mergemod.dll differ diff --git a/src/wixnative/ARM64/mergemod.dll b/src/wixnative/ARM64/mergemod.dll new file mode 100644 index 00000000..564a75fc Binary files /dev/null and b/src/wixnative/ARM64/mergemod.dll differ diff --git a/src/wixnative/Win32/mergemod.dll b/src/wixnative/Win32/mergemod.dll index 2bf647b2..4286df4d 100644 Binary files a/src/wixnative/Win32/mergemod.dll and b/src/wixnative/Win32/mergemod.dll differ diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index e53af651..373c8d06 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -5,18 +5,34 @@ - + Debug - Win32 + ARM - - Release + + Debug + ARM64 + + + Debug Win32 Debug x64 + + Release + ARM + + + Release + ARM64 + + + Release + Win32 + Release x64 @@ -28,7 +44,7 @@ Application Console wixnative - v141 + v142 Unicode diff --git a/src/wixnative/x64/mergemod.dll b/src/wixnative/x64/mergemod.dll index 2f4cdc7c..b6422174 100644 Binary files a/src/wixnative/x64/mergemod.dll and b/src/wixnative/x64/mergemod.dll differ -- cgit v1.2.3-55-g6feb From cca7952f92015a9a87a04c4a19ba21a131228087 Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Wed, 24 Jun 2020 14:34:45 +1000 Subject: Add SourceLink to C++ projects. --- src/winterop/packages.config | 3 +++ src/winterop/winterop.vcxproj | 18 +++++++++++++++--- src/wixnative/packages.config | 3 +++ src/wixnative/wixnative.vcxproj | 14 +++++++++++++- 4 files changed, 34 insertions(+), 4 deletions(-) diff --git a/src/winterop/packages.config b/src/winterop/packages.config index 38ce9597..f3abcd35 100644 --- a/src/winterop/packages.config +++ b/src/winterop/packages.config @@ -1,5 +1,8 @@  + + + \ No newline at end of file diff --git a/src/winterop/winterop.vcxproj b/src/winterop/winterop.vcxproj index be006c38..fbca525d 100644 --- a/src/winterop/winterop.vcxproj +++ b/src/winterop/winterop.vcxproj @@ -2,6 +2,9 @@ + + + @@ -56,6 +59,10 @@ + + + + @@ -84,12 +91,17 @@ - - This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}. + This project references NuGet package(s) that are missing on this computer. Enable NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105.The missing file is {0}. - + + + + + + + diff --git a/src/wixnative/packages.config b/src/wixnative/packages.config index 38ce9597..f3abcd35 100644 --- a/src/wixnative/packages.config +++ b/src/wixnative/packages.config @@ -1,5 +1,8 @@  + + + \ No newline at end of file diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index 373c8d06..430a63a3 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -2,6 +2,9 @@ + + + @@ -56,6 +59,9 @@ + + + @@ -91,5 +97,11 @@ + + + + + + - + \ No newline at end of file -- cgit v1.2.3-55-g6feb From bcda6dbaee97210a35cfbba0bd7081a8d18dbdba Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Wed, 24 Jun 2020 14:35:38 +1000 Subject: Turn tests off in appveyor.yml since they're run in appveyor.cmd --- appveyor.yml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/appveyor.yml b/appveyor.yml index 7c686b04..e4d25586 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -21,6 +21,8 @@ environment: build_script: - appveyor.cmd +test: off + pull_requests: do_not_increment_build_number: true -- cgit v1.2.3-55-g6feb From 1132e3c15314d3921a796464f07fe1398c1a05ec Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Tue, 14 Jul 2020 14:08:55 +1000 Subject: Update dependencies. --- src/WixToolset.Core.Native/WixToolset.Core.Native.csproj | 2 +- src/winterop/packages.config | 4 ++-- src/winterop/winterop.vcxproj | 8 ++++---- src/wixnative/packages.config | 4 ++-- src/wixnative/wixnative.vcxproj | 8 ++++---- 5 files changed, 13 insertions(+), 13 deletions(-) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index 2626f749..ab7c4020 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -33,7 +33,7 @@ - + diff --git a/src/winterop/packages.config b/src/winterop/packages.config index f3abcd35..321d628a 100644 --- a/src/winterop/packages.config +++ b/src/winterop/packages.config @@ -3,6 +3,6 @@ - - + + \ No newline at end of file diff --git a/src/winterop/winterop.vcxproj b/src/winterop/winterop.vcxproj index fbca525d..7eb2d078 100644 --- a/src/winterop/winterop.vcxproj +++ b/src/winterop/winterop.vcxproj @@ -5,7 +5,7 @@ - + @@ -59,7 +59,7 @@ - + @@ -95,8 +95,8 @@ This project references NuGet package(s) that are missing on this computer. Enable NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105.The missing file is {0}. - - + + diff --git a/src/wixnative/packages.config b/src/wixnative/packages.config index f3abcd35..321d628a 100644 --- a/src/wixnative/packages.config +++ b/src/wixnative/packages.config @@ -3,6 +3,6 @@ - - + + \ No newline at end of file diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index 430a63a3..feebdf5b 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -5,7 +5,7 @@ - + @@ -58,7 +58,7 @@ - + @@ -95,8 +95,8 @@ This project references NuGet package(s) that are missing on this computer. Enable NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105.The missing file is {0}. - - + + -- cgit v1.2.3-55-g6feb From 61d04cad656e47a185575af7b19bc02412f3b606 Mon Sep 17 00:00:00 2001 From: Bob Arnson Date: Sat, 19 Sep 2020 21:08:49 -0400 Subject: Remove 32-bit ARM support. --- src/WixToolset.Core.Native/WixToolset.Core.Native.csproj | 4 ---- src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec | 3 --- src/winterop/winterop.vcxproj | 8 -------- src/wixnative/wixnative.vcxproj | 8 -------- 4 files changed, 23 deletions(-) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index ab7c4020..f12d674c 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -9,10 +9,6 @@ - - All - Platform=ARM - All Platform=ARM64 diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index 2e4b491c..0ec4747d 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -19,9 +19,6 @@ - - - diff --git a/src/winterop/winterop.vcxproj b/src/winterop/winterop.vcxproj index 7eb2d078..28393dba 100644 --- a/src/winterop/winterop.vcxproj +++ b/src/winterop/winterop.vcxproj @@ -8,10 +8,6 @@ - - Debug - ARM - Debug ARM64 @@ -24,10 +20,6 @@ Debug x64 - - Release - ARM - Release ARM64 diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index feebdf5b..0d79b80c 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -8,10 +8,6 @@ - - Debug - ARM - Debug ARM64 @@ -24,10 +20,6 @@ Debug x64 - - Release - ARM - Release ARM64 -- cgit v1.2.3-55-g6feb From 2761d399bf45825f7c3c13ac74cd630dfa7c7717 Mon Sep 17 00:00:00 2001 From: Bob Arnson Date: Wed, 28 Oct 2020 19:13:52 -0400 Subject: Strong-name sign WiX assemblies. --- src/CSharp.Build.props | 11 +++++++++++ src/Directory.Build.props | 5 ++++- src/wix.snk | Bin 0 -> 596 bytes 3 files changed, 15 insertions(+), 1 deletion(-) create mode 100644 src/CSharp.Build.props create mode 100644 src/wix.snk diff --git a/src/CSharp.Build.props b/src/CSharp.Build.props new file mode 100644 index 00000000..b12f4c6e --- /dev/null +++ b/src/CSharp.Build.props @@ -0,0 +1,11 @@ + + + + + true + $([System.IO.Path]::GetFullPath($(MSBuildThisFileDirectory)wix.snk)) + + diff --git a/src/Directory.Build.props b/src/Directory.Build.props index e853e22d..f83cc154 100644 --- a/src/Directory.Build.props +++ b/src/Directory.Build.props @@ -8,6 +8,7 @@ Debug false + MSB3246 $(MSBuildProjectName) $([System.IO.Path]::GetFullPath($(MSBuildThisFileDirectory)..\build\)) @@ -21,6 +22,8 @@ WiX Toolset - + + + diff --git a/src/wix.snk b/src/wix.snk new file mode 100644 index 00000000..3908a66a Binary files /dev/null and b/src/wix.snk differ -- cgit v1.2.3-55-g6feb From c88d4e26faa12c2227679a5dc83cd78a1ddd02c0 Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Mon, 14 Dec 2020 14:38:53 -0600 Subject: Enable CheckForOverflowUnderflow. --- src/CSharp.Build.props | 1 + 1 file changed, 1 insertion(+) diff --git a/src/CSharp.Build.props b/src/CSharp.Build.props index b12f4c6e..bcd47a0c 100644 --- a/src/CSharp.Build.props +++ b/src/CSharp.Build.props @@ -5,6 +5,7 @@ --> + true true $([System.IO.Path]::GetFullPath($(MSBuildThisFileDirectory)wix.snk)) -- cgit v1.2.3-55-g6feb From 8296f237c69743d3e6cbdc58fc07e8470562b92e Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Fri, 18 Dec 2020 18:12:45 -0600 Subject: Enable XML doc. --- src/CSharp.Build.props | 1 + src/Directory.Build.targets | 8 ++++++++ src/WixToolset.Core.Native/CabInterop.cs | 4 +++- src/WixToolset.Core.Native/Cabinet.cs | 10 ++++++++-- src/WixToolset.Core.Native/MSIFILEHASHINFO.cs | 15 +++++++++++++++ src/WixToolset.Core.Native/MsmInterop.cs | 3 +++ src/WixToolset.Core.Native/WixToolset.Core.Native.csproj | 6 +++++- src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec | 1 + src/winterop/packages.config | 4 ++-- src/winterop/winterop.vcxproj | 8 ++++---- src/wixnative/packages.config | 4 ++-- src/wixnative/wixnative.vcxproj | 8 ++++---- 12 files changed, 56 insertions(+), 16 deletions(-) diff --git a/src/CSharp.Build.props b/src/CSharp.Build.props index bcd47a0c..81d24ad1 100644 --- a/src/CSharp.Build.props +++ b/src/CSharp.Build.props @@ -8,5 +8,6 @@ true true $([System.IO.Path]::GetFullPath($(MSBuildThisFileDirectory)wix.snk)) + false diff --git a/src/Directory.Build.targets b/src/Directory.Build.targets index dac7452a..cb988931 100644 --- a/src/Directory.Build.targets +++ b/src/Directory.Build.targets @@ -9,6 +9,11 @@ See the original here: https://github.com/dotnet/sdk/issues/1151#issuecomment-385133284 --> + + false + $(OutputPath)\$(AssemblyName).xml + + true $(SolutionPath) @@ -45,4 +50,7 @@ + + + diff --git a/src/WixToolset.Core.Native/CabInterop.cs b/src/WixToolset.Core.Native/CabInterop.cs index 4c1eb93d..69781047 100644 --- a/src/WixToolset.Core.Native/CabInterop.cs +++ b/src/WixToolset.Core.Native/CabInterop.cs @@ -29,6 +29,7 @@ namespace WixToolset.Core.Native /// /// Full path to file to add to cabinet. /// Name of file in cabinet. + /// /// Handle to open cabinet. [DllImport("winterop.dll", EntryPoint = "CreateCabAddFile", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] public static extern void CreateCabAddFile(string file, string token, MSIFILEHASHINFO fileHash, IntPtr contextHandle); @@ -101,7 +102,8 @@ namespace WixToolset.Core.Native /// to be exposed by .NET Frameowkr. /// /// Pointer to a CERT_CONTEXT struct with public key information to hash. - /// Number of file paths in array. + /// + /// [DllImport("winterop.dll", EntryPoint = "HashPublicKeyInfo", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] public static extern void HashPublicKeyInfo(IntPtr certContext, byte[] publicKeyInfoHashed, ref uint sizePublicKeyInfoHashed); diff --git a/src/WixToolset.Core.Native/Cabinet.cs b/src/WixToolset.Core.Native/Cabinet.cs index 2d624658..7e04cbc5 100644 --- a/src/WixToolset.Core.Native/Cabinet.cs +++ b/src/WixToolset.Core.Native/Cabinet.cs @@ -15,19 +15,25 @@ namespace WixToolset.Core.Native private const string CompressionLevelVariable = "WIX_COMPRESSION_LEVEL"; private static readonly char[] TextLineSplitter = new[] { '\t' }; + /// + /// + /// + /// Path of cabinet public Cabinet(string path) { this.Path = path; } + /// + /// Cabinet path. + /// public string Path { get; } /// /// Creates a cabinet. /// - /// Path of cabinet to create. + /// Files to compress. /// Level of compression to apply. - /// Maximum number of files that will be added to cabinet. /// Maximum size of cabinet. /// Maximum threshold for each cabinet. public void Compress(IEnumerable files, CompressionLevel compressionLevel, int maxSize = 0, int maxThresh = 0) diff --git a/src/WixToolset.Core.Native/MSIFILEHASHINFO.cs b/src/WixToolset.Core.Native/MSIFILEHASHINFO.cs index c11b44ea..d5ac1bc0 100644 --- a/src/WixToolset.Core.Native/MSIFILEHASHINFO.cs +++ b/src/WixToolset.Core.Native/MSIFILEHASHINFO.cs @@ -10,10 +10,25 @@ namespace WixToolset.Core.Native [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/MsmInterop.cs b/src/WixToolset.Core.Native/MsmInterop.cs index 6b1ad141..87ed6f02 100644 --- a/src/WixToolset.Core.Native/MsmInterop.cs +++ b/src/WixToolset.Core.Native/MsmInterop.cs @@ -481,6 +481,9 @@ namespace WixToolset.Core.Native [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] public interface IClassFactory { + /// + /// + /// [return: MarshalAs(UnmanagedType.IUnknown)] object CreateInstance(IntPtr unkOuter, [MarshalAs(UnmanagedType.LPStruct)] Guid iid); } diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index f12d674c..2c118d51 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -1,3 +1,6 @@ + + + @@ -6,6 +9,7 @@ Core Native embedded true + true @@ -29,7 +33,7 @@ - + diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index 0ec4747d..33b19bc1 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -18,6 +18,7 @@ + diff --git a/src/winterop/packages.config b/src/winterop/packages.config index 321d628a..eb65b3b3 100644 --- a/src/winterop/packages.config +++ b/src/winterop/packages.config @@ -3,6 +3,6 @@ - - + + \ No newline at end of file diff --git a/src/winterop/winterop.vcxproj b/src/winterop/winterop.vcxproj index 28393dba..ae844fdf 100644 --- a/src/winterop/winterop.vcxproj +++ b/src/winterop/winterop.vcxproj @@ -5,7 +5,7 @@ - + @@ -51,7 +51,7 @@ - + @@ -87,8 +87,8 @@ This project references NuGet package(s) that are missing on this computer. Enable NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105.The missing file is {0}. - - + + diff --git a/src/wixnative/packages.config b/src/wixnative/packages.config index 321d628a..eb65b3b3 100644 --- a/src/wixnative/packages.config +++ b/src/wixnative/packages.config @@ -3,6 +3,6 @@ - - + + \ No newline at end of file diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index 0d79b80c..681b2e1e 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -5,7 +5,7 @@ - + @@ -50,7 +50,7 @@ - + @@ -87,8 +87,8 @@ This project references NuGet package(s) that are missing on this computer. Enable NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105.The missing file is {0}. - - + + -- cgit v1.2.3-55-g6feb From 533fb3c24290f5c9684a661e2576d857fbee9fb6 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Mon, 15 Mar 2021 15:46:44 -0700 Subject: Modernize build system --- .gitignore | 43 ++++++-- appveyor.cmd | 12 ++- appveyor.yml | 6 +- src/CSharp.Build.props | 13 --- src/Cpp.Build.props | 86 ---------------- src/Directory.Build.props | 4 +- src/Directory.Build.targets | 9 +- src/Directory.csproj.props | 13 +++ src/Directory.csproj.targets | 26 +++++ src/Directory.vcxproj.props | 111 +++++++++++++++++++++ .../WixToolset.Core.Native.csproj | 83 ++++----------- .../WixToolset.Core.Native.nuspec | 14 +-- .../WixToolsetTest.Core.Native.csproj | 7 +- src/wixnative/wixnative.vcxproj | 49 +++------ 14 files changed, 237 insertions(+), 239 deletions(-) delete mode 100644 src/CSharp.Build.props delete mode 100644 src/Cpp.Build.props create mode 100644 src/Directory.csproj.props create mode 100644 src/Directory.csproj.targets create mode 100644 src/Directory.vcxproj.props diff --git a/.gitignore b/.gitignore index 3e8a1553..1ee53850 100644 --- a/.gitignore +++ b/.gitignore @@ -13,6 +13,9 @@ # User-specific files (MonoDevelop/Xamarin Studio) *.userprefs +# Mono auto generated files +mono_crash.* + # Build results [Dd]ebug/ [Dd]ebugPublic/ @@ -20,12 +23,14 @@ [Rr]eleases/ x64/ x86/ +[Ww][Ii][Nn]32/ [Aa][Rr][Mm]/ [Aa][Rr][Mm]64/ bld/ [Bb]in/ [Oo]bj/ [Ll]og/ +[Ll]ogs/ # Visual Studio 2015/2017 cache/options directory .vs/ @@ -39,9 +44,10 @@ Generated\ Files/ [Tt]est[Rr]esult*/ [Bb]uild[Ll]og.* -# NUNIT +# NUnit *.VisualState.xml TestResult.xml +nunit-*.xml # Build Results of an ATL Project [Dd]ebugPS/ @@ -56,6 +62,9 @@ project.lock.json project.fragment.lock.json artifacts/ +# ASP.NET Scaffolding +ScaffoldingReadMe.txt + # StyleCop StyleCopReport.xml @@ -122,9 +131,6 @@ _ReSharper*/ *.[Rr]e[Ss]harper *.DotSettings.user -# JustCode is a .NET coding add-in -.JustCode - # TeamCity is a build add-in _TeamCity* @@ -135,6 +141,11 @@ _TeamCity* .axoCover/* !.axoCover/settings.json +# Coverlet is a free, cross platform Code Coverage Tool +coverage*.json +coverage*.xml +coverage*.info + # Visual Studio code coverage results *.coverage *.coveragexml @@ -182,6 +193,8 @@ PublishScripts/ # NuGet Packages *.nupkg +# NuGet Symbol Packages +*.snupkg # The packages folder can be ignored because of Package Restore **/[Pp]ackages/* # except build/, which is used as an MSBuild target. @@ -206,6 +219,8 @@ BundleArtifacts/ Package.StoreAssociation.xml _pkginfo.txt *.appx +*.appxbundle +*.appxupload # Visual Studio cache files # files ending in .cache can be ignored @@ -231,8 +246,6 @@ orleans.codegen.cs # Since there are multiple workflows, uncomment next line to ignore bower_components # (https://github.com/github/gitignore/pull/1529#issuecomment-104372622) #bower_components/ -# ASP.NET Core default setup: bower directory is configured as wwwroot/lib/ and bower restore is true -**/wwwroot/lib/ # RIA/Silverlight projects Generated_Code/ @@ -257,6 +270,9 @@ ServiceFabricBackup/ *.bim.layout *.bim_*.settings *.rptproj.rsuser +*- [Bb]ackup.rdl +*- [Bb]ackup ([0-9]).rdl +*- [Bb]ackup ([0-9][0-9]).rdl # Microsoft Fakes FakesAssemblies/ @@ -292,10 +308,6 @@ paket-files/ # FAKE - F# Make .fake/ -# JetBrains Rider -.idea/ -*.sln.iml - # CodeRush personal settings .cr/personal @@ -337,5 +349,14 @@ ASALocalRun/ # Local History for Visual Studio .localhistory/ -# BeatPulse healthcheck temp database +# BeatPulse healthcheck temp database healthchecksdb + +# Backup folder for Package Reference Convert tool in Visual Studio 2017 +MigrationBackup/ + +# Ionide (cross platform F# VS Code tools) working folder +.ionide/ + +# Fody - auto-generated XML schema +FodyWeavers.xsd diff --git a/appveyor.cmd b/appveyor.cmd index 6de09dad..d9691a42 100644 --- a/appveyor.cmd +++ b/appveyor.cmd @@ -1,13 +1,19 @@ @setlocal @pushd %~dp0 @set _C=Release +@if /i "%1"=="debug" set _C=Debug -nuget restore || exit /b +:: 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 -msbuild -t:Pack -p:Configuration=%_C% src\WixToolset.Core.Native\WixToolset.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 \ No newline at end of file +@endlocal diff --git a/appveyor.yml b/appveyor.yml index e4d25586..dccc2071 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -21,8 +21,6 @@ environment: build_script: - appveyor.cmd -test: off - pull_requests: do_not_increment_build_number: true @@ -32,9 +30,13 @@ nuget: skip_branch_with_pr: true skip_tags: true +test: off + artifacts: - path: build\Release\**\*.nupkg name: nuget +- path: build\Release\**\*.snupkg + name: snupkg notifications: - provider: Slack diff --git a/src/CSharp.Build.props b/src/CSharp.Build.props deleted file mode 100644 index 81d24ad1..00000000 --- a/src/CSharp.Build.props +++ /dev/null @@ -1,13 +0,0 @@ - - - - - true - true - $([System.IO.Path]::GetFullPath($(MSBuildThisFileDirectory)wix.snk)) - false - - diff --git a/src/Cpp.Build.props b/src/Cpp.Build.props deleted file mode 100644 index 9b7a1bb5..00000000 --- a/src/Cpp.Build.props +++ /dev/null @@ -1,86 +0,0 @@ - - - - - - Win32 - $(BaseIntermediateOutputPath)$(Configuration)\$(Platform)\ - $(OutputPath)$(Platform)\ - - - - $([Microsoft.Build.Utilities.ToolLocationHelper]::GetLatestSDKTargetPlatformVersion('Windows', '10.0')) - - - - - $(DisableSpecificCompilerWarnings) - Level4 - $(ProjectDir)inc;$(MSBuildProjectDirectory);$(IntDir);$(SqlCESdkIncludePath);$(ProjectAdditionalIncludeDirectories);%(AdditionalIncludeDirectories) - WIN32;_WINDOWS;_WIN32_MSI=500;_WIN32_WINNT=0x0501;$(ArmPreprocessorDefinitions);$(UnicodePreprocessorDefinitions);_CRT_STDIO_LEGACY_WIDE_SPECIFIERS;_WINSOCK_DEPRECATED_NO_WARNINGS;%(PreprocessorDefinitions) - Use - precomp.h - StdCall - true - false - -YlprecompDefine - /Zc:threadSafeInit- %(AdditionalOptions) - true - - - $(ArmPreprocessorDefinitions);%(PreprocessorDefinitions) - $(ProjectAdditionalResourceIncludeDirectories);%(AdditionalIncludeDirectories) - - - $(OutDir);$(AdditionalMultiTargetLibraryPath);$(ProjectAdditionalLibraryDirectories);%(AdditionalLibraryDirectories) - - - $(ProjectSubSystem) - $(ProjectModuleDefinitionFile) - $(ResourceOnlyDll) - true - $(ProjectAdditionalLinkLibraries);advapi32.lib;comdlg32.lib;user32.lib;oleaut32.lib;gdi32.lib;shell32.lib;ole32.lib;version.lib;%(AdditionalDependencies) - $(OutDir);$(AdditionalMultiTargetLibraryPath);$(ArmLibraryDirectories);$(ProjectAdditionalLinkLibraryDirectories);%(AdditionalLibraryDirectories) - /IGNORE:4099 %(AdditionalOptions) - - - - - - NoExtensions - - - - - CDecl - - - - - OldStyle - true - true - - - - - Disabled - EnableFastChecks - _DEBUG;DEBUG;%(PreprocessorDefinitions) - MultiThreadedDebug - - - - - MinSpace - NDEBUG;%(PreprocessorDefinitions) - true - true - MultiThreaded - - - true - true - - - diff --git a/src/Directory.Build.props b/src/Directory.Build.props index f83cc154..b3c6287c 100644 --- a/src/Directory.Build.props +++ b/src/Directory.Build.props @@ -22,8 +22,6 @@ WiX Toolset - - - + diff --git a/src/Directory.Build.targets b/src/Directory.Build.targets index cb988931..2fcc765a 100644 --- a/src/Directory.Build.targets +++ b/src/Directory.Build.targets @@ -9,11 +9,6 @@ See the original here: https://github.com/dotnet/sdk/issues/1151#issuecomment-385133284 --> - - false - $(OutputPath)\$(AssemblyName).xml - - true $(SolutionPath) @@ -45,12 +40,12 @@ - + - + diff --git a/src/Directory.csproj.props b/src/Directory.csproj.props new file mode 100644 index 00000000..81d24ad1 --- /dev/null +++ b/src/Directory.csproj.props @@ -0,0 +1,13 @@ + + + + + true + true + $([System.IO.Path]::GetFullPath($(MSBuildThisFileDirectory)wix.snk)) + false + + diff --git a/src/Directory.csproj.targets b/src/Directory.csproj.targets new file mode 100644 index 00000000..c3270426 --- /dev/null +++ b/src/Directory.csproj.targets @@ -0,0 +1,26 @@ + + + + + false + $(OutputPath)\$(AssemblyName).xml + + + + + $(PrivateRepositoryUrl.Replace('.git','')) + + $(MSBuildProjectName).nuspec + $(OutputPath)..\ + $(NuspecProperties);Id=$(PackageId);Authors=$(Authors);Copyright=$(Copyright);Description=$(Description);Title=$(Title) + $(NuspecProperties);Version=$(PackageVersion);RepositoryCommit=$(SourceRevisionId);RepositoryType=$(RepositoryType);RepositoryUrl=$(PrivateRepositoryUrl);ProjectFolder=$(MSBuildProjectDirectory)\;ProjectUrl=$(ProjectUrl) + true + snupkg + + + + diff --git a/src/Directory.vcxproj.props b/src/Directory.vcxproj.props new file mode 100644 index 00000000..bcf26c57 --- /dev/null +++ b/src/Directory.vcxproj.props @@ -0,0 +1,111 @@ + + + + + + Win32 + $(BaseIntermediateOutputPath)$(Configuration)\$(Platform)\ + $(OutputPath)$(Platform)\ + + + $(Company) + $(Copyright) + + win-x86;win-x64;win-arm64 + native,Version=v0.0 + + + + $([Microsoft.Build.Utilities.ToolLocationHelper]::GetLatestSDKTargetPlatformVersion('Windows', '10.0')) + + + + + $(DisableSpecificCompilerWarnings) + Level4 + $(ProjectDir)inc;$(MSBuildProjectDirectory);$(IntDir);$(SqlCESdkIncludePath);$(ProjectAdditionalIncludeDirectories);%(AdditionalIncludeDirectories) + WIN32;_WINDOWS;_WIN32_MSI=500;_WIN32_WINNT=0x0501;$(ArmPreprocessorDefinitions);$(UnicodePreprocessorDefinitions);_CRT_STDIO_LEGACY_WIDE_SPECIFIERS;_WINSOCK_DEPRECATED_NO_WARNINGS;%(PreprocessorDefinitions) + Use + precomp.h + StdCall + true + false + -YlprecompDefine + /Zc:threadSafeInit- %(AdditionalOptions) + true + + + $(ArmPreprocessorDefinitions);%(PreprocessorDefinitions) + $(ProjectAdditionalResourceIncludeDirectories);%(AdditionalIncludeDirectories) + + + $(OutDir);$(AdditionalMultiTargetLibraryPath);$(ProjectAdditionalLibraryDirectories);%(AdditionalLibraryDirectories) + + + $(ProjectSubSystem) + $(ProjectModuleDefinitionFile) + $(ResourceOnlyDll) + true + $(ProjectAdditionalLinkLibraries);advapi32.lib;comdlg32.lib;user32.lib;oleaut32.lib;gdi32.lib;shell32.lib;ole32.lib;version.lib;%(AdditionalDependencies) + $(OutDir);$(AdditionalMultiTargetLibraryPath);$(ArmLibraryDirectories);$(ProjectAdditionalLinkLibraryDirectories);%(AdditionalLibraryDirectories) + /IGNORE:4099 %(AdditionalOptions) + + + + + + NoExtensions + + + + + CDecl + + + + + OldStyle + true + true + + + + + Disabled + EnableFastChecks + _DEBUG;DEBUG;%(PreprocessorDefinitions) + MultiThreadedDebug + + + + + + MultiThreadedDebugDll + + + + + MinSpace + NDEBUG;%(PreprocessorDefinitions) + true + true + MultiThreaded + + + true + true + + + + + + MultiThreadedDll + + + + + $(LinkKeyFile) + $(LinkDelaySign) + + + diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index 2c118d51..41e75f99 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -5,82 +5,35 @@ netstandard2.0 - $(MSBuildThisFileName).nuspec - Core Native embedded + WiX Toolset Native Processing true true - - - All - Platform=ARM64 - - - All - Platform=Win32 - - - All - Platform=x64 - + + + + + + + + + + + + + + + - + - + - - - $(MSBuildProjectDir)..\..\build\obj\$(ProjectName)\$(Configuration)\NativeFileList.txt - $(NCrunchOriginalProjectDir)..\..\build\obj\$(ProjectName)\$(Configuration)\NativeFileList.txt - $(MSBuildThisFileDirectory)..\wixnative\ - - - - - <_NativeProjectOutput Include="$(MergeModDirectory)x64\mergemod.dll" /> - <_NativeProjectOutput Include="$(OutputPath)..\x64\wixnative.exe" /> - <_NativeProjectOutput Include="$(OutputPath)..\x64\wixnative.pdb" /> - - - - - - - - - PreserveNewest - %(Filename)%(Extension) - - - - - - - - - - - - - - - - - - - - - - - $(OutputPath)..\ - Id=$(MSBuildThisFileName);Version=$(BuildVersionSimple);Authors=$(Authors);Copyright=$(Copyright);Description=$(Description);RepositoryCommit=$(SourceRevisionId);RepositoryType=$(RepositoryType);RepositoryUrl=$(PrivateRepositoryUrl);MergeModDir=$(MergeModDirectory) - - diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index 33b19bc1..b6fd9790 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -3,16 +3,16 @@ $id$ $version$ + $title$ + $description$ $authors$ - $authors$ MS-RL - https://github.com/wixtoolset/Core.Native false - $description$ $copyright$ + $projectUrl$ - + @@ -20,13 +20,13 @@ - + - + - + diff --git a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj index 9c493d84..77a53c29 100644 --- a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj +++ b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj @@ -9,18 +9,13 @@ - - + - - - - diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index 681b2e1e..64c2e7d0 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -2,11 +2,6 @@ - - - - - Debug @@ -37,34 +32,23 @@ {8497EC72-B8D0-4272-A9D0-7E9D871CEFBF} Application - Console - wixnative v142 Unicode + Console + wixnative + WiX Native Processing + 10.0 - - - - - - - - - - crypt32.lib;cabinet.lib;msi.lib - - - 4996 - + Create @@ -79,21 +63,14 @@ - + + + + + + + - - - This project references NuGet package(s) that are missing on this computer. Enable NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105.The missing file is {0}. - - - - - - - - - - - \ No newline at end of file + -- cgit v1.2.3-55-g6feb From 089a08fd6b9398b0e1040f96b8e24ba81acfe05b Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Tue, 16 Mar 2021 10:34:28 -0700 Subject: Migrate PInvoke to Core.Native out of Core --- src/WixToolset.Core.Native/CabInterop.cs | 248 +++--- src/WixToolset.Core.Native/CabinetFileInfo.cs | 6 +- src/WixToolset.Core.Native/FileSystem.cs | 61 ++ src/WixToolset.Core.Native/Msi/Database.cs | 257 ++++++ 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 | 113 +++ 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 | 397 +++++++++ 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 | 243 +++++ .../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/MsmInterop.cs | 510 ----------- src/WixToolset.Core.Native/Ole32/Storage.cs | 377 ++++++++ src/WixToolset.Core.Native/Ole32/StorageMode.cs | 55 ++ .../PatchAPI/PatchInterop.cs | 990 +++++++++++++++++++++ src/WixToolset.Core.Native/WixNativeExe.cs | 1 - src/test/WixToolsetTest.Core.Native/MsmFixture.cs | 5 +- 34 files changed, 4448 insertions(+), 644 deletions(-) create mode 100644 src/WixToolset.Core.Native/FileSystem.cs create mode 100644 src/WixToolset.Core.Native/Msi/Database.cs create mode 100644 src/WixToolset.Core.Native/Msi/InstallLogModes.cs create mode 100644 src/WixToolset.Core.Native/Msi/InstallMessage.cs create mode 100644 src/WixToolset.Core.Native/Msi/InstallUILevels.cs create mode 100644 src/WixToolset.Core.Native/Msi/Installer.cs create mode 100644 src/WixToolset.Core.Native/Msi/ModifyView.cs create mode 100644 src/WixToolset.Core.Native/Msi/MsiException.cs create mode 100644 src/WixToolset.Core.Native/Msi/MsiHandle.cs create mode 100644 src/WixToolset.Core.Native/Msi/MsiInterop.cs create mode 100644 src/WixToolset.Core.Native/Msi/OpenDatabase.cs create mode 100644 src/WixToolset.Core.Native/Msi/Record.cs create mode 100644 src/WixToolset.Core.Native/Msi/Session.cs create mode 100644 src/WixToolset.Core.Native/Msi/SummaryInformation.cs create mode 100644 src/WixToolset.Core.Native/Msi/TransformErrorConditions.cs create mode 100644 src/WixToolset.Core.Native/Msi/TransformValidations.cs create mode 100644 src/WixToolset.Core.Native/Msi/View.cs create mode 100644 src/WixToolset.Core.Native/Msi/WixInvalidIdtException.cs create mode 100644 src/WixToolset.Core.Native/Msm/ConfigurationCallback.cs create mode 100644 src/WixToolset.Core.Native/Msm/IMsmConfigureModule.cs create mode 100644 src/WixToolset.Core.Native/Msm/IMsmError.cs create mode 100644 src/WixToolset.Core.Native/Msm/IMsmErrors.cs create mode 100644 src/WixToolset.Core.Native/Msm/IMsmMerge2.cs create mode 100644 src/WixToolset.Core.Native/Msm/IMsmStrings.cs create mode 100644 src/WixToolset.Core.Native/Msm/MsmErrorType.cs create mode 100644 src/WixToolset.Core.Native/Msm/MsmInterop.cs delete mode 100644 src/WixToolset.Core.Native/MsmInterop.cs create mode 100644 src/WixToolset.Core.Native/Ole32/Storage.cs create mode 100644 src/WixToolset.Core.Native/Ole32/StorageMode.cs create mode 100644 src/WixToolset.Core.Native/PatchAPI/PatchInterop.cs diff --git a/src/WixToolset.Core.Native/CabInterop.cs b/src/WixToolset.Core.Native/CabInterop.cs index 69781047..e08c1b90 100644 --- a/src/WixToolset.Core.Native/CabInterop.cs +++ b/src/WixToolset.Core.Native/CabInterop.cs @@ -3,132 +3,8 @@ namespace WixToolset.Core.Native { using System; - using System.Diagnostics.CodeAnalysis; using System.Runtime.InteropServices; - /// - /// The native methods. - /// - public sealed class NativeMethods - { - /// - /// Starts creating a cabinet. - /// - /// Name of cabinet to create. - /// Directory to create cabinet in. - /// Maximum number of files that will be added to cabinet. - /// Maximum size of the cabinet. - /// Maximum threshold in the cabinet. - /// Type of compression to use in the cabinet. - /// Handle to opened cabinet. - [DllImport("winterop.dll", EntryPoint = "CreateCabBegin", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void CreateCabBegin(string cabinetName, string cabinetDirectory, uint maxFiles, uint maxSize, uint maxThreshold, uint compressionType, out IntPtr contextHandle); - - /// - /// Adds a file to an open cabinet. - /// - /// Full path to file to add to cabinet. - /// Name of file in cabinet. - /// - /// Handle to open cabinet. - [DllImport("winterop.dll", EntryPoint = "CreateCabAddFile", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void CreateCabAddFile(string file, string token, MSIFILEHASHINFO fileHash, IntPtr contextHandle); - - /// - /// Closes a cabinet. - /// - /// Handle to open cabinet to close. - /// Address of Binder's cabinet split callback - [DllImport("winterop.dll", EntryPoint = "CreateCabFinish", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void CreateCabFinish(IntPtr contextHandle, IntPtr newCabNamesCallBackAddress); - - /// - /// Cancels cabinet creation. - /// - /// Handle to open cabinet to cancel. - [DllImport("winterop.dll", EntryPoint = "CreateCabCancel", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void CreateCabCancel(IntPtr contextHandle); - - /// - /// Initializes cabinet extraction. - /// - [DllImport("winterop.dll", EntryPoint = "ExtractCabBegin", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void ExtractCabBegin(); - - /// - /// Extracts files from cabinet. - /// - /// Path to cabinet to extract files from. - /// Directory to extract files to. - [DllImport("winterop.dll", EntryPoint = "ExtractCab", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true, PreserveSig = false)] - public static extern void ExtractCab(string cabinet, string extractDirectory); - - /// - /// Cleans up after cabinet extraction. - /// - [DllImport("winterop.dll", EntryPoint = "ExtractCabFinish", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true)] - public static extern void ExtractCabFinish(); - - /// - /// Initializes cabinet enumeration. - /// - [DllImport("winterop.dll", EntryPoint = "EnumerateCabBegin", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void EnumerateCabBegin(); - - /// - /// Enumerates files from cabinet. - /// - /// Path to cabinet to enumerate files from. - /// callback that gets each file. - [DllImport("winterop.dll", EntryPoint = "EnumerateCab", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true, PreserveSig = false)] - public static extern void EnumerateCab(string cabinet, CabInterop.PFNNOTIFY notify); - - /// - /// Cleans up after cabinet enumeration. - /// - [DllImport("winterop.dll", EntryPoint = "EnumerateCabFinish", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true)] - public static extern void EnumerateCabFinish(); - - /// - /// Resets the DACL on an array of files to "empty". - /// - /// Array of file reset ACL to "empty". - /// Number of file paths in array. - [DllImport("winterop.dll", EntryPoint = "ResetAcls", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void ResetAcls(string[] files, uint fileCount); - - /// - /// Gets the hash of the pCertContext->pCertInfo->SubjectPublicKeyInfo using ::CryptHashPublicKeyInfo() which does not seem - /// to be exposed by .NET Frameowkr. - /// - /// Pointer to a CERT_CONTEXT struct with public key information to hash. - /// - /// - [DllImport("winterop.dll", EntryPoint = "HashPublicKeyInfo", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void HashPublicKeyInfo(IntPtr certContext, byte[] publicKeyInfoHashed, ref uint sizePublicKeyInfoHashed); - - /// - /// Converts file time to a local file time. - /// - /// file time - /// local file time - /// true if successful, false otherwise - [DllImport("kernel32.dll", SetLastError = true)] - [return: MarshalAs(UnmanagedType.Bool)] - public 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)] - [return: MarshalAs(UnmanagedType.Bool)] - public static extern bool FileTimeToDosDateTime(ref long fileTime, out ushort wFatDate, out ushort wFatTime); - } - /// /// Interop class for the winterop.dll. /// @@ -180,7 +56,6 @@ namespace WixToolset.Core.Native /// /// Wraps FDINOTIFICATION. /// - [SuppressMessage("Microsoft.Performance", "CA1812:AvoidUninstantiatedInternalClasses")] [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)] public class NOTIFICATION { @@ -307,5 +182,128 @@ namespace WixToolset.Core.Native get { return this.fdie; } } } + + /// + /// The native methods. + /// + private class NativeMethods + { + /// + /// Starts creating a cabinet. + /// + /// Name of cabinet to create. + /// Directory to create cabinet in. + /// Maximum number of files that will be added to cabinet. + /// Maximum size of the cabinet. + /// Maximum threshold in the cabinet. + /// Type of compression to use in the cabinet. + /// Handle to opened cabinet. + [DllImport("winterop.dll", EntryPoint = "CreateCabBegin", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void CreateCabBegin(string cabinetName, string cabinetDirectory, uint maxFiles, uint maxSize, uint maxThreshold, uint compressionType, out IntPtr contextHandle); + + /// + /// Adds a file to an open cabinet. + /// + /// Full path to file to add to cabinet. + /// Name of file in cabinet. + /// + /// Handle to open cabinet. + [DllImport("winterop.dll", EntryPoint = "CreateCabAddFile", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void CreateCabAddFile(string file, string token, MSIFILEHASHINFO fileHash, IntPtr contextHandle); + + /// + /// Closes a cabinet. + /// + /// Handle to open cabinet to close. + /// Address of Binder's cabinet split callback + [DllImport("winterop.dll", EntryPoint = "CreateCabFinish", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void CreateCabFinish(IntPtr contextHandle, IntPtr newCabNamesCallBackAddress); + + /// + /// Cancels cabinet creation. + /// + /// Handle to open cabinet to cancel. + [DllImport("winterop.dll", EntryPoint = "CreateCabCancel", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void CreateCabCancel(IntPtr contextHandle); + + /// + /// Initializes cabinet extraction. + /// + [DllImport("winterop.dll", EntryPoint = "ExtractCabBegin", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void ExtractCabBegin(); + + /// + /// Extracts files from cabinet. + /// + /// Path to cabinet to extract files from. + /// Directory to extract files to. + [DllImport("winterop.dll", EntryPoint = "ExtractCab", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true, PreserveSig = false)] + public static extern void ExtractCab(string cabinet, string extractDirectory); + + /// + /// Cleans up after cabinet extraction. + /// + [DllImport("winterop.dll", EntryPoint = "ExtractCabFinish", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true)] + public static extern void ExtractCabFinish(); + + /// + /// Initializes cabinet enumeration. + /// + [DllImport("winterop.dll", EntryPoint = "EnumerateCabBegin", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void EnumerateCabBegin(); + + /// + /// Enumerates files from cabinet. + /// + /// Path to cabinet to enumerate files from. + /// callback that gets each file. + [DllImport("winterop.dll", EntryPoint = "EnumerateCab", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true, PreserveSig = false)] + public static extern void EnumerateCab(string cabinet, CabInterop.PFNNOTIFY notify); + + /// + /// Cleans up after cabinet enumeration. + /// + [DllImport("winterop.dll", EntryPoint = "EnumerateCabFinish", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true)] + public static extern void EnumerateCabFinish(); + + /// + /// Resets the DACL on an array of files to "empty". + /// + /// Array of file reset ACL to "empty". + /// Number of file paths in array. + [DllImport("winterop.dll", EntryPoint = "ResetAcls", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void ResetAcls(string[] files, uint fileCount); + + /// + /// Gets the hash of the pCertContext->pCertInfo->SubjectPublicKeyInfo using ::CryptHashPublicKeyInfo() which does not seem + /// to be exposed by .NET Frameowkr. + /// + /// Pointer to a CERT_CONTEXT struct with public key information to hash. + /// + /// + [DllImport("winterop.dll", EntryPoint = "HashPublicKeyInfo", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] + public static extern void HashPublicKeyInfo(IntPtr certContext, byte[] publicKeyInfoHashed, ref uint sizePublicKeyInfoHashed); + + /// + /// Converts file time to a local file time. + /// + /// file time + /// local file time + /// true if successful, false otherwise + [DllImport("kernel32.dll", SetLastError = true)] + [return: MarshalAs(UnmanagedType.Bool)] + public 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)] + [return: MarshalAs(UnmanagedType.Bool)] + public static extern bool FileTimeToDosDateTime(ref long fileTime, out ushort wFatDate, out ushort wFatTime); + } } } diff --git a/src/WixToolset.Core.Native/CabinetFileInfo.cs b/src/WixToolset.Core.Native/CabinetFileInfo.cs index ea229121..52f28ad4 100644 --- a/src/WixToolset.Core.Native/CabinetFileInfo.cs +++ b/src/WixToolset.Core.Native/CabinetFileInfo.cs @@ -56,11 +56,7 @@ namespace WixToolset.Core.Native /// public bool SameAsDateTime(DateTime dateTime) { - long filetime = dateTime.ToFileTime(); - long localTime = 0; - NativeMethods.FileTimeToLocalFileTime(ref filetime, ref localTime); - NativeMethods.FileTimeToDosDateTime(ref localTime, out var cabDate, out var cabTime); - + CabInterop.DateTimeToCabDateAndTime(dateTime, out var cabDate, out var cabTime); return this.Date == cabDate && this.Time == cabTime; } } diff --git a/src/WixToolset.Core.Native/FileSystem.cs b/src/WixToolset.Core.Native/FileSystem.cs new file mode 100644 index 00000000..b9691d44 --- /dev/null +++ b/src/WixToolset.Core.Native/FileSystem.cs @@ -0,0 +1,61 @@ +// 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.Runtime.InteropServices; + + /// + /// 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) + { + if (File.Exists(destination)) + { + File.Delete(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) + { + if (File.Exists(destination)) + { + File.Delete(destination); + } + + var directory = Path.GetDirectoryName(destination); + if (!String.IsNullOrEmpty(directory)) + { + Directory.CreateDirectory(directory); + } + + File.Move(source, destination); + } + + [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/Msi/Database.cs b/src/WixToolset.Core.Native/Msi/Database.cs new file mode 100644 index 00000000..a44e8cf9 --- /dev/null +++ b/src/WixToolset.Core.Native/Msi/Database.cs @@ -0,0 +1,257 @@ +// 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.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)) + { + var value = summaryInfo.GetProperty((int)SummaryInformation.Transform.ValidationFlags); + try + { + var validationFlags = Int32.Parse(value, CultureInfo.InvariantCulture); + 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. + public void Merge(Database mergeDatabase, string tableName) + { + var error = MsiInterop.MsiDatabaseMerge(this.Handle, mergeDatabase.Handle, tableName); + if (0 != error) + { + throw new MsiException(error); + } + } + + /// + /// 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 new file mode 100644 index 00000000..f7012b35 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msi/InstallMessage.cs b/src/WixToolset.Core.Native/Msi/InstallMessage.cs new file mode 100644 index 00000000..35773e13 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msi/InstallUILevels.cs b/src/WixToolset.Core.Native/Msi/InstallUILevels.cs new file mode 100644 index 00000000..e84b5215 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msi/Installer.cs b/src/WixToolset.Core.Native/Msi/Installer.cs new file mode 100644 index 00000000..2bb41078 --- /dev/null +++ b/src/WixToolset.Core.Native/Msi/Installer.cs @@ -0,0 +1,113 @@ +// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. + +namespace WixToolset.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 + { + /// + /// 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/ModifyView.cs b/src/WixToolset.Core.Native/Msi/ModifyView.cs new file mode 100644 index 00000000..989de174 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msi/MsiException.cs b/src/WixToolset.Core.Native/Msi/MsiException.cs new file mode 100644 index 00000000..07c83d81 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msi/MsiHandle.cs b/src/WixToolset.Core.Native/Msi/MsiHandle.cs new file mode 100644 index 00000000..dc2ce605 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msi/MsiInterop.cs b/src/WixToolset.Core.Native/Msi/MsiInterop.cs new file mode 100644 index 00000000..0d16fcb2 --- /dev/null +++ b/src/WixToolset.Core.Native/Msi/MsiInterop.cs @@ -0,0 +1,397 @@ +// 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 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 new file mode 100644 index 00000000..18a78f77 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msi/Record.cs b/src/WixToolset.Core.Native/Msi/Record.cs new file mode 100644 index 00000000..c25e76e2 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msi/Session.cs b/src/WixToolset.Core.Native/Msi/Session.cs new file mode 100644 index 00000000..743fb2be --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msi/SummaryInformation.cs b/src/WixToolset.Core.Native/Msi/SummaryInformation.cs new file mode 100644 index 00000000..a7ba5717 --- /dev/null +++ b/src/WixToolset.Core.Native/Msi/SummaryInformation.cs @@ -0,0 +1,243 @@ +// 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 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("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("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 property. + /// + /// Index of the summary information property. + /// The summary information property. + public string GetProperty(int index) + { + var bufSize = 64; + var stringValue = new StringBuilder(bufSize); + + FILETIME timeValue; + timeValue.dwHighDateTime = 0; + timeValue.dwLowDateTime = 0; + + var error = MsiInterop.MsiSummaryInfoGetProperty(this.Handle, index, out var dataType, out var 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); + } + + 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(); + } + } + + /// + /// 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 new file mode 100644 index 00000000..313dceeb --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msi/TransformValidations.cs b/src/WixToolset.Core.Native/Msi/TransformValidations.cs new file mode 100644 index 00000000..52bddeaf --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msi/View.cs b/src/WixToolset.Core.Native/Msi/View.cs new file mode 100644 index 00000000..6305a9de --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msi/WixInvalidIdtException.cs b/src/WixToolset.Core.Native/Msi/WixInvalidIdtException.cs new file mode 100644 index 00000000..268ddc11 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msm/ConfigurationCallback.cs b/src/WixToolset.Core.Native/Msm/ConfigurationCallback.cs new file mode 100644 index 00000000..31b06d02 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msm/IMsmConfigureModule.cs b/src/WixToolset.Core.Native/Msm/IMsmConfigureModule.cs new file mode 100644 index 00000000..468fb1fc --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msm/IMsmError.cs b/src/WixToolset.Core.Native/Msm/IMsmError.cs new file mode 100644 index 00000000..4f1325a6 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msm/IMsmErrors.cs b/src/WixToolset.Core.Native/Msm/IMsmErrors.cs new file mode 100644 index 00000000..e1472376 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msm/IMsmMerge2.cs b/src/WixToolset.Core.Native/Msm/IMsmMerge2.cs new file mode 100644 index 00000000..400249e7 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msm/IMsmStrings.cs b/src/WixToolset.Core.Native/Msm/IMsmStrings.cs new file mode 100644 index 00000000..41063bfa --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msm/MsmErrorType.cs b/src/WixToolset.Core.Native/Msm/MsmErrorType.cs new file mode 100644 index 00000000..c67d37b4 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Msm/MsmInterop.cs b/src/WixToolset.Core.Native/Msm/MsmInterop.cs new file mode 100644 index 00000000..d2627904 --- /dev/null +++ b/src/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/WixToolset.Core.Native/MsmInterop.cs b/src/WixToolset.Core.Native/MsmInterop.cs deleted file mode 100644 index 87ed6f02..00000000 --- a/src/WixToolset.Core.Native/MsmInterop.cs +++ /dev/null @@ -1,510 +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.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, - } - - /// - /// 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; - } - } - - /// - /// 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; - } - } - - /// - /// 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; - } - } - - /// - /// 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; - } - } - - /// - /// 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); - } - - /// - /// 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 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 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 new file mode 100644 index 00000000..3e4c6af2 --- /dev/null +++ b/src/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/WixToolset.Core.Native/Ole32/StorageMode.cs b/src/WixToolset.Core.Native/Ole32/StorageMode.cs new file mode 100644 index 00000000..24b60e4d --- /dev/null +++ b/src/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/WixToolset.Core.Native/PatchAPI/PatchInterop.cs b/src/WixToolset.Core.Native/PatchAPI/PatchInterop.cs new file mode 100644 index 00000000..04f5a553 --- /dev/null +++ b/src/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/WixToolset.Core.Native/WixNativeExe.cs b/src/WixToolset.Core.Native/WixNativeExe.cs index eaa2b2e0..9ae758ca 100644 --- a/src/WixToolset.Core.Native/WixNativeExe.cs +++ b/src/WixToolset.Core.Native/WixNativeExe.cs @@ -8,7 +8,6 @@ namespace WixToolset.Core.Native using System.Diagnostics; using System.IO; using System.Reflection; - using System.Runtime.InteropServices; internal class WixNativeExe { diff --git a/src/test/WixToolsetTest.Core.Native/MsmFixture.cs b/src/test/WixToolsetTest.Core.Native/MsmFixture.cs index a1e42d94..709d4b93 100644 --- a/src/test/WixToolsetTest.Core.Native/MsmFixture.cs +++ b/src/test/WixToolsetTest.Core.Native/MsmFixture.cs @@ -2,7 +2,7 @@ namespace WixToolsetTest.CoreNative { - using WixToolset.Core.Native; + using WixToolset.Core.Native.Msm; using Xunit; public class MsmFixture @@ -10,8 +10,7 @@ namespace WixToolsetTest.CoreNative [Fact] public void CanCreateMsmInterface() { - var msm = new MsmInterop(); - var merge = msm.GetMsmMerge(); + var merge = MsmInterop.GetMsmMerge(); Assert.NotNull(merge); } } -- cgit v1.2.3-55-g6feb From ecf0f8e0a3038e65d18cb3ace71b845af27407ae Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Tue, 16 Mar 2021 10:48:29 -0700 Subject: Implement validation and fix abandoned validation mutex Fixes wixtoolset/issues#5946 Fixes wixtoolset/issues#6366 --- .../IWindowsInstallerValidatorCallback.cs | 27 ++ src/WixToolset.Core.Native/ValidationMessage.cs | 47 +++ .../ValidationMessageType.cs | 31 ++ .../WindowsInstallerValidator.cs | 423 +++++++++++++++++++++ .../WixToolset.Core.Native.csproj | 7 +- .../WixToolset.Core.Native.nuspec | 3 + src/WixToolset.Core.Native/cubes/darice.cub | Bin 0 -> 684032 bytes src/WixToolset.Core.Native/cubes/mergemod.cub | Bin 0 -> 483328 bytes 8 files changed, 537 insertions(+), 1 deletion(-) create mode 100644 src/WixToolset.Core.Native/IWindowsInstallerValidatorCallback.cs create mode 100644 src/WixToolset.Core.Native/ValidationMessage.cs create mode 100644 src/WixToolset.Core.Native/ValidationMessageType.cs create mode 100644 src/WixToolset.Core.Native/WindowsInstallerValidator.cs create mode 100644 src/WixToolset.Core.Native/cubes/darice.cub create mode 100644 src/WixToolset.Core.Native/cubes/mergemod.cub diff --git a/src/WixToolset.Core.Native/IWindowsInstallerValidatorCallback.cs b/src/WixToolset.Core.Native/IWindowsInstallerValidatorCallback.cs new file mode 100644 index 00000000..f4aff134 --- /dev/null +++ b/src/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/WixToolset.Core.Native/ValidationMessage.cs b/src/WixToolset.Core.Native/ValidationMessage.cs new file mode 100644 index 00000000..d7137326 --- /dev/null +++ b/src/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/WixToolset.Core.Native/ValidationMessageType.cs b/src/WixToolset.Core.Native/ValidationMessageType.cs new file mode 100644 index 00000000..98635294 --- /dev/null +++ b/src/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/WixToolset.Core.Native/WindowsInstallerValidator.cs b/src/WixToolset.Core.Native/WindowsInstallerValidator.cs new file mode 100644 index 00000000..d013e5f9 --- /dev/null +++ b/src/WixToolset.Core.Native/WindowsInstallerValidator.cs @@ -0,0 +1,423 @@ +// 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; + + var baseCubePath = Path.Combine(Path.GetDirectoryName(typeof(WindowsInstallerValidator).Assembly.Location), CubesFolder); + var cubeFiles = this.CubeFiles.Select(s => Path.Combine(baseCubePath, s)).ToList(); + + 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 cubeFiles) + { + try + { + using (var cubeDatabase = new Database(cubeFile, 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(cubeFile)); + } + + 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/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index 41e75f99..4069b6b4 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -11,6 +11,11 @@ true + + + + + @@ -29,7 +34,7 @@ - + diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index b6fd9790..cbc4f1be 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -20,6 +20,9 @@ + + + diff --git a/src/WixToolset.Core.Native/cubes/darice.cub b/src/WixToolset.Core.Native/cubes/darice.cub new file mode 100644 index 00000000..4292fede Binary files /dev/null and b/src/WixToolset.Core.Native/cubes/darice.cub differ diff --git a/src/WixToolset.Core.Native/cubes/mergemod.cub b/src/WixToolset.Core.Native/cubes/mergemod.cub new file mode 100644 index 00000000..def6dd1a Binary files /dev/null and b/src/WixToolset.Core.Native/cubes/mergemod.cub differ -- cgit v1.2.3-55-g6feb From d056d00e1e1111b6ef68c57f1f03ef6843204723 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Tue, 16 Mar 2021 14:43:36 -0700 Subject: Move ResetAcls to Core.Native from Core Also simplify copying/moving files directory creation. --- src/WixToolset.Core.Native/FileSystem.cs | 35 ++++++++++++++++------ .../WixToolset.Core.Native.csproj | 1 + 2 files changed, 27 insertions(+), 9 deletions(-) diff --git a/src/WixToolset.Core.Native/FileSystem.cs b/src/WixToolset.Core.Native/FileSystem.cs index b9691d44..d843a9e8 100644 --- a/src/WixToolset.Core.Native/FileSystem.cs +++ b/src/WixToolset.Core.Native/FileSystem.cs @@ -3,8 +3,10 @@ namespace WixToolset.Core.Native { using System; + using System.Collections.Generic; using System.IO; using System.Runtime.InteropServices; + using System.Security.AccessControl; /// /// File system helpers. @@ -19,10 +21,7 @@ namespace WixToolset.Core.Native /// Allow hardlinks. public static void CopyFile(string source, string destination, bool allowHardlink) { - if (File.Exists(destination)) - { - File.Delete(destination); - } + EnsureDirectoryWithoutFile(destination); if (!allowHardlink || !CreateHardLink(destination, source, IntPtr.Zero)) { @@ -41,18 +40,36 @@ namespace WixToolset.Core.Native /// The destination file. public static void MoveFile(string source, string destination) { - if (File.Exists(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) { - File.Delete(destination); + new FileInfo(file).SetAccessControl(aclReset); } + } + + private static void EnsureDirectoryWithoutFile(string path) + { + File.Delete(path); + + var directory = Path.GetDirectoryName(path); - var directory = Path.GetDirectoryName(destination); if (!String.IsNullOrEmpty(directory)) { Directory.CreateDirectory(directory); } - - File.Move(source, destination); } [DllImport("Kernel32.dll", CharSet = CharSet.Unicode, SetLastError = true)] diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index 4069b6b4..94ab812a 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -35,6 +35,7 @@ + -- cgit v1.2.3-55-g6feb From 4e3c12a45cb91828ac94963129495033a4402136 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Tue, 16 Mar 2021 16:02:09 -0700 Subject: Include dependency on System.IO.FileSystem.AccessControl --- src/WixToolset.Core.Native/WixToolset.Core.Native.csproj | 2 ++ src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec | 4 +++- 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index 94ab812a..079e9d93 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -35,6 +35,8 @@ + + diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index cbc4f1be..9e1fc72f 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -12,7 +12,9 @@ $projectUrl$ - + + + -- cgit v1.2.3-55-g6feb From 47d6dd96c1cca8668118a5ad9adfa8ed776b2715 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Thu, 18 Mar 2021 00:48:44 -0700 Subject: Ensure ICE cubs are copied to the output --- src/WixToolset.Core.Native/WixToolset.Core.Native.csproj | 1 + src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec | 1 + .../targets/WixToolset.Core.Native.targets | 9 +++++++++ 3 files changed, 11 insertions(+) create mode 100644 src/WixToolset.Core.Native/targets/WixToolset.Core.Native.targets diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index 079e9d93..14de87cd 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -14,6 +14,7 @@ + diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index 9e1fc72f..fa6a32b6 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -22,6 +22,7 @@ + diff --git a/src/WixToolset.Core.Native/targets/WixToolset.Core.Native.targets b/src/WixToolset.Core.Native/targets/WixToolset.Core.Native.targets new file mode 100644 index 00000000..aafbd405 --- /dev/null +++ b/src/WixToolset.Core.Native/targets/WixToolset.Core.Native.targets @@ -0,0 +1,9 @@ + + + + + PreserveNewest + cubes\%(RecursiveDir)%(Filename)%(Extension) + + + -- cgit v1.2.3-55-g6feb From 50829b410e96d8cb62f557342e33f1815df7d82b Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Thu, 18 Mar 2021 09:17:59 -0500 Subject: Deploy .targets file to buildTransitive folder since the assets are needed in the final project. --- src/WixToolset.Core.Native/WixToolset.Core.Native.csproj | 2 ++ src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec | 4 +++- 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index 14de87cd..d53a155b 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -9,6 +9,8 @@ WiX Toolset Native Processing true true + + NU5128 diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec index fa6a32b6..3091ccd5 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.nuspec @@ -22,10 +22,12 @@ - + + + -- cgit v1.2.3-55-g6feb From e62434acc6c6e41bbafd3f1d46b2ed2011a9ad98 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Fri, 19 Mar 2021 07:34:49 -0700 Subject: Remove a bunch of dead code related to winterop.dll --- src/WixToolset.Core.Native/CabInterop.cs | 309 --------------------- src/WixToolset.Core.Native/Cabinet.cs | 89 +----- src/WixToolset.Core.Native/CabinetFileInfo.cs | 2 +- src/WixToolset.Core.Native/DateTimeInterop.cs | 48 ++++ src/WixToolset.Core.Native/MSIFILEHASHINFO.cs | 34 --- src/WixToolset.Core.Native/Msi/MSIFILEHASHINFO.cs | 34 +++ .../WixToolsetTest.Core.Native/CabinetFixture.cs | 24 -- src/winterop/packages.config | 8 - src/winterop/precomp.h | 12 - .../runtime.win-xxx.WixToolset.Core.Native.nuspec | 20 -- src/winterop/winterop.cpp | 216 -------------- src/winterop/winterop.def | 18 -- src/winterop/winterop.vcxproj | 99 ------- 13 files changed, 85 insertions(+), 828 deletions(-) delete mode 100644 src/WixToolset.Core.Native/CabInterop.cs create mode 100644 src/WixToolset.Core.Native/DateTimeInterop.cs delete mode 100644 src/WixToolset.Core.Native/MSIFILEHASHINFO.cs create mode 100644 src/WixToolset.Core.Native/Msi/MSIFILEHASHINFO.cs delete mode 100644 src/winterop/packages.config delete mode 100644 src/winterop/precomp.h delete mode 100644 src/winterop/runtime.win-xxx.WixToolset.Core.Native.nuspec delete mode 100644 src/winterop/winterop.cpp delete mode 100644 src/winterop/winterop.def delete mode 100644 src/winterop/winterop.vcxproj diff --git a/src/WixToolset.Core.Native/CabInterop.cs b/src/WixToolset.Core.Native/CabInterop.cs deleted file mode 100644 index e08c1b90..00000000 --- a/src/WixToolset.Core.Native/CabInterop.cs +++ /dev/null @@ -1,309 +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 winterop.dll. - /// - public static class CabInterop - { - /// - /// Delegate type that's called by cabinet api for every file in cabinet. - /// - /// NOTIFICATIONTYPE - /// NOTIFICATION - /// 0 for success, -1 otherwise - public delegate Int32 PFNNOTIFY(NOTIFICATIONTYPE fdint, NOTIFICATION pfdin); - - /// - /// Wraps FDINOTIFICATIONTYPE. - /// - public enum NOTIFICATIONTYPE : int - { - /// Info about the cabinet. - CABINET_INFO, - /// One or more files are continued. - PARTIAL_FILE, - /// Called for each file in cabinet. - COPY_FILE, - /// Called after all of the data has been written to a target file. - CLOSE_FILE_INFO, - /// A file is continued to the next cabinet. - NEXT_CABINET, - /// Called once after a call to FDICopy() starts scanning a CAB's CFFILE entries, and again when there are no more CFFILE entries. - ENUMERATE, - } - - /// - /// 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; - NativeMethods.FileTimeToLocalFileTime(ref filetime, ref localTime); - NativeMethods.FileTimeToDosDateTime(ref localTime, out cabDate, out cabTime); - } - - /// - /// Wraps FDINOTIFICATION. - /// - [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)] - public class NOTIFICATION - { - private int cb; - [MarshalAs(UnmanagedType.LPStr)] - private string psz1; - [MarshalAs(UnmanagedType.LPStr)] - private string psz2; - [MarshalAs(UnmanagedType.LPStr)] - private string psz3; - private IntPtr pv; - - private IntPtr hf; - - private ushort date; - private ushort time; - private ushort attribs; - private ushort setID; - private ushort cabinet; - private ushort folder; - private int fdie; - - /// - /// Uncompressed size of file. - /// - public int Cb - { - get { return this.cb; } - } - - /// - /// File name in cabinet. - /// - public String Psz1 - { - get { return this.psz1; } - } - - /// - /// Name of next disk. - /// - public string Psz2 - { - get { return this.psz2; } - } - - /// - /// Points to a 256 character buffer. - /// - public string Psz3 - { - get { return this.psz3; } - } - - /// - /// Value for client. - /// - public IntPtr Pv - { - get { return this.pv; } - } - - /// - /// Not used. - /// - public Int32 Hf - { - get { return (Int32)this.hf; } - } - - /// - /// Last modified MS-DOS date. - /// - public ushort Date - { - get { return this.date; } - } - - /// - /// Last modified MS-DOS time. - /// - public ushort Time - { - get { return this.time; } - } - - /// - /// File attributes. - /// - public ushort Attribs - { - get { return this.attribs; } - } - - /// - /// Cabinet set ID (a random 16-bit number). - /// - public ushort SetID - { - get { return this.setID; } - } - - /// - /// Cabinet number within cabinet set (0-based). - /// - public ushort Cabinet - { - get { return this.cabinet; } - } - - /// - /// File's folder index. - /// - public ushort Folder - { - get { return this.folder; } - } - - /// - /// Error code. - /// - public int Fdie - { - get { return this.fdie; } - } - } - - /// - /// The native methods. - /// - private class NativeMethods - { - /// - /// Starts creating a cabinet. - /// - /// Name of cabinet to create. - /// Directory to create cabinet in. - /// Maximum number of files that will be added to cabinet. - /// Maximum size of the cabinet. - /// Maximum threshold in the cabinet. - /// Type of compression to use in the cabinet. - /// Handle to opened cabinet. - [DllImport("winterop.dll", EntryPoint = "CreateCabBegin", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void CreateCabBegin(string cabinetName, string cabinetDirectory, uint maxFiles, uint maxSize, uint maxThreshold, uint compressionType, out IntPtr contextHandle); - - /// - /// Adds a file to an open cabinet. - /// - /// Full path to file to add to cabinet. - /// Name of file in cabinet. - /// - /// Handle to open cabinet. - [DllImport("winterop.dll", EntryPoint = "CreateCabAddFile", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void CreateCabAddFile(string file, string token, MSIFILEHASHINFO fileHash, IntPtr contextHandle); - - /// - /// Closes a cabinet. - /// - /// Handle to open cabinet to close. - /// Address of Binder's cabinet split callback - [DllImport("winterop.dll", EntryPoint = "CreateCabFinish", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void CreateCabFinish(IntPtr contextHandle, IntPtr newCabNamesCallBackAddress); - - /// - /// Cancels cabinet creation. - /// - /// Handle to open cabinet to cancel. - [DllImport("winterop.dll", EntryPoint = "CreateCabCancel", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void CreateCabCancel(IntPtr contextHandle); - - /// - /// Initializes cabinet extraction. - /// - [DllImport("winterop.dll", EntryPoint = "ExtractCabBegin", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void ExtractCabBegin(); - - /// - /// Extracts files from cabinet. - /// - /// Path to cabinet to extract files from. - /// Directory to extract files to. - [DllImport("winterop.dll", EntryPoint = "ExtractCab", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true, PreserveSig = false)] - public static extern void ExtractCab(string cabinet, string extractDirectory); - - /// - /// Cleans up after cabinet extraction. - /// - [DllImport("winterop.dll", EntryPoint = "ExtractCabFinish", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true)] - public static extern void ExtractCabFinish(); - - /// - /// Initializes cabinet enumeration. - /// - [DllImport("winterop.dll", EntryPoint = "EnumerateCabBegin", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void EnumerateCabBegin(); - - /// - /// Enumerates files from cabinet. - /// - /// Path to cabinet to enumerate files from. - /// callback that gets each file. - [DllImport("winterop.dll", EntryPoint = "EnumerateCab", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true, PreserveSig = false)] - public static extern void EnumerateCab(string cabinet, CabInterop.PFNNOTIFY notify); - - /// - /// Cleans up after cabinet enumeration. - /// - [DllImport("winterop.dll", EntryPoint = "EnumerateCabFinish", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = true)] - public static extern void EnumerateCabFinish(); - - /// - /// Resets the DACL on an array of files to "empty". - /// - /// Array of file reset ACL to "empty". - /// Number of file paths in array. - [DllImport("winterop.dll", EntryPoint = "ResetAcls", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void ResetAcls(string[] files, uint fileCount); - - /// - /// Gets the hash of the pCertContext->pCertInfo->SubjectPublicKeyInfo using ::CryptHashPublicKeyInfo() which does not seem - /// to be exposed by .NET Frameowkr. - /// - /// Pointer to a CERT_CONTEXT struct with public key information to hash. - /// - /// - [DllImport("winterop.dll", EntryPoint = "HashPublicKeyInfo", CharSet = CharSet.Unicode, ExactSpelling = true, PreserveSig = false)] - public static extern void HashPublicKeyInfo(IntPtr certContext, byte[] publicKeyInfoHashed, ref uint sizePublicKeyInfoHashed); - - /// - /// Converts file time to a local file time. - /// - /// file time - /// local file time - /// true if successful, false otherwise - [DllImport("kernel32.dll", SetLastError = true)] - [return: MarshalAs(UnmanagedType.Bool)] - public 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)] - [return: MarshalAs(UnmanagedType.Bool)] - public static extern bool FileTimeToDosDateTime(ref long fileTime, out ushort wFatDate, out ushort wFatTime); - } - } -} diff --git a/src/WixToolset.Core.Native/Cabinet.cs b/src/WixToolset.Core.Native/Cabinet.cs index 7e04cbc5..9b77bd37 100644 --- a/src/WixToolset.Core.Native/Cabinet.cs +++ b/src/WixToolset.Core.Native/Cabinet.cs @@ -8,7 +8,7 @@ namespace WixToolset.Core.Native using WixToolset.Data; /// - /// Wrapper class around interop with wixcab.dll to compress files into a cabinet. + /// Cabinet create, enumerate and extract mechanism. /// public sealed class Cabinet { @@ -16,7 +16,7 @@ namespace WixToolset.Core.Native private static readonly char[] TextLineSplitter = new[] { '\t' }; /// - /// + /// Creates a cabinet creation, enumeration, extraction mechanism. /// /// Path of cabinet public Cabinet(string path) @@ -116,90 +116,5 @@ namespace WixToolset.Core.Native var wixnative = new WixNativeExe("extractcab", this.Path, outputFolder); return wixnative.Run().Where(output => !String.IsNullOrWhiteSpace(output)); } - -#if TOOD_ERROR_HANDLING - /// - /// Adds a file to the cabinet with an optional MSI file hash. - /// - /// The file to add. - /// The token for the file. - /// The MSI file hash of the file. - //private void AddFile(string file, string token, MsiInterop.MSIFILEHASHINFO fileHash) - //{ - // try - // { - // NativeMethods.CreateCabAddFile(file, token, fileHash, this.handle); - // } - // catch (COMException ce) - // { - // if (0x80004005 == unchecked((uint)ce.ErrorCode)) // E_FAIL - // { - // throw new WixException(WixErrors.CreateCabAddFileFailed()); - // } - // else if (0x80070070 == unchecked((uint)ce.ErrorCode)) // ERROR_DISK_FULL - // { - // throw new WixException(WixErrors.CreateCabInsufficientDiskSpace()); - // } - // else - // { - // throw; - // } - // } - // catch (DirectoryNotFoundException) - // { - // throw new WixFileNotFoundException(file); - // } - // catch (FileNotFoundException) - // { - // throw new WixFileNotFoundException(file); - // } - //} - - /// - /// Complete/commit the cabinet - this must be called before Dispose so that errors will be - /// reported on the same thread. - /// - /// Address of Binder's callback function for Cabinet Splitting - public void Complete(IntPtr newCabNamesCallBackAddress) - { - if (IntPtr.Zero != this.handle) - { - try - { - if (newCabNamesCallBackAddress != IntPtr.Zero && this.maxSize != 0) - { - NativeMethods.CreateCabFinish(this.handle, newCabNamesCallBackAddress); - } - else - { - NativeMethods.CreateCabFinish(this.handle, IntPtr.Zero); - } - - GC.SuppressFinalize(this); - this.disposed = true; - } - catch (COMException ce) - { - //if (0x80004005 == unchecked((uint)ce.ErrorCode)) // E_FAIL - //{ - // // This error seems to happen, among other situations, when cabbing more than 0xFFFF files - // throw new WixException(WixErrors.FinishCabFailed()); - //} - //else if (0x80070070 == unchecked((uint)ce.ErrorCode)) // ERROR_DISK_FULL - //{ - // throw new WixException(WixErrors.CreateCabInsufficientDiskSpace()); - //} - //else - //{ - // throw; - //} - } - finally - { - this.handle = IntPtr.Zero; - } - } - } -#endif } } diff --git a/src/WixToolset.Core.Native/CabinetFileInfo.cs b/src/WixToolset.Core.Native/CabinetFileInfo.cs index 52f28ad4..07387191 100644 --- a/src/WixToolset.Core.Native/CabinetFileInfo.cs +++ b/src/WixToolset.Core.Native/CabinetFileInfo.cs @@ -56,7 +56,7 @@ namespace WixToolset.Core.Native /// public bool SameAsDateTime(DateTime dateTime) { - CabInterop.DateTimeToCabDateAndTime(dateTime, out var cabDate, out var cabTime); + 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 new file mode 100644 index 00000000..d2a0ba2b --- /dev/null +++ b/src/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/WixToolset.Core.Native/MSIFILEHASHINFO.cs b/src/WixToolset.Core.Native/MSIFILEHASHINFO.cs deleted file mode 100644 index d5ac1bc0..00000000 --- a/src/WixToolset.Core.Native/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 -{ - 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/MSIFILEHASHINFO.cs b/src/WixToolset.Core.Native/Msi/MSIFILEHASHINFO.cs new file mode 100644 index 00000000..ae88ec7e --- /dev/null +++ b/src/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/test/WixToolsetTest.Core.Native/CabinetFixture.cs b/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs index 4f5a3427..2e43dce4 100644 --- a/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs +++ b/src/test/WixToolsetTest.Core.Native/CabinetFixture.cs @@ -46,30 +46,6 @@ namespace WixToolsetTest.CoreNative // Assert.True(file.SameAsDateTime(new DateTime(2017, 9, 28, 0, 19, 38))); } - [Fact] - public void CanExtractSingleFileCabinet() - { - var cabinetPath = TestData.Get(@"TestData\test.cab"); - - using (var fs = new DisposableFileSystem()) - { - var extractFolder = fs.GetFolder(true); - - var cabinet = new Cabinet(cabinetPath); - var reportedFiles = cabinet.Extract(extractFolder); - var files = Directory.EnumerateFiles(extractFolder); - Assert.Equal(reportedFiles, files.Select(f => Path.GetFileName(f))); - - var file = new FileInfo(files.Single()); - CabInterop.DateTimeToCabDateAndTime(file.CreationTime, out var date, out var time); - - Assert.Equal("test.txt", file.Name); - Assert.Equal(17, file.Length); - Assert.Equal(19259, date); - Assert.Equal(47731, time); - } - } - [Fact] public void IntegrationTest() { diff --git a/src/winterop/packages.config b/src/winterop/packages.config deleted file mode 100644 index eb65b3b3..00000000 --- a/src/winterop/packages.config +++ /dev/null @@ -1,8 +0,0 @@ - - - - - - - - \ No newline at end of file diff --git a/src/winterop/precomp.h b/src/winterop/precomp.h deleted file mode 100644 index eba996c7..00000000 --- a/src/winterop/precomp.h +++ /dev/null @@ -1,12 +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 "dutil.h" -#include "fileutil.h" -#include "memutil.h" -#include "strutil.h" -#include "cabcutil.h" -#include "cabutil.h" diff --git a/src/winterop/runtime.win-xxx.WixToolset.Core.Native.nuspec b/src/winterop/runtime.win-xxx.WixToolset.Core.Native.nuspec deleted file mode 100644 index 18676197..00000000 --- a/src/winterop/runtime.win-xxx.WixToolset.Core.Native.nuspec +++ /dev/null @@ -1,20 +0,0 @@ - - - - $id$ - $version$ - $authors$ - $authors$ - - https://licenses.nuget.org/MS-RL - https://github.com/wixtoolset/Core.Native - false - $description$ - $copyright$ - - - - - - - diff --git a/src/winterop/winterop.cpp b/src/winterop/winterop.cpp deleted file mode 100644 index 12d8ca3f..00000000 --- a/src/winterop/winterop.cpp +++ /dev/null @@ -1,216 +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 HashPublicKeyInfo( - __in PCERT_CONTEXT pCertContext, - __in_ecount(*pcbSubjectKeyIndentifier) BYTE* rgbSubjectKeyIdentifier, - __inout DWORD* pcbSubjectKeyIndentifier - ) -{ - HRESULT hr = S_OK; - - if (!::CryptHashPublicKeyInfo(NULL, CALG_SHA1, 0, X509_ASN_ENCODING, &pCertContext->pCertInfo->SubjectPublicKeyInfo, rgbSubjectKeyIdentifier, pcbSubjectKeyIndentifier)) - { - ExitWithLastError(hr, "Failed to hash public key information."); - } - -LExit: - return hr; -} - -HRESULT ResetAcls( - __in LPCWSTR pwzFiles[], - __in DWORD cFiles - ) -{ - HRESULT hr = S_OK; - ACL* pacl = NULL; - DWORD cbAcl = sizeof(ACL); - - OSVERSIONINFO osvi; - - osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO); - if (!::GetVersionExA(&osvi)) - { - ExitOnLastError(hr, "failed to get OS version"); - } - - // If we're running on NT 4 or earlier, or ME or earlier, don't reset ACLs. - if (4 >= osvi.dwMajorVersion) - { - ExitFunction1(hr = S_FALSE); - } - - // create an empty (not NULL!) ACL to use on all the files - pacl = static_cast(MemAlloc(cbAcl, FALSE)); - ExitOnNull(pacl, hr, E_OUTOFMEMORY, "failed to allocate ACL"); - -#pragma prefast(push) -#pragma prefast(disable:25029) - if (!::InitializeAcl(pacl, cbAcl, ACL_REVISION)) -#pragma prefast(op) - { - ExitOnLastError(hr, "failed to initialize ACL"); - } - - // reset the existing security permissions on each file - for (DWORD i = 0; i < cFiles; ++i) - { - hr = ::SetNamedSecurityInfoW(const_cast(pwzFiles[i]), 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) - { - ExitOnFailure(hr = HRESULT_FROM_WIN32(hr), "failed to set security descriptor for file: %S", pwzFiles[i]); - } - } - - // Setting to S_OK because we could end with ERROR_FILE_NOT_FOUND or ERROR_PATH_NOT_FOUND as valid return values. - hr = S_OK; - - AssertSz(::IsValidAcl(pacl), "ResetAcls() - created invalid ACL"); - -LExit: - if (pacl) - { - MemFree(pacl); - } - - return hr; -} - - -HRESULT CreateCabBegin( - __in LPCWSTR wzCab, - __in LPCWSTR wzCabDir, - __in DWORD dwMaxFiles, - __in DWORD dwMaxSize, - __in DWORD dwMaxThresh, - __in COMPRESSION_TYPE ct, - __out HANDLE *phContext - ) -{ - return CabCBegin(wzCab, wzCabDir, dwMaxFiles, dwMaxSize, dwMaxThresh, ct, phContext); -} - - -HRESULT CreateCabAddFile( - __in LPCWSTR wzFile, - __in_opt LPCWSTR wzToken, - __in_opt PMSIFILEHASHINFO pmfHash, - __in HANDLE hContext - ) -{ - return CabCAddFile(wzFile, wzToken, pmfHash, hContext); -} - - -HRESULT CreateCabAddFiles( - __in LPCWSTR pwzFiles[], - __in LPCWSTR pwzTokens[], - __in PMSIFILEHASHINFO pmfHash[], - __in DWORD cFiles, - __in HANDLE hContext - ) -{ - HRESULT hr = S_OK; - DWORD i; - - Assert(pwzFiles); - Assert(hContext); - - for (i = 0; i < cFiles; i++) - { - hr = CreateCabAddFile( - pwzFiles[i], - pwzTokens ? pwzTokens[i] : NULL, - pmfHash[i], - hContext - ); - ExitOnFailure(hr, "Failed to add file %S to cab", pwzFiles[i]); - } - -LExit: - return hr; -} - - -HRESULT CreateCabFinish( - __in HANDLE hContext, - __in_opt FileSplitCabNamesCallback newCabNamesCallBackAddress - ) -{ - // Convert address into Binder callback function - return CabCFinish(hContext, newCabNamesCallBackAddress); -} - - -void CreateCabCancel( - __in HANDLE hContext - ) -{ - CabCCancel(hContext); -} - - -HRESULT ExtractCabBegin() -{ - return CabInitialize(FALSE); -} - - -HRESULT ExtractCab( - __in LPCWSTR wzCabinet, - __in LPCWSTR wzExtractDir - ) -{ - return CabExtract(wzCabinet, L"*", wzExtractDir, NULL, NULL, 0); -} - - -void ExtractCabFinish() -{ - CabUninitialize(); - return; -} - - -HRESULT EnumerateCabBegin() -{ - return CabInitialize(FALSE); -} - - -HRESULT EnumerateCab( - __in LPCWSTR wzCabinet, - __in STDCALL_PFNFDINOTIFY pfnNotify - ) -{ - return CabEnumerate(wzCabinet, L"*", pfnNotify, 0); -} - - -void EnumerateCabFinish() -{ - CabUninitialize(); - return; -} - - -BOOL WINAPI DllMain( - __in HINSTANCE /*hInstance*/, - __in DWORD dwReason, - __in LPVOID /*lpvReserved*/ - ) -{ - switch(dwReason) - { - case DLL_PROCESS_ATTACH: - case DLL_PROCESS_DETACH: - case DLL_THREAD_ATTACH: - case DLL_THREAD_DETACH: - break; - } - - return TRUE; -} diff --git a/src/winterop/winterop.def b/src/winterop/winterop.def deleted file mode 100644 index dffa6268..00000000 --- a/src/winterop/winterop.def +++ /dev/null @@ -1,18 +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. - -LIBRARY "winterop.dll" - -EXPORTS - CreateCabBegin - CreateCabCancel - CreateCabAddFile - CreateCabAddFiles - CreateCabFinish - EnumerateCabBegin - EnumerateCab - EnumerateCabFinish - ExtractCabBegin - ExtractCab - ExtractCabFinish - ResetAcls - HashPublicKeyInfo diff --git a/src/winterop/winterop.vcxproj b/src/winterop/winterop.vcxproj deleted file mode 100644 index ae844fdf..00000000 --- a/src/winterop/winterop.vcxproj +++ /dev/null @@ -1,99 +0,0 @@ - - - - - - - - - - - - Debug - ARM64 - - - Debug - Win32 - - - Debug - x64 - - - Release - ARM64 - - - Release - Win32 - - - Release - x64 - - - - - {26D45E58-E703-431D-B67E-493C72C9DA0B} - DynamicLibrary - winterop - v142 - MultiByte - winterop.def - Native component of WixToolset.Core - - - - - - - - - - - - - - - - - crypt32.lib;cabinet.lib;msi.lib - - - - - - 4996 - Create - - - - - - - - - - - - - - - - - - - - This project references NuGet package(s) that are missing on this computer. Enable NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105.The missing file is {0}. - - - - - - - - - - - -- cgit v1.2.3-55-g6feb From 062d6387692d074f502176296f361c52026b96d5 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Fri, 19 Mar 2021 07:47:43 -0700 Subject: Improve finding files relative to Core.Native assembly Should fix unit testing issues when .cub files cannot be found. --- src/WixToolset.Core.Native/AssemblyExtensions.cs | 70 ++++++++++++++++++++++ .../WindowsInstallerValidator.cs | 16 +++-- src/WixToolset.Core.Native/WixNativeExe.cs | 28 ++------- 3 files changed, 84 insertions(+), 30 deletions(-) create mode 100644 src/WixToolset.Core.Native/AssemblyExtensions.cs diff --git a/src/WixToolset.Core.Native/AssemblyExtensions.cs b/src/WixToolset.Core.Native/AssemblyExtensions.cs new file mode 100644 index 00000000..590a6887 --- /dev/null +++ b/src/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/WixToolset.Core.Native/WindowsInstallerValidator.cs b/src/WixToolset.Core.Native/WindowsInstallerValidator.cs index d013e5f9..9f4b26a3 100644 --- a/src/WixToolset.Core.Native/WindowsInstallerValidator.cs +++ b/src/WixToolset.Core.Native/WindowsInstallerValidator.cs @@ -86,9 +86,6 @@ namespace WixToolset.Core.Native var previousHwnd = IntPtr.Zero; InstallUIHandler previousUIHandler = null; - var baseCubePath = Path.Combine(Path.GetDirectoryName(typeof(WindowsInstallerValidator).Assembly.Location), CubesFolder); - var cubeFiles = this.CubeFiles.Select(s => Path.Combine(baseCubePath, s)).ToList(); - try { using (var database = new Database(this.DatabasePath, OpenDatabase.Direct)) @@ -116,11 +113,18 @@ namespace WixToolset.Core.Native } // Merge in the cube databases. - foreach (var cubeFile in cubeFiles) + 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(cubeFile, OpenDatabase.ReadOnly)) + using (var cubeDatabase = new Database(findCubeFile.Path, OpenDatabase.ReadOnly)) { try { @@ -136,7 +140,7 @@ namespace WixToolset.Core.Native { if (0x6E == e.NativeErrorCode) // ERROR_OPEN_FAILED { - throw new WixException(ErrorMessages.CubeFileNotFound(cubeFile)); + throw new WixException(ErrorMessages.CubeFileNotFound(findCubeFile.Path)); } throw; diff --git a/src/WixToolset.Core.Native/WixNativeExe.cs b/src/WixToolset.Core.Native/WixNativeExe.cs index 9ae758ca..fb41b2f2 100644 --- a/src/WixToolset.Core.Native/WixNativeExe.cs +++ b/src/WixToolset.Core.Native/WixNativeExe.cs @@ -7,7 +7,6 @@ namespace WixToolset.Core.Native using System.ComponentModel; using System.Diagnostics; using System.IO; - using System.Reflection; internal class WixNativeExe { @@ -81,35 +80,16 @@ namespace WixToolset.Core.Native { if (String.IsNullOrEmpty(PathToWixNativeExe)) { - var path = Path.Combine(Path.GetDirectoryName(new Uri(Assembly.GetExecutingAssembly().CodeBase).LocalPath), WixNativeExeFileName); - var possiblePaths = path; + var result = typeof(WixNativeExe).Assembly.FindFileRelativeToAssembly(WixNativeExeFileName, searchNativeDllDirectories: true); - var found = File.Exists(path); - if (!found && AppContext.GetData("NATIVE_DLL_SEARCH_DIRECTORIES") is string searchDirectoriesString) - { - possiblePaths = searchDirectoriesString; - var separatorChar = Path.PathSeparator; - var searchDirectories = searchDirectoriesString?.Split(separatorChar); - foreach (var directoryPath in searchDirectories) - { - var possiblePath = Path.Combine(directoryPath, WixNativeExeFileName); - if (File.Exists(possiblePath)) - { - path = possiblePath; - found = true; - break; - } - } - } - - if (!found) + if (!result.Found) { throw new PlatformNotSupportedException( $"Could not find platform specific '{WixNativeExeFileName}'", - new FileNotFoundException($"Could not find internal piece of WiX Toolset from: {possiblePaths}", WixNativeExeFileName)); + new FileNotFoundException($"Could not find internal piece of WiX Toolset from: {result.PossiblePaths}", WixNativeExeFileName)); } - PathToWixNativeExe = path; + PathToWixNativeExe = result.Path; } } -- cgit v1.2.3-55-g6feb From 343038b42c1c6189ba90d2bcf96177123cb66c8d Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Thu, 25 Mar 2021 09:57:52 -0700 Subject: Add missing MSI functionality required by Core --- src/WixToolset.Core.Native/Msi/Database.cs | 15 +- src/WixToolset.Core.Native/Msi/Installer.cs | 25 +++ src/WixToolset.Core.Native/Msi/MsiInterop.cs | 11 ++ .../Msi/SummaryInformation.cs | 173 ++++++++++++++++++--- 4 files changed, 199 insertions(+), 25 deletions(-) diff --git a/src/WixToolset.Core.Native/Msi/Database.cs b/src/WixToolset.Core.Native/Msi/Database.cs index a44e8cf9..b9c5c35b 100644 --- a/src/WixToolset.Core.Native/Msi/Database.cs +++ b/src/WixToolset.Core.Native/Msi/Database.cs @@ -3,7 +3,6 @@ namespace WixToolset.Core.Native.Msi { using System; - using System.Globalization; using System.IO; using System.Threading; @@ -44,10 +43,9 @@ namespace WixToolset.Core.Native.Msi var conditions = TransformErrorConditions.None; using (var summaryInfo = new SummaryInformation(transformFile)) { - var value = summaryInfo.GetProperty((int)SummaryInformation.Transform.ValidationFlags); try { - var validationFlags = Int32.Parse(value, CultureInfo.InvariantCulture); + var validationFlags = summaryInfo.GetNumericProperty(SummaryInformation.Transform.ValidationFlags); conditions = (TransformErrorConditions)(validationFlags & 0xffff); } catch (FormatException) @@ -192,13 +190,20 @@ namespace WixToolset.Core.Native.Msi /// /// The database to merge into the base database. /// The name of the table to receive merge conflict information. - public void Merge(Database mergeDatabase, string tableName) + /// 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 (0 != error) + if (error == 1627) + { + return true; + } + else if (error != 0) { throw new MsiException(error); } + + return false; } /// diff --git a/src/WixToolset.Core.Native/Msi/Installer.cs b/src/WixToolset.Core.Native/Msi/Installer.cs index 2bb41078..8a45aaa8 100644 --- a/src/WixToolset.Core.Native/Msi/Installer.cs +++ b/src/WixToolset.Core.Native/Msi/Installer.cs @@ -23,6 +23,31 @@ namespace WixToolset.Core.Native.Msi /// 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. /// diff --git a/src/WixToolset.Core.Native/Msi/MsiInterop.cs b/src/WixToolset.Core.Native/Msi/MsiInterop.cs index 0d16fcb2..11ac4094 100644 --- a/src/WixToolset.Core.Native/Msi/MsiInterop.cs +++ b/src/WixToolset.Core.Native/Msi/MsiInterop.cs @@ -152,6 +152,17 @@ namespace WixToolset.Core.Native.Msi [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. /// diff --git a/src/WixToolset.Core.Native/Msi/SummaryInformation.cs b/src/WixToolset.Core.Native/Msi/SummaryInformation.cs index a7ba5717..da629df2 100644 --- a/src/WixToolset.Core.Native/Msi/SummaryInformation.cs +++ b/src/WixToolset.Core.Native/Msi/SummaryInformation.cs @@ -12,6 +12,63 @@ namespace WixToolset.Core.Native.Msi /// 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. /// @@ -140,7 +197,7 @@ namespace WixToolset.Core.Native.Msi { if (null == db) { - throw new ArgumentNullException("db"); + throw new ArgumentNullException(nameof(db)); } uint handle = 0; @@ -160,7 +217,7 @@ namespace WixToolset.Core.Native.Msi { if (null == databaseFile) { - throw new ArgumentNullException("databaseFile"); + throw new ArgumentNullException(nameof(databaseFile)); } uint handle = 0; @@ -173,49 +230,125 @@ namespace WixToolset.Core.Native.Msi } /// - /// Gets a summary information property. + /// Gets a summary information package property. /// - /// Index of the summary information property. + /// The summary information package property. /// The summary information property. - public string GetProperty(int index) + public string GetProperty(Package property) { - var bufSize = 64; - var stringValue = new StringBuilder(bufSize); + return this.GetProperty((int)property); + } - FILETIME timeValue; - timeValue.dwHighDateTime = 0; - timeValue.dwLowDateTime = 0; + /// + /// 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); + } - var error = MsiInterop.MsiSummaryInfoGetProperty(this.Handle, index, out var dataType, out var 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); - } + /// + /// 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); + } - if (0 != error) - { - throw new MsiException(error); - } + /// + /// 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. /// -- cgit v1.2.3-55-g6feb From 6a496fc90dd88ce45803e5cd541a3a0d7da62154 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Tue, 30 Mar 2021 11:47:17 -0700 Subject: Remove removed project from .sln --- WixToolset.Core.Native.sln | 15 --------------- 1 file changed, 15 deletions(-) diff --git a/WixToolset.Core.Native.sln b/WixToolset.Core.Native.sln index e6530399..0d7a5921 100644 --- a/WixToolset.Core.Native.sln +++ b/WixToolset.Core.Native.sln @@ -3,12 +3,7 @@ Microsoft Visual Studio Solution File, Format Version 12.00 # Visual Studio 15 VisualStudioVersion = 15.0.27004.2009 MinimumVisualStudioVersion = 15.0.26124.0 -Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "winterop", "src\winterop\winterop.vcxproj", "{26D45E58-E703-431D-B67E-493C72C9DA0B}" -EndProject Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "WixToolset.Core.Native", "src\WixToolset.Core.Native\WixToolset.Core.Native.csproj", "{C1F36B7C-6A5B-44CB-BD05-3C9CDEC2DD63}" - ProjectSection(ProjectDependencies) = postProject - {26D45E58-E703-431D-B67E-493C72C9DA0B} = {26D45E58-E703-431D-B67E-493C72C9DA0B} - EndProjectSection EndProject Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "wixnative", "src\wixnative\wixnative.vcxproj", "{8497EC72-B8D0-4272-A9D0-7E9D871CEFBF}" EndProject @@ -24,16 +19,6 @@ Global Release|x86 = Release|x86 EndGlobalSection GlobalSection(ProjectConfigurationPlatforms) = postSolution - {26D45E58-E703-431D-B67E-493C72C9DA0B}.Debug|Any CPU.ActiveCfg = Debug|Win32 - {26D45E58-E703-431D-B67E-493C72C9DA0B}.Debug|x64.ActiveCfg = Debug|x64 - {26D45E58-E703-431D-B67E-493C72C9DA0B}.Debug|x64.Build.0 = Debug|x64 - {26D45E58-E703-431D-B67E-493C72C9DA0B}.Debug|x86.ActiveCfg = Debug|Win32 - {26D45E58-E703-431D-B67E-493C72C9DA0B}.Debug|x86.Build.0 = Debug|Win32 - {26D45E58-E703-431D-B67E-493C72C9DA0B}.Release|Any CPU.ActiveCfg = Release|Win32 - {26D45E58-E703-431D-B67E-493C72C9DA0B}.Release|x64.ActiveCfg = Release|x64 - {26D45E58-E703-431D-B67E-493C72C9DA0B}.Release|x64.Build.0 = Release|x64 - {26D45E58-E703-431D-B67E-493C72C9DA0B}.Release|x86.ActiveCfg = Release|Win32 - {26D45E58-E703-431D-B67E-493C72C9DA0B}.Release|x86.Build.0 = Release|Win32 {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 -- cgit v1.2.3-55-g6feb From ae224f38e8349e2d0dfce95918571db4f18b5fba Mon Sep 17 00:00:00 2001 From: Sean Hall Date: Wed, 28 Apr 2021 16:58:44 -0500 Subject: Update to latest dutil. --- src/wixnative/packages.config | 2 +- src/wixnative/wixnative.vcxproj | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/wixnative/packages.config b/src/wixnative/packages.config index eb65b3b3..a98c0c8e 100644 --- a/src/wixnative/packages.config +++ b/src/wixnative/packages.config @@ -4,5 +4,5 @@ - + \ No newline at end of file diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index 64c2e7d0..09dc97fa 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -67,7 +67,7 @@ - + -- cgit v1.2.3-55-g6feb From a2f0de28fc0f1ab71d4685c77f0b21d946f3e702 Mon Sep 17 00:00:00 2001 From: Rob Mensching Date: Fri, 16 Apr 2021 23:35:06 -0700 Subject: Introduce new versioning system based on GitInfo --- appveyor.yml | 4 +- signing.json | 13 +++ src/Directory.Build.props | 5 +- src/Directory.Build.targets | 120 +++++++++++++++------ src/Directory.csproj.targets | 15 +-- src/Directory.vcxproj.props | 9 +- src/Directory.vcxproj.targets | 45 ++++++++ .../WixToolset.Core.Native.csproj | 3 +- .../WixToolsetTest.Core.Native.csproj | 2 + src/test/version.txt | 1 + src/ver.rc | 55 ++++++++++ src/wixnative/wixnative.vcxproj | 2 +- version.json | 11 -- version.txt | 1 + 14 files changed, 223 insertions(+), 63 deletions(-) create mode 100644 signing.json create mode 100644 src/Directory.vcxproj.targets create mode 100644 src/test/version.txt create mode 100644 src/ver.rc delete mode 100644 version.json create mode 100644 version.txt diff --git a/appveyor.yml b/appveyor.yml index dccc2071..364569cf 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -21,6 +21,8 @@ environment: build_script: - appveyor.cmd +test: off + pull_requests: do_not_increment_build_number: true @@ -30,8 +32,6 @@ nuget: skip_branch_with_pr: true skip_tags: true -test: off - artifacts: - path: build\Release\**\*.nupkg name: nuget diff --git a/signing.json b/signing.json new file mode 100644 index 00000000..fe1c8c9b --- /dev/null +++ b/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/Directory.Build.props b/src/Directory.Build.props index b3c6287c..dc78f888 100644 --- a/src/Directory.Build.props +++ b/src/Directory.Build.props @@ -8,7 +8,6 @@ Debug false - MSB3246 $(MSBuildProjectName) $([System.IO.Path]::GetFullPath($(MSBuildThisFileDirectory)..\build\)) @@ -20,6 +19,10 @@ Copyright (c) .NET Foundation and contributors. All rights reserved. MS-RL WiX Toolset + + false + version.txt + v?(?<MAJOR>\d+|{[\dA-za-z\-\.]+})\.(?<MINOR>(\d+|{[\dA-za-z\-\.]+}))(?:\-(?<LABEL>[\dA-Za-z\-\.{}]+))?$|^v?(?<MAJOR>\d+|{[\dA-za-z\-\.]+})\.(?<MINOR>(\d+|{[\dA-za-z\-\.]+}))\.(?<PATCH>\d+|{[\dA-za-z\-\.]+})(?:\-(?<LABEL>[\dA-Za-z\-\.{}]+))?$|^(?<LABEL>[\dA-Za-z\-\.{}]+)\-v?(?<MAJOR>\d+|{[\dA-za-z\-\.]+})\.(?<MINOR>\d+|{[\dA-za-z\-\.]+})\.(?<PATCH>\d+|{[\dA-za-z\-\.]+})$ diff --git a/src/Directory.Build.targets b/src/Directory.Build.targets index 2fcc765a..c426f25e 100644 --- a/src/Directory.Build.targets +++ b/src/Directory.Build.targets @@ -4,47 +4,105 @@ Do NOT modify this file. Update the canonical version in Home\repo-template\src\Directory.Build.targets then update all of the repos. --> - - true - $(SolutionPath) - $(NCrunchOriginalSolutionPath) + $(BaseOutputPath)obj\.tools + $(SigningToolFolder)\SignClient.exe + $(SigningToolFolder)\empty-filelist.txt + $([MSBuild]::GetDirectoryNameOfFileAbove($(MSBuildProjectDirectory), signing.json))\signing.json + + + + + $(GitBaseVersion.Replace('{apiversion}', '$(ApiVersion)')) + $(GitBaseVersion.Replace('{height}', '$(GitCommits)')) + $(GitBaseVersion.Replace('{commits}', '$(GitCommits)')) + + + + + + $(GitBaseVersionMajor).$(GitBaseVersionMinor).0.0 + $(GitBaseVersionMajor).$(GitBaseVersionMinor).$(GitBaseVersionPatch).$(GitCommits) + $(GitBaseVersionMajor).$(GitBaseVersionMinor).$(GitBaseVersionPatch)$(GitSemVerDashLabel) + $(GitBaseVersionMajor).$(GitBaseVersionMinor).$(GitBaseVersionPatch)$(GitSemVerDashLabel)+$(GitSha) + + false + + + + + + + + $(GenerateNuspecDependsOn); + __SetNuspecProperties + - - + + + $(GitRepositoryUrl.Replace('.git','')) + + $(MSBuildProjectName).nuspec + $([System.IO.Path]::GetFullPath($(OutputPath)..))\ + $(NuspecProperties);Id=$(PackageId);Authors="$(Authors)";Configuration=$(Configuration);Copyright="$(Copyright)";Description="$(Description)";Title="$(Title)" + $(NuspecProperties);Version=$(PackageVersion);RepositoryCommit=$(GitSha);RepositoryType=git;RepositoryUrl=$(GitRepositoryUrl);ProjectFolder=$(MSBuildProjectDirectory)\;ProjectUrl=$(ProjectUrl) + true + snupkg + + + + + + + + + + + + + + + + - - $([System.IO.File]::ReadAllText($(TheSolutionPath))) - $([System.IO.Path]::GetDirectoryName( $(TheSolutionPath) )) - (?<="[PackageName]", ")(.*)(?=", ") - + - - - - %(Identity) - $(SolutionFileContent.Contains('\%(Identity).csproj')) - + + - - - $(RegexPattern.Replace('[PackageName]','%(PackageName)') ) - $([System.Text.RegularExpressions.Regex]::Match('$(SolutionFileContent)', '%(Pattern)')) - + - + + - - - + + + + - - + + diff --git a/src/Directory.csproj.targets b/src/Directory.csproj.targets index c3270426..49303a1d 100644 --- a/src/Directory.csproj.targets +++ b/src/Directory.csproj.targets @@ -9,18 +9,5 @@ $(OutputPath)\$(AssemblyName).xml - - - $(PrivateRepositoryUrl.Replace('.git','')) - - $(MSBuildProjectName).nuspec - $(OutputPath)..\ - $(NuspecProperties);Id=$(PackageId);Authors=$(Authors);Copyright=$(Copyright);Description=$(Description);Title=$(Title) - $(NuspecProperties);Version=$(PackageVersion);RepositoryCommit=$(SourceRevisionId);RepositoryType=$(RepositoryType);RepositoryUrl=$(PrivateRepositoryUrl);ProjectFolder=$(MSBuildProjectDirectory)\;ProjectUrl=$(ProjectUrl) - true - snupkg - - - + diff --git a/src/Directory.vcxproj.props b/src/Directory.vcxproj.props index bcf26c57..63d73b36 100644 --- a/src/Directory.vcxproj.props +++ b/src/Directory.vcxproj.props @@ -19,17 +19,22 @@ $([Microsoft.Build.Utilities.ToolLocationHelper]::GetLatestSDKTargetPlatformVersion('Windows', '10.0')) + + $(MSBuildThisFileDirectory)CustomizedNativeRecommendedRules.ruleset + + $(DisableSpecificCompilerWarnings) Level4 $(ProjectDir)inc;$(MSBuildProjectDirectory);$(IntDir);$(SqlCESdkIncludePath);$(ProjectAdditionalIncludeDirectories);%(AdditionalIncludeDirectories) - WIN32;_WINDOWS;_WIN32_MSI=500;_WIN32_WINNT=0x0501;$(ArmPreprocessorDefinitions);$(UnicodePreprocessorDefinitions);_CRT_STDIO_LEGACY_WIDE_SPECIFIERS;_WINSOCK_DEPRECATED_NO_WARNINGS;%(PreprocessorDefinitions) + WIN32;_WINDOWS;_WIN32_MSI=500;_WIN32_WINNT=0x0600;$(ArmPreprocessorDefinitions);$(UnicodePreprocessorDefinitions);_CRT_STDIO_LEGACY_WIDE_SPECIFIERS;_WINSOCK_DEPRECATED_NO_WARNINGS;%(PreprocessorDefinitions) Use precomp.h StdCall true false + Guard -YlprecompDefine /Zc:threadSafeInit- %(AdditionalOptions) true @@ -79,6 +84,7 @@ + MultiThreadedDebugDll @@ -98,6 +104,7 @@ + MultiThreadedDll diff --git a/src/Directory.vcxproj.targets b/src/Directory.vcxproj.targets new file mode 100644 index 00000000..9f0689d5 --- /dev/null +++ b/src/Directory.vcxproj.targets @@ -0,0 +1,45 @@ + + + + + + $(PrepareForBuildDependsOn); + __SetVersionResource + + + + + + <_ResourceFileType Condition=" '$(ConfigurationType)'=='Application' ">VFT_APP + <_ResourceFileType Condition=" '$(ConfigurationType)'=='DynamicLibrary' ">VFT_DLL + $(TargetName) + $(Description) + + + + + + %(PreprocessorDefinitions); + GITVER_FILEVER=$(FileVersion.Replace('.', ',')); + GITVER_PRODUCTVER=$(AssemblyVersion.Replace('.', ',')); + GITVER_FILE_VERSION=$(FileVersion); + GITVER_PRODUCT_VERSION=$(AssemblyVersion); + GITVER_FILE_TYPE=$(_ResourceFileType); + GITVER_CODEPAGE=0; + GITVER_LCID=$([System.Convert]::ToInt32('%(Culture)', 16)); + GITVER_VERSION_BLOCK=$([System.Convert]::ToString($([MSBuild]::Multiply($([System.Convert]::ToUint64('%(Culture)', 16)), 65536)), 16).PadLeft(8, '0')); + GITVER_COMPANY=$(Company); + GITVER_COPYRIGHT=$(Copyright); + GITVER_TITLE=$(Title); + GITVER_PRODUCT=$(Product); + GITVER_INFORMATIONAL_VERSION=$(InformationalVersion); + GITVER_INTERNAL_NAME=$(TargetName); + GITVER_FILE_NAME=$(TargetFileName); + + + + + diff --git a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj index d53a155b..fea15922 100644 --- a/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj +++ b/src/WixToolset.Core.Native/WixToolset.Core.Native.csproj @@ -7,7 +7,6 @@ netstandard2.0 embedded WiX Toolset Native Processing - true true NU5128 @@ -45,6 +44,6 @@ - + diff --git a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj index 77a53c29..6068dbea 100644 --- a/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj +++ b/src/test/WixToolsetTest.Core.Native/WixToolsetTest.Core.Native.csproj @@ -17,6 +17,8 @@ + + diff --git a/src/test/version.txt b/src/test/version.txt new file mode 100644 index 00000000..cf138743 --- /dev/null +++ b/src/test/version.txt @@ -0,0 +1 @@ +v42.42.{height}-preview.0 \ No newline at end of file diff --git a/src/ver.rc b/src/ver.rc new file mode 100644 index 00000000..4dc64d37 --- /dev/null +++ b/src/ver.rc @@ -0,0 +1,55 @@ +#pragma once +// ------------------------------------------------------------------------------ +// +// This code was generated by a tool. +// Runtime Version:4.0.30319.42000 +// Changes to this file may cause incorrect behavior and will be lost if +// the code is regenerated. +// +// ------------------------------------------------------------------------------ + +#if defined(_UNICODE) +#define GITVER_VERSION_STRING(x) L ## #x +#else +#define GITVER_VERSION_STRING(x) #x +#endif + +#define GVS(x) GITVER_VERSION_STRING(x) + +#ifdef RC_INVOKED + +#include + +VS_VERSION_INFO VERSIONINFO + FILEVERSION GITVER_FILEVER + PRODUCTVERSION GITVER_PRODUCTVER + FILEFLAGSMASK 0x3FL +#ifdef _DEBUG + FILEFLAGS 0x1L +#else + FILEFLAGS 0x0L +#endif + FILEOS 0x4L + FILETYPE GITVER_FILE_TYPE + FILESUBTYPE 0x0L +BEGIN + BLOCK "StringFileInfo" + BEGIN + BLOCK GVS(GITVER_VERSION_BLOCK) + BEGIN + VALUE "CompanyName", GVS(GITVER_COMPANY) + VALUE "FileDescription", GVS(GITVER_TITLE) + VALUE "FileVersion", GVS(GITVER_FILE_VERSION) + VALUE "InternalName", GVS(GITVER_INTERNAL_NAME) + VALUE "OriginalFilename", GVS(GITVER_FILE_NAME) + VALUE "ProductName", GVS(GITVER_PRODUCT) + VALUE "ProductVersion", GVS(GITVER_INFORMATIONAL_VERSION) + VALUE "LegalCopyright", GVS(GITVER_COPYRIGHT) + END + END + BLOCK "VarFileInfo" + BEGIN + VALUE "Translation", GITVER_LCID, GITVER_CODEPAGE + END +END +#endif diff --git a/src/wixnative/wixnative.vcxproj b/src/wixnative/wixnative.vcxproj index 09dc97fa..20959827 100644 --- a/src/wixnative/wixnative.vcxproj +++ b/src/wixnative/wixnative.vcxproj @@ -69,7 +69,7 @@ - + diff --git a/version.json b/version.json deleted file mode 100644 index 5f857771..00000000 --- a/version.json +++ /dev/null @@ -1,11 +0,0 @@ -{ - "version": "4.0", - "publicReleaseRefSpec": [ - "^refs/heads/master$" - ], - "cloudBuild": { - "buildNumber": { - "enabled": true - } - } -} diff --git a/version.txt b/version.txt new file mode 100644 index 00000000..dc60d914 --- /dev/null +++ b/version.txt @@ -0,0 +1 @@ +v4.{apiversion}-preview.0-build.{height} -- cgit v1.2.3-55-g6feb 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 --- .editorconfig | 37 - README.md | 2 - WixToolset.Core.Native.sln | 63 -- WixToolset.Core.Native.v3.ncrunchsolution | 6 - appveyor.cmd | 19 - appveyor.yml | 44 - nuget.config | 9 - signing.json | 13 - 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 version.txt | 1 - 154 files changed, 6610 insertions(+), 6610 deletions(-) delete mode 100644 .editorconfig delete mode 100644 README.md delete mode 100644 WixToolset.Core.Native.sln delete mode 100644 WixToolset.Core.Native.v3.ncrunchsolution delete mode 100644 appveyor.cmd delete mode 100644 appveyor.yml delete mode 100644 nuget.config delete mode 100644 signing.json 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 delete mode 100644 version.txt diff --git a/.editorconfig b/.editorconfig deleted file mode 100644 index 1d72e683..00000000 --- a/.editorconfig +++ /dev/null @@ -1,37 +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. -# -# 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/README.md b/README.md deleted file mode 100644 index 787123cc..00000000 --- a/README.md +++ /dev/null @@ -1,2 +0,0 @@ -# Core.Native -Core.Native - native component of WixToolset.Core diff --git a/WixToolset.Core.Native.sln b/WixToolset.Core.Native.sln deleted file mode 100644 index 0d7a5921..00000000 --- a/WixToolset.Core.Native.sln +++ /dev/null @@ -1,63 +0,0 @@ - -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/WixToolset.Core.Native.v3.ncrunchsolution b/WixToolset.Core.Native.v3.ncrunchsolution deleted file mode 100644 index 10420ac9..00000000 --- a/WixToolset.Core.Native.v3.ncrunchsolution +++ /dev/null @@ -1,6 +0,0 @@ - - - True - True - - \ No newline at end of file diff --git a/appveyor.cmd b/appveyor.cmd deleted file mode 100644 index d9691a42..00000000 --- a/appveyor.cmd +++ /dev/null @@ -1,19 +0,0 @@ -@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/appveyor.yml b/appveyor.yml deleted file mode 100644 index 364569cf..00000000 --- a/appveyor.yml +++ /dev/null @@ -1,44 +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. -# -# 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/nuget.config b/nuget.config deleted file mode 100644 index 18404582..00000000 --- a/nuget.config +++ /dev/null @@ -1,9 +0,0 @@ - - - - - - - - - \ No newline at end of file diff --git a/signing.json b/signing.json deleted file mode 100644 index fe1c8c9b..00000000 --- a/signing.json +++ /dev/null @@ -1,13 +0,0 @@ -{ - "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/.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 diff --git a/version.txt b/version.txt deleted file mode 100644 index dc60d914..00000000 --- a/version.txt +++ /dev/null @@ -1 +0,0 @@ -v4.{apiversion}-preview.0-build.{height} -- cgit v1.2.3-55-g6feb