blob: c90801618feb68028b69e4cb608ba548c672e306 [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
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19 */
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"
37#include "wine/debug.h"
Aric Stewartb6bc6aa2005-02-24 12:47:43 +000038#include "msidefs.h"
Aric Stewart401bd3f2004-06-28 20:34:35 +000039#include "msipriv.h"
Aric Stewart401bd3f2004-06-28 20:34:35 +000040#include "winuser.h"
41#include "shlobj.h"
42#include "wine/unicode.h"
Steven Edwards98efef12005-04-11 16:10:33 +000043#include "winver.h"
Aric Stewart54c67dd2005-01-25 20:17:09 +000044#include "action.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};
Aric Stewartdb982e22005-06-16 15:51:44 +0000116const 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 Julliardd757bfe2005-05-27 09:11:18 +0000250static void ce_actiontext(MSIPACKAGE* package, LPCWSTR action)
251{
252 static const WCHAR szActionText[] =
253 {'A','c','t','i','o','n','T','e','x','t',0};
254 MSIRECORD *row;
255
256 row = MSI_CreateRecord(1);
257 MSI_RecordSetStringW(row,1,action);
258 ControlEvent_FireSubscribedEvent(package,szActionText, row);
259 msiobj_release(&row->hdr);
Aric Stewart51cf7d12005-01-03 14:29:09 +0000260}
261
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000262static void ui_actionstart(MSIPACKAGE *package, LPCWSTR action)
Aric Stewartd2c395a2004-07-06 18:48:15 +0000263{
264 static const WCHAR template_s[]=
Aric Stewart8e233e92005-03-01 11:45:19 +0000265 {'A','c','t','i','o','n',' ','%','s',':',' ','%','s','.',' ', '%','s',
266 '.',0};
Aric Stewartd2c395a2004-07-06 18:48:15 +0000267 static const WCHAR format[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000268 {'H','H','\'',':','\'','m','m','\'',':','\'','s','s',0};
Aric Stewartd2c395a2004-07-06 18:48:15 +0000269 static const WCHAR Query_t[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000270 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000271 '`','A','c','t','i','o', 'n','T','e','x','t','`',' ',
272 'W','H','E','R','E', ' ','`','A','c','t','i','o','n','`',' ','=',
273 ' ','\'','%','s','\'',0};
Aric Stewartd2c395a2004-07-06 18:48:15 +0000274 WCHAR message[1024];
275 WCHAR timet[0x100];
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000276 MSIRECORD * row = 0;
Mike McCormack20806c72005-06-07 21:34:05 +0000277 LPCWSTR ActionText;
Aric Stewart43ec0a42005-08-25 09:50:08 +0000278 LPWSTR deformated;
Aric Stewartd2c395a2004-07-06 18:48:15 +0000279
280 GetTimeFormatW(LOCALE_USER_DEFAULT, 0, NULL, format, timet, 0x100);
281
Mike McCormack0b352c72005-06-02 10:29:57 +0000282 row = MSI_QueryGetRecord( package->db, Query_t, action );
283 if (!row)
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000284 return;
Aric Stewartd2c395a2004-07-06 18:48:15 +0000285
Mike McCormack20806c72005-06-07 21:34:05 +0000286 ActionText = MSI_RecordGetString(row,2);
Aric Stewart43ec0a42005-08-25 09:50:08 +0000287 deformat_string(package, ActionText, &deformated);
Aric Stewartd2c395a2004-07-06 18:48:15 +0000288
Aric Stewart43ec0a42005-08-25 09:50:08 +0000289 sprintfW(message,template_s,timet,action,deformated);
290 ce_actiontext(package, deformated);
Mike McCormack20806c72005-06-07 21:34:05 +0000291 msiobj_release(&row->hdr);
Aric Stewartd2c395a2004-07-06 18:48:15 +0000292
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000293 row = MSI_CreateRecord(1);
294 MSI_RecordSetStringW(row,1,message);
Aric Stewartd2c395a2004-07-06 18:48:15 +0000295
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000296 MSI_ProcessMessage(package, INSTALLMESSAGE_ACTIONSTART, row);
297 msiobj_release(&row->hdr);
Mike McCormackee034ba2005-09-20 11:59:14 +0000298 msi_free(deformated);
Aric Stewartd2c395a2004-07-06 18:48:15 +0000299}
300
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000301static void ui_actioninfo(MSIPACKAGE *package, LPCWSTR action, BOOL start,
Aric Stewartd2c395a2004-07-06 18:48:15 +0000302 UINT rc)
303{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000304 MSIRECORD * row;
Aric Stewartd2c395a2004-07-06 18:48:15 +0000305 static const WCHAR template_s[]=
Aric Stewart8e233e92005-03-01 11:45:19 +0000306 {'A','c','t','i','o','n',' ','s','t','a','r','t',' ','%','s',':',' ',
307 '%','s', '.',0};
Aric Stewartd2c395a2004-07-06 18:48:15 +0000308 static const WCHAR template_e[]=
Aric Stewart8e233e92005-03-01 11:45:19 +0000309 {'A','c','t','i','o','n',' ','e','n','d','e','d',' ','%','s',':',' ',
310 '%','s', '.',' ','R','e','t','u','r','n',' ','v','a','l','u','e',' ',
311 '%','i','.',0};
Aric Stewartd2c395a2004-07-06 18:48:15 +0000312 static const WCHAR format[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000313 {'H','H','\'',':','\'','m','m','\'',':','\'','s','s',0};
Aric Stewartd2c395a2004-07-06 18:48:15 +0000314 WCHAR message[1024];
315 WCHAR timet[0x100];
316
317 GetTimeFormatW(LOCALE_USER_DEFAULT, 0, NULL, format, timet, 0x100);
318 if (start)
319 sprintfW(message,template_s,timet,action);
320 else
321 sprintfW(message,template_e,timet,action,rc);
322
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000323 row = MSI_CreateRecord(1);
324 MSI_RecordSetStringW(row,1,message);
Aric Stewartd2c395a2004-07-06 18:48:15 +0000325
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000326 MSI_ProcessMessage(package, INSTALLMESSAGE_INFO, row);
327 msiobj_release(&row->hdr);
Aric Stewartd2c395a2004-07-06 18:48:15 +0000328}
329
Mike McCormack74f0de92005-09-29 10:32:39 +0000330static int msi_get_property_int( MSIPACKAGE *package, LPCWSTR prop, int def )
331{
332 LPWSTR str = msi_dup_property( package, prop );
333 int val = str ? atoiW( str ) : def;
334 msi_free( str );
335 return val;
336}
337
Mike McCormacke3452222005-09-28 15:12:32 +0000338static UINT msi_parse_command_line( MSIPACKAGE *package, LPCWSTR szCommandLine )
339{
340 LPCWSTR ptr,ptr2;
341 BOOL quote;
342 DWORD len;
343 LPWSTR prop = NULL, val = NULL;
344
345 if (!szCommandLine)
346 return ERROR_SUCCESS;
347
348 ptr = szCommandLine;
349
350 while (*ptr)
351 {
352 if (*ptr==' ')
353 {
354 ptr++;
355 continue;
356 }
357
358 TRACE("Looking at %s\n",debugstr_w(ptr));
359
360 ptr2 = strchrW(ptr,'=');
361 if (!ptr2)
362 {
363 ERR("command line contains unknown string : %s\n", debugstr_w(ptr));
364 break;
365 }
366
367 quote = FALSE;
368
369 len = ptr2-ptr;
370 prop = msi_alloc((len+1)*sizeof(WCHAR));
371 memcpy(prop,ptr,len*sizeof(WCHAR));
372 prop[len]=0;
373 ptr2++;
374
375 len = 0;
376 ptr = ptr2;
377 while (*ptr && (quote || (!quote && *ptr!=' ')))
378 {
379 if (*ptr == '"')
380 quote = !quote;
381 ptr++;
382 len++;
383 }
384
385 if (*ptr2=='"')
386 {
387 ptr2++;
388 len -= 2;
389 }
390 val = msi_alloc((len+1)*sizeof(WCHAR));
391 memcpy(val,ptr2,len*sizeof(WCHAR));
392 val[len] = 0;
393
394 if (lstrlenW(prop) > 0)
395 {
396 TRACE("Found commandline property (%s) = (%s)\n",
397 debugstr_w(prop), debugstr_w(val));
398 MSI_SetPropertyW(package,prop,val);
399 }
400 msi_free(val);
401 msi_free(prop);
402 }
403
404 return ERROR_SUCCESS;
405}
406
Mike McCormack965a72a2005-10-26 12:06:21 +0000407
408static LPWSTR* msi_split_string( LPCWSTR str, WCHAR sep )
409{
410 LPWSTR p, *ret = NULL;
411 UINT count = 0;
412
413 if (!str)
414 return ret;
415
416 /* count the number of substrings */
417 for ( p = (LPWSTR)str, count = 0; p; count++ )
418 {
419 p = strchrW( p, sep );
420 if (p)
421 p++;
422 }
423
424 /* allocate space for an array of substring pointers and the substrings */
425 ret = msi_alloc( (count+1) * sizeof (LPWSTR) +
426 (lstrlenW(str)+1) * sizeof(WCHAR) );
427 if (!ret)
428 return ret;
429
430 /* copy the string and set the pointers */
431 p = (LPWSTR) &ret[count+1];
432 lstrcpyW( p, str );
433 for( count = 0; (ret[count] = p); count++ )
434 {
435 p = strchrW( p, sep );
436 if (p)
437 *p++ = 0;
438 }
439
440 return ret;
441}
442
443static UINT msi_apply_substorage_transform( MSIPACKAGE *package,
444 MSIDATABASE *patch_db, LPCWSTR name )
445{
446 UINT ret = ERROR_FUNCTION_FAILED;
447 IStorage *stg = NULL;
448 HRESULT r;
449
450 TRACE("%p %s\n", package, debugstr_w(name) );
451
452 if (*name++ != ':')
453 {
454 ERR("expected a colon in %s\n", debugstr_w(name));
455 return ERROR_FUNCTION_FAILED;
456 }
457
458 r = IStorage_OpenStorage( patch_db->storage, name, NULL, STGM_SHARE_EXCLUSIVE, NULL, 0, &stg );
459 if (SUCCEEDED(r))
460 {
Mike McCormackf8fef6e2005-10-26 13:56:33 +0000461 ret = msi_table_apply_transform( package->db, stg );
Mike McCormack965a72a2005-10-26 12:06:21 +0000462 IStorage_Release( stg );
463 ret = ERROR_SUCCESS;
464 }
465 else
466 ERR("failed to open substorage %s\n", debugstr_w(name));
467
468 return ret;
469}
470
471static UINT msi_check_patch_applicable( MSIPACKAGE *package, MSISUMMARYINFO *si )
472{
473 static const WCHAR szProdID[] = { 'P','r','o','d','u','c','t','I','D',0 };
474 LPWSTR guid_list, *guids, product_id;
475 UINT i, ret = ERROR_FUNCTION_FAILED;
476
477 product_id = msi_dup_property( package, szProdID );
478 if (!product_id)
479 {
480 /* FIXME: the property ProductID should be written into the DB somewhere */
481 ERR("no product ID to check\n");
482 return ERROR_SUCCESS;
483 }
484
485 guid_list = msi_suminfo_dup_string( si, PID_TEMPLATE );
486 guids = msi_split_string( guid_list, ';' );
487 for ( i = 0; guids[i] && ret != ERROR_SUCCESS; i++ )
488 {
489 if (!lstrcmpW( guids[i], product_id ))
490 ret = ERROR_SUCCESS;
491 }
492 msi_free( guids );
493 msi_free( guid_list );
494 msi_free( product_id );
495
496 return ret;
497}
498
499static UINT msi_parse_patch_summary( MSIPACKAGE *package, MSIDATABASE *patch_db )
500{
501 MSISUMMARYINFO *si;
502 LPWSTR str, *substorage;
503 UINT i, r = ERROR_SUCCESS;
504
505 si = MSI_GetSummaryInformationW( patch_db, 0 );
506 if (!si)
507 return ERROR_FUNCTION_FAILED;
508
509 msi_check_patch_applicable( package, si );
510
511 /* enumerate the substorage */
512 str = msi_suminfo_dup_string( si, PID_LASTAUTHOR );
513 substorage = msi_split_string( str, ';' );
514 for ( i = 0; substorage && substorage[i] && r == ERROR_SUCCESS; i++ )
515 r = msi_apply_substorage_transform( package, patch_db, substorage[i] );
516 msi_free( substorage );
517 msi_free( str );
518
519 /* FIXME: parse the sources in PID_REVNUMBER and do something with them... */
520
521 msiobj_release( &si->hdr );
522
523 return r;
524}
525
526static UINT msi_apply_patch_package( MSIPACKAGE *package, LPCWSTR file )
527{
528 MSIDATABASE *patch_db = NULL;
529 UINT r;
530
531 TRACE("%p %s\n", package, debugstr_w( file ) );
532
533 /* FIXME:
534 * We probably want to make sure we only open a patch collection here.
535 * Patch collections (.msp) and databases (.msi) have different GUIDs
536 * but currently MSI_OpenDatabaseW will accept both.
537 */
538 r = MSI_OpenDatabaseW( file, MSIDBOPEN_READONLY, &patch_db );
539 if ( r != ERROR_SUCCESS )
540 {
541 ERR("failed to open patch collection %s\n", debugstr_w( file ) );
542 return r;
543 }
544
545 msi_parse_patch_summary( package, patch_db );
546 msiobj_release( &patch_db->hdr );
547
548 return ERROR_SUCCESS;
549}
550
551/* get the PATCH property, and apply all the patches it specifies */
552static UINT msi_apply_patches( MSIPACKAGE *package )
553{
554 static const WCHAR szPatch[] = { 'P','A','T','C','H',0 };
555 LPWSTR patch_list, *patches;
556 UINT i, r = ERROR_SUCCESS;
557
558 patch_list = msi_dup_property( package, szPatch );
559
560 TRACE("patches to be applied: %s\n", debugstr_w( patch_list ) );
561
562 patches = msi_split_string( patch_list, ';' );
563 for( i=0; patches && patches[i] && r == ERROR_SUCCESS; i++ )
564 r = msi_apply_patch_package( package, patches[i] );
565
566 msi_free( patches );
567 msi_free( patch_list );
568
569 return r;
570}
571
Aric Stewart401bd3f2004-06-28 20:34:35 +0000572/****************************************************
573 * TOP level entry points
574 *****************************************************/
575
Mike McCormack61f24a42005-09-30 10:32:41 +0000576UINT MSI_InstallPackage( MSIPACKAGE *package, LPCWSTR szPackagePath,
577 LPCWSTR szCommandLine )
Aric Stewart401bd3f2004-06-28 20:34:35 +0000578{
Aric Stewart401bd3f2004-06-28 20:34:35 +0000579 UINT rc;
Aric Stewartf8f64402005-03-24 19:03:45 +0000580 BOOL ui = FALSE;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000581 static const WCHAR szUILevel[] = {'U','I','L','e','v','e','l',0};
Aric Stewart6269f002005-01-17 13:40:39 +0000582 static const WCHAR szAction[] = {'A','C','T','I','O','N',0};
583 static const WCHAR szInstall[] = {'I','N','S','T','A','L','L',0};
584
585 MSI_SetPropertyW(package, szAction, szInstall);
Aric Stewart9cd707d2005-05-27 19:24:22 +0000586
Mike McCormackee034ba2005-09-20 11:59:14 +0000587 package->script = msi_alloc(sizeof(MSISCRIPT));
Aric Stewart9cd707d2005-05-27 19:24:22 +0000588 memset(package->script,0,sizeof(MSISCRIPT));
Aric Stewart401bd3f2004-06-28 20:34:35 +0000589
Aric Stewartc9802932005-06-30 20:45:43 +0000590 package->script->InWhatSequence = SEQUENCE_INSTALL;
591
Aric Stewarte95136b2004-06-29 03:44:01 +0000592 if (szPackagePath)
593 {
Mike McCormackba8200b2004-12-22 15:25:30 +0000594 LPWSTR p, check, path;
Aric Stewarte95136b2004-06-29 03:44:01 +0000595
Mike McCormack95dea492005-03-16 11:31:35 +0000596 package->PackagePath = strdupW(szPackagePath);
597 path = strdupW(szPackagePath);
Mike McCormackba8200b2004-12-22 15:25:30 +0000598 p = strrchrW(path,'\\');
Aric Stewarte95136b2004-06-29 03:44:01 +0000599 if (p)
600 {
601 p++;
602 *p=0;
603 }
Aric Stewartc1e5c4a2005-02-08 14:26:49 +0000604 else
605 {
Mike McCormackee034ba2005-09-20 11:59:14 +0000606 msi_free(path);
607 path = msi_alloc(MAX_PATH*sizeof(WCHAR));
Aric Stewartc1e5c4a2005-02-08 14:26:49 +0000608 GetCurrentDirectoryW(MAX_PATH,path);
609 strcatW(path,cszbs);
610 }
Aric Stewarte95136b2004-06-29 03:44:01 +0000611
Mike McCormack062ad502005-09-15 15:04:08 +0000612 check = msi_dup_property( package, cszSourceDir );
Aric Stewartd0c971a2004-12-22 18:15:50 +0000613 if (!check)
Mike McCormackba8200b2004-12-22 15:25:30 +0000614 MSI_SetPropertyW(package, cszSourceDir, path);
Mike McCormackee034ba2005-09-20 11:59:14 +0000615 msi_free(check);
616 msi_free(path);
Aric Stewarte95136b2004-06-29 03:44:01 +0000617 }
Aric Stewart401bd3f2004-06-28 20:34:35 +0000618
Mike McCormacke3452222005-09-28 15:12:32 +0000619 msi_parse_command_line( package, szCommandLine );
Mike McCormack74f0de92005-09-29 10:32:39 +0000620
Mike McCormack965a72a2005-10-26 12:06:21 +0000621 msi_apply_patches( package );
622
Mike McCormack74f0de92005-09-29 10:32:39 +0000623 if ( msi_get_property_int(package, szUILevel, 0) >= INSTALLUILEVEL_REDUCED )
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000624 {
Mike McCormack74f0de92005-09-29 10:32:39 +0000625 package->script->InWhatSequence |= SEQUENCE_UI;
626 rc = ACTION_ProcessUISequence(package);
627 ui = TRUE;
628 if (rc == ERROR_SUCCESS)
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000629 {
Mike McCormack74f0de92005-09-29 10:32:39 +0000630 package->script->InWhatSequence |= SEQUENCE_EXEC;
631 rc = ACTION_ProcessExecSequence(package,TRUE);
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000632 }
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000633 }
634 else
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000635 rc = ACTION_ProcessExecSequence(package,FALSE);
Aric Stewart6b16f292005-01-27 11:12:56 +0000636
637 if (rc == -1)
638 {
639 /* install was halted but should be considered a success */
640 rc = ERROR_SUCCESS;
641 }
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000642
Aric Stewart9cd707d2005-05-27 19:24:22 +0000643 package->script->CurrentlyScripting= FALSE;
644
Aric Stewart09d35c32004-12-27 19:00:26 +0000645 /* process the ending type action */
646 if (rc == ERROR_SUCCESS)
Aric Stewartf8f64402005-03-24 19:03:45 +0000647 ACTION_PerformActionSequence(package,-1,ui);
Aric Stewart6b16f292005-01-27 11:12:56 +0000648 else if (rc == ERROR_INSTALL_USEREXIT)
Aric Stewartf8f64402005-03-24 19:03:45 +0000649 ACTION_PerformActionSequence(package,-2,ui);
Aric Stewart6b16f292005-01-27 11:12:56 +0000650 else if (rc == ERROR_INSTALL_SUSPEND)
Aric Stewartf8f64402005-03-24 19:03:45 +0000651 ACTION_PerformActionSequence(package,-4,ui);
Aric Stewart9cd707d2005-05-27 19:24:22 +0000652 else /* failed */
653 ACTION_PerformActionSequence(package,-3,ui);
Aric Stewart54c67dd2005-01-25 20:17:09 +0000654
655 /* finish up running custom actions */
656 ACTION_FinishCustomActions(package);
Aric Stewart09d35c32004-12-27 19:00:26 +0000657
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000658 return rc;
659}
660
Aric Stewartf8f64402005-03-24 19:03:45 +0000661static UINT ACTION_PerformActionSequence(MSIPACKAGE *package, UINT seq, BOOL UI)
Aric Stewart09d35c32004-12-27 19:00:26 +0000662{
Mike McCormack0b352c72005-06-02 10:29:57 +0000663 UINT rc = ERROR_SUCCESS;
Aric Stewart09d35c32004-12-27 19:00:26 +0000664 MSIRECORD * row = 0;
Aric Stewart8e233e92005-03-01 11:45:19 +0000665 static const WCHAR ExecSeqQuery[] =
666 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000667 '`','I','n','s','t','a','l','l','E','x','e','c','u','t','e',
668 'S','e','q','u','e','n','c','e','`',' ', 'W','H','E','R','E',' ',
669 '`','S','e','q','u','e','n','c','e','`',' ', '=',' ','%','i',0};
Aric Stewart09d35c32004-12-27 19:00:26 +0000670
Aric Stewartf8f64402005-03-24 19:03:45 +0000671 static const WCHAR UISeqQuery[] =
672 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000673 '`','I','n','s','t','a','l','l','U','I','S','e','q','u','e','n','c','e',
674 '`', ' ', 'W','H','E','R','E',' ','`','S','e','q','u','e','n','c','e','`',
Aric Stewartf8f64402005-03-24 19:03:45 +0000675 ' ', '=',' ','%','i',0};
676
677 if (UI)
Mike McCormack0b352c72005-06-02 10:29:57 +0000678 row = MSI_QueryGetRecord(package->db, UISeqQuery, seq);
Aric Stewartf8f64402005-03-24 19:03:45 +0000679 else
Mike McCormack0b352c72005-06-02 10:29:57 +0000680 row = MSI_QueryGetRecord(package->db, ExecSeqQuery, seq);
Aric Stewart09d35c32004-12-27 19:00:26 +0000681
Mike McCormack0b352c72005-06-02 10:29:57 +0000682 if (row)
Aric Stewart09d35c32004-12-27 19:00:26 +0000683 {
Mike McCormack20806c72005-06-07 21:34:05 +0000684 LPCWSTR action, cond;
685
Aric Stewart09d35c32004-12-27 19:00:26 +0000686 TRACE("Running the actions\n");
687
Aric Stewart09d35c32004-12-27 19:00:26 +0000688 /* check conditions */
Mike McCormack20806c72005-06-07 21:34:05 +0000689 cond = MSI_RecordGetString(row,2);
690 if (cond)
Aric Stewart09d35c32004-12-27 19:00:26 +0000691 {
Mike McCormack20806c72005-06-07 21:34:05 +0000692 /* this is a hack to skip errors in the condition code */
693 if (MSI_EvaluateConditionW(package, cond) == MSICONDITION_FALSE)
694 goto end;
Aric Stewart09d35c32004-12-27 19:00:26 +0000695 }
696
Mike McCormack20806c72005-06-07 21:34:05 +0000697 action = MSI_RecordGetString(row,1);
698 if (!action)
Aric Stewart09d35c32004-12-27 19:00:26 +0000699 {
Mike McCormack20806c72005-06-07 21:34:05 +0000700 ERR("failed to fetch action\n");
701 rc = ERROR_FUNCTION_FAILED;
Aric Stewart09d35c32004-12-27 19:00:26 +0000702 goto end;
703 }
704
Aric Stewartf8f64402005-03-24 19:03:45 +0000705 if (UI)
Mike McCormack20806c72005-06-07 21:34:05 +0000706 rc = ACTION_PerformUIAction(package,action);
Aric Stewartf8f64402005-03-24 19:03:45 +0000707 else
Mike McCormack20806c72005-06-07 21:34:05 +0000708 rc = ACTION_PerformAction(package,action,FALSE);
Aric Stewart09d35c32004-12-27 19:00:26 +0000709end:
Mike McCormack0b352c72005-06-02 10:29:57 +0000710 msiobj_release(&row->hdr);
Aric Stewart09d35c32004-12-27 19:00:26 +0000711 }
712 else
713 rc = ERROR_SUCCESS;
714
715 return rc;
716}
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000717
Aric Stewart2703d712005-06-20 15:33:10 +0000718typedef struct {
719 MSIPACKAGE* package;
720 BOOL UI;
721} iterate_action_param;
722
723static UINT ITERATE_Actions(MSIRECORD *row, LPVOID param)
724{
725 iterate_action_param *iap= (iterate_action_param*)param;
726 UINT rc;
727 LPCWSTR cond, action;
728
729 action = MSI_RecordGetString(row,1);
730 if (!action)
731 {
732 ERR("Error is retrieving action name\n");
733 return ERROR_FUNCTION_FAILED;
734 }
735
736 /* check conditions */
737 cond = MSI_RecordGetString(row,2);
738 if (cond)
739 {
740 /* this is a hack to skip errors in the condition code */
741 if (MSI_EvaluateConditionW(iap->package, cond) == MSICONDITION_FALSE)
742 {
743 TRACE("Skipping action: %s (condition is false)\n",
744 debugstr_w(action));
745 return ERROR_SUCCESS;
746 }
747 }
748
749 if (iap->UI)
750 rc = ACTION_PerformUIAction(iap->package,action);
751 else
752 rc = ACTION_PerformAction(iap->package,action,FALSE);
753
Mike McCormack4f634a32005-07-06 15:44:51 +0000754 msi_dialog_check_messages( NULL );
755
756 if (iap->package->CurrentInstallState != ERROR_SUCCESS )
757 rc = iap->package->CurrentInstallState;
758
Aric Stewart2703d712005-06-20 15:33:10 +0000759 if (rc == ERROR_FUNCTION_NOT_CALLED)
760 rc = ERROR_SUCCESS;
761
762 if (rc != ERROR_SUCCESS)
Mike McCormack558abec2005-10-27 12:39:28 +0000763 ERR("Execution halted, action %s returned %i\n", debugstr_w(action), rc);
Aric Stewart2703d712005-06-20 15:33:10 +0000764
765 return rc;
766}
767
Mike McCormackd34b1c22005-09-21 10:55:23 +0000768UINT MSI_Sequence( MSIPACKAGE *package, LPCWSTR szTable, INT iSequenceMode )
769{
770 MSIQUERY * view;
771 UINT r;
772 static const WCHAR query[] =
773 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
774 '`','%','s','`',
775 ' ','W','H','E','R','E',' ',
776 '`','S','e','q','u','e','n','c','e','`',' ',
777 '>',' ','0',' ','O','R','D','E','R',' ','B','Y',' ',
778 '`','S','e','q','u','e','n','c','e','`',0};
779 iterate_action_param iap;
780
781 /*
782 * FIXME: probably should be checking UILevel in the
783 * ACTION_PerformUIAction/ACTION_PerformAction
784 * rather than saving the UI level here. Those
785 * two functions can be merged too.
786 */
787 iap.package = package;
788 iap.UI = TRUE;
789
790 TRACE("%p %s %i\n", package, debugstr_w(szTable), iSequenceMode );
791
792 r = MSI_OpenQuery( package->db, &view, query, szTable );
793 if (r == ERROR_SUCCESS)
794 {
795 r = MSI_IterateRecords( view, NULL, ITERATE_Actions, &iap );
796 msiobj_release(&view->hdr);
797 }
798
799 return r;
800}
801
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000802static UINT ACTION_ProcessExecSequence(MSIPACKAGE *package, BOOL UIran)
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000803{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000804 MSIQUERY * view;
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000805 UINT rc;
Aric Stewart8e233e92005-03-01 11:45:19 +0000806 static const WCHAR ExecSeqQuery[] =
807 {'S','E','L','E','C','T',' ','*',' ', 'F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000808 '`','I','n','s','t','a','l','l','E','x','e','c','u','t','e',
809 'S','e','q','u','e','n','c','e','`',' ', 'W','H','E','R','E',' ',
810 '`','S','e','q','u','e','n','c','e','`',' ', '>',' ','%','i',' ',
Aric Stewart8e233e92005-03-01 11:45:19 +0000811 'O','R','D','E','R',' ', 'B','Y',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000812 '`','S','e','q','u','e','n','c','e','`',0 };
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000813 MSIRECORD * row = 0;
Aric Stewart8e233e92005-03-01 11:45:19 +0000814 static const WCHAR IVQuery[] =
Aric Stewart98e38082005-05-20 09:40:42 +0000815 {'S','E','L','E','C','T',' ','`','S','e','q','u','e','n','c','e','`',
816 ' ', 'F','R','O','M',' ','`','I','n','s','t','a','l','l',
817 'E','x','e','c','u','t','e','S','e','q','u','e','n','c','e','`',' ',
818 'W','H','E','R','E',' ','`','A','c','t','i','o','n','`',' ','=',
819 ' ','\'', 'I','n','s','t','a','l','l',
820 'V','a','l','i','d','a','t','e','\'', 0};
Mike McCormack9db0e072004-12-22 15:05:07 +0000821 INT seq = 0;
Aric Stewart2703d712005-06-20 15:33:10 +0000822 iterate_action_param iap;
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000823
Aric Stewart2703d712005-06-20 15:33:10 +0000824 iap.package = package;
825 iap.UI = FALSE;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +0000826
Aric Stewart9cd707d2005-05-27 19:24:22 +0000827 if (package->script->ExecuteSequenceRun)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +0000828 {
829 TRACE("Execute Sequence already Run\n");
830 return ERROR_SUCCESS;
831 }
832
Aric Stewart9cd707d2005-05-27 19:24:22 +0000833 package->script->ExecuteSequenceRun = TRUE;
Aric Stewart2703d712005-06-20 15:33:10 +0000834
Mike McCormack9db0e072004-12-22 15:05:07 +0000835 /* get the sequence number */
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000836 if (UIran)
837 {
Mike McCormack0b352c72005-06-02 10:29:57 +0000838 row = MSI_QueryGetRecord(package->db, IVQuery);
839 if( !row )
840 return ERROR_FUNCTION_FAILED;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000841 seq = MSI_RecordGetInteger(row,1);
842 msiobj_release(&row->hdr);
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000843 }
Mike McCormack9db0e072004-12-22 15:05:07 +0000844
Mike McCormack0c238852005-01-21 16:19:11 +0000845 rc = MSI_OpenQuery(package->db, &view, ExecSeqQuery, seq);
Aric Stewart401bd3f2004-06-28 20:34:35 +0000846 if (rc == ERROR_SUCCESS)
847 {
Aric Stewart2703d712005-06-20 15:33:10 +0000848 TRACE("Running the actions\n");
Aric Stewart401bd3f2004-06-28 20:34:35 +0000849
Aric Stewart2703d712005-06-20 15:33:10 +0000850 rc = MSI_IterateRecords(view, NULL, ITERATE_Actions, &iap);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000851 msiobj_release(&view->hdr);
Aric Stewart401bd3f2004-06-28 20:34:35 +0000852 }
853
Aric Stewart401bd3f2004-06-28 20:34:35 +0000854 return rc;
855}
856
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000857static UINT ACTION_ProcessUISequence(MSIPACKAGE *package)
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000858{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000859 MSIQUERY * view;
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000860 UINT rc;
Aric Stewart8e233e92005-03-01 11:45:19 +0000861 static const WCHAR ExecSeqQuery [] =
862 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000863 '`','I','n','s','t','a','l','l',
864 'U','I','S','e','q','u','e','n','c','e','`',
865 ' ','W','H','E','R','E',' ',
866 '`','S','e','q','u','e','n','c','e','`',' ',
Aric Stewart8e233e92005-03-01 11:45:19 +0000867 '>',' ','0',' ','O','R','D','E','R',' ','B','Y',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000868 '`','S','e','q','u','e','n','c','e','`',0};
Aric Stewart2703d712005-06-20 15:33:10 +0000869 iterate_action_param iap;
870
871 iap.package = package;
872 iap.UI = TRUE;
873
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000874 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000875
876 if (rc == ERROR_SUCCESS)
877 {
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000878 TRACE("Running the actions \n");
879
Aric Stewart2703d712005-06-20 15:33:10 +0000880 rc = MSI_IterateRecords(view, NULL, ITERATE_Actions, &iap);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000881 msiobj_release(&view->hdr);
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000882 }
883
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000884 return rc;
885}
886
Aric Stewart401bd3f2004-06-28 20:34:35 +0000887/********************************************************
888 * ACTION helper functions and functions that perform the actions
889 *******************************************************/
Aric Stewart9cd707d2005-05-27 19:24:22 +0000890static BOOL ACTION_HandleStandardAction(MSIPACKAGE *package, LPCWSTR action,
891 UINT* rc, BOOL force )
Aric Stewart3f318602005-02-01 18:46:26 +0000892{
893 BOOL ret = FALSE;
Aric Stewart9cd707d2005-05-27 19:24:22 +0000894 BOOL run = force;
Aric Stewart3f318602005-02-01 18:46:26 +0000895 int i;
Aric Stewart9cd707d2005-05-27 19:24:22 +0000896
Mike McCormacka977b2c2005-11-03 09:56:29 +0000897 if (!package)
898 {
899 ERR("package was null!\n");
900 return FALSE;
901 }
902
Aric Stewart9cd707d2005-05-27 19:24:22 +0000903 if (!run && !package->script->CurrentlyScripting)
904 run = TRUE;
905
906 if (!run)
907 {
908 if (strcmpW(action,szInstallFinalize) == 0 ||
909 strcmpW(action,szInstallExecute) == 0 ||
910 strcmpW(action,szInstallExecuteAgain) == 0)
911 run = TRUE;
912 }
913
Aric Stewart3f318602005-02-01 18:46:26 +0000914 i = 0;
915 while (StandardActions[i].action != NULL)
916 {
917 if (strcmpW(StandardActions[i].action, action)==0)
918 {
Aric Stewart9cd707d2005-05-27 19:24:22 +0000919 if (!run)
Aric Stewartd6ecf582005-02-02 09:29:30 +0000920 {
Aric Stewart9cd707d2005-05-27 19:24:22 +0000921 ui_actioninfo(package, action, TRUE, 0);
922 *rc = schedule_action(package,INSTALL_SCRIPT,action);
923 ui_actioninfo(package, action, FALSE, *rc);
Aric Stewartd6ecf582005-02-02 09:29:30 +0000924 }
925 else
926 {
Aric Stewart9cd707d2005-05-27 19:24:22 +0000927 ui_actionstart(package, action);
928 if (StandardActions[i].handler)
929 {
930 *rc = StandardActions[i].handler(package);
931 }
932 else
933 {
Mike McCormack54a28912005-09-06 09:23:18 +0000934 FIXME("unhandled standard action %s\n",debugstr_w(action));
Aric Stewart9cd707d2005-05-27 19:24:22 +0000935 *rc = ERROR_SUCCESS;
936 }
Aric Stewartd6ecf582005-02-02 09:29:30 +0000937 }
Aric Stewart3f318602005-02-01 18:46:26 +0000938 ret = TRUE;
939 break;
940 }
941 i++;
942 }
943 return ret;
944}
945
Mike McCormackf9acfe62005-06-07 20:29:51 +0000946static BOOL ACTION_HandleCustomAction( MSIPACKAGE* package, LPCWSTR action,
947 UINT* rc, BOOL force )
Aric Stewart3f318602005-02-01 18:46:26 +0000948{
949 BOOL ret=FALSE;
950 UINT arc;
951
Aric Stewart9cd707d2005-05-27 19:24:22 +0000952 arc = ACTION_CustomAction(package,action, force);
Aric Stewart3f318602005-02-01 18:46:26 +0000953
954 if (arc != ERROR_CALL_NOT_IMPLEMENTED)
955 {
956 *rc = arc;
957 ret = TRUE;
958 }
959 return ret;
960}
Aric Stewart401bd3f2004-06-28 20:34:35 +0000961
962/*
Francois Gougetda8b3dd2005-01-26 21:09:04 +0000963 * A lot of actions are really important even if they don't do anything
964 * explicit... Lots of properties are set at the beginning of the installation
965 * CostFinalize does a bunch of work to translate the directories and such
Aric Stewart401bd3f2004-06-28 20:34:35 +0000966 *
Mike McCormack6e2bca32004-07-04 00:25:00 +0000967 * But until I get write access to the database that is hard, so I am going to
Aric Stewart401bd3f2004-06-28 20:34:35 +0000968 * hack it to see if I can get something to run.
969 */
Aric Stewart9cd707d2005-05-27 19:24:22 +0000970UINT ACTION_PerformAction(MSIPACKAGE *package, const WCHAR *action, BOOL force)
Aric Stewart401bd3f2004-06-28 20:34:35 +0000971{
Aric Stewartd2c395a2004-07-06 18:48:15 +0000972 UINT rc = ERROR_SUCCESS;
Aric Stewart3f318602005-02-01 18:46:26 +0000973 BOOL handled;
Aric Stewart401bd3f2004-06-28 20:34:35 +0000974
975 TRACE("Performing action (%s)\n",debugstr_w(action));
976
Aric Stewart9cd707d2005-05-27 19:24:22 +0000977 handled = ACTION_HandleStandardAction(package, action, &rc, force);
Aric Stewart7d3e5972004-07-04 00:36:58 +0000978
Aric Stewart3f318602005-02-01 18:46:26 +0000979 if (!handled)
Aric Stewart9cd707d2005-05-27 19:24:22 +0000980 handled = ACTION_HandleCustomAction(package, action, &rc, force);
Aric Stewart3f318602005-02-01 18:46:26 +0000981
Aric Stewart90c57392005-01-31 16:23:12 +0000982 if (!handled)
983 {
Mike McCormack54a28912005-09-06 09:23:18 +0000984 FIXME("unhandled msi action %s\n",debugstr_w(action));
Aric Stewart3f318602005-02-01 18:46:26 +0000985 rc = ERROR_FUNCTION_NOT_CALLED;
986 }
Aric Stewart7d3e5972004-07-04 00:36:58 +0000987
Aric Stewart3f318602005-02-01 18:46:26 +0000988 return rc;
989}
990
991UINT ACTION_PerformUIAction(MSIPACKAGE *package, const WCHAR *action)
992{
993 UINT rc = ERROR_SUCCESS;
994 BOOL handled = FALSE;
995
996 TRACE("Performing action (%s)\n",debugstr_w(action));
997
Aric Stewart9cd707d2005-05-27 19:24:22 +0000998 handled = ACTION_HandleStandardAction(package, action, &rc,TRUE);
Aric Stewart3f318602005-02-01 18:46:26 +0000999
1000 if (!handled)
Aric Stewart9cd707d2005-05-27 19:24:22 +00001001 handled = ACTION_HandleCustomAction(package, action, &rc, FALSE);
Aric Stewart3f318602005-02-01 18:46:26 +00001002
Mike McCormack4f634a32005-07-06 15:44:51 +00001003 if( !handled && ACTION_DialogBox(package,action) == ERROR_SUCCESS )
1004 handled = TRUE;
Mike McCormack34d4a022005-02-09 13:24:31 +00001005
Aric Stewart3f318602005-02-01 18:46:26 +00001006 if (!handled)
1007 {
Mike McCormack54a28912005-09-06 09:23:18 +00001008 FIXME("unhandled msi action %s\n",debugstr_w(action));
Aric Stewart3f318602005-02-01 18:46:26 +00001009 rc = ERROR_FUNCTION_NOT_CALLED;
Aric Stewart90c57392005-01-31 16:23:12 +00001010 }
Aric Stewart401bd3f2004-06-28 20:34:35 +00001011
Aric Stewartd2c395a2004-07-06 18:48:15 +00001012 return rc;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001013}
1014
Aric Stewart2274ff12005-06-21 20:03:46 +00001015
1016/*
1017 * Actual Action Handlers
1018 */
1019
1020static UINT ITERATE_CreateFolders(MSIRECORD *row, LPVOID param)
1021{
1022 MSIPACKAGE *package = (MSIPACKAGE*)param;
1023 LPCWSTR dir;
1024 LPWSTR full_path;
1025 MSIRECORD *uirow;
1026 MSIFOLDER *folder;
1027
1028 dir = MSI_RecordGetString(row,1);
1029 if (!dir)
1030 {
1031 ERR("Unable to get folder id \n");
1032 return ERROR_SUCCESS;
1033 }
1034
1035 full_path = resolve_folder(package,dir,FALSE,FALSE,&folder);
1036 if (!full_path)
1037 {
1038 ERR("Unable to resolve folder id %s\n",debugstr_w(dir));
1039 return ERROR_SUCCESS;
1040 }
1041
1042 TRACE("Folder is %s\n",debugstr_w(full_path));
1043
1044 /* UI stuff */
1045 uirow = MSI_CreateRecord(1);
1046 MSI_RecordSetStringW(uirow,1,full_path);
1047 ui_actiondata(package,szCreateFolders,uirow);
1048 msiobj_release( &uirow->hdr );
1049
1050 if (folder->State == 0)
1051 create_full_pathW(full_path);
1052
1053 folder->State = 3;
1054
Mike McCormackee034ba2005-09-20 11:59:14 +00001055 msi_free(full_path);
Aric Stewart2274ff12005-06-21 20:03:46 +00001056 return ERROR_SUCCESS;
1057}
1058
Mike McCormack03b4dbb2005-10-28 09:39:29 +00001059/* FIXME: probably should merge this with the above function */
1060static UINT msi_create_directory( MSIPACKAGE* package, LPCWSTR dir )
1061{
1062 UINT rc = ERROR_SUCCESS;
1063 MSIFOLDER *folder;
1064 LPWSTR install_path;
1065
1066 install_path = resolve_folder(package, dir, FALSE, FALSE, &folder);
1067 if (!install_path)
1068 return ERROR_FUNCTION_FAILED;
1069
1070 /* create the path */
1071 if (folder->State == 0)
1072 {
1073 create_full_pathW(install_path);
1074 folder->State = 2;
1075 }
1076 msi_free(install_path);
1077
1078 return rc;
1079}
Aric Stewart2274ff12005-06-21 20:03:46 +00001080
Mike McCormack9c845852005-10-29 11:29:17 +00001081UINT msi_create_component_directories( MSIPACKAGE *package )
1082{
1083 MSICOMPONENT *comp;
1084
1085 /* create all the folders required by the components are going to install */
1086 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
1087 {
Mike McCormackd693f462005-10-29 11:36:48 +00001088 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
Mike McCormack9c845852005-10-29 11:29:17 +00001089 continue;
1090 msi_create_directory( package, comp->Directory );
1091 }
1092
1093 return ERROR_SUCCESS;
1094}
1095
Aric Stewart401bd3f2004-06-28 20:34:35 +00001096/*
1097 * Also we cannot enable/disable components either, so for now I am just going
Mike McCormack6e2bca32004-07-04 00:25:00 +00001098 * to do all the directories for all the components.
Aric Stewart401bd3f2004-06-28 20:34:35 +00001099 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001100static UINT ACTION_CreateFolders(MSIPACKAGE *package)
Aric Stewart401bd3f2004-06-28 20:34:35 +00001101{
Aric Stewart8e233e92005-03-01 11:45:19 +00001102 static const WCHAR ExecSeqQuery[] =
Aric Stewart98e38082005-05-20 09:40:42 +00001103 {'S','E','L','E','C','T',' ',
1104 '`','D','i','r','e','c','t','o','r','y','_','`',
Aric Stewart8e233e92005-03-01 11:45:19 +00001105 ' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001106 '`','C','r','e','a','t','e','F','o','l','d','e','r','`',0 };
Aric Stewart401bd3f2004-06-28 20:34:35 +00001107 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001108 MSIQUERY *view;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001109
Mike McCormack03b4dbb2005-10-28 09:39:29 +00001110 /* create all the empty folders specified in the CreateFolder table */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001111 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view );
Aric Stewart401bd3f2004-06-28 20:34:35 +00001112 if (rc != ERROR_SUCCESS)
Aric Stewart84837d92004-07-20 01:22:37 +00001113 return ERROR_SUCCESS;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001114
Aric Stewart2274ff12005-06-21 20:03:46 +00001115 rc = MSI_IterateRecords(view, NULL, ITERATE_CreateFolders, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001116 msiobj_release(&view->hdr);
Mike McCormack03b4dbb2005-10-28 09:39:29 +00001117
Mike McCormack9c845852005-10-29 11:29:17 +00001118 msi_create_component_directories( package );
Mike McCormack03b4dbb2005-10-28 09:39:29 +00001119
Aric Stewart401bd3f2004-06-28 20:34:35 +00001120 return rc;
1121}
1122
Mike McCormack38d67a42005-08-22 09:15:23 +00001123static MSICOMPONENT* load_component( MSIRECORD * row )
Aric Stewartbdb29552004-07-04 00:32:48 +00001124{
Mike McCormack38d67a42005-08-22 09:15:23 +00001125 MSICOMPONENT *comp;
Aric Stewartbdb29552004-07-04 00:32:48 +00001126
Mike McCormackee034ba2005-09-20 11:59:14 +00001127 comp = msi_alloc_zero( sizeof(MSICOMPONENT) );
Mike McCormack38d67a42005-08-22 09:15:23 +00001128 if (!comp)
1129 return comp;
1130
Aric Stewartbdb29552004-07-04 00:32:48 +00001131 /* fill in the data */
Mike McCormack51c66182005-10-27 12:36:12 +00001132 comp->Component = msi_dup_record_field( row, 1 );
Aric Stewartbdb29552004-07-04 00:32:48 +00001133
Mike McCormackefcc1ec2005-09-12 12:07:15 +00001134 TRACE("Loading Component %s\n", debugstr_w(comp->Component));
Aric Stewartbdb29552004-07-04 00:32:48 +00001135
Mike McCormack51c66182005-10-27 12:36:12 +00001136 comp->ComponentId = msi_dup_record_field( row, 2 );
1137 comp->Directory = msi_dup_record_field( row, 3 );
Mike McCormack38d67a42005-08-22 09:15:23 +00001138 comp->Attributes = MSI_RecordGetInteger(row,4);
Mike McCormack51c66182005-10-27 12:36:12 +00001139 comp->Condition = msi_dup_record_field( row, 5 );
1140 comp->KeyPath = msi_dup_record_field( row, 6 );
Aric Stewartbdb29552004-07-04 00:32:48 +00001141
Mike McCormack38d67a42005-08-22 09:15:23 +00001142 comp->Installed = INSTALLSTATE_ABSENT;
1143 comp->Action = INSTALLSTATE_UNKNOWN;
1144 comp->ActionRequest = INSTALLSTATE_UNKNOWN;
Aric Stewartfbdd7092004-12-27 19:06:22 +00001145
Mike McCormack38d67a42005-08-22 09:15:23 +00001146 comp->Enabled = TRUE;
Aric Stewartbdb29552004-07-04 00:32:48 +00001147
Mike McCormack38d67a42005-08-22 09:15:23 +00001148 return comp;
Aric Stewartbdb29552004-07-04 00:32:48 +00001149}
1150
Aric Stewart04598242005-06-23 16:43:24 +00001151typedef struct {
1152 MSIPACKAGE *package;
Mike McCormack1da28582005-08-22 14:09:17 +00001153 MSIFEATURE *feature;
Aric Stewart04598242005-06-23 16:43:24 +00001154} _ilfs;
1155
Mike McCormack38d67a42005-08-22 09:15:23 +00001156static UINT add_feature_component( MSIFEATURE *feature, MSICOMPONENT *comp )
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001157{
1158 ComponentList *cl;
1159
Mike McCormackee034ba2005-09-20 11:59:14 +00001160 cl = msi_alloc( sizeof (*cl) );
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001161 if ( !cl )
1162 return ERROR_NOT_ENOUGH_MEMORY;
Mike McCormack38d67a42005-08-22 09:15:23 +00001163 cl->component = comp;
Mike McCormack1da28582005-08-22 14:09:17 +00001164 list_add_tail( &feature->Components, &cl->entry );
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001165
1166 return ERROR_SUCCESS;
1167}
1168
Mike McCormack38d67a42005-08-22 09:15:23 +00001169static UINT iterate_component_check( MSIRECORD *row, LPVOID param )
Aric Stewartbdb29552004-07-04 00:32:48 +00001170{
Aric Stewart04598242005-06-23 16:43:24 +00001171 _ilfs* ilfs= (_ilfs*)param;
Mike McCormack38d67a42005-08-22 09:15:23 +00001172 MSIPACKAGE *package = ilfs->package;
Mike McCormack1da28582005-08-22 14:09:17 +00001173 MSIFEATURE *feature = ilfs->feature;
Mike McCormack38d67a42005-08-22 09:15:23 +00001174 MSICOMPONENT *comp;
Aric Stewart04598242005-06-23 16:43:24 +00001175
Mike McCormack38d67a42005-08-22 09:15:23 +00001176 comp = load_component( row );
1177 if (!comp)
1178 return ERROR_FUNCTION_FAILED;
Aric Stewart04598242005-06-23 16:43:24 +00001179
Mike McCormack38d67a42005-08-22 09:15:23 +00001180 list_add_tail( &package->components, &comp->entry );
Mike McCormack1da28582005-08-22 14:09:17 +00001181 add_feature_component( feature, comp );
Mike McCormack38d67a42005-08-22 09:15:23 +00001182
1183 TRACE("Loaded new component %p\n", comp);
Aric Stewart04598242005-06-23 16:43:24 +00001184
1185 return ERROR_SUCCESS;
1186}
1187
1188static UINT iterate_load_featurecomponents(MSIRECORD *row, LPVOID param)
1189{
1190 _ilfs* ilfs= (_ilfs*)param;
1191 LPCWSTR component;
1192 DWORD rc;
Mike McCormack38d67a42005-08-22 09:15:23 +00001193 MSICOMPONENT *comp;
Aric Stewart04598242005-06-23 16:43:24 +00001194 MSIQUERY * view;
1195 static const WCHAR Query[] =
1196 {'S','E','L','E','C','T',' ','*',' ','F','R', 'O','M',' ',
1197 '`','C','o','m','p','o','n','e','n','t','`',' ',
1198 'W','H','E','R','E',' ',
1199 '`','C','o','m','p','o','n','e','n','t','`',' ',
1200 '=','\'','%','s','\'',0};
1201
1202 component = MSI_RecordGetString(row,1);
1203
1204 /* check to see if the component is already loaded */
Mike McCormack38d67a42005-08-22 09:15:23 +00001205 comp = get_loaded_component( ilfs->package, component );
1206 if (comp)
Aric Stewart04598242005-06-23 16:43:24 +00001207 {
Mike McCormack38d67a42005-08-22 09:15:23 +00001208 TRACE("Component %s already loaded\n", debugstr_w(component) );
Mike McCormack1da28582005-08-22 14:09:17 +00001209 add_feature_component( ilfs->feature, comp );
Aric Stewart04598242005-06-23 16:43:24 +00001210 return ERROR_SUCCESS;
1211 }
1212
1213 rc = MSI_OpenQuery(ilfs->package->db, &view, Query, component);
1214 if (rc != ERROR_SUCCESS)
1215 return ERROR_SUCCESS;
1216
Aric Stewart04598242005-06-23 16:43:24 +00001217 rc = MSI_IterateRecords(view, NULL, iterate_component_check, ilfs);
1218 msiobj_release( &view->hdr );
1219
1220 return ERROR_SUCCESS;
1221}
1222
1223static UINT load_feature(MSIRECORD * row, LPVOID param)
1224{
1225 MSIPACKAGE* package = (MSIPACKAGE*)param;
Mike McCormack1da28582005-08-22 14:09:17 +00001226 MSIFEATURE* feature;
Aric Stewart8e233e92005-03-01 11:45:19 +00001227 static const WCHAR Query1[] =
Aric Stewart98e38082005-05-20 09:40:42 +00001228 {'S','E','L','E','C','T',' ',
1229 '`','C','o','m','p','o','n','e','n','t','_','`',
1230 ' ','F','R','O','M',' ','`','F','e','a','t','u','r','e',
1231 'C','o','m','p','o','n','e','n','t','s','`',' ',
1232 'W','H','E','R','E',' ',
1233 '`','F','e', 'a','t','u','r','e','_','`',' ','=','\'','%','s','\'',0};
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001234 MSIQUERY * view;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001235 UINT rc;
Aric Stewart04598242005-06-23 16:43:24 +00001236 _ilfs ilfs;
1237
Aric Stewartbdb29552004-07-04 00:32:48 +00001238 /* fill in the data */
1239
Mike McCormackee034ba2005-09-20 11:59:14 +00001240 feature = msi_alloc_zero( sizeof (MSIFEATURE) );
Mike McCormack1da28582005-08-22 14:09:17 +00001241 if (!feature)
1242 return ERROR_NOT_ENOUGH_MEMORY;
Aric Stewartbdb29552004-07-04 00:32:48 +00001243
Mike McCormack1da28582005-08-22 14:09:17 +00001244 list_init( &feature->Components );
Aric Stewartbdb29552004-07-04 00:32:48 +00001245
Mike McCormack51c66182005-10-27 12:36:12 +00001246 feature->Feature = msi_dup_record_field( row, 1 );
Aric Stewartbdb29552004-07-04 00:32:48 +00001247
Mike McCormack1da28582005-08-22 14:09:17 +00001248 TRACE("Loading feature %s\n",debugstr_w(feature->Feature));
Aric Stewartbdb29552004-07-04 00:32:48 +00001249
Mike McCormack51c66182005-10-27 12:36:12 +00001250 feature->Feature_Parent = msi_dup_record_field( row, 2 );
1251 feature->Title = msi_dup_record_field( row, 3 );
1252 feature->Description = msi_dup_record_field( row, 4 );
Aric Stewartbdb29552004-07-04 00:32:48 +00001253
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001254 if (!MSI_RecordIsNull(row,5))
Mike McCormack1da28582005-08-22 14:09:17 +00001255 feature->Display = MSI_RecordGetInteger(row,5);
Aric Stewartbdb29552004-07-04 00:32:48 +00001256
Mike McCormack1da28582005-08-22 14:09:17 +00001257 feature->Level= MSI_RecordGetInteger(row,6);
Mike McCormack51c66182005-10-27 12:36:12 +00001258 feature->Directory = msi_dup_record_field( row, 7 );
Mike McCormack1da28582005-08-22 14:09:17 +00001259 feature->Attributes = MSI_RecordGetInteger(row,8);
Aric Stewartfbdd7092004-12-27 19:06:22 +00001260
Mike McCormack1da28582005-08-22 14:09:17 +00001261 feature->Installed = INSTALLSTATE_ABSENT;
1262 feature->Action = INSTALLSTATE_UNKNOWN;
1263 feature->ActionRequest = INSTALLSTATE_UNKNOWN;
1264
1265 list_add_tail( &package->features, &feature->entry );
Aric Stewartbdb29552004-07-04 00:32:48 +00001266
1267 /* load feature components */
1268
Mike McCormack1da28582005-08-22 14:09:17 +00001269 rc = MSI_OpenQuery( package->db, &view, Query1, feature->Feature );
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001270 if (rc != ERROR_SUCCESS)
Aric Stewart04598242005-06-23 16:43:24 +00001271 return ERROR_SUCCESS;
Aric Stewartbdb29552004-07-04 00:32:48 +00001272
Mike McCormack1da28582005-08-22 14:09:17 +00001273 ilfs.package = package;
1274 ilfs.feature = feature;
1275
Aric Stewart04598242005-06-23 16:43:24 +00001276 MSI_IterateRecords(view, NULL, iterate_load_featurecomponents , &ilfs);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001277 msiobj_release(&view->hdr);
Aric Stewart04598242005-06-23 16:43:24 +00001278
1279 return ERROR_SUCCESS;
Aric Stewartbdb29552004-07-04 00:32:48 +00001280}
1281
Aric Stewart04598242005-06-23 16:43:24 +00001282static UINT load_file(MSIRECORD *row, LPVOID param)
Aric Stewartc5a14432005-05-18 17:46:12 +00001283{
Aric Stewart04598242005-06-23 16:43:24 +00001284 MSIPACKAGE* package = (MSIPACKAGE*)param;
Aric Stewart09b0aba2005-06-09 20:30:59 +00001285 LPCWSTR component;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001286 MSIFILE *file;
Aric Stewartc5a14432005-05-18 17:46:12 +00001287
1288 /* fill in the data */
1289
Mike McCormackee034ba2005-09-20 11:59:14 +00001290 file = msi_alloc_zero( sizeof (MSIFILE) );
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001291 if (!file)
1292 return ERROR_NOT_ENOUGH_MEMORY;
Aric Stewartc5a14432005-05-18 17:46:12 +00001293
Mike McCormack51c66182005-10-27 12:36:12 +00001294 file->File = msi_dup_record_field( row, 1 );
Aric Stewartc5a14432005-05-18 17:46:12 +00001295
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001296 component = MSI_RecordGetString( row, 2 );
1297 file->Component = get_loaded_component( package, component );
Aric Stewart09b0aba2005-06-09 20:30:59 +00001298
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001299 if (!file->Component)
Aric Stewart09b0aba2005-06-09 20:30:59 +00001300 ERR("Unfound Component %s\n",debugstr_w(component));
Aric Stewartc5a14432005-05-18 17:46:12 +00001301
Mike McCormack51c66182005-10-27 12:36:12 +00001302 file->FileName = msi_dup_record_field( row, 3 );
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001303 reduce_to_longfilename( file->FileName );
Aric Stewartc5a14432005-05-18 17:46:12 +00001304
Mike McCormack51c66182005-10-27 12:36:12 +00001305 file->ShortName = msi_dup_record_field( row, 3 );
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001306 reduce_to_shortfilename( file->ShortName );
Aric Stewartc5a14432005-05-18 17:46:12 +00001307
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001308 file->FileSize = MSI_RecordGetInteger( row, 4 );
Mike McCormack51c66182005-10-27 12:36:12 +00001309 file->Version = msi_dup_record_field( row, 5 );
1310 file->Language = msi_dup_record_field( row, 6 );
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001311 file->Attributes = MSI_RecordGetInteger( row, 7 );
1312 file->Sequence = MSI_RecordGetInteger( row, 8 );
Aric Stewartc5a14432005-05-18 17:46:12 +00001313
Mike McCormackdded8fb2005-11-02 10:56:42 +00001314 file->state = msifs_invalid;
Aric Stewartc5a14432005-05-18 17:46:12 +00001315
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001316 TRACE("File Loaded (%s)\n",debugstr_w(file->File));
1317
1318 list_add_tail( &package->files, &file->entry );
Aric Stewartc5a14432005-05-18 17:46:12 +00001319
1320 return ERROR_SUCCESS;
1321}
1322
1323static UINT load_all_files(MSIPACKAGE *package)
1324{
1325 MSIQUERY * view;
Aric Stewartc5a14432005-05-18 17:46:12 +00001326 UINT rc;
1327 static const WCHAR Query[] =
1328 {'S','E','L','E','C','T',' ','*',' ', 'F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001329 '`','F','i','l','e','`',' ', 'O','R','D','E','R',' ','B','Y',' ',
1330 '`','S','e','q','u','e','n','c','e','`', 0};
Aric Stewartc5a14432005-05-18 17:46:12 +00001331
1332 if (!package)
1333 return ERROR_INVALID_HANDLE;
1334
1335 rc = MSI_DatabaseOpenViewW(package->db, Query, &view);
1336 if (rc != ERROR_SUCCESS)
1337 return ERROR_SUCCESS;
Aric Stewartc5a14432005-05-18 17:46:12 +00001338
Aric Stewart04598242005-06-23 16:43:24 +00001339 rc = MSI_IterateRecords(view, NULL, load_file, package);
Aric Stewartc5a14432005-05-18 17:46:12 +00001340 msiobj_release(&view->hdr);
1341
1342 return ERROR_SUCCESS;
1343}
1344
1345
Aric Stewartbdb29552004-07-04 00:32:48 +00001346/*
1347 * I am not doing any of the costing functionality yet.
1348 * Mostly looking at doing the Component and Feature loading
1349 *
Francois Gougetda8b3dd2005-01-26 21:09:04 +00001350 * The native MSI does A LOT of modification to tables here. Mostly adding
1351 * a lot of temporary columns to the Feature and Component tables.
Aric Stewartbdb29552004-07-04 00:32:48 +00001352 *
Francois Gougetda8b3dd2005-01-26 21:09:04 +00001353 * note: Native msi also tracks the short filename. But I am only going to
Aric Stewartbdb29552004-07-04 00:32:48 +00001354 * track the long ones. Also looking at this directory table
1355 * it appears that the directory table does not get the parents
Francois Gougetda8b3dd2005-01-26 21:09:04 +00001356 * resolved base on property only based on their entries in the
Aric Stewartbdb29552004-07-04 00:32:48 +00001357 * directory table.
1358 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001359static UINT ACTION_CostInitialize(MSIPACKAGE *package)
Aric Stewartbdb29552004-07-04 00:32:48 +00001360{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001361 MSIQUERY * view;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001362 UINT rc;
Aric Stewart8e233e92005-03-01 11:45:19 +00001363 static const WCHAR Query_all[] =
1364 {'S','E','L','E','C','T',' ','*',' ', 'F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001365 '`','F','e','a','t','u','r','e','`',0};
Aric Stewart8e233e92005-03-01 11:45:19 +00001366 static const WCHAR szCosting[] =
1367 {'C','o','s','t','i','n','g','C','o','m','p','l','e','t','e',0 };
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001368 static const WCHAR szZero[] = { '0', 0 };
Aric Stewartbdb29552004-07-04 00:32:48 +00001369
Mike McCormack9293f862005-10-30 19:16:45 +00001370 if ( 1 == msi_get_property_int( package, szCosting, 0 ) )
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001371 return ERROR_SUCCESS;
1372
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001373 MSI_SetPropertyW(package, szCosting, szZero);
Juan Lang014ad3b2005-03-01 10:41:52 +00001374 MSI_SetPropertyW(package, cszRootDrive , c_colon);
Aric Stewartbdb29552004-07-04 00:32:48 +00001375
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001376 rc = MSI_DatabaseOpenViewW(package->db,Query_all,&view);
1377 if (rc != ERROR_SUCCESS)
1378 return rc;
Aric Stewartbdb29552004-07-04 00:32:48 +00001379
Aric Stewart04598242005-06-23 16:43:24 +00001380 rc = MSI_IterateRecords(view, NULL, load_feature, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001381 msiobj_release(&view->hdr);
Aric Stewartec688fb2004-07-04 00:35:52 +00001382
Aric Stewartc5a14432005-05-18 17:46:12 +00001383 load_all_files(package);
Aric Stewartec688fb2004-07-04 00:35:52 +00001384
Aric Stewartec688fb2004-07-04 00:35:52 +00001385 return ERROR_SUCCESS;
1386}
1387
Mike McCormackf9acfe62005-06-07 20:29:51 +00001388static UINT execute_script(MSIPACKAGE *package, UINT script )
Aric Stewart9cd707d2005-05-27 19:24:22 +00001389{
1390 int i;
1391 UINT rc = ERROR_SUCCESS;
1392
1393 TRACE("Executing Script %i\n",script);
1394
1395 for (i = 0; i < package->script->ActionCount[script]; i++)
1396 {
1397 LPWSTR action;
1398 action = package->script->Actions[script][i];
1399 ui_actionstart(package, action);
1400 TRACE("Executing Action (%s)\n",debugstr_w(action));
1401 rc = ACTION_PerformAction(package, action, TRUE);
Mike McCormackee034ba2005-09-20 11:59:14 +00001402 msi_free(package->script->Actions[script][i]);
Aric Stewart9cd707d2005-05-27 19:24:22 +00001403 if (rc != ERROR_SUCCESS)
1404 break;
1405 }
Mike McCormackee034ba2005-09-20 11:59:14 +00001406 msi_free(package->script->Actions[script]);
Aric Stewart9cd707d2005-05-27 19:24:22 +00001407
1408 package->script->ActionCount[script] = 0;
1409 package->script->Actions[script] = NULL;
1410 return rc;
1411}
1412
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001413static UINT ACTION_FileCost(MSIPACKAGE *package)
Aric Stewartec688fb2004-07-04 00:35:52 +00001414{
Aric Stewartbdb29552004-07-04 00:32:48 +00001415 return ERROR_SUCCESS;
1416}
1417
Aric Stewartc5a14432005-05-18 17:46:12 +00001418
Mike McCormack979511f2005-08-23 18:15:44 +00001419static MSIFOLDER *load_folder( MSIPACKAGE *package, LPCWSTR dir )
Aric Stewartbdb29552004-07-04 00:32:48 +00001420{
Mike McCormackba8200b2004-12-22 15:25:30 +00001421 static const WCHAR Query[] =
Aric Stewart8e233e92005-03-01 11:45:19 +00001422 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001423 '`','D','i','r','e','c', 't','o','r','y','`',' ',
1424 'W','H','E','R','E',' ', '`', 'D','i','r','e','c','t', 'o','r','y','`',
1425 ' ','=',' ','\'','%','s','\'',
Aric Stewart8e233e92005-03-01 11:45:19 +00001426 0};
Aric Stewart09b0aba2005-06-09 20:30:59 +00001427 LPWSTR ptargetdir, targetdir, srcdir;
1428 LPCWSTR parent;
Aric Stewart6186b2b2005-05-16 21:37:35 +00001429 LPWSTR shortname = NULL;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001430 MSIRECORD * row = 0;
Mike McCormack979511f2005-08-23 18:15:44 +00001431 MSIFOLDER *folder;
Aric Stewarte95136b2004-06-29 03:44:01 +00001432
Aric Stewartbdb29552004-07-04 00:32:48 +00001433 TRACE("Looking for dir %s\n",debugstr_w(dir));
Aric Stewart401bd3f2004-06-28 20:34:35 +00001434
Mike McCormack979511f2005-08-23 18:15:44 +00001435 folder = get_loaded_folder( package, dir );
1436 if (folder)
1437 return folder;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001438
Aric Stewartbdb29552004-07-04 00:32:48 +00001439 TRACE("Working to load %s\n",debugstr_w(dir));
1440
Mike McCormackee034ba2005-09-20 11:59:14 +00001441 folder = msi_alloc_zero( sizeof (MSIFOLDER) );
Mike McCormack979511f2005-08-23 18:15:44 +00001442 if (!folder)
1443 return NULL;
Aric Stewartbdb29552004-07-04 00:32:48 +00001444
Mike McCormack979511f2005-08-23 18:15:44 +00001445 folder->Directory = strdupW(dir);
Aric Stewartbdb29552004-07-04 00:32:48 +00001446
Mike McCormack0b352c72005-06-02 10:29:57 +00001447 row = MSI_QueryGetRecord(package->db, Query, dir);
1448 if (!row)
Mike McCormack979511f2005-08-23 18:15:44 +00001449 return NULL;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001450
Mike McCormack51c66182005-10-27 12:36:12 +00001451 ptargetdir = targetdir = msi_dup_record_field(row,3);
Aric Stewart401bd3f2004-06-28 20:34:35 +00001452
1453 /* split src and target dir */
1454 if (strchrW(targetdir,':'))
1455 {
1456 srcdir=strchrW(targetdir,':');
1457 *srcdir=0;
1458 srcdir ++;
1459 }
1460 else
1461 srcdir=NULL;
1462
1463 /* for now only pick long filename versions */
1464 if (strchrW(targetdir,'|'))
1465 {
Aric Stewart6186b2b2005-05-16 21:37:35 +00001466 shortname = targetdir;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001467 targetdir = strchrW(targetdir,'|');
1468 *targetdir = 0;
1469 targetdir ++;
1470 }
Aric Stewart6186b2b2005-05-16 21:37:35 +00001471 /* for the sourcedir pick the short filename */
Aric Stewart401bd3f2004-06-28 20:34:35 +00001472 if (srcdir && strchrW(srcdir,'|'))
1473 {
Aric Stewart6186b2b2005-05-16 21:37:35 +00001474 LPWSTR p = strchrW(srcdir,'|');
1475 *p = 0;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001476 }
1477
1478 /* now check for root dirs */
1479 if (targetdir[0] == '.' && targetdir[1] == 0)
1480 targetdir = NULL;
1481
Mike McCormackba8200b2004-12-22 15:25:30 +00001482 if (targetdir)
Aric Stewart401bd3f2004-06-28 20:34:35 +00001483 {
Mike McCormackba8200b2004-12-22 15:25:30 +00001484 TRACE(" TargetDefault = %s\n",debugstr_w(targetdir));
Mike McCormackee034ba2005-09-20 11:59:14 +00001485 msi_free( folder->TargetDefault);
Mike McCormack979511f2005-08-23 18:15:44 +00001486 folder->TargetDefault = strdupW(targetdir);
Aric Stewart401bd3f2004-06-28 20:34:35 +00001487 }
1488
Mike McCormackba8200b2004-12-22 15:25:30 +00001489 if (srcdir)
Mike McCormack979511f2005-08-23 18:15:44 +00001490 folder->SourceDefault = strdupW(srcdir);
Aric Stewart6186b2b2005-05-16 21:37:35 +00001491 else if (shortname)
Mike McCormack979511f2005-08-23 18:15:44 +00001492 folder->SourceDefault = strdupW(shortname);
Mike McCormackba8200b2004-12-22 15:25:30 +00001493 else if (targetdir)
Mike McCormack979511f2005-08-23 18:15:44 +00001494 folder->SourceDefault = strdupW(targetdir);
Mike McCormackee034ba2005-09-20 11:59:14 +00001495 msi_free(ptargetdir);
Mike McCormack979511f2005-08-23 18:15:44 +00001496 TRACE(" SourceDefault = %s\n", debugstr_w( folder->SourceDefault ));
Mike McCormackba8200b2004-12-22 15:25:30 +00001497
Aric Stewart09b0aba2005-06-09 20:30:59 +00001498 parent = MSI_RecordGetString(row,2);
Mike McCormackba8200b2004-12-22 15:25:30 +00001499 if (parent)
Aric Stewart401bd3f2004-06-28 20:34:35 +00001500 {
Mike McCormack979511f2005-08-23 18:15:44 +00001501 folder->Parent = load_folder( package, parent );
1502 if ( folder->Parent )
1503 TRACE("loaded parent %p %s\n", folder->Parent,
1504 debugstr_w(folder->Parent->Directory));
1505 else
1506 ERR("failed to load parent folder %s\n", debugstr_w(parent));
Aric Stewartbdb29552004-07-04 00:32:48 +00001507 }
Aric Stewart401bd3f2004-06-28 20:34:35 +00001508
Mike McCormack062ad502005-09-15 15:04:08 +00001509 folder->Property = msi_dup_property( package, dir );
Aric Stewart401bd3f2004-06-28 20:34:35 +00001510
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001511 msiobj_release(&row->hdr);
Mike McCormack979511f2005-08-23 18:15:44 +00001512
1513 list_add_tail( &package->folders, &folder->entry );
1514
1515 TRACE("%s returning %p\n",debugstr_w(dir),folder);
1516
1517 return folder;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001518}
1519
Aric Stewart78a04e32005-02-22 15:47:00 +00001520/* scan for and update current install states */
Mike McCormackf9acfe62005-06-07 20:29:51 +00001521static void ACTION_UpdateInstallStates(MSIPACKAGE *package)
Aric Stewart78a04e32005-02-22 15:47:00 +00001522{
Mike McCormack38d67a42005-08-22 09:15:23 +00001523 MSICOMPONENT *comp;
Mike McCormack1da28582005-08-22 14:09:17 +00001524 MSIFEATURE *feature;
Aric Stewart78a04e32005-02-22 15:47:00 +00001525
Mike McCormack38d67a42005-08-22 09:15:23 +00001526 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
Aric Stewart78a04e32005-02-22 15:47:00 +00001527 {
1528 INSTALLSTATE res;
Mike McCormack38d67a42005-08-22 09:15:23 +00001529 res = MsiGetComponentPathW( package->ProductCode,
1530 comp->ComponentId, NULL, NULL);
Aric Stewart78a04e32005-02-22 15:47:00 +00001531 if (res < 0)
1532 res = INSTALLSTATE_ABSENT;
Mike McCormack38d67a42005-08-22 09:15:23 +00001533 comp->Installed = res;
Aric Stewart78a04e32005-02-22 15:47:00 +00001534 }
1535
Mike McCormack1da28582005-08-22 14:09:17 +00001536 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewart78a04e32005-02-22 15:47:00 +00001537 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001538 ComponentList *cl;
Aric Stewart78a04e32005-02-22 15:47:00 +00001539 INSTALLSTATE res = -10;
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001540
Mike McCormack1da28582005-08-22 14:09:17 +00001541 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewart78a04e32005-02-22 15:47:00 +00001542 {
Mike McCormack38d67a42005-08-22 09:15:23 +00001543 comp= cl->component;
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001544
Aric Stewart78a04e32005-02-22 15:47:00 +00001545 if (res == -10)
Mike McCormack38d67a42005-08-22 09:15:23 +00001546 res = comp->Installed;
Aric Stewart78a04e32005-02-22 15:47:00 +00001547 else
1548 {
Mike McCormack38d67a42005-08-22 09:15:23 +00001549 if (res == comp->Installed)
Aric Stewart78a04e32005-02-22 15:47:00 +00001550 continue;
1551
Mike McCormack38d67a42005-08-22 09:15:23 +00001552 if (res != comp->Installed)
Aric Stewart78a04e32005-02-22 15:47:00 +00001553 res = INSTALLSTATE_INCOMPLETE;
1554 }
1555 }
Mike McCormack8aa1a912005-08-25 19:19:10 +00001556 feature->Installed = res;
Aric Stewart78a04e32005-02-22 15:47:00 +00001557 }
1558}
1559
Aric Stewart78a04e32005-02-22 15:47:00 +00001560static BOOL process_state_property (MSIPACKAGE* package, LPCWSTR property,
1561 INSTALLSTATE state)
Aric Stewartae1aa322004-12-27 19:02:59 +00001562{
Aric Stewartae1aa322004-12-27 19:02:59 +00001563 static const WCHAR all[]={'A','L','L',0};
Mike McCormack72faac02005-09-08 11:03:35 +00001564 LPWSTR override;
Mike McCormack1da28582005-08-22 14:09:17 +00001565 MSIFEATURE *feature;
Aric Stewartae1aa322004-12-27 19:02:59 +00001566
Mike McCormack062ad502005-09-15 15:04:08 +00001567 override = msi_dup_property( package, property );
Mike McCormack72faac02005-09-08 11:03:35 +00001568 if (!override)
1569 return FALSE;
1570
1571 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartae1aa322004-12-27 19:02:59 +00001572 {
Mike McCormack72faac02005-09-08 11:03:35 +00001573 if (strcmpiW(override,all)==0)
Aric Stewartfbdd7092004-12-27 19:06:22 +00001574 {
Mike McCormack72faac02005-09-08 11:03:35 +00001575 feature->ActionRequest= state;
1576 feature->Action = state;
1577 }
1578 else
1579 {
1580 LPWSTR ptr = override;
1581 LPWSTR ptr2 = strchrW(override,',');
Aric Stewartd900b532004-12-27 19:12:35 +00001582
Mike McCormack72faac02005-09-08 11:03:35 +00001583 while (ptr)
1584 {
1585 if ((ptr2 && strncmpW(ptr,feature->Feature, ptr2-ptr)==0)
1586 || (!ptr2 && strcmpW(ptr,feature->Feature)==0))
Aric Stewartd900b532004-12-27 19:12:35 +00001587 {
Mike McCormack72faac02005-09-08 11:03:35 +00001588 feature->ActionRequest= state;
1589 feature->Action = state;
1590 break;
Aric Stewartd900b532004-12-27 19:12:35 +00001591 }
Mike McCormack72faac02005-09-08 11:03:35 +00001592 if (ptr2)
1593 {
1594 ptr=ptr2+1;
1595 ptr2 = strchrW(ptr,',');
1596 }
1597 else
1598 break;
Aric Stewartd900b532004-12-27 19:12:35 +00001599 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001600 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001601 }
Mike McCormackee034ba2005-09-20 11:59:14 +00001602 msi_free(override);
Aric Stewart78a04e32005-02-22 15:47:00 +00001603
Mike McCormack72faac02005-09-08 11:03:35 +00001604 return TRUE;
Aric Stewart78a04e32005-02-22 15:47:00 +00001605}
1606
1607static UINT SetFeatureStates(MSIPACKAGE *package)
1608{
Mike McCormack74f0de92005-09-29 10:32:39 +00001609 int install_level;
Aric Stewart8e233e92005-03-01 11:45:19 +00001610 static const WCHAR szlevel[] =
1611 {'I','N','S','T','A','L','L','L','E','V','E','L',0};
1612 static const WCHAR szAddLocal[] =
1613 {'A','D','D','L','O','C','A','L',0};
1614 static const WCHAR szRemove[] =
1615 {'R','E','M','O','V','E',0};
Aric Stewartd5655f92005-11-02 14:21:17 +00001616 static const WCHAR szReinstall[] =
1617 {'R','E','I','N','S','T','A','L','L',0};
Aric Stewart78a04e32005-02-22 15:47:00 +00001618 BOOL override = FALSE;
Mike McCormack38d67a42005-08-22 09:15:23 +00001619 MSICOMPONENT* component;
Mike McCormack1da28582005-08-22 14:09:17 +00001620 MSIFEATURE *feature;
1621
Aric Stewart78a04e32005-02-22 15:47:00 +00001622
1623 /* I do not know if this is where it should happen.. but */
1624
1625 TRACE("Checking Install Level\n");
1626
Mike McCormack74f0de92005-09-29 10:32:39 +00001627 install_level = msi_get_property_int( package, szlevel, 1 );
Aric Stewart78a04e32005-02-22 15:47:00 +00001628
1629 /* ok hereis the _real_ rub
Francois Gougetfbb33432005-03-02 13:53:50 +00001630 * all these activation/deactivation things happen in order and things
1631 * later on the list override things earlier on the list.
Aric Stewart78a04e32005-02-22 15:47:00 +00001632 * 1) INSTALLLEVEL processing
1633 * 2) ADDLOCAL
1634 * 3) REMOVE
1635 * 4) ADDSOURCE
1636 * 5) ADDDEFAULT
1637 * 6) REINSTALL
1638 * 7) COMPADDLOCAL
1639 * 8) COMPADDSOURCE
1640 * 9) FILEADDLOCAL
1641 * 10) FILEADDSOURCE
1642 * 11) FILEADDDEFAULT
Francois Gougetfbb33432005-03-02 13:53:50 +00001643 * I have confirmed that if ADDLOCAL is stated then the INSTALLLEVEL is
1644 * ignored for all the features. seems strange, especially since it is not
Aric Stewart78a04e32005-02-22 15:47:00 +00001645 * documented anywhere, but it is how it works.
1646 *
Francois Gougetfbb33432005-03-02 13:53:50 +00001647 * I am still ignoring a lot of these. But that is ok for now, ADDLOCAL and
1648 * REMOVE are the big ones, since we don't handle administrative installs
1649 * yet anyway.
Aric Stewart78a04e32005-02-22 15:47:00 +00001650 */
1651 override |= process_state_property(package,szAddLocal,INSTALLSTATE_LOCAL);
1652 override |= process_state_property(package,szRemove,INSTALLSTATE_ABSENT);
Aric Stewartd5655f92005-11-02 14:21:17 +00001653 override |= process_state_property(package,szReinstall,INSTALLSTATE_LOCAL);
Aric Stewart78a04e32005-02-22 15:47:00 +00001654
1655 if (!override)
Aric Stewartfbdd7092004-12-27 19:06:22 +00001656 {
Mike McCormack1da28582005-08-22 14:09:17 +00001657 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartfbdd7092004-12-27 19:06:22 +00001658 {
Mike McCormack1da28582005-08-22 14:09:17 +00001659 BOOL feature_state = ((feature->Level > 0) &&
1660 (feature->Level <= install_level));
Aric Stewartae1aa322004-12-27 19:02:59 +00001661
Mike McCormack1da28582005-08-22 14:09:17 +00001662 if ((feature_state) && (feature->Action == INSTALLSTATE_UNKNOWN))
Aric Stewartfbdd7092004-12-27 19:06:22 +00001663 {
Mike McCormack1da28582005-08-22 14:09:17 +00001664 if (feature->Attributes & msidbFeatureAttributesFavorSource)
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001665 {
Mike McCormack1da28582005-08-22 14:09:17 +00001666 feature->ActionRequest = INSTALLSTATE_SOURCE;
1667 feature->Action = INSTALLSTATE_SOURCE;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001668 }
Mike McCormack1da28582005-08-22 14:09:17 +00001669 else if (feature->Attributes & msidbFeatureAttributesFavorAdvertise)
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001670 {
Mike McCormack1da28582005-08-22 14:09:17 +00001671 feature->ActionRequest = INSTALLSTATE_ADVERTISED;
1672 feature->Action = INSTALLSTATE_ADVERTISED;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001673 }
1674 else
1675 {
Mike McCormack1da28582005-08-22 14:09:17 +00001676 feature->ActionRequest = INSTALLSTATE_LOCAL;
1677 feature->Action = INSTALLSTATE_LOCAL;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001678 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001679 }
Aric Stewartae1aa322004-12-27 19:02:59 +00001680 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001681 }
Aric Stewart6999a042005-06-08 19:20:02 +00001682 else
1683 {
1684 /* set the Preselected Property */
1685 static const WCHAR szPreselected[] = {'P','r','e','s','e','l','e','c','t','e','d',0};
1686 static const WCHAR szOne[] = { '1', 0 };
1687
1688 MSI_SetPropertyW(package,szPreselected,szOne);
1689 }
Aric Stewartae1aa322004-12-27 19:02:59 +00001690
Aric Stewartfbdd7092004-12-27 19:06:22 +00001691 /*
1692 * now we want to enable or disable components base on feature
1693 */
1694
Mike McCormack1da28582005-08-22 14:09:17 +00001695 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartfbdd7092004-12-27 19:06:22 +00001696 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001697 ComponentList *cl;
1698
Aric Stewartfbdd7092004-12-27 19:06:22 +00001699 TRACE("Examining Feature %s (Installed %i, Action %i, Request %i)\n",
1700 debugstr_w(feature->Feature), feature->Installed, feature->Action,
1701 feature->ActionRequest);
1702
Mike McCormack1da28582005-08-22 14:09:17 +00001703 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewartae1aa322004-12-27 19:02:59 +00001704 {
Mike McCormack38d67a42005-08-22 09:15:23 +00001705 component = cl->component;
Aric Stewartfbdd7092004-12-27 19:06:22 +00001706
1707 if (!component->Enabled)
1708 {
Aric Stewart78a04e32005-02-22 15:47:00 +00001709 component->Action = INSTALLSTATE_UNKNOWN;
1710 component->ActionRequest = INSTALLSTATE_UNKNOWN;
Aric Stewartfbdd7092004-12-27 19:06:22 +00001711 }
1712 else
1713 {
1714 if (feature->Action == INSTALLSTATE_LOCAL)
Aric Stewart78a04e32005-02-22 15:47:00 +00001715 {
Aric Stewartfbdd7092004-12-27 19:06:22 +00001716 component->Action = INSTALLSTATE_LOCAL;
Aric Stewartfbdd7092004-12-27 19:06:22 +00001717 component->ActionRequest = INSTALLSTATE_LOCAL;
Aric Stewart78a04e32005-02-22 15:47:00 +00001718 }
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001719 else if (feature->ActionRequest == INSTALLSTATE_SOURCE)
1720 {
1721 if ((component->Action == INSTALLSTATE_UNKNOWN) ||
1722 (component->Action == INSTALLSTATE_ABSENT) ||
1723 (component->Action == INSTALLSTATE_ADVERTISED))
1724
1725 {
1726 component->Action = INSTALLSTATE_SOURCE;
1727 component->ActionRequest = INSTALLSTATE_SOURCE;
1728 }
1729 }
1730 else if (feature->ActionRequest == INSTALLSTATE_ADVERTISED)
1731 {
1732 if ((component->Action == INSTALLSTATE_UNKNOWN) ||
1733 (component->Action == INSTALLSTATE_ABSENT))
1734
1735 {
1736 component->Action = INSTALLSTATE_ADVERTISED;
1737 component->ActionRequest = INSTALLSTATE_ADVERTISED;
1738 }
1739 }
Aric Stewart78a04e32005-02-22 15:47:00 +00001740 else if (feature->ActionRequest == INSTALLSTATE_ABSENT)
1741 {
1742 if (component->Action == INSTALLSTATE_UNKNOWN)
1743 {
1744 component->Action = INSTALLSTATE_ABSENT;
1745 component->ActionRequest = INSTALLSTATE_ABSENT;
1746 }
1747 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001748 }
Aric Stewartae1aa322004-12-27 19:02:59 +00001749 }
1750 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001751
Mike McCormack38d67a42005-08-22 09:15:23 +00001752 LIST_FOR_EACH_ENTRY( component, &package->components, MSICOMPONENT, entry )
Aric Stewartfbdd7092004-12-27 19:06:22 +00001753 {
Aric Stewartfbdd7092004-12-27 19:06:22 +00001754 TRACE("Result: Component %s (Installed %i, Action %i, Request %i)\n",
1755 debugstr_w(component->Component), component->Installed,
1756 component->Action, component->ActionRequest);
1757 }
1758
1759
Aric Stewartae1aa322004-12-27 19:02:59 +00001760 return ERROR_SUCCESS;
1761}
1762
Aric Stewart443ad4d2005-06-21 20:50:12 +00001763static UINT ITERATE_CostFinalizeDirectories(MSIRECORD *row, LPVOID param)
1764{
1765 MSIPACKAGE *package = (MSIPACKAGE*)param;
1766 LPCWSTR name;
1767 LPWSTR path;
1768
1769 name = MSI_RecordGetString(row,1);
1770
1771 /* This helper function now does ALL the work */
1772 TRACE("Dir %s ...\n",debugstr_w(name));
1773 load_folder(package,name);
1774 path = resolve_folder(package,name,FALSE,TRUE,NULL);
1775 TRACE("resolves to %s\n",debugstr_w(path));
Mike McCormackee034ba2005-09-20 11:59:14 +00001776 msi_free(path);
Aric Stewart443ad4d2005-06-21 20:50:12 +00001777
1778 return ERROR_SUCCESS;
1779}
1780
1781static UINT ITERATE_CostFinalizeConditions(MSIRECORD *row, LPVOID param)
1782{
1783 MSIPACKAGE *package = (MSIPACKAGE*)param;
Mike McCormack1da28582005-08-22 14:09:17 +00001784 LPCWSTR name;
1785 MSIFEATURE *feature;
Aric Stewart443ad4d2005-06-21 20:50:12 +00001786
Mike McCormack1da28582005-08-22 14:09:17 +00001787 name = MSI_RecordGetString( row, 1 );
Aric Stewart443ad4d2005-06-21 20:50:12 +00001788
Mike McCormack1da28582005-08-22 14:09:17 +00001789 feature = get_loaded_feature( package, name );
1790 if (!feature)
1791 ERR("FAILED to find loaded feature %s\n",debugstr_w(name));
Aric Stewart443ad4d2005-06-21 20:50:12 +00001792 else
1793 {
1794 LPCWSTR Condition;
1795 Condition = MSI_RecordGetString(row,3);
1796
Aric Stewart0713f092005-06-24 11:51:29 +00001797 if (MSI_EvaluateConditionW(package,Condition) == MSICONDITION_TRUE)
Aric Stewart443ad4d2005-06-21 20:50:12 +00001798 {
1799 int level = MSI_RecordGetInteger(row,2);
Mike McCormack1da28582005-08-22 14:09:17 +00001800 TRACE("Reseting feature %s to level %i\n", debugstr_w(name), level);
1801 feature->Level = level;
Aric Stewart443ad4d2005-06-21 20:50:12 +00001802 }
1803 }
1804 return ERROR_SUCCESS;
1805}
1806
1807
Aric Stewart401bd3f2004-06-28 20:34:35 +00001808/*
Francois Gougetda8b3dd2005-01-26 21:09:04 +00001809 * A lot is done in this function aside from just the costing.
Mike McCormack6e2bca32004-07-04 00:25:00 +00001810 * The costing needs to be implemented at some point but for now I am going
Aric Stewart401bd3f2004-06-28 20:34:35 +00001811 * to focus on the directory building
1812 *
Aric Stewart401bd3f2004-06-28 20:34:35 +00001813 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001814static UINT ACTION_CostFinalize(MSIPACKAGE *package)
Aric Stewart401bd3f2004-06-28 20:34:35 +00001815{
Aric Stewart8e233e92005-03-01 11:45:19 +00001816 static const WCHAR ExecSeqQuery[] =
1817 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001818 '`','D','i','r','e','c','t','o','r','y','`',0};
Aric Stewart8e233e92005-03-01 11:45:19 +00001819 static const WCHAR ConditionQuery[] =
1820 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001821 '`','C','o','n','d','i','t','i','o','n','`',0};
Aric Stewart8e233e92005-03-01 11:45:19 +00001822 static const WCHAR szCosting[] =
1823 {'C','o','s','t','i','n','g','C','o','m','p','l','e','t','e',0 };
1824 static const WCHAR szlevel[] =
1825 {'I','N','S','T','A','L','L','L','E','V','E','L',0};
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001826 static const WCHAR szOne[] = { '1', 0 };
Mike McCormack38d67a42005-08-22 09:15:23 +00001827 MSICOMPONENT *comp;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001828 MSIFILE *file;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001829 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001830 MSIQUERY * view;
Aric Stewart8cc14a92004-12-27 18:56:30 +00001831 LPWSTR level;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001832
Mike McCormack9293f862005-10-30 19:16:45 +00001833 if ( 1 == msi_get_property_int( package, szCosting, 0 ) )
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001834 return ERROR_SUCCESS;
Mike McCormack9293f862005-10-30 19:16:45 +00001835
Aric Stewart401bd3f2004-06-28 20:34:35 +00001836 TRACE("Building Directory properties\n");
1837
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001838 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
Aric Stewart84837d92004-07-20 01:22:37 +00001839 if (rc == ERROR_SUCCESS)
Aric Stewart401bd3f2004-06-28 20:34:35 +00001840 {
Aric Stewart443ad4d2005-06-21 20:50:12 +00001841 rc = MSI_IterateRecords(view, NULL, ITERATE_CostFinalizeDirectories,
1842 package);
Aric Stewart84837d92004-07-20 01:22:37 +00001843 msiobj_release(&view->hdr);
1844 }
Aric Stewart401bd3f2004-06-28 20:34:35 +00001845
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001846 TRACE("File calculations\n");
Aric Stewartec688fb2004-07-04 00:35:52 +00001847
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001848 LIST_FOR_EACH_ENTRY( file, &package->files, MSIFILE, entry )
Aric Stewartec688fb2004-07-04 00:35:52 +00001849 {
Mike McCormackf11c8b02005-09-09 14:48:51 +00001850 MSICOMPONENT* comp = file->Component;
1851 LPWSTR p;
Aric Stewartec688fb2004-07-04 00:35:52 +00001852
Mike McCormackf11c8b02005-09-09 14:48:51 +00001853 if (!comp)
1854 continue;
Aric Stewartec688fb2004-07-04 00:35:52 +00001855
Mike McCormackf11c8b02005-09-09 14:48:51 +00001856 /* calculate target */
1857 p = resolve_folder(package, comp->Directory, FALSE, FALSE, NULL);
1858
Mike McCormackee034ba2005-09-20 11:59:14 +00001859 msi_free(file->TargetPath);
Mike McCormackf11c8b02005-09-09 14:48:51 +00001860
1861 TRACE("file %s is named %s\n",
1862 debugstr_w(file->File),debugstr_w(file->FileName));
1863
1864 file->TargetPath = build_directory_name(2, p, file->FileName);
1865
Mike McCormackee034ba2005-09-20 11:59:14 +00001866 msi_free(p);
Mike McCormackf11c8b02005-09-09 14:48:51 +00001867
1868 TRACE("file %s resolves to %s\n",
1869 debugstr_w(file->File),debugstr_w(file->TargetPath));
1870
1871 if (GetFileAttributesW(file->TargetPath) == INVALID_FILE_ATTRIBUTES)
Aric Stewartec688fb2004-07-04 00:35:52 +00001872 {
Mike McCormackdded8fb2005-11-02 10:56:42 +00001873 file->state = msifs_missing;
Mike McCormackf11c8b02005-09-09 14:48:51 +00001874 comp->Cost += file->FileSize;
1875 continue;
1876 }
Mike McCormackba8200b2004-12-22 15:25:30 +00001877
Mike McCormackf11c8b02005-09-09 14:48:51 +00001878 if (file->Version)
1879 {
1880 DWORD handle;
1881 DWORD versize;
1882 UINT sz;
1883 LPVOID version;
Ivan Leo Puoti5b22e4d2005-11-03 11:35:11 +00001884 static WCHAR name[] = {'\\',0};
Mike McCormackf11c8b02005-09-09 14:48:51 +00001885 static const WCHAR name_fmt[] =
1886 {'%','u','.','%','u','.','%','u','.','%','u',0};
1887 WCHAR filever[0x100];
1888 VS_FIXEDFILEINFO *lpVer;
Aric Stewartfa384f62004-12-22 18:46:17 +00001889
Mike McCormackf11c8b02005-09-09 14:48:51 +00001890 TRACE("Version comparison.. \n");
1891 versize = GetFileVersionInfoSizeW(file->TargetPath,&handle);
Mike McCormackee034ba2005-09-20 11:59:14 +00001892 version = msi_alloc(versize);
Mike McCormackf11c8b02005-09-09 14:48:51 +00001893 GetFileVersionInfoW(file->TargetPath, 0, versize, version);
Aric Stewartfa384f62004-12-22 18:46:17 +00001894
Mike McCormackf11c8b02005-09-09 14:48:51 +00001895 VerQueryValueW(version, name, (LPVOID*)&lpVer, &sz);
Aric Stewartfa384f62004-12-22 18:46:17 +00001896
Mike McCormackf11c8b02005-09-09 14:48:51 +00001897 sprintfW(filever,name_fmt,
1898 HIWORD(lpVer->dwFileVersionMS),
1899 LOWORD(lpVer->dwFileVersionMS),
1900 HIWORD(lpVer->dwFileVersionLS),
1901 LOWORD(lpVer->dwFileVersionLS));
Aric Stewartec688fb2004-07-04 00:35:52 +00001902
Mike McCormackf11c8b02005-09-09 14:48:51 +00001903 TRACE("new %s old %s\n", debugstr_w(file->Version),
1904 debugstr_w(filever));
1905 if (strcmpiW(filever,file->Version)<0)
Aric Stewartec688fb2004-07-04 00:35:52 +00001906 {
Mike McCormackdded8fb2005-11-02 10:56:42 +00001907 file->state = msifs_overwrite;
1908 /* FIXME: cost should be diff in size */
Aric Stewartec688fb2004-07-04 00:35:52 +00001909 comp->Cost += file->FileSize;
1910 }
1911 else
Mike McCormackdded8fb2005-11-02 10:56:42 +00001912 file->state = msifs_present;
Mike McCormackee034ba2005-09-20 11:59:14 +00001913 msi_free(version);
Mike McCormackf11c8b02005-09-09 14:48:51 +00001914 }
1915 else
Mike McCormackdded8fb2005-11-02 10:56:42 +00001916 file->state = msifs_present;
Aric Stewartec688fb2004-07-04 00:35:52 +00001917 }
1918
Aric Stewart7d3e5972004-07-04 00:36:58 +00001919 TRACE("Evaluating Condition Table\n");
Aric Stewart2e9b5f72004-07-04 00:31:17 +00001920
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001921 rc = MSI_DatabaseOpenViewW(package->db, ConditionQuery, &view);
Aric Stewart84837d92004-07-20 01:22:37 +00001922 if (rc == ERROR_SUCCESS)
1923 {
Aric Stewart443ad4d2005-06-21 20:50:12 +00001924 rc = MSI_IterateRecords(view, NULL, ITERATE_CostFinalizeConditions,
1925 package);
Mike McCormackac6a4132005-01-04 20:36:12 +00001926 msiobj_release(&view->hdr);
Aric Stewart84837d92004-07-20 01:22:37 +00001927 }
Aric Stewart7d3e5972004-07-04 00:36:58 +00001928
1929 TRACE("Enabling or Disabling Components\n");
Mike McCormack38d67a42005-08-22 09:15:23 +00001930 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
Aric Stewart7d3e5972004-07-04 00:36:58 +00001931 {
Mike McCormack298cdae2005-09-06 11:39:01 +00001932 if (comp->Condition)
Aric Stewart7d3e5972004-07-04 00:36:58 +00001933 {
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001934 if (MSI_EvaluateConditionW(package,
Mike McCormack38d67a42005-08-22 09:15:23 +00001935 comp->Condition) == MSICONDITION_FALSE)
Aric Stewart7d3e5972004-07-04 00:36:58 +00001936 {
Mike McCormack38d67a42005-08-22 09:15:23 +00001937 TRACE("Disabling component %s\n", debugstr_w(comp->Component));
1938 comp->Enabled = FALSE;
Aric Stewart7d3e5972004-07-04 00:36:58 +00001939 }
1940 }
1941 }
1942
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001943 MSI_SetPropertyW(package,szCosting,szOne);
Aric Stewart8cc14a92004-12-27 18:56:30 +00001944 /* set default run level if not set */
Mike McCormack062ad502005-09-15 15:04:08 +00001945 level = msi_dup_property( package, szlevel );
Aric Stewart8cc14a92004-12-27 18:56:30 +00001946 if (!level)
1947 MSI_SetPropertyW(package,szlevel, szOne);
Mike McCormackee034ba2005-09-20 11:59:14 +00001948 msi_free(level);
Aric Stewartae1aa322004-12-27 19:02:59 +00001949
Aric Stewart78a04e32005-02-22 15:47:00 +00001950 ACTION_UpdateInstallStates(package);
Aric Stewartae1aa322004-12-27 19:02:59 +00001951
Aric Stewart78a04e32005-02-22 15:47:00 +00001952 return SetFeatureStates(package);
Aric Stewart401bd3f2004-06-28 20:34:35 +00001953}
1954
Francois Gougetda8b3dd2005-01-26 21:09:04 +00001955/* OK this value is "interpreted" and then formatted based on the
Aric Stewart6e160f12004-06-29 04:07:22 +00001956 first few characters */
Aric Stewart09b0aba2005-06-09 20:30:59 +00001957static LPSTR parse_value(MSIPACKAGE *package, LPCWSTR value, DWORD *type,
Aric Stewart401bd3f2004-06-28 20:34:35 +00001958 DWORD *size)
1959{
1960 LPSTR data = NULL;
Aric Stewart6e160f12004-06-29 04:07:22 +00001961 if (value[0]=='#' && value[1]!='#' && value[1]!='%')
Aric Stewart401bd3f2004-06-28 20:34:35 +00001962 {
Aric Stewart6e160f12004-06-29 04:07:22 +00001963 if (value[1]=='x')
1964 {
1965 LPWSTR ptr;
1966 CHAR byte[5];
Aric Stewart6186b2b2005-05-16 21:37:35 +00001967 LPWSTR deformated = NULL;
Aric Stewart6e160f12004-06-29 04:07:22 +00001968 int count;
1969
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001970 deformat_string(package, &value[2], &deformated);
Aric Stewart6e160f12004-06-29 04:07:22 +00001971
1972 /* binary value type */
Aric Stewart6186b2b2005-05-16 21:37:35 +00001973 ptr = deformated;
1974 *type = REG_BINARY;
1975 if (strlenW(ptr)%2)
1976 *size = (strlenW(ptr)/2)+1;
1977 else
1978 *size = strlenW(ptr)/2;
1979
Mike McCormackee034ba2005-09-20 11:59:14 +00001980 data = msi_alloc(*size);
Aric Stewart6186b2b2005-05-16 21:37:35 +00001981
Aric Stewart6e160f12004-06-29 04:07:22 +00001982 byte[0] = '0';
1983 byte[1] = 'x';
1984 byte[4] = 0;
1985 count = 0;
Aric Stewart6186b2b2005-05-16 21:37:35 +00001986 /* if uneven pad with a zero in front */
1987 if (strlenW(ptr)%2)
1988 {
1989 byte[2]= '0';
1990 byte[3]= *ptr;
1991 ptr++;
1992 data[count] = (BYTE)strtol(byte,NULL,0);
1993 count ++;
1994 TRACE("Uneven byte count\n");
1995 }
Aric Stewart6e160f12004-06-29 04:07:22 +00001996 while (*ptr)
1997 {
1998 byte[2]= *ptr;
1999 ptr++;
2000 byte[3]= *ptr;
2001 ptr++;
2002 data[count] = (BYTE)strtol(byte,NULL,0);
2003 count ++;
2004 }
Mike McCormackee034ba2005-09-20 11:59:14 +00002005 msi_free(deformated);
Aric Stewart6e160f12004-06-29 04:07:22 +00002006
2007 TRACE("Data %li bytes(%i)\n",*size,count);
2008 }
2009 else
2010 {
2011 LPWSTR deformated;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002012 LPWSTR p;
2013 DWORD d = 0;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002014 deformat_string(package, &value[1], &deformated);
Aric Stewart6e160f12004-06-29 04:07:22 +00002015
2016 *type=REG_DWORD;
2017 *size = sizeof(DWORD);
Mike McCormackee034ba2005-09-20 11:59:14 +00002018 data = msi_alloc(*size);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002019 p = deformated;
2020 if (*p == '-')
2021 p++;
2022 while (*p)
2023 {
2024 if ( (*p < '0') || (*p > '9') )
2025 break;
2026 d *= 10;
2027 d += (*p - '0');
2028 p++;
2029 }
2030 if (deformated[0] == '-')
2031 d = -d;
2032 *(LPDWORD)data = d;
2033 TRACE("DWORD %li\n",*(LPDWORD)data);
Aric Stewart6e160f12004-06-29 04:07:22 +00002034
Mike McCormackee034ba2005-09-20 11:59:14 +00002035 msi_free(deformated);
Aric Stewart6e160f12004-06-29 04:07:22 +00002036 }
Aric Stewart401bd3f2004-06-28 20:34:35 +00002037 }
2038 else
2039 {
Aric Stewart54c67dd2005-01-25 20:17:09 +00002040 static const WCHAR szMulti[] = {'[','~',']',0};
Aric Stewart09b0aba2005-06-09 20:30:59 +00002041 LPCWSTR ptr;
Aric Stewart6e160f12004-06-29 04:07:22 +00002042 *type=REG_SZ;
2043
Aric Stewart401bd3f2004-06-28 20:34:35 +00002044 if (value[0]=='#')
Aric Stewart6e160f12004-06-29 04:07:22 +00002045 {
2046 if (value[1]=='%')
2047 {
2048 ptr = &value[2];
2049 *type=REG_EXPAND_SZ;
2050 }
2051 else
2052 ptr = &value[1];
2053 }
2054 else
Aric Stewart401bd3f2004-06-28 20:34:35 +00002055 ptr=value;
2056
Aric Stewart54c67dd2005-01-25 20:17:09 +00002057 if (strstrW(value,szMulti))
2058 *type = REG_MULTI_SZ;
2059
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002060 *size = deformat_string(package, ptr,(LPWSTR*)&data);
Aric Stewart401bd3f2004-06-28 20:34:35 +00002061 }
2062 return data;
2063}
2064
Aric Stewart92ef78e2005-06-21 20:21:18 +00002065static UINT ITERATE_WriteRegistryValues(MSIRECORD *row, LPVOID param)
2066{
2067 MSIPACKAGE *package = (MSIPACKAGE*)param;
2068 static const WCHAR szHCR[] =
2069 {'H','K','E','Y','_','C','L','A','S','S','E','S','_',
2070 'R','O','O','T','\\',0};
2071 static const WCHAR szHCU[] =
2072 {'H','K','E','Y','_','C','U','R','R','E','N','T','_',
2073 'U','S','E','R','\\',0};
2074 static const WCHAR szHLM[] =
2075 {'H','K','E','Y','_','L','O','C','A','L','_',
2076 'M','A','C','H','I','N','E','\\',0};
2077 static const WCHAR szHU[] =
2078 {'H','K','E','Y','_','U','S','E','R','S','\\',0};
2079
2080 LPSTR value_data = NULL;
2081 HKEY root_key, hkey;
2082 DWORD type,size;
2083 LPWSTR deformated;
2084 LPCWSTR szRoot, component, name, key, value;
Mike McCormack38d67a42005-08-22 09:15:23 +00002085 MSICOMPONENT *comp;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002086 MSIRECORD * uirow;
2087 LPWSTR uikey;
2088 INT root;
2089 BOOL check_first = FALSE;
2090 UINT rc;
2091
2092 ui_progress(package,2,0,0,0);
2093
2094 value = NULL;
2095 key = NULL;
2096 uikey = NULL;
2097 name = NULL;
2098
2099 component = MSI_RecordGetString(row, 6);
Mike McCormack38d67a42005-08-22 09:15:23 +00002100 comp = get_loaded_component(package,component);
Johan Dahlin0946c422005-08-24 10:57:27 +00002101 if (!comp)
2102 return ERROR_SUCCESS;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002103
Mike McCormackd693f462005-10-29 11:36:48 +00002104 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
Aric Stewart92ef78e2005-06-21 20:21:18 +00002105 {
2106 TRACE("Skipping write due to disabled component %s\n",
2107 debugstr_w(component));
2108
Mike McCormack38d67a42005-08-22 09:15:23 +00002109 comp->Action = comp->Installed;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002110
2111 return ERROR_SUCCESS;
2112 }
2113
Mike McCormack38d67a42005-08-22 09:15:23 +00002114 comp->Action = INSTALLSTATE_LOCAL;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002115
2116 name = MSI_RecordGetString(row, 4);
2117 if( MSI_RecordIsNull(row,5) && name )
2118 {
2119 /* null values can have special meanings */
2120 if (name[0]=='-' && name[1] == 0)
2121 return ERROR_SUCCESS;
2122 else if ((name[0]=='+' && name[1] == 0) ||
2123 (name[0] == '*' && name[1] == 0))
2124 name = NULL;
2125 check_first = TRUE;
2126 }
2127
2128 root = MSI_RecordGetInteger(row,2);
2129 key = MSI_RecordGetString(row, 3);
2130
2131 /* get the root key */
2132 switch (root)
2133 {
Aric Stewart0713f092005-06-24 11:51:29 +00002134 case -1:
2135 {
2136 static const WCHAR szALLUSER[] = {'A','L','L','U','S','E','R','S',0};
Mike McCormack062ad502005-09-15 15:04:08 +00002137 LPWSTR all_users = msi_dup_property( package, szALLUSER );
Aric Stewart0713f092005-06-24 11:51:29 +00002138 if (all_users && all_users[0] == '1')
2139 {
2140 root_key = HKEY_LOCAL_MACHINE;
2141 szRoot = szHLM;
2142 }
2143 else
2144 {
2145 root_key = HKEY_CURRENT_USER;
2146 szRoot = szHCU;
2147 }
Mike McCormackee034ba2005-09-20 11:59:14 +00002148 msi_free(all_users);
Aric Stewart0713f092005-06-24 11:51:29 +00002149 }
2150 break;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002151 case 0: root_key = HKEY_CLASSES_ROOT;
2152 szRoot = szHCR;
2153 break;
2154 case 1: root_key = HKEY_CURRENT_USER;
2155 szRoot = szHCU;
2156 break;
2157 case 2: root_key = HKEY_LOCAL_MACHINE;
2158 szRoot = szHLM;
2159 break;
2160 case 3: root_key = HKEY_USERS;
2161 szRoot = szHU;
2162 break;
2163 default:
2164 ERR("Unknown root %i\n",root);
2165 root_key=NULL;
2166 szRoot = NULL;
2167 break;
2168 }
2169 if (!root_key)
2170 return ERROR_SUCCESS;
2171
2172 deformat_string(package, key , &deformated);
2173 size = strlenW(deformated) + strlenW(szRoot) + 1;
Mike McCormackee034ba2005-09-20 11:59:14 +00002174 uikey = msi_alloc(size*sizeof(WCHAR));
Aric Stewart92ef78e2005-06-21 20:21:18 +00002175 strcpyW(uikey,szRoot);
2176 strcatW(uikey,deformated);
2177
2178 if (RegCreateKeyW( root_key, deformated, &hkey))
2179 {
2180 ERR("Could not create key %s\n",debugstr_w(deformated));
Mike McCormackee034ba2005-09-20 11:59:14 +00002181 msi_free(deformated);
2182 msi_free(uikey);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002183 return ERROR_SUCCESS;
2184 }
Mike McCormackee034ba2005-09-20 11:59:14 +00002185 msi_free(deformated);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002186
2187 value = MSI_RecordGetString(row,5);
2188 if (value)
2189 value_data = parse_value(package, value, &type, &size);
2190 else
2191 {
2192 static const WCHAR szEmpty[] = {0};
2193 value_data = (LPSTR)strdupW(szEmpty);
2194 size = 0;
2195 type = REG_SZ;
2196 }
2197
2198 deformat_string(package, name, &deformated);
2199
2200 /* get the double nulls to terminate SZ_MULTI */
2201 if (type == REG_MULTI_SZ)
2202 size +=sizeof(WCHAR);
2203
2204 if (!check_first)
2205 {
2206 TRACE("Setting value %s of %s\n",debugstr_w(deformated),
2207 debugstr_w(uikey));
Mike McCormack16466af2005-07-06 10:33:30 +00002208 RegSetValueExW(hkey, deformated, 0, type, (LPBYTE)value_data, size);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002209 }
2210 else
2211 {
2212 DWORD sz = 0;
2213 rc = RegQueryValueExW(hkey, deformated, NULL, NULL, NULL, &sz);
2214 if (rc == ERROR_SUCCESS || rc == ERROR_MORE_DATA)
2215 {
2216 TRACE("value %s of %s checked already exists\n",
2217 debugstr_w(deformated), debugstr_w(uikey));
2218 }
2219 else
2220 {
2221 TRACE("Checked and setting value %s of %s\n",
2222 debugstr_w(deformated), debugstr_w(uikey));
2223 if (deformated || size)
Mike McCormack16466af2005-07-06 10:33:30 +00002224 RegSetValueExW(hkey, deformated, 0, type, (LPBYTE) value_data, size);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002225 }
2226 }
2227 RegCloseKey(hkey);
2228
2229 uirow = MSI_CreateRecord(3);
2230 MSI_RecordSetStringW(uirow,2,deformated);
2231 MSI_RecordSetStringW(uirow,1,uikey);
2232
2233 if (type == REG_SZ)
2234 MSI_RecordSetStringW(uirow,3,(LPWSTR)value_data);
2235 else
2236 MSI_RecordSetStringW(uirow,3,value);
2237
2238 ui_actiondata(package,szWriteRegistryValues,uirow);
2239 msiobj_release( &uirow->hdr );
2240
Mike McCormackee034ba2005-09-20 11:59:14 +00002241 msi_free(value_data);
2242 msi_free(deformated);
2243 msi_free(uikey);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002244
2245 return ERROR_SUCCESS;
2246}
2247
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002248static UINT ACTION_WriteRegistryValues(MSIPACKAGE *package)
Aric Stewart401bd3f2004-06-28 20:34:35 +00002249{
2250 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002251 MSIQUERY * view;
Aric Stewart8e233e92005-03-01 11:45:19 +00002252 static const WCHAR ExecSeqQuery[] =
2253 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002254 '`','R','e','g','i','s','t','r','y','`',0 };
Aric Stewart401bd3f2004-06-28 20:34:35 +00002255
Aric Stewart7d3e5972004-07-04 00:36:58 +00002256 if (!package)
2257 return ERROR_INVALID_HANDLE;
2258
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002259 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
Aric Stewart401bd3f2004-06-28 20:34:35 +00002260 if (rc != ERROR_SUCCESS)
Aric Stewart84837d92004-07-20 01:22:37 +00002261 return ERROR_SUCCESS;
Aric Stewart401bd3f2004-06-28 20:34:35 +00002262
Aric Stewartd2c395a2004-07-06 18:48:15 +00002263 /* increment progress bar each time action data is sent */
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002264 ui_progress(package,1,REG_PROGRESS_VALUE,1,0);
Aric Stewartd2c395a2004-07-06 18:48:15 +00002265
Aric Stewart92ef78e2005-06-21 20:21:18 +00002266 rc = MSI_IterateRecords(view, NULL, ITERATE_WriteRegistryValues, package);
Aric Stewartd2c395a2004-07-06 18:48:15 +00002267
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002268 msiobj_release(&view->hdr);
Aric Stewart401bd3f2004-06-28 20:34:35 +00002269 return rc;
2270}
2271
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002272static UINT ACTION_InstallInitialize(MSIPACKAGE *package)
Aric Stewart7d3e5972004-07-04 00:36:58 +00002273{
Aric Stewart9cd707d2005-05-27 19:24:22 +00002274 package->script->CurrentlyScripting = TRUE;
2275
Aric Stewart7d3e5972004-07-04 00:36:58 +00002276 return ERROR_SUCCESS;
2277}
2278
Aric Stewartae1aa322004-12-27 19:02:59 +00002279
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002280static UINT ACTION_InstallValidate(MSIPACKAGE *package)
Aric Stewart7d3e5972004-07-04 00:36:58 +00002281{
Mike McCormack38d67a42005-08-22 09:15:23 +00002282 MSICOMPONENT *comp;
Aric Stewart7d3e5972004-07-04 00:36:58 +00002283 DWORD progress = 0;
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002284 DWORD total = 0;
Aric Stewart8e233e92005-03-01 11:45:19 +00002285 static const WCHAR q1[]=
2286 {'S','E','L','E','C','T',' ','*',' ', 'F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002287 '`','R','e','g','i','s','t','r','y','`',0};
Aric Stewart7d3e5972004-07-04 00:36:58 +00002288 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002289 MSIQUERY * view;
Mike McCormack1da28582005-08-22 14:09:17 +00002290 MSIFEATURE *feature;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002291 MSIFILE *file;
Aric Stewart7d3e5972004-07-04 00:36:58 +00002292
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002293 TRACE("InstallValidate\n");
Aric Stewart7d3e5972004-07-04 00:36:58 +00002294
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002295 rc = MSI_DatabaseOpenViewW(package->db, q1, &view);
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002296 if (rc == ERROR_SUCCESS)
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002297 {
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002298 MSI_IterateRecords( view, &progress, NULL, package );
2299 msiobj_release( &view->hdr );
2300 total += progress * REG_PROGRESS_VALUE;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002301 }
Aric Stewart7d3e5972004-07-04 00:36:58 +00002302
Mike McCormack38d67a42005-08-22 09:15:23 +00002303 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
Mike McCormack38d67a42005-08-22 09:15:23 +00002304 total += COMPONENT_PROGRESS_VALUE;
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002305
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002306 LIST_FOR_EACH_ENTRY( file, &package->files, MSIFILE, entry )
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002307 total += file->FileSize;
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002308
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002309 ui_progress(package,0,total,0,0);
Aric Stewart7d3e5972004-07-04 00:36:58 +00002310
Mike McCormack1da28582005-08-22 14:09:17 +00002311 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002312 {
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002313 TRACE("Feature: %s; Installed: %i; Action %i; Request %i\n",
2314 debugstr_w(feature->Feature), feature->Installed, feature->Action,
2315 feature->ActionRequest);
2316 }
2317
Aric Stewart7d3e5972004-07-04 00:36:58 +00002318 return ERROR_SUCCESS;
2319}
2320
Aric Stewartc79f4e22005-06-22 18:03:08 +00002321static UINT ITERATE_LaunchConditions(MSIRECORD *row, LPVOID param)
2322{
2323 MSIPACKAGE* package = (MSIPACKAGE*)param;
2324 LPCWSTR cond = NULL;
2325 LPCWSTR message = NULL;
2326 static const WCHAR title[]=
2327 {'I','n','s','t','a','l','l',' ','F','a', 'i','l','e','d',0};
2328
2329 cond = MSI_RecordGetString(row,1);
2330
2331 if (MSI_EvaluateConditionW(package,cond) != MSICONDITION_TRUE)
2332 {
2333 LPWSTR deformated;
2334 message = MSI_RecordGetString(row,2);
2335 deformat_string(package,message,&deformated);
2336 MessageBoxW(NULL,deformated,title,MB_OK);
Mike McCormackee034ba2005-09-20 11:59:14 +00002337 msi_free(deformated);
Aric Stewartc79f4e22005-06-22 18:03:08 +00002338 return ERROR_FUNCTION_FAILED;
2339 }
2340
2341 return ERROR_SUCCESS;
2342}
2343
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002344static UINT ACTION_LaunchConditions(MSIPACKAGE *package)
Aric Stewart5b936ca2004-07-06 18:47:09 +00002345{
2346 UINT rc;
Mike McCormackf3c8b832004-07-19 19:35:05 +00002347 MSIQUERY * view = NULL;
Aric Stewart8e233e92005-03-01 11:45:19 +00002348 static const WCHAR ExecSeqQuery[] =
2349 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002350 '`','L','a','u','n','c','h','C','o','n','d','i','t','i','o','n','`',0};
Aric Stewart7d3e5972004-07-04 00:36:58 +00002351
Aric Stewart5b936ca2004-07-06 18:47:09 +00002352 TRACE("Checking launch conditions\n");
Aric Stewart7d3e5972004-07-04 00:36:58 +00002353
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002354 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
Aric Stewart5b936ca2004-07-06 18:47:09 +00002355 if (rc != ERROR_SUCCESS)
Aric Stewart84837d92004-07-20 01:22:37 +00002356 return ERROR_SUCCESS;
Aric Stewart5b936ca2004-07-06 18:47:09 +00002357
Aric Stewartc79f4e22005-06-22 18:03:08 +00002358 rc = MSI_IterateRecords(view, NULL, ITERATE_LaunchConditions, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002359 msiobj_release(&view->hdr);
Aric Stewartc79f4e22005-06-22 18:03:08 +00002360
Aric Stewart5b936ca2004-07-06 18:47:09 +00002361 return rc;
2362}
Aric Stewart7d3e5972004-07-04 00:36:58 +00002363
Mike McCormack38d67a42005-08-22 09:15:23 +00002364static LPWSTR resolve_keypath( MSIPACKAGE* package, MSICOMPONENT *cmp )
Aric Stewartb942e182004-07-06 18:50:02 +00002365{
Aric Stewartb942e182004-07-06 18:50:02 +00002366
Mike McCormackefcc1ec2005-09-12 12:07:15 +00002367 if (!cmp->KeyPath)
2368 return resolve_folder(package,cmp->Directory,FALSE,FALSE,NULL);
2369
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002370 if (cmp->Attributes & msidbComponentAttributesRegistryKeyPath)
Aric Stewartb942e182004-07-06 18:50:02 +00002371 {
Aric Stewart6269f002005-01-17 13:40:39 +00002372 MSIRECORD * row = 0;
Mike McCormack0b352c72005-06-02 10:29:57 +00002373 UINT root,len;
Aric Stewart09b0aba2005-06-09 20:30:59 +00002374 LPWSTR deformated,buffer,deformated_name;
2375 LPCWSTR key,name;
Aric Stewart8e233e92005-03-01 11:45:19 +00002376 static const WCHAR ExecSeqQuery[] =
2377 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002378 '`','R','e','g','i','s','t','r','y','`',' ',
2379 'W','H','E','R','E',' ', '`','R','e','g','i','s','t','r','y','`',
2380 ' ','=',' ' ,'\'','%','s','\'',0 };
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002381 static const WCHAR fmt[]={'%','0','2','i',':','\\','%','s','\\',0};
Aric Stewart8e233e92005-03-01 11:45:19 +00002382 static const WCHAR fmt2[]=
2383 {'%','0','2','i',':','\\','%','s','\\','%','s',0};
Aric Stewart6269f002005-01-17 13:40:39 +00002384
Mike McCormack0b352c72005-06-02 10:29:57 +00002385 row = MSI_QueryGetRecord(package->db, ExecSeqQuery,cmp->KeyPath);
2386 if (!row)
Aric Stewart6269f002005-01-17 13:40:39 +00002387 return NULL;
2388
Aric Stewart6269f002005-01-17 13:40:39 +00002389 root = MSI_RecordGetInteger(row,2);
Aric Stewart09b0aba2005-06-09 20:30:59 +00002390 key = MSI_RecordGetString(row, 3);
2391 name = MSI_RecordGetString(row, 4);
Aric Stewart6269f002005-01-17 13:40:39 +00002392 deformat_string(package, key , &deformated);
2393 deformat_string(package, name, &deformated_name);
2394
Ulrich Czekallae15e5172005-03-08 16:44:51 +00002395 len = strlenW(deformated) + 6;
Aric Stewart6269f002005-01-17 13:40:39 +00002396 if (deformated_name)
2397 len+=strlenW(deformated_name);
2398
Mike McCormackee034ba2005-09-20 11:59:14 +00002399 buffer = msi_alloc( len *sizeof(WCHAR));
Aric Stewart6269f002005-01-17 13:40:39 +00002400
2401 if (deformated_name)
2402 sprintfW(buffer,fmt2,root,deformated,deformated_name);
2403 else
2404 sprintfW(buffer,fmt,root,deformated);
2405
Mike McCormackee034ba2005-09-20 11:59:14 +00002406 msi_free(deformated);
2407 msi_free(deformated_name);
Aric Stewart6269f002005-01-17 13:40:39 +00002408 msiobj_release(&row->hdr);
Aric Stewart6269f002005-01-17 13:40:39 +00002409
2410 return buffer;
2411 }
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002412 else if (cmp->Attributes & msidbComponentAttributesODBCDataSource)
Aric Stewart6269f002005-01-17 13:40:39 +00002413 {
2414 FIXME("UNIMPLEMENTED keypath as ODBC Source\n");
Aric Stewartfa384f62004-12-22 18:46:17 +00002415 return NULL;
Aric Stewartb942e182004-07-06 18:50:02 +00002416 }
2417 else
2418 {
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002419 MSIFILE *file = get_loaded_file( package, cmp->KeyPath );
Aric Stewartfcb20c52004-07-06 18:51:16 +00002420
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002421 if (file)
2422 return strdupW( file->TargetPath );
Aric Stewartb942e182004-07-06 18:50:02 +00002423 }
Aric Stewartfa384f62004-12-22 18:46:17 +00002424 return NULL;
Aric Stewartb942e182004-07-06 18:50:02 +00002425}
2426
Stefan Huehnerac6f5622005-06-20 14:18:03 +00002427static HKEY openSharedDLLsKey(void)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002428{
2429 HKEY hkey=0;
Aric Stewart8e233e92005-03-01 11:45:19 +00002430 static const WCHAR path[] =
2431 {'S','o','f','t','w','a','r','e','\\',
2432 'M','i','c','r','o','s','o','f','t','\\',
2433 'W','i','n','d','o','w','s','\\',
2434 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
2435 'S','h','a','r','e','d','D','L','L','s',0};
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002436
2437 RegCreateKeyW(HKEY_LOCAL_MACHINE,path,&hkey);
2438 return hkey;
2439}
2440
2441static UINT ACTION_GetSharedDLLsCount(LPCWSTR dll)
2442{
2443 HKEY hkey;
2444 DWORD count=0;
2445 DWORD type;
2446 DWORD sz = sizeof(count);
2447 DWORD rc;
2448
2449 hkey = openSharedDLLsKey();
2450 rc = RegQueryValueExW(hkey, dll, NULL, &type, (LPBYTE)&count, &sz);
2451 if (rc != ERROR_SUCCESS)
2452 count = 0;
2453 RegCloseKey(hkey);
2454 return count;
2455}
2456
2457static UINT ACTION_WriteSharedDLLsCount(LPCWSTR path, UINT count)
2458{
2459 HKEY hkey;
2460
2461 hkey = openSharedDLLsKey();
2462 if (count > 0)
Mike McCormack4db02cd2005-09-15 14:58:38 +00002463 msi_reg_set_val_dword( hkey, path, count );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002464 else
2465 RegDeleteValueW(hkey,path);
2466 RegCloseKey(hkey);
2467 return count;
2468}
2469
2470/*
2471 * Return TRUE if the count should be written out and FALSE if not
2472 */
Mike McCormack38d67a42005-08-22 09:15:23 +00002473static void ACTION_RefCountComponent( MSIPACKAGE* package, MSICOMPONENT *comp )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002474{
Mike McCormack1da28582005-08-22 14:09:17 +00002475 MSIFEATURE *feature;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002476 INT count = 0;
2477 BOOL write = FALSE;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002478
2479 /* only refcount DLLs */
Mike McCormackefcc1ec2005-09-12 12:07:15 +00002480 if (comp->KeyPath == NULL ||
Mike McCormack38d67a42005-08-22 09:15:23 +00002481 comp->Attributes & msidbComponentAttributesRegistryKeyPath ||
2482 comp->Attributes & msidbComponentAttributesODBCDataSource)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002483 write = FALSE;
2484 else
2485 {
Mike McCormack38d67a42005-08-22 09:15:23 +00002486 count = ACTION_GetSharedDLLsCount( comp->FullKeypath);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002487 write = (count > 0);
2488
Mike McCormack38d67a42005-08-22 09:15:23 +00002489 if (comp->Attributes & msidbComponentAttributesSharedDllRefCount)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002490 write = TRUE;
2491 }
2492
2493 /* increment counts */
Mike McCormack1da28582005-08-22 14:09:17 +00002494 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002495 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00002496 ComponentList *cl;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002497
Mike McCormack1da28582005-08-22 14:09:17 +00002498 if (!ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_LOCAL ))
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002499 continue;
2500
Mike McCormack1da28582005-08-22 14:09:17 +00002501 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002502 {
Mike McCormack38d67a42005-08-22 09:15:23 +00002503 if ( cl->component == comp )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002504 count++;
2505 }
2506 }
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002507
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002508 /* decrement counts */
Mike McCormack1da28582005-08-22 14:09:17 +00002509 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002510 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00002511 ComponentList *cl;
2512
Mike McCormack1da28582005-08-22 14:09:17 +00002513 if (!ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_ABSENT ))
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002514 continue;
2515
Mike McCormack1da28582005-08-22 14:09:17 +00002516 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002517 {
Mike McCormack38d67a42005-08-22 09:15:23 +00002518 if ( cl->component == comp )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002519 count--;
2520 }
2521 }
2522
2523 /* ref count all the files in the component */
2524 if (write)
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002525 {
2526 MSIFILE *file;
2527
2528 LIST_FOR_EACH_ENTRY( file, &package->files, MSIFILE, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002529 {
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002530 if (file->Component == comp)
2531 ACTION_WriteSharedDLLsCount( file->TargetPath, count );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002532 }
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002533 }
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002534
2535 /* add a count for permenent */
Mike McCormack38d67a42005-08-22 09:15:23 +00002536 if (comp->Attributes & msidbComponentAttributesPermanent)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002537 count ++;
2538
Mike McCormack38d67a42005-08-22 09:15:23 +00002539 comp->RefCount = count;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002540
2541 if (write)
Mike McCormack38d67a42005-08-22 09:15:23 +00002542 ACTION_WriteSharedDLLsCount( comp->FullKeypath, comp->RefCount );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002543}
2544
Aric Stewart2cf222f2004-07-06 19:00:23 +00002545/*
2546 * Ok further analysis makes me think that this work is
2547 * actually done in the PublishComponents and PublishFeatures
Mike McCormack3ece2462004-07-09 19:33:25 +00002548 * step, and not here. It appears like the keypath and all that is
2549 * resolved in this step, however actually written in the Publish steps.
Alexandre Julliard77b12762004-07-09 19:43:29 +00002550 * But we will leave it here for now because it is unclear
Aric Stewart2cf222f2004-07-06 19:00:23 +00002551 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002552static UINT ACTION_ProcessComponents(MSIPACKAGE *package)
Aric Stewartb942e182004-07-06 18:50:02 +00002553{
Aric Stewart68b07492005-01-25 11:05:37 +00002554 WCHAR squished_pc[GUID_SIZE];
2555 WCHAR squished_cc[GUID_SIZE];
Aric Stewartb942e182004-07-06 18:50:02 +00002556 UINT rc;
Mike McCormack38d67a42005-08-22 09:15:23 +00002557 MSICOMPONENT *comp;
Aric Stewart68b07492005-01-25 11:05:37 +00002558 HKEY hkey=0,hkey2=0;
Aric Stewartb942e182004-07-06 18:50:02 +00002559
Aric Stewartb942e182004-07-06 18:50:02 +00002560 /* writes the Component and Features values to the registry */
Aric Stewartb942e182004-07-06 18:50:02 +00002561
Aric Stewart68b07492005-01-25 11:05:37 +00002562 rc = MSIREG_OpenComponents(&hkey);
Aric Stewartb942e182004-07-06 18:50:02 +00002563 if (rc != ERROR_SUCCESS)
2564 goto end;
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002565
Aric Stewartadaef112005-07-07 20:27:06 +00002566 squash_guid(package->ProductCode,squished_pc);
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002567 ui_progress(package,1,COMPONENT_PROGRESS_VALUE,1,0);
Mike McCormack38d67a42005-08-22 09:15:23 +00002568
2569 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
Aric Stewartb942e182004-07-06 18:50:02 +00002570 {
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002571 ui_progress(package,2,0,0,0);
Mike McCormackefcc1ec2005-09-12 12:07:15 +00002572 if (comp->ComponentId)
Aric Stewartb942e182004-07-06 18:50:02 +00002573 {
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002574 MSIRECORD * uirow;
Aric Stewartb942e182004-07-06 18:50:02 +00002575
Mike McCormack38d67a42005-08-22 09:15:23 +00002576 squash_guid(comp->ComponentId,squished_cc);
Aric Stewartb942e182004-07-06 18:50:02 +00002577
Mike McCormack566c69e2005-09-22 10:49:17 +00002578 msi_free(comp->FullKeypath);
2579 comp->FullKeypath = resolve_keypath( package, comp );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002580
2581 /* do the refcounting */
Mike McCormack38d67a42005-08-22 09:15:23 +00002582 ACTION_RefCountComponent( package, comp );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002583
Aric Stewartc5a14432005-05-18 17:46:12 +00002584 TRACE("Component %s (%s), Keypath=%s, RefCount=%i\n",
Mike McCormack38d67a42005-08-22 09:15:23 +00002585 debugstr_w(comp->Component),
Aric Stewartc5a14432005-05-18 17:46:12 +00002586 debugstr_w(squished_cc),
Mike McCormack38d67a42005-08-22 09:15:23 +00002587 debugstr_w(comp->FullKeypath),
2588 comp->RefCount);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002589 /*
2590 * Write the keypath out if the component is to be registered
2591 * and delete the key if the component is to be deregistered
2592 */
Mike McCormackd693f462005-10-29 11:36:48 +00002593 if (ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
Aric Stewartfa384f62004-12-22 18:46:17 +00002594 {
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002595 rc = RegCreateKeyW(hkey,squished_cc,&hkey2);
2596 if (rc != ERROR_SUCCESS)
2597 continue;
2598
Mike McCormack566c69e2005-09-22 10:49:17 +00002599 if (comp->FullKeypath)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002600 {
Mike McCormack566c69e2005-09-22 10:49:17 +00002601 msi_reg_set_val_str( hkey2, squished_pc, comp->FullKeypath );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002602
Mike McCormack38d67a42005-08-22 09:15:23 +00002603 if (comp->Attributes & msidbComponentAttributesPermanent)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002604 {
Aric Stewart8e233e92005-03-01 11:45:19 +00002605 static const WCHAR szPermKey[] =
2606 { '0','0','0','0','0','0','0','0','0','0','0','0',
Mike McCormack4db02cd2005-09-15 14:58:38 +00002607 '0','0','0','0','0','0','0','0','0','0','0','0',
Aric Stewart8e233e92005-03-01 11:45:19 +00002608 '0','0','0','0','0','0','0','0',0};
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002609
Mike McCormack566c69e2005-09-22 10:49:17 +00002610 msi_reg_set_val_str( hkey2, szPermKey, comp->FullKeypath );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002611 }
2612
2613 RegCloseKey(hkey2);
2614
2615 /* UI stuff */
2616 uirow = MSI_CreateRecord(3);
Aric Stewartadaef112005-07-07 20:27:06 +00002617 MSI_RecordSetStringW(uirow,1,package->ProductCode);
Mike McCormack38d67a42005-08-22 09:15:23 +00002618 MSI_RecordSetStringW(uirow,2,comp->ComponentId);
Mike McCormack566c69e2005-09-22 10:49:17 +00002619 MSI_RecordSetStringW(uirow,3,comp->FullKeypath);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002620 ui_actiondata(package,szProcessComponents,uirow);
2621 msiobj_release( &uirow->hdr );
2622 }
2623 }
Mike McCormackd693f462005-10-29 11:36:48 +00002624 else if (ACTION_VerifyComponentForAction( comp, INSTALLSTATE_ABSENT))
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002625 {
2626 DWORD res;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002627
2628 rc = RegOpenKeyW(hkey,squished_cc,&hkey2);
2629 if (rc != ERROR_SUCCESS)
2630 continue;
2631
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002632 RegDeleteValueW(hkey2,squished_pc);
2633
2634 /* if the key is empty delete it */
2635 res = RegEnumKeyExW(hkey2,0,NULL,0,0,NULL,0,NULL);
Aric Stewart68b07492005-01-25 11:05:37 +00002636 RegCloseKey(hkey2);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002637 if (res == ERROR_NO_MORE_ITEMS)
2638 RegDeleteKeyW(hkey,squished_cc);
Aric Stewartb942e182004-07-06 18:50:02 +00002639
Aric Stewartfa384f62004-12-22 18:46:17 +00002640 /* UI stuff */
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002641 uirow = MSI_CreateRecord(2);
Aric Stewartadaef112005-07-07 20:27:06 +00002642 MSI_RecordSetStringW(uirow,1,package->ProductCode);
Mike McCormack38d67a42005-08-22 09:15:23 +00002643 MSI_RecordSetStringW(uirow,2,comp->ComponentId);
Aric Stewartfa384f62004-12-22 18:46:17 +00002644 ui_actiondata(package,szProcessComponents,uirow);
2645 msiobj_release( &uirow->hdr );
Aric Stewartfa384f62004-12-22 18:46:17 +00002646 }
Aric Stewartb942e182004-07-06 18:50:02 +00002647 }
2648 }
2649end:
Aric Stewartb942e182004-07-06 18:50:02 +00002650 RegCloseKey(hkey);
2651 return rc;
2652}
2653
Aric Stewart6e821732005-03-30 10:19:08 +00002654typedef struct {
2655 CLSID clsid;
2656 LPWSTR source;
2657
2658 LPWSTR path;
2659 ITypeLib *ptLib;
2660} typelib_struct;
2661
Mike McCormackf9acfe62005-06-07 20:29:51 +00002662static BOOL CALLBACK Typelib_EnumResNameProc( HMODULE hModule, LPCWSTR lpszType,
Aric Stewart6e821732005-03-30 10:19:08 +00002663 LPWSTR lpszName, LONG_PTR lParam)
2664{
2665 TLIBATTR *attr;
2666 typelib_struct *tl_struct = (typelib_struct*) lParam;
2667 static const WCHAR fmt[] = {'%','s','\\','%','i',0};
2668 int sz;
2669 HRESULT res;
2670
2671 if (!IS_INTRESOURCE(lpszName))
2672 {
2673 ERR("Not Int Resource Name %s\n",debugstr_w(lpszName));
2674 return TRUE;
2675 }
2676
2677 sz = strlenW(tl_struct->source)+4;
2678 sz *= sizeof(WCHAR);
2679
Aric Stewartca8c4e42005-06-02 15:13:57 +00002680 if ((INT)lpszName == 1)
2681 tl_struct->path = strdupW(tl_struct->source);
2682 else
2683 {
Mike McCormackee034ba2005-09-20 11:59:14 +00002684 tl_struct->path = msi_alloc(sz);
Aric Stewartca8c4e42005-06-02 15:13:57 +00002685 sprintfW(tl_struct->path,fmt,tl_struct->source, lpszName);
2686 }
Aric Stewart6e821732005-03-30 10:19:08 +00002687
2688 TRACE("trying %s\n", debugstr_w(tl_struct->path));
2689 res = LoadTypeLib(tl_struct->path,&tl_struct->ptLib);
2690 if (!SUCCEEDED(res))
2691 {
Mike McCormackee034ba2005-09-20 11:59:14 +00002692 msi_free(tl_struct->path);
Aric Stewart6e821732005-03-30 10:19:08 +00002693 tl_struct->path = NULL;
2694
2695 return TRUE;
2696 }
2697
2698 ITypeLib_GetLibAttr(tl_struct->ptLib, &attr);
2699 if (IsEqualGUID(&(tl_struct->clsid),&(attr->guid)))
2700 {
2701 ITypeLib_ReleaseTLibAttr(tl_struct->ptLib, attr);
2702 return FALSE;
2703 }
2704
Mike McCormackee034ba2005-09-20 11:59:14 +00002705 msi_free(tl_struct->path);
Aric Stewart6e821732005-03-30 10:19:08 +00002706 tl_struct->path = NULL;
2707
2708 ITypeLib_ReleaseTLibAttr(tl_struct->ptLib, attr);
2709 ITypeLib_Release(tl_struct->ptLib);
2710
2711 return TRUE;
2712}
2713
Aric Stewart234dc4b2005-06-22 18:27:34 +00002714static UINT ITERATE_RegisterTypeLibraries(MSIRECORD *row, LPVOID param)
2715{
2716 MSIPACKAGE* package = (MSIPACKAGE*)param;
2717 LPCWSTR component;
Mike McCormack38d67a42005-08-22 09:15:23 +00002718 MSICOMPONENT *comp;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002719 MSIFILE *file;
Aric Stewart234dc4b2005-06-22 18:27:34 +00002720 typelib_struct tl_struct;
2721 HMODULE module;
2722 static const WCHAR szTYPELIB[] = {'T','Y','P','E','L','I','B',0};
2723
2724 component = MSI_RecordGetString(row,3);
Mike McCormack38d67a42005-08-22 09:15:23 +00002725 comp = get_loaded_component(package,component);
2726 if (!comp)
Aric Stewart234dc4b2005-06-22 18:27:34 +00002727 return ERROR_SUCCESS;
2728
Mike McCormackd693f462005-10-29 11:36:48 +00002729 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
Aric Stewart234dc4b2005-06-22 18:27:34 +00002730 {
2731 TRACE("Skipping typelib reg due to disabled component\n");
2732
Mike McCormack38d67a42005-08-22 09:15:23 +00002733 comp->Action = comp->Installed;
Aric Stewart234dc4b2005-06-22 18:27:34 +00002734
2735 return ERROR_SUCCESS;
2736 }
2737
Mike McCormack38d67a42005-08-22 09:15:23 +00002738 comp->Action = INSTALLSTATE_LOCAL;
Aric Stewart234dc4b2005-06-22 18:27:34 +00002739
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002740 file = get_loaded_file( package, comp->KeyPath );
2741 if (!file)
Aric Stewart234dc4b2005-06-22 18:27:34 +00002742 return ERROR_SUCCESS;
2743
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002744 module = LoadLibraryExW( file->TargetPath, NULL, LOAD_LIBRARY_AS_DATAFILE );
Mike McCormack51c66182005-10-27 12:36:12 +00002745 if (module)
Aric Stewart234dc4b2005-06-22 18:27:34 +00002746 {
Mike McCormack51c66182005-10-27 12:36:12 +00002747 LPCWSTR guid;
2748 guid = MSI_RecordGetString(row,1);
2749 CLSIDFromString((LPWSTR)guid, &tl_struct.clsid);
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002750 tl_struct.source = strdupW( file->TargetPath );
Aric Stewart234dc4b2005-06-22 18:27:34 +00002751 tl_struct.path = NULL;
2752
2753 EnumResourceNamesW(module, szTYPELIB, Typelib_EnumResNameProc,
2754 (LONG_PTR)&tl_struct);
2755
Mike McCormack51c66182005-10-27 12:36:12 +00002756 if (tl_struct.path)
Aric Stewart234dc4b2005-06-22 18:27:34 +00002757 {
2758 LPWSTR help = NULL;
2759 LPCWSTR helpid;
2760 HRESULT res;
2761
2762 helpid = MSI_RecordGetString(row,6);
2763
2764 if (helpid)
2765 help = resolve_folder(package,helpid,FALSE,FALSE,NULL);
2766 res = RegisterTypeLib(tl_struct.ptLib,tl_struct.path,help);
Mike McCormackee034ba2005-09-20 11:59:14 +00002767 msi_free(help);
Aric Stewart234dc4b2005-06-22 18:27:34 +00002768
2769 if (!SUCCEEDED(res))
2770 ERR("Failed to register type library %s\n",
2771 debugstr_w(tl_struct.path));
2772 else
2773 {
2774 ui_actiondata(package,szRegisterTypeLibraries,row);
2775
2776 TRACE("Registered %s\n", debugstr_w(tl_struct.path));
2777 }
2778
2779 ITypeLib_Release(tl_struct.ptLib);
Mike McCormackee034ba2005-09-20 11:59:14 +00002780 msi_free(tl_struct.path);
Aric Stewart234dc4b2005-06-22 18:27:34 +00002781 }
2782 else
2783 ERR("Failed to load type library %s\n",
2784 debugstr_w(tl_struct.source));
2785
2786 FreeLibrary(module);
Mike McCormackee034ba2005-09-20 11:59:14 +00002787 msi_free(tl_struct.source);
Aric Stewart234dc4b2005-06-22 18:27:34 +00002788 }
2789 else
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002790 ERR("Could not load file! %s\n", debugstr_w(file->TargetPath));
Aric Stewart234dc4b2005-06-22 18:27:34 +00002791
2792 return ERROR_SUCCESS;
2793}
2794
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002795static UINT ACTION_RegisterTypeLibraries(MSIPACKAGE *package)
Aric Stewartfcb20c52004-07-06 18:51:16 +00002796{
2797 /*
Mike McCormackc90c7812004-07-09 22:58:27 +00002798 * OK this is a bit confusing.. I am given a _Component key and I believe
Aric Stewartfcb20c52004-07-06 18:51:16 +00002799 * that the file that is being registered as a type library is the "key file
Mike McCormackc90c7812004-07-09 22:58:27 +00002800 * of that component" which I interpret to mean "The file in the KeyPath of
2801 * that component".
Aric Stewartfcb20c52004-07-06 18:51:16 +00002802 */
2803 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002804 MSIQUERY * view;
Aric Stewart8e233e92005-03-01 11:45:19 +00002805 static const WCHAR Query[] =
2806 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002807 '`','T','y','p','e','L','i','b','`',0};
Aric Stewartfcb20c52004-07-06 18:51:16 +00002808
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002809 rc = MSI_DatabaseOpenViewW(package->db, Query, &view);
Aric Stewartfcb20c52004-07-06 18:51:16 +00002810 if (rc != ERROR_SUCCESS)
Aric Stewart84837d92004-07-20 01:22:37 +00002811 return ERROR_SUCCESS;
Aric Stewartfcb20c52004-07-06 18:51:16 +00002812
Aric Stewart234dc4b2005-06-22 18:27:34 +00002813 rc = MSI_IterateRecords(view, NULL, ITERATE_RegisterTypeLibraries, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002814 msiobj_release(&view->hdr);
Aric Stewartfcb20c52004-07-06 18:51:16 +00002815 return rc;
Aric Stewartfcb20c52004-07-06 18:51:16 +00002816}
2817
Aric Stewart9adacf62005-06-24 11:58:21 +00002818static UINT ITERATE_CreateShortcuts(MSIRECORD *row, LPVOID param)
Aric Stewart2cf222f2004-07-06 19:00:23 +00002819{
Aric Stewart9adacf62005-06-24 11:58:21 +00002820 MSIPACKAGE *package = (MSIPACKAGE*)param;
2821 LPWSTR target_file, target_folder;
2822 LPCWSTR buffer;
2823 WCHAR filename[0x100];
2824 DWORD sz;
Mike McCormack38d67a42005-08-22 09:15:23 +00002825 MSICOMPONENT *comp;
Aric Stewart9adacf62005-06-24 11:58:21 +00002826 static const WCHAR szlnk[]={'.','l','n','k',0};
Aric Stewart2cf222f2004-07-06 19:00:23 +00002827 IShellLinkW *sl;
2828 IPersistFile *pf;
2829 HRESULT res;
2830
Aric Stewart9adacf62005-06-24 11:58:21 +00002831 buffer = MSI_RecordGetString(row,4);
Mike McCormack38d67a42005-08-22 09:15:23 +00002832 comp = get_loaded_component(package,buffer);
2833 if (!comp)
Aric Stewart9adacf62005-06-24 11:58:21 +00002834 return ERROR_SUCCESS;
2835
Mike McCormackd693f462005-10-29 11:36:48 +00002836 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL ))
Aric Stewart9adacf62005-06-24 11:58:21 +00002837 {
2838 TRACE("Skipping shortcut creation due to disabled component\n");
2839
Mike McCormack38d67a42005-08-22 09:15:23 +00002840 comp->Action = comp->Installed;
Aric Stewart9adacf62005-06-24 11:58:21 +00002841
2842 return ERROR_SUCCESS;
2843 }
2844
Mike McCormack38d67a42005-08-22 09:15:23 +00002845 comp->Action = INSTALLSTATE_LOCAL;
Aric Stewart9adacf62005-06-24 11:58:21 +00002846
2847 ui_actiondata(package,szCreateShortcuts,row);
2848
2849 res = CoCreateInstance( &CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER,
2850 &IID_IShellLinkW, (LPVOID *) &sl );
2851
2852 if (FAILED(res))
2853 {
2854 ERR("Is IID_IShellLink\n");
2855 return ERROR_SUCCESS;
2856 }
2857
2858 res = IShellLinkW_QueryInterface( sl, &IID_IPersistFile,(LPVOID*) &pf );
2859 if( FAILED( res ) )
2860 {
2861 ERR("Is IID_IPersistFile\n");
2862 return ERROR_SUCCESS;
2863 }
2864
2865 buffer = MSI_RecordGetString(row,2);
2866 target_folder = resolve_folder(package, buffer,FALSE,FALSE,NULL);
2867
2868 /* may be needed because of a bug somehwere else */
2869 create_full_pathW(target_folder);
2870
2871 sz = 0x100;
2872 MSI_RecordGetStringW(row,3,filename,&sz);
2873 reduce_to_longfilename(filename);
2874 if (!strchrW(filename,'.') || strcmpiW(strchrW(filename,'.'),szlnk))
2875 strcatW(filename,szlnk);
2876 target_file = build_directory_name(2, target_folder, filename);
Mike McCormackee034ba2005-09-20 11:59:14 +00002877 msi_free(target_folder);
Aric Stewart9adacf62005-06-24 11:58:21 +00002878
2879 buffer = MSI_RecordGetString(row,5);
2880 if (strchrW(buffer,'['))
2881 {
2882 LPWSTR deformated;
2883 deformat_string(package,buffer,&deformated);
2884 IShellLinkW_SetPath(sl,deformated);
Mike McCormackee034ba2005-09-20 11:59:14 +00002885 msi_free(deformated);
Aric Stewart9adacf62005-06-24 11:58:21 +00002886 }
2887 else
2888 {
Aric Stewart9adacf62005-06-24 11:58:21 +00002889 FIXME("poorly handled shortcut format, advertised shortcut\n");
Mike McCormack566c69e2005-09-22 10:49:17 +00002890 IShellLinkW_SetPath(sl,comp->FullKeypath);
Aric Stewart9adacf62005-06-24 11:58:21 +00002891 }
2892
2893 if (!MSI_RecordIsNull(row,6))
2894 {
2895 LPWSTR deformated;
2896 buffer = MSI_RecordGetString(row,6);
2897 deformat_string(package,buffer,&deformated);
2898 IShellLinkW_SetArguments(sl,deformated);
Mike McCormackee034ba2005-09-20 11:59:14 +00002899 msi_free(deformated);
Aric Stewart9adacf62005-06-24 11:58:21 +00002900 }
2901
2902 if (!MSI_RecordIsNull(row,7))
2903 {
2904 buffer = MSI_RecordGetString(row,7);
2905 IShellLinkW_SetDescription(sl,buffer);
2906 }
2907
2908 if (!MSI_RecordIsNull(row,8))
2909 IShellLinkW_SetHotkey(sl,MSI_RecordGetInteger(row,8));
2910
2911 if (!MSI_RecordIsNull(row,9))
2912 {
Mike McCormack75658d72005-09-22 10:33:57 +00002913 LPWSTR Path;
Aric Stewart9adacf62005-06-24 11:58:21 +00002914 INT index;
2915
2916 buffer = MSI_RecordGetString(row,9);
2917
Mike McCormack75658d72005-09-22 10:33:57 +00002918 Path = build_icon_path(package,buffer);
Aric Stewart9adacf62005-06-24 11:58:21 +00002919 index = MSI_RecordGetInteger(row,10);
2920
2921 IShellLinkW_SetIconLocation(sl,Path,index);
Mike McCormackee034ba2005-09-20 11:59:14 +00002922 msi_free(Path);
Aric Stewart9adacf62005-06-24 11:58:21 +00002923 }
2924
2925 if (!MSI_RecordIsNull(row,11))
2926 IShellLinkW_SetShowCmd(sl,MSI_RecordGetInteger(row,11));
2927
2928 if (!MSI_RecordIsNull(row,12))
2929 {
2930 LPWSTR Path;
2931 buffer = MSI_RecordGetString(row,12);
2932 Path = resolve_folder(package, buffer, FALSE, FALSE, NULL);
2933 IShellLinkW_SetWorkingDirectory(sl,Path);
Mike McCormackee034ba2005-09-20 11:59:14 +00002934 msi_free(Path);
Aric Stewart9adacf62005-06-24 11:58:21 +00002935 }
2936
2937 TRACE("Writing shortcut to %s\n",debugstr_w(target_file));
2938 IPersistFile_Save(pf,target_file,FALSE);
2939
Mike McCormackee034ba2005-09-20 11:59:14 +00002940 msi_free(target_file);
Aric Stewart9adacf62005-06-24 11:58:21 +00002941
2942 IPersistFile_Release( pf );
2943 IShellLinkW_Release( sl );
2944
2945 return ERROR_SUCCESS;
2946}
2947
2948static UINT ACTION_CreateShortcuts(MSIPACKAGE *package)
2949{
2950 UINT rc;
2951 HRESULT res;
2952 MSIQUERY * view;
2953 static const WCHAR Query[] =
2954 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
2955 '`','S','h','o','r','t','c','u','t','`',0};
2956
Aric Stewart9adacf62005-06-24 11:58:21 +00002957 rc = MSI_DatabaseOpenViewW(package->db, Query, &view);
2958 if (rc != ERROR_SUCCESS)
2959 return ERROR_SUCCESS;
2960
Aric Stewart2cf222f2004-07-06 19:00:23 +00002961 res = CoInitialize( NULL );
2962 if (FAILED (res))
2963 {
2964 ERR("CoInitialize failed\n");
2965 return ERROR_FUNCTION_FAILED;
2966 }
2967
Aric Stewart9adacf62005-06-24 11:58:21 +00002968 rc = MSI_IterateRecords(view, NULL, ITERATE_CreateShortcuts, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002969 msiobj_release(&view->hdr);
Aric Stewart2cf222f2004-07-06 19:00:23 +00002970
Aric Stewart2cf222f2004-07-06 19:00:23 +00002971 CoUninitialize();
2972
2973 return rc;
2974}
2975
Aric Stewart916ef942005-06-22 18:42:19 +00002976static UINT ITERATE_PublishProduct(MSIRECORD *row, LPVOID param)
2977{
2978 MSIPACKAGE* package = (MSIPACKAGE*)param;
2979 HANDLE the_file;
Mike McCormack75658d72005-09-22 10:33:57 +00002980 LPWSTR FilePath;
2981 LPCWSTR FileName;
Aric Stewart916ef942005-06-22 18:42:19 +00002982 CHAR buffer[1024];
2983 DWORD sz;
2984 UINT rc;
2985
2986 FileName = MSI_RecordGetString(row,1);
2987 if (!FileName)
2988 {
2989 ERR("Unable to get FileName\n");
2990 return ERROR_SUCCESS;
2991 }
2992
Mike McCormack75658d72005-09-22 10:33:57 +00002993 FilePath = build_icon_path(package,FileName);
Aric Stewart916ef942005-06-22 18:42:19 +00002994
2995 TRACE("Creating icon file at %s\n",debugstr_w(FilePath));
2996
2997 the_file = CreateFileW(FilePath, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS,
2998 FILE_ATTRIBUTE_NORMAL, NULL);
2999
3000 if (the_file == INVALID_HANDLE_VALUE)
3001 {
3002 ERR("Unable to create file %s\n",debugstr_w(FilePath));
Mike McCormackee034ba2005-09-20 11:59:14 +00003003 msi_free(FilePath);
Aric Stewart916ef942005-06-22 18:42:19 +00003004 return ERROR_SUCCESS;
3005 }
3006
3007 do
3008 {
3009 DWORD write;
3010 sz = 1024;
3011 rc = MSI_RecordReadStream(row,2,buffer,&sz);
3012 if (rc != ERROR_SUCCESS)
3013 {
3014 ERR("Failed to get stream\n");
3015 CloseHandle(the_file);
3016 DeleteFileW(FilePath);
3017 break;
3018 }
3019 WriteFile(the_file,buffer,sz,&write,NULL);
3020 } while (sz == 1024);
3021
Mike McCormackee034ba2005-09-20 11:59:14 +00003022 msi_free(FilePath);
Aric Stewart916ef942005-06-22 18:42:19 +00003023
3024 CloseHandle(the_file);
3025 return ERROR_SUCCESS;
3026}
Aric Stewart2cf222f2004-07-06 19:00:23 +00003027
3028/*
3029 * 99% of the work done here is only done for
3030 * advertised installs. However this is where the
3031 * Icon table is processed and written out
Francois Gouget817c5202004-07-16 19:15:40 +00003032 * so that is what I am going to do here.
Aric Stewart2cf222f2004-07-06 19:00:23 +00003033 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00003034static UINT ACTION_PublishProduct(MSIPACKAGE *package)
Aric Stewart2cf222f2004-07-06 19:00:23 +00003035{
3036 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00003037 MSIQUERY * view;
Aric Stewart8e233e92005-03-01 11:45:19 +00003038 static const WCHAR Query[]=
3039 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00003040 '`','I','c','o','n','`',0};
Aric Stewart6269f002005-01-17 13:40:39 +00003041 /* for registry stuff */
Aric Stewart68b07492005-01-25 11:05:37 +00003042 HKEY hkey=0;
3043 HKEY hukey=0;
Aric Stewart6957e4a2005-06-08 19:16:45 +00003044 static const WCHAR szProductLanguage[] =
3045 {'P','r','o','d','u','c','t','L','a','n','g','u','a','g','e',0};
Aric Stewart6957e4a2005-06-08 19:16:45 +00003046 static const WCHAR szARPProductIcon[] =
3047 {'A','R','P','P','R','O','D','U','C','T','I','C','O','N',0};
Aric Stewartc28bb542005-06-09 15:49:11 +00003048 static const WCHAR szProductVersion[] =
3049 {'P','r','o','d','u','c','t','V','e','r','s','i','o','n',0};
Aric Stewart6957e4a2005-06-08 19:16:45 +00003050 DWORD langid;
Aric Stewart6269f002005-01-17 13:40:39 +00003051 LPWSTR buffer;
3052 DWORD size;
Aric Stewart68b07492005-01-25 11:05:37 +00003053 MSIHANDLE hDb, hSumInfo;
Aric Stewart2cf222f2004-07-06 19:00:23 +00003054
Aric Stewart916ef942005-06-22 18:42:19 +00003055 /* write out icon files */
3056
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00003057 rc = MSI_DatabaseOpenViewW(package->db, Query, &view);
Aric Stewart916ef942005-06-22 18:42:19 +00003058 if (rc == ERROR_SUCCESS)
Aric Stewart2cf222f2004-07-06 19:00:23 +00003059 {
Aric Stewart916ef942005-06-22 18:42:19 +00003060 MSI_IterateRecords(view, NULL, ITERATE_PublishProduct, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00003061 msiobj_release(&view->hdr);
Aric Stewart2cf222f2004-07-06 19:00:23 +00003062 }
3063
Francois Gougetda8b3dd2005-01-26 21:09:04 +00003064 /* ok there is a lot more done here but i need to figure out what */
Aric Stewart916ef942005-06-22 18:42:19 +00003065
Aric Stewartadaef112005-07-07 20:27:06 +00003066 rc = MSIREG_OpenProductsKey(package->ProductCode,&hkey,TRUE);
Aric Stewart6269f002005-01-17 13:40:39 +00003067 if (rc != ERROR_SUCCESS)
3068 goto end;
3069
Aric Stewartadaef112005-07-07 20:27:06 +00003070 rc = MSIREG_OpenUserProductsKey(package->ProductCode,&hukey,TRUE);
Aric Stewart6269f002005-01-17 13:40:39 +00003071 if (rc != ERROR_SUCCESS)
3072 goto end;
3073
Aric Stewart6269f002005-01-17 13:40:39 +00003074
Mike McCormack062ad502005-09-15 15:04:08 +00003075 buffer = msi_dup_property( package, INSTALLPROPERTY_PRODUCTNAMEW );
Mike McCormack4db02cd2005-09-15 14:58:38 +00003076 msi_reg_set_val_str( hukey, INSTALLPROPERTY_PRODUCTNAMEW, buffer );
Mike McCormackee034ba2005-09-20 11:59:14 +00003077 msi_free(buffer);
Aric Stewart6957e4a2005-06-08 19:16:45 +00003078
Mike McCormack74f0de92005-09-29 10:32:39 +00003079 langid = msi_get_property_int( package, szProductLanguage, 0 );
Mike McCormack4db02cd2005-09-15 14:58:38 +00003080 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_LANGUAGEW, langid );
Aric Stewart6957e4a2005-06-08 19:16:45 +00003081
Mike McCormack062ad502005-09-15 15:04:08 +00003082 buffer = msi_dup_property( package, szARPProductIcon );
Aric Stewart6957e4a2005-06-08 19:16:45 +00003083 if (buffer)
3084 {
Mike McCormack75658d72005-09-22 10:33:57 +00003085 LPWSTR path = build_icon_path(package,buffer);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003086 msi_reg_set_val_str( hukey, INSTALLPROPERTY_PRODUCTICONW, path );
Mike McCormack75658d72005-09-22 10:33:57 +00003087 msi_free( path );
Aric Stewart6957e4a2005-06-08 19:16:45 +00003088 }
Mike McCormackee034ba2005-09-20 11:59:14 +00003089 msi_free(buffer);
Aric Stewartc28bb542005-06-09 15:49:11 +00003090
Mike McCormack062ad502005-09-15 15:04:08 +00003091 buffer = msi_dup_property( package, szProductVersion );
Aric Stewartc28bb542005-06-09 15:49:11 +00003092 if (buffer)
3093 {
3094 DWORD verdword = build_version_dword(buffer);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003095 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_VERSIONW, verdword );
Aric Stewartc28bb542005-06-09 15:49:11 +00003096 }
Mike McCormackee034ba2005-09-20 11:59:14 +00003097 msi_free(buffer);
Aric Stewart6957e4a2005-06-08 19:16:45 +00003098
Aric Stewart2cae30b2005-01-19 19:07:40 +00003099 FIXME("Need to write more keys to the user registry\n");
Aric Stewart68b07492005-01-25 11:05:37 +00003100
Aric Stewart7e7b8cf2005-02-18 20:00:34 +00003101 hDb= alloc_msihandle( &package->db->hdr );
Aric Stewart68b07492005-01-25 11:05:37 +00003102 rc = MsiGetSummaryInformationW(hDb, NULL, 0, &hSumInfo);
Aric Stewart7e7b8cf2005-02-18 20:00:34 +00003103 MsiCloseHandle(hDb);
Aric Stewart68b07492005-01-25 11:05:37 +00003104 if (rc == ERROR_SUCCESS)
3105 {
3106 WCHAR guidbuffer[0x200];
3107 size = 0x200;
Aric Stewart7e7b8cf2005-02-18 20:00:34 +00003108 rc = MsiSummaryInfoGetPropertyW(hSumInfo, 9, NULL, NULL, NULL,
Aric Stewart68b07492005-01-25 11:05:37 +00003109 guidbuffer, &size);
3110 if (rc == ERROR_SUCCESS)
3111 {
3112 WCHAR squashed[GUID_SIZE];
3113 /* for now we only care about the first guid */
3114 LPWSTR ptr = strchrW(guidbuffer,';');
3115 if (ptr) *ptr = 0;
3116 squash_guid(guidbuffer,squashed);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003117 msi_reg_set_val_str( hukey, INSTALLPROPERTY_PACKAGECODEW, squashed );
Aric Stewart68b07492005-01-25 11:05:37 +00003118 }
3119 else
3120 {
3121 ERR("Unable to query Revision_Number... \n");
3122 rc = ERROR_SUCCESS;
3123 }
3124 MsiCloseHandle(hSumInfo);
3125 }
3126 else
3127 {
3128 ERR("Unable to open Summary Information\n");
3129 rc = ERROR_SUCCESS;
3130 }
Aric Stewart2cae30b2005-01-19 19:07:40 +00003131
Aric Stewart6269f002005-01-17 13:40:39 +00003132end:
3133
Aric Stewart6269f002005-01-17 13:40:39 +00003134 RegCloseKey(hkey);
Aric Stewart6269f002005-01-17 13:40:39 +00003135 RegCloseKey(hukey);
3136
Aric Stewart2cf222f2004-07-06 19:00:23 +00003137 return rc;
Aric Stewart2cf222f2004-07-06 19:00:23 +00003138}
3139
Aric Stewartaded32f2005-06-23 09:46:31 +00003140static UINT ITERATE_WriteIniValues(MSIRECORD *row, LPVOID param)
3141{
3142 MSIPACKAGE *package = (MSIPACKAGE*)param;
3143 LPCWSTR component,section,key,value,identifier,filename,dirproperty;
3144 LPWSTR deformated_section, deformated_key, deformated_value;
3145 LPWSTR folder, fullname = NULL;
3146 MSIRECORD * uirow;
Mike McCormack38d67a42005-08-22 09:15:23 +00003147 INT action;
3148 MSICOMPONENT *comp;
Aric Stewartaded32f2005-06-23 09:46:31 +00003149 static const WCHAR szWindowsFolder[] =
3150 {'W','i','n','d','o','w','s','F','o','l','d','e','r',0};
3151
3152 component = MSI_RecordGetString(row, 8);
Mike McCormack38d67a42005-08-22 09:15:23 +00003153 comp = get_loaded_component(package,component);
Aric Stewartaded32f2005-06-23 09:46:31 +00003154
Mike McCormackd693f462005-10-29 11:36:48 +00003155 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
Aric Stewartaded32f2005-06-23 09:46:31 +00003156 {
3157 TRACE("Skipping ini file due to disabled component %s\n",
3158 debugstr_w(component));
3159
Mike McCormack38d67a42005-08-22 09:15:23 +00003160 comp->Action = comp->Installed;
Aric Stewartaded32f2005-06-23 09:46:31 +00003161
3162 return ERROR_SUCCESS;
3163 }
3164
Mike McCormack38d67a42005-08-22 09:15:23 +00003165 comp->Action = INSTALLSTATE_LOCAL;
Aric Stewartaded32f2005-06-23 09:46:31 +00003166
3167 identifier = MSI_RecordGetString(row,1);
3168 filename = MSI_RecordGetString(row,2);
3169 dirproperty = MSI_RecordGetString(row,3);
3170 section = MSI_RecordGetString(row,4);
3171 key = MSI_RecordGetString(row,5);
3172 value = MSI_RecordGetString(row,6);
3173 action = MSI_RecordGetInteger(row,7);
3174
3175 deformat_string(package,section,&deformated_section);
3176 deformat_string(package,key,&deformated_key);
3177 deformat_string(package,value,&deformated_value);
3178
3179 if (dirproperty)
3180 {
3181 folder = resolve_folder(package, dirproperty, FALSE, FALSE, NULL);
3182 if (!folder)
Mike McCormack062ad502005-09-15 15:04:08 +00003183 folder = msi_dup_property( package, dirproperty );
Aric Stewartaded32f2005-06-23 09:46:31 +00003184 }
3185 else
Mike McCormack062ad502005-09-15 15:04:08 +00003186 folder = msi_dup_property( package, szWindowsFolder );
Aric Stewartaded32f2005-06-23 09:46:31 +00003187
3188 if (!folder)
3189 {
3190 ERR("Unable to resolve folder! (%s)\n",debugstr_w(dirproperty));
3191 goto cleanup;
3192 }
3193
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003194 fullname = build_directory_name(2, folder, filename);
Aric Stewartaded32f2005-06-23 09:46:31 +00003195
3196 if (action == 0)
3197 {
3198 TRACE("Adding value %s to section %s in %s\n",
3199 debugstr_w(deformated_key), debugstr_w(deformated_section),
3200 debugstr_w(fullname));
3201 WritePrivateProfileStringW(deformated_section, deformated_key,
3202 deformated_value, fullname);
3203 }
3204 else if (action == 1)
3205 {
3206 WCHAR returned[10];
3207 GetPrivateProfileStringW(deformated_section, deformated_key, NULL,
3208 returned, 10, fullname);
3209 if (returned[0] == 0)
3210 {
3211 TRACE("Adding value %s to section %s in %s\n",
3212 debugstr_w(deformated_key), debugstr_w(deformated_section),
3213 debugstr_w(fullname));
3214
3215 WritePrivateProfileStringW(deformated_section, deformated_key,
3216 deformated_value, fullname);
3217 }
3218 }
3219 else if (action == 3)
3220 FIXME("Append to existing section not yet implemented\n");
3221
3222 uirow = MSI_CreateRecord(4);
3223 MSI_RecordSetStringW(uirow,1,identifier);
3224 MSI_RecordSetStringW(uirow,2,deformated_section);
3225 MSI_RecordSetStringW(uirow,3,deformated_key);
3226 MSI_RecordSetStringW(uirow,4,deformated_value);
3227 ui_actiondata(package,szWriteIniValues,uirow);
3228 msiobj_release( &uirow->hdr );
3229cleanup:
Mike McCormackee034ba2005-09-20 11:59:14 +00003230 msi_free(fullname);
3231 msi_free(folder);
3232 msi_free(deformated_key);
3233 msi_free(deformated_value);
3234 msi_free(deformated_section);
Aric Stewartaded32f2005-06-23 09:46:31 +00003235 return ERROR_SUCCESS;
3236}
3237
Aric Stewart516a9c72005-01-14 15:59:26 +00003238static UINT ACTION_WriteIniValues(MSIPACKAGE *package)
3239{
3240 UINT rc;
3241 MSIQUERY * view;
Aric Stewart8e233e92005-03-01 11:45:19 +00003242 static const WCHAR ExecSeqQuery[] =
3243 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00003244 '`','I','n','i','F','i','l','e','`',0};
Aric Stewart516a9c72005-01-14 15:59:26 +00003245
3246 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
3247 if (rc != ERROR_SUCCESS)
3248 {
3249 TRACE("no IniFile table\n");
3250 return ERROR_SUCCESS;
3251 }
3252
Aric Stewartaded32f2005-06-23 09:46:31 +00003253 rc = MSI_IterateRecords(view, NULL, ITERATE_WriteIniValues, package);
Aric Stewart516a9c72005-01-14 15:59:26 +00003254 msiobj_release(&view->hdr);
3255 return rc;
3256}
3257
Aric Stewart854bfc42005-06-24 11:33:02 +00003258static UINT ITERATE_SelfRegModules(MSIRECORD *row, LPVOID param)
Aric Stewart6269f002005-01-17 13:40:39 +00003259{
Aric Stewart854bfc42005-06-24 11:33:02 +00003260 MSIPACKAGE *package = (MSIPACKAGE*)param;
3261 LPCWSTR filename;
3262 LPWSTR FullName;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003263 MSIFILE *file;
Aric Stewart854bfc42005-06-24 11:33:02 +00003264 DWORD len;
Aric Stewart8e233e92005-03-01 11:45:19 +00003265 static const WCHAR ExeStr[] =
Aric Stewartc5a14432005-05-18 17:46:12 +00003266 {'r','e','g','s','v','r','3','2','.','e','x','e',' ','\"',0};
3267 static const WCHAR close[] = {'\"',0};
Aric Stewart2cae30b2005-01-19 19:07:40 +00003268 STARTUPINFOW si;
3269 PROCESS_INFORMATION info;
3270 BOOL brc;
3271
3272 memset(&si,0,sizeof(STARTUPINFOW));
3273
Aric Stewart854bfc42005-06-24 11:33:02 +00003274 filename = MSI_RecordGetString(row,1);
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003275 file = get_loaded_file( package, filename );
Aric Stewart854bfc42005-06-24 11:33:02 +00003276
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003277 if (!file)
Aric Stewart854bfc42005-06-24 11:33:02 +00003278 {
3279 ERR("Unable to find file id %s\n",debugstr_w(filename));
3280 return ERROR_SUCCESS;
3281 }
3282
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003283 len = strlenW(ExeStr) + strlenW( file->TargetPath ) + 2;
Aric Stewart854bfc42005-06-24 11:33:02 +00003284
Mike McCormackee034ba2005-09-20 11:59:14 +00003285 FullName = msi_alloc(len*sizeof(WCHAR));
Aric Stewart854bfc42005-06-24 11:33:02 +00003286 strcpyW(FullName,ExeStr);
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003287 strcatW( FullName, file->TargetPath );
Aric Stewart854bfc42005-06-24 11:33:02 +00003288 strcatW(FullName,close);
3289
3290 TRACE("Registering %s\n",debugstr_w(FullName));
3291 brc = CreateProcessW(NULL, FullName, NULL, NULL, FALSE, 0, NULL, c_colon,
3292 &si, &info);
3293
3294 if (brc)
3295 msi_dialog_check_messages(info.hProcess);
3296
Mike McCormackee034ba2005-09-20 11:59:14 +00003297 msi_free(FullName);
Aric Stewart854bfc42005-06-24 11:33:02 +00003298 return ERROR_SUCCESS;
3299}
3300
3301static UINT ACTION_SelfRegModules(MSIPACKAGE *package)
3302{
3303 UINT rc;
3304 MSIQUERY * view;
3305 static const WCHAR ExecSeqQuery[] =
3306 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
3307 '`','S','e','l','f','R','e','g','`',0};
3308
Aric Stewart6269f002005-01-17 13:40:39 +00003309 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
3310 if (rc != ERROR_SUCCESS)
3311 {
3312 TRACE("no SelfReg table\n");
3313 return ERROR_SUCCESS;
3314 }
3315
Aric Stewart854bfc42005-06-24 11:33:02 +00003316 MSI_IterateRecords(view, NULL, ITERATE_SelfRegModules, package);
Aric Stewart6269f002005-01-17 13:40:39 +00003317 msiobj_release(&view->hdr);
Aric Stewart854bfc42005-06-24 11:33:02 +00003318
3319 return ERROR_SUCCESS;
Aric Stewart6269f002005-01-17 13:40:39 +00003320}
3321
3322static UINT ACTION_PublishFeatures(MSIPACKAGE *package)
3323{
Mike McCormack1da28582005-08-22 14:09:17 +00003324 MSIFEATURE *feature;
Aric Stewart6269f002005-01-17 13:40:39 +00003325 UINT rc;
Aric Stewart68b07492005-01-25 11:05:37 +00003326 HKEY hkey=0;
3327 HKEY hukey=0;
Aric Stewart68b07492005-01-25 11:05:37 +00003328
Aric Stewartadaef112005-07-07 20:27:06 +00003329 rc = MSIREG_OpenFeaturesKey(package->ProductCode,&hkey,TRUE);
Aric Stewart6269f002005-01-17 13:40:39 +00003330 if (rc != ERROR_SUCCESS)
3331 goto end;
3332
Aric Stewartadaef112005-07-07 20:27:06 +00003333 rc = MSIREG_OpenUserFeaturesKey(package->ProductCode,&hukey,TRUE);
Aric Stewart6269f002005-01-17 13:40:39 +00003334 if (rc != ERROR_SUCCESS)
3335 goto end;
3336
3337 /* here the guids are base 85 encoded */
Mike McCormack1da28582005-08-22 14:09:17 +00003338 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewart6269f002005-01-17 13:40:39 +00003339 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003340 ComponentList *cl;
Aric Stewart6269f002005-01-17 13:40:39 +00003341 LPWSTR data = NULL;
3342 GUID clsid;
Aric Stewart6269f002005-01-17 13:40:39 +00003343 INT size;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003344 BOOL absent = FALSE;
Aric Stewart6269f002005-01-17 13:40:39 +00003345
Mike McCormack1da28582005-08-22 14:09:17 +00003346 if (!ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_LOCAL ) &&
3347 !ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_SOURCE ) &&
3348 !ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_ADVERTISED ))
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003349 absent = TRUE;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00003350
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003351 size = 1;
Mike McCormack1da28582005-08-22 14:09:17 +00003352 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003353 {
3354 size += 21;
3355 }
Mike McCormack79ca56c2005-09-13 10:37:37 +00003356 if (feature->Feature_Parent)
Mike McCormack1da28582005-08-22 14:09:17 +00003357 size += strlenW( feature->Feature_Parent )+2;
Aric Stewart6269f002005-01-17 13:40:39 +00003358
Mike McCormackee034ba2005-09-20 11:59:14 +00003359 data = msi_alloc(size * sizeof(WCHAR));
Aric Stewart6269f002005-01-17 13:40:39 +00003360
3361 data[0] = 0;
Mike McCormack1da28582005-08-22 14:09:17 +00003362 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewart6269f002005-01-17 13:40:39 +00003363 {
Mike McCormack38d67a42005-08-22 09:15:23 +00003364 MSICOMPONENT* component = cl->component;
Aric Stewart6269f002005-01-17 13:40:39 +00003365 WCHAR buf[21];
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003366
Aric Stewart6269f002005-01-17 13:40:39 +00003367 memset(buf,0,sizeof(buf));
Mike McCormackefcc1ec2005-09-12 12:07:15 +00003368 if (component->ComponentId)
Aric Stewartc5a14432005-05-18 17:46:12 +00003369 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003370 TRACE("From %s\n",debugstr_w(component->ComponentId));
3371 CLSIDFromString(component->ComponentId, &clsid);
Aric Stewartc5a14432005-05-18 17:46:12 +00003372 encode_base85_guid(&clsid,buf);
3373 TRACE("to %s\n",debugstr_w(buf));
3374 strcatW(data,buf);
3375 }
Aric Stewart6269f002005-01-17 13:40:39 +00003376 }
Mike McCormack79ca56c2005-09-13 10:37:37 +00003377 if (feature->Feature_Parent)
Aric Stewart6269f002005-01-17 13:40:39 +00003378 {
3379 static const WCHAR sep[] = {'\2',0};
3380 strcatW(data,sep);
Mike McCormack1da28582005-08-22 14:09:17 +00003381 strcatW(data,feature->Feature_Parent);
Aric Stewart6269f002005-01-17 13:40:39 +00003382 }
3383
Mike McCormack4db02cd2005-09-15 14:58:38 +00003384 msi_reg_set_val_str( hkey, feature->Feature, data );
Mike McCormackee034ba2005-09-20 11:59:14 +00003385 msi_free(data);
Aric Stewart6269f002005-01-17 13:40:39 +00003386
Mike McCormack79ca56c2005-09-13 10:37:37 +00003387 size = 0;
3388 if (feature->Feature_Parent)
3389 size = strlenW(feature->Feature_Parent)*sizeof(WCHAR);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003390 if (!absent)
3391 {
Mike McCormack1da28582005-08-22 14:09:17 +00003392 RegSetValueExW(hukey,feature->Feature,0,REG_SZ,
3393 (LPBYTE)feature->Feature_Parent,size);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003394 }
3395 else
3396 {
Mike McCormack79ca56c2005-09-13 10:37:37 +00003397 size += 2*sizeof(WCHAR);
Mike McCormackee034ba2005-09-20 11:59:14 +00003398 data = msi_alloc(size);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003399 data[0] = 0x6;
Mike McCormack79ca56c2005-09-13 10:37:37 +00003400 data[1] = 0;
3401 if (feature->Feature_Parent)
3402 strcpyW( &data[1], feature->Feature_Parent );
Mike McCormack1da28582005-08-22 14:09:17 +00003403 RegSetValueExW(hukey,feature->Feature,0,REG_SZ,
Mike McCormack16466af2005-07-06 10:33:30 +00003404 (LPBYTE)data,size);
Mike McCormackee034ba2005-09-20 11:59:14 +00003405 msi_free(data);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003406 }
Aric Stewart6269f002005-01-17 13:40:39 +00003407 }
3408
Aric Stewart6269f002005-01-17 13:40:39 +00003409end:
Aric Stewart6269f002005-01-17 13:40:39 +00003410 RegCloseKey(hkey);
3411 RegCloseKey(hukey);
3412 return rc;
3413}
3414
Mike McCormack61f24a42005-09-30 10:32:41 +00003415static UINT msi_make_package_local( MSIPACKAGE *package, HKEY hkey )
3416{
3417 static const WCHAR installerPathFmt[] = {
3418 '%','s','\\','I','n','s','t','a','l','l','e','r','\\',0};
3419 static const WCHAR fmt[] = {
3420 '%','s','\\',
3421 'I','n','s','t','a','l','l','e','r','\\',
3422 '%','x','.','m','s','i',0};
3423 static const WCHAR szOriginalDatabase[] =
3424 {'O','r','i','g','i','n','a','l','D','a','t','a','b','a','s','e',0};
3425 WCHAR windir[MAX_PATH], path[MAX_PATH], packagefile[MAX_PATH];
3426 INT num, start;
3427 LPWSTR msiFilePath;
3428 BOOL r;
3429
3430 /* copy the package locally */
3431 num = GetTickCount() & 0xffff;
3432 if (!num)
3433 num = 1;
3434 start = num;
3435 GetWindowsDirectoryW( windir, MAX_PATH );
3436 snprintfW( packagefile, MAX_PATH, fmt, windir, num );
3437 do
3438 {
3439 HANDLE handle = CreateFileW(packagefile,GENERIC_WRITE, 0, NULL,
3440 CREATE_NEW, FILE_ATTRIBUTE_NORMAL, 0 );
3441 if (handle != INVALID_HANDLE_VALUE)
3442 {
3443 CloseHandle(handle);
3444 break;
3445 }
3446 if (GetLastError() != ERROR_FILE_EXISTS &&
3447 GetLastError() != ERROR_SHARING_VIOLATION)
3448 break;
3449 if (!(++num & 0xffff)) num = 1;
3450 sprintfW(packagefile,fmt,num);
3451 } while (num != start);
3452
3453 snprintfW( path, MAX_PATH, installerPathFmt, windir );
3454 create_full_pathW(path);
3455
3456 TRACE("Copying to local package %s\n",debugstr_w(packagefile));
3457
3458 msiFilePath = msi_dup_property( package, szOriginalDatabase );
3459 r = CopyFileW( msiFilePath, packagefile, FALSE);
3460 msi_free( msiFilePath );
3461
3462 if (!r)
3463 {
3464 ERR("Unable to copy package (%s -> %s) (error %ld)\n",
3465 debugstr_w(msiFilePath), debugstr_w(packagefile), GetLastError());
3466 return ERROR_FUNCTION_FAILED;
3467 }
3468
3469 /* FIXME: maybe set this key in ACTION_RegisterProduct instead */
3470 msi_reg_set_val_str( hkey, INSTALLPROPERTY_LOCALPACKAGEW, packagefile );
3471 return ERROR_SUCCESS;
3472}
3473
Mike McCormackba293ee2005-10-27 12:08:16 +00003474static UINT msi_write_uninstall_property_vals( MSIPACKAGE *package, HKEY hkey )
3475{
3476 LPWSTR prop, val, key;
3477 static const LPCSTR propval[] = {
3478 "ARPAUTHORIZEDCDFPREFIX", "AuthorizedCDFPrefix",
3479 "ARPCONTACT", "Contact",
3480 "ARPCOMMENTS", "Comments",
3481 "ProductName", "DisplayName",
3482 "ProductVersion", "DisplayVersion",
3483 "ARPHELPLINK", "HelpLink",
3484 "ARPHELPTELEPHONE", "HelpTelephone",
3485 "ARPINSTALLLOCATION", "InstallLocation",
3486 "SourceDir", "InstallSource",
3487 "Manufacturer", "Publisher",
3488 "ARPREADME", "Readme",
3489 "ARPSIZE", "Size",
3490 "ARPURLINFOABOUT", "URLInfoAbout",
3491 "ARPURLUPDATEINFO", "URLUpdateInfo",
3492 NULL,
3493 };
3494 const LPCSTR *p = propval;
3495
3496 while( *p )
3497 {
3498 prop = strdupAtoW( *p++ );
3499 key = strdupAtoW( *p++ );
3500 val = msi_dup_property( package, prop );
3501 msi_reg_set_val_str( hkey, key, val );
3502 msi_free(val);
3503 msi_free(key);
3504 msi_free(prop);
3505 }
3506 return ERROR_SUCCESS;
3507}
3508
Aric Stewart2cae30b2005-01-19 19:07:40 +00003509static UINT ACTION_RegisterProduct(MSIPACKAGE *package)
3510{
Aric Stewart68b07492005-01-25 11:05:37 +00003511 HKEY hkey=0;
Aric Stewarte9db87b2005-06-17 21:25:41 +00003512 LPWSTR buffer = NULL;
Mike McCormackba293ee2005-10-27 12:08:16 +00003513 UINT rc;
Mike McCormack74f0de92005-09-29 10:32:39 +00003514 DWORD size, langid;
Mike McCormack4db02cd2005-09-15 14:58:38 +00003515 static const WCHAR szWindowsInstaller[] =
Mike McCormackba293ee2005-10-27 12:08:16 +00003516 {'W','i','n','d','o','w','s','I','n','s','t','a','l','l','e','r',0};
Aric Stewart36a01502005-06-08 19:07:52 +00003517 static const WCHAR szUpgradeCode[] =
3518 {'U','p','g','r','a','d','e','C','o','d','e',0};
Aric Stewarte9db87b2005-06-17 21:25:41 +00003519 static const WCHAR modpath_fmt[] =
Mike McCormackba293ee2005-10-27 12:08:16 +00003520 {'M','s','i','E','x','e','c','.','e','x','e',' ',
3521 '/','I','[','P','r','o','d','u','c','t','C','o','d','e',']',0};
Aric Stewarte9db87b2005-06-17 21:25:41 +00003522 static const WCHAR szModifyPath[] =
3523 {'M','o','d','i','f','y','P','a','t','h',0};
3524 static const WCHAR szUninstallString[] =
3525 {'U','n','i','n','s','t','a','l','l','S','t','r','i','n','g',0};
3526 static const WCHAR szEstimatedSize[] =
3527 {'E','s','t','i','m','a','t','e','d','S','i','z','e',0};
Aric Stewarte9db87b2005-06-17 21:25:41 +00003528 static const WCHAR szProductLanguage[] =
3529 {'P','r','o','d','u','c','t','L','a','n','g','u','a','g','e',0};
3530 static const WCHAR szProductVersion[] =
3531 {'P','r','o','d','u','c','t','V','e','r','s','i','o','n',0};
Aric Stewarte9db87b2005-06-17 21:25:41 +00003532
3533 SYSTEMTIME systime;
3534 static const WCHAR date_fmt[] = {'%','i','%','i','%','i',0};
Aric Stewart36a01502005-06-08 19:07:52 +00003535 LPWSTR upgrade_code;
Mike McCormackba293ee2005-10-27 12:08:16 +00003536 WCHAR szDate[9];
Aric Stewart2cae30b2005-01-19 19:07:40 +00003537
Aric Stewartadaef112005-07-07 20:27:06 +00003538 rc = MSIREG_OpenUninstallKey(package->ProductCode,&hkey,TRUE);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003539 if (rc != ERROR_SUCCESS)
Mike McCormackba293ee2005-10-27 12:08:16 +00003540 return rc;
Aric Stewart2cae30b2005-01-19 19:07:40 +00003541
3542 /* dump all the info i can grab */
3543 FIXME("Flesh out more information \n");
3544
Mike McCormackba293ee2005-10-27 12:08:16 +00003545 msi_write_uninstall_property_vals( package, hkey );
Aric Stewart2cae30b2005-01-19 19:07:40 +00003546
Mike McCormack4db02cd2005-09-15 14:58:38 +00003547 msi_reg_set_val_dword( hkey, szWindowsInstaller, 1 );
Aric Stewart2cae30b2005-01-19 19:07:40 +00003548
Mike McCormack61f24a42005-09-30 10:32:41 +00003549 msi_make_package_local( package, hkey );
Aric Stewart36a01502005-06-08 19:07:52 +00003550
Aric Stewarte9db87b2005-06-17 21:25:41 +00003551 /* do ModifyPath and UninstallString */
3552 size = deformat_string(package,modpath_fmt,&buffer);
Mike McCormack16466af2005-07-06 10:33:30 +00003553 RegSetValueExW(hkey,szModifyPath,0,REG_EXPAND_SZ,(LPBYTE)buffer,size);
3554 RegSetValueExW(hkey,szUninstallString,0,REG_EXPAND_SZ,(LPBYTE)buffer,size);
Mike McCormackee034ba2005-09-20 11:59:14 +00003555 msi_free(buffer);
Aric Stewarte9db87b2005-06-17 21:25:41 +00003556
3557 FIXME("Write real Estimated Size when we have it\n");
Mike McCormack4db02cd2005-09-15 14:58:38 +00003558 msi_reg_set_val_dword( hkey, szEstimatedSize, 0 );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003559
3560 GetLocalTime(&systime);
Mike McCormackba293ee2005-10-27 12:08:16 +00003561 sprintfW(szDate,date_fmt,systime.wYear,systime.wMonth,systime.wDay);
3562 msi_reg_set_val_str( hkey, INSTALLPROPERTY_INSTALLDATEW, szDate );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003563
Mike McCormack74f0de92005-09-29 10:32:39 +00003564 langid = msi_get_property_int( package, szProductLanguage, 0 );
3565 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_LANGUAGEW, langid );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003566
Mike McCormack062ad502005-09-15 15:04:08 +00003567 buffer = msi_dup_property( package, szProductVersion );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003568 if (buffer)
3569 {
3570 DWORD verdword = build_version_dword(buffer);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003571
3572 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_VERSIONW, verdword );
3573 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_VERSIONMAJORW, verdword>>24 );
3574 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_VERSIONMINORW, (verdword>>16)&0x00FF );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003575 }
Mike McCormackee034ba2005-09-20 11:59:14 +00003576 msi_free(buffer);
Aric Stewarte9db87b2005-06-17 21:25:41 +00003577
Aric Stewart36a01502005-06-08 19:07:52 +00003578 /* Handle Upgrade Codes */
Mike McCormack062ad502005-09-15 15:04:08 +00003579 upgrade_code = msi_dup_property( package, szUpgradeCode );
Aric Stewart36a01502005-06-08 19:07:52 +00003580 if (upgrade_code)
3581 {
3582 HKEY hkey2;
3583 WCHAR squashed[33];
3584 MSIREG_OpenUpgradeCodesKey(upgrade_code, &hkey2, TRUE);
Aric Stewartadaef112005-07-07 20:27:06 +00003585 squash_guid(package->ProductCode,squashed);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003586 msi_reg_set_val_str( hkey2, squashed, NULL );
Aric Stewart36a01502005-06-08 19:07:52 +00003587 RegCloseKey(hkey2);
3588 MSIREG_OpenUserUpgradeCodesKey(upgrade_code, &hkey2, TRUE);
Aric Stewartadaef112005-07-07 20:27:06 +00003589 squash_guid(package->ProductCode,squashed);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003590 msi_reg_set_val_str( hkey2, squashed, NULL );
Aric Stewart36a01502005-06-08 19:07:52 +00003591 RegCloseKey(hkey2);
3592
Mike McCormackee034ba2005-09-20 11:59:14 +00003593 msi_free(upgrade_code);
Aric Stewart36a01502005-06-08 19:07:52 +00003594 }
Aric Stewart2cae30b2005-01-19 19:07:40 +00003595
Aric Stewart2cae30b2005-01-19 19:07:40 +00003596 RegCloseKey(hkey);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003597
3598 return ERROR_SUCCESS;
3599}
3600
3601static UINT ACTION_InstallExecute(MSIPACKAGE *package)
3602{
Mike McCormacka977b2c2005-11-03 09:56:29 +00003603 return execute_script(package,INSTALL_SCRIPT);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003604}
3605
3606static UINT ACTION_InstallFinalize(MSIPACKAGE *package)
3607{
Aric Stewart9cd707d2005-05-27 19:24:22 +00003608 UINT rc;
3609
Aric Stewart9cd707d2005-05-27 19:24:22 +00003610 /* turn off scheduleing */
3611 package->script->CurrentlyScripting= FALSE;
3612
Aric Stewart54c67dd2005-01-25 20:17:09 +00003613 /* first do the same as an InstallExecute */
Aric Stewart9cd707d2005-05-27 19:24:22 +00003614 rc = ACTION_InstallExecute(package);
3615 if (rc != ERROR_SUCCESS)
3616 return rc;
Aric Stewart54c67dd2005-01-25 20:17:09 +00003617
3618 /* then handle Commit Actions */
Aric Stewart9cd707d2005-05-27 19:24:22 +00003619 rc = execute_script(package,COMMIT_SCRIPT);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003620
Aric Stewart9cd707d2005-05-27 19:24:22 +00003621 return rc;
Aric Stewart2cae30b2005-01-19 19:07:40 +00003622}
3623
3624static UINT ACTION_ForceReboot(MSIPACKAGE *package)
3625{
3626 static const WCHAR RunOnce[] = {
3627 'S','o','f','t','w','a','r','e','\\',
3628 'M','i','c','r','o','s','o','f','t','\\',
3629 'W','i','n','d','o','w','s','\\',
3630 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
Mike McCormack09b82642005-02-22 19:31:45 +00003631 'R','u','n','O','n','c','e',0};
Aric Stewart2cae30b2005-01-19 19:07:40 +00003632 static const WCHAR InstallRunOnce[] = {
3633 'S','o','f','t','w','a','r','e','\\',
3634 'M','i','c','r','o','s','o','f','t','\\',
3635 'W','i','n','d','o','w','s','\\',
3636 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
3637 'I','n','s','t','a','l','l','e','r','\\',
Mike McCormack09b82642005-02-22 19:31:45 +00003638 'R','u','n','O','n','c','e','E','n','t','r','i','e','s',0};
Aric Stewart2cae30b2005-01-19 19:07:40 +00003639
3640 static const WCHAR msiexec_fmt[] = {
Juan Lang014ad3b2005-03-01 10:41:52 +00003641 '%','s',
Aric Stewart2cae30b2005-01-19 19:07:40 +00003642 '\\','M','s','i','E','x','e','c','.','e','x','e',' ','/','@',' ',
3643 '\"','%','s','\"',0};
3644 static const WCHAR install_fmt[] = {
3645 '/','I',' ','\"','%','s','\"',' ',
3646 'A','F','T','E','R','R','E','B','O','O','T','=','1',' ',
3647 'R','U','N','O','N','C','E','E','N','T','R','Y','=','\"','%','s','\"',0};
Juan Lang014ad3b2005-03-01 10:41:52 +00003648 WCHAR buffer[256], sysdir[MAX_PATH];
Aric Stewartadaef112005-07-07 20:27:06 +00003649 HKEY hkey;
Mike McCormack4db02cd2005-09-15 14:58:38 +00003650 WCHAR squished_pc[100];
Aric Stewart2cae30b2005-01-19 19:07:40 +00003651
Aric Stewartadaef112005-07-07 20:27:06 +00003652 squash_guid(package->ProductCode,squished_pc);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003653
Juan Lang014ad3b2005-03-01 10:41:52 +00003654 GetSystemDirectoryW(sysdir, sizeof(sysdir)/sizeof(sysdir[0]));
Aric Stewart2cae30b2005-01-19 19:07:40 +00003655 RegCreateKeyW(HKEY_LOCAL_MACHINE,RunOnce,&hkey);
Juan Lang014ad3b2005-03-01 10:41:52 +00003656 snprintfW(buffer,sizeof(buffer)/sizeof(buffer[0]),msiexec_fmt,sysdir,
3657 squished_pc);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003658
Mike McCormack4db02cd2005-09-15 14:58:38 +00003659 msi_reg_set_val_str( hkey, squished_pc, buffer );
Aric Stewart2cae30b2005-01-19 19:07:40 +00003660 RegCloseKey(hkey);
3661
3662 TRACE("Reboot command %s\n",debugstr_w(buffer));
3663
3664 RegCreateKeyW(HKEY_LOCAL_MACHINE,InstallRunOnce,&hkey);
Aric Stewartadaef112005-07-07 20:27:06 +00003665 sprintfW(buffer,install_fmt,package->ProductCode,squished_pc);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00003666
Mike McCormack4db02cd2005-09-15 14:58:38 +00003667 msi_reg_set_val_str( hkey, squished_pc, buffer );
Aric Stewart2cae30b2005-01-19 19:07:40 +00003668 RegCloseKey(hkey);
3669
Aric Stewart68b07492005-01-25 11:05:37 +00003670 return ERROR_INSTALL_SUSPEND;
Aric Stewart2cae30b2005-01-19 19:07:40 +00003671}
3672
Aric Stewart90c57392005-01-31 16:23:12 +00003673UINT ACTION_ResolveSource(MSIPACKAGE* package)
3674{
Aric Stewart94d68182005-08-15 20:50:06 +00003675 DWORD attrib;
3676 UINT rc;
Aric Stewart90c57392005-01-31 16:23:12 +00003677 /*
3678 * we are currently doing what should be done here in the top level Install
3679 * however for Adminastrative and uninstalls this step will be needed
3680 */
Aric Stewart94d68182005-08-15 20:50:06 +00003681 if (!package->PackagePath)
3682 return ERROR_SUCCESS;
3683
3684 attrib = GetFileAttributesW(package->PackagePath);
3685 if (attrib == INVALID_FILE_ATTRIBUTES)
3686 {
3687 LPWSTR prompt;
3688 LPWSTR msg;
3689 DWORD size = 0;
3690
3691 rc = MsiSourceListGetInfoW(package->ProductCode, NULL,
3692 MSIINSTALLCONTEXT_USERMANAGED, MSICODE_PRODUCT,
3693 INSTALLPROPERTY_DISKPROMPTW,NULL,&size);
3694 if (rc == ERROR_MORE_DATA)
3695 {
Mike McCormackee034ba2005-09-20 11:59:14 +00003696 prompt = msi_alloc(size * sizeof(WCHAR));
Aric Stewart94d68182005-08-15 20:50:06 +00003697 MsiSourceListGetInfoW(package->ProductCode, NULL,
3698 MSIINSTALLCONTEXT_USERMANAGED, MSICODE_PRODUCT,
3699 INSTALLPROPERTY_DISKPROMPTW,prompt,&size);
3700 }
3701 else
3702 prompt = strdupW(package->PackagePath);
3703
3704 msg = generate_error_string(package,1302,1,prompt);
3705 while(attrib == INVALID_FILE_ATTRIBUTES)
3706 {
3707 rc = MessageBoxW(NULL,msg,NULL,MB_OKCANCEL);
3708 if (rc == IDCANCEL)
3709 {
3710 rc = ERROR_INSTALL_USEREXIT;
3711 break;
3712 }
3713 attrib = GetFileAttributesW(package->PackagePath);
3714 }
Mike McCormackee034ba2005-09-20 11:59:14 +00003715 msi_free(prompt);
Aric Stewart94d68182005-08-15 20:50:06 +00003716 rc = ERROR_SUCCESS;
3717 }
3718 else
3719 return ERROR_SUCCESS;
3720
3721 return rc;
Aric Stewart90c57392005-01-31 16:23:12 +00003722}
3723
Aric Stewartc7e88e02005-02-10 17:09:44 +00003724static UINT ACTION_RegisterUser(MSIPACKAGE *package)
3725{
Aric Stewartc7e88e02005-02-10 17:09:44 +00003726 HKEY hkey=0;
3727 LPWSTR buffer;
Aric Stewartc7e88e02005-02-10 17:09:44 +00003728 LPWSTR productid;
3729 UINT rc,i;
Aric Stewartc7e88e02005-02-10 17:09:44 +00003730
3731 static const WCHAR szPropKeys[][80] =
3732 {
Aric Stewart8e233e92005-03-01 11:45:19 +00003733 {'P','r','o','d','u','c','t','I','D',0},
3734 {'U','S','E','R','N','A','M','E',0},
3735 {'C','O','M','P','A','N','Y','N','A','M','E',0},
3736 {0},
Aric Stewartc7e88e02005-02-10 17:09:44 +00003737 };
3738
3739 static const WCHAR szRegKeys[][80] =
3740 {
Aric Stewart8e233e92005-03-01 11:45:19 +00003741 {'P','r','o','d','u','c','t','I','D',0},
3742 {'R','e','g','O','w','n','e','r',0},
3743 {'R','e','g','C','o','m','p','a','n','y',0},
3744 {0},
Aric Stewartc7e88e02005-02-10 17:09:44 +00003745 };
3746
Mike McCormack062ad502005-09-15 15:04:08 +00003747 productid = msi_dup_property( package, INSTALLPROPERTY_PRODUCTIDW );
Aric Stewartc7e88e02005-02-10 17:09:44 +00003748 if (!productid)
3749 return ERROR_SUCCESS;
3750
Aric Stewartadaef112005-07-07 20:27:06 +00003751 rc = MSIREG_OpenUninstallKey(package->ProductCode,&hkey,TRUE);
Aric Stewartc7e88e02005-02-10 17:09:44 +00003752 if (rc != ERROR_SUCCESS)
3753 goto end;
3754
Mike McCormack67189f92005-09-16 18:45:19 +00003755 for( i = 0; szPropKeys[i][0]; i++ )
Aric Stewartc7e88e02005-02-10 17:09:44 +00003756 {
Mike McCormack062ad502005-09-15 15:04:08 +00003757 buffer = msi_dup_property( package, szPropKeys[i] );
Mike McCormack4db02cd2005-09-15 14:58:38 +00003758 msi_reg_set_val_str( hkey, szRegKeys[i], buffer );
Mike McCormackee034ba2005-09-20 11:59:14 +00003759 msi_free( buffer );
Aric Stewartc7e88e02005-02-10 17:09:44 +00003760 }
3761
3762end:
Mike McCormackee034ba2005-09-20 11:59:14 +00003763 msi_free(productid);
Aric Stewartc7e88e02005-02-10 17:09:44 +00003764 RegCloseKey(hkey);
3765
3766 return ERROR_SUCCESS;
3767}
3768
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00003769
3770static UINT ACTION_ExecuteAction(MSIPACKAGE *package)
3771{
3772 UINT rc;
Aric Stewart25f1e752005-06-24 12:14:52 +00003773
Aric Stewartc9802932005-06-30 20:45:43 +00003774 package->script->InWhatSequence |= SEQUENCE_EXEC;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003775 rc = ACTION_ProcessExecSequence(package,FALSE);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00003776 return rc;
3777}
3778
Aric Stewart0af24872005-02-25 14:00:09 +00003779
3780/*
3781 * Code based off of code located here
3782 * http://www.codeproject.com/gdi/fontnamefromfile.asp
3783 *
3784 * Using string index 4 (full font name) instead of 1 (family name)
3785 */
3786static LPWSTR load_ttfname_from(LPCWSTR filename)
3787{
3788 HANDLE handle;
3789 LPWSTR ret = NULL;
3790 int i;
3791
3792 typedef struct _tagTT_OFFSET_TABLE{
3793 USHORT uMajorVersion;
3794 USHORT uMinorVersion;
3795 USHORT uNumOfTables;
3796 USHORT uSearchRange;
3797 USHORT uEntrySelector;
3798 USHORT uRangeShift;
3799 }TT_OFFSET_TABLE;
3800
3801 typedef struct _tagTT_TABLE_DIRECTORY{
3802 char szTag[4]; /* table name */
3803 ULONG uCheckSum; /* Check sum */
3804 ULONG uOffset; /* Offset from beginning of file */
3805 ULONG uLength; /* length of the table in bytes */
3806 }TT_TABLE_DIRECTORY;
3807
3808 typedef struct _tagTT_NAME_TABLE_HEADER{
3809 USHORT uFSelector; /* format selector. Always 0 */
3810 USHORT uNRCount; /* Name Records count */
3811 USHORT uStorageOffset; /* Offset for strings storage,
3812 * from start of the table */
3813 }TT_NAME_TABLE_HEADER;
3814
3815 typedef struct _tagTT_NAME_RECORD{
3816 USHORT uPlatformID;
3817 USHORT uEncodingID;
3818 USHORT uLanguageID;
3819 USHORT uNameID;
3820 USHORT uStringLength;
3821 USHORT uStringOffset; /* from start of storage area */
3822 }TT_NAME_RECORD;
Alexandre Julliardd0ee9f92005-03-02 12:23:20 +00003823
3824#define SWAPWORD(x) MAKEWORD(HIBYTE(x), LOBYTE(x))
3825#define SWAPLONG(x) MAKELONG(SWAPWORD(HIWORD(x)), SWAPWORD(LOWORD(x)))
3826
Aric Stewart0af24872005-02-25 14:00:09 +00003827 handle = CreateFileW(filename ,GENERIC_READ, 0, NULL, OPEN_EXISTING,
3828 FILE_ATTRIBUTE_NORMAL, 0 );
3829 if (handle != INVALID_HANDLE_VALUE)
3830 {
3831 TT_TABLE_DIRECTORY tblDir;
3832 BOOL bFound = FALSE;
3833 TT_OFFSET_TABLE ttOffsetTable;
Thomas Weidenmueller561f2f02005-10-29 10:29:08 +00003834 DWORD dwRead;
Aric Stewart0af24872005-02-25 14:00:09 +00003835
Thomas Weidenmueller561f2f02005-10-29 10:29:08 +00003836 ReadFile(handle,&ttOffsetTable, sizeof(TT_OFFSET_TABLE),&dwRead,NULL);
Aric Stewart0af24872005-02-25 14:00:09 +00003837 ttOffsetTable.uNumOfTables = SWAPWORD(ttOffsetTable.uNumOfTables);
3838 ttOffsetTable.uMajorVersion = SWAPWORD(ttOffsetTable.uMajorVersion);
3839 ttOffsetTable.uMinorVersion = SWAPWORD(ttOffsetTable.uMinorVersion);
3840
3841 if (ttOffsetTable.uMajorVersion != 1 ||
3842 ttOffsetTable.uMinorVersion != 0)
3843 return NULL;
3844
3845 for (i=0; i< ttOffsetTable.uNumOfTables; i++)
3846 {
Thomas Weidenmueller561f2f02005-10-29 10:29:08 +00003847 ReadFile(handle,&tblDir, sizeof(TT_TABLE_DIRECTORY),&dwRead,NULL);
Aric Stewart0af24872005-02-25 14:00:09 +00003848 if (strncmp(tblDir.szTag,"name",4)==0)
3849 {
3850 bFound = TRUE;
3851 tblDir.uLength = SWAPLONG(tblDir.uLength);
3852 tblDir.uOffset = SWAPLONG(tblDir.uOffset);
3853 break;
3854 }
3855 }
3856
3857 if (bFound)
3858 {
3859 TT_NAME_TABLE_HEADER ttNTHeader;
3860 TT_NAME_RECORD ttRecord;
3861
3862 SetFilePointer(handle, tblDir.uOffset, NULL, FILE_BEGIN);
3863 ReadFile(handle,&ttNTHeader, sizeof(TT_NAME_TABLE_HEADER),
Thomas Weidenmueller561f2f02005-10-29 10:29:08 +00003864 &dwRead,NULL);
Aric Stewart0af24872005-02-25 14:00:09 +00003865
3866 ttNTHeader.uNRCount = SWAPWORD(ttNTHeader.uNRCount);
3867 ttNTHeader.uStorageOffset = SWAPWORD(ttNTHeader.uStorageOffset);
3868 bFound = FALSE;
3869 for(i=0; i<ttNTHeader.uNRCount; i++)
3870 {
Thomas Weidenmueller561f2f02005-10-29 10:29:08 +00003871 ReadFile(handle,&ttRecord, sizeof(TT_NAME_RECORD),&dwRead,NULL);
Aric Stewart0af24872005-02-25 14:00:09 +00003872 ttRecord.uNameID = SWAPWORD(ttRecord.uNameID);
3873 /* 4 is the Full Font Name */
3874 if(ttRecord.uNameID == 4)
3875 {
3876 int nPos;
3877 LPSTR buf;
Mike McCormackf9acfe62005-06-07 20:29:51 +00003878 static LPCSTR tt = " (TrueType)";
Aric Stewart0af24872005-02-25 14:00:09 +00003879
3880 ttRecord.uStringLength = SWAPWORD(ttRecord.uStringLength);
3881 ttRecord.uStringOffset = SWAPWORD(ttRecord.uStringOffset);
3882 nPos = SetFilePointer(handle, 0, NULL, FILE_CURRENT);
3883 SetFilePointer(handle, tblDir.uOffset +
3884 ttRecord.uStringOffset +
3885 ttNTHeader.uStorageOffset,
3886 NULL, FILE_BEGIN);
Mike McCormackee034ba2005-09-20 11:59:14 +00003887 buf = msi_alloc( ttRecord.uStringLength + 1 + strlen(tt) );
Aric Stewart0af24872005-02-25 14:00:09 +00003888 memset(buf, 0, ttRecord.uStringLength + 1 + strlen(tt));
Thomas Weidenmueller561f2f02005-10-29 10:29:08 +00003889 ReadFile(handle, buf, ttRecord.uStringLength, &dwRead, NULL);
Aric Stewart0af24872005-02-25 14:00:09 +00003890 if (strlen(buf) > 0)
3891 {
3892 strcat(buf,tt);
3893 ret = strdupAtoW(buf);
Mike McCormackee034ba2005-09-20 11:59:14 +00003894 msi_free(buf);
Aric Stewart0af24872005-02-25 14:00:09 +00003895 break;
3896 }
3897
Mike McCormackee034ba2005-09-20 11:59:14 +00003898 msi_free(buf);
Aric Stewart0af24872005-02-25 14:00:09 +00003899 SetFilePointer(handle,nPos, NULL, FILE_BEGIN);
3900 }
3901 }
3902 }
3903 CloseHandle(handle);
3904 }
Aric Stewart8a6a9a62005-05-18 09:44:38 +00003905 else
3906 ERR("Unable to open font file %s\n", debugstr_w(filename));
Aric Stewart0af24872005-02-25 14:00:09 +00003907
3908 TRACE("Returning fontname %s\n",debugstr_w(ret));
3909 return ret;
3910}
3911
Aric Stewarta9b50522005-06-23 11:04:09 +00003912static UINT ITERATE_RegisterFonts(MSIRECORD *row, LPVOID param)
Aric Stewart0af24872005-02-25 14:00:09 +00003913{
Aric Stewarta9b50522005-06-23 11:04:09 +00003914 MSIPACKAGE *package = (MSIPACKAGE*)param;
3915 LPWSTR name;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003916 LPCWSTR filename;
3917 MSIFILE *file;
Aric Stewart8e233e92005-03-01 11:45:19 +00003918 static const WCHAR regfont1[] =
3919 {'S','o','f','t','w','a','r','e','\\',
3920 'M','i','c','r','o','s','o','f','t','\\',
3921 'W','i','n','d','o','w','s',' ','N','T','\\',
3922 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
3923 'F','o','n','t','s',0};
3924 static const WCHAR regfont2[] =
3925 {'S','o','f','t','w','a','r','e','\\',
3926 'M','i','c','r','o','s','o','f','t','\\',
3927 'W','i','n','d','o','w','s','\\',
3928 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
3929 'F','o','n','t','s',0};
Aric Stewart0af24872005-02-25 14:00:09 +00003930 HKEY hkey1;
3931 HKEY hkey2;
3932
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003933 filename = MSI_RecordGetString( row, 1 );
3934 file = get_loaded_file( package, filename );
3935 if (!file)
Aric Stewarta9b50522005-06-23 11:04:09 +00003936 {
3937 ERR("Unable to load file\n");
3938 return ERROR_SUCCESS;
3939 }
3940
3941 /* check to make sure that component is installed */
Mike McCormackd693f462005-10-29 11:36:48 +00003942 if (!ACTION_VerifyComponentForAction( file->Component, INSTALLSTATE_LOCAL))
Aric Stewarta9b50522005-06-23 11:04:09 +00003943 {
3944 TRACE("Skipping: Component not scheduled for install\n");
3945 return ERROR_SUCCESS;
3946 }
3947
3948 RegCreateKeyW(HKEY_LOCAL_MACHINE,regfont1,&hkey1);
3949 RegCreateKeyW(HKEY_LOCAL_MACHINE,regfont2,&hkey2);
3950
3951 if (MSI_RecordIsNull(row,2))
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003952 name = load_ttfname_from( file->TargetPath );
Aric Stewarta9b50522005-06-23 11:04:09 +00003953 else
Mike McCormack51c66182005-10-27 12:36:12 +00003954 name = msi_dup_record_field(row,2);
Aric Stewarta9b50522005-06-23 11:04:09 +00003955
3956 if (name)
3957 {
Mike McCormack4db02cd2005-09-15 14:58:38 +00003958 msi_reg_set_val_str( hkey1, name, file->FileName );
3959 msi_reg_set_val_str( hkey2, name, file->FileName );
Aric Stewarta9b50522005-06-23 11:04:09 +00003960 }
3961
Mike McCormackee034ba2005-09-20 11:59:14 +00003962 msi_free(name);
Aric Stewarta9b50522005-06-23 11:04:09 +00003963 RegCloseKey(hkey1);
3964 RegCloseKey(hkey2);
3965 return ERROR_SUCCESS;
3966}
3967
3968static UINT ACTION_RegisterFonts(MSIPACKAGE *package)
3969{
3970 UINT rc;
3971 MSIQUERY * view;
3972 static const WCHAR ExecSeqQuery[] =
3973 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
3974 '`','F','o','n','t','`',0};
Juan Lang88adb532005-05-19 11:15:24 +00003975
Aric Stewart0af24872005-02-25 14:00:09 +00003976 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
3977 if (rc != ERROR_SUCCESS)
Juan Lang88adb532005-05-19 11:15:24 +00003978 {
3979 TRACE("MSI_DatabaseOpenViewW failed: %d\n", rc);
3980 return ERROR_SUCCESS;
3981 }
Aric Stewart0af24872005-02-25 14:00:09 +00003982
Aric Stewarta9b50522005-06-23 11:04:09 +00003983 MSI_IterateRecords(view, NULL, ITERATE_RegisterFonts, package);
Aric Stewart0af24872005-02-25 14:00:09 +00003984 msiobj_release(&view->hdr);
3985
Aric Stewarta9b50522005-06-23 11:04:09 +00003986 return ERROR_SUCCESS;
Aric Stewart0af24872005-02-25 14:00:09 +00003987}
3988
Aric Stewart072c5e52005-04-20 12:50:05 +00003989static UINT ITERATE_PublishComponent(MSIRECORD *rec, LPVOID param)
3990{
3991 MSIPACKAGE *package = (MSIPACKAGE*)param;
Aric Stewart09b0aba2005-06-09 20:30:59 +00003992 LPCWSTR compgroupid=NULL;
3993 LPCWSTR feature=NULL;
3994 LPCWSTR text = NULL;
3995 LPCWSTR qualifier = NULL;
3996 LPCWSTR component = NULL;
Aric Stewart6f43c182005-05-26 12:24:28 +00003997 LPWSTR advertise = NULL;
3998 LPWSTR output = NULL;
Aric Stewart072c5e52005-04-20 12:50:05 +00003999 HKEY hkey;
4000 UINT rc = ERROR_SUCCESS;
Mike McCormack38d67a42005-08-22 09:15:23 +00004001 MSICOMPONENT *comp;
Aric Stewart072c5e52005-04-20 12:50:05 +00004002 DWORD sz = 0;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00004003
Aric Stewart09b0aba2005-06-09 20:30:59 +00004004 component = MSI_RecordGetString(rec,3);
Mike McCormack38d67a42005-08-22 09:15:23 +00004005 comp = get_loaded_component(package,component);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00004006
Mike McCormackd693f462005-10-29 11:36:48 +00004007 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL ) &&
4008 !ACTION_VerifyComponentForAction( comp, INSTALLSTATE_SOURCE ) &&
4009 !ACTION_VerifyComponentForAction( comp, INSTALLSTATE_ADVERTISED ))
Aric Stewartb39d8fc2005-05-13 13:56:39 +00004010 {
4011 TRACE("Skipping: Component %s not scheduled for install\n",
4012 debugstr_w(component));
Aric Stewart6f43c182005-05-26 12:24:28 +00004013
Aric Stewartb39d8fc2005-05-13 13:56:39 +00004014 return ERROR_SUCCESS;
4015 }
Aric Stewart072c5e52005-04-20 12:50:05 +00004016
Aric Stewart09b0aba2005-06-09 20:30:59 +00004017 compgroupid = MSI_RecordGetString(rec,1);
Aric Stewart072c5e52005-04-20 12:50:05 +00004018
4019 rc = MSIREG_OpenUserComponentsKey(compgroupid, &hkey, TRUE);
4020 if (rc != ERROR_SUCCESS)
4021 goto end;
4022
Aric Stewart09b0aba2005-06-09 20:30:59 +00004023 text = MSI_RecordGetString(rec,4);
4024 qualifier = MSI_RecordGetString(rec,2);
4025 feature = MSI_RecordGetString(rec,5);
Aric Stewart072c5e52005-04-20 12:50:05 +00004026
Mike McCormack38d67a42005-08-22 09:15:23 +00004027 advertise = create_component_advertise_string(package, comp, feature);
Aric Stewart072c5e52005-04-20 12:50:05 +00004028
Aric Stewart6f43c182005-05-26 12:24:28 +00004029 sz = strlenW(advertise);
4030
Aric Stewart072c5e52005-04-20 12:50:05 +00004031 if (text)
4032 sz += lstrlenW(text);
Aric Stewart072c5e52005-04-20 12:50:05 +00004033
4034 sz+=3;
4035 sz *= sizeof(WCHAR);
4036
Mike McCormackee034ba2005-09-20 11:59:14 +00004037 output = msi_alloc(sz);
Aric Stewart072c5e52005-04-20 12:50:05 +00004038 memset(output,0,sz);
Aric Stewart6f43c182005-05-26 12:24:28 +00004039 strcpyW(output,advertise);
Mike McCormack470f23d2005-09-22 10:56:26 +00004040 msi_free(advertise);
Aric Stewart072c5e52005-04-20 12:50:05 +00004041
4042 if (text)
4043 strcatW(output,text);
4044
Mike McCormack4db02cd2005-09-15 14:58:38 +00004045 msi_reg_set_val_multi_str( hkey, qualifier, output );
Aric Stewart072c5e52005-04-20 12:50:05 +00004046
4047end:
4048 RegCloseKey(hkey);
Mike McCormackee034ba2005-09-20 11:59:14 +00004049 msi_free(output);
Aric Stewart072c5e52005-04-20 12:50:05 +00004050
4051 return rc;
4052}
4053
4054/*
4055 * At present I am ignorning the advertised components part of this and only
4056 * focusing on the qualified component sets
4057 */
4058static UINT ACTION_PublishComponents(MSIPACKAGE *package)
4059{
4060 UINT rc;
4061 MSIQUERY * view;
4062 static const WCHAR ExecSeqQuery[] =
4063 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00004064 '`','P','u','b','l','i','s','h',
4065 'C','o','m','p','o','n','e','n','t','`',0};
Aric Stewart072c5e52005-04-20 12:50:05 +00004066
4067 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
4068 if (rc != ERROR_SUCCESS)
4069 return ERROR_SUCCESS;
4070
4071 rc = MSI_IterateRecords(view, NULL, ITERATE_PublishComponent, package);
4072 msiobj_release(&view->hdr);
4073
4074 return rc;
4075}
Mike McCormack202166c2005-09-23 10:09:18 +00004076
Mike McCormack2586a092005-09-26 09:56:18 +00004077static UINT msi_unimplemented_action_stub( MSIPACKAGE *package,
4078 LPCSTR action, LPCWSTR table )
Mike McCormack567f0312005-09-23 11:06:57 +00004079{
4080 static const WCHAR query[] = {
Mike McCormack2586a092005-09-26 09:56:18 +00004081 'S','E','L','E','C','T',' ','*',' ',
4082 'F','R','O','M',' ','`','%','s','`',0 };
Mike McCormack567f0312005-09-23 11:06:57 +00004083 MSIQUERY *view = NULL;
4084 DWORD count = 0;
Mike McCormack2586a092005-09-26 09:56:18 +00004085 UINT r;
Mike McCormack567f0312005-09-23 11:06:57 +00004086
Mike McCormack2586a092005-09-26 09:56:18 +00004087 r = MSI_OpenQuery( package->db, &view, query, table );
4088 if (r == ERROR_SUCCESS)
Mike McCormack567f0312005-09-23 11:06:57 +00004089 {
Mike McCormack2586a092005-09-26 09:56:18 +00004090 r = MSI_IterateRecords(view, &count, NULL, package);
Mike McCormack567f0312005-09-23 11:06:57 +00004091 msiobj_release(&view->hdr);
4092 }
4093
Mike McCormack2586a092005-09-26 09:56:18 +00004094 if (count)
4095 FIXME("%s -> %lu ignored %s table values\n",
4096 action, count, debugstr_w(table));
4097
Mike McCormack567f0312005-09-23 11:06:57 +00004098 return ERROR_SUCCESS;
4099}
Mike McCormack94fbe092005-09-23 17:21:10 +00004100
Mike McCormack55942702005-10-30 19:23:28 +00004101static UINT ACTION_AllocateRegistrySpace( MSIPACKAGE *package )
4102{
4103 TRACE("%p\n", package);
4104 return ERROR_SUCCESS;
4105}
4106
Mike McCormack2586a092005-09-26 09:56:18 +00004107static UINT ACTION_RemoveIniValues( MSIPACKAGE *package )
Mike McCormack94fbe092005-09-23 17:21:10 +00004108{
Mike McCormack2586a092005-09-26 09:56:18 +00004109 static const WCHAR table[] =
4110 {'R','e','m','o','v','e','I','n','i','F','i','l','e',0 };
4111 return msi_unimplemented_action_stub( package, "RemoveIniValues", table );
Mike McCormack94fbe092005-09-23 17:21:10 +00004112}
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004113
Mike McCormack2586a092005-09-26 09:56:18 +00004114static UINT ACTION_MoveFiles( MSIPACKAGE *package )
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004115{
Mike McCormack2586a092005-09-26 09:56:18 +00004116 static const WCHAR table[] = { 'M','o','v','e','F','i','l','e',0 };
4117 return msi_unimplemented_action_stub( package, "MoveFiles", table );
4118}
4119
4120static UINT ACTION_PatchFiles( MSIPACKAGE *package )
4121{
4122 static const WCHAR table[] = { 'P','a','t','c','h',0 };
4123 return msi_unimplemented_action_stub( package, "PatchFiles", table );
4124}
4125
4126static UINT ACTION_BindImage( MSIPACKAGE *package )
4127{
4128 static const WCHAR table[] = { 'B','i','n','d','I','m','a','g','e',0 };
4129 return msi_unimplemented_action_stub( package, "BindImage", table );
4130}
4131
4132static UINT ACTION_IsolateComponents( MSIPACKAGE *package )
4133{
4134 static const WCHAR table[] = {
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004135 'I','s','o','l','a','t','e','C','o','m','p','o','n','e','n','t',0 };
Mike McCormack2586a092005-09-26 09:56:18 +00004136 return msi_unimplemented_action_stub( package, "IsolateComponents", table );
4137}
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004138
Mike McCormack2586a092005-09-26 09:56:18 +00004139static UINT ACTION_MigrateFeatureStates( MSIPACKAGE *package )
4140{
4141 static const WCHAR table[] = { 'U','p','g','r','a','d','e',0 };
4142 return msi_unimplemented_action_stub( package, "MigrateFeatureStates", table );
4143}
4144
4145static UINT ACTION_SelfUnregModules( MSIPACKAGE *package )
4146{
4147 static const WCHAR table[] = { 'S','e','l','f','R','e','g',0 };
4148 return msi_unimplemented_action_stub( package, "SelfUnregModules", table );
4149}
4150
4151static UINT ACTION_InstallServices( MSIPACKAGE *package )
4152{
4153 static const WCHAR table[] = {
4154 'S','e','r','v','i','c','e','I','n','s','t','a','l','l',0 };
4155 return msi_unimplemented_action_stub( package, "InstallServices", table );
4156}
4157
4158static UINT ACTION_StartServices( MSIPACKAGE *package )
4159{
4160 static const WCHAR table[] = {
4161 'S','e','r','v','i','c','e','C','o','n','t','r','o','l',0 };
4162 return msi_unimplemented_action_stub( package, "StartServices", table );
4163}
4164
4165static UINT ACTION_StopServices( MSIPACKAGE *package )
4166{
4167 static const WCHAR table[] = {
4168 'S','e','r','v','i','c','e','C','o','n','t','r','o','l',0 };
4169 return msi_unimplemented_action_stub( package, "StopServices", table );
4170}
4171
4172static UINT ACTION_DeleteServices( MSIPACKAGE *package )
4173{
4174 static const WCHAR table[] = {
4175 'S','e','r','v','i','c','e','C','o','n','t','r','o','l',0 };
4176 return msi_unimplemented_action_stub( package, "DeleteServices", table );
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004177}
Mike McCormack3b955152005-09-28 18:10:44 +00004178
4179static UINT ACTION_WriteEnvironmentStrings( MSIPACKAGE *package )
4180{
4181 static const WCHAR table[] = {
4182 'E','n','v','i','r','o','n','m','e','n','t',0 };
4183 return msi_unimplemented_action_stub( package, "WriteEnvironmentStrings", table );
4184}
4185
4186static UINT ACTION_RemoveEnvironmentStrings( MSIPACKAGE *package )
4187{
4188 static const WCHAR table[] = {
4189 'E','n','v','i','r','o','n','m','e','n','t',0 };
4190 return msi_unimplemented_action_stub( package, "RemoveEnvironmentStrings", table );
4191}
4192
4193static UINT ACTION_MsiPublishAssemblies( MSIPACKAGE *package )
4194{
4195 static const WCHAR table[] = {
4196 'M','s','i','A','s','s','e','m','b','l','y',0 };
4197 return msi_unimplemented_action_stub( package, "MsiPublishAssemblies", table );
4198}
4199
4200static UINT ACTION_MsiUnpublishAssemblies( MSIPACKAGE *package )
4201{
4202 static const WCHAR table[] = {
4203 'M','s','i','A','s','s','e','m','b','l','y',0 };
4204 return msi_unimplemented_action_stub( package, "MsiUnpublishAssemblies", table );
4205}
4206
4207static UINT ACTION_UnregisterFonts( MSIPACKAGE *package )
4208{
4209 static const WCHAR table[] = { 'F','o','n','t',0 };
4210 return msi_unimplemented_action_stub( package, "UnregisterFonts", table );
4211}
4212
4213static struct _actions StandardActions[] = {
Mike McCormack55942702005-10-30 19:23:28 +00004214 { szAllocateRegistrySpace, ACTION_AllocateRegistrySpace },
Mike McCormack3b955152005-09-28 18:10:44 +00004215 { szAppSearch, ACTION_AppSearch },
4216 { szBindImage, ACTION_BindImage },
4217 { szCCPSearch, NULL},
4218 { szCostFinalize, ACTION_CostFinalize },
4219 { szCostInitialize, ACTION_CostInitialize },
4220 { szCreateFolders, ACTION_CreateFolders },
4221 { szCreateShortcuts, ACTION_CreateShortcuts },
4222 { szDeleteServices, ACTION_DeleteServices },
4223 { szDisableRollback, NULL},
4224 { szDuplicateFiles, ACTION_DuplicateFiles },
4225 { szExecuteAction, ACTION_ExecuteAction },
4226 { szFileCost, ACTION_FileCost },
4227 { szFindRelatedProducts, ACTION_FindRelatedProducts },
4228 { szForceReboot, ACTION_ForceReboot },
4229 { szInstallAdminPackage, NULL},
4230 { szInstallExecute, ACTION_InstallExecute },
4231 { szInstallExecuteAgain, ACTION_InstallExecute },
4232 { szInstallFiles, ACTION_InstallFiles},
4233 { szInstallFinalize, ACTION_InstallFinalize },
4234 { szInstallInitialize, ACTION_InstallInitialize },
4235 { szInstallSFPCatalogFile, NULL},
4236 { szInstallValidate, ACTION_InstallValidate },
4237 { szIsolateComponents, ACTION_IsolateComponents },
4238 { szLaunchConditions, ACTION_LaunchConditions },
4239 { szMigrateFeatureStates, ACTION_MigrateFeatureStates },
4240 { szMoveFiles, ACTION_MoveFiles },
4241 { szMsiPublishAssemblies, ACTION_MsiPublishAssemblies },
4242 { szMsiUnpublishAssemblies, ACTION_MsiUnpublishAssemblies },
4243 { szInstallODBC, NULL},
4244 { szInstallServices, ACTION_InstallServices },
4245 { szPatchFiles, ACTION_PatchFiles },
4246 { szProcessComponents, ACTION_ProcessComponents },
4247 { szPublishComponents, ACTION_PublishComponents },
4248 { szPublishFeatures, ACTION_PublishFeatures },
4249 { szPublishProduct, ACTION_PublishProduct },
4250 { szRegisterClassInfo, ACTION_RegisterClassInfo },
4251 { szRegisterComPlus, NULL},
4252 { szRegisterExtensionInfo, ACTION_RegisterExtensionInfo },
4253 { szRegisterFonts, ACTION_RegisterFonts },
4254 { szRegisterMIMEInfo, ACTION_RegisterMIMEInfo },
4255 { szRegisterProduct, ACTION_RegisterProduct },
4256 { szRegisterProgIdInfo, ACTION_RegisterProgIdInfo },
4257 { szRegisterTypeLibraries, ACTION_RegisterTypeLibraries },
4258 { szRegisterUser, ACTION_RegisterUser},
4259 { szRemoveDuplicateFiles, NULL},
4260 { szRemoveEnvironmentStrings, ACTION_RemoveEnvironmentStrings },
4261 { szRemoveExistingProducts, NULL},
Mike McCormackfd4a2002005-11-02 11:42:56 +00004262 { szRemoveFiles, ACTION_RemoveFiles},
Mike McCormack3b955152005-09-28 18:10:44 +00004263 { szRemoveFolders, NULL},
4264 { szRemoveIniValues, ACTION_RemoveIniValues },
4265 { szRemoveODBC, NULL},
4266 { szRemoveRegistryValues, NULL},
4267 { szRemoveShortcuts, NULL},
4268 { szResolveSource, ACTION_ResolveSource},
4269 { szRMCCPSearch, NULL},
4270 { szScheduleReboot, NULL},
4271 { szSelfRegModules, ACTION_SelfRegModules },
4272 { szSelfUnregModules, ACTION_SelfUnregModules },
4273 { szSetODBCFolders, NULL},
4274 { szStartServices, ACTION_StartServices },
4275 { szStopServices, ACTION_StopServices },
4276 { szUnpublishComponents, NULL},
4277 { szUnpublishFeatures, NULL},
4278 { szUnregisterClassInfo, NULL},
4279 { szUnregisterComPlus, NULL},
4280 { szUnregisterExtensionInfo, NULL},
4281 { szUnregisterFonts, ACTION_UnregisterFonts },
4282 { szUnregisterMIMEInfo, NULL},
4283 { szUnregisterProgIdInfo, NULL},
4284 { szUnregisterTypeLibraries, NULL},
4285 { szValidateProductID, NULL},
4286 { szWriteEnvironmentStrings, ACTION_WriteEnvironmentStrings },
4287 { szWriteIniValues, ACTION_WriteIniValues },
4288 { szWriteRegistryValues, ACTION_WriteRegistryValues},
4289 { NULL, NULL},
4290};