blob: 92058842b4fd6d71e982ff28cff41e1ab1ac09f5 [file] [log] [blame]
Aric Stewart401bd3f2004-06-28 20:34:35 +00001/*
2 * Implementation of the Microsoft Installer (msi.dll)
3 *
Aric Stewartc6689522005-06-17 20:56:55 +00004 * Copyright 2004,2005 Aric Stewart for CodeWeavers
Aric Stewart401bd3f2004-06-28 20:34:35 +00005 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
Jonathan Ernst360a3f92006-05-18 14:49:52 +020018 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
Aric Stewart401bd3f2004-06-28 20:34:35 +000019 */
20
21/*
Mike McCormack6e2bca32004-07-04 00:25:00 +000022 * Pages I need
Aric Stewart401bd3f2004-06-28 20:34:35 +000023 *
24http://msdn.microsoft.com/library/default.asp?url=/library/en-us/msi/setup/installexecutesequence_table.asp
25
26http://msdn.microsoft.com/library/default.asp?url=/library/en-us/msi/setup/standard_actions_reference.asp
Aric Stewart401bd3f2004-06-28 20:34:35 +000027 */
28
29#include <stdarg.h>
Aric Stewart401bd3f2004-06-28 20:34:35 +000030
Francois Gouget486d0202004-10-07 03:06:48 +000031#define COBJMACROS
32
Aric Stewart401bd3f2004-06-28 20:34:35 +000033#include "windef.h"
34#include "winbase.h"
35#include "winerror.h"
36#include "winreg.h"
James Hawkins9bc12ad2006-10-19 15:49:54 -070037#include "winsvc.h"
Aric Stewart401bd3f2004-06-28 20:34:35 +000038#include "wine/debug.h"
Aric Stewartb6bc6aa2005-02-24 12:47:43 +000039#include "msidefs.h"
Aric Stewart401bd3f2004-06-28 20:34:35 +000040#include "msipriv.h"
Aric Stewart401bd3f2004-06-28 20:34:35 +000041#include "winuser.h"
42#include "shlobj.h"
43#include "wine/unicode.h"
Steven Edwards98efef12005-04-11 16:10:33 +000044#include "winver.h"
Aric Stewart401bd3f2004-06-28 20:34:35 +000045
Aric Stewartbd1bbc12005-01-03 20:00:13 +000046#define REG_PROGRESS_VALUE 13200
47#define COMPONENT_PROGRESS_VALUE 24000
Aric Stewart401bd3f2004-06-28 20:34:35 +000048
Aric Stewart401bd3f2004-06-28 20:34:35 +000049WINE_DEFAULT_DEBUG_CHANNEL(msi);
50
51/*
52 * Prototypes
53 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +000054static UINT ACTION_ProcessExecSequence(MSIPACKAGE *package, BOOL UIran);
55static UINT ACTION_ProcessUISequence(MSIPACKAGE *package);
Aric Stewartf8f64402005-03-24 19:03:45 +000056static UINT ACTION_PerformActionSequence(MSIPACKAGE *package, UINT seq, BOOL UI);
Aric Stewart90c57392005-01-31 16:23:12 +000057
Aric Stewart401bd3f2004-06-28 20:34:35 +000058/*
59 * consts and values used
60 */
Juan Lang014ad3b2005-03-01 10:41:52 +000061static const WCHAR c_colon[] = {'C',':','\\',0};
Aric Stewartc6689522005-06-17 20:56:55 +000062
Mike McCormack9ba8ba32005-10-30 19:04:26 +000063static const WCHAR szCreateFolders[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000064 {'C','r','e','a','t','e','F','o','l','d','e','r','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +000065static const WCHAR szCostFinalize[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000066 {'C','o','s','t','F','i','n','a','l','i','z','e',0};
Aric Stewartc6689522005-06-17 20:56:55 +000067const WCHAR szInstallFiles[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000068 {'I','n','s','t','a','l','l','F','i','l','e','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +000069const WCHAR szDuplicateFiles[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000070 {'D','u','p','l','i','c','a','t','e','F','i','l','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +000071static const WCHAR szWriteRegistryValues[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000072 {'W','r','i','t','e','R','e','g','i','s','t','r','y',
73 'V','a','l','u','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +000074static const WCHAR szCostInitialize[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000075 {'C','o','s','t','I','n','i','t','i','a','l','i','z','e',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +000076static const WCHAR szFileCost[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000077 {'F','i','l','e','C','o','s','t',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +000078static const WCHAR szInstallInitialize[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000079 {'I','n','s','t','a','l','l','I','n','i','t','i','a','l','i','z','e',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +000080static const WCHAR szInstallValidate[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000081 {'I','n','s','t','a','l','l','V','a','l','i','d','a','t','e',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +000082static const WCHAR szLaunchConditions[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000083 {'L','a','u','n','c','h','C','o','n','d','i','t','i','o','n','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +000084static const WCHAR szProcessComponents[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000085 {'P','r','o','c','e','s','s','C','o','m','p','o','n','e','n','t','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +000086static const WCHAR szRegisterTypeLibraries[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000087 {'R','e','g','i','s','t','e','r','T','y','p','e',
88 'L','i','b','r','a','r','i','e','s',0};
Aric Stewartdb982e22005-06-16 15:51:44 +000089const WCHAR szRegisterClassInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000090 {'R','e','g','i','s','t','e','r','C','l','a','s','s','I','n','f','o',0};
Aric Stewartdb982e22005-06-16 15:51:44 +000091const WCHAR szRegisterProgIdInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000092 {'R','e','g','i','s','t','e','r','P','r','o','g','I','d','I','n','f','o',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +000093static const WCHAR szCreateShortcuts[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000094 {'C','r','e','a','t','e','S','h','o','r','t','c','u','t','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +000095static const WCHAR szPublishProduct[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000096 {'P','u','b','l','i','s','h','P','r','o','d','u','c','t',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +000097static const WCHAR szWriteIniValues[] =
Aric Stewart8e233e92005-03-01 11:45:19 +000098 {'W','r','i','t','e','I','n','i','V','a','l','u','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +000099static const WCHAR szSelfRegModules[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000100 {'S','e','l','f','R','e','g','M','o','d','u','l','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000101static const WCHAR szPublishFeatures[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000102 {'P','u','b','l','i','s','h','F','e','a','t','u','r','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000103static const WCHAR szRegisterProduct[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000104 {'R','e','g','i','s','t','e','r','P','r','o','d','u','c','t',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000105static const WCHAR szInstallExecute[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000106 {'I','n','s','t','a','l','l','E','x','e','c','u','t','e',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000107static const WCHAR szInstallExecuteAgain[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000108 {'I','n','s','t','a','l','l','E','x','e','c','u','t','e',
109 'A','g','a','i','n',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000110static const WCHAR szInstallFinalize[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000111 {'I','n','s','t','a','l','l','F','i','n','a','l','i','z','e',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000112static const WCHAR szForceReboot[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000113 {'F','o','r','c','e','R','e','b','o','o','t',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000114static const WCHAR szResolveSource[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000115 {'R','e','s','o','l','v','e','S','o','u','r','c','e',0};
Alexandre Julliardb79a53e2006-06-12 13:39:59 +0200116static const WCHAR szAppSearch[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000117 {'A','p','p','S','e','a','r','c','h',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000118static const WCHAR szAllocateRegistrySpace[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000119 {'A','l','l','o','c','a','t','e','R','e','g','i','s','t','r','y',
120 'S','p','a','c','e',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000121static const WCHAR szBindImage[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000122 {'B','i','n','d','I','m','a','g','e',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000123static const WCHAR szCCPSearch[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000124 {'C','C','P','S','e','a','r','c','h',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000125static const WCHAR szDeleteServices[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000126 {'D','e','l','e','t','e','S','e','r','v','i','c','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000127static const WCHAR szDisableRollback[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000128 {'D','i','s','a','b','l','e','R','o','l','l','b','a','c','k',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000129static const WCHAR szExecuteAction[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000130 {'E','x','e','c','u','t','e','A','c','t','i','o','n',0};
Aric Stewartdb982e22005-06-16 15:51:44 +0000131const WCHAR szFindRelatedProducts[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000132 {'F','i','n','d','R','e','l','a','t','e','d',
133 'P','r','o','d','u','c','t','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000134static const WCHAR szInstallAdminPackage[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000135 {'I','n','s','t','a','l','l','A','d','m','i','n',
136 'P','a','c','k','a','g','e',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000137static const WCHAR szInstallSFPCatalogFile[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000138 {'I','n','s','t','a','l','l','S','F','P','C','a','t','a','l','o','g',
139 'F','i','l','e',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000140static const WCHAR szIsolateComponents[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000141 {'I','s','o','l','a','t','e','C','o','m','p','o','n','e','n','t','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000142const WCHAR szMigrateFeatureStates[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000143 {'M','i','g','r','a','t','e','F','e','a','t','u','r','e',
144 'S','t','a','t','e','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000145const WCHAR szMoveFiles[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000146 {'M','o','v','e','F','i','l','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000147static const WCHAR szMsiPublishAssemblies[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000148 {'M','s','i','P','u','b','l','i','s','h',
149 'A','s','s','e','m','b','l','i','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000150static const WCHAR szMsiUnpublishAssemblies[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000151 {'M','s','i','U','n','p','u','b','l','i','s','h',
152 'A','s','s','e','m','b','l','i','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000153static const WCHAR szInstallODBC[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000154 {'I','n','s','t','a','l','l','O','D','B','C',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000155static const WCHAR szInstallServices[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000156 {'I','n','s','t','a','l','l','S','e','r','v','i','c','e','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000157const WCHAR szPatchFiles[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000158 {'P','a','t','c','h','F','i','l','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000159static const WCHAR szPublishComponents[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000160 {'P','u','b','l','i','s','h','C','o','m','p','o','n','e','n','t','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000161static const WCHAR szRegisterComPlus[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000162 {'R','e','g','i','s','t','e','r','C','o','m','P','l','u','s',0};
Aric Stewartdb982e22005-06-16 15:51:44 +0000163const WCHAR szRegisterExtensionInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000164 {'R','e','g','i','s','t','e','r','E','x','t','e','n','s','i','o','n',
165 'I','n','f','o',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000166static const WCHAR szRegisterFonts[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000167 {'R','e','g','i','s','t','e','r','F','o','n','t','s',0};
Aric Stewartdb982e22005-06-16 15:51:44 +0000168const WCHAR szRegisterMIMEInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000169 {'R','e','g','i','s','t','e','r','M','I','M','E','I','n','f','o',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000170static const WCHAR szRegisterUser[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000171 {'R','e','g','i','s','t','e','r','U','s','e','r',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000172const WCHAR szRemoveDuplicateFiles[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000173 {'R','e','m','o','v','e','D','u','p','l','i','c','a','t','e',
174 'F','i','l','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000175static const WCHAR szRemoveEnvironmentStrings[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000176 {'R','e','m','o','v','e','E','n','v','i','r','o','n','m','e','n','t',
177 'S','t','r','i','n','g','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000178const WCHAR szRemoveExistingProducts[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000179 {'R','e','m','o','v','e','E','x','i','s','t','i','n','g',
180 'P','r','o','d','u','c','t','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000181const WCHAR szRemoveFiles[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000182 {'R','e','m','o','v','e','F','i','l','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000183static const WCHAR szRemoveFolders[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000184 {'R','e','m','o','v','e','F','o','l','d','e','r','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000185static const WCHAR szRemoveIniValues[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000186 {'R','e','m','o','v','e','I','n','i','V','a','l','u','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000187static const WCHAR szRemoveODBC[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000188 {'R','e','m','o','v','e','O','D','B','C',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000189static const WCHAR szRemoveRegistryValues[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000190 {'R','e','m','o','v','e','R','e','g','i','s','t','r','y',
191 'V','a','l','u','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000192static const WCHAR szRemoveShortcuts[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000193 {'R','e','m','o','v','e','S','h','o','r','t','c','u','t','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000194static const WCHAR szRMCCPSearch[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000195 {'R','M','C','C','P','S','e','a','r','c','h',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000196static const WCHAR szScheduleReboot[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000197 {'S','c','h','e','d','u','l','e','R','e','b','o','o','t',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000198static const WCHAR szSelfUnregModules[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000199 {'S','e','l','f','U','n','r','e','g','M','o','d','u','l','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000200static const WCHAR szSetODBCFolders[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000201 {'S','e','t','O','D','B','C','F','o','l','d','e','r','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000202static const WCHAR szStartServices[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000203 {'S','t','a','r','t','S','e','r','v','i','c','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000204static const WCHAR szStopServices[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000205 {'S','t','o','p','S','e','r','v','i','c','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000206static const WCHAR szUnpublishComponents[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000207 {'U','n','p','u','b','l','i','s','h',
208 'C','o','m','p','o','n','e','n','t','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000209static const WCHAR szUnpublishFeatures[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000210 {'U','n','p','u','b','l','i','s','h','F','e','a','t','u','r','e','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000211const WCHAR szUnregisterClassInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000212 {'U','n','r','e','g','i','s','t','e','r','C','l','a','s','s',
213 'I','n','f','o',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000214static const WCHAR szUnregisterComPlus[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000215 {'U','n','r','e','g','i','s','t','e','r','C','o','m','P','l','u','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000216const WCHAR szUnregisterExtensionInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000217 {'U','n','r','e','g','i','s','t','e','r',
218 'E','x','t','e','n','s','i','o','n','I','n','f','o',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000219static const WCHAR szUnregisterFonts[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000220 {'U','n','r','e','g','i','s','t','e','r','F','o','n','t','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000221const WCHAR szUnregisterMIMEInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000222 {'U','n','r','e','g','i','s','t','e','r','M','I','M','E','I','n','f','o',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000223const WCHAR szUnregisterProgIdInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000224 {'U','n','r','e','g','i','s','t','e','r','P','r','o','g','I','d',
225 'I','n','f','o',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000226static const WCHAR szUnregisterTypeLibraries[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000227 {'U','n','r','e','g','i','s','t','e','r','T','y','p','e',
228 'L','i','b','r','a','r','i','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000229static const WCHAR szValidateProductID[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000230 {'V','a','l','i','d','a','t','e','P','r','o','d','u','c','t','I','D',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000231static const WCHAR szWriteEnvironmentStrings[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000232 {'W','r','i','t','e','E','n','v','i','r','o','n','m','e','n','t',
233 'S','t','r','i','n','g','s',0};
Aric Stewart90c57392005-01-31 16:23:12 +0000234
Mike McCormack3b955152005-09-28 18:10:44 +0000235/* action handlers */
236typedef UINT (*STANDARDACTIONHANDLER)(MSIPACKAGE*);
237
Aric Stewart90c57392005-01-31 16:23:12 +0000238struct _actions {
239 LPCWSTR action;
240 STANDARDACTIONHANDLER handler;
241};
242
Mike McCormack3b955152005-09-28 18:10:44 +0000243static struct _actions StandardActions[];
Aric Stewart90c57392005-01-31 16:23:12 +0000244
Aric Stewart2703d712005-06-20 15:33:10 +0000245
246/********************************************************
247 * helper functions
248 ********************************************************/
249
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000250static void ui_actionstart(MSIPACKAGE *package, LPCWSTR action)
Aric Stewartd2c395a2004-07-06 18:48:15 +0000251{
Aric Stewartd2c395a2004-07-06 18:48:15 +0000252 static const WCHAR Query_t[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000253 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000254 '`','A','c','t','i','o', 'n','T','e','x','t','`',' ',
255 'W','H','E','R','E', ' ','`','A','c','t','i','o','n','`',' ','=',
256 ' ','\'','%','s','\'',0};
Robert Shearmand679bc52006-01-23 17:30:31 +0100257 MSIRECORD * row;
Aric Stewartd2c395a2004-07-06 18:48:15 +0000258
Mike McCormack0b352c72005-06-02 10:29:57 +0000259 row = MSI_QueryGetRecord( package->db, Query_t, action );
260 if (!row)
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000261 return;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000262 MSI_ProcessMessage(package, INSTALLMESSAGE_ACTIONSTART, row);
263 msiobj_release(&row->hdr);
Aric Stewartd2c395a2004-07-06 18:48:15 +0000264}
265
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000266static void ui_actioninfo(MSIPACKAGE *package, LPCWSTR action, BOOL start,
Aric Stewartd2c395a2004-07-06 18:48:15 +0000267 UINT rc)
268{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000269 MSIRECORD * row;
Aric Stewartd2c395a2004-07-06 18:48:15 +0000270 static const WCHAR template_s[]=
Aric Stewart8e233e92005-03-01 11:45:19 +0000271 {'A','c','t','i','o','n',' ','s','t','a','r','t',' ','%','s',':',' ',
272 '%','s', '.',0};
Aric Stewartd2c395a2004-07-06 18:48:15 +0000273 static const WCHAR template_e[]=
Aric Stewart8e233e92005-03-01 11:45:19 +0000274 {'A','c','t','i','o','n',' ','e','n','d','e','d',' ','%','s',':',' ',
275 '%','s', '.',' ','R','e','t','u','r','n',' ','v','a','l','u','e',' ',
276 '%','i','.',0};
Aric Stewartd2c395a2004-07-06 18:48:15 +0000277 static const WCHAR format[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000278 {'H','H','\'',':','\'','m','m','\'',':','\'','s','s',0};
Aric Stewartd2c395a2004-07-06 18:48:15 +0000279 WCHAR message[1024];
280 WCHAR timet[0x100];
281
282 GetTimeFormatW(LOCALE_USER_DEFAULT, 0, NULL, format, timet, 0x100);
283 if (start)
284 sprintfW(message,template_s,timet,action);
285 else
286 sprintfW(message,template_e,timet,action,rc);
287
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000288 row = MSI_CreateRecord(1);
289 MSI_RecordSetStringW(row,1,message);
Aric Stewartd2c395a2004-07-06 18:48:15 +0000290
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000291 MSI_ProcessMessage(package, INSTALLMESSAGE_INFO, row);
292 msiobj_release(&row->hdr);
Aric Stewartd2c395a2004-07-06 18:48:15 +0000293}
294
Mike McCormacke3452222005-09-28 15:12:32 +0000295static UINT msi_parse_command_line( MSIPACKAGE *package, LPCWSTR szCommandLine )
296{
297 LPCWSTR ptr,ptr2;
298 BOOL quote;
299 DWORD len;
300 LPWSTR prop = NULL, val = NULL;
301
302 if (!szCommandLine)
303 return ERROR_SUCCESS;
304
305 ptr = szCommandLine;
306
307 while (*ptr)
308 {
309 if (*ptr==' ')
310 {
311 ptr++;
312 continue;
313 }
314
315 TRACE("Looking at %s\n",debugstr_w(ptr));
316
317 ptr2 = strchrW(ptr,'=');
318 if (!ptr2)
319 {
320 ERR("command line contains unknown string : %s\n", debugstr_w(ptr));
321 break;
322 }
323
324 quote = FALSE;
325
326 len = ptr2-ptr;
327 prop = msi_alloc((len+1)*sizeof(WCHAR));
328 memcpy(prop,ptr,len*sizeof(WCHAR));
329 prop[len]=0;
330 ptr2++;
331
332 len = 0;
333 ptr = ptr2;
334 while (*ptr && (quote || (!quote && *ptr!=' ')))
335 {
336 if (*ptr == '"')
337 quote = !quote;
338 ptr++;
339 len++;
340 }
341
342 if (*ptr2=='"')
343 {
344 ptr2++;
345 len -= 2;
346 }
347 val = msi_alloc((len+1)*sizeof(WCHAR));
348 memcpy(val,ptr2,len*sizeof(WCHAR));
349 val[len] = 0;
350
351 if (lstrlenW(prop) > 0)
352 {
353 TRACE("Found commandline property (%s) = (%s)\n",
354 debugstr_w(prop), debugstr_w(val));
355 MSI_SetPropertyW(package,prop,val);
356 }
357 msi_free(val);
358 msi_free(prop);
359 }
360
361 return ERROR_SUCCESS;
362}
363
Mike McCormack965a72a2005-10-26 12:06:21 +0000364
365static LPWSTR* msi_split_string( LPCWSTR str, WCHAR sep )
366{
Andrew Talbot2e372c02006-09-17 08:52:49 +0100367 LPCWSTR pc;
Mike McCormack965a72a2005-10-26 12:06:21 +0000368 LPWSTR p, *ret = NULL;
369 UINT count = 0;
370
371 if (!str)
372 return ret;
373
374 /* count the number of substrings */
Andrew Talbot2e372c02006-09-17 08:52:49 +0100375 for ( pc = str, count = 0; pc; count++ )
Mike McCormack965a72a2005-10-26 12:06:21 +0000376 {
Andrew Talbot2e372c02006-09-17 08:52:49 +0100377 pc = strchrW( pc, sep );
378 if (pc)
379 pc++;
Mike McCormack965a72a2005-10-26 12:06:21 +0000380 }
381
382 /* allocate space for an array of substring pointers and the substrings */
383 ret = msi_alloc( (count+1) * sizeof (LPWSTR) +
384 (lstrlenW(str)+1) * sizeof(WCHAR) );
385 if (!ret)
386 return ret;
387
388 /* copy the string and set the pointers */
389 p = (LPWSTR) &ret[count+1];
390 lstrcpyW( p, str );
391 for( count = 0; (ret[count] = p); count++ )
392 {
393 p = strchrW( p, sep );
394 if (p)
395 *p++ = 0;
396 }
397
398 return ret;
399}
400
401static UINT msi_apply_substorage_transform( MSIPACKAGE *package,
402 MSIDATABASE *patch_db, LPCWSTR name )
403{
404 UINT ret = ERROR_FUNCTION_FAILED;
405 IStorage *stg = NULL;
406 HRESULT r;
407
408 TRACE("%p %s\n", package, debugstr_w(name) );
409
410 if (*name++ != ':')
411 {
412 ERR("expected a colon in %s\n", debugstr_w(name));
413 return ERROR_FUNCTION_FAILED;
414 }
415
416 r = IStorage_OpenStorage( patch_db->storage, name, NULL, STGM_SHARE_EXCLUSIVE, NULL, 0, &stg );
417 if (SUCCEEDED(r))
418 {
Mike McCormackf8fef6e2005-10-26 13:56:33 +0000419 ret = msi_table_apply_transform( package->db, stg );
Mike McCormack965a72a2005-10-26 12:06:21 +0000420 IStorage_Release( stg );
421 ret = ERROR_SUCCESS;
422 }
423 else
424 ERR("failed to open substorage %s\n", debugstr_w(name));
425
426 return ret;
427}
428
429static UINT msi_check_patch_applicable( MSIPACKAGE *package, MSISUMMARYINFO *si )
430{
431 static const WCHAR szProdID[] = { 'P','r','o','d','u','c','t','I','D',0 };
432 LPWSTR guid_list, *guids, product_id;
433 UINT i, ret = ERROR_FUNCTION_FAILED;
434
435 product_id = msi_dup_property( package, szProdID );
436 if (!product_id)
437 {
438 /* FIXME: the property ProductID should be written into the DB somewhere */
439 ERR("no product ID to check\n");
440 return ERROR_SUCCESS;
441 }
442
443 guid_list = msi_suminfo_dup_string( si, PID_TEMPLATE );
444 guids = msi_split_string( guid_list, ';' );
445 for ( i = 0; guids[i] && ret != ERROR_SUCCESS; i++ )
446 {
447 if (!lstrcmpW( guids[i], product_id ))
448 ret = ERROR_SUCCESS;
449 }
450 msi_free( guids );
451 msi_free( guid_list );
452 msi_free( product_id );
453
454 return ret;
455}
456
457static UINT msi_parse_patch_summary( MSIPACKAGE *package, MSIDATABASE *patch_db )
458{
459 MSISUMMARYINFO *si;
460 LPWSTR str, *substorage;
461 UINT i, r = ERROR_SUCCESS;
462
Mike McCormack7f98f1d2006-10-24 01:11:30 +0900463 si = MSI_GetSummaryInformationW( patch_db->storage, 0 );
Mike McCormack965a72a2005-10-26 12:06:21 +0000464 if (!si)
465 return ERROR_FUNCTION_FAILED;
466
467 msi_check_patch_applicable( package, si );
468
469 /* enumerate the substorage */
470 str = msi_suminfo_dup_string( si, PID_LASTAUTHOR );
471 substorage = msi_split_string( str, ';' );
472 for ( i = 0; substorage && substorage[i] && r == ERROR_SUCCESS; i++ )
473 r = msi_apply_substorage_transform( package, patch_db, substorage[i] );
474 msi_free( substorage );
475 msi_free( str );
476
477 /* FIXME: parse the sources in PID_REVNUMBER and do something with them... */
478
479 msiobj_release( &si->hdr );
480
481 return r;
482}
483
484static UINT msi_apply_patch_package( MSIPACKAGE *package, LPCWSTR file )
485{
486 MSIDATABASE *patch_db = NULL;
487 UINT r;
488
489 TRACE("%p %s\n", package, debugstr_w( file ) );
490
491 /* FIXME:
492 * We probably want to make sure we only open a patch collection here.
493 * Patch collections (.msp) and databases (.msi) have different GUIDs
494 * but currently MSI_OpenDatabaseW will accept both.
495 */
496 r = MSI_OpenDatabaseW( file, MSIDBOPEN_READONLY, &patch_db );
497 if ( r != ERROR_SUCCESS )
498 {
499 ERR("failed to open patch collection %s\n", debugstr_w( file ) );
500 return r;
501 }
502
503 msi_parse_patch_summary( package, patch_db );
504 msiobj_release( &patch_db->hdr );
505
506 return ERROR_SUCCESS;
507}
508
509/* get the PATCH property, and apply all the patches it specifies */
510static UINT msi_apply_patches( MSIPACKAGE *package )
511{
512 static const WCHAR szPatch[] = { 'P','A','T','C','H',0 };
513 LPWSTR patch_list, *patches;
514 UINT i, r = ERROR_SUCCESS;
515
516 patch_list = msi_dup_property( package, szPatch );
517
518 TRACE("patches to be applied: %s\n", debugstr_w( patch_list ) );
519
520 patches = msi_split_string( patch_list, ';' );
521 for( i=0; patches && patches[i] && r == ERROR_SUCCESS; i++ )
522 r = msi_apply_patch_package( package, patches[i] );
523
524 msi_free( patches );
525 msi_free( patch_list );
526
527 return r;
528}
529
Mike McCormacke534e772006-01-04 14:51:25 +0100530static UINT msi_apply_transforms( MSIPACKAGE *package )
531{
532 static const WCHAR szTransforms[] = {
533 'T','R','A','N','S','F','O','R','M','S',0 };
534 LPWSTR xform_list, *xforms;
535 UINT i, r = ERROR_SUCCESS;
536
537 xform_list = msi_dup_property( package, szTransforms );
538 xforms = msi_split_string( xform_list, ';' );
539
540 for( i=0; xforms && xforms[i] && r == ERROR_SUCCESS; i++ )
541 {
542 if (xforms[i][0] == ':')
543 r = msi_apply_substorage_transform( package, package->db, &xforms[i][1] );
544 else
545 r = MSI_DatabaseApplyTransformW( package->db, xforms[i], 0 );
546 }
547
548 msi_free( xforms );
549 msi_free( xform_list );
550
551 return r;
552}
553
Aric Stewart401bd3f2004-06-28 20:34:35 +0000554/****************************************************
555 * TOP level entry points
556 *****************************************************/
557
Mike McCormack61f24a42005-09-30 10:32:41 +0000558UINT MSI_InstallPackage( MSIPACKAGE *package, LPCWSTR szPackagePath,
559 LPCWSTR szCommandLine )
Aric Stewart401bd3f2004-06-28 20:34:35 +0000560{
Aric Stewart401bd3f2004-06-28 20:34:35 +0000561 UINT rc;
Aric Stewartf8f64402005-03-24 19:03:45 +0000562 BOOL ui = FALSE;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000563 static const WCHAR szUILevel[] = {'U','I','L','e','v','e','l',0};
Aric Stewart6269f002005-01-17 13:40:39 +0000564 static const WCHAR szAction[] = {'A','C','T','I','O','N',0};
565 static const WCHAR szInstall[] = {'I','N','S','T','A','L','L',0};
566
567 MSI_SetPropertyW(package, szAction, szInstall);
Aric Stewart9cd707d2005-05-27 19:24:22 +0000568
Mike McCormack3a940112006-04-19 02:29:03 +0900569 package->script = msi_alloc_zero(sizeof(MSISCRIPT));
Aric Stewart401bd3f2004-06-28 20:34:35 +0000570
Aric Stewartc9802932005-06-30 20:45:43 +0000571 package->script->InWhatSequence = SEQUENCE_INSTALL;
572
Aric Stewarte95136b2004-06-29 03:44:01 +0000573 if (szPackagePath)
574 {
Mike McCormackba8200b2004-12-22 15:25:30 +0000575 LPWSTR p, check, path;
Aric Stewarte95136b2004-06-29 03:44:01 +0000576
Mike McCormack95dea492005-03-16 11:31:35 +0000577 path = strdupW(szPackagePath);
Mike McCormackba8200b2004-12-22 15:25:30 +0000578 p = strrchrW(path,'\\');
Aric Stewarte95136b2004-06-29 03:44:01 +0000579 if (p)
580 {
581 p++;
582 *p=0;
583 }
Aric Stewartc1e5c4a2005-02-08 14:26:49 +0000584 else
585 {
Mike McCormackee034ba2005-09-20 11:59:14 +0000586 msi_free(path);
587 path = msi_alloc(MAX_PATH*sizeof(WCHAR));
Aric Stewartc1e5c4a2005-02-08 14:26:49 +0000588 GetCurrentDirectoryW(MAX_PATH,path);
589 strcatW(path,cszbs);
590 }
Aric Stewarte95136b2004-06-29 03:44:01 +0000591
Mike McCormack062ad502005-09-15 15:04:08 +0000592 check = msi_dup_property( package, cszSourceDir );
Aric Stewartd0c971a2004-12-22 18:15:50 +0000593 if (!check)
Mike McCormackba8200b2004-12-22 15:25:30 +0000594 MSI_SetPropertyW(package, cszSourceDir, path);
James Hawkinsc5075432006-10-10 13:39:50 -0700595
James Hawkins3797e6c2006-10-13 14:08:52 -0700596 check = msi_dup_property( package, cszSOURCEDIR );
597 if (!check)
598 MSI_SetPropertyW(package, cszSOURCEDIR, path);
599
Mike McCormack4047cc72006-10-17 16:11:42 +0900600 msi_free( package->PackagePath );
James Hawkinsc5075432006-10-10 13:39:50 -0700601 package->PackagePath = path;
602
Mike McCormackee034ba2005-09-20 11:59:14 +0000603 msi_free(check);
Aric Stewarte95136b2004-06-29 03:44:01 +0000604 }
Aric Stewart401bd3f2004-06-28 20:34:35 +0000605
Mike McCormacke3452222005-09-28 15:12:32 +0000606 msi_parse_command_line( package, szCommandLine );
Mike McCormack74f0de92005-09-29 10:32:39 +0000607
Mike McCormacke534e772006-01-04 14:51:25 +0100608 msi_apply_transforms( package );
Mike McCormack965a72a2005-10-26 12:06:21 +0000609 msi_apply_patches( package );
610
Mike McCormack74f0de92005-09-29 10:32:39 +0000611 if ( msi_get_property_int(package, szUILevel, 0) >= INSTALLUILEVEL_REDUCED )
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000612 {
Mike McCormack74f0de92005-09-29 10:32:39 +0000613 package->script->InWhatSequence |= SEQUENCE_UI;
614 rc = ACTION_ProcessUISequence(package);
615 ui = TRUE;
616 if (rc == ERROR_SUCCESS)
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000617 {
Mike McCormack74f0de92005-09-29 10:32:39 +0000618 package->script->InWhatSequence |= SEQUENCE_EXEC;
619 rc = ACTION_ProcessExecSequence(package,TRUE);
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000620 }
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000621 }
622 else
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000623 rc = ACTION_ProcessExecSequence(package,FALSE);
Aric Stewart6b16f292005-01-27 11:12:56 +0000624
625 if (rc == -1)
626 {
627 /* install was halted but should be considered a success */
628 rc = ERROR_SUCCESS;
629 }
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000630
Aric Stewart9cd707d2005-05-27 19:24:22 +0000631 package->script->CurrentlyScripting= FALSE;
632
Aric Stewart09d35c32004-12-27 19:00:26 +0000633 /* process the ending type action */
634 if (rc == ERROR_SUCCESS)
Aric Stewartf8f64402005-03-24 19:03:45 +0000635 ACTION_PerformActionSequence(package,-1,ui);
Aric Stewart6b16f292005-01-27 11:12:56 +0000636 else if (rc == ERROR_INSTALL_USEREXIT)
Aric Stewartf8f64402005-03-24 19:03:45 +0000637 ACTION_PerformActionSequence(package,-2,ui);
Aric Stewart6b16f292005-01-27 11:12:56 +0000638 else if (rc == ERROR_INSTALL_SUSPEND)
Aric Stewartf8f64402005-03-24 19:03:45 +0000639 ACTION_PerformActionSequence(package,-4,ui);
Aric Stewart9cd707d2005-05-27 19:24:22 +0000640 else /* failed */
641 ACTION_PerformActionSequence(package,-3,ui);
Aric Stewart54c67dd2005-01-25 20:17:09 +0000642
643 /* finish up running custom actions */
644 ACTION_FinishCustomActions(package);
Aric Stewart09d35c32004-12-27 19:00:26 +0000645
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000646 return rc;
647}
648
Aric Stewartf8f64402005-03-24 19:03:45 +0000649static UINT ACTION_PerformActionSequence(MSIPACKAGE *package, UINT seq, BOOL UI)
Aric Stewart09d35c32004-12-27 19:00:26 +0000650{
Mike McCormack0b352c72005-06-02 10:29:57 +0000651 UINT rc = ERROR_SUCCESS;
Aric Stewart09d35c32004-12-27 19:00:26 +0000652 MSIRECORD * row = 0;
Aric Stewart8e233e92005-03-01 11:45:19 +0000653 static const WCHAR ExecSeqQuery[] =
654 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000655 '`','I','n','s','t','a','l','l','E','x','e','c','u','t','e',
656 'S','e','q','u','e','n','c','e','`',' ', 'W','H','E','R','E',' ',
657 '`','S','e','q','u','e','n','c','e','`',' ', '=',' ','%','i',0};
Aric Stewart09d35c32004-12-27 19:00:26 +0000658
Aric Stewartf8f64402005-03-24 19:03:45 +0000659 static const WCHAR UISeqQuery[] =
660 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000661 '`','I','n','s','t','a','l','l','U','I','S','e','q','u','e','n','c','e',
662 '`', ' ', 'W','H','E','R','E',' ','`','S','e','q','u','e','n','c','e','`',
Aric Stewartf8f64402005-03-24 19:03:45 +0000663 ' ', '=',' ','%','i',0};
664
665 if (UI)
Mike McCormack0b352c72005-06-02 10:29:57 +0000666 row = MSI_QueryGetRecord(package->db, UISeqQuery, seq);
Aric Stewartf8f64402005-03-24 19:03:45 +0000667 else
Mike McCormack0b352c72005-06-02 10:29:57 +0000668 row = MSI_QueryGetRecord(package->db, ExecSeqQuery, seq);
Aric Stewart09d35c32004-12-27 19:00:26 +0000669
Mike McCormack0b352c72005-06-02 10:29:57 +0000670 if (row)
Aric Stewart09d35c32004-12-27 19:00:26 +0000671 {
Mike McCormack20806c72005-06-07 21:34:05 +0000672 LPCWSTR action, cond;
673
Aric Stewart09d35c32004-12-27 19:00:26 +0000674 TRACE("Running the actions\n");
675
Aric Stewart09d35c32004-12-27 19:00:26 +0000676 /* check conditions */
Mike McCormack20806c72005-06-07 21:34:05 +0000677 cond = MSI_RecordGetString(row,2);
678 if (cond)
Aric Stewart09d35c32004-12-27 19:00:26 +0000679 {
Mike McCormack20806c72005-06-07 21:34:05 +0000680 /* this is a hack to skip errors in the condition code */
681 if (MSI_EvaluateConditionW(package, cond) == MSICONDITION_FALSE)
682 goto end;
Aric Stewart09d35c32004-12-27 19:00:26 +0000683 }
684
Mike McCormack20806c72005-06-07 21:34:05 +0000685 action = MSI_RecordGetString(row,1);
686 if (!action)
Aric Stewart09d35c32004-12-27 19:00:26 +0000687 {
Mike McCormack20806c72005-06-07 21:34:05 +0000688 ERR("failed to fetch action\n");
689 rc = ERROR_FUNCTION_FAILED;
Aric Stewart09d35c32004-12-27 19:00:26 +0000690 goto end;
691 }
692
Aric Stewartf8f64402005-03-24 19:03:45 +0000693 if (UI)
Mike McCormack20806c72005-06-07 21:34:05 +0000694 rc = ACTION_PerformUIAction(package,action);
Aric Stewartf8f64402005-03-24 19:03:45 +0000695 else
Mike McCormack20806c72005-06-07 21:34:05 +0000696 rc = ACTION_PerformAction(package,action,FALSE);
Aric Stewart09d35c32004-12-27 19:00:26 +0000697end:
Mike McCormack0b352c72005-06-02 10:29:57 +0000698 msiobj_release(&row->hdr);
Aric Stewart09d35c32004-12-27 19:00:26 +0000699 }
700 else
701 rc = ERROR_SUCCESS;
702
703 return rc;
704}
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000705
Aric Stewart2703d712005-06-20 15:33:10 +0000706typedef struct {
707 MSIPACKAGE* package;
708 BOOL UI;
709} iterate_action_param;
710
711static UINT ITERATE_Actions(MSIRECORD *row, LPVOID param)
712{
713 iterate_action_param *iap= (iterate_action_param*)param;
714 UINT rc;
715 LPCWSTR cond, action;
716
717 action = MSI_RecordGetString(row,1);
718 if (!action)
719 {
720 ERR("Error is retrieving action name\n");
721 return ERROR_FUNCTION_FAILED;
722 }
723
724 /* check conditions */
725 cond = MSI_RecordGetString(row,2);
726 if (cond)
727 {
728 /* this is a hack to skip errors in the condition code */
729 if (MSI_EvaluateConditionW(iap->package, cond) == MSICONDITION_FALSE)
730 {
731 TRACE("Skipping action: %s (condition is false)\n",
732 debugstr_w(action));
733 return ERROR_SUCCESS;
734 }
735 }
736
737 if (iap->UI)
738 rc = ACTION_PerformUIAction(iap->package,action);
739 else
740 rc = ACTION_PerformAction(iap->package,action,FALSE);
741
Mike McCormack4f634a32005-07-06 15:44:51 +0000742 msi_dialog_check_messages( NULL );
743
744 if (iap->package->CurrentInstallState != ERROR_SUCCESS )
745 rc = iap->package->CurrentInstallState;
746
Aric Stewart2703d712005-06-20 15:33:10 +0000747 if (rc == ERROR_FUNCTION_NOT_CALLED)
748 rc = ERROR_SUCCESS;
749
750 if (rc != ERROR_SUCCESS)
Mike McCormack558abec2005-10-27 12:39:28 +0000751 ERR("Execution halted, action %s returned %i\n", debugstr_w(action), rc);
Aric Stewart2703d712005-06-20 15:33:10 +0000752
753 return rc;
754}
755
Mike McCormackd34b1c22005-09-21 10:55:23 +0000756UINT MSI_Sequence( MSIPACKAGE *package, LPCWSTR szTable, INT iSequenceMode )
757{
758 MSIQUERY * view;
759 UINT r;
760 static const WCHAR query[] =
761 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
762 '`','%','s','`',
763 ' ','W','H','E','R','E',' ',
764 '`','S','e','q','u','e','n','c','e','`',' ',
765 '>',' ','0',' ','O','R','D','E','R',' ','B','Y',' ',
766 '`','S','e','q','u','e','n','c','e','`',0};
767 iterate_action_param iap;
768
769 /*
770 * FIXME: probably should be checking UILevel in the
771 * ACTION_PerformUIAction/ACTION_PerformAction
772 * rather than saving the UI level here. Those
773 * two functions can be merged too.
774 */
775 iap.package = package;
776 iap.UI = TRUE;
777
778 TRACE("%p %s %i\n", package, debugstr_w(szTable), iSequenceMode );
779
780 r = MSI_OpenQuery( package->db, &view, query, szTable );
781 if (r == ERROR_SUCCESS)
782 {
783 r = MSI_IterateRecords( view, NULL, ITERATE_Actions, &iap );
784 msiobj_release(&view->hdr);
785 }
786
787 return r;
788}
789
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000790static UINT ACTION_ProcessExecSequence(MSIPACKAGE *package, BOOL UIran)
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000791{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000792 MSIQUERY * view;
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000793 UINT rc;
Aric Stewart8e233e92005-03-01 11:45:19 +0000794 static const WCHAR ExecSeqQuery[] =
795 {'S','E','L','E','C','T',' ','*',' ', 'F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000796 '`','I','n','s','t','a','l','l','E','x','e','c','u','t','e',
797 'S','e','q','u','e','n','c','e','`',' ', 'W','H','E','R','E',' ',
798 '`','S','e','q','u','e','n','c','e','`',' ', '>',' ','%','i',' ',
Aric Stewart8e233e92005-03-01 11:45:19 +0000799 'O','R','D','E','R',' ', 'B','Y',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000800 '`','S','e','q','u','e','n','c','e','`',0 };
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000801 MSIRECORD * row = 0;
Aric Stewart8e233e92005-03-01 11:45:19 +0000802 static const WCHAR IVQuery[] =
Aric Stewart98e38082005-05-20 09:40:42 +0000803 {'S','E','L','E','C','T',' ','`','S','e','q','u','e','n','c','e','`',
804 ' ', 'F','R','O','M',' ','`','I','n','s','t','a','l','l',
805 'E','x','e','c','u','t','e','S','e','q','u','e','n','c','e','`',' ',
806 'W','H','E','R','E',' ','`','A','c','t','i','o','n','`',' ','=',
807 ' ','\'', 'I','n','s','t','a','l','l',
808 'V','a','l','i','d','a','t','e','\'', 0};
Mike McCormack9db0e072004-12-22 15:05:07 +0000809 INT seq = 0;
Aric Stewart2703d712005-06-20 15:33:10 +0000810 iterate_action_param iap;
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000811
Aric Stewart2703d712005-06-20 15:33:10 +0000812 iap.package = package;
813 iap.UI = FALSE;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +0000814
Aric Stewart9cd707d2005-05-27 19:24:22 +0000815 if (package->script->ExecuteSequenceRun)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +0000816 {
817 TRACE("Execute Sequence already Run\n");
818 return ERROR_SUCCESS;
819 }
820
Aric Stewart9cd707d2005-05-27 19:24:22 +0000821 package->script->ExecuteSequenceRun = TRUE;
Aric Stewart2703d712005-06-20 15:33:10 +0000822
Mike McCormack9db0e072004-12-22 15:05:07 +0000823 /* get the sequence number */
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000824 if (UIran)
825 {
Mike McCormack0b352c72005-06-02 10:29:57 +0000826 row = MSI_QueryGetRecord(package->db, IVQuery);
827 if( !row )
828 return ERROR_FUNCTION_FAILED;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000829 seq = MSI_RecordGetInteger(row,1);
830 msiobj_release(&row->hdr);
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000831 }
Mike McCormack9db0e072004-12-22 15:05:07 +0000832
Mike McCormack0c238852005-01-21 16:19:11 +0000833 rc = MSI_OpenQuery(package->db, &view, ExecSeqQuery, seq);
Aric Stewart401bd3f2004-06-28 20:34:35 +0000834 if (rc == ERROR_SUCCESS)
835 {
Aric Stewart2703d712005-06-20 15:33:10 +0000836 TRACE("Running the actions\n");
Aric Stewart401bd3f2004-06-28 20:34:35 +0000837
Aric Stewart2703d712005-06-20 15:33:10 +0000838 rc = MSI_IterateRecords(view, NULL, ITERATE_Actions, &iap);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000839 msiobj_release(&view->hdr);
Aric Stewart401bd3f2004-06-28 20:34:35 +0000840 }
841
Aric Stewart401bd3f2004-06-28 20:34:35 +0000842 return rc;
843}
844
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000845static UINT ACTION_ProcessUISequence(MSIPACKAGE *package)
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000846{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000847 MSIQUERY * view;
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000848 UINT rc;
Aric Stewart8e233e92005-03-01 11:45:19 +0000849 static const WCHAR ExecSeqQuery [] =
850 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000851 '`','I','n','s','t','a','l','l',
852 'U','I','S','e','q','u','e','n','c','e','`',
853 ' ','W','H','E','R','E',' ',
854 '`','S','e','q','u','e','n','c','e','`',' ',
Aric Stewart8e233e92005-03-01 11:45:19 +0000855 '>',' ','0',' ','O','R','D','E','R',' ','B','Y',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000856 '`','S','e','q','u','e','n','c','e','`',0};
Aric Stewart2703d712005-06-20 15:33:10 +0000857 iterate_action_param iap;
858
859 iap.package = package;
860 iap.UI = TRUE;
861
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000862 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000863
864 if (rc == ERROR_SUCCESS)
865 {
Francois Gouget0edbaf72005-11-10 12:14:56 +0000866 TRACE("Running the actions\n");
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000867
Aric Stewart2703d712005-06-20 15:33:10 +0000868 rc = MSI_IterateRecords(view, NULL, ITERATE_Actions, &iap);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000869 msiobj_release(&view->hdr);
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000870 }
871
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000872 return rc;
873}
874
Aric Stewart401bd3f2004-06-28 20:34:35 +0000875/********************************************************
876 * ACTION helper functions and functions that perform the actions
877 *******************************************************/
Aric Stewart9cd707d2005-05-27 19:24:22 +0000878static BOOL ACTION_HandleStandardAction(MSIPACKAGE *package, LPCWSTR action,
879 UINT* rc, BOOL force )
Aric Stewart3f318602005-02-01 18:46:26 +0000880{
881 BOOL ret = FALSE;
Aric Stewart9cd707d2005-05-27 19:24:22 +0000882 BOOL run = force;
Aric Stewart3f318602005-02-01 18:46:26 +0000883 int i;
Aric Stewart9cd707d2005-05-27 19:24:22 +0000884
885 if (!run && !package->script->CurrentlyScripting)
886 run = TRUE;
887
888 if (!run)
889 {
890 if (strcmpW(action,szInstallFinalize) == 0 ||
891 strcmpW(action,szInstallExecute) == 0 ||
892 strcmpW(action,szInstallExecuteAgain) == 0)
893 run = TRUE;
894 }
895
Aric Stewart3f318602005-02-01 18:46:26 +0000896 i = 0;
897 while (StandardActions[i].action != NULL)
898 {
899 if (strcmpW(StandardActions[i].action, action)==0)
900 {
Aric Stewart9cd707d2005-05-27 19:24:22 +0000901 if (!run)
Aric Stewartd6ecf582005-02-02 09:29:30 +0000902 {
Aric Stewart9cd707d2005-05-27 19:24:22 +0000903 ui_actioninfo(package, action, TRUE, 0);
904 *rc = schedule_action(package,INSTALL_SCRIPT,action);
905 ui_actioninfo(package, action, FALSE, *rc);
Aric Stewartd6ecf582005-02-02 09:29:30 +0000906 }
907 else
908 {
Aric Stewart9cd707d2005-05-27 19:24:22 +0000909 ui_actionstart(package, action);
910 if (StandardActions[i].handler)
911 {
912 *rc = StandardActions[i].handler(package);
913 }
914 else
915 {
Mike McCormack54a28912005-09-06 09:23:18 +0000916 FIXME("unhandled standard action %s\n",debugstr_w(action));
Aric Stewart9cd707d2005-05-27 19:24:22 +0000917 *rc = ERROR_SUCCESS;
918 }
Aric Stewartd6ecf582005-02-02 09:29:30 +0000919 }
Aric Stewart3f318602005-02-01 18:46:26 +0000920 ret = TRUE;
921 break;
922 }
923 i++;
924 }
925 return ret;
926}
927
Mike McCormackf9acfe62005-06-07 20:29:51 +0000928static BOOL ACTION_HandleCustomAction( MSIPACKAGE* package, LPCWSTR action,
929 UINT* rc, BOOL force )
Aric Stewart3f318602005-02-01 18:46:26 +0000930{
931 BOOL ret=FALSE;
932 UINT arc;
933
Aric Stewart9cd707d2005-05-27 19:24:22 +0000934 arc = ACTION_CustomAction(package,action, force);
Aric Stewart3f318602005-02-01 18:46:26 +0000935
936 if (arc != ERROR_CALL_NOT_IMPLEMENTED)
937 {
938 *rc = arc;
939 ret = TRUE;
940 }
941 return ret;
942}
Aric Stewart401bd3f2004-06-28 20:34:35 +0000943
944/*
Francois Gougetda8b3dd2005-01-26 21:09:04 +0000945 * A lot of actions are really important even if they don't do anything
946 * explicit... Lots of properties are set at the beginning of the installation
947 * CostFinalize does a bunch of work to translate the directories and such
Aric Stewart401bd3f2004-06-28 20:34:35 +0000948 *
Mike McCormack6e2bca32004-07-04 00:25:00 +0000949 * But until I get write access to the database that is hard, so I am going to
Aric Stewart401bd3f2004-06-28 20:34:35 +0000950 * hack it to see if I can get something to run.
951 */
Aric Stewart9cd707d2005-05-27 19:24:22 +0000952UINT ACTION_PerformAction(MSIPACKAGE *package, const WCHAR *action, BOOL force)
Aric Stewart401bd3f2004-06-28 20:34:35 +0000953{
Aric Stewartd2c395a2004-07-06 18:48:15 +0000954 UINT rc = ERROR_SUCCESS;
Aric Stewart3f318602005-02-01 18:46:26 +0000955 BOOL handled;
Aric Stewart401bd3f2004-06-28 20:34:35 +0000956
957 TRACE("Performing action (%s)\n",debugstr_w(action));
958
Aric Stewart9cd707d2005-05-27 19:24:22 +0000959 handled = ACTION_HandleStandardAction(package, action, &rc, force);
Aric Stewart7d3e5972004-07-04 00:36:58 +0000960
Aric Stewart3f318602005-02-01 18:46:26 +0000961 if (!handled)
Aric Stewart9cd707d2005-05-27 19:24:22 +0000962 handled = ACTION_HandleCustomAction(package, action, &rc, force);
Aric Stewart3f318602005-02-01 18:46:26 +0000963
Aric Stewart90c57392005-01-31 16:23:12 +0000964 if (!handled)
965 {
Mike McCormack54a28912005-09-06 09:23:18 +0000966 FIXME("unhandled msi action %s\n",debugstr_w(action));
Aric Stewart3f318602005-02-01 18:46:26 +0000967 rc = ERROR_FUNCTION_NOT_CALLED;
968 }
Aric Stewart7d3e5972004-07-04 00:36:58 +0000969
Aric Stewart3f318602005-02-01 18:46:26 +0000970 return rc;
971}
972
973UINT ACTION_PerformUIAction(MSIPACKAGE *package, const WCHAR *action)
974{
975 UINT rc = ERROR_SUCCESS;
976 BOOL handled = FALSE;
977
978 TRACE("Performing action (%s)\n",debugstr_w(action));
979
Aric Stewart9cd707d2005-05-27 19:24:22 +0000980 handled = ACTION_HandleStandardAction(package, action, &rc,TRUE);
Aric Stewart3f318602005-02-01 18:46:26 +0000981
982 if (!handled)
Aric Stewart9cd707d2005-05-27 19:24:22 +0000983 handled = ACTION_HandleCustomAction(package, action, &rc, FALSE);
Aric Stewart3f318602005-02-01 18:46:26 +0000984
Mike McCormack4f634a32005-07-06 15:44:51 +0000985 if( !handled && ACTION_DialogBox(package,action) == ERROR_SUCCESS )
986 handled = TRUE;
Mike McCormack34d4a022005-02-09 13:24:31 +0000987
Aric Stewart3f318602005-02-01 18:46:26 +0000988 if (!handled)
989 {
Mike McCormack54a28912005-09-06 09:23:18 +0000990 FIXME("unhandled msi action %s\n",debugstr_w(action));
Aric Stewart3f318602005-02-01 18:46:26 +0000991 rc = ERROR_FUNCTION_NOT_CALLED;
Aric Stewart90c57392005-01-31 16:23:12 +0000992 }
Aric Stewart401bd3f2004-06-28 20:34:35 +0000993
Aric Stewartd2c395a2004-07-06 18:48:15 +0000994 return rc;
Aric Stewart401bd3f2004-06-28 20:34:35 +0000995}
996
Aric Stewart2274ff12005-06-21 20:03:46 +0000997
998/*
999 * Actual Action Handlers
1000 */
1001
1002static UINT ITERATE_CreateFolders(MSIRECORD *row, LPVOID param)
1003{
1004 MSIPACKAGE *package = (MSIPACKAGE*)param;
1005 LPCWSTR dir;
1006 LPWSTR full_path;
1007 MSIRECORD *uirow;
1008 MSIFOLDER *folder;
1009
1010 dir = MSI_RecordGetString(row,1);
1011 if (!dir)
1012 {
Francois Gouget0edbaf72005-11-10 12:14:56 +00001013 ERR("Unable to get folder id\n");
Aric Stewart2274ff12005-06-21 20:03:46 +00001014 return ERROR_SUCCESS;
1015 }
1016
1017 full_path = resolve_folder(package,dir,FALSE,FALSE,&folder);
1018 if (!full_path)
1019 {
1020 ERR("Unable to resolve folder id %s\n",debugstr_w(dir));
1021 return ERROR_SUCCESS;
1022 }
1023
1024 TRACE("Folder is %s\n",debugstr_w(full_path));
1025
1026 /* UI stuff */
1027 uirow = MSI_CreateRecord(1);
1028 MSI_RecordSetStringW(uirow,1,full_path);
1029 ui_actiondata(package,szCreateFolders,uirow);
1030 msiobj_release( &uirow->hdr );
1031
1032 if (folder->State == 0)
1033 create_full_pathW(full_path);
1034
1035 folder->State = 3;
1036
Mike McCormackee034ba2005-09-20 11:59:14 +00001037 msi_free(full_path);
Aric Stewart2274ff12005-06-21 20:03:46 +00001038 return ERROR_SUCCESS;
1039}
1040
Mike McCormack03b4dbb2005-10-28 09:39:29 +00001041/* FIXME: probably should merge this with the above function */
1042static UINT msi_create_directory( MSIPACKAGE* package, LPCWSTR dir )
1043{
1044 UINT rc = ERROR_SUCCESS;
1045 MSIFOLDER *folder;
1046 LPWSTR install_path;
1047
1048 install_path = resolve_folder(package, dir, FALSE, FALSE, &folder);
1049 if (!install_path)
1050 return ERROR_FUNCTION_FAILED;
1051
1052 /* create the path */
1053 if (folder->State == 0)
1054 {
1055 create_full_pathW(install_path);
1056 folder->State = 2;
1057 }
1058 msi_free(install_path);
1059
1060 return rc;
1061}
Aric Stewart2274ff12005-06-21 20:03:46 +00001062
Mike McCormack9c845852005-10-29 11:29:17 +00001063UINT msi_create_component_directories( MSIPACKAGE *package )
1064{
1065 MSICOMPONENT *comp;
1066
1067 /* create all the folders required by the components are going to install */
1068 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
1069 {
Mike McCormackd693f462005-10-29 11:36:48 +00001070 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
Mike McCormack9c845852005-10-29 11:29:17 +00001071 continue;
1072 msi_create_directory( package, comp->Directory );
1073 }
1074
1075 return ERROR_SUCCESS;
1076}
1077
Aric Stewart401bd3f2004-06-28 20:34:35 +00001078/*
1079 * Also we cannot enable/disable components either, so for now I am just going
Mike McCormack6e2bca32004-07-04 00:25:00 +00001080 * to do all the directories for all the components.
Aric Stewart401bd3f2004-06-28 20:34:35 +00001081 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001082static UINT ACTION_CreateFolders(MSIPACKAGE *package)
Aric Stewart401bd3f2004-06-28 20:34:35 +00001083{
Aric Stewart8e233e92005-03-01 11:45:19 +00001084 static const WCHAR ExecSeqQuery[] =
Aric Stewart98e38082005-05-20 09:40:42 +00001085 {'S','E','L','E','C','T',' ',
1086 '`','D','i','r','e','c','t','o','r','y','_','`',
Aric Stewart8e233e92005-03-01 11:45:19 +00001087 ' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001088 '`','C','r','e','a','t','e','F','o','l','d','e','r','`',0 };
Aric Stewart401bd3f2004-06-28 20:34:35 +00001089 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001090 MSIQUERY *view;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001091
Mike McCormack03b4dbb2005-10-28 09:39:29 +00001092 /* create all the empty folders specified in the CreateFolder table */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001093 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view );
Aric Stewart401bd3f2004-06-28 20:34:35 +00001094 if (rc != ERROR_SUCCESS)
Aric Stewart84837d92004-07-20 01:22:37 +00001095 return ERROR_SUCCESS;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001096
Aric Stewart2274ff12005-06-21 20:03:46 +00001097 rc = MSI_IterateRecords(view, NULL, ITERATE_CreateFolders, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001098 msiobj_release(&view->hdr);
Mike McCormack03b4dbb2005-10-28 09:39:29 +00001099
Mike McCormack9c845852005-10-29 11:29:17 +00001100 msi_create_component_directories( package );
Mike McCormack03b4dbb2005-10-28 09:39:29 +00001101
Aric Stewart401bd3f2004-06-28 20:34:35 +00001102 return rc;
1103}
1104
Mike McCormack1d46cdf2006-07-25 21:49:44 +09001105static UINT load_component( MSIRECORD *row, LPVOID param )
Aric Stewartbdb29552004-07-04 00:32:48 +00001106{
Mike McCormack1d46cdf2006-07-25 21:49:44 +09001107 MSIPACKAGE *package = param;
Mike McCormack38d67a42005-08-22 09:15:23 +00001108 MSICOMPONENT *comp;
Aric Stewartbdb29552004-07-04 00:32:48 +00001109
Mike McCormackee034ba2005-09-20 11:59:14 +00001110 comp = msi_alloc_zero( sizeof(MSICOMPONENT) );
Mike McCormack38d67a42005-08-22 09:15:23 +00001111 if (!comp)
Mike McCormack1d46cdf2006-07-25 21:49:44 +09001112 return ERROR_FUNCTION_FAILED;
1113
1114 list_add_tail( &package->components, &comp->entry );
Mike McCormack38d67a42005-08-22 09:15:23 +00001115
Aric Stewartbdb29552004-07-04 00:32:48 +00001116 /* fill in the data */
Mike McCormack51c66182005-10-27 12:36:12 +00001117 comp->Component = msi_dup_record_field( row, 1 );
Aric Stewartbdb29552004-07-04 00:32:48 +00001118
Mike McCormackefcc1ec2005-09-12 12:07:15 +00001119 TRACE("Loading Component %s\n", debugstr_w(comp->Component));
Aric Stewartbdb29552004-07-04 00:32:48 +00001120
Mike McCormack51c66182005-10-27 12:36:12 +00001121 comp->ComponentId = msi_dup_record_field( row, 2 );
1122 comp->Directory = msi_dup_record_field( row, 3 );
Mike McCormack38d67a42005-08-22 09:15:23 +00001123 comp->Attributes = MSI_RecordGetInteger(row,4);
Mike McCormack51c66182005-10-27 12:36:12 +00001124 comp->Condition = msi_dup_record_field( row, 5 );
1125 comp->KeyPath = msi_dup_record_field( row, 6 );
Aric Stewartbdb29552004-07-04 00:32:48 +00001126
James Hawkinsd893cb72006-09-20 19:55:01 -07001127 comp->Installed = INSTALLSTATE_UNKNOWN;
1128 comp->Action = INSTALLSTATE_UNKNOWN;
1129 comp->ActionRequest = INSTALLSTATE_UNKNOWN;
Aric Stewartfbdd7092004-12-27 19:06:22 +00001130
Mike McCormack1d46cdf2006-07-25 21:49:44 +09001131 return ERROR_SUCCESS;
1132}
Aric Stewartbdb29552004-07-04 00:32:48 +00001133
Mike McCormack1d46cdf2006-07-25 21:49:44 +09001134static UINT load_all_components( MSIPACKAGE *package )
1135{
1136 static const WCHAR query[] = {
1137 'S','E','L','E','C','T',' ','*',' ','F','R', 'O','M',' ',
1138 '`','C','o','m','p','o','n','e','n','t','`',0 };
1139 MSIQUERY *view;
1140 UINT r;
1141
1142 if (!list_empty(&package->components))
1143 return ERROR_SUCCESS;
1144
1145 r = MSI_DatabaseOpenViewW( package->db, query, &view );
1146 if (r != ERROR_SUCCESS)
1147 return r;
1148
1149 r = MSI_IterateRecords(view, NULL, load_component, package);
1150 msiobj_release(&view->hdr);
1151 return r;
Aric Stewartbdb29552004-07-04 00:32:48 +00001152}
1153
Aric Stewart04598242005-06-23 16:43:24 +00001154typedef struct {
1155 MSIPACKAGE *package;
Mike McCormack1da28582005-08-22 14:09:17 +00001156 MSIFEATURE *feature;
Aric Stewart04598242005-06-23 16:43:24 +00001157} _ilfs;
1158
Mike McCormack38d67a42005-08-22 09:15:23 +00001159static UINT add_feature_component( MSIFEATURE *feature, MSICOMPONENT *comp )
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001160{
1161 ComponentList *cl;
1162
Mike McCormackee034ba2005-09-20 11:59:14 +00001163 cl = msi_alloc( sizeof (*cl) );
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001164 if ( !cl )
1165 return ERROR_NOT_ENOUGH_MEMORY;
Mike McCormack38d67a42005-08-22 09:15:23 +00001166 cl->component = comp;
Mike McCormack1da28582005-08-22 14:09:17 +00001167 list_add_tail( &feature->Components, &cl->entry );
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001168
1169 return ERROR_SUCCESS;
1170}
1171
James Hawkins545d0e72006-09-20 19:59:19 -07001172static UINT add_feature_child( MSIFEATURE *parent, MSIFEATURE *child )
1173{
1174 FeatureList *fl;
1175
1176 fl = msi_alloc( sizeof(*fl) );
1177 if ( !fl )
1178 return ERROR_NOT_ENOUGH_MEMORY;
1179 fl->feature = child;
1180 list_add_tail( &parent->Children, &fl->entry );
1181
1182 return ERROR_SUCCESS;
1183}
1184
Aric Stewart04598242005-06-23 16:43:24 +00001185static UINT iterate_load_featurecomponents(MSIRECORD *row, LPVOID param)
1186{
1187 _ilfs* ilfs= (_ilfs*)param;
1188 LPCWSTR component;
Mike McCormack38d67a42005-08-22 09:15:23 +00001189 MSICOMPONENT *comp;
Aric Stewart04598242005-06-23 16:43:24 +00001190
1191 component = MSI_RecordGetString(row,1);
1192
1193 /* check to see if the component is already loaded */
Mike McCormack38d67a42005-08-22 09:15:23 +00001194 comp = get_loaded_component( ilfs->package, component );
Mike McCormack1d46cdf2006-07-25 21:49:44 +09001195 if (!comp)
Aric Stewart04598242005-06-23 16:43:24 +00001196 {
Mike McCormack1d46cdf2006-07-25 21:49:44 +09001197 ERR("unknown component %s\n", debugstr_w(component));
1198 return ERROR_FUNCTION_FAILED;
Aric Stewart04598242005-06-23 16:43:24 +00001199 }
1200
Mike McCormack1d46cdf2006-07-25 21:49:44 +09001201 add_feature_component( ilfs->feature, comp );
1202 comp->Enabled = TRUE;
Aric Stewart04598242005-06-23 16:43:24 +00001203
1204 return ERROR_SUCCESS;
1205}
1206
James Hawkins545d0e72006-09-20 19:59:19 -07001207static MSIFEATURE *find_feature_by_name( MSIPACKAGE *package, LPCWSTR name )
1208{
1209 MSIFEATURE *feature;
1210
1211 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
1212 {
1213 if ( !lstrcmpW( feature->Feature, name ) )
1214 return feature;
1215 }
1216
1217 return NULL;
1218}
1219
Aric Stewart04598242005-06-23 16:43:24 +00001220static UINT load_feature(MSIRECORD * row, LPVOID param)
1221{
1222 MSIPACKAGE* package = (MSIPACKAGE*)param;
Mike McCormack1da28582005-08-22 14:09:17 +00001223 MSIFEATURE* feature;
Aric Stewart8e233e92005-03-01 11:45:19 +00001224 static const WCHAR Query1[] =
Aric Stewart98e38082005-05-20 09:40:42 +00001225 {'S','E','L','E','C','T',' ',
1226 '`','C','o','m','p','o','n','e','n','t','_','`',
1227 ' ','F','R','O','M',' ','`','F','e','a','t','u','r','e',
1228 'C','o','m','p','o','n','e','n','t','s','`',' ',
1229 'W','H','E','R','E',' ',
1230 '`','F','e', 'a','t','u','r','e','_','`',' ','=','\'','%','s','\'',0};
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001231 MSIQUERY * view;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001232 UINT rc;
Aric Stewart04598242005-06-23 16:43:24 +00001233 _ilfs ilfs;
1234
Aric Stewartbdb29552004-07-04 00:32:48 +00001235 /* fill in the data */
1236
Mike McCormackee034ba2005-09-20 11:59:14 +00001237 feature = msi_alloc_zero( sizeof (MSIFEATURE) );
Mike McCormack1da28582005-08-22 14:09:17 +00001238 if (!feature)
1239 return ERROR_NOT_ENOUGH_MEMORY;
Aric Stewartbdb29552004-07-04 00:32:48 +00001240
James Hawkins545d0e72006-09-20 19:59:19 -07001241 list_init( &feature->Children );
Mike McCormack1da28582005-08-22 14:09:17 +00001242 list_init( &feature->Components );
Aric Stewartbdb29552004-07-04 00:32:48 +00001243
Mike McCormack51c66182005-10-27 12:36:12 +00001244 feature->Feature = msi_dup_record_field( row, 1 );
Aric Stewartbdb29552004-07-04 00:32:48 +00001245
Mike McCormack1da28582005-08-22 14:09:17 +00001246 TRACE("Loading feature %s\n",debugstr_w(feature->Feature));
Aric Stewartbdb29552004-07-04 00:32:48 +00001247
Mike McCormack51c66182005-10-27 12:36:12 +00001248 feature->Feature_Parent = msi_dup_record_field( row, 2 );
1249 feature->Title = msi_dup_record_field( row, 3 );
1250 feature->Description = msi_dup_record_field( row, 4 );
Aric Stewartbdb29552004-07-04 00:32:48 +00001251
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001252 if (!MSI_RecordIsNull(row,5))
Mike McCormack1da28582005-08-22 14:09:17 +00001253 feature->Display = MSI_RecordGetInteger(row,5);
Aric Stewartbdb29552004-07-04 00:32:48 +00001254
Mike McCormack1da28582005-08-22 14:09:17 +00001255 feature->Level= MSI_RecordGetInteger(row,6);
Mike McCormack51c66182005-10-27 12:36:12 +00001256 feature->Directory = msi_dup_record_field( row, 7 );
Mike McCormack1da28582005-08-22 14:09:17 +00001257 feature->Attributes = MSI_RecordGetInteger(row,8);
Aric Stewartfbdd7092004-12-27 19:06:22 +00001258
James Hawkinsca5c1102006-09-20 19:53:56 -07001259 feature->Installed = INSTALLSTATE_UNKNOWN;
Mike McCormack1da28582005-08-22 14:09:17 +00001260 feature->Action = INSTALLSTATE_UNKNOWN;
1261 feature->ActionRequest = INSTALLSTATE_UNKNOWN;
1262
1263 list_add_tail( &package->features, &feature->entry );
Aric Stewartbdb29552004-07-04 00:32:48 +00001264
1265 /* load feature components */
1266
Mike McCormack1da28582005-08-22 14:09:17 +00001267 rc = MSI_OpenQuery( package->db, &view, Query1, feature->Feature );
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001268 if (rc != ERROR_SUCCESS)
Aric Stewart04598242005-06-23 16:43:24 +00001269 return ERROR_SUCCESS;
Aric Stewartbdb29552004-07-04 00:32:48 +00001270
Mike McCormack1da28582005-08-22 14:09:17 +00001271 ilfs.package = package;
1272 ilfs.feature = feature;
1273
Aric Stewart04598242005-06-23 16:43:24 +00001274 MSI_IterateRecords(view, NULL, iterate_load_featurecomponents , &ilfs);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001275 msiobj_release(&view->hdr);
Aric Stewart04598242005-06-23 16:43:24 +00001276
1277 return ERROR_SUCCESS;
Aric Stewartbdb29552004-07-04 00:32:48 +00001278}
1279
James Hawkins545d0e72006-09-20 19:59:19 -07001280static UINT find_feature_children(MSIRECORD * row, LPVOID param)
1281{
1282 MSIPACKAGE* package = (MSIPACKAGE*)param;
1283 MSIFEATURE *parent, *child;
1284
1285 child = find_feature_by_name( package, MSI_RecordGetString( row, 1 ) );
1286 if (!child)
1287 return ERROR_FUNCTION_FAILED;
1288
1289 if (!child->Feature_Parent)
1290 return ERROR_SUCCESS;
1291
1292 parent = find_feature_by_name( package, child->Feature_Parent );
1293 if (!parent)
1294 return ERROR_FUNCTION_FAILED;
1295
1296 add_feature_child( parent, child );
1297 return ERROR_SUCCESS;
1298}
1299
Mike McCormack1d46cdf2006-07-25 21:49:44 +09001300static UINT load_all_features( MSIPACKAGE *package )
1301{
1302 static const WCHAR query[] = {
1303 'S','E','L','E','C','T',' ','*',' ', 'F','R','O','M',' ',
1304 '`','F','e','a','t','u','r','e','`',' ','O','R','D','E','R',
1305 ' ','B','Y',' ','`','D','i','s','p','l','a','y','`',0};
1306 MSIQUERY *view;
1307 UINT r;
1308
1309 if (!list_empty(&package->features))
1310 return ERROR_SUCCESS;
1311
1312 r = MSI_DatabaseOpenViewW( package->db, query, &view );
1313 if (r != ERROR_SUCCESS)
1314 return r;
1315
1316 r = MSI_IterateRecords( view, NULL, load_feature, package );
James Hawkins545d0e72006-09-20 19:59:19 -07001317 if (r != ERROR_SUCCESS)
1318 return r;
1319
1320 r = MSI_IterateRecords( view, NULL, find_feature_children, package );
Mike McCormack1d46cdf2006-07-25 21:49:44 +09001321 msiobj_release( &view->hdr );
James Hawkins545d0e72006-09-20 19:59:19 -07001322
Mike McCormack1d46cdf2006-07-25 21:49:44 +09001323 return r;
1324}
1325
Mike McCormackc1513be2006-03-21 19:40:36 +09001326static LPWSTR folder_split_path(LPWSTR p, WCHAR ch)
1327{
1328 if (!p)
1329 return p;
1330 p = strchrW(p, ch);
1331 if (!p)
1332 return p;
1333 *p = 0;
1334 return p+1;
1335}
1336
Aric Stewart04598242005-06-23 16:43:24 +00001337static UINT load_file(MSIRECORD *row, LPVOID param)
Aric Stewartc5a14432005-05-18 17:46:12 +00001338{
Aric Stewart04598242005-06-23 16:43:24 +00001339 MSIPACKAGE* package = (MSIPACKAGE*)param;
Aric Stewart09b0aba2005-06-09 20:30:59 +00001340 LPCWSTR component;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001341 MSIFILE *file;
Aric Stewartc5a14432005-05-18 17:46:12 +00001342
1343 /* fill in the data */
1344
Mike McCormackee034ba2005-09-20 11:59:14 +00001345 file = msi_alloc_zero( sizeof (MSIFILE) );
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001346 if (!file)
1347 return ERROR_NOT_ENOUGH_MEMORY;
Aric Stewartc5a14432005-05-18 17:46:12 +00001348
Mike McCormack51c66182005-10-27 12:36:12 +00001349 file->File = msi_dup_record_field( row, 1 );
Aric Stewartc5a14432005-05-18 17:46:12 +00001350
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001351 component = MSI_RecordGetString( row, 2 );
1352 file->Component = get_loaded_component( package, component );
Aric Stewart09b0aba2005-06-09 20:30:59 +00001353
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001354 if (!file->Component)
Aric Stewart09b0aba2005-06-09 20:30:59 +00001355 ERR("Unfound Component %s\n",debugstr_w(component));
Aric Stewartc5a14432005-05-18 17:46:12 +00001356
Mike McCormack51c66182005-10-27 12:36:12 +00001357 file->FileName = msi_dup_record_field( row, 3 );
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001358 reduce_to_longfilename( file->FileName );
Aric Stewartc5a14432005-05-18 17:46:12 +00001359
Mike McCormack51c66182005-10-27 12:36:12 +00001360 file->ShortName = msi_dup_record_field( row, 3 );
Mike McCormackc1513be2006-03-21 19:40:36 +09001361 file->LongName = strdupW( folder_split_path(file->ShortName, '|'));
Aric Stewartc5a14432005-05-18 17:46:12 +00001362
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001363 file->FileSize = MSI_RecordGetInteger( row, 4 );
Mike McCormack51c66182005-10-27 12:36:12 +00001364 file->Version = msi_dup_record_field( row, 5 );
1365 file->Language = msi_dup_record_field( row, 6 );
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001366 file->Attributes = MSI_RecordGetInteger( row, 7 );
1367 file->Sequence = MSI_RecordGetInteger( row, 8 );
Aric Stewartc5a14432005-05-18 17:46:12 +00001368
Mike McCormackdded8fb2005-11-02 10:56:42 +00001369 file->state = msifs_invalid;
Aric Stewartc5a14432005-05-18 17:46:12 +00001370
James Hawkinsf84fa0c2006-08-07 11:37:49 -07001371 /* if the compressed bits are not set in the file attributes,
1372 * then read the information from the package word count property
1373 */
James Hawkins98d14862006-07-31 11:15:52 -07001374 if (file->Attributes & msidbFileAttributesCompressed)
1375 {
James Hawkinsf84fa0c2006-08-07 11:37:49 -07001376 file->IsCompressed = TRUE;
1377 }
1378 else if (file->Attributes & msidbFileAttributesNoncompressed)
1379 {
1380 file->IsCompressed = FALSE;
1381 }
1382 else
1383 {
1384 file->IsCompressed = package->WordCount & MSIWORDCOUNT_COMPRESSED;
1385 }
1386
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001387 TRACE("File Loaded (%s)\n",debugstr_w(file->File));
1388
1389 list_add_tail( &package->files, &file->entry );
Aric Stewartc5a14432005-05-18 17:46:12 +00001390
1391 return ERROR_SUCCESS;
1392}
1393
1394static UINT load_all_files(MSIPACKAGE *package)
1395{
1396 MSIQUERY * view;
Aric Stewartc5a14432005-05-18 17:46:12 +00001397 UINT rc;
1398 static const WCHAR Query[] =
1399 {'S','E','L','E','C','T',' ','*',' ', 'F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001400 '`','F','i','l','e','`',' ', 'O','R','D','E','R',' ','B','Y',' ',
1401 '`','S','e','q','u','e','n','c','e','`', 0};
Aric Stewartc5a14432005-05-18 17:46:12 +00001402
Mike McCormack9a9195d2006-07-19 17:01:07 +09001403 if (!list_empty(&package->files))
1404 return ERROR_SUCCESS;
1405
Aric Stewartc5a14432005-05-18 17:46:12 +00001406 rc = MSI_DatabaseOpenViewW(package->db, Query, &view);
1407 if (rc != ERROR_SUCCESS)
1408 return ERROR_SUCCESS;
Aric Stewartc5a14432005-05-18 17:46:12 +00001409
Aric Stewart04598242005-06-23 16:43:24 +00001410 rc = MSI_IterateRecords(view, NULL, load_file, package);
Aric Stewartc5a14432005-05-18 17:46:12 +00001411 msiobj_release(&view->hdr);
1412
1413 return ERROR_SUCCESS;
1414}
1415
1416
Aric Stewartbdb29552004-07-04 00:32:48 +00001417/*
Mike McCormack9a9195d2006-07-19 17:01:07 +09001418 * I am not doing any of the costing functionality yet.
Aric Stewartbdb29552004-07-04 00:32:48 +00001419 * Mostly looking at doing the Component and Feature loading
1420 *
Francois Gougetda8b3dd2005-01-26 21:09:04 +00001421 * The native MSI does A LOT of modification to tables here. Mostly adding
Mike McCormack9a9195d2006-07-19 17:01:07 +09001422 * a lot of temporary columns to the Feature and Component tables.
Aric Stewartbdb29552004-07-04 00:32:48 +00001423 *
Francois Gougetda8b3dd2005-01-26 21:09:04 +00001424 * note: Native msi also tracks the short filename. But I am only going to
Aric Stewartbdb29552004-07-04 00:32:48 +00001425 * track the long ones. Also looking at this directory table
1426 * it appears that the directory table does not get the parents
Mike McCormack9a9195d2006-07-19 17:01:07 +09001427 * resolved base on property only based on their entries in the
Aric Stewartbdb29552004-07-04 00:32:48 +00001428 * directory table.
1429 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001430static UINT ACTION_CostInitialize(MSIPACKAGE *package)
Aric Stewartbdb29552004-07-04 00:32:48 +00001431{
Aric Stewart8e233e92005-03-01 11:45:19 +00001432 static const WCHAR szCosting[] =
1433 {'C','o','s','t','i','n','g','C','o','m','p','l','e','t','e',0 };
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001434 static const WCHAR szZero[] = { '0', 0 };
Aric Stewartbdb29552004-07-04 00:32:48 +00001435
Mike McCormack9293f862005-10-30 19:16:45 +00001436 if ( 1 == msi_get_property_int( package, szCosting, 0 ) )
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001437 return ERROR_SUCCESS;
Mike McCormack9a9195d2006-07-19 17:01:07 +09001438
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001439 MSI_SetPropertyW(package, szCosting, szZero);
Mike McCormack1d46cdf2006-07-25 21:49:44 +09001440 MSI_SetPropertyW(package, cszRootDrive, c_colon);
Aric Stewartbdb29552004-07-04 00:32:48 +00001441
Mike McCormack1d46cdf2006-07-25 21:49:44 +09001442 load_all_components( package );
1443 load_all_features( package );
1444 load_all_files( package );
Aric Stewartec688fb2004-07-04 00:35:52 +00001445
Aric Stewartec688fb2004-07-04 00:35:52 +00001446 return ERROR_SUCCESS;
1447}
1448
Mike McCormackf9acfe62005-06-07 20:29:51 +00001449static UINT execute_script(MSIPACKAGE *package, UINT script )
Aric Stewart9cd707d2005-05-27 19:24:22 +00001450{
1451 int i;
1452 UINT rc = ERROR_SUCCESS;
1453
1454 TRACE("Executing Script %i\n",script);
1455
Mike McCormackaa81e4f2006-01-10 12:09:19 +01001456 if (!package->script)
1457 {
1458 ERR("no script!\n");
1459 return ERROR_FUNCTION_FAILED;
1460 }
1461
Aric Stewart9cd707d2005-05-27 19:24:22 +00001462 for (i = 0; i < package->script->ActionCount[script]; i++)
1463 {
1464 LPWSTR action;
1465 action = package->script->Actions[script][i];
1466 ui_actionstart(package, action);
1467 TRACE("Executing Action (%s)\n",debugstr_w(action));
1468 rc = ACTION_PerformAction(package, action, TRUE);
Mike McCormackee034ba2005-09-20 11:59:14 +00001469 msi_free(package->script->Actions[script][i]);
Aric Stewart9cd707d2005-05-27 19:24:22 +00001470 if (rc != ERROR_SUCCESS)
1471 break;
1472 }
Mike McCormackee034ba2005-09-20 11:59:14 +00001473 msi_free(package->script->Actions[script]);
Aric Stewart9cd707d2005-05-27 19:24:22 +00001474
1475 package->script->ActionCount[script] = 0;
1476 package->script->Actions[script] = NULL;
1477 return rc;
1478}
1479
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001480static UINT ACTION_FileCost(MSIPACKAGE *package)
Aric Stewartec688fb2004-07-04 00:35:52 +00001481{
Aric Stewartbdb29552004-07-04 00:32:48 +00001482 return ERROR_SUCCESS;
1483}
1484
Mike McCormack979511f2005-08-23 18:15:44 +00001485static MSIFOLDER *load_folder( MSIPACKAGE *package, LPCWSTR dir )
Aric Stewartbdb29552004-07-04 00:32:48 +00001486{
Mike McCormackba8200b2004-12-22 15:25:30 +00001487 static const WCHAR Query[] =
Aric Stewart8e233e92005-03-01 11:45:19 +00001488 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001489 '`','D','i','r','e','c', 't','o','r','y','`',' ',
1490 'W','H','E','R','E',' ', '`', 'D','i','r','e','c','t', 'o','r','y','`',
1491 ' ','=',' ','\'','%','s','\'',
Aric Stewart8e233e92005-03-01 11:45:19 +00001492 0};
Mike McCormackc1513be2006-03-21 19:40:36 +09001493 static const WCHAR szDot[] = { '.',0 };
James Hawkinsa1910e12006-06-20 15:46:41 -07001494 static WCHAR szEmpty[] = { 0 };
Mike McCormackc1513be2006-03-21 19:40:36 +09001495 LPWSTR p, tgt_short, tgt_long, src_short, src_long;
Aric Stewart09b0aba2005-06-09 20:30:59 +00001496 LPCWSTR parent;
Mike McCormackc1513be2006-03-21 19:40:36 +09001497 MSIRECORD *row;
Mike McCormack979511f2005-08-23 18:15:44 +00001498 MSIFOLDER *folder;
Aric Stewarte95136b2004-06-29 03:44:01 +00001499
Aric Stewartbdb29552004-07-04 00:32:48 +00001500 TRACE("Looking for dir %s\n",debugstr_w(dir));
Aric Stewart401bd3f2004-06-28 20:34:35 +00001501
Mike McCormack979511f2005-08-23 18:15:44 +00001502 folder = get_loaded_folder( package, dir );
1503 if (folder)
1504 return folder;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001505
Aric Stewartbdb29552004-07-04 00:32:48 +00001506 TRACE("Working to load %s\n",debugstr_w(dir));
1507
Mike McCormackee034ba2005-09-20 11:59:14 +00001508 folder = msi_alloc_zero( sizeof (MSIFOLDER) );
Mike McCormack979511f2005-08-23 18:15:44 +00001509 if (!folder)
1510 return NULL;
Aric Stewartbdb29552004-07-04 00:32:48 +00001511
Mike McCormack979511f2005-08-23 18:15:44 +00001512 folder->Directory = strdupW(dir);
Aric Stewartbdb29552004-07-04 00:32:48 +00001513
Mike McCormack0b352c72005-06-02 10:29:57 +00001514 row = MSI_QueryGetRecord(package->db, Query, dir);
1515 if (!row)
Mike McCormack979511f2005-08-23 18:15:44 +00001516 return NULL;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001517
Mike McCormackc1513be2006-03-21 19:40:36 +09001518 p = msi_dup_record_field(row, 3);
Aric Stewart401bd3f2004-06-28 20:34:35 +00001519
1520 /* split src and target dir */
Mike McCormackc1513be2006-03-21 19:40:36 +09001521 tgt_short = p;
1522 src_short = folder_split_path( p, ':' );
Aric Stewart401bd3f2004-06-28 20:34:35 +00001523
Francois Gougete739ba92006-05-12 00:06:31 +02001524 /* split the long and short paths */
Mike McCormackc1513be2006-03-21 19:40:36 +09001525 tgt_long = folder_split_path( tgt_short, '|' );
1526 src_long = folder_split_path( src_short, '|' );
Aric Stewart401bd3f2004-06-28 20:34:35 +00001527
James Hawkinsa1910e12006-06-20 15:46:41 -07001528 /* check for no-op dirs */
Mike McCormackc1513be2006-03-21 19:40:36 +09001529 if (!lstrcmpW(szDot, tgt_short))
James Hawkinsa1910e12006-06-20 15:46:41 -07001530 tgt_short = szEmpty;
1531 if (!lstrcmpW(szDot, src_short))
1532 src_short = szEmpty;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001533
Mike McCormackc1513be2006-03-21 19:40:36 +09001534 if (!tgt_long)
1535 tgt_long = tgt_short;
Labrousse Jeand0862c32006-05-21 17:27:29 +02001536
1537 if (!src_short) {
1538 src_short = tgt_short;
1539 src_long = tgt_long;
1540 }
1541
Mike McCormackc1513be2006-03-21 19:40:36 +09001542 if (!src_long)
1543 src_long = src_short;
Mike McCormackba8200b2004-12-22 15:25:30 +00001544
Mike McCormackc1513be2006-03-21 19:40:36 +09001545 /* FIXME: use the target short path too */
1546 folder->TargetDefault = strdupW(tgt_long);
Labrousse Jeand0862c32006-05-21 17:27:29 +02001547 folder->SourceShortPath = strdupW(src_short);
Mike McCormackc1513be2006-03-21 19:40:36 +09001548 folder->SourceLongPath = strdupW(src_long);
1549 msi_free(p);
1550
1551 TRACE("TargetDefault = %s\n",debugstr_w( folder->TargetDefault ));
1552 TRACE("SourceLong = %s\n", debugstr_w( folder->SourceLongPath ));
1553 TRACE("SourceShort = %s\n", debugstr_w( folder->SourceShortPath ));
1554
1555 parent = MSI_RecordGetString(row, 2);
Mike McCormackba8200b2004-12-22 15:25:30 +00001556 if (parent)
Aric Stewart401bd3f2004-06-28 20:34:35 +00001557 {
Mike McCormack979511f2005-08-23 18:15:44 +00001558 folder->Parent = load_folder( package, parent );
1559 if ( folder->Parent )
1560 TRACE("loaded parent %p %s\n", folder->Parent,
1561 debugstr_w(folder->Parent->Directory));
1562 else
1563 ERR("failed to load parent folder %s\n", debugstr_w(parent));
Aric Stewartbdb29552004-07-04 00:32:48 +00001564 }
Aric Stewart401bd3f2004-06-28 20:34:35 +00001565
Mike McCormack062ad502005-09-15 15:04:08 +00001566 folder->Property = msi_dup_property( package, dir );
Aric Stewart401bd3f2004-06-28 20:34:35 +00001567
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001568 msiobj_release(&row->hdr);
Mike McCormack979511f2005-08-23 18:15:44 +00001569
1570 list_add_tail( &package->folders, &folder->entry );
1571
1572 TRACE("%s returning %p\n",debugstr_w(dir),folder);
1573
1574 return folder;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001575}
1576
Aric Stewart78a04e32005-02-22 15:47:00 +00001577/* scan for and update current install states */
Mike McCormackf9acfe62005-06-07 20:29:51 +00001578static void ACTION_UpdateInstallStates(MSIPACKAGE *package)
Aric Stewart78a04e32005-02-22 15:47:00 +00001579{
Mike McCormack38d67a42005-08-22 09:15:23 +00001580 MSICOMPONENT *comp;
Mike McCormack1da28582005-08-22 14:09:17 +00001581 MSIFEATURE *feature;
Aric Stewart78a04e32005-02-22 15:47:00 +00001582
James Hawkins32f57022006-09-20 19:57:03 -07001583 /* FIXME: component's installed state should be determined
1584 * by the component's registration
1585 */
Mike McCormack38d67a42005-08-22 09:15:23 +00001586 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
Aric Stewart78a04e32005-02-22 15:47:00 +00001587 {
1588 INSTALLSTATE res;
James Hawkins32f57022006-09-20 19:57:03 -07001589
1590 if (!comp->ComponentId)
1591 continue;
1592
Mike McCormack38d67a42005-08-22 09:15:23 +00001593 res = MsiGetComponentPathW( package->ProductCode,
1594 comp->ComponentId, NULL, NULL);
Aric Stewart78a04e32005-02-22 15:47:00 +00001595 if (res < 0)
1596 res = INSTALLSTATE_ABSENT;
Mike McCormack38d67a42005-08-22 09:15:23 +00001597 comp->Installed = res;
Aric Stewart78a04e32005-02-22 15:47:00 +00001598 }
1599
Mike McCormack1da28582005-08-22 14:09:17 +00001600 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewart78a04e32005-02-22 15:47:00 +00001601 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001602 ComponentList *cl;
James Hawkins545d0e72006-09-20 19:59:19 -07001603 INSTALLSTATE res = INSTALLSTATE_ABSENT;
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001604
Mike McCormack1da28582005-08-22 14:09:17 +00001605 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewart78a04e32005-02-22 15:47:00 +00001606 {
Mike McCormack38d67a42005-08-22 09:15:23 +00001607 comp= cl->component;
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001608
James Hawkins32f57022006-09-20 19:57:03 -07001609 if (!comp->ComponentId)
1610 {
1611 res = INSTALLSTATE_ABSENT;
1612 break;
1613 }
1614
James Hawkins545d0e72006-09-20 19:59:19 -07001615 if (res == INSTALLSTATE_ABSENT)
Mike McCormack38d67a42005-08-22 09:15:23 +00001616 res = comp->Installed;
Aric Stewart78a04e32005-02-22 15:47:00 +00001617 else
1618 {
Mike McCormack38d67a42005-08-22 09:15:23 +00001619 if (res == comp->Installed)
Aric Stewart78a04e32005-02-22 15:47:00 +00001620 continue;
1621
James Hawkins32f57022006-09-20 19:57:03 -07001622 if (res != INSTALLSTATE_DEFAULT || res != INSTALLSTATE_LOCAL ||
1623 res != INSTALLSTATE_SOURCE)
1624 {
1625 res = INSTALLSTATE_INCOMPLETE;
1626 }
Aric Stewart78a04e32005-02-22 15:47:00 +00001627 }
1628 }
Mike McCormack8aa1a912005-08-25 19:19:10 +00001629 feature->Installed = res;
Aric Stewart78a04e32005-02-22 15:47:00 +00001630 }
1631}
1632
Aric Stewart78a04e32005-02-22 15:47:00 +00001633static BOOL process_state_property (MSIPACKAGE* package, LPCWSTR property,
1634 INSTALLSTATE state)
Aric Stewartae1aa322004-12-27 19:02:59 +00001635{
Aric Stewartae1aa322004-12-27 19:02:59 +00001636 static const WCHAR all[]={'A','L','L',0};
Mike McCormack72faac02005-09-08 11:03:35 +00001637 LPWSTR override;
Mike McCormack1da28582005-08-22 14:09:17 +00001638 MSIFEATURE *feature;
Aric Stewartae1aa322004-12-27 19:02:59 +00001639
Mike McCormack062ad502005-09-15 15:04:08 +00001640 override = msi_dup_property( package, property );
Mike McCormack72faac02005-09-08 11:03:35 +00001641 if (!override)
1642 return FALSE;
1643
1644 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartae1aa322004-12-27 19:02:59 +00001645 {
Mike McCormack72faac02005-09-08 11:03:35 +00001646 if (strcmpiW(override,all)==0)
Aric Stewartfbdd7092004-12-27 19:06:22 +00001647 {
Mike McCormack72faac02005-09-08 11:03:35 +00001648 feature->ActionRequest= state;
1649 feature->Action = state;
1650 }
1651 else
1652 {
1653 LPWSTR ptr = override;
1654 LPWSTR ptr2 = strchrW(override,',');
Aric Stewartd900b532004-12-27 19:12:35 +00001655
Mike McCormack72faac02005-09-08 11:03:35 +00001656 while (ptr)
1657 {
1658 if ((ptr2 && strncmpW(ptr,feature->Feature, ptr2-ptr)==0)
1659 || (!ptr2 && strcmpW(ptr,feature->Feature)==0))
Aric Stewartd900b532004-12-27 19:12:35 +00001660 {
Mike McCormack72faac02005-09-08 11:03:35 +00001661 feature->ActionRequest= state;
1662 feature->Action = state;
1663 break;
Aric Stewartd900b532004-12-27 19:12:35 +00001664 }
Mike McCormack72faac02005-09-08 11:03:35 +00001665 if (ptr2)
1666 {
1667 ptr=ptr2+1;
1668 ptr2 = strchrW(ptr,',');
1669 }
1670 else
1671 break;
Aric Stewartd900b532004-12-27 19:12:35 +00001672 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001673 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001674 }
Mike McCormackee034ba2005-09-20 11:59:14 +00001675 msi_free(override);
Aric Stewart78a04e32005-02-22 15:47:00 +00001676
Mike McCormack72faac02005-09-08 11:03:35 +00001677 return TRUE;
Aric Stewart78a04e32005-02-22 15:47:00 +00001678}
1679
James Hawkins7bcac312006-07-19 11:17:16 -07001680UINT MSI_SetFeatureStates(MSIPACKAGE *package)
Aric Stewart78a04e32005-02-22 15:47:00 +00001681{
Mike McCormack74f0de92005-09-29 10:32:39 +00001682 int install_level;
Aric Stewart8e233e92005-03-01 11:45:19 +00001683 static const WCHAR szlevel[] =
1684 {'I','N','S','T','A','L','L','L','E','V','E','L',0};
1685 static const WCHAR szAddLocal[] =
1686 {'A','D','D','L','O','C','A','L',0};
1687 static const WCHAR szRemove[] =
1688 {'R','E','M','O','V','E',0};
Aric Stewartd5655f92005-11-02 14:21:17 +00001689 static const WCHAR szReinstall[] =
1690 {'R','E','I','N','S','T','A','L','L',0};
Aric Stewart78a04e32005-02-22 15:47:00 +00001691 BOOL override = FALSE;
Mike McCormack38d67a42005-08-22 09:15:23 +00001692 MSICOMPONENT* component;
Mike McCormack1da28582005-08-22 14:09:17 +00001693 MSIFEATURE *feature;
1694
Aric Stewart78a04e32005-02-22 15:47:00 +00001695
1696 /* I do not know if this is where it should happen.. but */
1697
1698 TRACE("Checking Install Level\n");
1699
Mike McCormack74f0de92005-09-29 10:32:39 +00001700 install_level = msi_get_property_int( package, szlevel, 1 );
Aric Stewart78a04e32005-02-22 15:47:00 +00001701
Francois Gouget58162f82006-10-13 02:19:42 +02001702 /* ok here is the _real_ rub
Francois Gougetfbb33432005-03-02 13:53:50 +00001703 * all these activation/deactivation things happen in order and things
1704 * later on the list override things earlier on the list.
Aric Stewart78a04e32005-02-22 15:47:00 +00001705 * 1) INSTALLLEVEL processing
1706 * 2) ADDLOCAL
1707 * 3) REMOVE
1708 * 4) ADDSOURCE
1709 * 5) ADDDEFAULT
1710 * 6) REINSTALL
1711 * 7) COMPADDLOCAL
1712 * 8) COMPADDSOURCE
1713 * 9) FILEADDLOCAL
1714 * 10) FILEADDSOURCE
1715 * 11) FILEADDDEFAULT
Francois Gougetfbb33432005-03-02 13:53:50 +00001716 * I have confirmed that if ADDLOCAL is stated then the INSTALLLEVEL is
1717 * ignored for all the features. seems strange, especially since it is not
Aric Stewart78a04e32005-02-22 15:47:00 +00001718 * documented anywhere, but it is how it works.
1719 *
Francois Gougetfbb33432005-03-02 13:53:50 +00001720 * I am still ignoring a lot of these. But that is ok for now, ADDLOCAL and
1721 * REMOVE are the big ones, since we don't handle administrative installs
1722 * yet anyway.
Aric Stewart78a04e32005-02-22 15:47:00 +00001723 */
1724 override |= process_state_property(package,szAddLocal,INSTALLSTATE_LOCAL);
1725 override |= process_state_property(package,szRemove,INSTALLSTATE_ABSENT);
Aric Stewartd5655f92005-11-02 14:21:17 +00001726 override |= process_state_property(package,szReinstall,INSTALLSTATE_LOCAL);
Aric Stewart78a04e32005-02-22 15:47:00 +00001727
1728 if (!override)
Aric Stewartfbdd7092004-12-27 19:06:22 +00001729 {
Mike McCormack1da28582005-08-22 14:09:17 +00001730 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartfbdd7092004-12-27 19:06:22 +00001731 {
Mike McCormack1da28582005-08-22 14:09:17 +00001732 BOOL feature_state = ((feature->Level > 0) &&
1733 (feature->Level <= install_level));
Aric Stewartae1aa322004-12-27 19:02:59 +00001734
Mike McCormack1da28582005-08-22 14:09:17 +00001735 if ((feature_state) && (feature->Action == INSTALLSTATE_UNKNOWN))
Aric Stewartfbdd7092004-12-27 19:06:22 +00001736 {
Mike McCormack1da28582005-08-22 14:09:17 +00001737 if (feature->Attributes & msidbFeatureAttributesFavorSource)
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001738 {
Mike McCormack1da28582005-08-22 14:09:17 +00001739 feature->ActionRequest = INSTALLSTATE_SOURCE;
1740 feature->Action = INSTALLSTATE_SOURCE;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001741 }
Mike McCormack1da28582005-08-22 14:09:17 +00001742 else if (feature->Attributes & msidbFeatureAttributesFavorAdvertise)
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001743 {
Mike McCormack1da28582005-08-22 14:09:17 +00001744 feature->ActionRequest = INSTALLSTATE_ADVERTISED;
1745 feature->Action = INSTALLSTATE_ADVERTISED;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001746 }
1747 else
1748 {
Mike McCormack1da28582005-08-22 14:09:17 +00001749 feature->ActionRequest = INSTALLSTATE_LOCAL;
1750 feature->Action = INSTALLSTATE_LOCAL;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001751 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001752 }
Aric Stewartae1aa322004-12-27 19:02:59 +00001753 }
James Hawkins545d0e72006-09-20 19:59:19 -07001754
1755 /* disable child features of unselected parent features */
1756 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
1757 {
1758 FeatureList *fl;
1759
1760 if (feature->Level > 0 && feature->Level <= install_level)
1761 continue;
1762
1763 LIST_FOR_EACH_ENTRY( fl, &feature->Children, FeatureList, entry )
1764 {
1765 fl->feature->ActionRequest = INSTALLSTATE_UNKNOWN;
1766 fl->feature->Action = INSTALLSTATE_UNKNOWN;
1767 }
1768 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001769 }
Aric Stewart6999a042005-06-08 19:20:02 +00001770 else
1771 {
1772 /* set the Preselected Property */
1773 static const WCHAR szPreselected[] = {'P','r','e','s','e','l','e','c','t','e','d',0};
1774 static const WCHAR szOne[] = { '1', 0 };
1775
1776 MSI_SetPropertyW(package,szPreselected,szOne);
1777 }
Aric Stewartae1aa322004-12-27 19:02:59 +00001778
Aric Stewartfbdd7092004-12-27 19:06:22 +00001779 /*
1780 * now we want to enable or disable components base on feature
1781 */
1782
Mike McCormack1da28582005-08-22 14:09:17 +00001783 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartfbdd7092004-12-27 19:06:22 +00001784 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001785 ComponentList *cl;
1786
Aric Stewartfbdd7092004-12-27 19:06:22 +00001787 TRACE("Examining Feature %s (Installed %i, Action %i, Request %i)\n",
1788 debugstr_w(feature->Feature), feature->Installed, feature->Action,
1789 feature->ActionRequest);
1790
Mike McCormack1da28582005-08-22 14:09:17 +00001791 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewartae1aa322004-12-27 19:02:59 +00001792 {
Mike McCormack38d67a42005-08-22 09:15:23 +00001793 component = cl->component;
Aric Stewartfbdd7092004-12-27 19:06:22 +00001794
James Hawkins929395c2006-10-19 15:51:33 -07001795 switch (component->Attributes)
1796 {
1797 case msidbComponentAttributesLocalOnly:
1798 component->Action = INSTALLSTATE_LOCAL;
1799 component->ActionRequest = INSTALLSTATE_LOCAL;
1800 break;
1801 case msidbComponentAttributesSourceOnly:
1802 component->Action = INSTALLSTATE_SOURCE;
1803 component->ActionRequest = INSTALLSTATE_SOURCE;
1804 break;
1805 case msidbComponentAttributesOptional:
1806 component->Action = INSTALLSTATE_DEFAULT;
1807 component->ActionRequest = INSTALLSTATE_DEFAULT;
1808 break;
1809 default:
1810 component->Action = INSTALLSTATE_LOCAL;
1811 component->ActionRequest = INSTALLSTATE_LOCAL;
1812 }
1813
1814 if (component->ForceLocalState)
1815 {
1816 component->Action = INSTALLSTATE_LOCAL;
1817 component->ActionRequest = INSTALLSTATE_LOCAL;
1818 }
1819
Aric Stewartfbdd7092004-12-27 19:06:22 +00001820 if (!component->Enabled)
1821 {
Aric Stewart78a04e32005-02-22 15:47:00 +00001822 component->Action = INSTALLSTATE_UNKNOWN;
1823 component->ActionRequest = INSTALLSTATE_UNKNOWN;
Aric Stewartfbdd7092004-12-27 19:06:22 +00001824 }
1825 else
1826 {
James Hawkins38f2ba22006-07-31 14:50:44 -07001827 if (feature->Attributes == msidbFeatureAttributesFavorLocal)
Aric Stewart78a04e32005-02-22 15:47:00 +00001828 {
James Hawkins38f2ba22006-07-31 14:50:44 -07001829 if (!(component->Attributes & msidbComponentAttributesSourceOnly))
1830 {
1831 component->Action = INSTALLSTATE_LOCAL;
1832 component->ActionRequest = INSTALLSTATE_LOCAL;
1833 }
Aric Stewart78a04e32005-02-22 15:47:00 +00001834 }
James Hawkins38f2ba22006-07-31 14:50:44 -07001835 else if (feature->Attributes == msidbFeatureAttributesFavorSource)
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001836 {
1837 if ((component->Action == INSTALLSTATE_UNKNOWN) ||
1838 (component->Action == INSTALLSTATE_ABSENT) ||
James Hawkins38f2ba22006-07-31 14:50:44 -07001839 (component->Action == INSTALLSTATE_ADVERTISED) ||
1840 (component->Action == INSTALLSTATE_DEFAULT))
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001841
1842 {
1843 component->Action = INSTALLSTATE_SOURCE;
1844 component->ActionRequest = INSTALLSTATE_SOURCE;
1845 }
1846 }
1847 else if (feature->ActionRequest == INSTALLSTATE_ADVERTISED)
1848 {
1849 if ((component->Action == INSTALLSTATE_UNKNOWN) ||
1850 (component->Action == INSTALLSTATE_ABSENT))
1851
1852 {
1853 component->Action = INSTALLSTATE_ADVERTISED;
1854 component->ActionRequest = INSTALLSTATE_ADVERTISED;
1855 }
1856 }
Aric Stewart78a04e32005-02-22 15:47:00 +00001857 else if (feature->ActionRequest == INSTALLSTATE_ABSENT)
1858 {
1859 if (component->Action == INSTALLSTATE_UNKNOWN)
1860 {
1861 component->Action = INSTALLSTATE_ABSENT;
1862 component->ActionRequest = INSTALLSTATE_ABSENT;
1863 }
1864 }
James Hawkinsbbd4d1e2006-09-25 20:05:25 -07001865 else if (feature->ActionRequest == INSTALLSTATE_UNKNOWN)
1866 {
1867 component->Action = INSTALLSTATE_UNKNOWN;
1868 component->ActionRequest = INSTALLSTATE_UNKNOWN;
1869 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001870 }
James Hawkins7edea0c2006-08-01 10:14:15 -07001871
James Hawkins937b9b22006-09-20 19:53:07 -07001872 if (component->ForceLocalState && feature->Action == INSTALLSTATE_SOURCE)
James Hawkins7edea0c2006-08-01 10:14:15 -07001873 {
1874 feature->Action = INSTALLSTATE_LOCAL;
1875 feature->ActionRequest = INSTALLSTATE_LOCAL;
1876 }
Aric Stewartae1aa322004-12-27 19:02:59 +00001877 }
1878 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001879
Mike McCormack38d67a42005-08-22 09:15:23 +00001880 LIST_FOR_EACH_ENTRY( component, &package->components, MSICOMPONENT, entry )
Aric Stewartfbdd7092004-12-27 19:06:22 +00001881 {
Aric Stewartfbdd7092004-12-27 19:06:22 +00001882 TRACE("Result: Component %s (Installed %i, Action %i, Request %i)\n",
1883 debugstr_w(component->Component), component->Installed,
1884 component->Action, component->ActionRequest);
1885 }
1886
1887
Aric Stewartae1aa322004-12-27 19:02:59 +00001888 return ERROR_SUCCESS;
1889}
1890
Aric Stewart443ad4d2005-06-21 20:50:12 +00001891static UINT ITERATE_CostFinalizeDirectories(MSIRECORD *row, LPVOID param)
1892{
1893 MSIPACKAGE *package = (MSIPACKAGE*)param;
1894 LPCWSTR name;
1895 LPWSTR path;
1896
1897 name = MSI_RecordGetString(row,1);
1898
1899 /* This helper function now does ALL the work */
1900 TRACE("Dir %s ...\n",debugstr_w(name));
1901 load_folder(package,name);
1902 path = resolve_folder(package,name,FALSE,TRUE,NULL);
1903 TRACE("resolves to %s\n",debugstr_w(path));
Mike McCormackee034ba2005-09-20 11:59:14 +00001904 msi_free(path);
Aric Stewart443ad4d2005-06-21 20:50:12 +00001905
1906 return ERROR_SUCCESS;
1907}
1908
1909static UINT ITERATE_CostFinalizeConditions(MSIRECORD *row, LPVOID param)
1910{
1911 MSIPACKAGE *package = (MSIPACKAGE*)param;
Mike McCormack1da28582005-08-22 14:09:17 +00001912 LPCWSTR name;
1913 MSIFEATURE *feature;
Aric Stewart443ad4d2005-06-21 20:50:12 +00001914
Mike McCormack1da28582005-08-22 14:09:17 +00001915 name = MSI_RecordGetString( row, 1 );
Aric Stewart443ad4d2005-06-21 20:50:12 +00001916
Mike McCormack1da28582005-08-22 14:09:17 +00001917 feature = get_loaded_feature( package, name );
1918 if (!feature)
1919 ERR("FAILED to find loaded feature %s\n",debugstr_w(name));
Aric Stewart443ad4d2005-06-21 20:50:12 +00001920 else
1921 {
1922 LPCWSTR Condition;
1923 Condition = MSI_RecordGetString(row,3);
1924
Aric Stewart0713f092005-06-24 11:51:29 +00001925 if (MSI_EvaluateConditionW(package,Condition) == MSICONDITION_TRUE)
Aric Stewart443ad4d2005-06-21 20:50:12 +00001926 {
1927 int level = MSI_RecordGetInteger(row,2);
Mike McCormack1da28582005-08-22 14:09:17 +00001928 TRACE("Reseting feature %s to level %i\n", debugstr_w(name), level);
1929 feature->Level = level;
Aric Stewart443ad4d2005-06-21 20:50:12 +00001930 }
1931 }
1932 return ERROR_SUCCESS;
1933}
1934
Aric Stewart401bd3f2004-06-28 20:34:35 +00001935/*
Francois Gougetda8b3dd2005-01-26 21:09:04 +00001936 * A lot is done in this function aside from just the costing.
Mike McCormack6e2bca32004-07-04 00:25:00 +00001937 * The costing needs to be implemented at some point but for now I am going
Aric Stewart401bd3f2004-06-28 20:34:35 +00001938 * to focus on the directory building
1939 *
Aric Stewart401bd3f2004-06-28 20:34:35 +00001940 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001941static UINT ACTION_CostFinalize(MSIPACKAGE *package)
Aric Stewart401bd3f2004-06-28 20:34:35 +00001942{
Aric Stewart8e233e92005-03-01 11:45:19 +00001943 static const WCHAR ExecSeqQuery[] =
1944 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001945 '`','D','i','r','e','c','t','o','r','y','`',0};
Aric Stewart8e233e92005-03-01 11:45:19 +00001946 static const WCHAR ConditionQuery[] =
1947 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001948 '`','C','o','n','d','i','t','i','o','n','`',0};
Aric Stewart8e233e92005-03-01 11:45:19 +00001949 static const WCHAR szCosting[] =
1950 {'C','o','s','t','i','n','g','C','o','m','p','l','e','t','e',0 };
1951 static const WCHAR szlevel[] =
1952 {'I','N','S','T','A','L','L','L','E','V','E','L',0};
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001953 static const WCHAR szOne[] = { '1', 0 };
Mike McCormack38d67a42005-08-22 09:15:23 +00001954 MSICOMPONENT *comp;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001955 MSIFILE *file;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001956 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001957 MSIQUERY * view;
Aric Stewart8cc14a92004-12-27 18:56:30 +00001958 LPWSTR level;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001959
Mike McCormack9293f862005-10-30 19:16:45 +00001960 if ( 1 == msi_get_property_int( package, szCosting, 0 ) )
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001961 return ERROR_SUCCESS;
Mike McCormack9293f862005-10-30 19:16:45 +00001962
Aric Stewart401bd3f2004-06-28 20:34:35 +00001963 TRACE("Building Directory properties\n");
1964
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001965 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
Aric Stewart84837d92004-07-20 01:22:37 +00001966 if (rc == ERROR_SUCCESS)
Aric Stewart401bd3f2004-06-28 20:34:35 +00001967 {
Aric Stewart443ad4d2005-06-21 20:50:12 +00001968 rc = MSI_IterateRecords(view, NULL, ITERATE_CostFinalizeDirectories,
1969 package);
Aric Stewart84837d92004-07-20 01:22:37 +00001970 msiobj_release(&view->hdr);
1971 }
Aric Stewart401bd3f2004-06-28 20:34:35 +00001972
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001973 TRACE("File calculations\n");
Aric Stewartec688fb2004-07-04 00:35:52 +00001974
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001975 LIST_FOR_EACH_ENTRY( file, &package->files, MSIFILE, entry )
Aric Stewartec688fb2004-07-04 00:35:52 +00001976 {
Mike McCormackf11c8b02005-09-09 14:48:51 +00001977 MSICOMPONENT* comp = file->Component;
1978 LPWSTR p;
Aric Stewartec688fb2004-07-04 00:35:52 +00001979
Mike McCormackf11c8b02005-09-09 14:48:51 +00001980 if (!comp)
1981 continue;
Aric Stewartec688fb2004-07-04 00:35:52 +00001982
James Hawkinsd893cb72006-09-20 19:55:01 -07001983 if (file->IsCompressed)
James Hawkinsd893cb72006-09-20 19:55:01 -07001984 comp->ForceLocalState = TRUE;
James Hawkinsd893cb72006-09-20 19:55:01 -07001985
Mike McCormackf11c8b02005-09-09 14:48:51 +00001986 /* calculate target */
1987 p = resolve_folder(package, comp->Directory, FALSE, FALSE, NULL);
1988
Mike McCormackee034ba2005-09-20 11:59:14 +00001989 msi_free(file->TargetPath);
Mike McCormackf11c8b02005-09-09 14:48:51 +00001990
1991 TRACE("file %s is named %s\n",
1992 debugstr_w(file->File),debugstr_w(file->FileName));
1993
1994 file->TargetPath = build_directory_name(2, p, file->FileName);
1995
Mike McCormackee034ba2005-09-20 11:59:14 +00001996 msi_free(p);
Mike McCormackf11c8b02005-09-09 14:48:51 +00001997
1998 TRACE("file %s resolves to %s\n",
1999 debugstr_w(file->File),debugstr_w(file->TargetPath));
2000
2001 if (GetFileAttributesW(file->TargetPath) == INVALID_FILE_ATTRIBUTES)
Aric Stewartec688fb2004-07-04 00:35:52 +00002002 {
Mike McCormackdded8fb2005-11-02 10:56:42 +00002003 file->state = msifs_missing;
Mike McCormackf11c8b02005-09-09 14:48:51 +00002004 comp->Cost += file->FileSize;
2005 continue;
2006 }
Mike McCormackba8200b2004-12-22 15:25:30 +00002007
Mike McCormackf11c8b02005-09-09 14:48:51 +00002008 if (file->Version)
2009 {
2010 DWORD handle;
2011 DWORD versize;
2012 UINT sz;
2013 LPVOID version;
Ivan Leo Puoti5b22e4d2005-11-03 11:35:11 +00002014 static WCHAR name[] = {'\\',0};
Mike McCormackf11c8b02005-09-09 14:48:51 +00002015 static const WCHAR name_fmt[] =
2016 {'%','u','.','%','u','.','%','u','.','%','u',0};
2017 WCHAR filever[0x100];
2018 VS_FIXEDFILEINFO *lpVer;
Aric Stewartfa384f62004-12-22 18:46:17 +00002019
Francois Gouget0edbaf72005-11-10 12:14:56 +00002020 TRACE("Version comparison..\n");
Mike McCormackf11c8b02005-09-09 14:48:51 +00002021 versize = GetFileVersionInfoSizeW(file->TargetPath,&handle);
Mike McCormackee034ba2005-09-20 11:59:14 +00002022 version = msi_alloc(versize);
Mike McCormackf11c8b02005-09-09 14:48:51 +00002023 GetFileVersionInfoW(file->TargetPath, 0, versize, version);
Aric Stewartfa384f62004-12-22 18:46:17 +00002024
Mike McCormackf11c8b02005-09-09 14:48:51 +00002025 VerQueryValueW(version, name, (LPVOID*)&lpVer, &sz);
Aric Stewartfa384f62004-12-22 18:46:17 +00002026
Mike McCormackf11c8b02005-09-09 14:48:51 +00002027 sprintfW(filever,name_fmt,
2028 HIWORD(lpVer->dwFileVersionMS),
2029 LOWORD(lpVer->dwFileVersionMS),
2030 HIWORD(lpVer->dwFileVersionLS),
2031 LOWORD(lpVer->dwFileVersionLS));
Aric Stewartec688fb2004-07-04 00:35:52 +00002032
Mike McCormackf11c8b02005-09-09 14:48:51 +00002033 TRACE("new %s old %s\n", debugstr_w(file->Version),
2034 debugstr_w(filever));
2035 if (strcmpiW(filever,file->Version)<0)
Aric Stewartec688fb2004-07-04 00:35:52 +00002036 {
Mike McCormackdded8fb2005-11-02 10:56:42 +00002037 file->state = msifs_overwrite;
2038 /* FIXME: cost should be diff in size */
Aric Stewartec688fb2004-07-04 00:35:52 +00002039 comp->Cost += file->FileSize;
2040 }
2041 else
Mike McCormackdded8fb2005-11-02 10:56:42 +00002042 file->state = msifs_present;
Mike McCormackee034ba2005-09-20 11:59:14 +00002043 msi_free(version);
Mike McCormackf11c8b02005-09-09 14:48:51 +00002044 }
2045 else
Mike McCormackdded8fb2005-11-02 10:56:42 +00002046 file->state = msifs_present;
Aric Stewartec688fb2004-07-04 00:35:52 +00002047 }
2048
Aric Stewart7d3e5972004-07-04 00:36:58 +00002049 TRACE("Evaluating Condition Table\n");
Aric Stewart2e9b5f72004-07-04 00:31:17 +00002050
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002051 rc = MSI_DatabaseOpenViewW(package->db, ConditionQuery, &view);
Aric Stewart84837d92004-07-20 01:22:37 +00002052 if (rc == ERROR_SUCCESS)
2053 {
Aric Stewart443ad4d2005-06-21 20:50:12 +00002054 rc = MSI_IterateRecords(view, NULL, ITERATE_CostFinalizeConditions,
2055 package);
Mike McCormackac6a4132005-01-04 20:36:12 +00002056 msiobj_release(&view->hdr);
Aric Stewart84837d92004-07-20 01:22:37 +00002057 }
Aric Stewart7d3e5972004-07-04 00:36:58 +00002058
2059 TRACE("Enabling or Disabling Components\n");
Mike McCormack38d67a42005-08-22 09:15:23 +00002060 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
Aric Stewart7d3e5972004-07-04 00:36:58 +00002061 {
Mike McCormack298cdae2005-09-06 11:39:01 +00002062 if (comp->Condition)
Aric Stewart7d3e5972004-07-04 00:36:58 +00002063 {
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002064 if (MSI_EvaluateConditionW(package,
Mike McCormack38d67a42005-08-22 09:15:23 +00002065 comp->Condition) == MSICONDITION_FALSE)
Aric Stewart7d3e5972004-07-04 00:36:58 +00002066 {
Mike McCormack38d67a42005-08-22 09:15:23 +00002067 TRACE("Disabling component %s\n", debugstr_w(comp->Component));
2068 comp->Enabled = FALSE;
Aric Stewart7d3e5972004-07-04 00:36:58 +00002069 }
2070 }
2071 }
2072
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002073 MSI_SetPropertyW(package,szCosting,szOne);
Aric Stewart8cc14a92004-12-27 18:56:30 +00002074 /* set default run level if not set */
Mike McCormack062ad502005-09-15 15:04:08 +00002075 level = msi_dup_property( package, szlevel );
Aric Stewart8cc14a92004-12-27 18:56:30 +00002076 if (!level)
2077 MSI_SetPropertyW(package,szlevel, szOne);
Mike McCormackee034ba2005-09-20 11:59:14 +00002078 msi_free(level);
Aric Stewartae1aa322004-12-27 19:02:59 +00002079
Aric Stewart78a04e32005-02-22 15:47:00 +00002080 ACTION_UpdateInstallStates(package);
Aric Stewartae1aa322004-12-27 19:02:59 +00002081
James Hawkins7c7f0bb2006-07-19 11:15:37 -07002082 return MSI_SetFeatureStates(package);
Aric Stewart401bd3f2004-06-28 20:34:35 +00002083}
2084
Francois Gougetda8b3dd2005-01-26 21:09:04 +00002085/* OK this value is "interpreted" and then formatted based on the
Aric Stewart6e160f12004-06-29 04:07:22 +00002086 first few characters */
Aric Stewart09b0aba2005-06-09 20:30:59 +00002087static LPSTR parse_value(MSIPACKAGE *package, LPCWSTR value, DWORD *type,
Aric Stewart401bd3f2004-06-28 20:34:35 +00002088 DWORD *size)
2089{
2090 LPSTR data = NULL;
Aric Stewart6e160f12004-06-29 04:07:22 +00002091 if (value[0]=='#' && value[1]!='#' && value[1]!='%')
Aric Stewart401bd3f2004-06-28 20:34:35 +00002092 {
Aric Stewart6e160f12004-06-29 04:07:22 +00002093 if (value[1]=='x')
2094 {
2095 LPWSTR ptr;
2096 CHAR byte[5];
Aric Stewart6186b2b2005-05-16 21:37:35 +00002097 LPWSTR deformated = NULL;
Aric Stewart6e160f12004-06-29 04:07:22 +00002098 int count;
2099
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002100 deformat_string(package, &value[2], &deformated);
Aric Stewart6e160f12004-06-29 04:07:22 +00002101
2102 /* binary value type */
Aric Stewart6186b2b2005-05-16 21:37:35 +00002103 ptr = deformated;
2104 *type = REG_BINARY;
2105 if (strlenW(ptr)%2)
2106 *size = (strlenW(ptr)/2)+1;
2107 else
2108 *size = strlenW(ptr)/2;
2109
Mike McCormackee034ba2005-09-20 11:59:14 +00002110 data = msi_alloc(*size);
Aric Stewart6186b2b2005-05-16 21:37:35 +00002111
Aric Stewart6e160f12004-06-29 04:07:22 +00002112 byte[0] = '0';
2113 byte[1] = 'x';
2114 byte[4] = 0;
2115 count = 0;
Aric Stewart6186b2b2005-05-16 21:37:35 +00002116 /* if uneven pad with a zero in front */
2117 if (strlenW(ptr)%2)
2118 {
2119 byte[2]= '0';
2120 byte[3]= *ptr;
2121 ptr++;
2122 data[count] = (BYTE)strtol(byte,NULL,0);
2123 count ++;
2124 TRACE("Uneven byte count\n");
2125 }
Aric Stewart6e160f12004-06-29 04:07:22 +00002126 while (*ptr)
2127 {
2128 byte[2]= *ptr;
2129 ptr++;
2130 byte[3]= *ptr;
2131 ptr++;
2132 data[count] = (BYTE)strtol(byte,NULL,0);
2133 count ++;
2134 }
Mike McCormackee034ba2005-09-20 11:59:14 +00002135 msi_free(deformated);
Aric Stewart6e160f12004-06-29 04:07:22 +00002136
Mike McCormackf1d46462006-10-05 13:41:22 +09002137 TRACE("Data %i bytes(%i)\n",*size,count);
Aric Stewart6e160f12004-06-29 04:07:22 +00002138 }
2139 else
2140 {
2141 LPWSTR deformated;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002142 LPWSTR p;
2143 DWORD d = 0;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002144 deformat_string(package, &value[1], &deformated);
Aric Stewart6e160f12004-06-29 04:07:22 +00002145
2146 *type=REG_DWORD;
2147 *size = sizeof(DWORD);
Mike McCormackee034ba2005-09-20 11:59:14 +00002148 data = msi_alloc(*size);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002149 p = deformated;
2150 if (*p == '-')
2151 p++;
2152 while (*p)
2153 {
2154 if ( (*p < '0') || (*p > '9') )
2155 break;
2156 d *= 10;
2157 d += (*p - '0');
2158 p++;
2159 }
2160 if (deformated[0] == '-')
2161 d = -d;
2162 *(LPDWORD)data = d;
Mike McCormackf1d46462006-10-05 13:41:22 +09002163 TRACE("DWORD %i\n",*(LPDWORD)data);
Aric Stewart6e160f12004-06-29 04:07:22 +00002164
Mike McCormackee034ba2005-09-20 11:59:14 +00002165 msi_free(deformated);
Aric Stewart6e160f12004-06-29 04:07:22 +00002166 }
Aric Stewart401bd3f2004-06-28 20:34:35 +00002167 }
2168 else
2169 {
Aric Stewart54c67dd2005-01-25 20:17:09 +00002170 static const WCHAR szMulti[] = {'[','~',']',0};
Aric Stewart09b0aba2005-06-09 20:30:59 +00002171 LPCWSTR ptr;
Aric Stewart6e160f12004-06-29 04:07:22 +00002172 *type=REG_SZ;
2173
Aric Stewart401bd3f2004-06-28 20:34:35 +00002174 if (value[0]=='#')
Aric Stewart6e160f12004-06-29 04:07:22 +00002175 {
2176 if (value[1]=='%')
2177 {
2178 ptr = &value[2];
2179 *type=REG_EXPAND_SZ;
2180 }
2181 else
2182 ptr = &value[1];
2183 }
2184 else
Aric Stewart401bd3f2004-06-28 20:34:35 +00002185 ptr=value;
2186
Aric Stewart54c67dd2005-01-25 20:17:09 +00002187 if (strstrW(value,szMulti))
2188 *type = REG_MULTI_SZ;
2189
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002190 *size = deformat_string(package, ptr,(LPWSTR*)&data);
Aric Stewart401bd3f2004-06-28 20:34:35 +00002191 }
2192 return data;
2193}
2194
Aric Stewart92ef78e2005-06-21 20:21:18 +00002195static UINT ITERATE_WriteRegistryValues(MSIRECORD *row, LPVOID param)
2196{
2197 MSIPACKAGE *package = (MSIPACKAGE*)param;
2198 static const WCHAR szHCR[] =
2199 {'H','K','E','Y','_','C','L','A','S','S','E','S','_',
2200 'R','O','O','T','\\',0};
2201 static const WCHAR szHCU[] =
2202 {'H','K','E','Y','_','C','U','R','R','E','N','T','_',
2203 'U','S','E','R','\\',0};
2204 static const WCHAR szHLM[] =
2205 {'H','K','E','Y','_','L','O','C','A','L','_',
2206 'M','A','C','H','I','N','E','\\',0};
2207 static const WCHAR szHU[] =
2208 {'H','K','E','Y','_','U','S','E','R','S','\\',0};
2209
2210 LPSTR value_data = NULL;
2211 HKEY root_key, hkey;
2212 DWORD type,size;
2213 LPWSTR deformated;
2214 LPCWSTR szRoot, component, name, key, value;
Mike McCormack38d67a42005-08-22 09:15:23 +00002215 MSICOMPONENT *comp;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002216 MSIRECORD * uirow;
2217 LPWSTR uikey;
2218 INT root;
2219 BOOL check_first = FALSE;
2220 UINT rc;
2221
2222 ui_progress(package,2,0,0,0);
2223
2224 value = NULL;
2225 key = NULL;
2226 uikey = NULL;
2227 name = NULL;
2228
2229 component = MSI_RecordGetString(row, 6);
Mike McCormack38d67a42005-08-22 09:15:23 +00002230 comp = get_loaded_component(package,component);
Johan Dahlin0946c422005-08-24 10:57:27 +00002231 if (!comp)
2232 return ERROR_SUCCESS;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002233
Mike McCormackd693f462005-10-29 11:36:48 +00002234 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
Aric Stewart92ef78e2005-06-21 20:21:18 +00002235 {
2236 TRACE("Skipping write due to disabled component %s\n",
2237 debugstr_w(component));
2238
Mike McCormack38d67a42005-08-22 09:15:23 +00002239 comp->Action = comp->Installed;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002240
2241 return ERROR_SUCCESS;
2242 }
2243
Mike McCormack38d67a42005-08-22 09:15:23 +00002244 comp->Action = INSTALLSTATE_LOCAL;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002245
2246 name = MSI_RecordGetString(row, 4);
2247 if( MSI_RecordIsNull(row,5) && name )
2248 {
2249 /* null values can have special meanings */
2250 if (name[0]=='-' && name[1] == 0)
2251 return ERROR_SUCCESS;
2252 else if ((name[0]=='+' && name[1] == 0) ||
2253 (name[0] == '*' && name[1] == 0))
2254 name = NULL;
2255 check_first = TRUE;
2256 }
2257
2258 root = MSI_RecordGetInteger(row,2);
2259 key = MSI_RecordGetString(row, 3);
2260
2261 /* get the root key */
2262 switch (root)
2263 {
Aric Stewart0713f092005-06-24 11:51:29 +00002264 case -1:
2265 {
2266 static const WCHAR szALLUSER[] = {'A','L','L','U','S','E','R','S',0};
Mike McCormack062ad502005-09-15 15:04:08 +00002267 LPWSTR all_users = msi_dup_property( package, szALLUSER );
Aric Stewart0713f092005-06-24 11:51:29 +00002268 if (all_users && all_users[0] == '1')
2269 {
2270 root_key = HKEY_LOCAL_MACHINE;
2271 szRoot = szHLM;
2272 }
2273 else
2274 {
2275 root_key = HKEY_CURRENT_USER;
2276 szRoot = szHCU;
2277 }
Mike McCormackee034ba2005-09-20 11:59:14 +00002278 msi_free(all_users);
Aric Stewart0713f092005-06-24 11:51:29 +00002279 }
2280 break;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002281 case 0: root_key = HKEY_CLASSES_ROOT;
2282 szRoot = szHCR;
2283 break;
2284 case 1: root_key = HKEY_CURRENT_USER;
2285 szRoot = szHCU;
2286 break;
2287 case 2: root_key = HKEY_LOCAL_MACHINE;
2288 szRoot = szHLM;
2289 break;
2290 case 3: root_key = HKEY_USERS;
2291 szRoot = szHU;
2292 break;
2293 default:
2294 ERR("Unknown root %i\n",root);
2295 root_key=NULL;
2296 szRoot = NULL;
2297 break;
2298 }
2299 if (!root_key)
2300 return ERROR_SUCCESS;
2301
2302 deformat_string(package, key , &deformated);
2303 size = strlenW(deformated) + strlenW(szRoot) + 1;
Mike McCormackee034ba2005-09-20 11:59:14 +00002304 uikey = msi_alloc(size*sizeof(WCHAR));
Aric Stewart92ef78e2005-06-21 20:21:18 +00002305 strcpyW(uikey,szRoot);
2306 strcatW(uikey,deformated);
2307
2308 if (RegCreateKeyW( root_key, deformated, &hkey))
2309 {
2310 ERR("Could not create key %s\n",debugstr_w(deformated));
Mike McCormackee034ba2005-09-20 11:59:14 +00002311 msi_free(deformated);
2312 msi_free(uikey);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002313 return ERROR_SUCCESS;
2314 }
Mike McCormackee034ba2005-09-20 11:59:14 +00002315 msi_free(deformated);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002316
2317 value = MSI_RecordGetString(row,5);
2318 if (value)
2319 value_data = parse_value(package, value, &type, &size);
2320 else
2321 {
2322 static const WCHAR szEmpty[] = {0};
2323 value_data = (LPSTR)strdupW(szEmpty);
2324 size = 0;
2325 type = REG_SZ;
2326 }
2327
2328 deformat_string(package, name, &deformated);
2329
2330 /* get the double nulls to terminate SZ_MULTI */
2331 if (type == REG_MULTI_SZ)
2332 size +=sizeof(WCHAR);
2333
2334 if (!check_first)
2335 {
2336 TRACE("Setting value %s of %s\n",debugstr_w(deformated),
2337 debugstr_w(uikey));
Mike McCormack16466af2005-07-06 10:33:30 +00002338 RegSetValueExW(hkey, deformated, 0, type, (LPBYTE)value_data, size);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002339 }
2340 else
2341 {
2342 DWORD sz = 0;
2343 rc = RegQueryValueExW(hkey, deformated, NULL, NULL, NULL, &sz);
2344 if (rc == ERROR_SUCCESS || rc == ERROR_MORE_DATA)
2345 {
2346 TRACE("value %s of %s checked already exists\n",
2347 debugstr_w(deformated), debugstr_w(uikey));
2348 }
2349 else
2350 {
2351 TRACE("Checked and setting value %s of %s\n",
2352 debugstr_w(deformated), debugstr_w(uikey));
2353 if (deformated || size)
Mike McCormack16466af2005-07-06 10:33:30 +00002354 RegSetValueExW(hkey, deformated, 0, type, (LPBYTE) value_data, size);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002355 }
2356 }
2357 RegCloseKey(hkey);
2358
2359 uirow = MSI_CreateRecord(3);
2360 MSI_RecordSetStringW(uirow,2,deformated);
2361 MSI_RecordSetStringW(uirow,1,uikey);
2362
2363 if (type == REG_SZ)
2364 MSI_RecordSetStringW(uirow,3,(LPWSTR)value_data);
2365 else
2366 MSI_RecordSetStringW(uirow,3,value);
2367
2368 ui_actiondata(package,szWriteRegistryValues,uirow);
2369 msiobj_release( &uirow->hdr );
2370
Mike McCormackee034ba2005-09-20 11:59:14 +00002371 msi_free(value_data);
2372 msi_free(deformated);
2373 msi_free(uikey);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002374
2375 return ERROR_SUCCESS;
2376}
2377
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002378static UINT ACTION_WriteRegistryValues(MSIPACKAGE *package)
Aric Stewart401bd3f2004-06-28 20:34:35 +00002379{
2380 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002381 MSIQUERY * view;
Aric Stewart8e233e92005-03-01 11:45:19 +00002382 static const WCHAR ExecSeqQuery[] =
2383 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002384 '`','R','e','g','i','s','t','r','y','`',0 };
Aric Stewart401bd3f2004-06-28 20:34:35 +00002385
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002386 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
Aric Stewart401bd3f2004-06-28 20:34:35 +00002387 if (rc != ERROR_SUCCESS)
Aric Stewart84837d92004-07-20 01:22:37 +00002388 return ERROR_SUCCESS;
Aric Stewart401bd3f2004-06-28 20:34:35 +00002389
Aric Stewartd2c395a2004-07-06 18:48:15 +00002390 /* increment progress bar each time action data is sent */
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002391 ui_progress(package,1,REG_PROGRESS_VALUE,1,0);
Aric Stewartd2c395a2004-07-06 18:48:15 +00002392
Aric Stewart92ef78e2005-06-21 20:21:18 +00002393 rc = MSI_IterateRecords(view, NULL, ITERATE_WriteRegistryValues, package);
Aric Stewartd2c395a2004-07-06 18:48:15 +00002394
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002395 msiobj_release(&view->hdr);
Aric Stewart401bd3f2004-06-28 20:34:35 +00002396 return rc;
2397}
2398
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002399static UINT ACTION_InstallInitialize(MSIPACKAGE *package)
Aric Stewart7d3e5972004-07-04 00:36:58 +00002400{
Aric Stewart9cd707d2005-05-27 19:24:22 +00002401 package->script->CurrentlyScripting = TRUE;
2402
Aric Stewart7d3e5972004-07-04 00:36:58 +00002403 return ERROR_SUCCESS;
2404}
2405
Aric Stewartae1aa322004-12-27 19:02:59 +00002406
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002407static UINT ACTION_InstallValidate(MSIPACKAGE *package)
Aric Stewart7d3e5972004-07-04 00:36:58 +00002408{
Mike McCormack38d67a42005-08-22 09:15:23 +00002409 MSICOMPONENT *comp;
Aric Stewart7d3e5972004-07-04 00:36:58 +00002410 DWORD progress = 0;
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002411 DWORD total = 0;
Aric Stewart8e233e92005-03-01 11:45:19 +00002412 static const WCHAR q1[]=
2413 {'S','E','L','E','C','T',' ','*',' ', 'F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002414 '`','R','e','g','i','s','t','r','y','`',0};
Aric Stewart7d3e5972004-07-04 00:36:58 +00002415 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002416 MSIQUERY * view;
Mike McCormack1da28582005-08-22 14:09:17 +00002417 MSIFEATURE *feature;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002418 MSIFILE *file;
Aric Stewart7d3e5972004-07-04 00:36:58 +00002419
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002420 TRACE("InstallValidate\n");
Aric Stewart7d3e5972004-07-04 00:36:58 +00002421
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002422 rc = MSI_DatabaseOpenViewW(package->db, q1, &view);
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002423 if (rc == ERROR_SUCCESS)
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002424 {
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002425 MSI_IterateRecords( view, &progress, NULL, package );
2426 msiobj_release( &view->hdr );
2427 total += progress * REG_PROGRESS_VALUE;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002428 }
Aric Stewart7d3e5972004-07-04 00:36:58 +00002429
Mike McCormack38d67a42005-08-22 09:15:23 +00002430 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
Mike McCormack38d67a42005-08-22 09:15:23 +00002431 total += COMPONENT_PROGRESS_VALUE;
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002432
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002433 LIST_FOR_EACH_ENTRY( file, &package->files, MSIFILE, entry )
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002434 total += file->FileSize;
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002435
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002436 ui_progress(package,0,total,0,0);
Aric Stewart7d3e5972004-07-04 00:36:58 +00002437
Mike McCormack1da28582005-08-22 14:09:17 +00002438 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002439 {
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002440 TRACE("Feature: %s; Installed: %i; Action %i; Request %i\n",
2441 debugstr_w(feature->Feature), feature->Installed, feature->Action,
2442 feature->ActionRequest);
2443 }
2444
Aric Stewart7d3e5972004-07-04 00:36:58 +00002445 return ERROR_SUCCESS;
2446}
2447
Aric Stewartc79f4e22005-06-22 18:03:08 +00002448static UINT ITERATE_LaunchConditions(MSIRECORD *row, LPVOID param)
2449{
2450 MSIPACKAGE* package = (MSIPACKAGE*)param;
2451 LPCWSTR cond = NULL;
2452 LPCWSTR message = NULL;
2453 static const WCHAR title[]=
2454 {'I','n','s','t','a','l','l',' ','F','a', 'i','l','e','d',0};
2455
2456 cond = MSI_RecordGetString(row,1);
2457
2458 if (MSI_EvaluateConditionW(package,cond) != MSICONDITION_TRUE)
2459 {
2460 LPWSTR deformated;
2461 message = MSI_RecordGetString(row,2);
2462 deformat_string(package,message,&deformated);
2463 MessageBoxW(NULL,deformated,title,MB_OK);
Mike McCormackee034ba2005-09-20 11:59:14 +00002464 msi_free(deformated);
Aric Stewartc79f4e22005-06-22 18:03:08 +00002465 return ERROR_FUNCTION_FAILED;
2466 }
2467
2468 return ERROR_SUCCESS;
2469}
2470
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002471static UINT ACTION_LaunchConditions(MSIPACKAGE *package)
Aric Stewart5b936ca2004-07-06 18:47:09 +00002472{
2473 UINT rc;
Mike McCormackf3c8b832004-07-19 19:35:05 +00002474 MSIQUERY * view = NULL;
Aric Stewart8e233e92005-03-01 11:45:19 +00002475 static const WCHAR ExecSeqQuery[] =
2476 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002477 '`','L','a','u','n','c','h','C','o','n','d','i','t','i','o','n','`',0};
Aric Stewart7d3e5972004-07-04 00:36:58 +00002478
Aric Stewart5b936ca2004-07-06 18:47:09 +00002479 TRACE("Checking launch conditions\n");
Aric Stewart7d3e5972004-07-04 00:36:58 +00002480
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002481 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
Aric Stewart5b936ca2004-07-06 18:47:09 +00002482 if (rc != ERROR_SUCCESS)
Aric Stewart84837d92004-07-20 01:22:37 +00002483 return ERROR_SUCCESS;
Aric Stewart5b936ca2004-07-06 18:47:09 +00002484
Aric Stewartc79f4e22005-06-22 18:03:08 +00002485 rc = MSI_IterateRecords(view, NULL, ITERATE_LaunchConditions, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002486 msiobj_release(&view->hdr);
Aric Stewartc79f4e22005-06-22 18:03:08 +00002487
Aric Stewart5b936ca2004-07-06 18:47:09 +00002488 return rc;
2489}
Aric Stewart7d3e5972004-07-04 00:36:58 +00002490
Mike McCormack38d67a42005-08-22 09:15:23 +00002491static LPWSTR resolve_keypath( MSIPACKAGE* package, MSICOMPONENT *cmp )
Aric Stewartb942e182004-07-06 18:50:02 +00002492{
Aric Stewartb942e182004-07-06 18:50:02 +00002493
Mike McCormackefcc1ec2005-09-12 12:07:15 +00002494 if (!cmp->KeyPath)
2495 return resolve_folder(package,cmp->Directory,FALSE,FALSE,NULL);
2496
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002497 if (cmp->Attributes & msidbComponentAttributesRegistryKeyPath)
Aric Stewartb942e182004-07-06 18:50:02 +00002498 {
Aric Stewart6269f002005-01-17 13:40:39 +00002499 MSIRECORD * row = 0;
Mike McCormack0b352c72005-06-02 10:29:57 +00002500 UINT root,len;
Aric Stewart09b0aba2005-06-09 20:30:59 +00002501 LPWSTR deformated,buffer,deformated_name;
2502 LPCWSTR key,name;
Aric Stewart8e233e92005-03-01 11:45:19 +00002503 static const WCHAR ExecSeqQuery[] =
2504 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002505 '`','R','e','g','i','s','t','r','y','`',' ',
2506 'W','H','E','R','E',' ', '`','R','e','g','i','s','t','r','y','`',
2507 ' ','=',' ' ,'\'','%','s','\'',0 };
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002508 static const WCHAR fmt[]={'%','0','2','i',':','\\','%','s','\\',0};
Aric Stewart8e233e92005-03-01 11:45:19 +00002509 static const WCHAR fmt2[]=
2510 {'%','0','2','i',':','\\','%','s','\\','%','s',0};
Aric Stewart6269f002005-01-17 13:40:39 +00002511
Mike McCormack0b352c72005-06-02 10:29:57 +00002512 row = MSI_QueryGetRecord(package->db, ExecSeqQuery,cmp->KeyPath);
2513 if (!row)
Aric Stewart6269f002005-01-17 13:40:39 +00002514 return NULL;
2515
Aric Stewart6269f002005-01-17 13:40:39 +00002516 root = MSI_RecordGetInteger(row,2);
Aric Stewart09b0aba2005-06-09 20:30:59 +00002517 key = MSI_RecordGetString(row, 3);
2518 name = MSI_RecordGetString(row, 4);
Aric Stewart6269f002005-01-17 13:40:39 +00002519 deformat_string(package, key , &deformated);
2520 deformat_string(package, name, &deformated_name);
2521
Ulrich Czekallae15e5172005-03-08 16:44:51 +00002522 len = strlenW(deformated) + 6;
Aric Stewart6269f002005-01-17 13:40:39 +00002523 if (deformated_name)
2524 len+=strlenW(deformated_name);
2525
Mike McCormackee034ba2005-09-20 11:59:14 +00002526 buffer = msi_alloc( len *sizeof(WCHAR));
Aric Stewart6269f002005-01-17 13:40:39 +00002527
2528 if (deformated_name)
2529 sprintfW(buffer,fmt2,root,deformated,deformated_name);
2530 else
2531 sprintfW(buffer,fmt,root,deformated);
2532
Mike McCormackee034ba2005-09-20 11:59:14 +00002533 msi_free(deformated);
2534 msi_free(deformated_name);
Aric Stewart6269f002005-01-17 13:40:39 +00002535 msiobj_release(&row->hdr);
Aric Stewart6269f002005-01-17 13:40:39 +00002536
2537 return buffer;
2538 }
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002539 else if (cmp->Attributes & msidbComponentAttributesODBCDataSource)
Aric Stewart6269f002005-01-17 13:40:39 +00002540 {
2541 FIXME("UNIMPLEMENTED keypath as ODBC Source\n");
Aric Stewartfa384f62004-12-22 18:46:17 +00002542 return NULL;
Aric Stewartb942e182004-07-06 18:50:02 +00002543 }
2544 else
2545 {
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002546 MSIFILE *file = get_loaded_file( package, cmp->KeyPath );
Aric Stewartfcb20c52004-07-06 18:51:16 +00002547
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002548 if (file)
2549 return strdupW( file->TargetPath );
Aric Stewartb942e182004-07-06 18:50:02 +00002550 }
Aric Stewartfa384f62004-12-22 18:46:17 +00002551 return NULL;
Aric Stewartb942e182004-07-06 18:50:02 +00002552}
2553
Stefan Huehnerac6f5622005-06-20 14:18:03 +00002554static HKEY openSharedDLLsKey(void)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002555{
2556 HKEY hkey=0;
Aric Stewart8e233e92005-03-01 11:45:19 +00002557 static const WCHAR path[] =
2558 {'S','o','f','t','w','a','r','e','\\',
2559 'M','i','c','r','o','s','o','f','t','\\',
2560 'W','i','n','d','o','w','s','\\',
2561 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
2562 'S','h','a','r','e','d','D','L','L','s',0};
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002563
2564 RegCreateKeyW(HKEY_LOCAL_MACHINE,path,&hkey);
2565 return hkey;
2566}
2567
2568static UINT ACTION_GetSharedDLLsCount(LPCWSTR dll)
2569{
2570 HKEY hkey;
2571 DWORD count=0;
2572 DWORD type;
2573 DWORD sz = sizeof(count);
2574 DWORD rc;
2575
2576 hkey = openSharedDLLsKey();
2577 rc = RegQueryValueExW(hkey, dll, NULL, &type, (LPBYTE)&count, &sz);
2578 if (rc != ERROR_SUCCESS)
2579 count = 0;
2580 RegCloseKey(hkey);
2581 return count;
2582}
2583
2584static UINT ACTION_WriteSharedDLLsCount(LPCWSTR path, UINT count)
2585{
2586 HKEY hkey;
2587
2588 hkey = openSharedDLLsKey();
2589 if (count > 0)
Mike McCormack4db02cd2005-09-15 14:58:38 +00002590 msi_reg_set_val_dword( hkey, path, count );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002591 else
2592 RegDeleteValueW(hkey,path);
2593 RegCloseKey(hkey);
2594 return count;
2595}
2596
2597/*
2598 * Return TRUE if the count should be written out and FALSE if not
2599 */
Mike McCormack38d67a42005-08-22 09:15:23 +00002600static void ACTION_RefCountComponent( MSIPACKAGE* package, MSICOMPONENT *comp )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002601{
Mike McCormack1da28582005-08-22 14:09:17 +00002602 MSIFEATURE *feature;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002603 INT count = 0;
2604 BOOL write = FALSE;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002605
2606 /* only refcount DLLs */
Mike McCormackefcc1ec2005-09-12 12:07:15 +00002607 if (comp->KeyPath == NULL ||
Mike McCormack38d67a42005-08-22 09:15:23 +00002608 comp->Attributes & msidbComponentAttributesRegistryKeyPath ||
2609 comp->Attributes & msidbComponentAttributesODBCDataSource)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002610 write = FALSE;
2611 else
2612 {
Mike McCormack38d67a42005-08-22 09:15:23 +00002613 count = ACTION_GetSharedDLLsCount( comp->FullKeypath);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002614 write = (count > 0);
2615
Mike McCormack38d67a42005-08-22 09:15:23 +00002616 if (comp->Attributes & msidbComponentAttributesSharedDllRefCount)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002617 write = TRUE;
2618 }
2619
2620 /* increment counts */
Mike McCormack1da28582005-08-22 14:09:17 +00002621 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002622 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00002623 ComponentList *cl;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002624
Mike McCormack1da28582005-08-22 14:09:17 +00002625 if (!ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_LOCAL ))
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002626 continue;
2627
Mike McCormack1da28582005-08-22 14:09:17 +00002628 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002629 {
Mike McCormack38d67a42005-08-22 09:15:23 +00002630 if ( cl->component == comp )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002631 count++;
2632 }
2633 }
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002634
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002635 /* decrement counts */
Mike McCormack1da28582005-08-22 14:09:17 +00002636 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002637 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00002638 ComponentList *cl;
2639
Mike McCormack1da28582005-08-22 14:09:17 +00002640 if (!ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_ABSENT ))
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002641 continue;
2642
Mike McCormack1da28582005-08-22 14:09:17 +00002643 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002644 {
Mike McCormack38d67a42005-08-22 09:15:23 +00002645 if ( cl->component == comp )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002646 count--;
2647 }
2648 }
2649
2650 /* ref count all the files in the component */
2651 if (write)
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002652 {
2653 MSIFILE *file;
2654
2655 LIST_FOR_EACH_ENTRY( file, &package->files, MSIFILE, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002656 {
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002657 if (file->Component == comp)
2658 ACTION_WriteSharedDLLsCount( file->TargetPath, count );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002659 }
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002660 }
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002661
2662 /* add a count for permenent */
Mike McCormack38d67a42005-08-22 09:15:23 +00002663 if (comp->Attributes & msidbComponentAttributesPermanent)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002664 count ++;
2665
Mike McCormack38d67a42005-08-22 09:15:23 +00002666 comp->RefCount = count;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002667
2668 if (write)
Mike McCormack38d67a42005-08-22 09:15:23 +00002669 ACTION_WriteSharedDLLsCount( comp->FullKeypath, comp->RefCount );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002670}
2671
Aric Stewart2cf222f2004-07-06 19:00:23 +00002672/*
2673 * Ok further analysis makes me think that this work is
2674 * actually done in the PublishComponents and PublishFeatures
Mike McCormack3ece2462004-07-09 19:33:25 +00002675 * step, and not here. It appears like the keypath and all that is
2676 * resolved in this step, however actually written in the Publish steps.
Alexandre Julliard77b12762004-07-09 19:43:29 +00002677 * But we will leave it here for now because it is unclear
Aric Stewart2cf222f2004-07-06 19:00:23 +00002678 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002679static UINT ACTION_ProcessComponents(MSIPACKAGE *package)
Aric Stewartb942e182004-07-06 18:50:02 +00002680{
Aric Stewart68b07492005-01-25 11:05:37 +00002681 WCHAR squished_pc[GUID_SIZE];
2682 WCHAR squished_cc[GUID_SIZE];
Aric Stewartb942e182004-07-06 18:50:02 +00002683 UINT rc;
Mike McCormack38d67a42005-08-22 09:15:23 +00002684 MSICOMPONENT *comp;
Aric Stewart68b07492005-01-25 11:05:37 +00002685 HKEY hkey=0,hkey2=0;
Aric Stewartb942e182004-07-06 18:50:02 +00002686
Aric Stewartb942e182004-07-06 18:50:02 +00002687 /* writes the Component and Features values to the registry */
Aric Stewartb942e182004-07-06 18:50:02 +00002688
Aric Stewart68b07492005-01-25 11:05:37 +00002689 rc = MSIREG_OpenComponents(&hkey);
Aric Stewartb942e182004-07-06 18:50:02 +00002690 if (rc != ERROR_SUCCESS)
Mike McCormackfe8cd382006-03-09 14:21:37 +09002691 return rc;
2692
Aric Stewartadaef112005-07-07 20:27:06 +00002693 squash_guid(package->ProductCode,squished_pc);
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002694 ui_progress(package,1,COMPONENT_PROGRESS_VALUE,1,0);
Mike McCormack38d67a42005-08-22 09:15:23 +00002695
2696 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
Aric Stewartb942e182004-07-06 18:50:02 +00002697 {
Mike McCormackfe8cd382006-03-09 14:21:37 +09002698 MSIRECORD * uirow;
2699
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002700 ui_progress(package,2,0,0,0);
Mike McCormackfe8cd382006-03-09 14:21:37 +09002701 if (!comp->ComponentId)
2702 continue;
Aric Stewartb942e182004-07-06 18:50:02 +00002703
Mike McCormackfe8cd382006-03-09 14:21:37 +09002704 squash_guid(comp->ComponentId,squished_cc);
Aric Stewartb942e182004-07-06 18:50:02 +00002705
Mike McCormackfe8cd382006-03-09 14:21:37 +09002706 msi_free(comp->FullKeypath);
2707 comp->FullKeypath = resolve_keypath( package, comp );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002708
Mike McCormackfe8cd382006-03-09 14:21:37 +09002709 /* do the refcounting */
2710 ACTION_RefCountComponent( package, comp );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002711
Mike McCormackfe8cd382006-03-09 14:21:37 +09002712 TRACE("Component %s (%s), Keypath=%s, RefCount=%i\n",
Mike McCormack38d67a42005-08-22 09:15:23 +00002713 debugstr_w(comp->Component),
Aric Stewartc5a14432005-05-18 17:46:12 +00002714 debugstr_w(squished_cc),
Mike McCormackfe8cd382006-03-09 14:21:37 +09002715 debugstr_w(comp->FullKeypath),
Mike McCormack38d67a42005-08-22 09:15:23 +00002716 comp->RefCount);
Mike McCormackfe8cd382006-03-09 14:21:37 +09002717 /*
2718 * Write the keypath out if the component is to be registered
2719 * and delete the key if the component is to be deregistered
2720 */
2721 if (ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
2722 {
2723 rc = RegCreateKeyW(hkey,squished_cc,&hkey2);
2724 if (rc != ERROR_SUCCESS)
2725 continue;
2726
2727 if (!comp->FullKeypath)
2728 continue;
2729
2730 msi_reg_set_val_str( hkey2, squished_pc, comp->FullKeypath );
2731
2732 if (comp->Attributes & msidbComponentAttributesPermanent)
Aric Stewartfa384f62004-12-22 18:46:17 +00002733 {
Mike McCormackfe8cd382006-03-09 14:21:37 +09002734 static const WCHAR szPermKey[] =
2735 { '0','0','0','0','0','0','0','0','0','0','0','0',
2736 '0','0','0','0','0','0','0','0','0','0','0','0',
2737 '0','0','0','0','0','0','0','0',0 };
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002738
Mike McCormackfe8cd382006-03-09 14:21:37 +09002739 msi_reg_set_val_str( hkey2, szPermKey, comp->FullKeypath );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002740 }
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002741
Mike McCormackfe8cd382006-03-09 14:21:37 +09002742 RegCloseKey(hkey2);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002743
Mike McCormackfe8cd382006-03-09 14:21:37 +09002744 /* UI stuff */
2745 uirow = MSI_CreateRecord(3);
2746 MSI_RecordSetStringW(uirow,1,package->ProductCode);
2747 MSI_RecordSetStringW(uirow,2,comp->ComponentId);
2748 MSI_RecordSetStringW(uirow,3,comp->FullKeypath);
2749 ui_actiondata(package,szProcessComponents,uirow);
2750 msiobj_release( &uirow->hdr );
2751 }
2752 else if (ACTION_VerifyComponentForAction( comp, INSTALLSTATE_ABSENT))
2753 {
2754 DWORD res;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002755
Mike McCormackfe8cd382006-03-09 14:21:37 +09002756 rc = RegOpenKeyW(hkey,squished_cc,&hkey2);
2757 if (rc != ERROR_SUCCESS)
2758 continue;
2759
2760 RegDeleteValueW(hkey2,squished_pc);
2761
2762 /* if the key is empty delete it */
2763 res = RegEnumKeyExW(hkey2,0,NULL,0,0,NULL,0,NULL);
2764 RegCloseKey(hkey2);
2765 if (res == ERROR_NO_MORE_ITEMS)
2766 RegDeleteKeyW(hkey,squished_cc);
2767
2768 /* UI stuff */
2769 uirow = MSI_CreateRecord(2);
2770 MSI_RecordSetStringW(uirow,1,package->ProductCode);
2771 MSI_RecordSetStringW(uirow,2,comp->ComponentId);
2772 ui_actiondata(package,szProcessComponents,uirow);
2773 msiobj_release( &uirow->hdr );
Aric Stewartb942e182004-07-06 18:50:02 +00002774 }
2775 }
Aric Stewartb942e182004-07-06 18:50:02 +00002776 RegCloseKey(hkey);
2777 return rc;
2778}
2779
Aric Stewart6e821732005-03-30 10:19:08 +00002780typedef struct {
2781 CLSID clsid;
2782 LPWSTR source;
2783
2784 LPWSTR path;
2785 ITypeLib *ptLib;
2786} typelib_struct;
2787
Mike McCormackf9acfe62005-06-07 20:29:51 +00002788static BOOL CALLBACK Typelib_EnumResNameProc( HMODULE hModule, LPCWSTR lpszType,
Aric Stewart6e821732005-03-30 10:19:08 +00002789 LPWSTR lpszName, LONG_PTR lParam)
2790{
2791 TLIBATTR *attr;
2792 typelib_struct *tl_struct = (typelib_struct*) lParam;
2793 static const WCHAR fmt[] = {'%','s','\\','%','i',0};
2794 int sz;
2795 HRESULT res;
2796
2797 if (!IS_INTRESOURCE(lpszName))
2798 {
2799 ERR("Not Int Resource Name %s\n",debugstr_w(lpszName));
2800 return TRUE;
2801 }
2802
2803 sz = strlenW(tl_struct->source)+4;
2804 sz *= sizeof(WCHAR);
2805
Mike McCormack2acf8002006-05-25 11:41:39 +09002806 if ((INT_PTR)lpszName == 1)
Aric Stewartca8c4e42005-06-02 15:13:57 +00002807 tl_struct->path = strdupW(tl_struct->source);
2808 else
2809 {
Mike McCormackee034ba2005-09-20 11:59:14 +00002810 tl_struct->path = msi_alloc(sz);
Aric Stewartca8c4e42005-06-02 15:13:57 +00002811 sprintfW(tl_struct->path,fmt,tl_struct->source, lpszName);
2812 }
Aric Stewart6e821732005-03-30 10:19:08 +00002813
2814 TRACE("trying %s\n", debugstr_w(tl_struct->path));
2815 res = LoadTypeLib(tl_struct->path,&tl_struct->ptLib);
2816 if (!SUCCEEDED(res))
2817 {
Mike McCormackee034ba2005-09-20 11:59:14 +00002818 msi_free(tl_struct->path);
Aric Stewart6e821732005-03-30 10:19:08 +00002819 tl_struct->path = NULL;
2820
2821 return TRUE;
2822 }
2823
2824 ITypeLib_GetLibAttr(tl_struct->ptLib, &attr);
2825 if (IsEqualGUID(&(tl_struct->clsid),&(attr->guid)))
2826 {
2827 ITypeLib_ReleaseTLibAttr(tl_struct->ptLib, attr);
2828 return FALSE;
2829 }
2830
Mike McCormackee034ba2005-09-20 11:59:14 +00002831 msi_free(tl_struct->path);
Aric Stewart6e821732005-03-30 10:19:08 +00002832 tl_struct->path = NULL;
2833
2834 ITypeLib_ReleaseTLibAttr(tl_struct->ptLib, attr);
2835 ITypeLib_Release(tl_struct->ptLib);
2836
2837 return TRUE;
2838}
2839
Aric Stewart234dc4b2005-06-22 18:27:34 +00002840static UINT ITERATE_RegisterTypeLibraries(MSIRECORD *row, LPVOID param)
2841{
2842 MSIPACKAGE* package = (MSIPACKAGE*)param;
2843 LPCWSTR component;
Mike McCormack38d67a42005-08-22 09:15:23 +00002844 MSICOMPONENT *comp;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002845 MSIFILE *file;
Aric Stewart234dc4b2005-06-22 18:27:34 +00002846 typelib_struct tl_struct;
2847 HMODULE module;
2848 static const WCHAR szTYPELIB[] = {'T','Y','P','E','L','I','B',0};
2849
2850 component = MSI_RecordGetString(row,3);
Mike McCormack38d67a42005-08-22 09:15:23 +00002851 comp = get_loaded_component(package,component);
2852 if (!comp)
Aric Stewart234dc4b2005-06-22 18:27:34 +00002853 return ERROR_SUCCESS;
2854
Mike McCormackd693f462005-10-29 11:36:48 +00002855 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
Aric Stewart234dc4b2005-06-22 18:27:34 +00002856 {
2857 TRACE("Skipping typelib reg due to disabled component\n");
2858
Mike McCormack38d67a42005-08-22 09:15:23 +00002859 comp->Action = comp->Installed;
Aric Stewart234dc4b2005-06-22 18:27:34 +00002860
2861 return ERROR_SUCCESS;
2862 }
2863
Mike McCormack38d67a42005-08-22 09:15:23 +00002864 comp->Action = INSTALLSTATE_LOCAL;
Aric Stewart234dc4b2005-06-22 18:27:34 +00002865
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002866 file = get_loaded_file( package, comp->KeyPath );
2867 if (!file)
Aric Stewart234dc4b2005-06-22 18:27:34 +00002868 return ERROR_SUCCESS;
2869
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002870 module = LoadLibraryExW( file->TargetPath, NULL, LOAD_LIBRARY_AS_DATAFILE );
Mike McCormack51c66182005-10-27 12:36:12 +00002871 if (module)
Aric Stewart234dc4b2005-06-22 18:27:34 +00002872 {
Mike McCormack51c66182005-10-27 12:36:12 +00002873 LPCWSTR guid;
2874 guid = MSI_RecordGetString(row,1);
2875 CLSIDFromString((LPWSTR)guid, &tl_struct.clsid);
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002876 tl_struct.source = strdupW( file->TargetPath );
Aric Stewart234dc4b2005-06-22 18:27:34 +00002877 tl_struct.path = NULL;
2878
2879 EnumResourceNamesW(module, szTYPELIB, Typelib_EnumResNameProc,
2880 (LONG_PTR)&tl_struct);
2881
Mike McCormack51c66182005-10-27 12:36:12 +00002882 if (tl_struct.path)
Aric Stewart234dc4b2005-06-22 18:27:34 +00002883 {
2884 LPWSTR help = NULL;
2885 LPCWSTR helpid;
2886 HRESULT res;
2887
2888 helpid = MSI_RecordGetString(row,6);
2889
2890 if (helpid)
2891 help = resolve_folder(package,helpid,FALSE,FALSE,NULL);
2892 res = RegisterTypeLib(tl_struct.ptLib,tl_struct.path,help);
Mike McCormackee034ba2005-09-20 11:59:14 +00002893 msi_free(help);
Aric Stewart234dc4b2005-06-22 18:27:34 +00002894
2895 if (!SUCCEEDED(res))
2896 ERR("Failed to register type library %s\n",
2897 debugstr_w(tl_struct.path));
2898 else
2899 {
2900 ui_actiondata(package,szRegisterTypeLibraries,row);
2901
2902 TRACE("Registered %s\n", debugstr_w(tl_struct.path));
2903 }
2904
2905 ITypeLib_Release(tl_struct.ptLib);
Mike McCormackee034ba2005-09-20 11:59:14 +00002906 msi_free(tl_struct.path);
Aric Stewart234dc4b2005-06-22 18:27:34 +00002907 }
2908 else
2909 ERR("Failed to load type library %s\n",
2910 debugstr_w(tl_struct.source));
2911
2912 FreeLibrary(module);
Mike McCormackee034ba2005-09-20 11:59:14 +00002913 msi_free(tl_struct.source);
Aric Stewart234dc4b2005-06-22 18:27:34 +00002914 }
2915 else
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002916 ERR("Could not load file! %s\n", debugstr_w(file->TargetPath));
Aric Stewart234dc4b2005-06-22 18:27:34 +00002917
2918 return ERROR_SUCCESS;
2919}
2920
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002921static UINT ACTION_RegisterTypeLibraries(MSIPACKAGE *package)
Aric Stewartfcb20c52004-07-06 18:51:16 +00002922{
2923 /*
Mike McCormackc90c7812004-07-09 22:58:27 +00002924 * OK this is a bit confusing.. I am given a _Component key and I believe
Aric Stewartfcb20c52004-07-06 18:51:16 +00002925 * that the file that is being registered as a type library is the "key file
Mike McCormackc90c7812004-07-09 22:58:27 +00002926 * of that component" which I interpret to mean "The file in the KeyPath of
2927 * that component".
Aric Stewartfcb20c52004-07-06 18:51:16 +00002928 */
2929 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002930 MSIQUERY * view;
Aric Stewart8e233e92005-03-01 11:45:19 +00002931 static const WCHAR Query[] =
2932 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002933 '`','T','y','p','e','L','i','b','`',0};
Aric Stewartfcb20c52004-07-06 18:51:16 +00002934
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002935 rc = MSI_DatabaseOpenViewW(package->db, Query, &view);
Aric Stewartfcb20c52004-07-06 18:51:16 +00002936 if (rc != ERROR_SUCCESS)
Aric Stewart84837d92004-07-20 01:22:37 +00002937 return ERROR_SUCCESS;
Aric Stewartfcb20c52004-07-06 18:51:16 +00002938
Aric Stewart234dc4b2005-06-22 18:27:34 +00002939 rc = MSI_IterateRecords(view, NULL, ITERATE_RegisterTypeLibraries, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002940 msiobj_release(&view->hdr);
Aric Stewartfcb20c52004-07-06 18:51:16 +00002941 return rc;
Aric Stewartfcb20c52004-07-06 18:51:16 +00002942}
2943
Aric Stewart9adacf62005-06-24 11:58:21 +00002944static UINT ITERATE_CreateShortcuts(MSIRECORD *row, LPVOID param)
Aric Stewart2cf222f2004-07-06 19:00:23 +00002945{
Aric Stewart9adacf62005-06-24 11:58:21 +00002946 MSIPACKAGE *package = (MSIPACKAGE*)param;
Mike McCormack477bce32006-01-16 20:38:28 +01002947 LPWSTR target_file, target_folder, filename;
Robert Shearman4ac85672006-02-22 16:31:00 +00002948 LPCWSTR buffer, extension;
Mike McCormack38d67a42005-08-22 09:15:23 +00002949 MSICOMPONENT *comp;
Aric Stewart9adacf62005-06-24 11:58:21 +00002950 static const WCHAR szlnk[]={'.','l','n','k',0};
Mike McCormack20c57462006-05-24 17:41:04 +09002951 IShellLinkW *sl = NULL;
2952 IPersistFile *pf = NULL;
Aric Stewart2cf222f2004-07-06 19:00:23 +00002953 HRESULT res;
2954
Aric Stewart9adacf62005-06-24 11:58:21 +00002955 buffer = MSI_RecordGetString(row,4);
Mike McCormack38d67a42005-08-22 09:15:23 +00002956 comp = get_loaded_component(package,buffer);
2957 if (!comp)
Aric Stewart9adacf62005-06-24 11:58:21 +00002958 return ERROR_SUCCESS;
2959
Mike McCormackd693f462005-10-29 11:36:48 +00002960 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL ))
Aric Stewart9adacf62005-06-24 11:58:21 +00002961 {
2962 TRACE("Skipping shortcut creation due to disabled component\n");
2963
Mike McCormack38d67a42005-08-22 09:15:23 +00002964 comp->Action = comp->Installed;
Aric Stewart9adacf62005-06-24 11:58:21 +00002965
2966 return ERROR_SUCCESS;
2967 }
2968
Mike McCormack38d67a42005-08-22 09:15:23 +00002969 comp->Action = INSTALLSTATE_LOCAL;
Aric Stewart9adacf62005-06-24 11:58:21 +00002970
2971 ui_actiondata(package,szCreateShortcuts,row);
2972
2973 res = CoCreateInstance( &CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER,
2974 &IID_IShellLinkW, (LPVOID *) &sl );
2975
Mike McCormack20c57462006-05-24 17:41:04 +09002976 if (FAILED( res ))
Aric Stewart9adacf62005-06-24 11:58:21 +00002977 {
Mike McCormack20c57462006-05-24 17:41:04 +09002978 ERR("CLSID_ShellLink not available\n");
2979 goto err;
Aric Stewart9adacf62005-06-24 11:58:21 +00002980 }
2981
2982 res = IShellLinkW_QueryInterface( sl, &IID_IPersistFile,(LPVOID*) &pf );
Mike McCormack20c57462006-05-24 17:41:04 +09002983 if (FAILED( res ))
Aric Stewart9adacf62005-06-24 11:58:21 +00002984 {
Mike McCormack20c57462006-05-24 17:41:04 +09002985 ERR("QueryInterface(IID_IPersistFile) failed\n");
2986 goto err;
Aric Stewart9adacf62005-06-24 11:58:21 +00002987 }
2988
2989 buffer = MSI_RecordGetString(row,2);
2990 target_folder = resolve_folder(package, buffer,FALSE,FALSE,NULL);
2991
2992 /* may be needed because of a bug somehwere else */
2993 create_full_pathW(target_folder);
2994
Mike McCormack477bce32006-01-16 20:38:28 +01002995 filename = msi_dup_record_field( row, 3 );
Aric Stewart9adacf62005-06-24 11:58:21 +00002996 reduce_to_longfilename(filename);
Robert Shearman4ac85672006-02-22 16:31:00 +00002997
2998 extension = strchrW(filename,'.');
2999 if (!extension || strcmpiW(extension,szlnk))
3000 {
3001 int len = strlenW(filename);
3002 filename = msi_realloc(filename, len * sizeof(WCHAR) + sizeof(szlnk));
3003 memcpy(filename + len, szlnk, sizeof(szlnk));
3004 }
Aric Stewart9adacf62005-06-24 11:58:21 +00003005 target_file = build_directory_name(2, target_folder, filename);
Mike McCormackee034ba2005-09-20 11:59:14 +00003006 msi_free(target_folder);
Mike McCormack477bce32006-01-16 20:38:28 +01003007 msi_free(filename);
Aric Stewart9adacf62005-06-24 11:58:21 +00003008
3009 buffer = MSI_RecordGetString(row,5);
3010 if (strchrW(buffer,'['))
3011 {
3012 LPWSTR deformated;
3013 deformat_string(package,buffer,&deformated);
3014 IShellLinkW_SetPath(sl,deformated);
Mike McCormackee034ba2005-09-20 11:59:14 +00003015 msi_free(deformated);
Aric Stewart9adacf62005-06-24 11:58:21 +00003016 }
3017 else
3018 {
Aric Stewart9adacf62005-06-24 11:58:21 +00003019 FIXME("poorly handled shortcut format, advertised shortcut\n");
Mike McCormack566c69e2005-09-22 10:49:17 +00003020 IShellLinkW_SetPath(sl,comp->FullKeypath);
Aric Stewart9adacf62005-06-24 11:58:21 +00003021 }
3022
3023 if (!MSI_RecordIsNull(row,6))
3024 {
3025 LPWSTR deformated;
3026 buffer = MSI_RecordGetString(row,6);
3027 deformat_string(package,buffer,&deformated);
3028 IShellLinkW_SetArguments(sl,deformated);
Mike McCormackee034ba2005-09-20 11:59:14 +00003029 msi_free(deformated);
Aric Stewart9adacf62005-06-24 11:58:21 +00003030 }
3031
3032 if (!MSI_RecordIsNull(row,7))
3033 {
3034 buffer = MSI_RecordGetString(row,7);
3035 IShellLinkW_SetDescription(sl,buffer);
3036 }
3037
3038 if (!MSI_RecordIsNull(row,8))
3039 IShellLinkW_SetHotkey(sl,MSI_RecordGetInteger(row,8));
3040
3041 if (!MSI_RecordIsNull(row,9))
3042 {
Mike McCormack75658d72005-09-22 10:33:57 +00003043 LPWSTR Path;
Aric Stewart9adacf62005-06-24 11:58:21 +00003044 INT index;
3045
3046 buffer = MSI_RecordGetString(row,9);
3047
Mike McCormack75658d72005-09-22 10:33:57 +00003048 Path = build_icon_path(package,buffer);
Aric Stewart9adacf62005-06-24 11:58:21 +00003049 index = MSI_RecordGetInteger(row,10);
3050
Robert Shearmanab378802006-08-03 20:24:10 +01003051 /* no value means 0 */
3052 if (index == MSI_NULL_INTEGER)
3053 index = 0;
3054
Aric Stewart9adacf62005-06-24 11:58:21 +00003055 IShellLinkW_SetIconLocation(sl,Path,index);
Mike McCormackee034ba2005-09-20 11:59:14 +00003056 msi_free(Path);
Aric Stewart9adacf62005-06-24 11:58:21 +00003057 }
3058
3059 if (!MSI_RecordIsNull(row,11))
3060 IShellLinkW_SetShowCmd(sl,MSI_RecordGetInteger(row,11));
3061
3062 if (!MSI_RecordIsNull(row,12))
3063 {
3064 LPWSTR Path;
3065 buffer = MSI_RecordGetString(row,12);
3066 Path = resolve_folder(package, buffer, FALSE, FALSE, NULL);
Mike McCormack43f7f3e2006-07-27 23:18:15 +09003067 if (Path)
3068 IShellLinkW_SetWorkingDirectory(sl,Path);
Mike McCormackee034ba2005-09-20 11:59:14 +00003069 msi_free(Path);
Aric Stewart9adacf62005-06-24 11:58:21 +00003070 }
3071
3072 TRACE("Writing shortcut to %s\n",debugstr_w(target_file));
3073 IPersistFile_Save(pf,target_file,FALSE);
3074
Mike McCormackee034ba2005-09-20 11:59:14 +00003075 msi_free(target_file);
Aric Stewart9adacf62005-06-24 11:58:21 +00003076
Mike McCormack20c57462006-05-24 17:41:04 +09003077err:
3078 if (pf)
3079 IPersistFile_Release( pf );
3080 if (sl)
3081 IShellLinkW_Release( sl );
Aric Stewart9adacf62005-06-24 11:58:21 +00003082
3083 return ERROR_SUCCESS;
3084}
3085
3086static UINT ACTION_CreateShortcuts(MSIPACKAGE *package)
3087{
3088 UINT rc;
3089 HRESULT res;
3090 MSIQUERY * view;
3091 static const WCHAR Query[] =
3092 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
3093 '`','S','h','o','r','t','c','u','t','`',0};
3094
Aric Stewart9adacf62005-06-24 11:58:21 +00003095 rc = MSI_DatabaseOpenViewW(package->db, Query, &view);
3096 if (rc != ERROR_SUCCESS)
3097 return ERROR_SUCCESS;
3098
Aric Stewart2cf222f2004-07-06 19:00:23 +00003099 res = CoInitialize( NULL );
3100 if (FAILED (res))
3101 {
3102 ERR("CoInitialize failed\n");
3103 return ERROR_FUNCTION_FAILED;
3104 }
3105
Aric Stewart9adacf62005-06-24 11:58:21 +00003106 rc = MSI_IterateRecords(view, NULL, ITERATE_CreateShortcuts, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00003107 msiobj_release(&view->hdr);
Aric Stewart2cf222f2004-07-06 19:00:23 +00003108
Aric Stewart2cf222f2004-07-06 19:00:23 +00003109 CoUninitialize();
3110
3111 return rc;
3112}
3113
Aric Stewart916ef942005-06-22 18:42:19 +00003114static UINT ITERATE_PublishProduct(MSIRECORD *row, LPVOID param)
3115{
3116 MSIPACKAGE* package = (MSIPACKAGE*)param;
3117 HANDLE the_file;
Mike McCormack75658d72005-09-22 10:33:57 +00003118 LPWSTR FilePath;
3119 LPCWSTR FileName;
Aric Stewart916ef942005-06-22 18:42:19 +00003120 CHAR buffer[1024];
3121 DWORD sz;
3122 UINT rc;
Robert Shearmand2e48e02006-01-23 17:29:50 +01003123 MSIRECORD *uirow;
Aric Stewart916ef942005-06-22 18:42:19 +00003124
3125 FileName = MSI_RecordGetString(row,1);
3126 if (!FileName)
3127 {
3128 ERR("Unable to get FileName\n");
3129 return ERROR_SUCCESS;
3130 }
3131
Mike McCormack75658d72005-09-22 10:33:57 +00003132 FilePath = build_icon_path(package,FileName);
Aric Stewart916ef942005-06-22 18:42:19 +00003133
3134 TRACE("Creating icon file at %s\n",debugstr_w(FilePath));
3135
3136 the_file = CreateFileW(FilePath, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS,
3137 FILE_ATTRIBUTE_NORMAL, NULL);
3138
3139 if (the_file == INVALID_HANDLE_VALUE)
3140 {
3141 ERR("Unable to create file %s\n",debugstr_w(FilePath));
Mike McCormackee034ba2005-09-20 11:59:14 +00003142 msi_free(FilePath);
Aric Stewart916ef942005-06-22 18:42:19 +00003143 return ERROR_SUCCESS;
3144 }
3145
3146 do
3147 {
3148 DWORD write;
3149 sz = 1024;
3150 rc = MSI_RecordReadStream(row,2,buffer,&sz);
3151 if (rc != ERROR_SUCCESS)
3152 {
3153 ERR("Failed to get stream\n");
3154 CloseHandle(the_file);
3155 DeleteFileW(FilePath);
3156 break;
3157 }
3158 WriteFile(the_file,buffer,sz,&write,NULL);
3159 } while (sz == 1024);
3160
Mike McCormackee034ba2005-09-20 11:59:14 +00003161 msi_free(FilePath);
Aric Stewart916ef942005-06-22 18:42:19 +00003162
3163 CloseHandle(the_file);
Robert Shearmand2e48e02006-01-23 17:29:50 +01003164
3165 uirow = MSI_CreateRecord(1);
3166 MSI_RecordSetStringW(uirow,1,FileName);
3167 ui_actiondata(package,szPublishProduct,uirow);
3168 msiobj_release( &uirow->hdr );
3169
Aric Stewart916ef942005-06-22 18:42:19 +00003170 return ERROR_SUCCESS;
3171}
Aric Stewart2cf222f2004-07-06 19:00:23 +00003172
3173/*
3174 * 99% of the work done here is only done for
3175 * advertised installs. However this is where the
3176 * Icon table is processed and written out
Francois Gouget817c5202004-07-16 19:15:40 +00003177 * so that is what I am going to do here.
Aric Stewart2cf222f2004-07-06 19:00:23 +00003178 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00003179static UINT ACTION_PublishProduct(MSIPACKAGE *package)
Aric Stewart2cf222f2004-07-06 19:00:23 +00003180{
3181 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00003182 MSIQUERY * view;
Aric Stewart8e233e92005-03-01 11:45:19 +00003183 static const WCHAR Query[]=
3184 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00003185 '`','I','c','o','n','`',0};
Aric Stewart6269f002005-01-17 13:40:39 +00003186 /* for registry stuff */
Aric Stewart68b07492005-01-25 11:05:37 +00003187 HKEY hkey=0;
3188 HKEY hukey=0;
Aric Stewart6957e4a2005-06-08 19:16:45 +00003189 static const WCHAR szProductLanguage[] =
3190 {'P','r','o','d','u','c','t','L','a','n','g','u','a','g','e',0};
Aric Stewart6957e4a2005-06-08 19:16:45 +00003191 static const WCHAR szARPProductIcon[] =
3192 {'A','R','P','P','R','O','D','U','C','T','I','C','O','N',0};
Aric Stewartc28bb542005-06-09 15:49:11 +00003193 static const WCHAR szProductVersion[] =
3194 {'P','r','o','d','u','c','t','V','e','r','s','i','o','n',0};
Aric Stewart6957e4a2005-06-08 19:16:45 +00003195 DWORD langid;
Aric Stewart6269f002005-01-17 13:40:39 +00003196 LPWSTR buffer;
3197 DWORD size;
Aric Stewart68b07492005-01-25 11:05:37 +00003198 MSIHANDLE hDb, hSumInfo;
Aric Stewart2cf222f2004-07-06 19:00:23 +00003199
Aric Stewart916ef942005-06-22 18:42:19 +00003200 /* write out icon files */
3201
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00003202 rc = MSI_DatabaseOpenViewW(package->db, Query, &view);
Aric Stewart916ef942005-06-22 18:42:19 +00003203 if (rc == ERROR_SUCCESS)
Aric Stewart2cf222f2004-07-06 19:00:23 +00003204 {
Aric Stewart916ef942005-06-22 18:42:19 +00003205 MSI_IterateRecords(view, NULL, ITERATE_PublishProduct, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00003206 msiobj_release(&view->hdr);
Aric Stewart2cf222f2004-07-06 19:00:23 +00003207 }
3208
Francois Gougetda8b3dd2005-01-26 21:09:04 +00003209 /* ok there is a lot more done here but i need to figure out what */
Aric Stewart916ef942005-06-22 18:42:19 +00003210
Aric Stewartadaef112005-07-07 20:27:06 +00003211 rc = MSIREG_OpenProductsKey(package->ProductCode,&hkey,TRUE);
Aric Stewart6269f002005-01-17 13:40:39 +00003212 if (rc != ERROR_SUCCESS)
3213 goto end;
3214
Aric Stewartadaef112005-07-07 20:27:06 +00003215 rc = MSIREG_OpenUserProductsKey(package->ProductCode,&hukey,TRUE);
Aric Stewart6269f002005-01-17 13:40:39 +00003216 if (rc != ERROR_SUCCESS)
3217 goto end;
3218
Aric Stewart6269f002005-01-17 13:40:39 +00003219
Mike McCormack062ad502005-09-15 15:04:08 +00003220 buffer = msi_dup_property( package, INSTALLPROPERTY_PRODUCTNAMEW );
Mike McCormack4db02cd2005-09-15 14:58:38 +00003221 msi_reg_set_val_str( hukey, INSTALLPROPERTY_PRODUCTNAMEW, buffer );
Mike McCormackee034ba2005-09-20 11:59:14 +00003222 msi_free(buffer);
Aric Stewart6957e4a2005-06-08 19:16:45 +00003223
Mike McCormack74f0de92005-09-29 10:32:39 +00003224 langid = msi_get_property_int( package, szProductLanguage, 0 );
Mike McCormack4db02cd2005-09-15 14:58:38 +00003225 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_LANGUAGEW, langid );
Aric Stewart6957e4a2005-06-08 19:16:45 +00003226
Mike McCormack062ad502005-09-15 15:04:08 +00003227 buffer = msi_dup_property( package, szARPProductIcon );
Aric Stewart6957e4a2005-06-08 19:16:45 +00003228 if (buffer)
3229 {
Mike McCormack75658d72005-09-22 10:33:57 +00003230 LPWSTR path = build_icon_path(package,buffer);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003231 msi_reg_set_val_str( hukey, INSTALLPROPERTY_PRODUCTICONW, path );
Mike McCormack75658d72005-09-22 10:33:57 +00003232 msi_free( path );
Aric Stewart6957e4a2005-06-08 19:16:45 +00003233 }
Mike McCormackee034ba2005-09-20 11:59:14 +00003234 msi_free(buffer);
Aric Stewartc28bb542005-06-09 15:49:11 +00003235
Mike McCormack062ad502005-09-15 15:04:08 +00003236 buffer = msi_dup_property( package, szProductVersion );
Aric Stewartc28bb542005-06-09 15:49:11 +00003237 if (buffer)
3238 {
Mike McCormack230af9d2006-07-14 15:19:08 +09003239 DWORD verdword = msi_version_str_to_dword(buffer);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003240 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_VERSIONW, verdword );
Aric Stewartc28bb542005-06-09 15:49:11 +00003241 }
Mike McCormackee034ba2005-09-20 11:59:14 +00003242 msi_free(buffer);
Aric Stewart6957e4a2005-06-08 19:16:45 +00003243
Mike McCormackb7270b82005-12-31 13:18:11 +01003244 /* FIXME: Need to write more keys to the user registry */
Aric Stewart68b07492005-01-25 11:05:37 +00003245
Aric Stewart7e7b8cf2005-02-18 20:00:34 +00003246 hDb= alloc_msihandle( &package->db->hdr );
Dan Kegel337e1e22006-08-28 09:44:35 -07003247 if (!hDb) {
3248 rc = ERROR_NOT_ENOUGH_MEMORY;
3249 goto end;
3250 }
Aric Stewart68b07492005-01-25 11:05:37 +00003251 rc = MsiGetSummaryInformationW(hDb, NULL, 0, &hSumInfo);
Aric Stewart7e7b8cf2005-02-18 20:00:34 +00003252 MsiCloseHandle(hDb);
Aric Stewart68b07492005-01-25 11:05:37 +00003253 if (rc == ERROR_SUCCESS)
3254 {
3255 WCHAR guidbuffer[0x200];
3256 size = 0x200;
Aric Stewart7e7b8cf2005-02-18 20:00:34 +00003257 rc = MsiSummaryInfoGetPropertyW(hSumInfo, 9, NULL, NULL, NULL,
Aric Stewart68b07492005-01-25 11:05:37 +00003258 guidbuffer, &size);
3259 if (rc == ERROR_SUCCESS)
3260 {
3261 WCHAR squashed[GUID_SIZE];
3262 /* for now we only care about the first guid */
3263 LPWSTR ptr = strchrW(guidbuffer,';');
3264 if (ptr) *ptr = 0;
3265 squash_guid(guidbuffer,squashed);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003266 msi_reg_set_val_str( hukey, INSTALLPROPERTY_PACKAGECODEW, squashed );
Aric Stewart68b07492005-01-25 11:05:37 +00003267 }
3268 else
3269 {
Francois Gouget0edbaf72005-11-10 12:14:56 +00003270 ERR("Unable to query Revision_Number...\n");
Aric Stewart68b07492005-01-25 11:05:37 +00003271 rc = ERROR_SUCCESS;
3272 }
3273 MsiCloseHandle(hSumInfo);
3274 }
3275 else
3276 {
3277 ERR("Unable to open Summary Information\n");
3278 rc = ERROR_SUCCESS;
3279 }
Aric Stewart2cae30b2005-01-19 19:07:40 +00003280
Aric Stewart6269f002005-01-17 13:40:39 +00003281end:
3282
Aric Stewart6269f002005-01-17 13:40:39 +00003283 RegCloseKey(hkey);
Aric Stewart6269f002005-01-17 13:40:39 +00003284 RegCloseKey(hukey);
3285
Aric Stewart2cf222f2004-07-06 19:00:23 +00003286 return rc;
Aric Stewart2cf222f2004-07-06 19:00:23 +00003287}
3288
Aric Stewartaded32f2005-06-23 09:46:31 +00003289static UINT ITERATE_WriteIniValues(MSIRECORD *row, LPVOID param)
3290{
3291 MSIPACKAGE *package = (MSIPACKAGE*)param;
3292 LPCWSTR component,section,key,value,identifier,filename,dirproperty;
3293 LPWSTR deformated_section, deformated_key, deformated_value;
3294 LPWSTR folder, fullname = NULL;
3295 MSIRECORD * uirow;
Mike McCormack38d67a42005-08-22 09:15:23 +00003296 INT action;
3297 MSICOMPONENT *comp;
Aric Stewartaded32f2005-06-23 09:46:31 +00003298 static const WCHAR szWindowsFolder[] =
3299 {'W','i','n','d','o','w','s','F','o','l','d','e','r',0};
3300
3301 component = MSI_RecordGetString(row, 8);
Mike McCormack38d67a42005-08-22 09:15:23 +00003302 comp = get_loaded_component(package,component);
Aric Stewartaded32f2005-06-23 09:46:31 +00003303
Mike McCormackd693f462005-10-29 11:36:48 +00003304 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
Aric Stewartaded32f2005-06-23 09:46:31 +00003305 {
3306 TRACE("Skipping ini file due to disabled component %s\n",
3307 debugstr_w(component));
3308
Mike McCormack38d67a42005-08-22 09:15:23 +00003309 comp->Action = comp->Installed;
Aric Stewartaded32f2005-06-23 09:46:31 +00003310
3311 return ERROR_SUCCESS;
3312 }
3313
Mike McCormack38d67a42005-08-22 09:15:23 +00003314 comp->Action = INSTALLSTATE_LOCAL;
Aric Stewartaded32f2005-06-23 09:46:31 +00003315
3316 identifier = MSI_RecordGetString(row,1);
3317 filename = MSI_RecordGetString(row,2);
3318 dirproperty = MSI_RecordGetString(row,3);
3319 section = MSI_RecordGetString(row,4);
3320 key = MSI_RecordGetString(row,5);
3321 value = MSI_RecordGetString(row,6);
3322 action = MSI_RecordGetInteger(row,7);
3323
3324 deformat_string(package,section,&deformated_section);
3325 deformat_string(package,key,&deformated_key);
3326 deformat_string(package,value,&deformated_value);
3327
3328 if (dirproperty)
3329 {
3330 folder = resolve_folder(package, dirproperty, FALSE, FALSE, NULL);
3331 if (!folder)
Mike McCormack062ad502005-09-15 15:04:08 +00003332 folder = msi_dup_property( package, dirproperty );
Aric Stewartaded32f2005-06-23 09:46:31 +00003333 }
3334 else
Mike McCormack062ad502005-09-15 15:04:08 +00003335 folder = msi_dup_property( package, szWindowsFolder );
Aric Stewartaded32f2005-06-23 09:46:31 +00003336
3337 if (!folder)
3338 {
3339 ERR("Unable to resolve folder! (%s)\n",debugstr_w(dirproperty));
3340 goto cleanup;
3341 }
3342
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003343 fullname = build_directory_name(2, folder, filename);
Aric Stewartaded32f2005-06-23 09:46:31 +00003344
3345 if (action == 0)
3346 {
3347 TRACE("Adding value %s to section %s in %s\n",
3348 debugstr_w(deformated_key), debugstr_w(deformated_section),
3349 debugstr_w(fullname));
3350 WritePrivateProfileStringW(deformated_section, deformated_key,
3351 deformated_value, fullname);
3352 }
3353 else if (action == 1)
3354 {
3355 WCHAR returned[10];
3356 GetPrivateProfileStringW(deformated_section, deformated_key, NULL,
3357 returned, 10, fullname);
3358 if (returned[0] == 0)
3359 {
3360 TRACE("Adding value %s to section %s in %s\n",
3361 debugstr_w(deformated_key), debugstr_w(deformated_section),
3362 debugstr_w(fullname));
3363
3364 WritePrivateProfileStringW(deformated_section, deformated_key,
3365 deformated_value, fullname);
3366 }
3367 }
3368 else if (action == 3)
3369 FIXME("Append to existing section not yet implemented\n");
3370
3371 uirow = MSI_CreateRecord(4);
3372 MSI_RecordSetStringW(uirow,1,identifier);
3373 MSI_RecordSetStringW(uirow,2,deformated_section);
3374 MSI_RecordSetStringW(uirow,3,deformated_key);
3375 MSI_RecordSetStringW(uirow,4,deformated_value);
3376 ui_actiondata(package,szWriteIniValues,uirow);
3377 msiobj_release( &uirow->hdr );
3378cleanup:
Mike McCormackee034ba2005-09-20 11:59:14 +00003379 msi_free(fullname);
3380 msi_free(folder);
3381 msi_free(deformated_key);
3382 msi_free(deformated_value);
3383 msi_free(deformated_section);
Aric Stewartaded32f2005-06-23 09:46:31 +00003384 return ERROR_SUCCESS;
3385}
3386
Aric Stewart516a9c72005-01-14 15:59:26 +00003387static UINT ACTION_WriteIniValues(MSIPACKAGE *package)
3388{
3389 UINT rc;
3390 MSIQUERY * view;
Aric Stewart8e233e92005-03-01 11:45:19 +00003391 static const WCHAR ExecSeqQuery[] =
3392 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00003393 '`','I','n','i','F','i','l','e','`',0};
Aric Stewart516a9c72005-01-14 15:59:26 +00003394
3395 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
3396 if (rc != ERROR_SUCCESS)
3397 {
3398 TRACE("no IniFile table\n");
3399 return ERROR_SUCCESS;
3400 }
3401
Aric Stewartaded32f2005-06-23 09:46:31 +00003402 rc = MSI_IterateRecords(view, NULL, ITERATE_WriteIniValues, package);
Aric Stewart516a9c72005-01-14 15:59:26 +00003403 msiobj_release(&view->hdr);
3404 return rc;
3405}
3406
Aric Stewart854bfc42005-06-24 11:33:02 +00003407static UINT ITERATE_SelfRegModules(MSIRECORD *row, LPVOID param)
Aric Stewart6269f002005-01-17 13:40:39 +00003408{
Aric Stewart854bfc42005-06-24 11:33:02 +00003409 MSIPACKAGE *package = (MSIPACKAGE*)param;
3410 LPCWSTR filename;
3411 LPWSTR FullName;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003412 MSIFILE *file;
Aric Stewart854bfc42005-06-24 11:33:02 +00003413 DWORD len;
Aric Stewart8e233e92005-03-01 11:45:19 +00003414 static const WCHAR ExeStr[] =
Aric Stewartc5a14432005-05-18 17:46:12 +00003415 {'r','e','g','s','v','r','3','2','.','e','x','e',' ','\"',0};
3416 static const WCHAR close[] = {'\"',0};
Aric Stewart2cae30b2005-01-19 19:07:40 +00003417 STARTUPINFOW si;
3418 PROCESS_INFORMATION info;
3419 BOOL brc;
Robert Shearmand2e48e02006-01-23 17:29:50 +01003420 MSIRECORD *uirow;
3421 LPWSTR uipath, p;
Aric Stewart2cae30b2005-01-19 19:07:40 +00003422
3423 memset(&si,0,sizeof(STARTUPINFOW));
3424
Aric Stewart854bfc42005-06-24 11:33:02 +00003425 filename = MSI_RecordGetString(row,1);
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003426 file = get_loaded_file( package, filename );
Aric Stewart854bfc42005-06-24 11:33:02 +00003427
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003428 if (!file)
Aric Stewart854bfc42005-06-24 11:33:02 +00003429 {
3430 ERR("Unable to find file id %s\n",debugstr_w(filename));
3431 return ERROR_SUCCESS;
3432 }
3433
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003434 len = strlenW(ExeStr) + strlenW( file->TargetPath ) + 2;
Aric Stewart854bfc42005-06-24 11:33:02 +00003435
Mike McCormackee034ba2005-09-20 11:59:14 +00003436 FullName = msi_alloc(len*sizeof(WCHAR));
Aric Stewart854bfc42005-06-24 11:33:02 +00003437 strcpyW(FullName,ExeStr);
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003438 strcatW( FullName, file->TargetPath );
Aric Stewart854bfc42005-06-24 11:33:02 +00003439 strcatW(FullName,close);
3440
3441 TRACE("Registering %s\n",debugstr_w(FullName));
3442 brc = CreateProcessW(NULL, FullName, NULL, NULL, FALSE, 0, NULL, c_colon,
3443 &si, &info);
3444
3445 if (brc)
3446 msi_dialog_check_messages(info.hProcess);
3447
Mike McCormackee034ba2005-09-20 11:59:14 +00003448 msi_free(FullName);
Robert Shearmand2e48e02006-01-23 17:29:50 +01003449
3450 /* the UI chunk */
3451 uirow = MSI_CreateRecord( 2 );
3452 uipath = strdupW( file->TargetPath );
3453 p = strrchrW(uipath,'\\');
3454 if (p)
3455 p[1]=0;
3456 MSI_RecordSetStringW( uirow, 1, &p[2] );
3457 MSI_RecordSetStringW( uirow, 2, uipath);
3458 ui_actiondata( package, szSelfRegModules, uirow);
3459 msiobj_release( &uirow->hdr );
3460 msi_free( uipath );
3461 /* FIXME: call ui_progress? */
3462
Aric Stewart854bfc42005-06-24 11:33:02 +00003463 return ERROR_SUCCESS;
3464}
3465
3466static UINT ACTION_SelfRegModules(MSIPACKAGE *package)
3467{
3468 UINT rc;
3469 MSIQUERY * view;
3470 static const WCHAR ExecSeqQuery[] =
3471 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
3472 '`','S','e','l','f','R','e','g','`',0};
3473
Aric Stewart6269f002005-01-17 13:40:39 +00003474 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
3475 if (rc != ERROR_SUCCESS)
3476 {
3477 TRACE("no SelfReg table\n");
3478 return ERROR_SUCCESS;
3479 }
3480
Aric Stewart854bfc42005-06-24 11:33:02 +00003481 MSI_IterateRecords(view, NULL, ITERATE_SelfRegModules, package);
Aric Stewart6269f002005-01-17 13:40:39 +00003482 msiobj_release(&view->hdr);
Aric Stewart854bfc42005-06-24 11:33:02 +00003483
3484 return ERROR_SUCCESS;
Aric Stewart6269f002005-01-17 13:40:39 +00003485}
3486
3487static UINT ACTION_PublishFeatures(MSIPACKAGE *package)
3488{
Mike McCormack1da28582005-08-22 14:09:17 +00003489 MSIFEATURE *feature;
Aric Stewart6269f002005-01-17 13:40:39 +00003490 UINT rc;
Aric Stewart68b07492005-01-25 11:05:37 +00003491 HKEY hkey=0;
3492 HKEY hukey=0;
Aric Stewart68b07492005-01-25 11:05:37 +00003493
Aric Stewartadaef112005-07-07 20:27:06 +00003494 rc = MSIREG_OpenFeaturesKey(package->ProductCode,&hkey,TRUE);
Aric Stewart6269f002005-01-17 13:40:39 +00003495 if (rc != ERROR_SUCCESS)
3496 goto end;
3497
Aric Stewartadaef112005-07-07 20:27:06 +00003498 rc = MSIREG_OpenUserFeaturesKey(package->ProductCode,&hukey,TRUE);
Aric Stewart6269f002005-01-17 13:40:39 +00003499 if (rc != ERROR_SUCCESS)
3500 goto end;
3501
3502 /* here the guids are base 85 encoded */
Mike McCormack1da28582005-08-22 14:09:17 +00003503 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewart6269f002005-01-17 13:40:39 +00003504 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003505 ComponentList *cl;
Aric Stewart6269f002005-01-17 13:40:39 +00003506 LPWSTR data = NULL;
3507 GUID clsid;
Aric Stewart6269f002005-01-17 13:40:39 +00003508 INT size;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003509 BOOL absent = FALSE;
Robert Shearmand2e48e02006-01-23 17:29:50 +01003510 MSIRECORD *uirow;
Aric Stewart6269f002005-01-17 13:40:39 +00003511
Mike McCormack1da28582005-08-22 14:09:17 +00003512 if (!ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_LOCAL ) &&
3513 !ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_SOURCE ) &&
3514 !ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_ADVERTISED ))
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003515 absent = TRUE;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00003516
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003517 size = 1;
Mike McCormack1da28582005-08-22 14:09:17 +00003518 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003519 {
3520 size += 21;
3521 }
Mike McCormack79ca56c2005-09-13 10:37:37 +00003522 if (feature->Feature_Parent)
Mike McCormack1da28582005-08-22 14:09:17 +00003523 size += strlenW( feature->Feature_Parent )+2;
Aric Stewart6269f002005-01-17 13:40:39 +00003524
Mike McCormackee034ba2005-09-20 11:59:14 +00003525 data = msi_alloc(size * sizeof(WCHAR));
Aric Stewart6269f002005-01-17 13:40:39 +00003526
3527 data[0] = 0;
Mike McCormack1da28582005-08-22 14:09:17 +00003528 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewart6269f002005-01-17 13:40:39 +00003529 {
Mike McCormack38d67a42005-08-22 09:15:23 +00003530 MSICOMPONENT* component = cl->component;
Aric Stewart6269f002005-01-17 13:40:39 +00003531 WCHAR buf[21];
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003532
Mike McCormack3a940112006-04-19 02:29:03 +09003533 buf[0] = 0;
Mike McCormackefcc1ec2005-09-12 12:07:15 +00003534 if (component->ComponentId)
Aric Stewartc5a14432005-05-18 17:46:12 +00003535 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003536 TRACE("From %s\n",debugstr_w(component->ComponentId));
3537 CLSIDFromString(component->ComponentId, &clsid);
Aric Stewartc5a14432005-05-18 17:46:12 +00003538 encode_base85_guid(&clsid,buf);
3539 TRACE("to %s\n",debugstr_w(buf));
3540 strcatW(data,buf);
3541 }
Aric Stewart6269f002005-01-17 13:40:39 +00003542 }
Mike McCormack79ca56c2005-09-13 10:37:37 +00003543 if (feature->Feature_Parent)
Aric Stewart6269f002005-01-17 13:40:39 +00003544 {
3545 static const WCHAR sep[] = {'\2',0};
3546 strcatW(data,sep);
Mike McCormack1da28582005-08-22 14:09:17 +00003547 strcatW(data,feature->Feature_Parent);
Aric Stewart6269f002005-01-17 13:40:39 +00003548 }
3549
Mike McCormack4db02cd2005-09-15 14:58:38 +00003550 msi_reg_set_val_str( hkey, feature->Feature, data );
Mike McCormackee034ba2005-09-20 11:59:14 +00003551 msi_free(data);
Aric Stewart6269f002005-01-17 13:40:39 +00003552
Mike McCormack79ca56c2005-09-13 10:37:37 +00003553 size = 0;
3554 if (feature->Feature_Parent)
3555 size = strlenW(feature->Feature_Parent)*sizeof(WCHAR);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003556 if (!absent)
3557 {
Mike McCormack1da28582005-08-22 14:09:17 +00003558 RegSetValueExW(hukey,feature->Feature,0,REG_SZ,
3559 (LPBYTE)feature->Feature_Parent,size);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003560 }
3561 else
3562 {
Mike McCormack79ca56c2005-09-13 10:37:37 +00003563 size += 2*sizeof(WCHAR);
Mike McCormackee034ba2005-09-20 11:59:14 +00003564 data = msi_alloc(size);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003565 data[0] = 0x6;
Mike McCormack79ca56c2005-09-13 10:37:37 +00003566 data[1] = 0;
3567 if (feature->Feature_Parent)
3568 strcpyW( &data[1], feature->Feature_Parent );
Mike McCormack1da28582005-08-22 14:09:17 +00003569 RegSetValueExW(hukey,feature->Feature,0,REG_SZ,
Mike McCormack16466af2005-07-06 10:33:30 +00003570 (LPBYTE)data,size);
Mike McCormackee034ba2005-09-20 11:59:14 +00003571 msi_free(data);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003572 }
Robert Shearmand2e48e02006-01-23 17:29:50 +01003573
3574 /* the UI chunk */
3575 uirow = MSI_CreateRecord( 1 );
3576 MSI_RecordSetStringW( uirow, 1, feature->Feature );
3577 ui_actiondata( package, szPublishFeatures, uirow);
3578 msiobj_release( &uirow->hdr );
3579 /* FIXME: call ui_progress? */
Aric Stewart6269f002005-01-17 13:40:39 +00003580 }
3581
Aric Stewart6269f002005-01-17 13:40:39 +00003582end:
Aric Stewart6269f002005-01-17 13:40:39 +00003583 RegCloseKey(hkey);
3584 RegCloseKey(hukey);
3585 return rc;
3586}
3587
Mike McCormack5f830692006-09-08 16:20:46 +09003588static UINT msi_get_local_package_name( LPWSTR path )
Mike McCormack61f24a42005-09-30 10:32:41 +00003589{
Mike McCormack5f830692006-09-08 16:20:46 +09003590 static const WCHAR szInstaller[] = {
3591 '\\','I','n','s','t','a','l','l','e','r','\\',0};
3592 static const WCHAR fmt[] = { '%','x','.','m','s','i',0};
3593 DWORD time, len, i;
3594 HANDLE handle;
Mike McCormack61f24a42005-09-30 10:32:41 +00003595
Mike McCormack5f830692006-09-08 16:20:46 +09003596 time = GetTickCount();
3597 GetWindowsDirectoryW( path, MAX_PATH );
3598 lstrcatW( path, szInstaller );
3599 CreateDirectoryW( path, NULL );
3600
3601 len = lstrlenW(path);
3602 for (i=0; i<0x10000; i++)
Mike McCormack61f24a42005-09-30 10:32:41 +00003603 {
Mike McCormack5f830692006-09-08 16:20:46 +09003604 snprintfW( &path[len], MAX_PATH - len, fmt, (time+i)&0xffff );
3605 handle = CreateFileW( path, GENERIC_WRITE, 0, NULL,
3606 CREATE_NEW, FILE_ATTRIBUTE_NORMAL, 0 );
Mike McCormack61f24a42005-09-30 10:32:41 +00003607 if (handle != INVALID_HANDLE_VALUE)
3608 {
3609 CloseHandle(handle);
3610 break;
3611 }
3612 if (GetLastError() != ERROR_FILE_EXISTS &&
3613 GetLastError() != ERROR_SHARING_VIOLATION)
Mike McCormack5f830692006-09-08 16:20:46 +09003614 return ERROR_FUNCTION_FAILED;
3615 }
Mike McCormack61f24a42005-09-30 10:32:41 +00003616
Mike McCormack5f830692006-09-08 16:20:46 +09003617 return ERROR_SUCCESS;
3618}
3619
3620static UINT msi_make_package_local( MSIPACKAGE *package, HKEY hkey )
3621{
3622 static const WCHAR szOriginalDatabase[] =
3623 {'O','r','i','g','i','n','a','l','D','a','t','a','b','a','s','e',0};
3624 WCHAR packagefile[MAX_PATH];
3625 LPWSTR msiFilePath;
3626 UINT r;
3627
3628 r = msi_get_local_package_name( packagefile );
3629 if (r != ERROR_SUCCESS)
3630 return r;
Mike McCormack61f24a42005-09-30 10:32:41 +00003631
3632 TRACE("Copying to local package %s\n",debugstr_w(packagefile));
3633
3634 msiFilePath = msi_dup_property( package, szOriginalDatabase );
3635 r = CopyFileW( msiFilePath, packagefile, FALSE);
3636 msi_free( msiFilePath );
3637
3638 if (!r)
3639 {
Mike McCormackf1d46462006-10-05 13:41:22 +09003640 ERR("Unable to copy package (%s -> %s) (error %d)\n",
Mike McCormack61f24a42005-09-30 10:32:41 +00003641 debugstr_w(msiFilePath), debugstr_w(packagefile), GetLastError());
3642 return ERROR_FUNCTION_FAILED;
3643 }
3644
3645 /* FIXME: maybe set this key in ACTION_RegisterProduct instead */
3646 msi_reg_set_val_str( hkey, INSTALLPROPERTY_LOCALPACKAGEW, packagefile );
3647 return ERROR_SUCCESS;
3648}
3649
Mike McCormackba293ee2005-10-27 12:08:16 +00003650static UINT msi_write_uninstall_property_vals( MSIPACKAGE *package, HKEY hkey )
3651{
3652 LPWSTR prop, val, key;
3653 static const LPCSTR propval[] = {
3654 "ARPAUTHORIZEDCDFPREFIX", "AuthorizedCDFPrefix",
3655 "ARPCONTACT", "Contact",
3656 "ARPCOMMENTS", "Comments",
3657 "ProductName", "DisplayName",
3658 "ProductVersion", "DisplayVersion",
3659 "ARPHELPLINK", "HelpLink",
3660 "ARPHELPTELEPHONE", "HelpTelephone",
3661 "ARPINSTALLLOCATION", "InstallLocation",
3662 "SourceDir", "InstallSource",
3663 "Manufacturer", "Publisher",
3664 "ARPREADME", "Readme",
3665 "ARPSIZE", "Size",
3666 "ARPURLINFOABOUT", "URLInfoAbout",
3667 "ARPURLUPDATEINFO", "URLUpdateInfo",
3668 NULL,
3669 };
3670 const LPCSTR *p = propval;
3671
3672 while( *p )
3673 {
3674 prop = strdupAtoW( *p++ );
3675 key = strdupAtoW( *p++ );
3676 val = msi_dup_property( package, prop );
3677 msi_reg_set_val_str( hkey, key, val );
3678 msi_free(val);
3679 msi_free(key);
3680 msi_free(prop);
3681 }
3682 return ERROR_SUCCESS;
3683}
3684
Aric Stewart2cae30b2005-01-19 19:07:40 +00003685static UINT ACTION_RegisterProduct(MSIPACKAGE *package)
3686{
Aric Stewart68b07492005-01-25 11:05:37 +00003687 HKEY hkey=0;
Aric Stewarte9db87b2005-06-17 21:25:41 +00003688 LPWSTR buffer = NULL;
Mike McCormackba293ee2005-10-27 12:08:16 +00003689 UINT rc;
Mike McCormack74f0de92005-09-29 10:32:39 +00003690 DWORD size, langid;
Mike McCormack4db02cd2005-09-15 14:58:38 +00003691 static const WCHAR szWindowsInstaller[] =
Mike McCormackba293ee2005-10-27 12:08:16 +00003692 {'W','i','n','d','o','w','s','I','n','s','t','a','l','l','e','r',0};
Aric Stewart36a01502005-06-08 19:07:52 +00003693 static const WCHAR szUpgradeCode[] =
3694 {'U','p','g','r','a','d','e','C','o','d','e',0};
Aric Stewarte9db87b2005-06-17 21:25:41 +00003695 static const WCHAR modpath_fmt[] =
Mike McCormackba293ee2005-10-27 12:08:16 +00003696 {'M','s','i','E','x','e','c','.','e','x','e',' ',
3697 '/','I','[','P','r','o','d','u','c','t','C','o','d','e',']',0};
Aric Stewarte9db87b2005-06-17 21:25:41 +00003698 static const WCHAR szModifyPath[] =
3699 {'M','o','d','i','f','y','P','a','t','h',0};
3700 static const WCHAR szUninstallString[] =
3701 {'U','n','i','n','s','t','a','l','l','S','t','r','i','n','g',0};
3702 static const WCHAR szEstimatedSize[] =
3703 {'E','s','t','i','m','a','t','e','d','S','i','z','e',0};
Aric Stewarte9db87b2005-06-17 21:25:41 +00003704 static const WCHAR szProductLanguage[] =
3705 {'P','r','o','d','u','c','t','L','a','n','g','u','a','g','e',0};
3706 static const WCHAR szProductVersion[] =
3707 {'P','r','o','d','u','c','t','V','e','r','s','i','o','n',0};
Aric Stewarte9db87b2005-06-17 21:25:41 +00003708
3709 SYSTEMTIME systime;
3710 static const WCHAR date_fmt[] = {'%','i','%','i','%','i',0};
Aric Stewart36a01502005-06-08 19:07:52 +00003711 LPWSTR upgrade_code;
Mike McCormackba293ee2005-10-27 12:08:16 +00003712 WCHAR szDate[9];
Aric Stewart2cae30b2005-01-19 19:07:40 +00003713
Aric Stewartadaef112005-07-07 20:27:06 +00003714 rc = MSIREG_OpenUninstallKey(package->ProductCode,&hkey,TRUE);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003715 if (rc != ERROR_SUCCESS)
Mike McCormackba293ee2005-10-27 12:08:16 +00003716 return rc;
Aric Stewart2cae30b2005-01-19 19:07:40 +00003717
3718 /* dump all the info i can grab */
Mike McCormackb7270b82005-12-31 13:18:11 +01003719 /* FIXME: Flesh out more information */
Aric Stewart2cae30b2005-01-19 19:07:40 +00003720
Mike McCormackba293ee2005-10-27 12:08:16 +00003721 msi_write_uninstall_property_vals( package, hkey );
Aric Stewart2cae30b2005-01-19 19:07:40 +00003722
Mike McCormack4db02cd2005-09-15 14:58:38 +00003723 msi_reg_set_val_dword( hkey, szWindowsInstaller, 1 );
Aric Stewart2cae30b2005-01-19 19:07:40 +00003724
Mike McCormack61f24a42005-09-30 10:32:41 +00003725 msi_make_package_local( package, hkey );
Aric Stewart36a01502005-06-08 19:07:52 +00003726
Aric Stewarte9db87b2005-06-17 21:25:41 +00003727 /* do ModifyPath and UninstallString */
3728 size = deformat_string(package,modpath_fmt,&buffer);
Mike McCormack16466af2005-07-06 10:33:30 +00003729 RegSetValueExW(hkey,szModifyPath,0,REG_EXPAND_SZ,(LPBYTE)buffer,size);
3730 RegSetValueExW(hkey,szUninstallString,0,REG_EXPAND_SZ,(LPBYTE)buffer,size);
Mike McCormackee034ba2005-09-20 11:59:14 +00003731 msi_free(buffer);
Aric Stewarte9db87b2005-06-17 21:25:41 +00003732
Mike McCormackb7270b82005-12-31 13:18:11 +01003733 /* FIXME: Write real Estimated Size when we have it */
Mike McCormack4db02cd2005-09-15 14:58:38 +00003734 msi_reg_set_val_dword( hkey, szEstimatedSize, 0 );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003735
3736 GetLocalTime(&systime);
Mike McCormackba293ee2005-10-27 12:08:16 +00003737 sprintfW(szDate,date_fmt,systime.wYear,systime.wMonth,systime.wDay);
3738 msi_reg_set_val_str( hkey, INSTALLPROPERTY_INSTALLDATEW, szDate );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003739
Mike McCormack74f0de92005-09-29 10:32:39 +00003740 langid = msi_get_property_int( package, szProductLanguage, 0 );
3741 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_LANGUAGEW, langid );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003742
Mike McCormack062ad502005-09-15 15:04:08 +00003743 buffer = msi_dup_property( package, szProductVersion );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003744 if (buffer)
3745 {
Mike McCormack230af9d2006-07-14 15:19:08 +09003746 DWORD verdword = msi_version_str_to_dword(buffer);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003747
3748 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_VERSIONW, verdword );
3749 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_VERSIONMAJORW, verdword>>24 );
3750 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_VERSIONMINORW, (verdword>>16)&0x00FF );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003751 }
Mike McCormackee034ba2005-09-20 11:59:14 +00003752 msi_free(buffer);
Aric Stewarte9db87b2005-06-17 21:25:41 +00003753
Aric Stewart36a01502005-06-08 19:07:52 +00003754 /* Handle Upgrade Codes */
Mike McCormack062ad502005-09-15 15:04:08 +00003755 upgrade_code = msi_dup_property( package, szUpgradeCode );
Aric Stewart36a01502005-06-08 19:07:52 +00003756 if (upgrade_code)
3757 {
3758 HKEY hkey2;
3759 WCHAR squashed[33];
3760 MSIREG_OpenUpgradeCodesKey(upgrade_code, &hkey2, TRUE);
Aric Stewartadaef112005-07-07 20:27:06 +00003761 squash_guid(package->ProductCode,squashed);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003762 msi_reg_set_val_str( hkey2, squashed, NULL );
Aric Stewart36a01502005-06-08 19:07:52 +00003763 RegCloseKey(hkey2);
3764 MSIREG_OpenUserUpgradeCodesKey(upgrade_code, &hkey2, TRUE);
Aric Stewartadaef112005-07-07 20:27:06 +00003765 squash_guid(package->ProductCode,squashed);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003766 msi_reg_set_val_str( hkey2, squashed, NULL );
Aric Stewart36a01502005-06-08 19:07:52 +00003767 RegCloseKey(hkey2);
3768
Mike McCormackee034ba2005-09-20 11:59:14 +00003769 msi_free(upgrade_code);
Aric Stewart36a01502005-06-08 19:07:52 +00003770 }
Aric Stewart2cae30b2005-01-19 19:07:40 +00003771
Aric Stewart2cae30b2005-01-19 19:07:40 +00003772 RegCloseKey(hkey);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003773
Robert Shearmand2e48e02006-01-23 17:29:50 +01003774 /* FIXME: call ui_actiondata */
3775
Aric Stewart2cae30b2005-01-19 19:07:40 +00003776 return ERROR_SUCCESS;
3777}
3778
3779static UINT ACTION_InstallExecute(MSIPACKAGE *package)
3780{
Mike McCormacka977b2c2005-11-03 09:56:29 +00003781 return execute_script(package,INSTALL_SCRIPT);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003782}
3783
3784static UINT ACTION_InstallFinalize(MSIPACKAGE *package)
3785{
Aric Stewart9cd707d2005-05-27 19:24:22 +00003786 UINT rc;
3787
Aric Stewart9cd707d2005-05-27 19:24:22 +00003788 /* turn off scheduleing */
3789 package->script->CurrentlyScripting= FALSE;
3790
Aric Stewart54c67dd2005-01-25 20:17:09 +00003791 /* first do the same as an InstallExecute */
Aric Stewart9cd707d2005-05-27 19:24:22 +00003792 rc = ACTION_InstallExecute(package);
3793 if (rc != ERROR_SUCCESS)
3794 return rc;
Aric Stewart54c67dd2005-01-25 20:17:09 +00003795
3796 /* then handle Commit Actions */
Aric Stewart9cd707d2005-05-27 19:24:22 +00003797 rc = execute_script(package,COMMIT_SCRIPT);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003798
Aric Stewart9cd707d2005-05-27 19:24:22 +00003799 return rc;
Aric Stewart2cae30b2005-01-19 19:07:40 +00003800}
3801
3802static UINT ACTION_ForceReboot(MSIPACKAGE *package)
3803{
3804 static const WCHAR RunOnce[] = {
3805 'S','o','f','t','w','a','r','e','\\',
3806 'M','i','c','r','o','s','o','f','t','\\',
3807 'W','i','n','d','o','w','s','\\',
3808 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
Mike McCormack09b82642005-02-22 19:31:45 +00003809 'R','u','n','O','n','c','e',0};
Aric Stewart2cae30b2005-01-19 19:07:40 +00003810 static const WCHAR InstallRunOnce[] = {
3811 'S','o','f','t','w','a','r','e','\\',
3812 'M','i','c','r','o','s','o','f','t','\\',
3813 'W','i','n','d','o','w','s','\\',
3814 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
3815 'I','n','s','t','a','l','l','e','r','\\',
Mike McCormack09b82642005-02-22 19:31:45 +00003816 'R','u','n','O','n','c','e','E','n','t','r','i','e','s',0};
Aric Stewart2cae30b2005-01-19 19:07:40 +00003817
3818 static const WCHAR msiexec_fmt[] = {
Juan Lang014ad3b2005-03-01 10:41:52 +00003819 '%','s',
Aric Stewart2cae30b2005-01-19 19:07:40 +00003820 '\\','M','s','i','E','x','e','c','.','e','x','e',' ','/','@',' ',
3821 '\"','%','s','\"',0};
3822 static const WCHAR install_fmt[] = {
3823 '/','I',' ','\"','%','s','\"',' ',
3824 'A','F','T','E','R','R','E','B','O','O','T','=','1',' ',
3825 'R','U','N','O','N','C','E','E','N','T','R','Y','=','\"','%','s','\"',0};
Juan Lang014ad3b2005-03-01 10:41:52 +00003826 WCHAR buffer[256], sysdir[MAX_PATH];
Aric Stewartadaef112005-07-07 20:27:06 +00003827 HKEY hkey;
Mike McCormack4db02cd2005-09-15 14:58:38 +00003828 WCHAR squished_pc[100];
Aric Stewart2cae30b2005-01-19 19:07:40 +00003829
Aric Stewartadaef112005-07-07 20:27:06 +00003830 squash_guid(package->ProductCode,squished_pc);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003831
Juan Lang014ad3b2005-03-01 10:41:52 +00003832 GetSystemDirectoryW(sysdir, sizeof(sysdir)/sizeof(sysdir[0]));
Aric Stewart2cae30b2005-01-19 19:07:40 +00003833 RegCreateKeyW(HKEY_LOCAL_MACHINE,RunOnce,&hkey);
Juan Lang014ad3b2005-03-01 10:41:52 +00003834 snprintfW(buffer,sizeof(buffer)/sizeof(buffer[0]),msiexec_fmt,sysdir,
3835 squished_pc);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003836
Mike McCormack4db02cd2005-09-15 14:58:38 +00003837 msi_reg_set_val_str( hkey, squished_pc, buffer );
Aric Stewart2cae30b2005-01-19 19:07:40 +00003838 RegCloseKey(hkey);
3839
3840 TRACE("Reboot command %s\n",debugstr_w(buffer));
3841
3842 RegCreateKeyW(HKEY_LOCAL_MACHINE,InstallRunOnce,&hkey);
Aric Stewartadaef112005-07-07 20:27:06 +00003843 sprintfW(buffer,install_fmt,package->ProductCode,squished_pc);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00003844
Mike McCormack4db02cd2005-09-15 14:58:38 +00003845 msi_reg_set_val_str( hkey, squished_pc, buffer );
Aric Stewart2cae30b2005-01-19 19:07:40 +00003846 RegCloseKey(hkey);
3847
Aric Stewart68b07492005-01-25 11:05:37 +00003848 return ERROR_INSTALL_SUSPEND;
Aric Stewart2cae30b2005-01-19 19:07:40 +00003849}
3850
James Hawkins563a50a2006-10-09 00:05:04 -07003851static UINT ACTION_ResolveSource(MSIPACKAGE* package)
Aric Stewart90c57392005-01-31 16:23:12 +00003852{
James Hawkinsc5075432006-10-10 13:39:50 -07003853 DWORD attrib, len;
3854 LPWSTR ptr, source;
Aric Stewart94d68182005-08-15 20:50:06 +00003855 UINT rc;
James Hawkinsc5075432006-10-10 13:39:50 -07003856
Aric Stewart90c57392005-01-31 16:23:12 +00003857 /*
3858 * we are currently doing what should be done here in the top level Install
3859 * however for Adminastrative and uninstalls this step will be needed
3860 */
Aric Stewart94d68182005-08-15 20:50:06 +00003861 if (!package->PackagePath)
3862 return ERROR_SUCCESS;
3863
James Hawkinsc5075432006-10-10 13:39:50 -07003864 ptr = strrchrW(package->PackagePath, '\\');
3865 if (!ptr)
3866 return ERROR_SUCCESS;
3867
3868 len = ptr - package->PackagePath + 2;
3869 source = msi_alloc(len * sizeof(WCHAR));
3870 lstrcpynW(source, package->PackagePath, len);
3871
3872 MSI_SetPropertyW(package, cszSourceDir, source);
3873 MSI_SetPropertyW(package, cszSOURCEDIR, source);
3874
3875 msi_free(source);
3876
Aric Stewart94d68182005-08-15 20:50:06 +00003877 attrib = GetFileAttributesW(package->PackagePath);
3878 if (attrib == INVALID_FILE_ATTRIBUTES)
3879 {
3880 LPWSTR prompt;
3881 LPWSTR msg;
3882 DWORD size = 0;
3883
3884 rc = MsiSourceListGetInfoW(package->ProductCode, NULL,
3885 MSIINSTALLCONTEXT_USERMANAGED, MSICODE_PRODUCT,
3886 INSTALLPROPERTY_DISKPROMPTW,NULL,&size);
3887 if (rc == ERROR_MORE_DATA)
3888 {
Mike McCormackee034ba2005-09-20 11:59:14 +00003889 prompt = msi_alloc(size * sizeof(WCHAR));
Aric Stewart94d68182005-08-15 20:50:06 +00003890 MsiSourceListGetInfoW(package->ProductCode, NULL,
3891 MSIINSTALLCONTEXT_USERMANAGED, MSICODE_PRODUCT,
3892 INSTALLPROPERTY_DISKPROMPTW,prompt,&size);
3893 }
3894 else
3895 prompt = strdupW(package->PackagePath);
3896
3897 msg = generate_error_string(package,1302,1,prompt);
3898 while(attrib == INVALID_FILE_ATTRIBUTES)
3899 {
3900 rc = MessageBoxW(NULL,msg,NULL,MB_OKCANCEL);
3901 if (rc == IDCANCEL)
3902 {
3903 rc = ERROR_INSTALL_USEREXIT;
3904 break;
3905 }
3906 attrib = GetFileAttributesW(package->PackagePath);
3907 }
Mike McCormackee034ba2005-09-20 11:59:14 +00003908 msi_free(prompt);
Aric Stewart94d68182005-08-15 20:50:06 +00003909 rc = ERROR_SUCCESS;
3910 }
3911 else
3912 return ERROR_SUCCESS;
3913
3914 return rc;
Aric Stewart90c57392005-01-31 16:23:12 +00003915}
3916
Aric Stewartc7e88e02005-02-10 17:09:44 +00003917static UINT ACTION_RegisterUser(MSIPACKAGE *package)
3918{
Aric Stewartc7e88e02005-02-10 17:09:44 +00003919 HKEY hkey=0;
3920 LPWSTR buffer;
Aric Stewartc7e88e02005-02-10 17:09:44 +00003921 LPWSTR productid;
3922 UINT rc,i;
Aric Stewartc7e88e02005-02-10 17:09:44 +00003923
3924 static const WCHAR szPropKeys[][80] =
3925 {
Aric Stewart8e233e92005-03-01 11:45:19 +00003926 {'P','r','o','d','u','c','t','I','D',0},
3927 {'U','S','E','R','N','A','M','E',0},
3928 {'C','O','M','P','A','N','Y','N','A','M','E',0},
3929 {0},
Aric Stewartc7e88e02005-02-10 17:09:44 +00003930 };
3931
3932 static const WCHAR szRegKeys[][80] =
3933 {
Aric Stewart8e233e92005-03-01 11:45:19 +00003934 {'P','r','o','d','u','c','t','I','D',0},
3935 {'R','e','g','O','w','n','e','r',0},
3936 {'R','e','g','C','o','m','p','a','n','y',0},
3937 {0},
Aric Stewartc7e88e02005-02-10 17:09:44 +00003938 };
3939
Mike McCormack062ad502005-09-15 15:04:08 +00003940 productid = msi_dup_property( package, INSTALLPROPERTY_PRODUCTIDW );
Aric Stewartc7e88e02005-02-10 17:09:44 +00003941 if (!productid)
3942 return ERROR_SUCCESS;
3943
Aric Stewartadaef112005-07-07 20:27:06 +00003944 rc = MSIREG_OpenUninstallKey(package->ProductCode,&hkey,TRUE);
Aric Stewartc7e88e02005-02-10 17:09:44 +00003945 if (rc != ERROR_SUCCESS)
3946 goto end;
3947
Mike McCormack67189f92005-09-16 18:45:19 +00003948 for( i = 0; szPropKeys[i][0]; i++ )
Aric Stewartc7e88e02005-02-10 17:09:44 +00003949 {
Mike McCormack062ad502005-09-15 15:04:08 +00003950 buffer = msi_dup_property( package, szPropKeys[i] );
Mike McCormack4db02cd2005-09-15 14:58:38 +00003951 msi_reg_set_val_str( hkey, szRegKeys[i], buffer );
Mike McCormackee034ba2005-09-20 11:59:14 +00003952 msi_free( buffer );
Aric Stewartc7e88e02005-02-10 17:09:44 +00003953 }
3954
3955end:
Mike McCormackee034ba2005-09-20 11:59:14 +00003956 msi_free(productid);
Aric Stewartc7e88e02005-02-10 17:09:44 +00003957 RegCloseKey(hkey);
3958
Robert Shearmand2e48e02006-01-23 17:29:50 +01003959 /* FIXME: call ui_actiondata */
3960
Aric Stewartc7e88e02005-02-10 17:09:44 +00003961 return ERROR_SUCCESS;
3962}
3963
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00003964
3965static UINT ACTION_ExecuteAction(MSIPACKAGE *package)
3966{
3967 UINT rc;
Aric Stewart25f1e752005-06-24 12:14:52 +00003968
Aric Stewartc9802932005-06-30 20:45:43 +00003969 package->script->InWhatSequence |= SEQUENCE_EXEC;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003970 rc = ACTION_ProcessExecSequence(package,FALSE);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00003971 return rc;
3972}
3973
Aric Stewart0af24872005-02-25 14:00:09 +00003974
Aric Stewart072c5e52005-04-20 12:50:05 +00003975static UINT ITERATE_PublishComponent(MSIRECORD *rec, LPVOID param)
3976{
3977 MSIPACKAGE *package = (MSIPACKAGE*)param;
Aric Stewart09b0aba2005-06-09 20:30:59 +00003978 LPCWSTR compgroupid=NULL;
3979 LPCWSTR feature=NULL;
3980 LPCWSTR text = NULL;
3981 LPCWSTR qualifier = NULL;
3982 LPCWSTR component = NULL;
Aric Stewart6f43c182005-05-26 12:24:28 +00003983 LPWSTR advertise = NULL;
3984 LPWSTR output = NULL;
Aric Stewart072c5e52005-04-20 12:50:05 +00003985 HKEY hkey;
3986 UINT rc = ERROR_SUCCESS;
Mike McCormack38d67a42005-08-22 09:15:23 +00003987 MSICOMPONENT *comp;
Aric Stewart072c5e52005-04-20 12:50:05 +00003988 DWORD sz = 0;
Robert Shearmand2e48e02006-01-23 17:29:50 +01003989 MSIRECORD *uirow;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003990
Aric Stewart09b0aba2005-06-09 20:30:59 +00003991 component = MSI_RecordGetString(rec,3);
Mike McCormack38d67a42005-08-22 09:15:23 +00003992 comp = get_loaded_component(package,component);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003993
Mike McCormackd693f462005-10-29 11:36:48 +00003994 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL ) &&
3995 !ACTION_VerifyComponentForAction( comp, INSTALLSTATE_SOURCE ) &&
3996 !ACTION_VerifyComponentForAction( comp, INSTALLSTATE_ADVERTISED ))
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003997 {
3998 TRACE("Skipping: Component %s not scheduled for install\n",
3999 debugstr_w(component));
Aric Stewart6f43c182005-05-26 12:24:28 +00004000
Aric Stewartb39d8fc2005-05-13 13:56:39 +00004001 return ERROR_SUCCESS;
4002 }
Aric Stewart072c5e52005-04-20 12:50:05 +00004003
Aric Stewart09b0aba2005-06-09 20:30:59 +00004004 compgroupid = MSI_RecordGetString(rec,1);
Robert Shearmand2e48e02006-01-23 17:29:50 +01004005 qualifier = MSI_RecordGetString(rec,2);
Aric Stewart072c5e52005-04-20 12:50:05 +00004006
4007 rc = MSIREG_OpenUserComponentsKey(compgroupid, &hkey, TRUE);
4008 if (rc != ERROR_SUCCESS)
4009 goto end;
4010
Aric Stewart09b0aba2005-06-09 20:30:59 +00004011 text = MSI_RecordGetString(rec,4);
Aric Stewart09b0aba2005-06-09 20:30:59 +00004012 feature = MSI_RecordGetString(rec,5);
Aric Stewart072c5e52005-04-20 12:50:05 +00004013
Mike McCormack38d67a42005-08-22 09:15:23 +00004014 advertise = create_component_advertise_string(package, comp, feature);
Aric Stewart072c5e52005-04-20 12:50:05 +00004015
Aric Stewart6f43c182005-05-26 12:24:28 +00004016 sz = strlenW(advertise);
4017
Aric Stewart072c5e52005-04-20 12:50:05 +00004018 if (text)
4019 sz += lstrlenW(text);
Aric Stewart072c5e52005-04-20 12:50:05 +00004020
4021 sz+=3;
4022 sz *= sizeof(WCHAR);
4023
Mike McCormack3a940112006-04-19 02:29:03 +09004024 output = msi_alloc_zero(sz);
Aric Stewart6f43c182005-05-26 12:24:28 +00004025 strcpyW(output,advertise);
Mike McCormack470f23d2005-09-22 10:56:26 +00004026 msi_free(advertise);
Aric Stewart072c5e52005-04-20 12:50:05 +00004027
4028 if (text)
4029 strcatW(output,text);
4030
Mike McCormack4db02cd2005-09-15 14:58:38 +00004031 msi_reg_set_val_multi_str( hkey, qualifier, output );
Aric Stewart072c5e52005-04-20 12:50:05 +00004032
4033end:
4034 RegCloseKey(hkey);
Mike McCormackee034ba2005-09-20 11:59:14 +00004035 msi_free(output);
Robert Shearmand2e48e02006-01-23 17:29:50 +01004036
4037 /* the UI chunk */
4038 uirow = MSI_CreateRecord( 2 );
4039 MSI_RecordSetStringW( uirow, 1, compgroupid );
4040 MSI_RecordSetStringW( uirow, 2, qualifier);
4041 ui_actiondata( package, szPublishComponents, uirow);
4042 msiobj_release( &uirow->hdr );
4043 /* FIXME: call ui_progress? */
4044
Aric Stewart072c5e52005-04-20 12:50:05 +00004045 return rc;
4046}
4047
4048/*
4049 * At present I am ignorning the advertised components part of this and only
4050 * focusing on the qualified component sets
4051 */
4052static UINT ACTION_PublishComponents(MSIPACKAGE *package)
4053{
4054 UINT rc;
4055 MSIQUERY * view;
4056 static const WCHAR ExecSeqQuery[] =
4057 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00004058 '`','P','u','b','l','i','s','h',
4059 'C','o','m','p','o','n','e','n','t','`',0};
Aric Stewart072c5e52005-04-20 12:50:05 +00004060
4061 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
4062 if (rc != ERROR_SUCCESS)
4063 return ERROR_SUCCESS;
4064
4065 rc = MSI_IterateRecords(view, NULL, ITERATE_PublishComponent, package);
4066 msiobj_release(&view->hdr);
4067
4068 return rc;
4069}
Mike McCormack202166c2005-09-23 10:09:18 +00004070
James Hawkins9bc12ad2006-10-19 15:49:54 -07004071static UINT ITERATE_InstallService(MSIRECORD *rec, LPVOID param)
4072{
4073 MSIPACKAGE *package = (MSIPACKAGE*)param;
4074 MSIRECORD *row;
4075 MSIFILE *file;
4076 SC_HANDLE hscm, service = NULL;
4077 LPCWSTR name, disp, comp, depends, pass;
4078 LPCWSTR load_order, serv_name, key;
4079 DWORD serv_type, start_type;
4080 DWORD err_control;
4081
4082 static const WCHAR query[] =
4083 {'S','E','L','E','C','T',' ','*',' ','F','R', 'O','M',' ',
4084 '`','C','o','m','p','o','n','e','n','t','`',' ',
4085 'W','H','E','R','E',' ',
4086 '`','C','o','m','p','o','n','e','n','t','`',' ',
4087 '=','\'','%','s','\'',0};
4088
4089 hscm = OpenSCManagerW(NULL, SERVICES_ACTIVE_DATABASEW, GENERIC_WRITE);
4090 if (!hscm)
4091 {
4092 ERR("Failed to open the SC Manager!\n");
4093 goto done;
4094 }
4095
4096 start_type = MSI_RecordGetInteger(rec, 5);
4097 if (start_type == SERVICE_BOOT_START || start_type == SERVICE_SYSTEM_START)
4098 goto done;
4099
4100 depends = MSI_RecordGetString(rec, 8);
4101 if (depends && *depends)
4102 FIXME("Dependency list unhandled!\n");
4103
4104 name = MSI_RecordGetString(rec, 2);
4105 disp = MSI_RecordGetString(rec, 3);
4106 serv_type = MSI_RecordGetInteger(rec, 4);
4107 err_control = MSI_RecordGetInteger(rec, 6);
4108 load_order = MSI_RecordGetString(rec, 7);
4109 serv_name = MSI_RecordGetString(rec, 9);
4110 pass = MSI_RecordGetString(rec, 10);
4111 comp = MSI_RecordGetString(rec, 12);
4112
4113 /* fetch the service path */
4114 row = MSI_QueryGetRecord(package->db, query, comp);
4115 if (!row)
4116 {
4117 ERR("Control query failed!\n");
4118 goto done;
4119 }
4120
4121 key = MSI_RecordGetString(row, 6);
4122 msiobj_release(&row->hdr);
4123
4124 file = get_loaded_file(package, key);
4125 if (!file)
4126 {
4127 ERR("Failed to load the service file\n");
4128 goto done;
4129 }
4130
4131 service = CreateServiceW(hscm, name, disp, GENERIC_ALL, serv_type,
4132 start_type, err_control, file->TargetPath,
4133 load_order, NULL, NULL, serv_name, pass);
4134 if (!service)
4135 {
4136 if (GetLastError() != ERROR_SERVICE_EXISTS)
4137 ERR("Failed to create service %s: %d\n", debugstr_w(name), GetLastError());
4138 }
4139
4140done:
4141 CloseServiceHandle(service);
4142 CloseServiceHandle(hscm);
4143
4144 return ERROR_SUCCESS;
4145}
4146
4147static UINT ACTION_InstallServices( MSIPACKAGE *package )
4148{
4149 UINT rc;
4150 MSIQUERY * view;
4151 static const WCHAR ExecSeqQuery[] =
4152 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
4153 'S','e','r','v','i','c','e','I','n','s','t','a','l','l',0};
4154
4155 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
4156 if (rc != ERROR_SUCCESS)
4157 return ERROR_SUCCESS;
4158
4159 rc = MSI_IterateRecords(view, NULL, ITERATE_InstallService, package);
4160 msiobj_release(&view->hdr);
4161
4162 return rc;
4163}
4164
Mike McCormack2586a092005-09-26 09:56:18 +00004165static UINT msi_unimplemented_action_stub( MSIPACKAGE *package,
4166 LPCSTR action, LPCWSTR table )
Mike McCormack567f0312005-09-23 11:06:57 +00004167{
4168 static const WCHAR query[] = {
Mike McCormack2586a092005-09-26 09:56:18 +00004169 'S','E','L','E','C','T',' ','*',' ',
4170 'F','R','O','M',' ','`','%','s','`',0 };
Mike McCormack567f0312005-09-23 11:06:57 +00004171 MSIQUERY *view = NULL;
4172 DWORD count = 0;
Mike McCormack2586a092005-09-26 09:56:18 +00004173 UINT r;
Mike McCormack567f0312005-09-23 11:06:57 +00004174
Mike McCormack2586a092005-09-26 09:56:18 +00004175 r = MSI_OpenQuery( package->db, &view, query, table );
4176 if (r == ERROR_SUCCESS)
Mike McCormack567f0312005-09-23 11:06:57 +00004177 {
Mike McCormack2586a092005-09-26 09:56:18 +00004178 r = MSI_IterateRecords(view, &count, NULL, package);
Mike McCormack567f0312005-09-23 11:06:57 +00004179 msiobj_release(&view->hdr);
4180 }
4181
Mike McCormack2586a092005-09-26 09:56:18 +00004182 if (count)
Mike McCormackf1d46462006-10-05 13:41:22 +09004183 FIXME("%s -> %u ignored %s table values\n",
Mike McCormack2586a092005-09-26 09:56:18 +00004184 action, count, debugstr_w(table));
4185
Mike McCormack567f0312005-09-23 11:06:57 +00004186 return ERROR_SUCCESS;
4187}
Mike McCormack94fbe092005-09-23 17:21:10 +00004188
Mike McCormack55942702005-10-30 19:23:28 +00004189static UINT ACTION_AllocateRegistrySpace( MSIPACKAGE *package )
4190{
4191 TRACE("%p\n", package);
4192 return ERROR_SUCCESS;
4193}
4194
Mike McCormack2586a092005-09-26 09:56:18 +00004195static UINT ACTION_RemoveIniValues( MSIPACKAGE *package )
Mike McCormack94fbe092005-09-23 17:21:10 +00004196{
Mike McCormack2586a092005-09-26 09:56:18 +00004197 static const WCHAR table[] =
4198 {'R','e','m','o','v','e','I','n','i','F','i','l','e',0 };
4199 return msi_unimplemented_action_stub( package, "RemoveIniValues", table );
Mike McCormack94fbe092005-09-23 17:21:10 +00004200}
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004201
Mike McCormack2586a092005-09-26 09:56:18 +00004202static UINT ACTION_MoveFiles( MSIPACKAGE *package )
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004203{
Mike McCormack2586a092005-09-26 09:56:18 +00004204 static const WCHAR table[] = { 'M','o','v','e','F','i','l','e',0 };
4205 return msi_unimplemented_action_stub( package, "MoveFiles", table );
4206}
4207
4208static UINT ACTION_PatchFiles( MSIPACKAGE *package )
4209{
4210 static const WCHAR table[] = { 'P','a','t','c','h',0 };
4211 return msi_unimplemented_action_stub( package, "PatchFiles", table );
4212}
4213
4214static UINT ACTION_BindImage( MSIPACKAGE *package )
4215{
4216 static const WCHAR table[] = { 'B','i','n','d','I','m','a','g','e',0 };
4217 return msi_unimplemented_action_stub( package, "BindImage", table );
4218}
4219
4220static UINT ACTION_IsolateComponents( MSIPACKAGE *package )
4221{
4222 static const WCHAR table[] = {
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004223 'I','s','o','l','a','t','e','C','o','m','p','o','n','e','n','t',0 };
Mike McCormack2586a092005-09-26 09:56:18 +00004224 return msi_unimplemented_action_stub( package, "IsolateComponents", table );
4225}
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004226
Mike McCormack2586a092005-09-26 09:56:18 +00004227static UINT ACTION_MigrateFeatureStates( MSIPACKAGE *package )
4228{
4229 static const WCHAR table[] = { 'U','p','g','r','a','d','e',0 };
4230 return msi_unimplemented_action_stub( package, "MigrateFeatureStates", table );
4231}
4232
4233static UINT ACTION_SelfUnregModules( MSIPACKAGE *package )
4234{
4235 static const WCHAR table[] = { 'S','e','l','f','R','e','g',0 };
4236 return msi_unimplemented_action_stub( package, "SelfUnregModules", table );
4237}
4238
Mike McCormack2586a092005-09-26 09:56:18 +00004239static UINT ACTION_StartServices( MSIPACKAGE *package )
4240{
4241 static const WCHAR table[] = {
4242 'S','e','r','v','i','c','e','C','o','n','t','r','o','l',0 };
4243 return msi_unimplemented_action_stub( package, "StartServices", table );
4244}
4245
4246static UINT ACTION_StopServices( MSIPACKAGE *package )
4247{
4248 static const WCHAR table[] = {
4249 'S','e','r','v','i','c','e','C','o','n','t','r','o','l',0 };
4250 return msi_unimplemented_action_stub( package, "StopServices", table );
4251}
4252
4253static UINT ACTION_DeleteServices( MSIPACKAGE *package )
4254{
4255 static const WCHAR table[] = {
4256 'S','e','r','v','i','c','e','C','o','n','t','r','o','l',0 };
4257 return msi_unimplemented_action_stub( package, "DeleteServices", table );
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004258}
Mike McCormack3b955152005-09-28 18:10:44 +00004259
4260static UINT ACTION_WriteEnvironmentStrings( MSIPACKAGE *package )
4261{
4262 static const WCHAR table[] = {
4263 'E','n','v','i','r','o','n','m','e','n','t',0 };
4264 return msi_unimplemented_action_stub( package, "WriteEnvironmentStrings", table );
4265}
4266
4267static UINT ACTION_RemoveEnvironmentStrings( MSIPACKAGE *package )
4268{
4269 static const WCHAR table[] = {
4270 'E','n','v','i','r','o','n','m','e','n','t',0 };
4271 return msi_unimplemented_action_stub( package, "RemoveEnvironmentStrings", table );
4272}
4273
4274static UINT ACTION_MsiPublishAssemblies( MSIPACKAGE *package )
4275{
4276 static const WCHAR table[] = {
4277 'M','s','i','A','s','s','e','m','b','l','y',0 };
4278 return msi_unimplemented_action_stub( package, "MsiPublishAssemblies", table );
4279}
4280
4281static UINT ACTION_MsiUnpublishAssemblies( MSIPACKAGE *package )
4282{
4283 static const WCHAR table[] = {
4284 'M','s','i','A','s','s','e','m','b','l','y',0 };
4285 return msi_unimplemented_action_stub( package, "MsiUnpublishAssemblies", table );
4286}
4287
4288static UINT ACTION_UnregisterFonts( MSIPACKAGE *package )
4289{
4290 static const WCHAR table[] = { 'F','o','n','t',0 };
4291 return msi_unimplemented_action_stub( package, "UnregisterFonts", table );
4292}
4293
Mike McCormackf24a9e22005-12-31 13:14:23 +01004294static UINT ACTION_CCPSearch( MSIPACKAGE *package )
4295{
4296 static const WCHAR table[] = { 'C','C','P','S','e','a','r','c','h',0 };
4297 return msi_unimplemented_action_stub( package, "CCPSearch", table );
4298}
4299
4300static UINT ACTION_RMCCPSearch( MSIPACKAGE *package )
4301{
4302 static const WCHAR table[] = { 'C','C','P','S','e','a','r','c','h',0 };
4303 return msi_unimplemented_action_stub( package, "RMCCPSearch", table );
4304}
4305
Mike McCormack88603662006-03-22 23:01:56 +09004306static UINT ACTION_RegisterComPlus( MSIPACKAGE *package )
4307{
4308 static const WCHAR table[] = { 'C','o','m','p','l','u','s',0 };
4309 return msi_unimplemented_action_stub( package, "RegisterComPlus", table );
4310}
4311
4312static UINT ACTION_UnregisterComPlus( MSIPACKAGE *package )
4313{
4314 static const WCHAR table[] = { 'C','o','m','p','l','u','s',0 };
4315 return msi_unimplemented_action_stub( package, "UnregisterComPlus", table );
4316}
4317
Mike McCormack3b955152005-09-28 18:10:44 +00004318static struct _actions StandardActions[] = {
Mike McCormack55942702005-10-30 19:23:28 +00004319 { szAllocateRegistrySpace, ACTION_AllocateRegistrySpace },
Mike McCormack3b955152005-09-28 18:10:44 +00004320 { szAppSearch, ACTION_AppSearch },
4321 { szBindImage, ACTION_BindImage },
Mike McCormackf24a9e22005-12-31 13:14:23 +01004322 { szCCPSearch, ACTION_CCPSearch},
Mike McCormack3b955152005-09-28 18:10:44 +00004323 { szCostFinalize, ACTION_CostFinalize },
4324 { szCostInitialize, ACTION_CostInitialize },
4325 { szCreateFolders, ACTION_CreateFolders },
4326 { szCreateShortcuts, ACTION_CreateShortcuts },
4327 { szDeleteServices, ACTION_DeleteServices },
4328 { szDisableRollback, NULL},
4329 { szDuplicateFiles, ACTION_DuplicateFiles },
4330 { szExecuteAction, ACTION_ExecuteAction },
4331 { szFileCost, ACTION_FileCost },
4332 { szFindRelatedProducts, ACTION_FindRelatedProducts },
4333 { szForceReboot, ACTION_ForceReboot },
4334 { szInstallAdminPackage, NULL},
4335 { szInstallExecute, ACTION_InstallExecute },
4336 { szInstallExecuteAgain, ACTION_InstallExecute },
4337 { szInstallFiles, ACTION_InstallFiles},
4338 { szInstallFinalize, ACTION_InstallFinalize },
4339 { szInstallInitialize, ACTION_InstallInitialize },
4340 { szInstallSFPCatalogFile, NULL},
4341 { szInstallValidate, ACTION_InstallValidate },
4342 { szIsolateComponents, ACTION_IsolateComponents },
4343 { szLaunchConditions, ACTION_LaunchConditions },
4344 { szMigrateFeatureStates, ACTION_MigrateFeatureStates },
4345 { szMoveFiles, ACTION_MoveFiles },
4346 { szMsiPublishAssemblies, ACTION_MsiPublishAssemblies },
4347 { szMsiUnpublishAssemblies, ACTION_MsiUnpublishAssemblies },
4348 { szInstallODBC, NULL},
4349 { szInstallServices, ACTION_InstallServices },
4350 { szPatchFiles, ACTION_PatchFiles },
4351 { szProcessComponents, ACTION_ProcessComponents },
4352 { szPublishComponents, ACTION_PublishComponents },
4353 { szPublishFeatures, ACTION_PublishFeatures },
4354 { szPublishProduct, ACTION_PublishProduct },
4355 { szRegisterClassInfo, ACTION_RegisterClassInfo },
Mike McCormack88603662006-03-22 23:01:56 +09004356 { szRegisterComPlus, ACTION_RegisterComPlus},
Mike McCormack3b955152005-09-28 18:10:44 +00004357 { szRegisterExtensionInfo, ACTION_RegisterExtensionInfo },
4358 { szRegisterFonts, ACTION_RegisterFonts },
4359 { szRegisterMIMEInfo, ACTION_RegisterMIMEInfo },
4360 { szRegisterProduct, ACTION_RegisterProduct },
4361 { szRegisterProgIdInfo, ACTION_RegisterProgIdInfo },
4362 { szRegisterTypeLibraries, ACTION_RegisterTypeLibraries },
4363 { szRegisterUser, ACTION_RegisterUser},
4364 { szRemoveDuplicateFiles, NULL},
4365 { szRemoveEnvironmentStrings, ACTION_RemoveEnvironmentStrings },
4366 { szRemoveExistingProducts, NULL},
Mike McCormackfd4a2002005-11-02 11:42:56 +00004367 { szRemoveFiles, ACTION_RemoveFiles},
Mike McCormack3b955152005-09-28 18:10:44 +00004368 { szRemoveFolders, NULL},
4369 { szRemoveIniValues, ACTION_RemoveIniValues },
4370 { szRemoveODBC, NULL},
4371 { szRemoveRegistryValues, NULL},
4372 { szRemoveShortcuts, NULL},
4373 { szResolveSource, ACTION_ResolveSource},
Mike McCormackf24a9e22005-12-31 13:14:23 +01004374 { szRMCCPSearch, ACTION_RMCCPSearch},
Mike McCormack3b955152005-09-28 18:10:44 +00004375 { szScheduleReboot, NULL},
4376 { szSelfRegModules, ACTION_SelfRegModules },
4377 { szSelfUnregModules, ACTION_SelfUnregModules },
4378 { szSetODBCFolders, NULL},
4379 { szStartServices, ACTION_StartServices },
4380 { szStopServices, ACTION_StopServices },
4381 { szUnpublishComponents, NULL},
4382 { szUnpublishFeatures, NULL},
4383 { szUnregisterClassInfo, NULL},
Mike McCormack88603662006-03-22 23:01:56 +09004384 { szUnregisterComPlus, ACTION_UnregisterComPlus},
Mike McCormack3b955152005-09-28 18:10:44 +00004385 { szUnregisterExtensionInfo, NULL},
4386 { szUnregisterFonts, ACTION_UnregisterFonts },
4387 { szUnregisterMIMEInfo, NULL},
4388 { szUnregisterProgIdInfo, NULL},
4389 { szUnregisterTypeLibraries, NULL},
4390 { szValidateProductID, NULL},
4391 { szWriteEnvironmentStrings, ACTION_WriteEnvironmentStrings },
4392 { szWriteIniValues, ACTION_WriteIniValues },
4393 { szWriteRegistryValues, ACTION_WriteRegistryValues},
4394 { NULL, NULL},
4395};