blob: d7be3b52b92d721563b9a06642d741b39e04cc33 [file] [log] [blame]
Aric Stewart401bd3f2004-06-28 20:34:35 +00001/*
2 * Implementation of the Microsoft Installer (msi.dll)
3 *
Aric Stewartc6689522005-06-17 20:56:55 +00004 * Copyright 2004,2005 Aric Stewart for CodeWeavers
Aric Stewart401bd3f2004-06-28 20:34:35 +00005 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
Jonathan Ernst360a3f92006-05-18 14:49:52 +020018 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
Aric Stewart401bd3f2004-06-28 20:34:35 +000019 */
20
21/*
Mike McCormack6e2bca32004-07-04 00:25:00 +000022 * Pages I need
Aric Stewart401bd3f2004-06-28 20:34:35 +000023 *
24http://msdn.microsoft.com/library/default.asp?url=/library/en-us/msi/setup/installexecutesequence_table.asp
25
26http://msdn.microsoft.com/library/default.asp?url=/library/en-us/msi/setup/standard_actions_reference.asp
Aric Stewart401bd3f2004-06-28 20:34:35 +000027 */
28
29#include <stdarg.h>
Aric Stewart401bd3f2004-06-28 20:34:35 +000030
Francois Gouget486d0202004-10-07 03:06:48 +000031#define COBJMACROS
32
Aric Stewart401bd3f2004-06-28 20:34:35 +000033#include "windef.h"
34#include "winbase.h"
35#include "winerror.h"
36#include "winreg.h"
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};
Alexandre Julliardb79a53e2006-06-12 13:39:59 +0200116static const WCHAR szAppSearch[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000117 {'A','p','p','S','e','a','r','c','h',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000118static const WCHAR szAllocateRegistrySpace[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000119 {'A','l','l','o','c','a','t','e','R','e','g','i','s','t','r','y',
120 'S','p','a','c','e',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000121static const WCHAR szBindImage[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000122 {'B','i','n','d','I','m','a','g','e',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000123static const WCHAR szCCPSearch[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000124 {'C','C','P','S','e','a','r','c','h',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000125static const WCHAR szDeleteServices[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000126 {'D','e','l','e','t','e','S','e','r','v','i','c','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000127static const WCHAR szDisableRollback[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000128 {'D','i','s','a','b','l','e','R','o','l','l','b','a','c','k',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000129static const WCHAR szExecuteAction[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000130 {'E','x','e','c','u','t','e','A','c','t','i','o','n',0};
Aric Stewartdb982e22005-06-16 15:51:44 +0000131const WCHAR szFindRelatedProducts[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000132 {'F','i','n','d','R','e','l','a','t','e','d',
133 'P','r','o','d','u','c','t','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000134static const WCHAR szInstallAdminPackage[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000135 {'I','n','s','t','a','l','l','A','d','m','i','n',
136 'P','a','c','k','a','g','e',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000137static const WCHAR szInstallSFPCatalogFile[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000138 {'I','n','s','t','a','l','l','S','F','P','C','a','t','a','l','o','g',
139 'F','i','l','e',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000140static const WCHAR szIsolateComponents[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000141 {'I','s','o','l','a','t','e','C','o','m','p','o','n','e','n','t','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000142const WCHAR szMigrateFeatureStates[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000143 {'M','i','g','r','a','t','e','F','e','a','t','u','r','e',
144 'S','t','a','t','e','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000145const WCHAR szMoveFiles[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000146 {'M','o','v','e','F','i','l','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000147static const WCHAR szMsiPublishAssemblies[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000148 {'M','s','i','P','u','b','l','i','s','h',
149 'A','s','s','e','m','b','l','i','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000150static const WCHAR szMsiUnpublishAssemblies[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000151 {'M','s','i','U','n','p','u','b','l','i','s','h',
152 'A','s','s','e','m','b','l','i','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000153static const WCHAR szInstallODBC[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000154 {'I','n','s','t','a','l','l','O','D','B','C',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000155static const WCHAR szInstallServices[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000156 {'I','n','s','t','a','l','l','S','e','r','v','i','c','e','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000157const WCHAR szPatchFiles[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000158 {'P','a','t','c','h','F','i','l','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000159static const WCHAR szPublishComponents[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000160 {'P','u','b','l','i','s','h','C','o','m','p','o','n','e','n','t','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000161static const WCHAR szRegisterComPlus[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000162 {'R','e','g','i','s','t','e','r','C','o','m','P','l','u','s',0};
Aric Stewartdb982e22005-06-16 15:51:44 +0000163const WCHAR szRegisterExtensionInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000164 {'R','e','g','i','s','t','e','r','E','x','t','e','n','s','i','o','n',
165 'I','n','f','o',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000166static const WCHAR szRegisterFonts[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000167 {'R','e','g','i','s','t','e','r','F','o','n','t','s',0};
Aric Stewartdb982e22005-06-16 15:51:44 +0000168const WCHAR szRegisterMIMEInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000169 {'R','e','g','i','s','t','e','r','M','I','M','E','I','n','f','o',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000170static const WCHAR szRegisterUser[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000171 {'R','e','g','i','s','t','e','r','U','s','e','r',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000172const WCHAR szRemoveDuplicateFiles[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000173 {'R','e','m','o','v','e','D','u','p','l','i','c','a','t','e',
174 'F','i','l','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000175static const WCHAR szRemoveEnvironmentStrings[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000176 {'R','e','m','o','v','e','E','n','v','i','r','o','n','m','e','n','t',
177 'S','t','r','i','n','g','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000178const WCHAR szRemoveExistingProducts[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000179 {'R','e','m','o','v','e','E','x','i','s','t','i','n','g',
180 'P','r','o','d','u','c','t','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000181const WCHAR szRemoveFiles[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000182 {'R','e','m','o','v','e','F','i','l','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000183static const WCHAR szRemoveFolders[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000184 {'R','e','m','o','v','e','F','o','l','d','e','r','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000185static const WCHAR szRemoveIniValues[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000186 {'R','e','m','o','v','e','I','n','i','V','a','l','u','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000187static const WCHAR szRemoveODBC[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000188 {'R','e','m','o','v','e','O','D','B','C',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000189static const WCHAR szRemoveRegistryValues[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000190 {'R','e','m','o','v','e','R','e','g','i','s','t','r','y',
191 'V','a','l','u','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000192static const WCHAR szRemoveShortcuts[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000193 {'R','e','m','o','v','e','S','h','o','r','t','c','u','t','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000194static const WCHAR szRMCCPSearch[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000195 {'R','M','C','C','P','S','e','a','r','c','h',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000196static const WCHAR szScheduleReboot[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000197 {'S','c','h','e','d','u','l','e','R','e','b','o','o','t',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000198static const WCHAR szSelfUnregModules[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000199 {'S','e','l','f','U','n','r','e','g','M','o','d','u','l','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000200static const WCHAR szSetODBCFolders[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000201 {'S','e','t','O','D','B','C','F','o','l','d','e','r','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000202static const WCHAR szStartServices[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000203 {'S','t','a','r','t','S','e','r','v','i','c','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000204static const WCHAR szStopServices[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000205 {'S','t','o','p','S','e','r','v','i','c','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000206static const WCHAR szUnpublishComponents[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000207 {'U','n','p','u','b','l','i','s','h',
208 'C','o','m','p','o','n','e','n','t','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000209static const WCHAR szUnpublishFeatures[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000210 {'U','n','p','u','b','l','i','s','h','F','e','a','t','u','r','e','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000211const WCHAR szUnregisterClassInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000212 {'U','n','r','e','g','i','s','t','e','r','C','l','a','s','s',
213 'I','n','f','o',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000214static const WCHAR szUnregisterComPlus[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000215 {'U','n','r','e','g','i','s','t','e','r','C','o','m','P','l','u','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000216const WCHAR szUnregisterExtensionInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000217 {'U','n','r','e','g','i','s','t','e','r',
218 'E','x','t','e','n','s','i','o','n','I','n','f','o',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000219static const WCHAR szUnregisterFonts[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000220 {'U','n','r','e','g','i','s','t','e','r','F','o','n','t','s',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000221const WCHAR szUnregisterMIMEInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000222 {'U','n','r','e','g','i','s','t','e','r','M','I','M','E','I','n','f','o',0};
Aric Stewartc6689522005-06-17 20:56:55 +0000223const WCHAR szUnregisterProgIdInfo[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000224 {'U','n','r','e','g','i','s','t','e','r','P','r','o','g','I','d',
225 'I','n','f','o',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000226static const WCHAR szUnregisterTypeLibraries[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000227 {'U','n','r','e','g','i','s','t','e','r','T','y','p','e',
228 'L','i','b','r','a','r','i','e','s',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000229static const WCHAR szValidateProductID[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000230 {'V','a','l','i','d','a','t','e','P','r','o','d','u','c','t','I','D',0};
Mike McCormack9ba8ba32005-10-30 19:04:26 +0000231static const WCHAR szWriteEnvironmentStrings[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000232 {'W','r','i','t','e','E','n','v','i','r','o','n','m','e','n','t',
233 'S','t','r','i','n','g','s',0};
Aric Stewart90c57392005-01-31 16:23:12 +0000234
Mike McCormack3b955152005-09-28 18:10:44 +0000235/* action handlers */
236typedef UINT (*STANDARDACTIONHANDLER)(MSIPACKAGE*);
237
Aric Stewart90c57392005-01-31 16:23:12 +0000238struct _actions {
239 LPCWSTR action;
240 STANDARDACTIONHANDLER handler;
241};
242
Mike McCormack3b955152005-09-28 18:10:44 +0000243static struct _actions StandardActions[];
Aric Stewart90c57392005-01-31 16:23:12 +0000244
Aric Stewart2703d712005-06-20 15:33:10 +0000245
246/********************************************************
247 * helper functions
248 ********************************************************/
249
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000250static void ui_actionstart(MSIPACKAGE *package, LPCWSTR action)
Aric Stewartd2c395a2004-07-06 18:48:15 +0000251{
Aric Stewartd2c395a2004-07-06 18:48:15 +0000252 static const WCHAR Query_t[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000253 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000254 '`','A','c','t','i','o', 'n','T','e','x','t','`',' ',
255 'W','H','E','R','E', ' ','`','A','c','t','i','o','n','`',' ','=',
256 ' ','\'','%','s','\'',0};
Robert Shearmand679bc52006-01-23 17:30:31 +0100257 MSIRECORD * row;
Aric Stewartd2c395a2004-07-06 18:48:15 +0000258
Mike McCormack0b352c72005-06-02 10:29:57 +0000259 row = MSI_QueryGetRecord( package->db, Query_t, action );
260 if (!row)
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000261 return;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000262 MSI_ProcessMessage(package, INSTALLMESSAGE_ACTIONSTART, row);
263 msiobj_release(&row->hdr);
Aric Stewartd2c395a2004-07-06 18:48:15 +0000264}
265
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000266static void ui_actioninfo(MSIPACKAGE *package, LPCWSTR action, BOOL start,
Aric Stewartd2c395a2004-07-06 18:48:15 +0000267 UINT rc)
268{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000269 MSIRECORD * row;
Aric Stewartd2c395a2004-07-06 18:48:15 +0000270 static const WCHAR template_s[]=
Aric Stewart8e233e92005-03-01 11:45:19 +0000271 {'A','c','t','i','o','n',' ','s','t','a','r','t',' ','%','s',':',' ',
272 '%','s', '.',0};
Aric Stewartd2c395a2004-07-06 18:48:15 +0000273 static const WCHAR template_e[]=
Aric Stewart8e233e92005-03-01 11:45:19 +0000274 {'A','c','t','i','o','n',' ','e','n','d','e','d',' ','%','s',':',' ',
275 '%','s', '.',' ','R','e','t','u','r','n',' ','v','a','l','u','e',' ',
276 '%','i','.',0};
Aric Stewartd2c395a2004-07-06 18:48:15 +0000277 static const WCHAR format[] =
Aric Stewart8e233e92005-03-01 11:45:19 +0000278 {'H','H','\'',':','\'','m','m','\'',':','\'','s','s',0};
Aric Stewartd2c395a2004-07-06 18:48:15 +0000279 WCHAR message[1024];
280 WCHAR timet[0x100];
281
282 GetTimeFormatW(LOCALE_USER_DEFAULT, 0, NULL, format, timet, 0x100);
283 if (start)
284 sprintfW(message,template_s,timet,action);
285 else
286 sprintfW(message,template_e,timet,action,rc);
287
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000288 row = MSI_CreateRecord(1);
289 MSI_RecordSetStringW(row,1,message);
Aric Stewartd2c395a2004-07-06 18:48:15 +0000290
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000291 MSI_ProcessMessage(package, INSTALLMESSAGE_INFO, row);
292 msiobj_release(&row->hdr);
Aric Stewartd2c395a2004-07-06 18:48:15 +0000293}
294
Mike McCormacke3452222005-09-28 15:12:32 +0000295static UINT msi_parse_command_line( MSIPACKAGE *package, LPCWSTR szCommandLine )
296{
297 LPCWSTR ptr,ptr2;
298 BOOL quote;
299 DWORD len;
300 LPWSTR prop = NULL, val = NULL;
301
302 if (!szCommandLine)
303 return ERROR_SUCCESS;
304
305 ptr = szCommandLine;
306
307 while (*ptr)
308 {
309 if (*ptr==' ')
310 {
311 ptr++;
312 continue;
313 }
314
315 TRACE("Looking at %s\n",debugstr_w(ptr));
316
317 ptr2 = strchrW(ptr,'=');
318 if (!ptr2)
319 {
320 ERR("command line contains unknown string : %s\n", debugstr_w(ptr));
321 break;
322 }
323
324 quote = FALSE;
325
326 len = ptr2-ptr;
327 prop = msi_alloc((len+1)*sizeof(WCHAR));
328 memcpy(prop,ptr,len*sizeof(WCHAR));
329 prop[len]=0;
330 ptr2++;
331
332 len = 0;
333 ptr = ptr2;
334 while (*ptr && (quote || (!quote && *ptr!=' ')))
335 {
336 if (*ptr == '"')
337 quote = !quote;
338 ptr++;
339 len++;
340 }
341
342 if (*ptr2=='"')
343 {
344 ptr2++;
345 len -= 2;
346 }
347 val = msi_alloc((len+1)*sizeof(WCHAR));
348 memcpy(val,ptr2,len*sizeof(WCHAR));
349 val[len] = 0;
350
351 if (lstrlenW(prop) > 0)
352 {
353 TRACE("Found commandline property (%s) = (%s)\n",
354 debugstr_w(prop), debugstr_w(val));
355 MSI_SetPropertyW(package,prop,val);
356 }
357 msi_free(val);
358 msi_free(prop);
359 }
360
361 return ERROR_SUCCESS;
362}
363
Mike McCormack965a72a2005-10-26 12:06:21 +0000364
365static LPWSTR* msi_split_string( LPCWSTR str, WCHAR sep )
366{
367 LPWSTR p, *ret = NULL;
368 UINT count = 0;
369
370 if (!str)
371 return ret;
372
373 /* count the number of substrings */
374 for ( p = (LPWSTR)str, count = 0; p; count++ )
375 {
376 p = strchrW( p, sep );
377 if (p)
378 p++;
379 }
380
381 /* allocate space for an array of substring pointers and the substrings */
382 ret = msi_alloc( (count+1) * sizeof (LPWSTR) +
383 (lstrlenW(str)+1) * sizeof(WCHAR) );
384 if (!ret)
385 return ret;
386
387 /* copy the string and set the pointers */
388 p = (LPWSTR) &ret[count+1];
389 lstrcpyW( p, str );
390 for( count = 0; (ret[count] = p); count++ )
391 {
392 p = strchrW( p, sep );
393 if (p)
394 *p++ = 0;
395 }
396
397 return ret;
398}
399
400static UINT msi_apply_substorage_transform( MSIPACKAGE *package,
401 MSIDATABASE *patch_db, LPCWSTR name )
402{
403 UINT ret = ERROR_FUNCTION_FAILED;
404 IStorage *stg = NULL;
405 HRESULT r;
406
407 TRACE("%p %s\n", package, debugstr_w(name) );
408
409 if (*name++ != ':')
410 {
411 ERR("expected a colon in %s\n", debugstr_w(name));
412 return ERROR_FUNCTION_FAILED;
413 }
414
415 r = IStorage_OpenStorage( patch_db->storage, name, NULL, STGM_SHARE_EXCLUSIVE, NULL, 0, &stg );
416 if (SUCCEEDED(r))
417 {
Mike McCormackf8fef6e2005-10-26 13:56:33 +0000418 ret = msi_table_apply_transform( package->db, stg );
Mike McCormack965a72a2005-10-26 12:06:21 +0000419 IStorage_Release( stg );
420 ret = ERROR_SUCCESS;
421 }
422 else
423 ERR("failed to open substorage %s\n", debugstr_w(name));
424
425 return ret;
426}
427
428static UINT msi_check_patch_applicable( MSIPACKAGE *package, MSISUMMARYINFO *si )
429{
430 static const WCHAR szProdID[] = { 'P','r','o','d','u','c','t','I','D',0 };
431 LPWSTR guid_list, *guids, product_id;
432 UINT i, ret = ERROR_FUNCTION_FAILED;
433
434 product_id = msi_dup_property( package, szProdID );
435 if (!product_id)
436 {
437 /* FIXME: the property ProductID should be written into the DB somewhere */
438 ERR("no product ID to check\n");
439 return ERROR_SUCCESS;
440 }
441
442 guid_list = msi_suminfo_dup_string( si, PID_TEMPLATE );
443 guids = msi_split_string( guid_list, ';' );
444 for ( i = 0; guids[i] && ret != ERROR_SUCCESS; i++ )
445 {
446 if (!lstrcmpW( guids[i], product_id ))
447 ret = ERROR_SUCCESS;
448 }
449 msi_free( guids );
450 msi_free( guid_list );
451 msi_free( product_id );
452
453 return ret;
454}
455
456static UINT msi_parse_patch_summary( MSIPACKAGE *package, MSIDATABASE *patch_db )
457{
458 MSISUMMARYINFO *si;
459 LPWSTR str, *substorage;
460 UINT i, r = ERROR_SUCCESS;
461
462 si = MSI_GetSummaryInformationW( patch_db, 0 );
463 if (!si)
464 return ERROR_FUNCTION_FAILED;
465
466 msi_check_patch_applicable( package, si );
467
468 /* enumerate the substorage */
469 str = msi_suminfo_dup_string( si, PID_LASTAUTHOR );
470 substorage = msi_split_string( str, ';' );
471 for ( i = 0; substorage && substorage[i] && r == ERROR_SUCCESS; i++ )
472 r = msi_apply_substorage_transform( package, patch_db, substorage[i] );
473 msi_free( substorage );
474 msi_free( str );
475
476 /* FIXME: parse the sources in PID_REVNUMBER and do something with them... */
477
478 msiobj_release( &si->hdr );
479
480 return r;
481}
482
483static UINT msi_apply_patch_package( MSIPACKAGE *package, LPCWSTR file )
484{
485 MSIDATABASE *patch_db = NULL;
486 UINT r;
487
488 TRACE("%p %s\n", package, debugstr_w( file ) );
489
490 /* FIXME:
491 * We probably want to make sure we only open a patch collection here.
492 * Patch collections (.msp) and databases (.msi) have different GUIDs
493 * but currently MSI_OpenDatabaseW will accept both.
494 */
495 r = MSI_OpenDatabaseW( file, MSIDBOPEN_READONLY, &patch_db );
496 if ( r != ERROR_SUCCESS )
497 {
498 ERR("failed to open patch collection %s\n", debugstr_w( file ) );
499 return r;
500 }
501
502 msi_parse_patch_summary( package, patch_db );
503 msiobj_release( &patch_db->hdr );
504
505 return ERROR_SUCCESS;
506}
507
508/* get the PATCH property, and apply all the patches it specifies */
509static UINT msi_apply_patches( MSIPACKAGE *package )
510{
511 static const WCHAR szPatch[] = { 'P','A','T','C','H',0 };
512 LPWSTR patch_list, *patches;
513 UINT i, r = ERROR_SUCCESS;
514
515 patch_list = msi_dup_property( package, szPatch );
516
517 TRACE("patches to be applied: %s\n", debugstr_w( patch_list ) );
518
519 patches = msi_split_string( patch_list, ';' );
520 for( i=0; patches && patches[i] && r == ERROR_SUCCESS; i++ )
521 r = msi_apply_patch_package( package, patches[i] );
522
523 msi_free( patches );
524 msi_free( patch_list );
525
526 return r;
527}
528
Mike McCormacke534e772006-01-04 14:51:25 +0100529static UINT msi_apply_transforms( MSIPACKAGE *package )
530{
531 static const WCHAR szTransforms[] = {
532 'T','R','A','N','S','F','O','R','M','S',0 };
533 LPWSTR xform_list, *xforms;
534 UINT i, r = ERROR_SUCCESS;
535
536 xform_list = msi_dup_property( package, szTransforms );
537 xforms = msi_split_string( xform_list, ';' );
538
539 for( i=0; xforms && xforms[i] && r == ERROR_SUCCESS; i++ )
540 {
541 if (xforms[i][0] == ':')
542 r = msi_apply_substorage_transform( package, package->db, &xforms[i][1] );
543 else
544 r = MSI_DatabaseApplyTransformW( package->db, xforms[i], 0 );
545 }
546
547 msi_free( xforms );
548 msi_free( xform_list );
549
550 return r;
551}
552
Aric Stewart401bd3f2004-06-28 20:34:35 +0000553/****************************************************
554 * TOP level entry points
555 *****************************************************/
556
Mike McCormack61f24a42005-09-30 10:32:41 +0000557UINT MSI_InstallPackage( MSIPACKAGE *package, LPCWSTR szPackagePath,
558 LPCWSTR szCommandLine )
Aric Stewart401bd3f2004-06-28 20:34:35 +0000559{
Aric Stewart401bd3f2004-06-28 20:34:35 +0000560 UINT rc;
Aric Stewartf8f64402005-03-24 19:03:45 +0000561 BOOL ui = FALSE;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000562 static const WCHAR szUILevel[] = {'U','I','L','e','v','e','l',0};
Aric Stewart6269f002005-01-17 13:40:39 +0000563 static const WCHAR szAction[] = {'A','C','T','I','O','N',0};
564 static const WCHAR szInstall[] = {'I','N','S','T','A','L','L',0};
565
566 MSI_SetPropertyW(package, szAction, szInstall);
Aric Stewart9cd707d2005-05-27 19:24:22 +0000567
Mike McCormack3a940112006-04-19 02:29:03 +0900568 package->script = msi_alloc_zero(sizeof(MSISCRIPT));
Aric Stewart401bd3f2004-06-28 20:34:35 +0000569
Aric Stewartc9802932005-06-30 20:45:43 +0000570 package->script->InWhatSequence = SEQUENCE_INSTALL;
571
Aric Stewarte95136b2004-06-29 03:44:01 +0000572 if (szPackagePath)
573 {
Mike McCormackba8200b2004-12-22 15:25:30 +0000574 LPWSTR p, check, path;
Aric Stewarte95136b2004-06-29 03:44:01 +0000575
Mike McCormack95dea492005-03-16 11:31:35 +0000576 package->PackagePath = strdupW(szPackagePath);
577 path = strdupW(szPackagePath);
Mike McCormackba8200b2004-12-22 15:25:30 +0000578 p = strrchrW(path,'\\');
Aric Stewarte95136b2004-06-29 03:44:01 +0000579 if (p)
580 {
581 p++;
582 *p=0;
583 }
Aric Stewartc1e5c4a2005-02-08 14:26:49 +0000584 else
585 {
Mike McCormackee034ba2005-09-20 11:59:14 +0000586 msi_free(path);
587 path = msi_alloc(MAX_PATH*sizeof(WCHAR));
Aric Stewartc1e5c4a2005-02-08 14:26:49 +0000588 GetCurrentDirectoryW(MAX_PATH,path);
589 strcatW(path,cszbs);
590 }
Aric Stewarte95136b2004-06-29 03:44:01 +0000591
Mike McCormack062ad502005-09-15 15:04:08 +0000592 check = msi_dup_property( package, cszSourceDir );
Aric Stewartd0c971a2004-12-22 18:15:50 +0000593 if (!check)
Mike McCormackba8200b2004-12-22 15:25:30 +0000594 MSI_SetPropertyW(package, cszSourceDir, path);
Mike McCormackee034ba2005-09-20 11:59:14 +0000595 msi_free(check);
596 msi_free(path);
Aric Stewarte95136b2004-06-29 03:44:01 +0000597 }
Aric Stewart401bd3f2004-06-28 20:34:35 +0000598
Mike McCormacke3452222005-09-28 15:12:32 +0000599 msi_parse_command_line( package, szCommandLine );
Mike McCormack74f0de92005-09-29 10:32:39 +0000600
Mike McCormacke534e772006-01-04 14:51:25 +0100601 msi_apply_transforms( package );
Mike McCormack965a72a2005-10-26 12:06:21 +0000602 msi_apply_patches( package );
603
Mike McCormack74f0de92005-09-29 10:32:39 +0000604 if ( msi_get_property_int(package, szUILevel, 0) >= INSTALLUILEVEL_REDUCED )
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000605 {
Mike McCormack74f0de92005-09-29 10:32:39 +0000606 package->script->InWhatSequence |= SEQUENCE_UI;
607 rc = ACTION_ProcessUISequence(package);
608 ui = TRUE;
609 if (rc == ERROR_SUCCESS)
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000610 {
Mike McCormack74f0de92005-09-29 10:32:39 +0000611 package->script->InWhatSequence |= SEQUENCE_EXEC;
612 rc = ACTION_ProcessExecSequence(package,TRUE);
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000613 }
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000614 }
615 else
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000616 rc = ACTION_ProcessExecSequence(package,FALSE);
Aric Stewart6b16f292005-01-27 11:12:56 +0000617
618 if (rc == -1)
619 {
620 /* install was halted but should be considered a success */
621 rc = ERROR_SUCCESS;
622 }
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000623
Aric Stewart9cd707d2005-05-27 19:24:22 +0000624 package->script->CurrentlyScripting= FALSE;
625
Aric Stewart09d35c32004-12-27 19:00:26 +0000626 /* process the ending type action */
627 if (rc == ERROR_SUCCESS)
Aric Stewartf8f64402005-03-24 19:03:45 +0000628 ACTION_PerformActionSequence(package,-1,ui);
Aric Stewart6b16f292005-01-27 11:12:56 +0000629 else if (rc == ERROR_INSTALL_USEREXIT)
Aric Stewartf8f64402005-03-24 19:03:45 +0000630 ACTION_PerformActionSequence(package,-2,ui);
Aric Stewart6b16f292005-01-27 11:12:56 +0000631 else if (rc == ERROR_INSTALL_SUSPEND)
Aric Stewartf8f64402005-03-24 19:03:45 +0000632 ACTION_PerformActionSequence(package,-4,ui);
Aric Stewart9cd707d2005-05-27 19:24:22 +0000633 else /* failed */
634 ACTION_PerformActionSequence(package,-3,ui);
Aric Stewart54c67dd2005-01-25 20:17:09 +0000635
636 /* finish up running custom actions */
637 ACTION_FinishCustomActions(package);
Aric Stewart09d35c32004-12-27 19:00:26 +0000638
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000639 return rc;
640}
641
Aric Stewartf8f64402005-03-24 19:03:45 +0000642static UINT ACTION_PerformActionSequence(MSIPACKAGE *package, UINT seq, BOOL UI)
Aric Stewart09d35c32004-12-27 19:00:26 +0000643{
Mike McCormack0b352c72005-06-02 10:29:57 +0000644 UINT rc = ERROR_SUCCESS;
Aric Stewart09d35c32004-12-27 19:00:26 +0000645 MSIRECORD * row = 0;
Aric Stewart8e233e92005-03-01 11:45:19 +0000646 static const WCHAR ExecSeqQuery[] =
647 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000648 '`','I','n','s','t','a','l','l','E','x','e','c','u','t','e',
649 'S','e','q','u','e','n','c','e','`',' ', 'W','H','E','R','E',' ',
650 '`','S','e','q','u','e','n','c','e','`',' ', '=',' ','%','i',0};
Aric Stewart09d35c32004-12-27 19:00:26 +0000651
Aric Stewartf8f64402005-03-24 19:03:45 +0000652 static const WCHAR UISeqQuery[] =
653 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000654 '`','I','n','s','t','a','l','l','U','I','S','e','q','u','e','n','c','e',
655 '`', ' ', 'W','H','E','R','E',' ','`','S','e','q','u','e','n','c','e','`',
Aric Stewartf8f64402005-03-24 19:03:45 +0000656 ' ', '=',' ','%','i',0};
657
658 if (UI)
Mike McCormack0b352c72005-06-02 10:29:57 +0000659 row = MSI_QueryGetRecord(package->db, UISeqQuery, seq);
Aric Stewartf8f64402005-03-24 19:03:45 +0000660 else
Mike McCormack0b352c72005-06-02 10:29:57 +0000661 row = MSI_QueryGetRecord(package->db, ExecSeqQuery, seq);
Aric Stewart09d35c32004-12-27 19:00:26 +0000662
Mike McCormack0b352c72005-06-02 10:29:57 +0000663 if (row)
Aric Stewart09d35c32004-12-27 19:00:26 +0000664 {
Mike McCormack20806c72005-06-07 21:34:05 +0000665 LPCWSTR action, cond;
666
Aric Stewart09d35c32004-12-27 19:00:26 +0000667 TRACE("Running the actions\n");
668
Aric Stewart09d35c32004-12-27 19:00:26 +0000669 /* check conditions */
Mike McCormack20806c72005-06-07 21:34:05 +0000670 cond = MSI_RecordGetString(row,2);
671 if (cond)
Aric Stewart09d35c32004-12-27 19:00:26 +0000672 {
Mike McCormack20806c72005-06-07 21:34:05 +0000673 /* this is a hack to skip errors in the condition code */
674 if (MSI_EvaluateConditionW(package, cond) == MSICONDITION_FALSE)
675 goto end;
Aric Stewart09d35c32004-12-27 19:00:26 +0000676 }
677
Mike McCormack20806c72005-06-07 21:34:05 +0000678 action = MSI_RecordGetString(row,1);
679 if (!action)
Aric Stewart09d35c32004-12-27 19:00:26 +0000680 {
Mike McCormack20806c72005-06-07 21:34:05 +0000681 ERR("failed to fetch action\n");
682 rc = ERROR_FUNCTION_FAILED;
Aric Stewart09d35c32004-12-27 19:00:26 +0000683 goto end;
684 }
685
Aric Stewartf8f64402005-03-24 19:03:45 +0000686 if (UI)
Mike McCormack20806c72005-06-07 21:34:05 +0000687 rc = ACTION_PerformUIAction(package,action);
Aric Stewartf8f64402005-03-24 19:03:45 +0000688 else
Mike McCormack20806c72005-06-07 21:34:05 +0000689 rc = ACTION_PerformAction(package,action,FALSE);
Aric Stewart09d35c32004-12-27 19:00:26 +0000690end:
Mike McCormack0b352c72005-06-02 10:29:57 +0000691 msiobj_release(&row->hdr);
Aric Stewart09d35c32004-12-27 19:00:26 +0000692 }
693 else
694 rc = ERROR_SUCCESS;
695
696 return rc;
697}
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000698
Aric Stewart2703d712005-06-20 15:33:10 +0000699typedef struct {
700 MSIPACKAGE* package;
701 BOOL UI;
702} iterate_action_param;
703
704static UINT ITERATE_Actions(MSIRECORD *row, LPVOID param)
705{
706 iterate_action_param *iap= (iterate_action_param*)param;
707 UINT rc;
708 LPCWSTR cond, action;
709
710 action = MSI_RecordGetString(row,1);
711 if (!action)
712 {
713 ERR("Error is retrieving action name\n");
714 return ERROR_FUNCTION_FAILED;
715 }
716
717 /* check conditions */
718 cond = MSI_RecordGetString(row,2);
719 if (cond)
720 {
721 /* this is a hack to skip errors in the condition code */
722 if (MSI_EvaluateConditionW(iap->package, cond) == MSICONDITION_FALSE)
723 {
724 TRACE("Skipping action: %s (condition is false)\n",
725 debugstr_w(action));
726 return ERROR_SUCCESS;
727 }
728 }
729
730 if (iap->UI)
731 rc = ACTION_PerformUIAction(iap->package,action);
732 else
733 rc = ACTION_PerformAction(iap->package,action,FALSE);
734
Mike McCormack4f634a32005-07-06 15:44:51 +0000735 msi_dialog_check_messages( NULL );
736
737 if (iap->package->CurrentInstallState != ERROR_SUCCESS )
738 rc = iap->package->CurrentInstallState;
739
Aric Stewart2703d712005-06-20 15:33:10 +0000740 if (rc == ERROR_FUNCTION_NOT_CALLED)
741 rc = ERROR_SUCCESS;
742
743 if (rc != ERROR_SUCCESS)
Mike McCormack558abec2005-10-27 12:39:28 +0000744 ERR("Execution halted, action %s returned %i\n", debugstr_w(action), rc);
Aric Stewart2703d712005-06-20 15:33:10 +0000745
746 return rc;
747}
748
Mike McCormackd34b1c22005-09-21 10:55:23 +0000749UINT MSI_Sequence( MSIPACKAGE *package, LPCWSTR szTable, INT iSequenceMode )
750{
751 MSIQUERY * view;
752 UINT r;
753 static const WCHAR query[] =
754 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
755 '`','%','s','`',
756 ' ','W','H','E','R','E',' ',
757 '`','S','e','q','u','e','n','c','e','`',' ',
758 '>',' ','0',' ','O','R','D','E','R',' ','B','Y',' ',
759 '`','S','e','q','u','e','n','c','e','`',0};
760 iterate_action_param iap;
761
762 /*
763 * FIXME: probably should be checking UILevel in the
764 * ACTION_PerformUIAction/ACTION_PerformAction
765 * rather than saving the UI level here. Those
766 * two functions can be merged too.
767 */
768 iap.package = package;
769 iap.UI = TRUE;
770
771 TRACE("%p %s %i\n", package, debugstr_w(szTable), iSequenceMode );
772
773 r = MSI_OpenQuery( package->db, &view, query, szTable );
774 if (r == ERROR_SUCCESS)
775 {
776 r = MSI_IterateRecords( view, NULL, ITERATE_Actions, &iap );
777 msiobj_release(&view->hdr);
778 }
779
780 return r;
781}
782
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000783static UINT ACTION_ProcessExecSequence(MSIPACKAGE *package, BOOL UIran)
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000784{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000785 MSIQUERY * view;
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000786 UINT rc;
Aric Stewart8e233e92005-03-01 11:45:19 +0000787 static const WCHAR ExecSeqQuery[] =
788 {'S','E','L','E','C','T',' ','*',' ', 'F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000789 '`','I','n','s','t','a','l','l','E','x','e','c','u','t','e',
790 'S','e','q','u','e','n','c','e','`',' ', 'W','H','E','R','E',' ',
791 '`','S','e','q','u','e','n','c','e','`',' ', '>',' ','%','i',' ',
Aric Stewart8e233e92005-03-01 11:45:19 +0000792 'O','R','D','E','R',' ', 'B','Y',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000793 '`','S','e','q','u','e','n','c','e','`',0 };
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000794 MSIRECORD * row = 0;
Aric Stewart8e233e92005-03-01 11:45:19 +0000795 static const WCHAR IVQuery[] =
Aric Stewart98e38082005-05-20 09:40:42 +0000796 {'S','E','L','E','C','T',' ','`','S','e','q','u','e','n','c','e','`',
797 ' ', 'F','R','O','M',' ','`','I','n','s','t','a','l','l',
798 'E','x','e','c','u','t','e','S','e','q','u','e','n','c','e','`',' ',
799 'W','H','E','R','E',' ','`','A','c','t','i','o','n','`',' ','=',
800 ' ','\'', 'I','n','s','t','a','l','l',
801 'V','a','l','i','d','a','t','e','\'', 0};
Mike McCormack9db0e072004-12-22 15:05:07 +0000802 INT seq = 0;
Aric Stewart2703d712005-06-20 15:33:10 +0000803 iterate_action_param iap;
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000804
Aric Stewart2703d712005-06-20 15:33:10 +0000805 iap.package = package;
806 iap.UI = FALSE;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +0000807
Aric Stewart9cd707d2005-05-27 19:24:22 +0000808 if (package->script->ExecuteSequenceRun)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +0000809 {
810 TRACE("Execute Sequence already Run\n");
811 return ERROR_SUCCESS;
812 }
813
Aric Stewart9cd707d2005-05-27 19:24:22 +0000814 package->script->ExecuteSequenceRun = TRUE;
Aric Stewart2703d712005-06-20 15:33:10 +0000815
Mike McCormack9db0e072004-12-22 15:05:07 +0000816 /* get the sequence number */
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000817 if (UIran)
818 {
Mike McCormack0b352c72005-06-02 10:29:57 +0000819 row = MSI_QueryGetRecord(package->db, IVQuery);
820 if( !row )
821 return ERROR_FUNCTION_FAILED;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000822 seq = MSI_RecordGetInteger(row,1);
823 msiobj_release(&row->hdr);
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000824 }
Mike McCormack9db0e072004-12-22 15:05:07 +0000825
Mike McCormack0c238852005-01-21 16:19:11 +0000826 rc = MSI_OpenQuery(package->db, &view, ExecSeqQuery, seq);
Aric Stewart401bd3f2004-06-28 20:34:35 +0000827 if (rc == ERROR_SUCCESS)
828 {
Aric Stewart2703d712005-06-20 15:33:10 +0000829 TRACE("Running the actions\n");
Aric Stewart401bd3f2004-06-28 20:34:35 +0000830
Aric Stewart2703d712005-06-20 15:33:10 +0000831 rc = MSI_IterateRecords(view, NULL, ITERATE_Actions, &iap);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000832 msiobj_release(&view->hdr);
Aric Stewart401bd3f2004-06-28 20:34:35 +0000833 }
834
Aric Stewart401bd3f2004-06-28 20:34:35 +0000835 return rc;
836}
837
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000838static UINT ACTION_ProcessUISequence(MSIPACKAGE *package)
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000839{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000840 MSIQUERY * view;
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000841 UINT rc;
Aric Stewart8e233e92005-03-01 11:45:19 +0000842 static const WCHAR ExecSeqQuery [] =
843 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000844 '`','I','n','s','t','a','l','l',
845 'U','I','S','e','q','u','e','n','c','e','`',
846 ' ','W','H','E','R','E',' ',
847 '`','S','e','q','u','e','n','c','e','`',' ',
Aric Stewart8e233e92005-03-01 11:45:19 +0000848 '>',' ','0',' ','O','R','D','E','R',' ','B','Y',' ',
Aric Stewart98e38082005-05-20 09:40:42 +0000849 '`','S','e','q','u','e','n','c','e','`',0};
Aric Stewart2703d712005-06-20 15:33:10 +0000850 iterate_action_param iap;
851
852 iap.package = package;
853 iap.UI = TRUE;
854
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000855 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000856
857 if (rc == ERROR_SUCCESS)
858 {
Francois Gouget0edbaf72005-11-10 12:14:56 +0000859 TRACE("Running the actions\n");
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000860
Aric Stewart2703d712005-06-20 15:33:10 +0000861 rc = MSI_IterateRecords(view, NULL, ITERATE_Actions, &iap);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000862 msiobj_release(&view->hdr);
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000863 }
864
Aric Stewarted7c4bc2004-07-04 00:26:54 +0000865 return rc;
866}
867
Aric Stewart401bd3f2004-06-28 20:34:35 +0000868/********************************************************
869 * ACTION helper functions and functions that perform the actions
870 *******************************************************/
Aric Stewart9cd707d2005-05-27 19:24:22 +0000871static BOOL ACTION_HandleStandardAction(MSIPACKAGE *package, LPCWSTR action,
872 UINT* rc, BOOL force )
Aric Stewart3f318602005-02-01 18:46:26 +0000873{
874 BOOL ret = FALSE;
Aric Stewart9cd707d2005-05-27 19:24:22 +0000875 BOOL run = force;
Aric Stewart3f318602005-02-01 18:46:26 +0000876 int i;
Aric Stewart9cd707d2005-05-27 19:24:22 +0000877
878 if (!run && !package->script->CurrentlyScripting)
879 run = TRUE;
880
881 if (!run)
882 {
883 if (strcmpW(action,szInstallFinalize) == 0 ||
884 strcmpW(action,szInstallExecute) == 0 ||
885 strcmpW(action,szInstallExecuteAgain) == 0)
886 run = TRUE;
887 }
888
Aric Stewart3f318602005-02-01 18:46:26 +0000889 i = 0;
890 while (StandardActions[i].action != NULL)
891 {
892 if (strcmpW(StandardActions[i].action, action)==0)
893 {
Aric Stewart9cd707d2005-05-27 19:24:22 +0000894 if (!run)
Aric Stewartd6ecf582005-02-02 09:29:30 +0000895 {
Aric Stewart9cd707d2005-05-27 19:24:22 +0000896 ui_actioninfo(package, action, TRUE, 0);
897 *rc = schedule_action(package,INSTALL_SCRIPT,action);
898 ui_actioninfo(package, action, FALSE, *rc);
Aric Stewartd6ecf582005-02-02 09:29:30 +0000899 }
900 else
901 {
Aric Stewart9cd707d2005-05-27 19:24:22 +0000902 ui_actionstart(package, action);
903 if (StandardActions[i].handler)
904 {
905 *rc = StandardActions[i].handler(package);
906 }
907 else
908 {
Mike McCormack54a28912005-09-06 09:23:18 +0000909 FIXME("unhandled standard action %s\n",debugstr_w(action));
Aric Stewart9cd707d2005-05-27 19:24:22 +0000910 *rc = ERROR_SUCCESS;
911 }
Aric Stewartd6ecf582005-02-02 09:29:30 +0000912 }
Aric Stewart3f318602005-02-01 18:46:26 +0000913 ret = TRUE;
914 break;
915 }
916 i++;
917 }
918 return ret;
919}
920
Mike McCormackf9acfe62005-06-07 20:29:51 +0000921static BOOL ACTION_HandleCustomAction( MSIPACKAGE* package, LPCWSTR action,
922 UINT* rc, BOOL force )
Aric Stewart3f318602005-02-01 18:46:26 +0000923{
924 BOOL ret=FALSE;
925 UINT arc;
926
Aric Stewart9cd707d2005-05-27 19:24:22 +0000927 arc = ACTION_CustomAction(package,action, force);
Aric Stewart3f318602005-02-01 18:46:26 +0000928
929 if (arc != ERROR_CALL_NOT_IMPLEMENTED)
930 {
931 *rc = arc;
932 ret = TRUE;
933 }
934 return ret;
935}
Aric Stewart401bd3f2004-06-28 20:34:35 +0000936
937/*
Francois Gougetda8b3dd2005-01-26 21:09:04 +0000938 * A lot of actions are really important even if they don't do anything
939 * explicit... Lots of properties are set at the beginning of the installation
940 * CostFinalize does a bunch of work to translate the directories and such
Aric Stewart401bd3f2004-06-28 20:34:35 +0000941 *
Mike McCormack6e2bca32004-07-04 00:25:00 +0000942 * But until I get write access to the database that is hard, so I am going to
Aric Stewart401bd3f2004-06-28 20:34:35 +0000943 * hack it to see if I can get something to run.
944 */
Aric Stewart9cd707d2005-05-27 19:24:22 +0000945UINT ACTION_PerformAction(MSIPACKAGE *package, const WCHAR *action, BOOL force)
Aric Stewart401bd3f2004-06-28 20:34:35 +0000946{
Aric Stewartd2c395a2004-07-06 18:48:15 +0000947 UINT rc = ERROR_SUCCESS;
Aric Stewart3f318602005-02-01 18:46:26 +0000948 BOOL handled;
Aric Stewart401bd3f2004-06-28 20:34:35 +0000949
950 TRACE("Performing action (%s)\n",debugstr_w(action));
951
Aric Stewart9cd707d2005-05-27 19:24:22 +0000952 handled = ACTION_HandleStandardAction(package, action, &rc, force);
Aric Stewart7d3e5972004-07-04 00:36:58 +0000953
Aric Stewart3f318602005-02-01 18:46:26 +0000954 if (!handled)
Aric Stewart9cd707d2005-05-27 19:24:22 +0000955 handled = ACTION_HandleCustomAction(package, action, &rc, force);
Aric Stewart3f318602005-02-01 18:46:26 +0000956
Aric Stewart90c57392005-01-31 16:23:12 +0000957 if (!handled)
958 {
Mike McCormack54a28912005-09-06 09:23:18 +0000959 FIXME("unhandled msi action %s\n",debugstr_w(action));
Aric Stewart3f318602005-02-01 18:46:26 +0000960 rc = ERROR_FUNCTION_NOT_CALLED;
961 }
Aric Stewart7d3e5972004-07-04 00:36:58 +0000962
Aric Stewart3f318602005-02-01 18:46:26 +0000963 return rc;
964}
965
966UINT ACTION_PerformUIAction(MSIPACKAGE *package, const WCHAR *action)
967{
968 UINT rc = ERROR_SUCCESS;
969 BOOL handled = FALSE;
970
971 TRACE("Performing action (%s)\n",debugstr_w(action));
972
Aric Stewart9cd707d2005-05-27 19:24:22 +0000973 handled = ACTION_HandleStandardAction(package, action, &rc,TRUE);
Aric Stewart3f318602005-02-01 18:46:26 +0000974
975 if (!handled)
Aric Stewart9cd707d2005-05-27 19:24:22 +0000976 handled = ACTION_HandleCustomAction(package, action, &rc, FALSE);
Aric Stewart3f318602005-02-01 18:46:26 +0000977
Mike McCormack4f634a32005-07-06 15:44:51 +0000978 if( !handled && ACTION_DialogBox(package,action) == ERROR_SUCCESS )
979 handled = TRUE;
Mike McCormack34d4a022005-02-09 13:24:31 +0000980
Aric Stewart3f318602005-02-01 18:46:26 +0000981 if (!handled)
982 {
Mike McCormack54a28912005-09-06 09:23:18 +0000983 FIXME("unhandled msi action %s\n",debugstr_w(action));
Aric Stewart3f318602005-02-01 18:46:26 +0000984 rc = ERROR_FUNCTION_NOT_CALLED;
Aric Stewart90c57392005-01-31 16:23:12 +0000985 }
Aric Stewart401bd3f2004-06-28 20:34:35 +0000986
Aric Stewartd2c395a2004-07-06 18:48:15 +0000987 return rc;
Aric Stewart401bd3f2004-06-28 20:34:35 +0000988}
989
Aric Stewart2274ff12005-06-21 20:03:46 +0000990
991/*
992 * Actual Action Handlers
993 */
994
995static UINT ITERATE_CreateFolders(MSIRECORD *row, LPVOID param)
996{
997 MSIPACKAGE *package = (MSIPACKAGE*)param;
998 LPCWSTR dir;
999 LPWSTR full_path;
1000 MSIRECORD *uirow;
1001 MSIFOLDER *folder;
1002
1003 dir = MSI_RecordGetString(row,1);
1004 if (!dir)
1005 {
Francois Gouget0edbaf72005-11-10 12:14:56 +00001006 ERR("Unable to get folder id\n");
Aric Stewart2274ff12005-06-21 20:03:46 +00001007 return ERROR_SUCCESS;
1008 }
1009
1010 full_path = resolve_folder(package,dir,FALSE,FALSE,&folder);
1011 if (!full_path)
1012 {
1013 ERR("Unable to resolve folder id %s\n",debugstr_w(dir));
1014 return ERROR_SUCCESS;
1015 }
1016
1017 TRACE("Folder is %s\n",debugstr_w(full_path));
1018
1019 /* UI stuff */
1020 uirow = MSI_CreateRecord(1);
1021 MSI_RecordSetStringW(uirow,1,full_path);
1022 ui_actiondata(package,szCreateFolders,uirow);
1023 msiobj_release( &uirow->hdr );
1024
1025 if (folder->State == 0)
1026 create_full_pathW(full_path);
1027
1028 folder->State = 3;
1029
Mike McCormackee034ba2005-09-20 11:59:14 +00001030 msi_free(full_path);
Aric Stewart2274ff12005-06-21 20:03:46 +00001031 return ERROR_SUCCESS;
1032}
1033
Mike McCormack03b4dbb2005-10-28 09:39:29 +00001034/* FIXME: probably should merge this with the above function */
1035static UINT msi_create_directory( MSIPACKAGE* package, LPCWSTR dir )
1036{
1037 UINT rc = ERROR_SUCCESS;
1038 MSIFOLDER *folder;
1039 LPWSTR install_path;
1040
1041 install_path = resolve_folder(package, dir, FALSE, FALSE, &folder);
1042 if (!install_path)
1043 return ERROR_FUNCTION_FAILED;
1044
1045 /* create the path */
1046 if (folder->State == 0)
1047 {
1048 create_full_pathW(install_path);
1049 folder->State = 2;
1050 }
1051 msi_free(install_path);
1052
1053 return rc;
1054}
Aric Stewart2274ff12005-06-21 20:03:46 +00001055
Mike McCormack9c845852005-10-29 11:29:17 +00001056UINT msi_create_component_directories( MSIPACKAGE *package )
1057{
1058 MSICOMPONENT *comp;
1059
1060 /* create all the folders required by the components are going to install */
1061 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
1062 {
Mike McCormackd693f462005-10-29 11:36:48 +00001063 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
Mike McCormack9c845852005-10-29 11:29:17 +00001064 continue;
1065 msi_create_directory( package, comp->Directory );
1066 }
1067
1068 return ERROR_SUCCESS;
1069}
1070
Aric Stewart401bd3f2004-06-28 20:34:35 +00001071/*
1072 * Also we cannot enable/disable components either, so for now I am just going
Mike McCormack6e2bca32004-07-04 00:25:00 +00001073 * to do all the directories for all the components.
Aric Stewart401bd3f2004-06-28 20:34:35 +00001074 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001075static UINT ACTION_CreateFolders(MSIPACKAGE *package)
Aric Stewart401bd3f2004-06-28 20:34:35 +00001076{
Aric Stewart8e233e92005-03-01 11:45:19 +00001077 static const WCHAR ExecSeqQuery[] =
Aric Stewart98e38082005-05-20 09:40:42 +00001078 {'S','E','L','E','C','T',' ',
1079 '`','D','i','r','e','c','t','o','r','y','_','`',
Aric Stewart8e233e92005-03-01 11:45:19 +00001080 ' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001081 '`','C','r','e','a','t','e','F','o','l','d','e','r','`',0 };
Aric Stewart401bd3f2004-06-28 20:34:35 +00001082 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001083 MSIQUERY *view;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001084
Mike McCormack03b4dbb2005-10-28 09:39:29 +00001085 /* create all the empty folders specified in the CreateFolder table */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001086 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view );
Aric Stewart401bd3f2004-06-28 20:34:35 +00001087 if (rc != ERROR_SUCCESS)
Aric Stewart84837d92004-07-20 01:22:37 +00001088 return ERROR_SUCCESS;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001089
Aric Stewart2274ff12005-06-21 20:03:46 +00001090 rc = MSI_IterateRecords(view, NULL, ITERATE_CreateFolders, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001091 msiobj_release(&view->hdr);
Mike McCormack03b4dbb2005-10-28 09:39:29 +00001092
Mike McCormack9c845852005-10-29 11:29:17 +00001093 msi_create_component_directories( package );
Mike McCormack03b4dbb2005-10-28 09:39:29 +00001094
Aric Stewart401bd3f2004-06-28 20:34:35 +00001095 return rc;
1096}
1097
Mike McCormack38d67a42005-08-22 09:15:23 +00001098static MSICOMPONENT* load_component( MSIRECORD * row )
Aric Stewartbdb29552004-07-04 00:32:48 +00001099{
Mike McCormack38d67a42005-08-22 09:15:23 +00001100 MSICOMPONENT *comp;
Aric Stewartbdb29552004-07-04 00:32:48 +00001101
Mike McCormackee034ba2005-09-20 11:59:14 +00001102 comp = msi_alloc_zero( sizeof(MSICOMPONENT) );
Mike McCormack38d67a42005-08-22 09:15:23 +00001103 if (!comp)
1104 return comp;
1105
Aric Stewartbdb29552004-07-04 00:32:48 +00001106 /* fill in the data */
Mike McCormack51c66182005-10-27 12:36:12 +00001107 comp->Component = msi_dup_record_field( row, 1 );
Aric Stewartbdb29552004-07-04 00:32:48 +00001108
Mike McCormackefcc1ec2005-09-12 12:07:15 +00001109 TRACE("Loading Component %s\n", debugstr_w(comp->Component));
Aric Stewartbdb29552004-07-04 00:32:48 +00001110
Mike McCormack51c66182005-10-27 12:36:12 +00001111 comp->ComponentId = msi_dup_record_field( row, 2 );
1112 comp->Directory = msi_dup_record_field( row, 3 );
Mike McCormack38d67a42005-08-22 09:15:23 +00001113 comp->Attributes = MSI_RecordGetInteger(row,4);
Mike McCormack51c66182005-10-27 12:36:12 +00001114 comp->Condition = msi_dup_record_field( row, 5 );
1115 comp->KeyPath = msi_dup_record_field( row, 6 );
Aric Stewartbdb29552004-07-04 00:32:48 +00001116
Mike McCormack38d67a42005-08-22 09:15:23 +00001117 comp->Installed = INSTALLSTATE_ABSENT;
1118 comp->Action = INSTALLSTATE_UNKNOWN;
1119 comp->ActionRequest = INSTALLSTATE_UNKNOWN;
Aric Stewartfbdd7092004-12-27 19:06:22 +00001120
Mike McCormack38d67a42005-08-22 09:15:23 +00001121 comp->Enabled = TRUE;
Aric Stewartbdb29552004-07-04 00:32:48 +00001122
Mike McCormack38d67a42005-08-22 09:15:23 +00001123 return comp;
Aric Stewartbdb29552004-07-04 00:32:48 +00001124}
1125
Aric Stewart04598242005-06-23 16:43:24 +00001126typedef struct {
1127 MSIPACKAGE *package;
Mike McCormack1da28582005-08-22 14:09:17 +00001128 MSIFEATURE *feature;
Aric Stewart04598242005-06-23 16:43:24 +00001129} _ilfs;
1130
Mike McCormack38d67a42005-08-22 09:15:23 +00001131static UINT add_feature_component( MSIFEATURE *feature, MSICOMPONENT *comp )
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001132{
1133 ComponentList *cl;
1134
Mike McCormackee034ba2005-09-20 11:59:14 +00001135 cl = msi_alloc( sizeof (*cl) );
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001136 if ( !cl )
1137 return ERROR_NOT_ENOUGH_MEMORY;
Mike McCormack38d67a42005-08-22 09:15:23 +00001138 cl->component = comp;
Mike McCormack1da28582005-08-22 14:09:17 +00001139 list_add_tail( &feature->Components, &cl->entry );
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001140
1141 return ERROR_SUCCESS;
1142}
1143
Mike McCormack38d67a42005-08-22 09:15:23 +00001144static UINT iterate_component_check( MSIRECORD *row, LPVOID param )
Aric Stewartbdb29552004-07-04 00:32:48 +00001145{
Aric Stewart04598242005-06-23 16:43:24 +00001146 _ilfs* ilfs= (_ilfs*)param;
Mike McCormack38d67a42005-08-22 09:15:23 +00001147 MSIPACKAGE *package = ilfs->package;
Mike McCormack1da28582005-08-22 14:09:17 +00001148 MSIFEATURE *feature = ilfs->feature;
Mike McCormack38d67a42005-08-22 09:15:23 +00001149 MSICOMPONENT *comp;
Aric Stewart04598242005-06-23 16:43:24 +00001150
Mike McCormack38d67a42005-08-22 09:15:23 +00001151 comp = load_component( row );
1152 if (!comp)
1153 return ERROR_FUNCTION_FAILED;
Aric Stewart04598242005-06-23 16:43:24 +00001154
Mike McCormack38d67a42005-08-22 09:15:23 +00001155 list_add_tail( &package->components, &comp->entry );
Mike McCormack1da28582005-08-22 14:09:17 +00001156 add_feature_component( feature, comp );
Mike McCormack38d67a42005-08-22 09:15:23 +00001157
1158 TRACE("Loaded new component %p\n", comp);
Aric Stewart04598242005-06-23 16:43:24 +00001159
1160 return ERROR_SUCCESS;
1161}
1162
1163static UINT iterate_load_featurecomponents(MSIRECORD *row, LPVOID param)
1164{
1165 _ilfs* ilfs= (_ilfs*)param;
1166 LPCWSTR component;
1167 DWORD rc;
Mike McCormack38d67a42005-08-22 09:15:23 +00001168 MSICOMPONENT *comp;
Aric Stewart04598242005-06-23 16:43:24 +00001169 MSIQUERY * view;
1170 static const WCHAR Query[] =
1171 {'S','E','L','E','C','T',' ','*',' ','F','R', 'O','M',' ',
1172 '`','C','o','m','p','o','n','e','n','t','`',' ',
1173 'W','H','E','R','E',' ',
1174 '`','C','o','m','p','o','n','e','n','t','`',' ',
1175 '=','\'','%','s','\'',0};
1176
1177 component = MSI_RecordGetString(row,1);
1178
1179 /* check to see if the component is already loaded */
Mike McCormack38d67a42005-08-22 09:15:23 +00001180 comp = get_loaded_component( ilfs->package, component );
1181 if (comp)
Aric Stewart04598242005-06-23 16:43:24 +00001182 {
Mike McCormack38d67a42005-08-22 09:15:23 +00001183 TRACE("Component %s already loaded\n", debugstr_w(component) );
Mike McCormack1da28582005-08-22 14:09:17 +00001184 add_feature_component( ilfs->feature, comp );
Aric Stewart04598242005-06-23 16:43:24 +00001185 return ERROR_SUCCESS;
1186 }
1187
1188 rc = MSI_OpenQuery(ilfs->package->db, &view, Query, component);
1189 if (rc != ERROR_SUCCESS)
1190 return ERROR_SUCCESS;
1191
Aric Stewart04598242005-06-23 16:43:24 +00001192 rc = MSI_IterateRecords(view, NULL, iterate_component_check, ilfs);
1193 msiobj_release( &view->hdr );
1194
1195 return ERROR_SUCCESS;
1196}
1197
1198static UINT load_feature(MSIRECORD * row, LPVOID param)
1199{
1200 MSIPACKAGE* package = (MSIPACKAGE*)param;
Mike McCormack1da28582005-08-22 14:09:17 +00001201 MSIFEATURE* feature;
Aric Stewart8e233e92005-03-01 11:45:19 +00001202 static const WCHAR Query1[] =
Aric Stewart98e38082005-05-20 09:40:42 +00001203 {'S','E','L','E','C','T',' ',
1204 '`','C','o','m','p','o','n','e','n','t','_','`',
1205 ' ','F','R','O','M',' ','`','F','e','a','t','u','r','e',
1206 'C','o','m','p','o','n','e','n','t','s','`',' ',
1207 'W','H','E','R','E',' ',
1208 '`','F','e', 'a','t','u','r','e','_','`',' ','=','\'','%','s','\'',0};
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001209 MSIQUERY * view;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001210 UINT rc;
Aric Stewart04598242005-06-23 16:43:24 +00001211 _ilfs ilfs;
1212
Aric Stewartbdb29552004-07-04 00:32:48 +00001213 /* fill in the data */
1214
Mike McCormackee034ba2005-09-20 11:59:14 +00001215 feature = msi_alloc_zero( sizeof (MSIFEATURE) );
Mike McCormack1da28582005-08-22 14:09:17 +00001216 if (!feature)
1217 return ERROR_NOT_ENOUGH_MEMORY;
Aric Stewartbdb29552004-07-04 00:32:48 +00001218
Mike McCormack1da28582005-08-22 14:09:17 +00001219 list_init( &feature->Components );
Aric Stewartbdb29552004-07-04 00:32:48 +00001220
Mike McCormack51c66182005-10-27 12:36:12 +00001221 feature->Feature = msi_dup_record_field( row, 1 );
Aric Stewartbdb29552004-07-04 00:32:48 +00001222
Mike McCormack1da28582005-08-22 14:09:17 +00001223 TRACE("Loading feature %s\n",debugstr_w(feature->Feature));
Aric Stewartbdb29552004-07-04 00:32:48 +00001224
Mike McCormack51c66182005-10-27 12:36:12 +00001225 feature->Feature_Parent = msi_dup_record_field( row, 2 );
1226 feature->Title = msi_dup_record_field( row, 3 );
1227 feature->Description = msi_dup_record_field( row, 4 );
Aric Stewartbdb29552004-07-04 00:32:48 +00001228
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001229 if (!MSI_RecordIsNull(row,5))
Mike McCormack1da28582005-08-22 14:09:17 +00001230 feature->Display = MSI_RecordGetInteger(row,5);
Aric Stewartbdb29552004-07-04 00:32:48 +00001231
Mike McCormack1da28582005-08-22 14:09:17 +00001232 feature->Level= MSI_RecordGetInteger(row,6);
Mike McCormack51c66182005-10-27 12:36:12 +00001233 feature->Directory = msi_dup_record_field( row, 7 );
Mike McCormack1da28582005-08-22 14:09:17 +00001234 feature->Attributes = MSI_RecordGetInteger(row,8);
Aric Stewartfbdd7092004-12-27 19:06:22 +00001235
Mike McCormack1da28582005-08-22 14:09:17 +00001236 feature->Installed = INSTALLSTATE_ABSENT;
1237 feature->Action = INSTALLSTATE_UNKNOWN;
1238 feature->ActionRequest = INSTALLSTATE_UNKNOWN;
1239
1240 list_add_tail( &package->features, &feature->entry );
Aric Stewartbdb29552004-07-04 00:32:48 +00001241
1242 /* load feature components */
1243
Mike McCormack1da28582005-08-22 14:09:17 +00001244 rc = MSI_OpenQuery( package->db, &view, Query1, feature->Feature );
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001245 if (rc != ERROR_SUCCESS)
Aric Stewart04598242005-06-23 16:43:24 +00001246 return ERROR_SUCCESS;
Aric Stewartbdb29552004-07-04 00:32:48 +00001247
Mike McCormack1da28582005-08-22 14:09:17 +00001248 ilfs.package = package;
1249 ilfs.feature = feature;
1250
Aric Stewart04598242005-06-23 16:43:24 +00001251 MSI_IterateRecords(view, NULL, iterate_load_featurecomponents , &ilfs);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001252 msiobj_release(&view->hdr);
Aric Stewart04598242005-06-23 16:43:24 +00001253
1254 return ERROR_SUCCESS;
Aric Stewartbdb29552004-07-04 00:32:48 +00001255}
1256
Mike McCormackc1513be2006-03-21 19:40:36 +09001257static LPWSTR folder_split_path(LPWSTR p, WCHAR ch)
1258{
1259 if (!p)
1260 return p;
1261 p = strchrW(p, ch);
1262 if (!p)
1263 return p;
1264 *p = 0;
1265 return p+1;
1266}
1267
Aric Stewart04598242005-06-23 16:43:24 +00001268static UINT load_file(MSIRECORD *row, LPVOID param)
Aric Stewartc5a14432005-05-18 17:46:12 +00001269{
Aric Stewart04598242005-06-23 16:43:24 +00001270 MSIPACKAGE* package = (MSIPACKAGE*)param;
Aric Stewart09b0aba2005-06-09 20:30:59 +00001271 LPCWSTR component;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001272 MSIFILE *file;
Aric Stewartc5a14432005-05-18 17:46:12 +00001273
1274 /* fill in the data */
1275
Mike McCormackee034ba2005-09-20 11:59:14 +00001276 file = msi_alloc_zero( sizeof (MSIFILE) );
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001277 if (!file)
1278 return ERROR_NOT_ENOUGH_MEMORY;
Aric Stewartc5a14432005-05-18 17:46:12 +00001279
Mike McCormack51c66182005-10-27 12:36:12 +00001280 file->File = msi_dup_record_field( row, 1 );
Aric Stewartc5a14432005-05-18 17:46:12 +00001281
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001282 component = MSI_RecordGetString( row, 2 );
1283 file->Component = get_loaded_component( package, component );
Aric Stewart09b0aba2005-06-09 20:30:59 +00001284
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001285 if (!file->Component)
Aric Stewart09b0aba2005-06-09 20:30:59 +00001286 ERR("Unfound Component %s\n",debugstr_w(component));
Aric Stewartc5a14432005-05-18 17:46:12 +00001287
Mike McCormack51c66182005-10-27 12:36:12 +00001288 file->FileName = msi_dup_record_field( row, 3 );
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001289 reduce_to_longfilename( file->FileName );
Aric Stewartc5a14432005-05-18 17:46:12 +00001290
Mike McCormack51c66182005-10-27 12:36:12 +00001291 file->ShortName = msi_dup_record_field( row, 3 );
Mike McCormackc1513be2006-03-21 19:40:36 +09001292 file->LongName = strdupW( folder_split_path(file->ShortName, '|'));
Aric Stewartc5a14432005-05-18 17:46:12 +00001293
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001294 file->FileSize = MSI_RecordGetInteger( row, 4 );
Mike McCormack51c66182005-10-27 12:36:12 +00001295 file->Version = msi_dup_record_field( row, 5 );
1296 file->Language = msi_dup_record_field( row, 6 );
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001297 file->Attributes = MSI_RecordGetInteger( row, 7 );
1298 file->Sequence = MSI_RecordGetInteger( row, 8 );
Aric Stewartc5a14432005-05-18 17:46:12 +00001299
Mike McCormackdded8fb2005-11-02 10:56:42 +00001300 file->state = msifs_invalid;
Aric Stewartc5a14432005-05-18 17:46:12 +00001301
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001302 TRACE("File Loaded (%s)\n",debugstr_w(file->File));
1303
1304 list_add_tail( &package->files, &file->entry );
Aric Stewartc5a14432005-05-18 17:46:12 +00001305
1306 return ERROR_SUCCESS;
1307}
1308
1309static UINT load_all_files(MSIPACKAGE *package)
1310{
1311 MSIQUERY * view;
Aric Stewartc5a14432005-05-18 17:46:12 +00001312 UINT rc;
1313 static const WCHAR Query[] =
1314 {'S','E','L','E','C','T',' ','*',' ', 'F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001315 '`','F','i','l','e','`',' ', 'O','R','D','E','R',' ','B','Y',' ',
1316 '`','S','e','q','u','e','n','c','e','`', 0};
Aric Stewartc5a14432005-05-18 17:46:12 +00001317
Mike McCormack9a9195d2006-07-19 17:01:07 +09001318 if (!list_empty(&package->files))
1319 return ERROR_SUCCESS;
1320
Aric Stewartc5a14432005-05-18 17:46:12 +00001321 rc = MSI_DatabaseOpenViewW(package->db, Query, &view);
1322 if (rc != ERROR_SUCCESS)
1323 return ERROR_SUCCESS;
Aric Stewartc5a14432005-05-18 17:46:12 +00001324
Aric Stewart04598242005-06-23 16:43:24 +00001325 rc = MSI_IterateRecords(view, NULL, load_file, package);
Aric Stewartc5a14432005-05-18 17:46:12 +00001326 msiobj_release(&view->hdr);
1327
1328 return ERROR_SUCCESS;
1329}
1330
1331
Aric Stewartbdb29552004-07-04 00:32:48 +00001332/*
Mike McCormack9a9195d2006-07-19 17:01:07 +09001333 * I am not doing any of the costing functionality yet.
Aric Stewartbdb29552004-07-04 00:32:48 +00001334 * Mostly looking at doing the Component and Feature loading
1335 *
Francois Gougetda8b3dd2005-01-26 21:09:04 +00001336 * The native MSI does A LOT of modification to tables here. Mostly adding
Mike McCormack9a9195d2006-07-19 17:01:07 +09001337 * a lot of temporary columns to the Feature and Component tables.
Aric Stewartbdb29552004-07-04 00:32:48 +00001338 *
Francois Gougetda8b3dd2005-01-26 21:09:04 +00001339 * note: Native msi also tracks the short filename. But I am only going to
Aric Stewartbdb29552004-07-04 00:32:48 +00001340 * track the long ones. Also looking at this directory table
1341 * it appears that the directory table does not get the parents
Mike McCormack9a9195d2006-07-19 17:01:07 +09001342 * resolved base on property only based on their entries in the
Aric Stewartbdb29552004-07-04 00:32:48 +00001343 * directory table.
1344 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001345static UINT ACTION_CostInitialize(MSIPACKAGE *package)
Aric Stewartbdb29552004-07-04 00:32:48 +00001346{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001347 MSIQUERY * view;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001348 UINT rc;
Aric Stewart8e233e92005-03-01 11:45:19 +00001349 static const WCHAR Query_all[] =
1350 {'S','E','L','E','C','T',' ','*',' ', 'F','R','O','M',' ',
James Hawkins2396e2a2006-07-18 11:40:44 -07001351 '`','F','e','a','t','u','r','e','`',' ','O','R','D','E','R',
1352 ' ','B','Y',' ','`','D','i','s','p','l','a','y','`',0};
Aric Stewart8e233e92005-03-01 11:45:19 +00001353 static const WCHAR szCosting[] =
1354 {'C','o','s','t','i','n','g','C','o','m','p','l','e','t','e',0 };
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001355 static const WCHAR szZero[] = { '0', 0 };
Aric Stewartbdb29552004-07-04 00:32:48 +00001356
Mike McCormack9293f862005-10-30 19:16:45 +00001357 if ( 1 == msi_get_property_int( package, szCosting, 0 ) )
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001358 return ERROR_SUCCESS;
Mike McCormack9a9195d2006-07-19 17:01:07 +09001359
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001360 MSI_SetPropertyW(package, szCosting, szZero);
Juan Lang014ad3b2005-03-01 10:41:52 +00001361 MSI_SetPropertyW(package, cszRootDrive , c_colon);
Aric Stewartbdb29552004-07-04 00:32:48 +00001362
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001363 rc = MSI_DatabaseOpenViewW(package->db,Query_all,&view);
1364 if (rc != ERROR_SUCCESS)
1365 return rc;
Aric Stewartbdb29552004-07-04 00:32:48 +00001366
Mike McCormack9a9195d2006-07-19 17:01:07 +09001367 if (list_empty(&package->features))
1368 {
1369 rc = MSI_IterateRecords(view, NULL, load_feature, package);
1370 msiobj_release(&view->hdr);
1371 }
Aric Stewartec688fb2004-07-04 00:35:52 +00001372
Aric Stewartc5a14432005-05-18 17:46:12 +00001373 load_all_files(package);
Aric Stewartec688fb2004-07-04 00:35:52 +00001374
Aric Stewartec688fb2004-07-04 00:35:52 +00001375 return ERROR_SUCCESS;
1376}
1377
Mike McCormackf9acfe62005-06-07 20:29:51 +00001378static UINT execute_script(MSIPACKAGE *package, UINT script )
Aric Stewart9cd707d2005-05-27 19:24:22 +00001379{
1380 int i;
1381 UINT rc = ERROR_SUCCESS;
1382
1383 TRACE("Executing Script %i\n",script);
1384
Mike McCormackaa81e4f2006-01-10 12:09:19 +01001385 if (!package->script)
1386 {
1387 ERR("no script!\n");
1388 return ERROR_FUNCTION_FAILED;
1389 }
1390
Aric Stewart9cd707d2005-05-27 19:24:22 +00001391 for (i = 0; i < package->script->ActionCount[script]; i++)
1392 {
1393 LPWSTR action;
1394 action = package->script->Actions[script][i];
1395 ui_actionstart(package, action);
1396 TRACE("Executing Action (%s)\n",debugstr_w(action));
1397 rc = ACTION_PerformAction(package, action, TRUE);
Mike McCormackee034ba2005-09-20 11:59:14 +00001398 msi_free(package->script->Actions[script][i]);
Aric Stewart9cd707d2005-05-27 19:24:22 +00001399 if (rc != ERROR_SUCCESS)
1400 break;
1401 }
Mike McCormackee034ba2005-09-20 11:59:14 +00001402 msi_free(package->script->Actions[script]);
Aric Stewart9cd707d2005-05-27 19:24:22 +00001403
1404 package->script->ActionCount[script] = 0;
1405 package->script->Actions[script] = NULL;
1406 return rc;
1407}
1408
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001409static UINT ACTION_FileCost(MSIPACKAGE *package)
Aric Stewartec688fb2004-07-04 00:35:52 +00001410{
Aric Stewartbdb29552004-07-04 00:32:48 +00001411 return ERROR_SUCCESS;
1412}
1413
Mike McCormack979511f2005-08-23 18:15:44 +00001414static MSIFOLDER *load_folder( MSIPACKAGE *package, LPCWSTR dir )
Aric Stewartbdb29552004-07-04 00:32:48 +00001415{
Mike McCormackba8200b2004-12-22 15:25:30 +00001416 static const WCHAR Query[] =
Aric Stewart8e233e92005-03-01 11:45:19 +00001417 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001418 '`','D','i','r','e','c', 't','o','r','y','`',' ',
1419 'W','H','E','R','E',' ', '`', 'D','i','r','e','c','t', 'o','r','y','`',
1420 ' ','=',' ','\'','%','s','\'',
Aric Stewart8e233e92005-03-01 11:45:19 +00001421 0};
Mike McCormackc1513be2006-03-21 19:40:36 +09001422 static const WCHAR szDot[] = { '.',0 };
James Hawkinsa1910e12006-06-20 15:46:41 -07001423 static WCHAR szEmpty[] = { 0 };
Mike McCormackc1513be2006-03-21 19:40:36 +09001424 LPWSTR p, tgt_short, tgt_long, src_short, src_long;
Aric Stewart09b0aba2005-06-09 20:30:59 +00001425 LPCWSTR parent;
Mike McCormackc1513be2006-03-21 19:40:36 +09001426 MSIRECORD *row;
Mike McCormack979511f2005-08-23 18:15:44 +00001427 MSIFOLDER *folder;
Aric Stewarte95136b2004-06-29 03:44:01 +00001428
Aric Stewartbdb29552004-07-04 00:32:48 +00001429 TRACE("Looking for dir %s\n",debugstr_w(dir));
Aric Stewart401bd3f2004-06-28 20:34:35 +00001430
Mike McCormack979511f2005-08-23 18:15:44 +00001431 folder = get_loaded_folder( package, dir );
1432 if (folder)
1433 return folder;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001434
Aric Stewartbdb29552004-07-04 00:32:48 +00001435 TRACE("Working to load %s\n",debugstr_w(dir));
1436
Mike McCormackee034ba2005-09-20 11:59:14 +00001437 folder = msi_alloc_zero( sizeof (MSIFOLDER) );
Mike McCormack979511f2005-08-23 18:15:44 +00001438 if (!folder)
1439 return NULL;
Aric Stewartbdb29552004-07-04 00:32:48 +00001440
Mike McCormack979511f2005-08-23 18:15:44 +00001441 folder->Directory = strdupW(dir);
Aric Stewartbdb29552004-07-04 00:32:48 +00001442
Mike McCormack0b352c72005-06-02 10:29:57 +00001443 row = MSI_QueryGetRecord(package->db, Query, dir);
1444 if (!row)
Mike McCormack979511f2005-08-23 18:15:44 +00001445 return NULL;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001446
Mike McCormackc1513be2006-03-21 19:40:36 +09001447 p = msi_dup_record_field(row, 3);
Aric Stewart401bd3f2004-06-28 20:34:35 +00001448
1449 /* split src and target dir */
Mike McCormackc1513be2006-03-21 19:40:36 +09001450 tgt_short = p;
1451 src_short = folder_split_path( p, ':' );
Aric Stewart401bd3f2004-06-28 20:34:35 +00001452
Francois Gougete739ba92006-05-12 00:06:31 +02001453 /* split the long and short paths */
Mike McCormackc1513be2006-03-21 19:40:36 +09001454 tgt_long = folder_split_path( tgt_short, '|' );
1455 src_long = folder_split_path( src_short, '|' );
Aric Stewart401bd3f2004-06-28 20:34:35 +00001456
James Hawkinsa1910e12006-06-20 15:46:41 -07001457 /* check for no-op dirs */
Mike McCormackc1513be2006-03-21 19:40:36 +09001458 if (!lstrcmpW(szDot, tgt_short))
James Hawkinsa1910e12006-06-20 15:46:41 -07001459 tgt_short = szEmpty;
1460 if (!lstrcmpW(szDot, src_short))
1461 src_short = szEmpty;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001462
Mike McCormackc1513be2006-03-21 19:40:36 +09001463 if (!tgt_long)
1464 tgt_long = tgt_short;
Labrousse Jeand0862c32006-05-21 17:27:29 +02001465
1466 if (!src_short) {
1467 src_short = tgt_short;
1468 src_long = tgt_long;
1469 }
1470
Mike McCormackc1513be2006-03-21 19:40:36 +09001471 if (!src_long)
1472 src_long = src_short;
Mike McCormackba8200b2004-12-22 15:25:30 +00001473
Mike McCormackc1513be2006-03-21 19:40:36 +09001474 /* FIXME: use the target short path too */
1475 folder->TargetDefault = strdupW(tgt_long);
Labrousse Jeand0862c32006-05-21 17:27:29 +02001476 folder->SourceShortPath = strdupW(src_short);
Mike McCormackc1513be2006-03-21 19:40:36 +09001477 folder->SourceLongPath = strdupW(src_long);
1478 msi_free(p);
1479
1480 TRACE("TargetDefault = %s\n",debugstr_w( folder->TargetDefault ));
1481 TRACE("SourceLong = %s\n", debugstr_w( folder->SourceLongPath ));
1482 TRACE("SourceShort = %s\n", debugstr_w( folder->SourceShortPath ));
1483
1484 parent = MSI_RecordGetString(row, 2);
Mike McCormackba8200b2004-12-22 15:25:30 +00001485 if (parent)
Aric Stewart401bd3f2004-06-28 20:34:35 +00001486 {
Mike McCormack979511f2005-08-23 18:15:44 +00001487 folder->Parent = load_folder( package, parent );
1488 if ( folder->Parent )
1489 TRACE("loaded parent %p %s\n", folder->Parent,
1490 debugstr_w(folder->Parent->Directory));
1491 else
1492 ERR("failed to load parent folder %s\n", debugstr_w(parent));
Aric Stewartbdb29552004-07-04 00:32:48 +00001493 }
Aric Stewart401bd3f2004-06-28 20:34:35 +00001494
Mike McCormack062ad502005-09-15 15:04:08 +00001495 folder->Property = msi_dup_property( package, dir );
Aric Stewart401bd3f2004-06-28 20:34:35 +00001496
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001497 msiobj_release(&row->hdr);
Mike McCormack979511f2005-08-23 18:15:44 +00001498
1499 list_add_tail( &package->folders, &folder->entry );
1500
1501 TRACE("%s returning %p\n",debugstr_w(dir),folder);
1502
1503 return folder;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001504}
1505
Aric Stewart78a04e32005-02-22 15:47:00 +00001506/* scan for and update current install states */
Mike McCormackf9acfe62005-06-07 20:29:51 +00001507static void ACTION_UpdateInstallStates(MSIPACKAGE *package)
Aric Stewart78a04e32005-02-22 15:47:00 +00001508{
Mike McCormack38d67a42005-08-22 09:15:23 +00001509 MSICOMPONENT *comp;
Mike McCormack1da28582005-08-22 14:09:17 +00001510 MSIFEATURE *feature;
Aric Stewart78a04e32005-02-22 15:47:00 +00001511
Mike McCormack38d67a42005-08-22 09:15:23 +00001512 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
Aric Stewart78a04e32005-02-22 15:47:00 +00001513 {
1514 INSTALLSTATE res;
Mike McCormack38d67a42005-08-22 09:15:23 +00001515 res = MsiGetComponentPathW( package->ProductCode,
1516 comp->ComponentId, NULL, NULL);
Aric Stewart78a04e32005-02-22 15:47:00 +00001517 if (res < 0)
1518 res = INSTALLSTATE_ABSENT;
Mike McCormack38d67a42005-08-22 09:15:23 +00001519 comp->Installed = res;
Aric Stewart78a04e32005-02-22 15:47:00 +00001520 }
1521
Mike McCormack1da28582005-08-22 14:09:17 +00001522 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewart78a04e32005-02-22 15:47:00 +00001523 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001524 ComponentList *cl;
Aric Stewart78a04e32005-02-22 15:47:00 +00001525 INSTALLSTATE res = -10;
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001526
Mike McCormack1da28582005-08-22 14:09:17 +00001527 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewart78a04e32005-02-22 15:47:00 +00001528 {
Mike McCormack38d67a42005-08-22 09:15:23 +00001529 comp= cl->component;
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001530
Aric Stewart78a04e32005-02-22 15:47:00 +00001531 if (res == -10)
Mike McCormack38d67a42005-08-22 09:15:23 +00001532 res = comp->Installed;
Aric Stewart78a04e32005-02-22 15:47:00 +00001533 else
1534 {
Mike McCormack38d67a42005-08-22 09:15:23 +00001535 if (res == comp->Installed)
Aric Stewart78a04e32005-02-22 15:47:00 +00001536 continue;
1537
Mike McCormack38d67a42005-08-22 09:15:23 +00001538 if (res != comp->Installed)
Aric Stewart78a04e32005-02-22 15:47:00 +00001539 res = INSTALLSTATE_INCOMPLETE;
1540 }
1541 }
Mike McCormack8aa1a912005-08-25 19:19:10 +00001542 feature->Installed = res;
Aric Stewart78a04e32005-02-22 15:47:00 +00001543 }
1544}
1545
Aric Stewart78a04e32005-02-22 15:47:00 +00001546static BOOL process_state_property (MSIPACKAGE* package, LPCWSTR property,
1547 INSTALLSTATE state)
Aric Stewartae1aa322004-12-27 19:02:59 +00001548{
Aric Stewartae1aa322004-12-27 19:02:59 +00001549 static const WCHAR all[]={'A','L','L',0};
Mike McCormack72faac02005-09-08 11:03:35 +00001550 LPWSTR override;
Mike McCormack1da28582005-08-22 14:09:17 +00001551 MSIFEATURE *feature;
Aric Stewartae1aa322004-12-27 19:02:59 +00001552
Mike McCormack062ad502005-09-15 15:04:08 +00001553 override = msi_dup_property( package, property );
Mike McCormack72faac02005-09-08 11:03:35 +00001554 if (!override)
1555 return FALSE;
1556
1557 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartae1aa322004-12-27 19:02:59 +00001558 {
Mike McCormack72faac02005-09-08 11:03:35 +00001559 if (strcmpiW(override,all)==0)
Aric Stewartfbdd7092004-12-27 19:06:22 +00001560 {
Mike McCormack72faac02005-09-08 11:03:35 +00001561 feature->ActionRequest= state;
1562 feature->Action = state;
1563 }
1564 else
1565 {
1566 LPWSTR ptr = override;
1567 LPWSTR ptr2 = strchrW(override,',');
Aric Stewartd900b532004-12-27 19:12:35 +00001568
Mike McCormack72faac02005-09-08 11:03:35 +00001569 while (ptr)
1570 {
1571 if ((ptr2 && strncmpW(ptr,feature->Feature, ptr2-ptr)==0)
1572 || (!ptr2 && strcmpW(ptr,feature->Feature)==0))
Aric Stewartd900b532004-12-27 19:12:35 +00001573 {
Mike McCormack72faac02005-09-08 11:03:35 +00001574 feature->ActionRequest= state;
1575 feature->Action = state;
1576 break;
Aric Stewartd900b532004-12-27 19:12:35 +00001577 }
Mike McCormack72faac02005-09-08 11:03:35 +00001578 if (ptr2)
1579 {
1580 ptr=ptr2+1;
1581 ptr2 = strchrW(ptr,',');
1582 }
1583 else
1584 break;
Aric Stewartd900b532004-12-27 19:12:35 +00001585 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001586 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001587 }
Mike McCormackee034ba2005-09-20 11:59:14 +00001588 msi_free(override);
Aric Stewart78a04e32005-02-22 15:47:00 +00001589
Mike McCormack72faac02005-09-08 11:03:35 +00001590 return TRUE;
Aric Stewart78a04e32005-02-22 15:47:00 +00001591}
1592
James Hawkins7bcac312006-07-19 11:17:16 -07001593UINT MSI_SetFeatureStates(MSIPACKAGE *package)
Aric Stewart78a04e32005-02-22 15:47:00 +00001594{
Mike McCormack74f0de92005-09-29 10:32:39 +00001595 int install_level;
Aric Stewart8e233e92005-03-01 11:45:19 +00001596 static const WCHAR szlevel[] =
1597 {'I','N','S','T','A','L','L','L','E','V','E','L',0};
1598 static const WCHAR szAddLocal[] =
1599 {'A','D','D','L','O','C','A','L',0};
1600 static const WCHAR szRemove[] =
1601 {'R','E','M','O','V','E',0};
Aric Stewartd5655f92005-11-02 14:21:17 +00001602 static const WCHAR szReinstall[] =
1603 {'R','E','I','N','S','T','A','L','L',0};
Aric Stewart78a04e32005-02-22 15:47:00 +00001604 BOOL override = FALSE;
Mike McCormack38d67a42005-08-22 09:15:23 +00001605 MSICOMPONENT* component;
Mike McCormack1da28582005-08-22 14:09:17 +00001606 MSIFEATURE *feature;
1607
Aric Stewart78a04e32005-02-22 15:47:00 +00001608
1609 /* I do not know if this is where it should happen.. but */
1610
1611 TRACE("Checking Install Level\n");
1612
Mike McCormack74f0de92005-09-29 10:32:39 +00001613 install_level = msi_get_property_int( package, szlevel, 1 );
Aric Stewart78a04e32005-02-22 15:47:00 +00001614
1615 /* ok hereis the _real_ rub
Francois Gougetfbb33432005-03-02 13:53:50 +00001616 * all these activation/deactivation things happen in order and things
1617 * later on the list override things earlier on the list.
Aric Stewart78a04e32005-02-22 15:47:00 +00001618 * 1) INSTALLLEVEL processing
1619 * 2) ADDLOCAL
1620 * 3) REMOVE
1621 * 4) ADDSOURCE
1622 * 5) ADDDEFAULT
1623 * 6) REINSTALL
1624 * 7) COMPADDLOCAL
1625 * 8) COMPADDSOURCE
1626 * 9) FILEADDLOCAL
1627 * 10) FILEADDSOURCE
1628 * 11) FILEADDDEFAULT
Francois Gougetfbb33432005-03-02 13:53:50 +00001629 * I have confirmed that if ADDLOCAL is stated then the INSTALLLEVEL is
1630 * ignored for all the features. seems strange, especially since it is not
Aric Stewart78a04e32005-02-22 15:47:00 +00001631 * documented anywhere, but it is how it works.
1632 *
Francois Gougetfbb33432005-03-02 13:53:50 +00001633 * I am still ignoring a lot of these. But that is ok for now, ADDLOCAL and
1634 * REMOVE are the big ones, since we don't handle administrative installs
1635 * yet anyway.
Aric Stewart78a04e32005-02-22 15:47:00 +00001636 */
1637 override |= process_state_property(package,szAddLocal,INSTALLSTATE_LOCAL);
1638 override |= process_state_property(package,szRemove,INSTALLSTATE_ABSENT);
Aric Stewartd5655f92005-11-02 14:21:17 +00001639 override |= process_state_property(package,szReinstall,INSTALLSTATE_LOCAL);
Aric Stewart78a04e32005-02-22 15:47:00 +00001640
1641 if (!override)
Aric Stewartfbdd7092004-12-27 19:06:22 +00001642 {
Mike McCormack1da28582005-08-22 14:09:17 +00001643 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartfbdd7092004-12-27 19:06:22 +00001644 {
Mike McCormack1da28582005-08-22 14:09:17 +00001645 BOOL feature_state = ((feature->Level > 0) &&
1646 (feature->Level <= install_level));
Aric Stewartae1aa322004-12-27 19:02:59 +00001647
Mike McCormack1da28582005-08-22 14:09:17 +00001648 if ((feature_state) && (feature->Action == INSTALLSTATE_UNKNOWN))
Aric Stewartfbdd7092004-12-27 19:06:22 +00001649 {
Mike McCormack1da28582005-08-22 14:09:17 +00001650 if (feature->Attributes & msidbFeatureAttributesFavorSource)
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001651 {
Mike McCormack1da28582005-08-22 14:09:17 +00001652 feature->ActionRequest = INSTALLSTATE_SOURCE;
1653 feature->Action = INSTALLSTATE_SOURCE;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001654 }
Mike McCormack1da28582005-08-22 14:09:17 +00001655 else if (feature->Attributes & msidbFeatureAttributesFavorAdvertise)
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001656 {
Mike McCormack1da28582005-08-22 14:09:17 +00001657 feature->ActionRequest = INSTALLSTATE_ADVERTISED;
1658 feature->Action = INSTALLSTATE_ADVERTISED;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001659 }
1660 else
1661 {
Mike McCormack1da28582005-08-22 14:09:17 +00001662 feature->ActionRequest = INSTALLSTATE_LOCAL;
1663 feature->Action = INSTALLSTATE_LOCAL;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001664 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001665 }
Aric Stewartae1aa322004-12-27 19:02:59 +00001666 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001667 }
Aric Stewart6999a042005-06-08 19:20:02 +00001668 else
1669 {
1670 /* set the Preselected Property */
1671 static const WCHAR szPreselected[] = {'P','r','e','s','e','l','e','c','t','e','d',0};
1672 static const WCHAR szOne[] = { '1', 0 };
1673
1674 MSI_SetPropertyW(package,szPreselected,szOne);
1675 }
Aric Stewartae1aa322004-12-27 19:02:59 +00001676
Aric Stewartfbdd7092004-12-27 19:06:22 +00001677 /*
1678 * now we want to enable or disable components base on feature
1679 */
1680
Mike McCormack1da28582005-08-22 14:09:17 +00001681 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartfbdd7092004-12-27 19:06:22 +00001682 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00001683 ComponentList *cl;
1684
Aric Stewartfbdd7092004-12-27 19:06:22 +00001685 TRACE("Examining Feature %s (Installed %i, Action %i, Request %i)\n",
1686 debugstr_w(feature->Feature), feature->Installed, feature->Action,
1687 feature->ActionRequest);
1688
Mike McCormack1da28582005-08-22 14:09:17 +00001689 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewartae1aa322004-12-27 19:02:59 +00001690 {
Mike McCormack38d67a42005-08-22 09:15:23 +00001691 component = cl->component;
Aric Stewartfbdd7092004-12-27 19:06:22 +00001692
1693 if (!component->Enabled)
1694 {
Aric Stewart78a04e32005-02-22 15:47:00 +00001695 component->Action = INSTALLSTATE_UNKNOWN;
1696 component->ActionRequest = INSTALLSTATE_UNKNOWN;
Aric Stewartfbdd7092004-12-27 19:06:22 +00001697 }
1698 else
1699 {
1700 if (feature->Action == INSTALLSTATE_LOCAL)
Aric Stewart78a04e32005-02-22 15:47:00 +00001701 {
Aric Stewartfbdd7092004-12-27 19:06:22 +00001702 component->Action = INSTALLSTATE_LOCAL;
Aric Stewartfbdd7092004-12-27 19:06:22 +00001703 component->ActionRequest = INSTALLSTATE_LOCAL;
Aric Stewart78a04e32005-02-22 15:47:00 +00001704 }
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001705 else if (feature->ActionRequest == INSTALLSTATE_SOURCE)
1706 {
1707 if ((component->Action == INSTALLSTATE_UNKNOWN) ||
1708 (component->Action == INSTALLSTATE_ABSENT) ||
1709 (component->Action == INSTALLSTATE_ADVERTISED))
1710
1711 {
1712 component->Action = INSTALLSTATE_SOURCE;
1713 component->ActionRequest = INSTALLSTATE_SOURCE;
1714 }
1715 }
1716 else if (feature->ActionRequest == INSTALLSTATE_ADVERTISED)
1717 {
1718 if ((component->Action == INSTALLSTATE_UNKNOWN) ||
1719 (component->Action == INSTALLSTATE_ABSENT))
1720
1721 {
1722 component->Action = INSTALLSTATE_ADVERTISED;
1723 component->ActionRequest = INSTALLSTATE_ADVERTISED;
1724 }
1725 }
Aric Stewart78a04e32005-02-22 15:47:00 +00001726 else if (feature->ActionRequest == INSTALLSTATE_ABSENT)
1727 {
1728 if (component->Action == INSTALLSTATE_UNKNOWN)
1729 {
1730 component->Action = INSTALLSTATE_ABSENT;
1731 component->ActionRequest = INSTALLSTATE_ABSENT;
1732 }
1733 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001734 }
Aric Stewartae1aa322004-12-27 19:02:59 +00001735 }
1736 }
Aric Stewartfbdd7092004-12-27 19:06:22 +00001737
Mike McCormack38d67a42005-08-22 09:15:23 +00001738 LIST_FOR_EACH_ENTRY( component, &package->components, MSICOMPONENT, entry )
Aric Stewartfbdd7092004-12-27 19:06:22 +00001739 {
Aric Stewartfbdd7092004-12-27 19:06:22 +00001740 TRACE("Result: Component %s (Installed %i, Action %i, Request %i)\n",
1741 debugstr_w(component->Component), component->Installed,
1742 component->Action, component->ActionRequest);
1743 }
1744
1745
Aric Stewartae1aa322004-12-27 19:02:59 +00001746 return ERROR_SUCCESS;
1747}
1748
Aric Stewart443ad4d2005-06-21 20:50:12 +00001749static UINT ITERATE_CostFinalizeDirectories(MSIRECORD *row, LPVOID param)
1750{
1751 MSIPACKAGE *package = (MSIPACKAGE*)param;
1752 LPCWSTR name;
1753 LPWSTR path;
1754
1755 name = MSI_RecordGetString(row,1);
1756
1757 /* This helper function now does ALL the work */
1758 TRACE("Dir %s ...\n",debugstr_w(name));
1759 load_folder(package,name);
1760 path = resolve_folder(package,name,FALSE,TRUE,NULL);
1761 TRACE("resolves to %s\n",debugstr_w(path));
Mike McCormackee034ba2005-09-20 11:59:14 +00001762 msi_free(path);
Aric Stewart443ad4d2005-06-21 20:50:12 +00001763
1764 return ERROR_SUCCESS;
1765}
1766
1767static UINT ITERATE_CostFinalizeConditions(MSIRECORD *row, LPVOID param)
1768{
1769 MSIPACKAGE *package = (MSIPACKAGE*)param;
Mike McCormack1da28582005-08-22 14:09:17 +00001770 LPCWSTR name;
1771 MSIFEATURE *feature;
Aric Stewart443ad4d2005-06-21 20:50:12 +00001772
Mike McCormack1da28582005-08-22 14:09:17 +00001773 name = MSI_RecordGetString( row, 1 );
Aric Stewart443ad4d2005-06-21 20:50:12 +00001774
Mike McCormack1da28582005-08-22 14:09:17 +00001775 feature = get_loaded_feature( package, name );
1776 if (!feature)
1777 ERR("FAILED to find loaded feature %s\n",debugstr_w(name));
Aric Stewart443ad4d2005-06-21 20:50:12 +00001778 else
1779 {
1780 LPCWSTR Condition;
1781 Condition = MSI_RecordGetString(row,3);
1782
Aric Stewart0713f092005-06-24 11:51:29 +00001783 if (MSI_EvaluateConditionW(package,Condition) == MSICONDITION_TRUE)
Aric Stewart443ad4d2005-06-21 20:50:12 +00001784 {
1785 int level = MSI_RecordGetInteger(row,2);
Mike McCormack1da28582005-08-22 14:09:17 +00001786 TRACE("Reseting feature %s to level %i\n", debugstr_w(name), level);
1787 feature->Level = level;
Aric Stewart443ad4d2005-06-21 20:50:12 +00001788 }
1789 }
1790 return ERROR_SUCCESS;
1791}
1792
1793
Aric Stewart401bd3f2004-06-28 20:34:35 +00001794/*
Francois Gougetda8b3dd2005-01-26 21:09:04 +00001795 * A lot is done in this function aside from just the costing.
Mike McCormack6e2bca32004-07-04 00:25:00 +00001796 * The costing needs to be implemented at some point but for now I am going
Aric Stewart401bd3f2004-06-28 20:34:35 +00001797 * to focus on the directory building
1798 *
Aric Stewart401bd3f2004-06-28 20:34:35 +00001799 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001800static UINT ACTION_CostFinalize(MSIPACKAGE *package)
Aric Stewart401bd3f2004-06-28 20:34:35 +00001801{
Aric Stewart8e233e92005-03-01 11:45:19 +00001802 static const WCHAR ExecSeqQuery[] =
1803 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001804 '`','D','i','r','e','c','t','o','r','y','`',0};
Aric Stewart8e233e92005-03-01 11:45:19 +00001805 static const WCHAR ConditionQuery[] =
1806 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00001807 '`','C','o','n','d','i','t','i','o','n','`',0};
Aric Stewart8e233e92005-03-01 11:45:19 +00001808 static const WCHAR szCosting[] =
1809 {'C','o','s','t','i','n','g','C','o','m','p','l','e','t','e',0 };
1810 static const WCHAR szlevel[] =
1811 {'I','N','S','T','A','L','L','L','E','V','E','L',0};
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001812 static const WCHAR szOne[] = { '1', 0 };
Mike McCormack38d67a42005-08-22 09:15:23 +00001813 MSICOMPONENT *comp;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001814 MSIFILE *file;
Aric Stewart401bd3f2004-06-28 20:34:35 +00001815 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001816 MSIQUERY * view;
Aric Stewart8cc14a92004-12-27 18:56:30 +00001817 LPWSTR level;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001818
Mike McCormack9293f862005-10-30 19:16:45 +00001819 if ( 1 == msi_get_property_int( package, szCosting, 0 ) )
Aric Stewartb39d8fc2005-05-13 13:56:39 +00001820 return ERROR_SUCCESS;
Mike McCormack9293f862005-10-30 19:16:45 +00001821
Aric Stewart401bd3f2004-06-28 20:34:35 +00001822 TRACE("Building Directory properties\n");
1823
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001824 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
Aric Stewart84837d92004-07-20 01:22:37 +00001825 if (rc == ERROR_SUCCESS)
Aric Stewart401bd3f2004-06-28 20:34:35 +00001826 {
Aric Stewart443ad4d2005-06-21 20:50:12 +00001827 rc = MSI_IterateRecords(view, NULL, ITERATE_CostFinalizeDirectories,
1828 package);
Aric Stewart84837d92004-07-20 01:22:37 +00001829 msiobj_release(&view->hdr);
1830 }
Aric Stewart401bd3f2004-06-28 20:34:35 +00001831
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001832 TRACE("File calculations\n");
Aric Stewartec688fb2004-07-04 00:35:52 +00001833
Mike McCormacke18f8ab2005-08-23 10:03:17 +00001834 LIST_FOR_EACH_ENTRY( file, &package->files, MSIFILE, entry )
Aric Stewartec688fb2004-07-04 00:35:52 +00001835 {
Mike McCormackf11c8b02005-09-09 14:48:51 +00001836 MSICOMPONENT* comp = file->Component;
1837 LPWSTR p;
Aric Stewartec688fb2004-07-04 00:35:52 +00001838
Mike McCormackf11c8b02005-09-09 14:48:51 +00001839 if (!comp)
1840 continue;
Aric Stewartec688fb2004-07-04 00:35:52 +00001841
Mike McCormackf11c8b02005-09-09 14:48:51 +00001842 /* calculate target */
1843 p = resolve_folder(package, comp->Directory, FALSE, FALSE, NULL);
1844
Mike McCormackee034ba2005-09-20 11:59:14 +00001845 msi_free(file->TargetPath);
Mike McCormackf11c8b02005-09-09 14:48:51 +00001846
1847 TRACE("file %s is named %s\n",
1848 debugstr_w(file->File),debugstr_w(file->FileName));
1849
1850 file->TargetPath = build_directory_name(2, p, file->FileName);
1851
Mike McCormackee034ba2005-09-20 11:59:14 +00001852 msi_free(p);
Mike McCormackf11c8b02005-09-09 14:48:51 +00001853
1854 TRACE("file %s resolves to %s\n",
1855 debugstr_w(file->File),debugstr_w(file->TargetPath));
1856
1857 if (GetFileAttributesW(file->TargetPath) == INVALID_FILE_ATTRIBUTES)
Aric Stewartec688fb2004-07-04 00:35:52 +00001858 {
Mike McCormackdded8fb2005-11-02 10:56:42 +00001859 file->state = msifs_missing;
Mike McCormackf11c8b02005-09-09 14:48:51 +00001860 comp->Cost += file->FileSize;
1861 continue;
1862 }
Mike McCormackba8200b2004-12-22 15:25:30 +00001863
Mike McCormackf11c8b02005-09-09 14:48:51 +00001864 if (file->Version)
1865 {
1866 DWORD handle;
1867 DWORD versize;
1868 UINT sz;
1869 LPVOID version;
Ivan Leo Puoti5b22e4d2005-11-03 11:35:11 +00001870 static WCHAR name[] = {'\\',0};
Mike McCormackf11c8b02005-09-09 14:48:51 +00001871 static const WCHAR name_fmt[] =
1872 {'%','u','.','%','u','.','%','u','.','%','u',0};
1873 WCHAR filever[0x100];
1874 VS_FIXEDFILEINFO *lpVer;
Aric Stewartfa384f62004-12-22 18:46:17 +00001875
Francois Gouget0edbaf72005-11-10 12:14:56 +00001876 TRACE("Version comparison..\n");
Mike McCormackf11c8b02005-09-09 14:48:51 +00001877 versize = GetFileVersionInfoSizeW(file->TargetPath,&handle);
Mike McCormackee034ba2005-09-20 11:59:14 +00001878 version = msi_alloc(versize);
Mike McCormackf11c8b02005-09-09 14:48:51 +00001879 GetFileVersionInfoW(file->TargetPath, 0, versize, version);
Aric Stewartfa384f62004-12-22 18:46:17 +00001880
Mike McCormackf11c8b02005-09-09 14:48:51 +00001881 VerQueryValueW(version, name, (LPVOID*)&lpVer, &sz);
Aric Stewartfa384f62004-12-22 18:46:17 +00001882
Mike McCormackf11c8b02005-09-09 14:48:51 +00001883 sprintfW(filever,name_fmt,
1884 HIWORD(lpVer->dwFileVersionMS),
1885 LOWORD(lpVer->dwFileVersionMS),
1886 HIWORD(lpVer->dwFileVersionLS),
1887 LOWORD(lpVer->dwFileVersionLS));
Aric Stewartec688fb2004-07-04 00:35:52 +00001888
Mike McCormackf11c8b02005-09-09 14:48:51 +00001889 TRACE("new %s old %s\n", debugstr_w(file->Version),
1890 debugstr_w(filever));
1891 if (strcmpiW(filever,file->Version)<0)
Aric Stewartec688fb2004-07-04 00:35:52 +00001892 {
Mike McCormackdded8fb2005-11-02 10:56:42 +00001893 file->state = msifs_overwrite;
1894 /* FIXME: cost should be diff in size */
Aric Stewartec688fb2004-07-04 00:35:52 +00001895 comp->Cost += file->FileSize;
1896 }
1897 else
Mike McCormackdded8fb2005-11-02 10:56:42 +00001898 file->state = msifs_present;
Mike McCormackee034ba2005-09-20 11:59:14 +00001899 msi_free(version);
Mike McCormackf11c8b02005-09-09 14:48:51 +00001900 }
1901 else
Mike McCormackdded8fb2005-11-02 10:56:42 +00001902 file->state = msifs_present;
Aric Stewartec688fb2004-07-04 00:35:52 +00001903 }
1904
Aric Stewart7d3e5972004-07-04 00:36:58 +00001905 TRACE("Evaluating Condition Table\n");
Aric Stewart2e9b5f72004-07-04 00:31:17 +00001906
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001907 rc = MSI_DatabaseOpenViewW(package->db, ConditionQuery, &view);
Aric Stewart84837d92004-07-20 01:22:37 +00001908 if (rc == ERROR_SUCCESS)
1909 {
Aric Stewart443ad4d2005-06-21 20:50:12 +00001910 rc = MSI_IterateRecords(view, NULL, ITERATE_CostFinalizeConditions,
1911 package);
Mike McCormackac6a4132005-01-04 20:36:12 +00001912 msiobj_release(&view->hdr);
Aric Stewart84837d92004-07-20 01:22:37 +00001913 }
Aric Stewart7d3e5972004-07-04 00:36:58 +00001914
1915 TRACE("Enabling or Disabling Components\n");
Mike McCormack38d67a42005-08-22 09:15:23 +00001916 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
Aric Stewart7d3e5972004-07-04 00:36:58 +00001917 {
Mike McCormack298cdae2005-09-06 11:39:01 +00001918 if (comp->Condition)
Aric Stewart7d3e5972004-07-04 00:36:58 +00001919 {
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001920 if (MSI_EvaluateConditionW(package,
Mike McCormack38d67a42005-08-22 09:15:23 +00001921 comp->Condition) == MSICONDITION_FALSE)
Aric Stewart7d3e5972004-07-04 00:36:58 +00001922 {
Mike McCormack38d67a42005-08-22 09:15:23 +00001923 TRACE("Disabling component %s\n", debugstr_w(comp->Component));
1924 comp->Enabled = FALSE;
Aric Stewart7d3e5972004-07-04 00:36:58 +00001925 }
1926 }
1927 }
1928
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001929 MSI_SetPropertyW(package,szCosting,szOne);
Aric Stewart8cc14a92004-12-27 18:56:30 +00001930 /* set default run level if not set */
Mike McCormack062ad502005-09-15 15:04:08 +00001931 level = msi_dup_property( package, szlevel );
Aric Stewart8cc14a92004-12-27 18:56:30 +00001932 if (!level)
1933 MSI_SetPropertyW(package,szlevel, szOne);
Mike McCormackee034ba2005-09-20 11:59:14 +00001934 msi_free(level);
Aric Stewartae1aa322004-12-27 19:02:59 +00001935
Aric Stewart78a04e32005-02-22 15:47:00 +00001936 ACTION_UpdateInstallStates(package);
Aric Stewartae1aa322004-12-27 19:02:59 +00001937
James Hawkins7c7f0bb2006-07-19 11:15:37 -07001938 return MSI_SetFeatureStates(package);
Aric Stewart401bd3f2004-06-28 20:34:35 +00001939}
1940
Francois Gougetda8b3dd2005-01-26 21:09:04 +00001941/* OK this value is "interpreted" and then formatted based on the
Aric Stewart6e160f12004-06-29 04:07:22 +00001942 first few characters */
Aric Stewart09b0aba2005-06-09 20:30:59 +00001943static LPSTR parse_value(MSIPACKAGE *package, LPCWSTR value, DWORD *type,
Aric Stewart401bd3f2004-06-28 20:34:35 +00001944 DWORD *size)
1945{
1946 LPSTR data = NULL;
Aric Stewart6e160f12004-06-29 04:07:22 +00001947 if (value[0]=='#' && value[1]!='#' && value[1]!='%')
Aric Stewart401bd3f2004-06-28 20:34:35 +00001948 {
Aric Stewart6e160f12004-06-29 04:07:22 +00001949 if (value[1]=='x')
1950 {
1951 LPWSTR ptr;
1952 CHAR byte[5];
Aric Stewart6186b2b2005-05-16 21:37:35 +00001953 LPWSTR deformated = NULL;
Aric Stewart6e160f12004-06-29 04:07:22 +00001954 int count;
1955
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00001956 deformat_string(package, &value[2], &deformated);
Aric Stewart6e160f12004-06-29 04:07:22 +00001957
1958 /* binary value type */
Aric Stewart6186b2b2005-05-16 21:37:35 +00001959 ptr = deformated;
1960 *type = REG_BINARY;
1961 if (strlenW(ptr)%2)
1962 *size = (strlenW(ptr)/2)+1;
1963 else
1964 *size = strlenW(ptr)/2;
1965
Mike McCormackee034ba2005-09-20 11:59:14 +00001966 data = msi_alloc(*size);
Aric Stewart6186b2b2005-05-16 21:37:35 +00001967
Aric Stewart6e160f12004-06-29 04:07:22 +00001968 byte[0] = '0';
1969 byte[1] = 'x';
1970 byte[4] = 0;
1971 count = 0;
Aric Stewart6186b2b2005-05-16 21:37:35 +00001972 /* if uneven pad with a zero in front */
1973 if (strlenW(ptr)%2)
1974 {
1975 byte[2]= '0';
1976 byte[3]= *ptr;
1977 ptr++;
1978 data[count] = (BYTE)strtol(byte,NULL,0);
1979 count ++;
1980 TRACE("Uneven byte count\n");
1981 }
Aric Stewart6e160f12004-06-29 04:07:22 +00001982 while (*ptr)
1983 {
1984 byte[2]= *ptr;
1985 ptr++;
1986 byte[3]= *ptr;
1987 ptr++;
1988 data[count] = (BYTE)strtol(byte,NULL,0);
1989 count ++;
1990 }
Mike McCormackee034ba2005-09-20 11:59:14 +00001991 msi_free(deformated);
Aric Stewart6e160f12004-06-29 04:07:22 +00001992
1993 TRACE("Data %li bytes(%i)\n",*size,count);
1994 }
1995 else
1996 {
1997 LPWSTR deformated;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00001998 LPWSTR p;
1999 DWORD d = 0;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002000 deformat_string(package, &value[1], &deformated);
Aric Stewart6e160f12004-06-29 04:07:22 +00002001
2002 *type=REG_DWORD;
2003 *size = sizeof(DWORD);
Mike McCormackee034ba2005-09-20 11:59:14 +00002004 data = msi_alloc(*size);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002005 p = deformated;
2006 if (*p == '-')
2007 p++;
2008 while (*p)
2009 {
2010 if ( (*p < '0') || (*p > '9') )
2011 break;
2012 d *= 10;
2013 d += (*p - '0');
2014 p++;
2015 }
2016 if (deformated[0] == '-')
2017 d = -d;
2018 *(LPDWORD)data = d;
2019 TRACE("DWORD %li\n",*(LPDWORD)data);
Aric Stewart6e160f12004-06-29 04:07:22 +00002020
Mike McCormackee034ba2005-09-20 11:59:14 +00002021 msi_free(deformated);
Aric Stewart6e160f12004-06-29 04:07:22 +00002022 }
Aric Stewart401bd3f2004-06-28 20:34:35 +00002023 }
2024 else
2025 {
Aric Stewart54c67dd2005-01-25 20:17:09 +00002026 static const WCHAR szMulti[] = {'[','~',']',0};
Aric Stewart09b0aba2005-06-09 20:30:59 +00002027 LPCWSTR ptr;
Aric Stewart6e160f12004-06-29 04:07:22 +00002028 *type=REG_SZ;
2029
Aric Stewart401bd3f2004-06-28 20:34:35 +00002030 if (value[0]=='#')
Aric Stewart6e160f12004-06-29 04:07:22 +00002031 {
2032 if (value[1]=='%')
2033 {
2034 ptr = &value[2];
2035 *type=REG_EXPAND_SZ;
2036 }
2037 else
2038 ptr = &value[1];
2039 }
2040 else
Aric Stewart401bd3f2004-06-28 20:34:35 +00002041 ptr=value;
2042
Aric Stewart54c67dd2005-01-25 20:17:09 +00002043 if (strstrW(value,szMulti))
2044 *type = REG_MULTI_SZ;
2045
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002046 *size = deformat_string(package, ptr,(LPWSTR*)&data);
Aric Stewart401bd3f2004-06-28 20:34:35 +00002047 }
2048 return data;
2049}
2050
Aric Stewart92ef78e2005-06-21 20:21:18 +00002051static UINT ITERATE_WriteRegistryValues(MSIRECORD *row, LPVOID param)
2052{
2053 MSIPACKAGE *package = (MSIPACKAGE*)param;
2054 static const WCHAR szHCR[] =
2055 {'H','K','E','Y','_','C','L','A','S','S','E','S','_',
2056 'R','O','O','T','\\',0};
2057 static const WCHAR szHCU[] =
2058 {'H','K','E','Y','_','C','U','R','R','E','N','T','_',
2059 'U','S','E','R','\\',0};
2060 static const WCHAR szHLM[] =
2061 {'H','K','E','Y','_','L','O','C','A','L','_',
2062 'M','A','C','H','I','N','E','\\',0};
2063 static const WCHAR szHU[] =
2064 {'H','K','E','Y','_','U','S','E','R','S','\\',0};
2065
2066 LPSTR value_data = NULL;
2067 HKEY root_key, hkey;
2068 DWORD type,size;
2069 LPWSTR deformated;
2070 LPCWSTR szRoot, component, name, key, value;
Mike McCormack38d67a42005-08-22 09:15:23 +00002071 MSICOMPONENT *comp;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002072 MSIRECORD * uirow;
2073 LPWSTR uikey;
2074 INT root;
2075 BOOL check_first = FALSE;
2076 UINT rc;
2077
2078 ui_progress(package,2,0,0,0);
2079
2080 value = NULL;
2081 key = NULL;
2082 uikey = NULL;
2083 name = NULL;
2084
2085 component = MSI_RecordGetString(row, 6);
Mike McCormack38d67a42005-08-22 09:15:23 +00002086 comp = get_loaded_component(package,component);
Johan Dahlin0946c422005-08-24 10:57:27 +00002087 if (!comp)
2088 return ERROR_SUCCESS;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002089
Mike McCormackd693f462005-10-29 11:36:48 +00002090 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
Aric Stewart92ef78e2005-06-21 20:21:18 +00002091 {
2092 TRACE("Skipping write due to disabled component %s\n",
2093 debugstr_w(component));
2094
Mike McCormack38d67a42005-08-22 09:15:23 +00002095 comp->Action = comp->Installed;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002096
2097 return ERROR_SUCCESS;
2098 }
2099
Mike McCormack38d67a42005-08-22 09:15:23 +00002100 comp->Action = INSTALLSTATE_LOCAL;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002101
2102 name = MSI_RecordGetString(row, 4);
2103 if( MSI_RecordIsNull(row,5) && name )
2104 {
2105 /* null values can have special meanings */
2106 if (name[0]=='-' && name[1] == 0)
2107 return ERROR_SUCCESS;
2108 else if ((name[0]=='+' && name[1] == 0) ||
2109 (name[0] == '*' && name[1] == 0))
2110 name = NULL;
2111 check_first = TRUE;
2112 }
2113
2114 root = MSI_RecordGetInteger(row,2);
2115 key = MSI_RecordGetString(row, 3);
2116
2117 /* get the root key */
2118 switch (root)
2119 {
Aric Stewart0713f092005-06-24 11:51:29 +00002120 case -1:
2121 {
2122 static const WCHAR szALLUSER[] = {'A','L','L','U','S','E','R','S',0};
Mike McCormack062ad502005-09-15 15:04:08 +00002123 LPWSTR all_users = msi_dup_property( package, szALLUSER );
Aric Stewart0713f092005-06-24 11:51:29 +00002124 if (all_users && all_users[0] == '1')
2125 {
2126 root_key = HKEY_LOCAL_MACHINE;
2127 szRoot = szHLM;
2128 }
2129 else
2130 {
2131 root_key = HKEY_CURRENT_USER;
2132 szRoot = szHCU;
2133 }
Mike McCormackee034ba2005-09-20 11:59:14 +00002134 msi_free(all_users);
Aric Stewart0713f092005-06-24 11:51:29 +00002135 }
2136 break;
Aric Stewart92ef78e2005-06-21 20:21:18 +00002137 case 0: root_key = HKEY_CLASSES_ROOT;
2138 szRoot = szHCR;
2139 break;
2140 case 1: root_key = HKEY_CURRENT_USER;
2141 szRoot = szHCU;
2142 break;
2143 case 2: root_key = HKEY_LOCAL_MACHINE;
2144 szRoot = szHLM;
2145 break;
2146 case 3: root_key = HKEY_USERS;
2147 szRoot = szHU;
2148 break;
2149 default:
2150 ERR("Unknown root %i\n",root);
2151 root_key=NULL;
2152 szRoot = NULL;
2153 break;
2154 }
2155 if (!root_key)
2156 return ERROR_SUCCESS;
2157
2158 deformat_string(package, key , &deformated);
2159 size = strlenW(deformated) + strlenW(szRoot) + 1;
Mike McCormackee034ba2005-09-20 11:59:14 +00002160 uikey = msi_alloc(size*sizeof(WCHAR));
Aric Stewart92ef78e2005-06-21 20:21:18 +00002161 strcpyW(uikey,szRoot);
2162 strcatW(uikey,deformated);
2163
2164 if (RegCreateKeyW( root_key, deformated, &hkey))
2165 {
2166 ERR("Could not create key %s\n",debugstr_w(deformated));
Mike McCormackee034ba2005-09-20 11:59:14 +00002167 msi_free(deformated);
2168 msi_free(uikey);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002169 return ERROR_SUCCESS;
2170 }
Mike McCormackee034ba2005-09-20 11:59:14 +00002171 msi_free(deformated);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002172
2173 value = MSI_RecordGetString(row,5);
2174 if (value)
2175 value_data = parse_value(package, value, &type, &size);
2176 else
2177 {
2178 static const WCHAR szEmpty[] = {0};
2179 value_data = (LPSTR)strdupW(szEmpty);
2180 size = 0;
2181 type = REG_SZ;
2182 }
2183
2184 deformat_string(package, name, &deformated);
2185
2186 /* get the double nulls to terminate SZ_MULTI */
2187 if (type == REG_MULTI_SZ)
2188 size +=sizeof(WCHAR);
2189
2190 if (!check_first)
2191 {
2192 TRACE("Setting value %s of %s\n",debugstr_w(deformated),
2193 debugstr_w(uikey));
Mike McCormack16466af2005-07-06 10:33:30 +00002194 RegSetValueExW(hkey, deformated, 0, type, (LPBYTE)value_data, size);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002195 }
2196 else
2197 {
2198 DWORD sz = 0;
2199 rc = RegQueryValueExW(hkey, deformated, NULL, NULL, NULL, &sz);
2200 if (rc == ERROR_SUCCESS || rc == ERROR_MORE_DATA)
2201 {
2202 TRACE("value %s of %s checked already exists\n",
2203 debugstr_w(deformated), debugstr_w(uikey));
2204 }
2205 else
2206 {
2207 TRACE("Checked and setting value %s of %s\n",
2208 debugstr_w(deformated), debugstr_w(uikey));
2209 if (deformated || size)
Mike McCormack16466af2005-07-06 10:33:30 +00002210 RegSetValueExW(hkey, deformated, 0, type, (LPBYTE) value_data, size);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002211 }
2212 }
2213 RegCloseKey(hkey);
2214
2215 uirow = MSI_CreateRecord(3);
2216 MSI_RecordSetStringW(uirow,2,deformated);
2217 MSI_RecordSetStringW(uirow,1,uikey);
2218
2219 if (type == REG_SZ)
2220 MSI_RecordSetStringW(uirow,3,(LPWSTR)value_data);
2221 else
2222 MSI_RecordSetStringW(uirow,3,value);
2223
2224 ui_actiondata(package,szWriteRegistryValues,uirow);
2225 msiobj_release( &uirow->hdr );
2226
Mike McCormackee034ba2005-09-20 11:59:14 +00002227 msi_free(value_data);
2228 msi_free(deformated);
2229 msi_free(uikey);
Aric Stewart92ef78e2005-06-21 20:21:18 +00002230
2231 return ERROR_SUCCESS;
2232}
2233
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002234static UINT ACTION_WriteRegistryValues(MSIPACKAGE *package)
Aric Stewart401bd3f2004-06-28 20:34:35 +00002235{
2236 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002237 MSIQUERY * view;
Aric Stewart8e233e92005-03-01 11:45:19 +00002238 static const WCHAR ExecSeqQuery[] =
2239 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002240 '`','R','e','g','i','s','t','r','y','`',0 };
Aric Stewart401bd3f2004-06-28 20:34:35 +00002241
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002242 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
Aric Stewart401bd3f2004-06-28 20:34:35 +00002243 if (rc != ERROR_SUCCESS)
Aric Stewart84837d92004-07-20 01:22:37 +00002244 return ERROR_SUCCESS;
Aric Stewart401bd3f2004-06-28 20:34:35 +00002245
Aric Stewartd2c395a2004-07-06 18:48:15 +00002246 /* increment progress bar each time action data is sent */
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002247 ui_progress(package,1,REG_PROGRESS_VALUE,1,0);
Aric Stewartd2c395a2004-07-06 18:48:15 +00002248
Aric Stewart92ef78e2005-06-21 20:21:18 +00002249 rc = MSI_IterateRecords(view, NULL, ITERATE_WriteRegistryValues, package);
Aric Stewartd2c395a2004-07-06 18:48:15 +00002250
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002251 msiobj_release(&view->hdr);
Aric Stewart401bd3f2004-06-28 20:34:35 +00002252 return rc;
2253}
2254
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002255static UINT ACTION_InstallInitialize(MSIPACKAGE *package)
Aric Stewart7d3e5972004-07-04 00:36:58 +00002256{
Aric Stewart9cd707d2005-05-27 19:24:22 +00002257 package->script->CurrentlyScripting = TRUE;
2258
Aric Stewart7d3e5972004-07-04 00:36:58 +00002259 return ERROR_SUCCESS;
2260}
2261
Aric Stewartae1aa322004-12-27 19:02:59 +00002262
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002263static UINT ACTION_InstallValidate(MSIPACKAGE *package)
Aric Stewart7d3e5972004-07-04 00:36:58 +00002264{
Mike McCormack38d67a42005-08-22 09:15:23 +00002265 MSICOMPONENT *comp;
Aric Stewart7d3e5972004-07-04 00:36:58 +00002266 DWORD progress = 0;
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002267 DWORD total = 0;
Aric Stewart8e233e92005-03-01 11:45:19 +00002268 static const WCHAR q1[]=
2269 {'S','E','L','E','C','T',' ','*',' ', 'F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002270 '`','R','e','g','i','s','t','r','y','`',0};
Aric Stewart7d3e5972004-07-04 00:36:58 +00002271 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002272 MSIQUERY * view;
Mike McCormack1da28582005-08-22 14:09:17 +00002273 MSIFEATURE *feature;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002274 MSIFILE *file;
Aric Stewart7d3e5972004-07-04 00:36:58 +00002275
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002276 TRACE("InstallValidate\n");
Aric Stewart7d3e5972004-07-04 00:36:58 +00002277
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002278 rc = MSI_DatabaseOpenViewW(package->db, q1, &view);
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002279 if (rc == ERROR_SUCCESS)
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002280 {
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002281 MSI_IterateRecords( view, &progress, NULL, package );
2282 msiobj_release( &view->hdr );
2283 total += progress * REG_PROGRESS_VALUE;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002284 }
Aric Stewart7d3e5972004-07-04 00:36:58 +00002285
Mike McCormack38d67a42005-08-22 09:15:23 +00002286 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
Mike McCormack38d67a42005-08-22 09:15:23 +00002287 total += COMPONENT_PROGRESS_VALUE;
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002288
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002289 LIST_FOR_EACH_ENTRY( file, &package->files, MSIFILE, entry )
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002290 total += file->FileSize;
Mike McCormackf3f12ab2005-09-21 10:20:03 +00002291
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002292 ui_progress(package,0,total,0,0);
Aric Stewart7d3e5972004-07-04 00:36:58 +00002293
Mike McCormack1da28582005-08-22 14:09:17 +00002294 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002295 {
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002296 TRACE("Feature: %s; Installed: %i; Action %i; Request %i\n",
2297 debugstr_w(feature->Feature), feature->Installed, feature->Action,
2298 feature->ActionRequest);
2299 }
2300
Aric Stewart7d3e5972004-07-04 00:36:58 +00002301 return ERROR_SUCCESS;
2302}
2303
Aric Stewartc79f4e22005-06-22 18:03:08 +00002304static UINT ITERATE_LaunchConditions(MSIRECORD *row, LPVOID param)
2305{
2306 MSIPACKAGE* package = (MSIPACKAGE*)param;
2307 LPCWSTR cond = NULL;
2308 LPCWSTR message = NULL;
2309 static const WCHAR title[]=
2310 {'I','n','s','t','a','l','l',' ','F','a', 'i','l','e','d',0};
2311
2312 cond = MSI_RecordGetString(row,1);
2313
2314 if (MSI_EvaluateConditionW(package,cond) != MSICONDITION_TRUE)
2315 {
2316 LPWSTR deformated;
2317 message = MSI_RecordGetString(row,2);
2318 deformat_string(package,message,&deformated);
2319 MessageBoxW(NULL,deformated,title,MB_OK);
Mike McCormackee034ba2005-09-20 11:59:14 +00002320 msi_free(deformated);
Aric Stewartc79f4e22005-06-22 18:03:08 +00002321 return ERROR_FUNCTION_FAILED;
2322 }
2323
2324 return ERROR_SUCCESS;
2325}
2326
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002327static UINT ACTION_LaunchConditions(MSIPACKAGE *package)
Aric Stewart5b936ca2004-07-06 18:47:09 +00002328{
2329 UINT rc;
Mike McCormackf3c8b832004-07-19 19:35:05 +00002330 MSIQUERY * view = NULL;
Aric Stewart8e233e92005-03-01 11:45:19 +00002331 static const WCHAR ExecSeqQuery[] =
2332 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002333 '`','L','a','u','n','c','h','C','o','n','d','i','t','i','o','n','`',0};
Aric Stewart7d3e5972004-07-04 00:36:58 +00002334
Aric Stewart5b936ca2004-07-06 18:47:09 +00002335 TRACE("Checking launch conditions\n");
Aric Stewart7d3e5972004-07-04 00:36:58 +00002336
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002337 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
Aric Stewart5b936ca2004-07-06 18:47:09 +00002338 if (rc != ERROR_SUCCESS)
Aric Stewart84837d92004-07-20 01:22:37 +00002339 return ERROR_SUCCESS;
Aric Stewart5b936ca2004-07-06 18:47:09 +00002340
Aric Stewartc79f4e22005-06-22 18:03:08 +00002341 rc = MSI_IterateRecords(view, NULL, ITERATE_LaunchConditions, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002342 msiobj_release(&view->hdr);
Aric Stewartc79f4e22005-06-22 18:03:08 +00002343
Aric Stewart5b936ca2004-07-06 18:47:09 +00002344 return rc;
2345}
Aric Stewart7d3e5972004-07-04 00:36:58 +00002346
Mike McCormack38d67a42005-08-22 09:15:23 +00002347static LPWSTR resolve_keypath( MSIPACKAGE* package, MSICOMPONENT *cmp )
Aric Stewartb942e182004-07-06 18:50:02 +00002348{
Aric Stewartb942e182004-07-06 18:50:02 +00002349
Mike McCormackefcc1ec2005-09-12 12:07:15 +00002350 if (!cmp->KeyPath)
2351 return resolve_folder(package,cmp->Directory,FALSE,FALSE,NULL);
2352
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002353 if (cmp->Attributes & msidbComponentAttributesRegistryKeyPath)
Aric Stewartb942e182004-07-06 18:50:02 +00002354 {
Aric Stewart6269f002005-01-17 13:40:39 +00002355 MSIRECORD * row = 0;
Mike McCormack0b352c72005-06-02 10:29:57 +00002356 UINT root,len;
Aric Stewart09b0aba2005-06-09 20:30:59 +00002357 LPWSTR deformated,buffer,deformated_name;
2358 LPCWSTR key,name;
Aric Stewart8e233e92005-03-01 11:45:19 +00002359 static const WCHAR ExecSeqQuery[] =
2360 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002361 '`','R','e','g','i','s','t','r','y','`',' ',
2362 'W','H','E','R','E',' ', '`','R','e','g','i','s','t','r','y','`',
2363 ' ','=',' ' ,'\'','%','s','\'',0 };
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002364 static const WCHAR fmt[]={'%','0','2','i',':','\\','%','s','\\',0};
Aric Stewart8e233e92005-03-01 11:45:19 +00002365 static const WCHAR fmt2[]=
2366 {'%','0','2','i',':','\\','%','s','\\','%','s',0};
Aric Stewart6269f002005-01-17 13:40:39 +00002367
Mike McCormack0b352c72005-06-02 10:29:57 +00002368 row = MSI_QueryGetRecord(package->db, ExecSeqQuery,cmp->KeyPath);
2369 if (!row)
Aric Stewart6269f002005-01-17 13:40:39 +00002370 return NULL;
2371
Aric Stewart6269f002005-01-17 13:40:39 +00002372 root = MSI_RecordGetInteger(row,2);
Aric Stewart09b0aba2005-06-09 20:30:59 +00002373 key = MSI_RecordGetString(row, 3);
2374 name = MSI_RecordGetString(row, 4);
Aric Stewart6269f002005-01-17 13:40:39 +00002375 deformat_string(package, key , &deformated);
2376 deformat_string(package, name, &deformated_name);
2377
Ulrich Czekallae15e5172005-03-08 16:44:51 +00002378 len = strlenW(deformated) + 6;
Aric Stewart6269f002005-01-17 13:40:39 +00002379 if (deformated_name)
2380 len+=strlenW(deformated_name);
2381
Mike McCormackee034ba2005-09-20 11:59:14 +00002382 buffer = msi_alloc( len *sizeof(WCHAR));
Aric Stewart6269f002005-01-17 13:40:39 +00002383
2384 if (deformated_name)
2385 sprintfW(buffer,fmt2,root,deformated,deformated_name);
2386 else
2387 sprintfW(buffer,fmt,root,deformated);
2388
Mike McCormackee034ba2005-09-20 11:59:14 +00002389 msi_free(deformated);
2390 msi_free(deformated_name);
Aric Stewart6269f002005-01-17 13:40:39 +00002391 msiobj_release(&row->hdr);
Aric Stewart6269f002005-01-17 13:40:39 +00002392
2393 return buffer;
2394 }
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002395 else if (cmp->Attributes & msidbComponentAttributesODBCDataSource)
Aric Stewart6269f002005-01-17 13:40:39 +00002396 {
2397 FIXME("UNIMPLEMENTED keypath as ODBC Source\n");
Aric Stewartfa384f62004-12-22 18:46:17 +00002398 return NULL;
Aric Stewartb942e182004-07-06 18:50:02 +00002399 }
2400 else
2401 {
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002402 MSIFILE *file = get_loaded_file( package, cmp->KeyPath );
Aric Stewartfcb20c52004-07-06 18:51:16 +00002403
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002404 if (file)
2405 return strdupW( file->TargetPath );
Aric Stewartb942e182004-07-06 18:50:02 +00002406 }
Aric Stewartfa384f62004-12-22 18:46:17 +00002407 return NULL;
Aric Stewartb942e182004-07-06 18:50:02 +00002408}
2409
Stefan Huehnerac6f5622005-06-20 14:18:03 +00002410static HKEY openSharedDLLsKey(void)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002411{
2412 HKEY hkey=0;
Aric Stewart8e233e92005-03-01 11:45:19 +00002413 static const WCHAR path[] =
2414 {'S','o','f','t','w','a','r','e','\\',
2415 'M','i','c','r','o','s','o','f','t','\\',
2416 'W','i','n','d','o','w','s','\\',
2417 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
2418 'S','h','a','r','e','d','D','L','L','s',0};
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002419
2420 RegCreateKeyW(HKEY_LOCAL_MACHINE,path,&hkey);
2421 return hkey;
2422}
2423
2424static UINT ACTION_GetSharedDLLsCount(LPCWSTR dll)
2425{
2426 HKEY hkey;
2427 DWORD count=0;
2428 DWORD type;
2429 DWORD sz = sizeof(count);
2430 DWORD rc;
2431
2432 hkey = openSharedDLLsKey();
2433 rc = RegQueryValueExW(hkey, dll, NULL, &type, (LPBYTE)&count, &sz);
2434 if (rc != ERROR_SUCCESS)
2435 count = 0;
2436 RegCloseKey(hkey);
2437 return count;
2438}
2439
2440static UINT ACTION_WriteSharedDLLsCount(LPCWSTR path, UINT count)
2441{
2442 HKEY hkey;
2443
2444 hkey = openSharedDLLsKey();
2445 if (count > 0)
Mike McCormack4db02cd2005-09-15 14:58:38 +00002446 msi_reg_set_val_dword( hkey, path, count );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002447 else
2448 RegDeleteValueW(hkey,path);
2449 RegCloseKey(hkey);
2450 return count;
2451}
2452
2453/*
2454 * Return TRUE if the count should be written out and FALSE if not
2455 */
Mike McCormack38d67a42005-08-22 09:15:23 +00002456static void ACTION_RefCountComponent( MSIPACKAGE* package, MSICOMPONENT *comp )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002457{
Mike McCormack1da28582005-08-22 14:09:17 +00002458 MSIFEATURE *feature;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002459 INT count = 0;
2460 BOOL write = FALSE;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002461
2462 /* only refcount DLLs */
Mike McCormackefcc1ec2005-09-12 12:07:15 +00002463 if (comp->KeyPath == NULL ||
Mike McCormack38d67a42005-08-22 09:15:23 +00002464 comp->Attributes & msidbComponentAttributesRegistryKeyPath ||
2465 comp->Attributes & msidbComponentAttributesODBCDataSource)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002466 write = FALSE;
2467 else
2468 {
Mike McCormack38d67a42005-08-22 09:15:23 +00002469 count = ACTION_GetSharedDLLsCount( comp->FullKeypath);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002470 write = (count > 0);
2471
Mike McCormack38d67a42005-08-22 09:15:23 +00002472 if (comp->Attributes & msidbComponentAttributesSharedDllRefCount)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002473 write = TRUE;
2474 }
2475
2476 /* increment counts */
Mike McCormack1da28582005-08-22 14:09:17 +00002477 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002478 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00002479 ComponentList *cl;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002480
Mike McCormack1da28582005-08-22 14:09:17 +00002481 if (!ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_LOCAL ))
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002482 continue;
2483
Mike McCormack1da28582005-08-22 14:09:17 +00002484 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002485 {
Mike McCormack38d67a42005-08-22 09:15:23 +00002486 if ( cl->component == comp )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002487 count++;
2488 }
2489 }
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002490
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002491 /* decrement counts */
Mike McCormack1da28582005-08-22 14:09:17 +00002492 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002493 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00002494 ComponentList *cl;
2495
Mike McCormack1da28582005-08-22 14:09:17 +00002496 if (!ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_ABSENT ))
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002497 continue;
2498
Mike McCormack1da28582005-08-22 14:09:17 +00002499 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002500 {
Mike McCormack38d67a42005-08-22 09:15:23 +00002501 if ( cl->component == comp )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002502 count--;
2503 }
2504 }
2505
2506 /* ref count all the files in the component */
2507 if (write)
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002508 {
2509 MSIFILE *file;
2510
2511 LIST_FOR_EACH_ENTRY( file, &package->files, MSIFILE, entry )
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002512 {
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002513 if (file->Component == comp)
2514 ACTION_WriteSharedDLLsCount( file->TargetPath, count );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002515 }
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002516 }
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002517
2518 /* add a count for permenent */
Mike McCormack38d67a42005-08-22 09:15:23 +00002519 if (comp->Attributes & msidbComponentAttributesPermanent)
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002520 count ++;
2521
Mike McCormack38d67a42005-08-22 09:15:23 +00002522 comp->RefCount = count;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002523
2524 if (write)
Mike McCormack38d67a42005-08-22 09:15:23 +00002525 ACTION_WriteSharedDLLsCount( comp->FullKeypath, comp->RefCount );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002526}
2527
Aric Stewart2cf222f2004-07-06 19:00:23 +00002528/*
2529 * Ok further analysis makes me think that this work is
2530 * actually done in the PublishComponents and PublishFeatures
Mike McCormack3ece2462004-07-09 19:33:25 +00002531 * step, and not here. It appears like the keypath and all that is
2532 * resolved in this step, however actually written in the Publish steps.
Alexandre Julliard77b12762004-07-09 19:43:29 +00002533 * But we will leave it here for now because it is unclear
Aric Stewart2cf222f2004-07-06 19:00:23 +00002534 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002535static UINT ACTION_ProcessComponents(MSIPACKAGE *package)
Aric Stewartb942e182004-07-06 18:50:02 +00002536{
Aric Stewart68b07492005-01-25 11:05:37 +00002537 WCHAR squished_pc[GUID_SIZE];
2538 WCHAR squished_cc[GUID_SIZE];
Aric Stewartb942e182004-07-06 18:50:02 +00002539 UINT rc;
Mike McCormack38d67a42005-08-22 09:15:23 +00002540 MSICOMPONENT *comp;
Aric Stewart68b07492005-01-25 11:05:37 +00002541 HKEY hkey=0,hkey2=0;
Aric Stewartb942e182004-07-06 18:50:02 +00002542
Aric Stewartb942e182004-07-06 18:50:02 +00002543 /* writes the Component and Features values to the registry */
Aric Stewartb942e182004-07-06 18:50:02 +00002544
Aric Stewart68b07492005-01-25 11:05:37 +00002545 rc = MSIREG_OpenComponents(&hkey);
Aric Stewartb942e182004-07-06 18:50:02 +00002546 if (rc != ERROR_SUCCESS)
Mike McCormackfe8cd382006-03-09 14:21:37 +09002547 return rc;
2548
Aric Stewartadaef112005-07-07 20:27:06 +00002549 squash_guid(package->ProductCode,squished_pc);
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002550 ui_progress(package,1,COMPONENT_PROGRESS_VALUE,1,0);
Mike McCormack38d67a42005-08-22 09:15:23 +00002551
2552 LIST_FOR_EACH_ENTRY( comp, &package->components, MSICOMPONENT, entry )
Aric Stewartb942e182004-07-06 18:50:02 +00002553 {
Mike McCormackfe8cd382006-03-09 14:21:37 +09002554 MSIRECORD * uirow;
2555
Aric Stewartbd1bbc12005-01-03 20:00:13 +00002556 ui_progress(package,2,0,0,0);
Mike McCormackfe8cd382006-03-09 14:21:37 +09002557 if (!comp->ComponentId)
2558 continue;
Aric Stewartb942e182004-07-06 18:50:02 +00002559
Mike McCormackfe8cd382006-03-09 14:21:37 +09002560 squash_guid(comp->ComponentId,squished_cc);
Aric Stewartb942e182004-07-06 18:50:02 +00002561
Mike McCormackfe8cd382006-03-09 14:21:37 +09002562 msi_free(comp->FullKeypath);
2563 comp->FullKeypath = resolve_keypath( package, comp );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002564
Mike McCormackfe8cd382006-03-09 14:21:37 +09002565 /* do the refcounting */
2566 ACTION_RefCountComponent( package, comp );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002567
Mike McCormackfe8cd382006-03-09 14:21:37 +09002568 TRACE("Component %s (%s), Keypath=%s, RefCount=%i\n",
Mike McCormack38d67a42005-08-22 09:15:23 +00002569 debugstr_w(comp->Component),
Aric Stewartc5a14432005-05-18 17:46:12 +00002570 debugstr_w(squished_cc),
Mike McCormackfe8cd382006-03-09 14:21:37 +09002571 debugstr_w(comp->FullKeypath),
Mike McCormack38d67a42005-08-22 09:15:23 +00002572 comp->RefCount);
Mike McCormackfe8cd382006-03-09 14:21:37 +09002573 /*
2574 * Write the keypath out if the component is to be registered
2575 * and delete the key if the component is to be deregistered
2576 */
2577 if (ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
2578 {
2579 rc = RegCreateKeyW(hkey,squished_cc,&hkey2);
2580 if (rc != ERROR_SUCCESS)
2581 continue;
2582
2583 if (!comp->FullKeypath)
2584 continue;
2585
2586 msi_reg_set_val_str( hkey2, squished_pc, comp->FullKeypath );
2587
2588 if (comp->Attributes & msidbComponentAttributesPermanent)
Aric Stewartfa384f62004-12-22 18:46:17 +00002589 {
Mike McCormackfe8cd382006-03-09 14:21:37 +09002590 static const WCHAR szPermKey[] =
2591 { '0','0','0','0','0','0','0','0','0','0','0','0',
2592 '0','0','0','0','0','0','0','0','0','0','0','0',
2593 '0','0','0','0','0','0','0','0',0 };
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002594
Mike McCormackfe8cd382006-03-09 14:21:37 +09002595 msi_reg_set_val_str( hkey2, szPermKey, comp->FullKeypath );
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002596 }
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002597
Mike McCormackfe8cd382006-03-09 14:21:37 +09002598 RegCloseKey(hkey2);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00002599
Mike McCormackfe8cd382006-03-09 14:21:37 +09002600 /* UI stuff */
2601 uirow = MSI_CreateRecord(3);
2602 MSI_RecordSetStringW(uirow,1,package->ProductCode);
2603 MSI_RecordSetStringW(uirow,2,comp->ComponentId);
2604 MSI_RecordSetStringW(uirow,3,comp->FullKeypath);
2605 ui_actiondata(package,szProcessComponents,uirow);
2606 msiobj_release( &uirow->hdr );
2607 }
2608 else if (ACTION_VerifyComponentForAction( comp, INSTALLSTATE_ABSENT))
2609 {
2610 DWORD res;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00002611
Mike McCormackfe8cd382006-03-09 14:21:37 +09002612 rc = RegOpenKeyW(hkey,squished_cc,&hkey2);
2613 if (rc != ERROR_SUCCESS)
2614 continue;
2615
2616 RegDeleteValueW(hkey2,squished_pc);
2617
2618 /* if the key is empty delete it */
2619 res = RegEnumKeyExW(hkey2,0,NULL,0,0,NULL,0,NULL);
2620 RegCloseKey(hkey2);
2621 if (res == ERROR_NO_MORE_ITEMS)
2622 RegDeleteKeyW(hkey,squished_cc);
2623
2624 /* UI stuff */
2625 uirow = MSI_CreateRecord(2);
2626 MSI_RecordSetStringW(uirow,1,package->ProductCode);
2627 MSI_RecordSetStringW(uirow,2,comp->ComponentId);
2628 ui_actiondata(package,szProcessComponents,uirow);
2629 msiobj_release( &uirow->hdr );
Aric Stewartb942e182004-07-06 18:50:02 +00002630 }
2631 }
Aric Stewartb942e182004-07-06 18:50:02 +00002632 RegCloseKey(hkey);
2633 return rc;
2634}
2635
Aric Stewart6e821732005-03-30 10:19:08 +00002636typedef struct {
2637 CLSID clsid;
2638 LPWSTR source;
2639
2640 LPWSTR path;
2641 ITypeLib *ptLib;
2642} typelib_struct;
2643
Mike McCormackf9acfe62005-06-07 20:29:51 +00002644static BOOL CALLBACK Typelib_EnumResNameProc( HMODULE hModule, LPCWSTR lpszType,
Aric Stewart6e821732005-03-30 10:19:08 +00002645 LPWSTR lpszName, LONG_PTR lParam)
2646{
2647 TLIBATTR *attr;
2648 typelib_struct *tl_struct = (typelib_struct*) lParam;
2649 static const WCHAR fmt[] = {'%','s','\\','%','i',0};
2650 int sz;
2651 HRESULT res;
2652
2653 if (!IS_INTRESOURCE(lpszName))
2654 {
2655 ERR("Not Int Resource Name %s\n",debugstr_w(lpszName));
2656 return TRUE;
2657 }
2658
2659 sz = strlenW(tl_struct->source)+4;
2660 sz *= sizeof(WCHAR);
2661
Mike McCormack2acf8002006-05-25 11:41:39 +09002662 if ((INT_PTR)lpszName == 1)
Aric Stewartca8c4e42005-06-02 15:13:57 +00002663 tl_struct->path = strdupW(tl_struct->source);
2664 else
2665 {
Mike McCormackee034ba2005-09-20 11:59:14 +00002666 tl_struct->path = msi_alloc(sz);
Aric Stewartca8c4e42005-06-02 15:13:57 +00002667 sprintfW(tl_struct->path,fmt,tl_struct->source, lpszName);
2668 }
Aric Stewart6e821732005-03-30 10:19:08 +00002669
2670 TRACE("trying %s\n", debugstr_w(tl_struct->path));
2671 res = LoadTypeLib(tl_struct->path,&tl_struct->ptLib);
2672 if (!SUCCEEDED(res))
2673 {
Mike McCormackee034ba2005-09-20 11:59:14 +00002674 msi_free(tl_struct->path);
Aric Stewart6e821732005-03-30 10:19:08 +00002675 tl_struct->path = NULL;
2676
2677 return TRUE;
2678 }
2679
2680 ITypeLib_GetLibAttr(tl_struct->ptLib, &attr);
2681 if (IsEqualGUID(&(tl_struct->clsid),&(attr->guid)))
2682 {
2683 ITypeLib_ReleaseTLibAttr(tl_struct->ptLib, attr);
2684 return FALSE;
2685 }
2686
Mike McCormackee034ba2005-09-20 11:59:14 +00002687 msi_free(tl_struct->path);
Aric Stewart6e821732005-03-30 10:19:08 +00002688 tl_struct->path = NULL;
2689
2690 ITypeLib_ReleaseTLibAttr(tl_struct->ptLib, attr);
2691 ITypeLib_Release(tl_struct->ptLib);
2692
2693 return TRUE;
2694}
2695
Aric Stewart234dc4b2005-06-22 18:27:34 +00002696static UINT ITERATE_RegisterTypeLibraries(MSIRECORD *row, LPVOID param)
2697{
2698 MSIPACKAGE* package = (MSIPACKAGE*)param;
2699 LPCWSTR component;
Mike McCormack38d67a42005-08-22 09:15:23 +00002700 MSICOMPONENT *comp;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002701 MSIFILE *file;
Aric Stewart234dc4b2005-06-22 18:27:34 +00002702 typelib_struct tl_struct;
2703 HMODULE module;
2704 static const WCHAR szTYPELIB[] = {'T','Y','P','E','L','I','B',0};
2705
2706 component = MSI_RecordGetString(row,3);
Mike McCormack38d67a42005-08-22 09:15:23 +00002707 comp = get_loaded_component(package,component);
2708 if (!comp)
Aric Stewart234dc4b2005-06-22 18:27:34 +00002709 return ERROR_SUCCESS;
2710
Mike McCormackd693f462005-10-29 11:36:48 +00002711 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
Aric Stewart234dc4b2005-06-22 18:27:34 +00002712 {
2713 TRACE("Skipping typelib reg due to disabled component\n");
2714
Mike McCormack38d67a42005-08-22 09:15:23 +00002715 comp->Action = comp->Installed;
Aric Stewart234dc4b2005-06-22 18:27:34 +00002716
2717 return ERROR_SUCCESS;
2718 }
2719
Mike McCormack38d67a42005-08-22 09:15:23 +00002720 comp->Action = INSTALLSTATE_LOCAL;
Aric Stewart234dc4b2005-06-22 18:27:34 +00002721
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002722 file = get_loaded_file( package, comp->KeyPath );
2723 if (!file)
Aric Stewart234dc4b2005-06-22 18:27:34 +00002724 return ERROR_SUCCESS;
2725
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002726 module = LoadLibraryExW( file->TargetPath, NULL, LOAD_LIBRARY_AS_DATAFILE );
Mike McCormack51c66182005-10-27 12:36:12 +00002727 if (module)
Aric Stewart234dc4b2005-06-22 18:27:34 +00002728 {
Mike McCormack51c66182005-10-27 12:36:12 +00002729 LPCWSTR guid;
2730 guid = MSI_RecordGetString(row,1);
2731 CLSIDFromString((LPWSTR)guid, &tl_struct.clsid);
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002732 tl_struct.source = strdupW( file->TargetPath );
Aric Stewart234dc4b2005-06-22 18:27:34 +00002733 tl_struct.path = NULL;
2734
2735 EnumResourceNamesW(module, szTYPELIB, Typelib_EnumResNameProc,
2736 (LONG_PTR)&tl_struct);
2737
Mike McCormack51c66182005-10-27 12:36:12 +00002738 if (tl_struct.path)
Aric Stewart234dc4b2005-06-22 18:27:34 +00002739 {
2740 LPWSTR help = NULL;
2741 LPCWSTR helpid;
2742 HRESULT res;
2743
2744 helpid = MSI_RecordGetString(row,6);
2745
2746 if (helpid)
2747 help = resolve_folder(package,helpid,FALSE,FALSE,NULL);
2748 res = RegisterTypeLib(tl_struct.ptLib,tl_struct.path,help);
Mike McCormackee034ba2005-09-20 11:59:14 +00002749 msi_free(help);
Aric Stewart234dc4b2005-06-22 18:27:34 +00002750
2751 if (!SUCCEEDED(res))
2752 ERR("Failed to register type library %s\n",
2753 debugstr_w(tl_struct.path));
2754 else
2755 {
2756 ui_actiondata(package,szRegisterTypeLibraries,row);
2757
2758 TRACE("Registered %s\n", debugstr_w(tl_struct.path));
2759 }
2760
2761 ITypeLib_Release(tl_struct.ptLib);
Mike McCormackee034ba2005-09-20 11:59:14 +00002762 msi_free(tl_struct.path);
Aric Stewart234dc4b2005-06-22 18:27:34 +00002763 }
2764 else
2765 ERR("Failed to load type library %s\n",
2766 debugstr_w(tl_struct.source));
2767
2768 FreeLibrary(module);
Mike McCormackee034ba2005-09-20 11:59:14 +00002769 msi_free(tl_struct.source);
Aric Stewart234dc4b2005-06-22 18:27:34 +00002770 }
2771 else
Mike McCormacke18f8ab2005-08-23 10:03:17 +00002772 ERR("Could not load file! %s\n", debugstr_w(file->TargetPath));
Aric Stewart234dc4b2005-06-22 18:27:34 +00002773
2774 return ERROR_SUCCESS;
2775}
2776
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002777static UINT ACTION_RegisterTypeLibraries(MSIPACKAGE *package)
Aric Stewartfcb20c52004-07-06 18:51:16 +00002778{
2779 /*
Mike McCormackc90c7812004-07-09 22:58:27 +00002780 * OK this is a bit confusing.. I am given a _Component key and I believe
Aric Stewartfcb20c52004-07-06 18:51:16 +00002781 * that the file that is being registered as a type library is the "key file
Mike McCormackc90c7812004-07-09 22:58:27 +00002782 * of that component" which I interpret to mean "The file in the KeyPath of
2783 * that component".
Aric Stewartfcb20c52004-07-06 18:51:16 +00002784 */
2785 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002786 MSIQUERY * view;
Aric Stewart8e233e92005-03-01 11:45:19 +00002787 static const WCHAR Query[] =
2788 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00002789 '`','T','y','p','e','L','i','b','`',0};
Aric Stewartfcb20c52004-07-06 18:51:16 +00002790
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002791 rc = MSI_DatabaseOpenViewW(package->db, Query, &view);
Aric Stewartfcb20c52004-07-06 18:51:16 +00002792 if (rc != ERROR_SUCCESS)
Aric Stewart84837d92004-07-20 01:22:37 +00002793 return ERROR_SUCCESS;
Aric Stewartfcb20c52004-07-06 18:51:16 +00002794
Aric Stewart234dc4b2005-06-22 18:27:34 +00002795 rc = MSI_IterateRecords(view, NULL, ITERATE_RegisterTypeLibraries, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002796 msiobj_release(&view->hdr);
Aric Stewartfcb20c52004-07-06 18:51:16 +00002797 return rc;
Aric Stewartfcb20c52004-07-06 18:51:16 +00002798}
2799
Aric Stewart9adacf62005-06-24 11:58:21 +00002800static UINT ITERATE_CreateShortcuts(MSIRECORD *row, LPVOID param)
Aric Stewart2cf222f2004-07-06 19:00:23 +00002801{
Aric Stewart9adacf62005-06-24 11:58:21 +00002802 MSIPACKAGE *package = (MSIPACKAGE*)param;
Mike McCormack477bce32006-01-16 20:38:28 +01002803 LPWSTR target_file, target_folder, filename;
Robert Shearman4ac85672006-02-22 16:31:00 +00002804 LPCWSTR buffer, extension;
Mike McCormack38d67a42005-08-22 09:15:23 +00002805 MSICOMPONENT *comp;
Aric Stewart9adacf62005-06-24 11:58:21 +00002806 static const WCHAR szlnk[]={'.','l','n','k',0};
Mike McCormack20c57462006-05-24 17:41:04 +09002807 IShellLinkW *sl = NULL;
2808 IPersistFile *pf = NULL;
Aric Stewart2cf222f2004-07-06 19:00:23 +00002809 HRESULT res;
2810
Aric Stewart9adacf62005-06-24 11:58:21 +00002811 buffer = MSI_RecordGetString(row,4);
Mike McCormack38d67a42005-08-22 09:15:23 +00002812 comp = get_loaded_component(package,buffer);
2813 if (!comp)
Aric Stewart9adacf62005-06-24 11:58:21 +00002814 return ERROR_SUCCESS;
2815
Mike McCormackd693f462005-10-29 11:36:48 +00002816 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL ))
Aric Stewart9adacf62005-06-24 11:58:21 +00002817 {
2818 TRACE("Skipping shortcut creation due to disabled component\n");
2819
Mike McCormack38d67a42005-08-22 09:15:23 +00002820 comp->Action = comp->Installed;
Aric Stewart9adacf62005-06-24 11:58:21 +00002821
2822 return ERROR_SUCCESS;
2823 }
2824
Mike McCormack38d67a42005-08-22 09:15:23 +00002825 comp->Action = INSTALLSTATE_LOCAL;
Aric Stewart9adacf62005-06-24 11:58:21 +00002826
2827 ui_actiondata(package,szCreateShortcuts,row);
2828
2829 res = CoCreateInstance( &CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER,
2830 &IID_IShellLinkW, (LPVOID *) &sl );
2831
Mike McCormack20c57462006-05-24 17:41:04 +09002832 if (FAILED( res ))
Aric Stewart9adacf62005-06-24 11:58:21 +00002833 {
Mike McCormack20c57462006-05-24 17:41:04 +09002834 ERR("CLSID_ShellLink not available\n");
2835 goto err;
Aric Stewart9adacf62005-06-24 11:58:21 +00002836 }
2837
2838 res = IShellLinkW_QueryInterface( sl, &IID_IPersistFile,(LPVOID*) &pf );
Mike McCormack20c57462006-05-24 17:41:04 +09002839 if (FAILED( res ))
Aric Stewart9adacf62005-06-24 11:58:21 +00002840 {
Mike McCormack20c57462006-05-24 17:41:04 +09002841 ERR("QueryInterface(IID_IPersistFile) failed\n");
2842 goto err;
Aric Stewart9adacf62005-06-24 11:58:21 +00002843 }
2844
2845 buffer = MSI_RecordGetString(row,2);
2846 target_folder = resolve_folder(package, buffer,FALSE,FALSE,NULL);
2847
2848 /* may be needed because of a bug somehwere else */
2849 create_full_pathW(target_folder);
2850
Mike McCormack477bce32006-01-16 20:38:28 +01002851 filename = msi_dup_record_field( row, 3 );
Aric Stewart9adacf62005-06-24 11:58:21 +00002852 reduce_to_longfilename(filename);
Robert Shearman4ac85672006-02-22 16:31:00 +00002853
2854 extension = strchrW(filename,'.');
2855 if (!extension || strcmpiW(extension,szlnk))
2856 {
2857 int len = strlenW(filename);
2858 filename = msi_realloc(filename, len * sizeof(WCHAR) + sizeof(szlnk));
2859 memcpy(filename + len, szlnk, sizeof(szlnk));
2860 }
Aric Stewart9adacf62005-06-24 11:58:21 +00002861 target_file = build_directory_name(2, target_folder, filename);
Mike McCormackee034ba2005-09-20 11:59:14 +00002862 msi_free(target_folder);
Mike McCormack477bce32006-01-16 20:38:28 +01002863 msi_free(filename);
Aric Stewart9adacf62005-06-24 11:58:21 +00002864
2865 buffer = MSI_RecordGetString(row,5);
2866 if (strchrW(buffer,'['))
2867 {
2868 LPWSTR deformated;
2869 deformat_string(package,buffer,&deformated);
2870 IShellLinkW_SetPath(sl,deformated);
Mike McCormackee034ba2005-09-20 11:59:14 +00002871 msi_free(deformated);
Aric Stewart9adacf62005-06-24 11:58:21 +00002872 }
2873 else
2874 {
Aric Stewart9adacf62005-06-24 11:58:21 +00002875 FIXME("poorly handled shortcut format, advertised shortcut\n");
Mike McCormack566c69e2005-09-22 10:49:17 +00002876 IShellLinkW_SetPath(sl,comp->FullKeypath);
Aric Stewart9adacf62005-06-24 11:58:21 +00002877 }
2878
2879 if (!MSI_RecordIsNull(row,6))
2880 {
2881 LPWSTR deformated;
2882 buffer = MSI_RecordGetString(row,6);
2883 deformat_string(package,buffer,&deformated);
2884 IShellLinkW_SetArguments(sl,deformated);
Mike McCormackee034ba2005-09-20 11:59:14 +00002885 msi_free(deformated);
Aric Stewart9adacf62005-06-24 11:58:21 +00002886 }
2887
2888 if (!MSI_RecordIsNull(row,7))
2889 {
2890 buffer = MSI_RecordGetString(row,7);
2891 IShellLinkW_SetDescription(sl,buffer);
2892 }
2893
2894 if (!MSI_RecordIsNull(row,8))
2895 IShellLinkW_SetHotkey(sl,MSI_RecordGetInteger(row,8));
2896
2897 if (!MSI_RecordIsNull(row,9))
2898 {
Mike McCormack75658d72005-09-22 10:33:57 +00002899 LPWSTR Path;
Aric Stewart9adacf62005-06-24 11:58:21 +00002900 INT index;
2901
2902 buffer = MSI_RecordGetString(row,9);
2903
Mike McCormack75658d72005-09-22 10:33:57 +00002904 Path = build_icon_path(package,buffer);
Aric Stewart9adacf62005-06-24 11:58:21 +00002905 index = MSI_RecordGetInteger(row,10);
2906
2907 IShellLinkW_SetIconLocation(sl,Path,index);
Mike McCormackee034ba2005-09-20 11:59:14 +00002908 msi_free(Path);
Aric Stewart9adacf62005-06-24 11:58:21 +00002909 }
2910
2911 if (!MSI_RecordIsNull(row,11))
2912 IShellLinkW_SetShowCmd(sl,MSI_RecordGetInteger(row,11));
2913
2914 if (!MSI_RecordIsNull(row,12))
2915 {
2916 LPWSTR Path;
2917 buffer = MSI_RecordGetString(row,12);
2918 Path = resolve_folder(package, buffer, FALSE, FALSE, NULL);
2919 IShellLinkW_SetWorkingDirectory(sl,Path);
Mike McCormackee034ba2005-09-20 11:59:14 +00002920 msi_free(Path);
Aric Stewart9adacf62005-06-24 11:58:21 +00002921 }
2922
2923 TRACE("Writing shortcut to %s\n",debugstr_w(target_file));
2924 IPersistFile_Save(pf,target_file,FALSE);
2925
Mike McCormackee034ba2005-09-20 11:59:14 +00002926 msi_free(target_file);
Aric Stewart9adacf62005-06-24 11:58:21 +00002927
Mike McCormack20c57462006-05-24 17:41:04 +09002928err:
2929 if (pf)
2930 IPersistFile_Release( pf );
2931 if (sl)
2932 IShellLinkW_Release( sl );
Aric Stewart9adacf62005-06-24 11:58:21 +00002933
2934 return ERROR_SUCCESS;
2935}
2936
2937static UINT ACTION_CreateShortcuts(MSIPACKAGE *package)
2938{
2939 UINT rc;
2940 HRESULT res;
2941 MSIQUERY * view;
2942 static const WCHAR Query[] =
2943 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
2944 '`','S','h','o','r','t','c','u','t','`',0};
2945
Aric Stewart9adacf62005-06-24 11:58:21 +00002946 rc = MSI_DatabaseOpenViewW(package->db, Query, &view);
2947 if (rc != ERROR_SUCCESS)
2948 return ERROR_SUCCESS;
2949
Aric Stewart2cf222f2004-07-06 19:00:23 +00002950 res = CoInitialize( NULL );
2951 if (FAILED (res))
2952 {
2953 ERR("CoInitialize failed\n");
2954 return ERROR_FUNCTION_FAILED;
2955 }
2956
Aric Stewart9adacf62005-06-24 11:58:21 +00002957 rc = MSI_IterateRecords(view, NULL, ITERATE_CreateShortcuts, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00002958 msiobj_release(&view->hdr);
Aric Stewart2cf222f2004-07-06 19:00:23 +00002959
Aric Stewart2cf222f2004-07-06 19:00:23 +00002960 CoUninitialize();
2961
2962 return rc;
2963}
2964
Aric Stewart916ef942005-06-22 18:42:19 +00002965static UINT ITERATE_PublishProduct(MSIRECORD *row, LPVOID param)
2966{
2967 MSIPACKAGE* package = (MSIPACKAGE*)param;
2968 HANDLE the_file;
Mike McCormack75658d72005-09-22 10:33:57 +00002969 LPWSTR FilePath;
2970 LPCWSTR FileName;
Aric Stewart916ef942005-06-22 18:42:19 +00002971 CHAR buffer[1024];
2972 DWORD sz;
2973 UINT rc;
Robert Shearmand2e48e02006-01-23 17:29:50 +01002974 MSIRECORD *uirow;
Aric Stewart916ef942005-06-22 18:42:19 +00002975
2976 FileName = MSI_RecordGetString(row,1);
2977 if (!FileName)
2978 {
2979 ERR("Unable to get FileName\n");
2980 return ERROR_SUCCESS;
2981 }
2982
Mike McCormack75658d72005-09-22 10:33:57 +00002983 FilePath = build_icon_path(package,FileName);
Aric Stewart916ef942005-06-22 18:42:19 +00002984
2985 TRACE("Creating icon file at %s\n",debugstr_w(FilePath));
2986
2987 the_file = CreateFileW(FilePath, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS,
2988 FILE_ATTRIBUTE_NORMAL, NULL);
2989
2990 if (the_file == INVALID_HANDLE_VALUE)
2991 {
2992 ERR("Unable to create file %s\n",debugstr_w(FilePath));
Mike McCormackee034ba2005-09-20 11:59:14 +00002993 msi_free(FilePath);
Aric Stewart916ef942005-06-22 18:42:19 +00002994 return ERROR_SUCCESS;
2995 }
2996
2997 do
2998 {
2999 DWORD write;
3000 sz = 1024;
3001 rc = MSI_RecordReadStream(row,2,buffer,&sz);
3002 if (rc != ERROR_SUCCESS)
3003 {
3004 ERR("Failed to get stream\n");
3005 CloseHandle(the_file);
3006 DeleteFileW(FilePath);
3007 break;
3008 }
3009 WriteFile(the_file,buffer,sz,&write,NULL);
3010 } while (sz == 1024);
3011
Mike McCormackee034ba2005-09-20 11:59:14 +00003012 msi_free(FilePath);
Aric Stewart916ef942005-06-22 18:42:19 +00003013
3014 CloseHandle(the_file);
Robert Shearmand2e48e02006-01-23 17:29:50 +01003015
3016 uirow = MSI_CreateRecord(1);
3017 MSI_RecordSetStringW(uirow,1,FileName);
3018 ui_actiondata(package,szPublishProduct,uirow);
3019 msiobj_release( &uirow->hdr );
3020
Aric Stewart916ef942005-06-22 18:42:19 +00003021 return ERROR_SUCCESS;
3022}
Aric Stewart2cf222f2004-07-06 19:00:23 +00003023
3024/*
3025 * 99% of the work done here is only done for
3026 * advertised installs. However this is where the
3027 * Icon table is processed and written out
Francois Gouget817c5202004-07-16 19:15:40 +00003028 * so that is what I am going to do here.
Aric Stewart2cf222f2004-07-06 19:00:23 +00003029 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00003030static UINT ACTION_PublishProduct(MSIPACKAGE *package)
Aric Stewart2cf222f2004-07-06 19:00:23 +00003031{
3032 UINT rc;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00003033 MSIQUERY * view;
Aric Stewart8e233e92005-03-01 11:45:19 +00003034 static const WCHAR Query[]=
3035 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00003036 '`','I','c','o','n','`',0};
Aric Stewart6269f002005-01-17 13:40:39 +00003037 /* for registry stuff */
Aric Stewart68b07492005-01-25 11:05:37 +00003038 HKEY hkey=0;
3039 HKEY hukey=0;
Aric Stewart6957e4a2005-06-08 19:16:45 +00003040 static const WCHAR szProductLanguage[] =
3041 {'P','r','o','d','u','c','t','L','a','n','g','u','a','g','e',0};
Aric Stewart6957e4a2005-06-08 19:16:45 +00003042 static const WCHAR szARPProductIcon[] =
3043 {'A','R','P','P','R','O','D','U','C','T','I','C','O','N',0};
Aric Stewartc28bb542005-06-09 15:49:11 +00003044 static const WCHAR szProductVersion[] =
3045 {'P','r','o','d','u','c','t','V','e','r','s','i','o','n',0};
Aric Stewart6957e4a2005-06-08 19:16:45 +00003046 DWORD langid;
Aric Stewart6269f002005-01-17 13:40:39 +00003047 LPWSTR buffer;
3048 DWORD size;
Aric Stewart68b07492005-01-25 11:05:37 +00003049 MSIHANDLE hDb, hSumInfo;
Aric Stewart2cf222f2004-07-06 19:00:23 +00003050
Aric Stewart916ef942005-06-22 18:42:19 +00003051 /* write out icon files */
3052
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00003053 rc = MSI_DatabaseOpenViewW(package->db, Query, &view);
Aric Stewart916ef942005-06-22 18:42:19 +00003054 if (rc == ERROR_SUCCESS)
Aric Stewart2cf222f2004-07-06 19:00:23 +00003055 {
Aric Stewart916ef942005-06-22 18:42:19 +00003056 MSI_IterateRecords(view, NULL, ITERATE_PublishProduct, package);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00003057 msiobj_release(&view->hdr);
Aric Stewart2cf222f2004-07-06 19:00:23 +00003058 }
3059
Francois Gougetda8b3dd2005-01-26 21:09:04 +00003060 /* ok there is a lot more done here but i need to figure out what */
Aric Stewart916ef942005-06-22 18:42:19 +00003061
Aric Stewartadaef112005-07-07 20:27:06 +00003062 rc = MSIREG_OpenProductsKey(package->ProductCode,&hkey,TRUE);
Aric Stewart6269f002005-01-17 13:40:39 +00003063 if (rc != ERROR_SUCCESS)
3064 goto end;
3065
Aric Stewartadaef112005-07-07 20:27:06 +00003066 rc = MSIREG_OpenUserProductsKey(package->ProductCode,&hukey,TRUE);
Aric Stewart6269f002005-01-17 13:40:39 +00003067 if (rc != ERROR_SUCCESS)
3068 goto end;
3069
Aric Stewart6269f002005-01-17 13:40:39 +00003070
Mike McCormack062ad502005-09-15 15:04:08 +00003071 buffer = msi_dup_property( package, INSTALLPROPERTY_PRODUCTNAMEW );
Mike McCormack4db02cd2005-09-15 14:58:38 +00003072 msi_reg_set_val_str( hukey, INSTALLPROPERTY_PRODUCTNAMEW, buffer );
Mike McCormackee034ba2005-09-20 11:59:14 +00003073 msi_free(buffer);
Aric Stewart6957e4a2005-06-08 19:16:45 +00003074
Mike McCormack74f0de92005-09-29 10:32:39 +00003075 langid = msi_get_property_int( package, szProductLanguage, 0 );
Mike McCormack4db02cd2005-09-15 14:58:38 +00003076 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_LANGUAGEW, langid );
Aric Stewart6957e4a2005-06-08 19:16:45 +00003077
Mike McCormack062ad502005-09-15 15:04:08 +00003078 buffer = msi_dup_property( package, szARPProductIcon );
Aric Stewart6957e4a2005-06-08 19:16:45 +00003079 if (buffer)
3080 {
Mike McCormack75658d72005-09-22 10:33:57 +00003081 LPWSTR path = build_icon_path(package,buffer);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003082 msi_reg_set_val_str( hukey, INSTALLPROPERTY_PRODUCTICONW, path );
Mike McCormack75658d72005-09-22 10:33:57 +00003083 msi_free( path );
Aric Stewart6957e4a2005-06-08 19:16:45 +00003084 }
Mike McCormackee034ba2005-09-20 11:59:14 +00003085 msi_free(buffer);
Aric Stewartc28bb542005-06-09 15:49:11 +00003086
Mike McCormack062ad502005-09-15 15:04:08 +00003087 buffer = msi_dup_property( package, szProductVersion );
Aric Stewartc28bb542005-06-09 15:49:11 +00003088 if (buffer)
3089 {
Mike McCormack230af9d2006-07-14 15:19:08 +09003090 DWORD verdword = msi_version_str_to_dword(buffer);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003091 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_VERSIONW, verdword );
Aric Stewartc28bb542005-06-09 15:49:11 +00003092 }
Mike McCormackee034ba2005-09-20 11:59:14 +00003093 msi_free(buffer);
Aric Stewart6957e4a2005-06-08 19:16:45 +00003094
Mike McCormackb7270b82005-12-31 13:18:11 +01003095 /* FIXME: Need to write more keys to the user registry */
Aric Stewart68b07492005-01-25 11:05:37 +00003096
Aric Stewart7e7b8cf2005-02-18 20:00:34 +00003097 hDb= alloc_msihandle( &package->db->hdr );
Aric Stewart68b07492005-01-25 11:05:37 +00003098 rc = MsiGetSummaryInformationW(hDb, NULL, 0, &hSumInfo);
Aric Stewart7e7b8cf2005-02-18 20:00:34 +00003099 MsiCloseHandle(hDb);
Aric Stewart68b07492005-01-25 11:05:37 +00003100 if (rc == ERROR_SUCCESS)
3101 {
3102 WCHAR guidbuffer[0x200];
3103 size = 0x200;
Aric Stewart7e7b8cf2005-02-18 20:00:34 +00003104 rc = MsiSummaryInfoGetPropertyW(hSumInfo, 9, NULL, NULL, NULL,
Aric Stewart68b07492005-01-25 11:05:37 +00003105 guidbuffer, &size);
3106 if (rc == ERROR_SUCCESS)
3107 {
3108 WCHAR squashed[GUID_SIZE];
3109 /* for now we only care about the first guid */
3110 LPWSTR ptr = strchrW(guidbuffer,';');
3111 if (ptr) *ptr = 0;
3112 squash_guid(guidbuffer,squashed);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003113 msi_reg_set_val_str( hukey, INSTALLPROPERTY_PACKAGECODEW, squashed );
Aric Stewart68b07492005-01-25 11:05:37 +00003114 }
3115 else
3116 {
Francois Gouget0edbaf72005-11-10 12:14:56 +00003117 ERR("Unable to query Revision_Number...\n");
Aric Stewart68b07492005-01-25 11:05:37 +00003118 rc = ERROR_SUCCESS;
3119 }
3120 MsiCloseHandle(hSumInfo);
3121 }
3122 else
3123 {
3124 ERR("Unable to open Summary Information\n");
3125 rc = ERROR_SUCCESS;
3126 }
Aric Stewart2cae30b2005-01-19 19:07:40 +00003127
Aric Stewart6269f002005-01-17 13:40:39 +00003128end:
3129
Aric Stewart6269f002005-01-17 13:40:39 +00003130 RegCloseKey(hkey);
Aric Stewart6269f002005-01-17 13:40:39 +00003131 RegCloseKey(hukey);
3132
Aric Stewart2cf222f2004-07-06 19:00:23 +00003133 return rc;
Aric Stewart2cf222f2004-07-06 19:00:23 +00003134}
3135
Aric Stewartaded32f2005-06-23 09:46:31 +00003136static UINT ITERATE_WriteIniValues(MSIRECORD *row, LPVOID param)
3137{
3138 MSIPACKAGE *package = (MSIPACKAGE*)param;
3139 LPCWSTR component,section,key,value,identifier,filename,dirproperty;
3140 LPWSTR deformated_section, deformated_key, deformated_value;
3141 LPWSTR folder, fullname = NULL;
3142 MSIRECORD * uirow;
Mike McCormack38d67a42005-08-22 09:15:23 +00003143 INT action;
3144 MSICOMPONENT *comp;
Aric Stewartaded32f2005-06-23 09:46:31 +00003145 static const WCHAR szWindowsFolder[] =
3146 {'W','i','n','d','o','w','s','F','o','l','d','e','r',0};
3147
3148 component = MSI_RecordGetString(row, 8);
Mike McCormack38d67a42005-08-22 09:15:23 +00003149 comp = get_loaded_component(package,component);
Aric Stewartaded32f2005-06-23 09:46:31 +00003150
Mike McCormackd693f462005-10-29 11:36:48 +00003151 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL))
Aric Stewartaded32f2005-06-23 09:46:31 +00003152 {
3153 TRACE("Skipping ini file due to disabled component %s\n",
3154 debugstr_w(component));
3155
Mike McCormack38d67a42005-08-22 09:15:23 +00003156 comp->Action = comp->Installed;
Aric Stewartaded32f2005-06-23 09:46:31 +00003157
3158 return ERROR_SUCCESS;
3159 }
3160
Mike McCormack38d67a42005-08-22 09:15:23 +00003161 comp->Action = INSTALLSTATE_LOCAL;
Aric Stewartaded32f2005-06-23 09:46:31 +00003162
3163 identifier = MSI_RecordGetString(row,1);
3164 filename = MSI_RecordGetString(row,2);
3165 dirproperty = MSI_RecordGetString(row,3);
3166 section = MSI_RecordGetString(row,4);
3167 key = MSI_RecordGetString(row,5);
3168 value = MSI_RecordGetString(row,6);
3169 action = MSI_RecordGetInteger(row,7);
3170
3171 deformat_string(package,section,&deformated_section);
3172 deformat_string(package,key,&deformated_key);
3173 deformat_string(package,value,&deformated_value);
3174
3175 if (dirproperty)
3176 {
3177 folder = resolve_folder(package, dirproperty, FALSE, FALSE, NULL);
3178 if (!folder)
Mike McCormack062ad502005-09-15 15:04:08 +00003179 folder = msi_dup_property( package, dirproperty );
Aric Stewartaded32f2005-06-23 09:46:31 +00003180 }
3181 else
Mike McCormack062ad502005-09-15 15:04:08 +00003182 folder = msi_dup_property( package, szWindowsFolder );
Aric Stewartaded32f2005-06-23 09:46:31 +00003183
3184 if (!folder)
3185 {
3186 ERR("Unable to resolve folder! (%s)\n",debugstr_w(dirproperty));
3187 goto cleanup;
3188 }
3189
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003190 fullname = build_directory_name(2, folder, filename);
Aric Stewartaded32f2005-06-23 09:46:31 +00003191
3192 if (action == 0)
3193 {
3194 TRACE("Adding value %s to section %s in %s\n",
3195 debugstr_w(deformated_key), debugstr_w(deformated_section),
3196 debugstr_w(fullname));
3197 WritePrivateProfileStringW(deformated_section, deformated_key,
3198 deformated_value, fullname);
3199 }
3200 else if (action == 1)
3201 {
3202 WCHAR returned[10];
3203 GetPrivateProfileStringW(deformated_section, deformated_key, NULL,
3204 returned, 10, fullname);
3205 if (returned[0] == 0)
3206 {
3207 TRACE("Adding value %s to section %s in %s\n",
3208 debugstr_w(deformated_key), debugstr_w(deformated_section),
3209 debugstr_w(fullname));
3210
3211 WritePrivateProfileStringW(deformated_section, deformated_key,
3212 deformated_value, fullname);
3213 }
3214 }
3215 else if (action == 3)
3216 FIXME("Append to existing section not yet implemented\n");
3217
3218 uirow = MSI_CreateRecord(4);
3219 MSI_RecordSetStringW(uirow,1,identifier);
3220 MSI_RecordSetStringW(uirow,2,deformated_section);
3221 MSI_RecordSetStringW(uirow,3,deformated_key);
3222 MSI_RecordSetStringW(uirow,4,deformated_value);
3223 ui_actiondata(package,szWriteIniValues,uirow);
3224 msiobj_release( &uirow->hdr );
3225cleanup:
Mike McCormackee034ba2005-09-20 11:59:14 +00003226 msi_free(fullname);
3227 msi_free(folder);
3228 msi_free(deformated_key);
3229 msi_free(deformated_value);
3230 msi_free(deformated_section);
Aric Stewartaded32f2005-06-23 09:46:31 +00003231 return ERROR_SUCCESS;
3232}
3233
Aric Stewart516a9c72005-01-14 15:59:26 +00003234static UINT ACTION_WriteIniValues(MSIPACKAGE *package)
3235{
3236 UINT rc;
3237 MSIQUERY * view;
Aric Stewart8e233e92005-03-01 11:45:19 +00003238 static const WCHAR ExecSeqQuery[] =
3239 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00003240 '`','I','n','i','F','i','l','e','`',0};
Aric Stewart516a9c72005-01-14 15:59:26 +00003241
3242 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
3243 if (rc != ERROR_SUCCESS)
3244 {
3245 TRACE("no IniFile table\n");
3246 return ERROR_SUCCESS;
3247 }
3248
Aric Stewartaded32f2005-06-23 09:46:31 +00003249 rc = MSI_IterateRecords(view, NULL, ITERATE_WriteIniValues, package);
Aric Stewart516a9c72005-01-14 15:59:26 +00003250 msiobj_release(&view->hdr);
3251 return rc;
3252}
3253
Aric Stewart854bfc42005-06-24 11:33:02 +00003254static UINT ITERATE_SelfRegModules(MSIRECORD *row, LPVOID param)
Aric Stewart6269f002005-01-17 13:40:39 +00003255{
Aric Stewart854bfc42005-06-24 11:33:02 +00003256 MSIPACKAGE *package = (MSIPACKAGE*)param;
3257 LPCWSTR filename;
3258 LPWSTR FullName;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003259 MSIFILE *file;
Aric Stewart854bfc42005-06-24 11:33:02 +00003260 DWORD len;
Aric Stewart8e233e92005-03-01 11:45:19 +00003261 static const WCHAR ExeStr[] =
Aric Stewartc5a14432005-05-18 17:46:12 +00003262 {'r','e','g','s','v','r','3','2','.','e','x','e',' ','\"',0};
3263 static const WCHAR close[] = {'\"',0};
Aric Stewart2cae30b2005-01-19 19:07:40 +00003264 STARTUPINFOW si;
3265 PROCESS_INFORMATION info;
3266 BOOL brc;
Robert Shearmand2e48e02006-01-23 17:29:50 +01003267 MSIRECORD *uirow;
3268 LPWSTR uipath, p;
Aric Stewart2cae30b2005-01-19 19:07:40 +00003269
3270 memset(&si,0,sizeof(STARTUPINFOW));
3271
Aric Stewart854bfc42005-06-24 11:33:02 +00003272 filename = MSI_RecordGetString(row,1);
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003273 file = get_loaded_file( package, filename );
Aric Stewart854bfc42005-06-24 11:33:02 +00003274
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003275 if (!file)
Aric Stewart854bfc42005-06-24 11:33:02 +00003276 {
3277 ERR("Unable to find file id %s\n",debugstr_w(filename));
3278 return ERROR_SUCCESS;
3279 }
3280
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003281 len = strlenW(ExeStr) + strlenW( file->TargetPath ) + 2;
Aric Stewart854bfc42005-06-24 11:33:02 +00003282
Mike McCormackee034ba2005-09-20 11:59:14 +00003283 FullName = msi_alloc(len*sizeof(WCHAR));
Aric Stewart854bfc42005-06-24 11:33:02 +00003284 strcpyW(FullName,ExeStr);
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003285 strcatW( FullName, file->TargetPath );
Aric Stewart854bfc42005-06-24 11:33:02 +00003286 strcatW(FullName,close);
3287
3288 TRACE("Registering %s\n",debugstr_w(FullName));
3289 brc = CreateProcessW(NULL, FullName, NULL, NULL, FALSE, 0, NULL, c_colon,
3290 &si, &info);
3291
3292 if (brc)
3293 msi_dialog_check_messages(info.hProcess);
3294
Mike McCormackee034ba2005-09-20 11:59:14 +00003295 msi_free(FullName);
Robert Shearmand2e48e02006-01-23 17:29:50 +01003296
3297 /* the UI chunk */
3298 uirow = MSI_CreateRecord( 2 );
3299 uipath = strdupW( file->TargetPath );
3300 p = strrchrW(uipath,'\\');
3301 if (p)
3302 p[1]=0;
3303 MSI_RecordSetStringW( uirow, 1, &p[2] );
3304 MSI_RecordSetStringW( uirow, 2, uipath);
3305 ui_actiondata( package, szSelfRegModules, uirow);
3306 msiobj_release( &uirow->hdr );
3307 msi_free( uipath );
3308 /* FIXME: call ui_progress? */
3309
Aric Stewart854bfc42005-06-24 11:33:02 +00003310 return ERROR_SUCCESS;
3311}
3312
3313static UINT ACTION_SelfRegModules(MSIPACKAGE *package)
3314{
3315 UINT rc;
3316 MSIQUERY * view;
3317 static const WCHAR ExecSeqQuery[] =
3318 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
3319 '`','S','e','l','f','R','e','g','`',0};
3320
Aric Stewart6269f002005-01-17 13:40:39 +00003321 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
3322 if (rc != ERROR_SUCCESS)
3323 {
3324 TRACE("no SelfReg table\n");
3325 return ERROR_SUCCESS;
3326 }
3327
Aric Stewart854bfc42005-06-24 11:33:02 +00003328 MSI_IterateRecords(view, NULL, ITERATE_SelfRegModules, package);
Aric Stewart6269f002005-01-17 13:40:39 +00003329 msiobj_release(&view->hdr);
Aric Stewart854bfc42005-06-24 11:33:02 +00003330
3331 return ERROR_SUCCESS;
Aric Stewart6269f002005-01-17 13:40:39 +00003332}
3333
3334static UINT ACTION_PublishFeatures(MSIPACKAGE *package)
3335{
Mike McCormack1da28582005-08-22 14:09:17 +00003336 MSIFEATURE *feature;
Aric Stewart6269f002005-01-17 13:40:39 +00003337 UINT rc;
Aric Stewart68b07492005-01-25 11:05:37 +00003338 HKEY hkey=0;
3339 HKEY hukey=0;
Aric Stewart68b07492005-01-25 11:05:37 +00003340
Aric Stewartadaef112005-07-07 20:27:06 +00003341 rc = MSIREG_OpenFeaturesKey(package->ProductCode,&hkey,TRUE);
Aric Stewart6269f002005-01-17 13:40:39 +00003342 if (rc != ERROR_SUCCESS)
3343 goto end;
3344
Aric Stewartadaef112005-07-07 20:27:06 +00003345 rc = MSIREG_OpenUserFeaturesKey(package->ProductCode,&hukey,TRUE);
Aric Stewart6269f002005-01-17 13:40:39 +00003346 if (rc != ERROR_SUCCESS)
3347 goto end;
3348
3349 /* here the guids are base 85 encoded */
Mike McCormack1da28582005-08-22 14:09:17 +00003350 LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
Aric Stewart6269f002005-01-17 13:40:39 +00003351 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003352 ComponentList *cl;
Aric Stewart6269f002005-01-17 13:40:39 +00003353 LPWSTR data = NULL;
3354 GUID clsid;
Aric Stewart6269f002005-01-17 13:40:39 +00003355 INT size;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003356 BOOL absent = FALSE;
Robert Shearmand2e48e02006-01-23 17:29:50 +01003357 MSIRECORD *uirow;
Aric Stewart6269f002005-01-17 13:40:39 +00003358
Mike McCormack1da28582005-08-22 14:09:17 +00003359 if (!ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_LOCAL ) &&
3360 !ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_SOURCE ) &&
3361 !ACTION_VerifyFeatureForAction( feature, INSTALLSTATE_ADVERTISED ))
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003362 absent = TRUE;
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00003363
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003364 size = 1;
Mike McCormack1da28582005-08-22 14:09:17 +00003365 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003366 {
3367 size += 21;
3368 }
Mike McCormack79ca56c2005-09-13 10:37:37 +00003369 if (feature->Feature_Parent)
Mike McCormack1da28582005-08-22 14:09:17 +00003370 size += strlenW( feature->Feature_Parent )+2;
Aric Stewart6269f002005-01-17 13:40:39 +00003371
Mike McCormackee034ba2005-09-20 11:59:14 +00003372 data = msi_alloc(size * sizeof(WCHAR));
Aric Stewart6269f002005-01-17 13:40:39 +00003373
3374 data[0] = 0;
Mike McCormack1da28582005-08-22 14:09:17 +00003375 LIST_FOR_EACH_ENTRY( cl, &feature->Components, ComponentList, entry )
Aric Stewart6269f002005-01-17 13:40:39 +00003376 {
Mike McCormack38d67a42005-08-22 09:15:23 +00003377 MSICOMPONENT* component = cl->component;
Aric Stewart6269f002005-01-17 13:40:39 +00003378 WCHAR buf[21];
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003379
Mike McCormack3a940112006-04-19 02:29:03 +09003380 buf[0] = 0;
Mike McCormackefcc1ec2005-09-12 12:07:15 +00003381 if (component->ComponentId)
Aric Stewartc5a14432005-05-18 17:46:12 +00003382 {
Mike McCormack3f2d5d72005-08-19 10:03:11 +00003383 TRACE("From %s\n",debugstr_w(component->ComponentId));
3384 CLSIDFromString(component->ComponentId, &clsid);
Aric Stewartc5a14432005-05-18 17:46:12 +00003385 encode_base85_guid(&clsid,buf);
3386 TRACE("to %s\n",debugstr_w(buf));
3387 strcatW(data,buf);
3388 }
Aric Stewart6269f002005-01-17 13:40:39 +00003389 }
Mike McCormack79ca56c2005-09-13 10:37:37 +00003390 if (feature->Feature_Parent)
Aric Stewart6269f002005-01-17 13:40:39 +00003391 {
3392 static const WCHAR sep[] = {'\2',0};
3393 strcatW(data,sep);
Mike McCormack1da28582005-08-22 14:09:17 +00003394 strcatW(data,feature->Feature_Parent);
Aric Stewart6269f002005-01-17 13:40:39 +00003395 }
3396
Mike McCormack4db02cd2005-09-15 14:58:38 +00003397 msi_reg_set_val_str( hkey, feature->Feature, data );
Mike McCormackee034ba2005-09-20 11:59:14 +00003398 msi_free(data);
Aric Stewart6269f002005-01-17 13:40:39 +00003399
Mike McCormack79ca56c2005-09-13 10:37:37 +00003400 size = 0;
3401 if (feature->Feature_Parent)
3402 size = strlenW(feature->Feature_Parent)*sizeof(WCHAR);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003403 if (!absent)
3404 {
Mike McCormack1da28582005-08-22 14:09:17 +00003405 RegSetValueExW(hukey,feature->Feature,0,REG_SZ,
3406 (LPBYTE)feature->Feature_Parent,size);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003407 }
3408 else
3409 {
Mike McCormack79ca56c2005-09-13 10:37:37 +00003410 size += 2*sizeof(WCHAR);
Mike McCormackee034ba2005-09-20 11:59:14 +00003411 data = msi_alloc(size);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003412 data[0] = 0x6;
Mike McCormack79ca56c2005-09-13 10:37:37 +00003413 data[1] = 0;
3414 if (feature->Feature_Parent)
3415 strcpyW( &data[1], feature->Feature_Parent );
Mike McCormack1da28582005-08-22 14:09:17 +00003416 RegSetValueExW(hukey,feature->Feature,0,REG_SZ,
Mike McCormack16466af2005-07-06 10:33:30 +00003417 (LPBYTE)data,size);
Mike McCormackee034ba2005-09-20 11:59:14 +00003418 msi_free(data);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003419 }
Robert Shearmand2e48e02006-01-23 17:29:50 +01003420
3421 /* the UI chunk */
3422 uirow = MSI_CreateRecord( 1 );
3423 MSI_RecordSetStringW( uirow, 1, feature->Feature );
3424 ui_actiondata( package, szPublishFeatures, uirow);
3425 msiobj_release( &uirow->hdr );
3426 /* FIXME: call ui_progress? */
Aric Stewart6269f002005-01-17 13:40:39 +00003427 }
3428
Aric Stewart6269f002005-01-17 13:40:39 +00003429end:
Aric Stewart6269f002005-01-17 13:40:39 +00003430 RegCloseKey(hkey);
3431 RegCloseKey(hukey);
3432 return rc;
3433}
3434
Mike McCormack61f24a42005-09-30 10:32:41 +00003435static UINT msi_make_package_local( MSIPACKAGE *package, HKEY hkey )
3436{
3437 static const WCHAR installerPathFmt[] = {
3438 '%','s','\\','I','n','s','t','a','l','l','e','r','\\',0};
3439 static const WCHAR fmt[] = {
3440 '%','s','\\',
3441 'I','n','s','t','a','l','l','e','r','\\',
3442 '%','x','.','m','s','i',0};
3443 static const WCHAR szOriginalDatabase[] =
3444 {'O','r','i','g','i','n','a','l','D','a','t','a','b','a','s','e',0};
3445 WCHAR windir[MAX_PATH], path[MAX_PATH], packagefile[MAX_PATH];
3446 INT num, start;
3447 LPWSTR msiFilePath;
3448 BOOL r;
3449
3450 /* copy the package locally */
3451 num = GetTickCount() & 0xffff;
3452 if (!num)
3453 num = 1;
3454 start = num;
3455 GetWindowsDirectoryW( windir, MAX_PATH );
3456 snprintfW( packagefile, MAX_PATH, fmt, windir, num );
3457 do
3458 {
3459 HANDLE handle = CreateFileW(packagefile,GENERIC_WRITE, 0, NULL,
3460 CREATE_NEW, FILE_ATTRIBUTE_NORMAL, 0 );
3461 if (handle != INVALID_HANDLE_VALUE)
3462 {
3463 CloseHandle(handle);
3464 break;
3465 }
3466 if (GetLastError() != ERROR_FILE_EXISTS &&
3467 GetLastError() != ERROR_SHARING_VIOLATION)
3468 break;
3469 if (!(++num & 0xffff)) num = 1;
3470 sprintfW(packagefile,fmt,num);
3471 } while (num != start);
3472
3473 snprintfW( path, MAX_PATH, installerPathFmt, windir );
3474 create_full_pathW(path);
3475
3476 TRACE("Copying to local package %s\n",debugstr_w(packagefile));
3477
3478 msiFilePath = msi_dup_property( package, szOriginalDatabase );
3479 r = CopyFileW( msiFilePath, packagefile, FALSE);
3480 msi_free( msiFilePath );
3481
3482 if (!r)
3483 {
3484 ERR("Unable to copy package (%s -> %s) (error %ld)\n",
3485 debugstr_w(msiFilePath), debugstr_w(packagefile), GetLastError());
3486 return ERROR_FUNCTION_FAILED;
3487 }
3488
3489 /* FIXME: maybe set this key in ACTION_RegisterProduct instead */
3490 msi_reg_set_val_str( hkey, INSTALLPROPERTY_LOCALPACKAGEW, packagefile );
3491 return ERROR_SUCCESS;
3492}
3493
Mike McCormackba293ee2005-10-27 12:08:16 +00003494static UINT msi_write_uninstall_property_vals( MSIPACKAGE *package, HKEY hkey )
3495{
3496 LPWSTR prop, val, key;
3497 static const LPCSTR propval[] = {
3498 "ARPAUTHORIZEDCDFPREFIX", "AuthorizedCDFPrefix",
3499 "ARPCONTACT", "Contact",
3500 "ARPCOMMENTS", "Comments",
3501 "ProductName", "DisplayName",
3502 "ProductVersion", "DisplayVersion",
3503 "ARPHELPLINK", "HelpLink",
3504 "ARPHELPTELEPHONE", "HelpTelephone",
3505 "ARPINSTALLLOCATION", "InstallLocation",
3506 "SourceDir", "InstallSource",
3507 "Manufacturer", "Publisher",
3508 "ARPREADME", "Readme",
3509 "ARPSIZE", "Size",
3510 "ARPURLINFOABOUT", "URLInfoAbout",
3511 "ARPURLUPDATEINFO", "URLUpdateInfo",
3512 NULL,
3513 };
3514 const LPCSTR *p = propval;
3515
3516 while( *p )
3517 {
3518 prop = strdupAtoW( *p++ );
3519 key = strdupAtoW( *p++ );
3520 val = msi_dup_property( package, prop );
3521 msi_reg_set_val_str( hkey, key, val );
3522 msi_free(val);
3523 msi_free(key);
3524 msi_free(prop);
3525 }
3526 return ERROR_SUCCESS;
3527}
3528
Aric Stewart2cae30b2005-01-19 19:07:40 +00003529static UINT ACTION_RegisterProduct(MSIPACKAGE *package)
3530{
Aric Stewart68b07492005-01-25 11:05:37 +00003531 HKEY hkey=0;
Aric Stewarte9db87b2005-06-17 21:25:41 +00003532 LPWSTR buffer = NULL;
Mike McCormackba293ee2005-10-27 12:08:16 +00003533 UINT rc;
Mike McCormack74f0de92005-09-29 10:32:39 +00003534 DWORD size, langid;
Mike McCormack4db02cd2005-09-15 14:58:38 +00003535 static const WCHAR szWindowsInstaller[] =
Mike McCormackba293ee2005-10-27 12:08:16 +00003536 {'W','i','n','d','o','w','s','I','n','s','t','a','l','l','e','r',0};
Aric Stewart36a01502005-06-08 19:07:52 +00003537 static const WCHAR szUpgradeCode[] =
3538 {'U','p','g','r','a','d','e','C','o','d','e',0};
Aric Stewarte9db87b2005-06-17 21:25:41 +00003539 static const WCHAR modpath_fmt[] =
Mike McCormackba293ee2005-10-27 12:08:16 +00003540 {'M','s','i','E','x','e','c','.','e','x','e',' ',
3541 '/','I','[','P','r','o','d','u','c','t','C','o','d','e',']',0};
Aric Stewarte9db87b2005-06-17 21:25:41 +00003542 static const WCHAR szModifyPath[] =
3543 {'M','o','d','i','f','y','P','a','t','h',0};
3544 static const WCHAR szUninstallString[] =
3545 {'U','n','i','n','s','t','a','l','l','S','t','r','i','n','g',0};
3546 static const WCHAR szEstimatedSize[] =
3547 {'E','s','t','i','m','a','t','e','d','S','i','z','e',0};
Aric Stewarte9db87b2005-06-17 21:25:41 +00003548 static const WCHAR szProductLanguage[] =
3549 {'P','r','o','d','u','c','t','L','a','n','g','u','a','g','e',0};
3550 static const WCHAR szProductVersion[] =
3551 {'P','r','o','d','u','c','t','V','e','r','s','i','o','n',0};
Aric Stewarte9db87b2005-06-17 21:25:41 +00003552
3553 SYSTEMTIME systime;
3554 static const WCHAR date_fmt[] = {'%','i','%','i','%','i',0};
Aric Stewart36a01502005-06-08 19:07:52 +00003555 LPWSTR upgrade_code;
Mike McCormackba293ee2005-10-27 12:08:16 +00003556 WCHAR szDate[9];
Aric Stewart2cae30b2005-01-19 19:07:40 +00003557
Aric Stewartadaef112005-07-07 20:27:06 +00003558 rc = MSIREG_OpenUninstallKey(package->ProductCode,&hkey,TRUE);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003559 if (rc != ERROR_SUCCESS)
Mike McCormackba293ee2005-10-27 12:08:16 +00003560 return rc;
Aric Stewart2cae30b2005-01-19 19:07:40 +00003561
3562 /* dump all the info i can grab */
Mike McCormackb7270b82005-12-31 13:18:11 +01003563 /* FIXME: Flesh out more information */
Aric Stewart2cae30b2005-01-19 19:07:40 +00003564
Mike McCormackba293ee2005-10-27 12:08:16 +00003565 msi_write_uninstall_property_vals( package, hkey );
Aric Stewart2cae30b2005-01-19 19:07:40 +00003566
Mike McCormack4db02cd2005-09-15 14:58:38 +00003567 msi_reg_set_val_dword( hkey, szWindowsInstaller, 1 );
Aric Stewart2cae30b2005-01-19 19:07:40 +00003568
Mike McCormack61f24a42005-09-30 10:32:41 +00003569 msi_make_package_local( package, hkey );
Aric Stewart36a01502005-06-08 19:07:52 +00003570
Aric Stewarte9db87b2005-06-17 21:25:41 +00003571 /* do ModifyPath and UninstallString */
3572 size = deformat_string(package,modpath_fmt,&buffer);
Mike McCormack16466af2005-07-06 10:33:30 +00003573 RegSetValueExW(hkey,szModifyPath,0,REG_EXPAND_SZ,(LPBYTE)buffer,size);
3574 RegSetValueExW(hkey,szUninstallString,0,REG_EXPAND_SZ,(LPBYTE)buffer,size);
Mike McCormackee034ba2005-09-20 11:59:14 +00003575 msi_free(buffer);
Aric Stewarte9db87b2005-06-17 21:25:41 +00003576
Mike McCormackb7270b82005-12-31 13:18:11 +01003577 /* FIXME: Write real Estimated Size when we have it */
Mike McCormack4db02cd2005-09-15 14:58:38 +00003578 msi_reg_set_val_dword( hkey, szEstimatedSize, 0 );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003579
3580 GetLocalTime(&systime);
Mike McCormackba293ee2005-10-27 12:08:16 +00003581 sprintfW(szDate,date_fmt,systime.wYear,systime.wMonth,systime.wDay);
3582 msi_reg_set_val_str( hkey, INSTALLPROPERTY_INSTALLDATEW, szDate );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003583
Mike McCormack74f0de92005-09-29 10:32:39 +00003584 langid = msi_get_property_int( package, szProductLanguage, 0 );
3585 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_LANGUAGEW, langid );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003586
Mike McCormack062ad502005-09-15 15:04:08 +00003587 buffer = msi_dup_property( package, szProductVersion );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003588 if (buffer)
3589 {
Mike McCormack230af9d2006-07-14 15:19:08 +09003590 DWORD verdword = msi_version_str_to_dword(buffer);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003591
3592 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_VERSIONW, verdword );
3593 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_VERSIONMAJORW, verdword>>24 );
3594 msi_reg_set_val_dword( hkey, INSTALLPROPERTY_VERSIONMINORW, (verdword>>16)&0x00FF );
Aric Stewarte9db87b2005-06-17 21:25:41 +00003595 }
Mike McCormackee034ba2005-09-20 11:59:14 +00003596 msi_free(buffer);
Aric Stewarte9db87b2005-06-17 21:25:41 +00003597
Aric Stewart36a01502005-06-08 19:07:52 +00003598 /* Handle Upgrade Codes */
Mike McCormack062ad502005-09-15 15:04:08 +00003599 upgrade_code = msi_dup_property( package, szUpgradeCode );
Aric Stewart36a01502005-06-08 19:07:52 +00003600 if (upgrade_code)
3601 {
3602 HKEY hkey2;
3603 WCHAR squashed[33];
3604 MSIREG_OpenUpgradeCodesKey(upgrade_code, &hkey2, TRUE);
Aric Stewartadaef112005-07-07 20:27:06 +00003605 squash_guid(package->ProductCode,squashed);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003606 msi_reg_set_val_str( hkey2, squashed, NULL );
Aric Stewart36a01502005-06-08 19:07:52 +00003607 RegCloseKey(hkey2);
3608 MSIREG_OpenUserUpgradeCodesKey(upgrade_code, &hkey2, TRUE);
Aric Stewartadaef112005-07-07 20:27:06 +00003609 squash_guid(package->ProductCode,squashed);
Mike McCormack4db02cd2005-09-15 14:58:38 +00003610 msi_reg_set_val_str( hkey2, squashed, NULL );
Aric Stewart36a01502005-06-08 19:07:52 +00003611 RegCloseKey(hkey2);
3612
Mike McCormackee034ba2005-09-20 11:59:14 +00003613 msi_free(upgrade_code);
Aric Stewart36a01502005-06-08 19:07:52 +00003614 }
Aric Stewart2cae30b2005-01-19 19:07:40 +00003615
Aric Stewart2cae30b2005-01-19 19:07:40 +00003616 RegCloseKey(hkey);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003617
Robert Shearmand2e48e02006-01-23 17:29:50 +01003618 /* FIXME: call ui_actiondata */
3619
Aric Stewart2cae30b2005-01-19 19:07:40 +00003620 return ERROR_SUCCESS;
3621}
3622
3623static UINT ACTION_InstallExecute(MSIPACKAGE *package)
3624{
Mike McCormacka977b2c2005-11-03 09:56:29 +00003625 return execute_script(package,INSTALL_SCRIPT);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003626}
3627
3628static UINT ACTION_InstallFinalize(MSIPACKAGE *package)
3629{
Aric Stewart9cd707d2005-05-27 19:24:22 +00003630 UINT rc;
3631
Aric Stewart9cd707d2005-05-27 19:24:22 +00003632 /* turn off scheduleing */
3633 package->script->CurrentlyScripting= FALSE;
3634
Aric Stewart54c67dd2005-01-25 20:17:09 +00003635 /* first do the same as an InstallExecute */
Aric Stewart9cd707d2005-05-27 19:24:22 +00003636 rc = ACTION_InstallExecute(package);
3637 if (rc != ERROR_SUCCESS)
3638 return rc;
Aric Stewart54c67dd2005-01-25 20:17:09 +00003639
3640 /* then handle Commit Actions */
Aric Stewart9cd707d2005-05-27 19:24:22 +00003641 rc = execute_script(package,COMMIT_SCRIPT);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003642
Aric Stewart9cd707d2005-05-27 19:24:22 +00003643 return rc;
Aric Stewart2cae30b2005-01-19 19:07:40 +00003644}
3645
3646static UINT ACTION_ForceReboot(MSIPACKAGE *package)
3647{
3648 static const WCHAR RunOnce[] = {
3649 'S','o','f','t','w','a','r','e','\\',
3650 'M','i','c','r','o','s','o','f','t','\\',
3651 'W','i','n','d','o','w','s','\\',
3652 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
Mike McCormack09b82642005-02-22 19:31:45 +00003653 'R','u','n','O','n','c','e',0};
Aric Stewart2cae30b2005-01-19 19:07:40 +00003654 static const WCHAR InstallRunOnce[] = {
3655 'S','o','f','t','w','a','r','e','\\',
3656 'M','i','c','r','o','s','o','f','t','\\',
3657 'W','i','n','d','o','w','s','\\',
3658 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
3659 'I','n','s','t','a','l','l','e','r','\\',
Mike McCormack09b82642005-02-22 19:31:45 +00003660 'R','u','n','O','n','c','e','E','n','t','r','i','e','s',0};
Aric Stewart2cae30b2005-01-19 19:07:40 +00003661
3662 static const WCHAR msiexec_fmt[] = {
Juan Lang014ad3b2005-03-01 10:41:52 +00003663 '%','s',
Aric Stewart2cae30b2005-01-19 19:07:40 +00003664 '\\','M','s','i','E','x','e','c','.','e','x','e',' ','/','@',' ',
3665 '\"','%','s','\"',0};
3666 static const WCHAR install_fmt[] = {
3667 '/','I',' ','\"','%','s','\"',' ',
3668 'A','F','T','E','R','R','E','B','O','O','T','=','1',' ',
3669 'R','U','N','O','N','C','E','E','N','T','R','Y','=','\"','%','s','\"',0};
Juan Lang014ad3b2005-03-01 10:41:52 +00003670 WCHAR buffer[256], sysdir[MAX_PATH];
Aric Stewartadaef112005-07-07 20:27:06 +00003671 HKEY hkey;
Mike McCormack4db02cd2005-09-15 14:58:38 +00003672 WCHAR squished_pc[100];
Aric Stewart2cae30b2005-01-19 19:07:40 +00003673
Aric Stewartadaef112005-07-07 20:27:06 +00003674 squash_guid(package->ProductCode,squished_pc);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003675
Juan Lang014ad3b2005-03-01 10:41:52 +00003676 GetSystemDirectoryW(sysdir, sizeof(sysdir)/sizeof(sysdir[0]));
Aric Stewart2cae30b2005-01-19 19:07:40 +00003677 RegCreateKeyW(HKEY_LOCAL_MACHINE,RunOnce,&hkey);
Juan Lang014ad3b2005-03-01 10:41:52 +00003678 snprintfW(buffer,sizeof(buffer)/sizeof(buffer[0]),msiexec_fmt,sysdir,
3679 squished_pc);
Aric Stewart2cae30b2005-01-19 19:07:40 +00003680
Mike McCormack4db02cd2005-09-15 14:58:38 +00003681 msi_reg_set_val_str( hkey, squished_pc, buffer );
Aric Stewart2cae30b2005-01-19 19:07:40 +00003682 RegCloseKey(hkey);
3683
3684 TRACE("Reboot command %s\n",debugstr_w(buffer));
3685
3686 RegCreateKeyW(HKEY_LOCAL_MACHINE,InstallRunOnce,&hkey);
Aric Stewartadaef112005-07-07 20:27:06 +00003687 sprintfW(buffer,install_fmt,package->ProductCode,squished_pc);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00003688
Mike McCormack4db02cd2005-09-15 14:58:38 +00003689 msi_reg_set_val_str( hkey, squished_pc, buffer );
Aric Stewart2cae30b2005-01-19 19:07:40 +00003690 RegCloseKey(hkey);
3691
Aric Stewart68b07492005-01-25 11:05:37 +00003692 return ERROR_INSTALL_SUSPEND;
Aric Stewart2cae30b2005-01-19 19:07:40 +00003693}
3694
Aric Stewart90c57392005-01-31 16:23:12 +00003695UINT ACTION_ResolveSource(MSIPACKAGE* package)
3696{
Aric Stewart94d68182005-08-15 20:50:06 +00003697 DWORD attrib;
3698 UINT rc;
Aric Stewart90c57392005-01-31 16:23:12 +00003699 /*
3700 * we are currently doing what should be done here in the top level Install
3701 * however for Adminastrative and uninstalls this step will be needed
3702 */
Aric Stewart94d68182005-08-15 20:50:06 +00003703 if (!package->PackagePath)
3704 return ERROR_SUCCESS;
3705
3706 attrib = GetFileAttributesW(package->PackagePath);
3707 if (attrib == INVALID_FILE_ATTRIBUTES)
3708 {
3709 LPWSTR prompt;
3710 LPWSTR msg;
3711 DWORD size = 0;
3712
3713 rc = MsiSourceListGetInfoW(package->ProductCode, NULL,
3714 MSIINSTALLCONTEXT_USERMANAGED, MSICODE_PRODUCT,
3715 INSTALLPROPERTY_DISKPROMPTW,NULL,&size);
3716 if (rc == ERROR_MORE_DATA)
3717 {
Mike McCormackee034ba2005-09-20 11:59:14 +00003718 prompt = msi_alloc(size * sizeof(WCHAR));
Aric Stewart94d68182005-08-15 20:50:06 +00003719 MsiSourceListGetInfoW(package->ProductCode, NULL,
3720 MSIINSTALLCONTEXT_USERMANAGED, MSICODE_PRODUCT,
3721 INSTALLPROPERTY_DISKPROMPTW,prompt,&size);
3722 }
3723 else
3724 prompt = strdupW(package->PackagePath);
3725
3726 msg = generate_error_string(package,1302,1,prompt);
3727 while(attrib == INVALID_FILE_ATTRIBUTES)
3728 {
3729 rc = MessageBoxW(NULL,msg,NULL,MB_OKCANCEL);
3730 if (rc == IDCANCEL)
3731 {
3732 rc = ERROR_INSTALL_USEREXIT;
3733 break;
3734 }
3735 attrib = GetFileAttributesW(package->PackagePath);
3736 }
Mike McCormackee034ba2005-09-20 11:59:14 +00003737 msi_free(prompt);
Aric Stewart94d68182005-08-15 20:50:06 +00003738 rc = ERROR_SUCCESS;
3739 }
3740 else
3741 return ERROR_SUCCESS;
3742
3743 return rc;
Aric Stewart90c57392005-01-31 16:23:12 +00003744}
3745
Aric Stewartc7e88e02005-02-10 17:09:44 +00003746static UINT ACTION_RegisterUser(MSIPACKAGE *package)
3747{
Aric Stewartc7e88e02005-02-10 17:09:44 +00003748 HKEY hkey=0;
3749 LPWSTR buffer;
Aric Stewartc7e88e02005-02-10 17:09:44 +00003750 LPWSTR productid;
3751 UINT rc,i;
Aric Stewartc7e88e02005-02-10 17:09:44 +00003752
3753 static const WCHAR szPropKeys[][80] =
3754 {
Aric Stewart8e233e92005-03-01 11:45:19 +00003755 {'P','r','o','d','u','c','t','I','D',0},
3756 {'U','S','E','R','N','A','M','E',0},
3757 {'C','O','M','P','A','N','Y','N','A','M','E',0},
3758 {0},
Aric Stewartc7e88e02005-02-10 17:09:44 +00003759 };
3760
3761 static const WCHAR szRegKeys[][80] =
3762 {
Aric Stewart8e233e92005-03-01 11:45:19 +00003763 {'P','r','o','d','u','c','t','I','D',0},
3764 {'R','e','g','O','w','n','e','r',0},
3765 {'R','e','g','C','o','m','p','a','n','y',0},
3766 {0},
Aric Stewartc7e88e02005-02-10 17:09:44 +00003767 };
3768
Mike McCormack062ad502005-09-15 15:04:08 +00003769 productid = msi_dup_property( package, INSTALLPROPERTY_PRODUCTIDW );
Aric Stewartc7e88e02005-02-10 17:09:44 +00003770 if (!productid)
3771 return ERROR_SUCCESS;
3772
Aric Stewartadaef112005-07-07 20:27:06 +00003773 rc = MSIREG_OpenUninstallKey(package->ProductCode,&hkey,TRUE);
Aric Stewartc7e88e02005-02-10 17:09:44 +00003774 if (rc != ERROR_SUCCESS)
3775 goto end;
3776
Mike McCormack67189f92005-09-16 18:45:19 +00003777 for( i = 0; szPropKeys[i][0]; i++ )
Aric Stewartc7e88e02005-02-10 17:09:44 +00003778 {
Mike McCormack062ad502005-09-15 15:04:08 +00003779 buffer = msi_dup_property( package, szPropKeys[i] );
Mike McCormack4db02cd2005-09-15 14:58:38 +00003780 msi_reg_set_val_str( hkey, szRegKeys[i], buffer );
Mike McCormackee034ba2005-09-20 11:59:14 +00003781 msi_free( buffer );
Aric Stewartc7e88e02005-02-10 17:09:44 +00003782 }
3783
3784end:
Mike McCormackee034ba2005-09-20 11:59:14 +00003785 msi_free(productid);
Aric Stewartc7e88e02005-02-10 17:09:44 +00003786 RegCloseKey(hkey);
3787
Robert Shearmand2e48e02006-01-23 17:29:50 +01003788 /* FIXME: call ui_actiondata */
3789
Aric Stewartc7e88e02005-02-10 17:09:44 +00003790 return ERROR_SUCCESS;
3791}
3792
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00003793
3794static UINT ACTION_ExecuteAction(MSIPACKAGE *package)
3795{
3796 UINT rc;
Aric Stewart25f1e752005-06-24 12:14:52 +00003797
Aric Stewartc9802932005-06-30 20:45:43 +00003798 package->script->InWhatSequence |= SEQUENCE_EXEC;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00003799 rc = ACTION_ProcessExecSequence(package,FALSE);
Aric Stewartb6bc6aa2005-02-24 12:47:43 +00003800 return rc;
3801}
3802
Aric Stewart0af24872005-02-25 14:00:09 +00003803
3804/*
3805 * Code based off of code located here
3806 * http://www.codeproject.com/gdi/fontnamefromfile.asp
3807 *
3808 * Using string index 4 (full font name) instead of 1 (family name)
3809 */
3810static LPWSTR load_ttfname_from(LPCWSTR filename)
3811{
3812 HANDLE handle;
3813 LPWSTR ret = NULL;
3814 int i;
3815
3816 typedef struct _tagTT_OFFSET_TABLE{
3817 USHORT uMajorVersion;
3818 USHORT uMinorVersion;
3819 USHORT uNumOfTables;
3820 USHORT uSearchRange;
3821 USHORT uEntrySelector;
3822 USHORT uRangeShift;
3823 }TT_OFFSET_TABLE;
3824
3825 typedef struct _tagTT_TABLE_DIRECTORY{
3826 char szTag[4]; /* table name */
3827 ULONG uCheckSum; /* Check sum */
3828 ULONG uOffset; /* Offset from beginning of file */
3829 ULONG uLength; /* length of the table in bytes */
3830 }TT_TABLE_DIRECTORY;
3831
3832 typedef struct _tagTT_NAME_TABLE_HEADER{
3833 USHORT uFSelector; /* format selector. Always 0 */
3834 USHORT uNRCount; /* Name Records count */
3835 USHORT uStorageOffset; /* Offset for strings storage,
3836 * from start of the table */
3837 }TT_NAME_TABLE_HEADER;
3838
3839 typedef struct _tagTT_NAME_RECORD{
3840 USHORT uPlatformID;
3841 USHORT uEncodingID;
3842 USHORT uLanguageID;
3843 USHORT uNameID;
3844 USHORT uStringLength;
3845 USHORT uStringOffset; /* from start of storage area */
3846 }TT_NAME_RECORD;
Alexandre Julliardd0ee9f92005-03-02 12:23:20 +00003847
3848#define SWAPWORD(x) MAKEWORD(HIBYTE(x), LOBYTE(x))
3849#define SWAPLONG(x) MAKELONG(SWAPWORD(HIWORD(x)), SWAPWORD(LOWORD(x)))
3850
Aric Stewart0af24872005-02-25 14:00:09 +00003851 handle = CreateFileW(filename ,GENERIC_READ, 0, NULL, OPEN_EXISTING,
3852 FILE_ATTRIBUTE_NORMAL, 0 );
3853 if (handle != INVALID_HANDLE_VALUE)
3854 {
3855 TT_TABLE_DIRECTORY tblDir;
3856 BOOL bFound = FALSE;
3857 TT_OFFSET_TABLE ttOffsetTable;
Thomas Weidenmueller561f2f02005-10-29 10:29:08 +00003858 DWORD dwRead;
Aric Stewart0af24872005-02-25 14:00:09 +00003859
Thomas Weidenmueller561f2f02005-10-29 10:29:08 +00003860 ReadFile(handle,&ttOffsetTable, sizeof(TT_OFFSET_TABLE),&dwRead,NULL);
Aric Stewart0af24872005-02-25 14:00:09 +00003861 ttOffsetTable.uNumOfTables = SWAPWORD(ttOffsetTable.uNumOfTables);
3862 ttOffsetTable.uMajorVersion = SWAPWORD(ttOffsetTable.uMajorVersion);
3863 ttOffsetTable.uMinorVersion = SWAPWORD(ttOffsetTable.uMinorVersion);
3864
3865 if (ttOffsetTable.uMajorVersion != 1 ||
3866 ttOffsetTable.uMinorVersion != 0)
3867 return NULL;
3868
3869 for (i=0; i< ttOffsetTable.uNumOfTables; i++)
3870 {
Thomas Weidenmueller561f2f02005-10-29 10:29:08 +00003871 ReadFile(handle,&tblDir, sizeof(TT_TABLE_DIRECTORY),&dwRead,NULL);
Aric Stewart0af24872005-02-25 14:00:09 +00003872 if (strncmp(tblDir.szTag,"name",4)==0)
3873 {
3874 bFound = TRUE;
3875 tblDir.uLength = SWAPLONG(tblDir.uLength);
3876 tblDir.uOffset = SWAPLONG(tblDir.uOffset);
3877 break;
3878 }
3879 }
3880
3881 if (bFound)
3882 {
3883 TT_NAME_TABLE_HEADER ttNTHeader;
3884 TT_NAME_RECORD ttRecord;
3885
3886 SetFilePointer(handle, tblDir.uOffset, NULL, FILE_BEGIN);
3887 ReadFile(handle,&ttNTHeader, sizeof(TT_NAME_TABLE_HEADER),
Thomas Weidenmueller561f2f02005-10-29 10:29:08 +00003888 &dwRead,NULL);
Aric Stewart0af24872005-02-25 14:00:09 +00003889
3890 ttNTHeader.uNRCount = SWAPWORD(ttNTHeader.uNRCount);
3891 ttNTHeader.uStorageOffset = SWAPWORD(ttNTHeader.uStorageOffset);
3892 bFound = FALSE;
3893 for(i=0; i<ttNTHeader.uNRCount; i++)
3894 {
Thomas Weidenmueller561f2f02005-10-29 10:29:08 +00003895 ReadFile(handle,&ttRecord, sizeof(TT_NAME_RECORD),&dwRead,NULL);
Aric Stewart0af24872005-02-25 14:00:09 +00003896 ttRecord.uNameID = SWAPWORD(ttRecord.uNameID);
3897 /* 4 is the Full Font Name */
3898 if(ttRecord.uNameID == 4)
3899 {
3900 int nPos;
3901 LPSTR buf;
Mike McCormackf9acfe62005-06-07 20:29:51 +00003902 static LPCSTR tt = " (TrueType)";
Aric Stewart0af24872005-02-25 14:00:09 +00003903
3904 ttRecord.uStringLength = SWAPWORD(ttRecord.uStringLength);
3905 ttRecord.uStringOffset = SWAPWORD(ttRecord.uStringOffset);
3906 nPos = SetFilePointer(handle, 0, NULL, FILE_CURRENT);
3907 SetFilePointer(handle, tblDir.uOffset +
3908 ttRecord.uStringOffset +
3909 ttNTHeader.uStorageOffset,
3910 NULL, FILE_BEGIN);
Mike McCormack3a940112006-04-19 02:29:03 +09003911 buf = msi_alloc_zero( ttRecord.uStringLength + 1 + strlen(tt) );
Thomas Weidenmueller561f2f02005-10-29 10:29:08 +00003912 ReadFile(handle, buf, ttRecord.uStringLength, &dwRead, NULL);
Aric Stewart0af24872005-02-25 14:00:09 +00003913 if (strlen(buf) > 0)
3914 {
3915 strcat(buf,tt);
3916 ret = strdupAtoW(buf);
Mike McCormackee034ba2005-09-20 11:59:14 +00003917 msi_free(buf);
Aric Stewart0af24872005-02-25 14:00:09 +00003918 break;
3919 }
3920
Mike McCormackee034ba2005-09-20 11:59:14 +00003921 msi_free(buf);
Aric Stewart0af24872005-02-25 14:00:09 +00003922 SetFilePointer(handle,nPos, NULL, FILE_BEGIN);
3923 }
3924 }
3925 }
3926 CloseHandle(handle);
3927 }
Aric Stewart8a6a9a62005-05-18 09:44:38 +00003928 else
3929 ERR("Unable to open font file %s\n", debugstr_w(filename));
Aric Stewart0af24872005-02-25 14:00:09 +00003930
3931 TRACE("Returning fontname %s\n",debugstr_w(ret));
3932 return ret;
3933}
3934
Aric Stewarta9b50522005-06-23 11:04:09 +00003935static UINT ITERATE_RegisterFonts(MSIRECORD *row, LPVOID param)
Aric Stewart0af24872005-02-25 14:00:09 +00003936{
Aric Stewarta9b50522005-06-23 11:04:09 +00003937 MSIPACKAGE *package = (MSIPACKAGE*)param;
3938 LPWSTR name;
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003939 LPCWSTR filename;
3940 MSIFILE *file;
Aric Stewart8e233e92005-03-01 11:45:19 +00003941 static const WCHAR regfont1[] =
3942 {'S','o','f','t','w','a','r','e','\\',
3943 'M','i','c','r','o','s','o','f','t','\\',
3944 'W','i','n','d','o','w','s',' ','N','T','\\',
3945 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
3946 'F','o','n','t','s',0};
3947 static const WCHAR regfont2[] =
3948 {'S','o','f','t','w','a','r','e','\\',
3949 'M','i','c','r','o','s','o','f','t','\\',
3950 'W','i','n','d','o','w','s','\\',
3951 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
3952 'F','o','n','t','s',0};
Aric Stewart0af24872005-02-25 14:00:09 +00003953 HKEY hkey1;
3954 HKEY hkey2;
Robert Shearmand2e48e02006-01-23 17:29:50 +01003955 MSIRECORD *uirow;
3956 LPWSTR uipath, p;
Aric Stewart0af24872005-02-25 14:00:09 +00003957
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003958 filename = MSI_RecordGetString( row, 1 );
3959 file = get_loaded_file( package, filename );
3960 if (!file)
Aric Stewarta9b50522005-06-23 11:04:09 +00003961 {
3962 ERR("Unable to load file\n");
3963 return ERROR_SUCCESS;
3964 }
3965
3966 /* check to make sure that component is installed */
Mike McCormackd693f462005-10-29 11:36:48 +00003967 if (!ACTION_VerifyComponentForAction( file->Component, INSTALLSTATE_LOCAL))
Aric Stewarta9b50522005-06-23 11:04:09 +00003968 {
3969 TRACE("Skipping: Component not scheduled for install\n");
3970 return ERROR_SUCCESS;
3971 }
3972
3973 RegCreateKeyW(HKEY_LOCAL_MACHINE,regfont1,&hkey1);
3974 RegCreateKeyW(HKEY_LOCAL_MACHINE,regfont2,&hkey2);
3975
3976 if (MSI_RecordIsNull(row,2))
Mike McCormacke18f8ab2005-08-23 10:03:17 +00003977 name = load_ttfname_from( file->TargetPath );
Aric Stewarta9b50522005-06-23 11:04:09 +00003978 else
Mike McCormack51c66182005-10-27 12:36:12 +00003979 name = msi_dup_record_field(row,2);
Aric Stewarta9b50522005-06-23 11:04:09 +00003980
3981 if (name)
3982 {
Mike McCormack4db02cd2005-09-15 14:58:38 +00003983 msi_reg_set_val_str( hkey1, name, file->FileName );
3984 msi_reg_set_val_str( hkey2, name, file->FileName );
Aric Stewarta9b50522005-06-23 11:04:09 +00003985 }
3986
Mike McCormackee034ba2005-09-20 11:59:14 +00003987 msi_free(name);
Aric Stewarta9b50522005-06-23 11:04:09 +00003988 RegCloseKey(hkey1);
3989 RegCloseKey(hkey2);
Robert Shearmand2e48e02006-01-23 17:29:50 +01003990
3991 /* the UI chunk */
3992 uirow = MSI_CreateRecord( 1 );
3993 uipath = strdupW( file->TargetPath );
3994 p = strrchrW(uipath,'\\');
3995 if (p) p++;
3996 else p = uipath;
3997 MSI_RecordSetStringW( uirow, 1, p );
3998 ui_actiondata( package, szRegisterFonts, uirow);
3999 msiobj_release( &uirow->hdr );
4000 msi_free( uipath );
4001 /* FIXME: call ui_progress? */
4002
Aric Stewarta9b50522005-06-23 11:04:09 +00004003 return ERROR_SUCCESS;
4004}
4005
4006static UINT ACTION_RegisterFonts(MSIPACKAGE *package)
4007{
4008 UINT rc;
4009 MSIQUERY * view;
4010 static const WCHAR ExecSeqQuery[] =
4011 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
4012 '`','F','o','n','t','`',0};
Juan Lang88adb532005-05-19 11:15:24 +00004013
Aric Stewart0af24872005-02-25 14:00:09 +00004014 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
4015 if (rc != ERROR_SUCCESS)
Juan Lang88adb532005-05-19 11:15:24 +00004016 {
4017 TRACE("MSI_DatabaseOpenViewW failed: %d\n", rc);
4018 return ERROR_SUCCESS;
4019 }
Aric Stewart0af24872005-02-25 14:00:09 +00004020
Aric Stewarta9b50522005-06-23 11:04:09 +00004021 MSI_IterateRecords(view, NULL, ITERATE_RegisterFonts, package);
Aric Stewart0af24872005-02-25 14:00:09 +00004022 msiobj_release(&view->hdr);
4023
Aric Stewarta9b50522005-06-23 11:04:09 +00004024 return ERROR_SUCCESS;
Aric Stewart0af24872005-02-25 14:00:09 +00004025}
4026
Aric Stewart072c5e52005-04-20 12:50:05 +00004027static UINT ITERATE_PublishComponent(MSIRECORD *rec, LPVOID param)
4028{
4029 MSIPACKAGE *package = (MSIPACKAGE*)param;
Aric Stewart09b0aba2005-06-09 20:30:59 +00004030 LPCWSTR compgroupid=NULL;
4031 LPCWSTR feature=NULL;
4032 LPCWSTR text = NULL;
4033 LPCWSTR qualifier = NULL;
4034 LPCWSTR component = NULL;
Aric Stewart6f43c182005-05-26 12:24:28 +00004035 LPWSTR advertise = NULL;
4036 LPWSTR output = NULL;
Aric Stewart072c5e52005-04-20 12:50:05 +00004037 HKEY hkey;
4038 UINT rc = ERROR_SUCCESS;
Mike McCormack38d67a42005-08-22 09:15:23 +00004039 MSICOMPONENT *comp;
Aric Stewart072c5e52005-04-20 12:50:05 +00004040 DWORD sz = 0;
Robert Shearmand2e48e02006-01-23 17:29:50 +01004041 MSIRECORD *uirow;
Aric Stewartb39d8fc2005-05-13 13:56:39 +00004042
Aric Stewart09b0aba2005-06-09 20:30:59 +00004043 component = MSI_RecordGetString(rec,3);
Mike McCormack38d67a42005-08-22 09:15:23 +00004044 comp = get_loaded_component(package,component);
Aric Stewartb39d8fc2005-05-13 13:56:39 +00004045
Mike McCormackd693f462005-10-29 11:36:48 +00004046 if (!ACTION_VerifyComponentForAction( comp, INSTALLSTATE_LOCAL ) &&
4047 !ACTION_VerifyComponentForAction( comp, INSTALLSTATE_SOURCE ) &&
4048 !ACTION_VerifyComponentForAction( comp, INSTALLSTATE_ADVERTISED ))
Aric Stewartb39d8fc2005-05-13 13:56:39 +00004049 {
4050 TRACE("Skipping: Component %s not scheduled for install\n",
4051 debugstr_w(component));
Aric Stewart6f43c182005-05-26 12:24:28 +00004052
Aric Stewartb39d8fc2005-05-13 13:56:39 +00004053 return ERROR_SUCCESS;
4054 }
Aric Stewart072c5e52005-04-20 12:50:05 +00004055
Aric Stewart09b0aba2005-06-09 20:30:59 +00004056 compgroupid = MSI_RecordGetString(rec,1);
Robert Shearmand2e48e02006-01-23 17:29:50 +01004057 qualifier = MSI_RecordGetString(rec,2);
Aric Stewart072c5e52005-04-20 12:50:05 +00004058
4059 rc = MSIREG_OpenUserComponentsKey(compgroupid, &hkey, TRUE);
4060 if (rc != ERROR_SUCCESS)
4061 goto end;
4062
Aric Stewart09b0aba2005-06-09 20:30:59 +00004063 text = MSI_RecordGetString(rec,4);
Aric Stewart09b0aba2005-06-09 20:30:59 +00004064 feature = MSI_RecordGetString(rec,5);
Aric Stewart072c5e52005-04-20 12:50:05 +00004065
Mike McCormack38d67a42005-08-22 09:15:23 +00004066 advertise = create_component_advertise_string(package, comp, feature);
Aric Stewart072c5e52005-04-20 12:50:05 +00004067
Aric Stewart6f43c182005-05-26 12:24:28 +00004068 sz = strlenW(advertise);
4069
Aric Stewart072c5e52005-04-20 12:50:05 +00004070 if (text)
4071 sz += lstrlenW(text);
Aric Stewart072c5e52005-04-20 12:50:05 +00004072
4073 sz+=3;
4074 sz *= sizeof(WCHAR);
4075
Mike McCormack3a940112006-04-19 02:29:03 +09004076 output = msi_alloc_zero(sz);
Aric Stewart6f43c182005-05-26 12:24:28 +00004077 strcpyW(output,advertise);
Mike McCormack470f23d2005-09-22 10:56:26 +00004078 msi_free(advertise);
Aric Stewart072c5e52005-04-20 12:50:05 +00004079
4080 if (text)
4081 strcatW(output,text);
4082
Mike McCormack4db02cd2005-09-15 14:58:38 +00004083 msi_reg_set_val_multi_str( hkey, qualifier, output );
Aric Stewart072c5e52005-04-20 12:50:05 +00004084
4085end:
4086 RegCloseKey(hkey);
Mike McCormackee034ba2005-09-20 11:59:14 +00004087 msi_free(output);
Robert Shearmand2e48e02006-01-23 17:29:50 +01004088
4089 /* the UI chunk */
4090 uirow = MSI_CreateRecord( 2 );
4091 MSI_RecordSetStringW( uirow, 1, compgroupid );
4092 MSI_RecordSetStringW( uirow, 2, qualifier);
4093 ui_actiondata( package, szPublishComponents, uirow);
4094 msiobj_release( &uirow->hdr );
4095 /* FIXME: call ui_progress? */
4096
Aric Stewart072c5e52005-04-20 12:50:05 +00004097 return rc;
4098}
4099
4100/*
4101 * At present I am ignorning the advertised components part of this and only
4102 * focusing on the qualified component sets
4103 */
4104static UINT ACTION_PublishComponents(MSIPACKAGE *package)
4105{
4106 UINT rc;
4107 MSIQUERY * view;
4108 static const WCHAR ExecSeqQuery[] =
4109 {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
Aric Stewart98e38082005-05-20 09:40:42 +00004110 '`','P','u','b','l','i','s','h',
4111 'C','o','m','p','o','n','e','n','t','`',0};
Aric Stewart072c5e52005-04-20 12:50:05 +00004112
4113 rc = MSI_DatabaseOpenViewW(package->db, ExecSeqQuery, &view);
4114 if (rc != ERROR_SUCCESS)
4115 return ERROR_SUCCESS;
4116
4117 rc = MSI_IterateRecords(view, NULL, ITERATE_PublishComponent, package);
4118 msiobj_release(&view->hdr);
4119
4120 return rc;
4121}
Mike McCormack202166c2005-09-23 10:09:18 +00004122
Mike McCormack2586a092005-09-26 09:56:18 +00004123static UINT msi_unimplemented_action_stub( MSIPACKAGE *package,
4124 LPCSTR action, LPCWSTR table )
Mike McCormack567f0312005-09-23 11:06:57 +00004125{
4126 static const WCHAR query[] = {
Mike McCormack2586a092005-09-26 09:56:18 +00004127 'S','E','L','E','C','T',' ','*',' ',
4128 'F','R','O','M',' ','`','%','s','`',0 };
Mike McCormack567f0312005-09-23 11:06:57 +00004129 MSIQUERY *view = NULL;
4130 DWORD count = 0;
Mike McCormack2586a092005-09-26 09:56:18 +00004131 UINT r;
Mike McCormack567f0312005-09-23 11:06:57 +00004132
Mike McCormack2586a092005-09-26 09:56:18 +00004133 r = MSI_OpenQuery( package->db, &view, query, table );
4134 if (r == ERROR_SUCCESS)
Mike McCormack567f0312005-09-23 11:06:57 +00004135 {
Mike McCormack2586a092005-09-26 09:56:18 +00004136 r = MSI_IterateRecords(view, &count, NULL, package);
Mike McCormack567f0312005-09-23 11:06:57 +00004137 msiobj_release(&view->hdr);
4138 }
4139
Mike McCormack2586a092005-09-26 09:56:18 +00004140 if (count)
4141 FIXME("%s -> %lu ignored %s table values\n",
4142 action, count, debugstr_w(table));
4143
Mike McCormack567f0312005-09-23 11:06:57 +00004144 return ERROR_SUCCESS;
4145}
Mike McCormack94fbe092005-09-23 17:21:10 +00004146
Mike McCormack55942702005-10-30 19:23:28 +00004147static UINT ACTION_AllocateRegistrySpace( MSIPACKAGE *package )
4148{
4149 TRACE("%p\n", package);
4150 return ERROR_SUCCESS;
4151}
4152
Mike McCormack2586a092005-09-26 09:56:18 +00004153static UINT ACTION_RemoveIniValues( MSIPACKAGE *package )
Mike McCormack94fbe092005-09-23 17:21:10 +00004154{
Mike McCormack2586a092005-09-26 09:56:18 +00004155 static const WCHAR table[] =
4156 {'R','e','m','o','v','e','I','n','i','F','i','l','e',0 };
4157 return msi_unimplemented_action_stub( package, "RemoveIniValues", table );
Mike McCormack94fbe092005-09-23 17:21:10 +00004158}
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004159
Mike McCormack2586a092005-09-26 09:56:18 +00004160static UINT ACTION_MoveFiles( MSIPACKAGE *package )
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004161{
Mike McCormack2586a092005-09-26 09:56:18 +00004162 static const WCHAR table[] = { 'M','o','v','e','F','i','l','e',0 };
4163 return msi_unimplemented_action_stub( package, "MoveFiles", table );
4164}
4165
4166static UINT ACTION_PatchFiles( MSIPACKAGE *package )
4167{
4168 static const WCHAR table[] = { 'P','a','t','c','h',0 };
4169 return msi_unimplemented_action_stub( package, "PatchFiles", table );
4170}
4171
4172static UINT ACTION_BindImage( MSIPACKAGE *package )
4173{
4174 static const WCHAR table[] = { 'B','i','n','d','I','m','a','g','e',0 };
4175 return msi_unimplemented_action_stub( package, "BindImage", table );
4176}
4177
4178static UINT ACTION_IsolateComponents( MSIPACKAGE *package )
4179{
4180 static const WCHAR table[] = {
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004181 'I','s','o','l','a','t','e','C','o','m','p','o','n','e','n','t',0 };
Mike McCormack2586a092005-09-26 09:56:18 +00004182 return msi_unimplemented_action_stub( package, "IsolateComponents", table );
4183}
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004184
Mike McCormack2586a092005-09-26 09:56:18 +00004185static UINT ACTION_MigrateFeatureStates( MSIPACKAGE *package )
4186{
4187 static const WCHAR table[] = { 'U','p','g','r','a','d','e',0 };
4188 return msi_unimplemented_action_stub( package, "MigrateFeatureStates", table );
4189}
4190
4191static UINT ACTION_SelfUnregModules( MSIPACKAGE *package )
4192{
4193 static const WCHAR table[] = { 'S','e','l','f','R','e','g',0 };
4194 return msi_unimplemented_action_stub( package, "SelfUnregModules", table );
4195}
4196
4197static UINT ACTION_InstallServices( MSIPACKAGE *package )
4198{
4199 static const WCHAR table[] = {
4200 'S','e','r','v','i','c','e','I','n','s','t','a','l','l',0 };
4201 return msi_unimplemented_action_stub( package, "InstallServices", table );
4202}
4203
4204static UINT ACTION_StartServices( MSIPACKAGE *package )
4205{
4206 static const WCHAR table[] = {
4207 'S','e','r','v','i','c','e','C','o','n','t','r','o','l',0 };
4208 return msi_unimplemented_action_stub( package, "StartServices", table );
4209}
4210
4211static UINT ACTION_StopServices( MSIPACKAGE *package )
4212{
4213 static const WCHAR table[] = {
4214 'S','e','r','v','i','c','e','C','o','n','t','r','o','l',0 };
4215 return msi_unimplemented_action_stub( package, "StopServices", table );
4216}
4217
4218static UINT ACTION_DeleteServices( MSIPACKAGE *package )
4219{
4220 static const WCHAR table[] = {
4221 'S','e','r','v','i','c','e','C','o','n','t','r','o','l',0 };
4222 return msi_unimplemented_action_stub( package, "DeleteServices", table );
Mike McCormackb9a3a7a2005-09-25 15:14:03 +00004223}
Mike McCormack3b955152005-09-28 18:10:44 +00004224
4225static UINT ACTION_WriteEnvironmentStrings( MSIPACKAGE *package )
4226{
4227 static const WCHAR table[] = {
4228 'E','n','v','i','r','o','n','m','e','n','t',0 };
4229 return msi_unimplemented_action_stub( package, "WriteEnvironmentStrings", table );
4230}
4231
4232static UINT ACTION_RemoveEnvironmentStrings( MSIPACKAGE *package )
4233{
4234 static const WCHAR table[] = {
4235 'E','n','v','i','r','o','n','m','e','n','t',0 };
4236 return msi_unimplemented_action_stub( package, "RemoveEnvironmentStrings", table );
4237}
4238
4239static UINT ACTION_MsiPublishAssemblies( MSIPACKAGE *package )
4240{
4241 static const WCHAR table[] = {
4242 'M','s','i','A','s','s','e','m','b','l','y',0 };
4243 return msi_unimplemented_action_stub( package, "MsiPublishAssemblies", table );
4244}
4245
4246static UINT ACTION_MsiUnpublishAssemblies( MSIPACKAGE *package )
4247{
4248 static const WCHAR table[] = {
4249 'M','s','i','A','s','s','e','m','b','l','y',0 };
4250 return msi_unimplemented_action_stub( package, "MsiUnpublishAssemblies", table );
4251}
4252
4253static UINT ACTION_UnregisterFonts( MSIPACKAGE *package )
4254{
4255 static const WCHAR table[] = { 'F','o','n','t',0 };
4256 return msi_unimplemented_action_stub( package, "UnregisterFonts", table );
4257}
4258
Mike McCormackf24a9e22005-12-31 13:14:23 +01004259static UINT ACTION_CCPSearch( MSIPACKAGE *package )
4260{
4261 static const WCHAR table[] = { 'C','C','P','S','e','a','r','c','h',0 };
4262 return msi_unimplemented_action_stub( package, "CCPSearch", table );
4263}
4264
4265static UINT ACTION_RMCCPSearch( MSIPACKAGE *package )
4266{
4267 static const WCHAR table[] = { 'C','C','P','S','e','a','r','c','h',0 };
4268 return msi_unimplemented_action_stub( package, "RMCCPSearch", table );
4269}
4270
Mike McCormack88603662006-03-22 23:01:56 +09004271static UINT ACTION_RegisterComPlus( MSIPACKAGE *package )
4272{
4273 static const WCHAR table[] = { 'C','o','m','p','l','u','s',0 };
4274 return msi_unimplemented_action_stub( package, "RegisterComPlus", table );
4275}
4276
4277static UINT ACTION_UnregisterComPlus( MSIPACKAGE *package )
4278{
4279 static const WCHAR table[] = { 'C','o','m','p','l','u','s',0 };
4280 return msi_unimplemented_action_stub( package, "UnregisterComPlus", table );
4281}
4282
Mike McCormack3b955152005-09-28 18:10:44 +00004283static struct _actions StandardActions[] = {
Mike McCormack55942702005-10-30 19:23:28 +00004284 { szAllocateRegistrySpace, ACTION_AllocateRegistrySpace },
Mike McCormack3b955152005-09-28 18:10:44 +00004285 { szAppSearch, ACTION_AppSearch },
4286 { szBindImage, ACTION_BindImage },
Mike McCormackf24a9e22005-12-31 13:14:23 +01004287 { szCCPSearch, ACTION_CCPSearch},
Mike McCormack3b955152005-09-28 18:10:44 +00004288 { szCostFinalize, ACTION_CostFinalize },
4289 { szCostInitialize, ACTION_CostInitialize },
4290 { szCreateFolders, ACTION_CreateFolders },
4291 { szCreateShortcuts, ACTION_CreateShortcuts },
4292 { szDeleteServices, ACTION_DeleteServices },
4293 { szDisableRollback, NULL},
4294 { szDuplicateFiles, ACTION_DuplicateFiles },
4295 { szExecuteAction, ACTION_ExecuteAction },
4296 { szFileCost, ACTION_FileCost },
4297 { szFindRelatedProducts, ACTION_FindRelatedProducts },
4298 { szForceReboot, ACTION_ForceReboot },
4299 { szInstallAdminPackage, NULL},
4300 { szInstallExecute, ACTION_InstallExecute },
4301 { szInstallExecuteAgain, ACTION_InstallExecute },
4302 { szInstallFiles, ACTION_InstallFiles},
4303 { szInstallFinalize, ACTION_InstallFinalize },
4304 { szInstallInitialize, ACTION_InstallInitialize },
4305 { szInstallSFPCatalogFile, NULL},
4306 { szInstallValidate, ACTION_InstallValidate },
4307 { szIsolateComponents, ACTION_IsolateComponents },
4308 { szLaunchConditions, ACTION_LaunchConditions },
4309 { szMigrateFeatureStates, ACTION_MigrateFeatureStates },
4310 { szMoveFiles, ACTION_MoveFiles },
4311 { szMsiPublishAssemblies, ACTION_MsiPublishAssemblies },
4312 { szMsiUnpublishAssemblies, ACTION_MsiUnpublishAssemblies },
4313 { szInstallODBC, NULL},
4314 { szInstallServices, ACTION_InstallServices },
4315 { szPatchFiles, ACTION_PatchFiles },
4316 { szProcessComponents, ACTION_ProcessComponents },
4317 { szPublishComponents, ACTION_PublishComponents },
4318 { szPublishFeatures, ACTION_PublishFeatures },
4319 { szPublishProduct, ACTION_PublishProduct },
4320 { szRegisterClassInfo, ACTION_RegisterClassInfo },
Mike McCormack88603662006-03-22 23:01:56 +09004321 { szRegisterComPlus, ACTION_RegisterComPlus},
Mike McCormack3b955152005-09-28 18:10:44 +00004322 { szRegisterExtensionInfo, ACTION_RegisterExtensionInfo },
4323 { szRegisterFonts, ACTION_RegisterFonts },
4324 { szRegisterMIMEInfo, ACTION_RegisterMIMEInfo },
4325 { szRegisterProduct, ACTION_RegisterProduct },
4326 { szRegisterProgIdInfo, ACTION_RegisterProgIdInfo },
4327 { szRegisterTypeLibraries, ACTION_RegisterTypeLibraries },
4328 { szRegisterUser, ACTION_RegisterUser},
4329 { szRemoveDuplicateFiles, NULL},
4330 { szRemoveEnvironmentStrings, ACTION_RemoveEnvironmentStrings },
4331 { szRemoveExistingProducts, NULL},
Mike McCormackfd4a2002005-11-02 11:42:56 +00004332 { szRemoveFiles, ACTION_RemoveFiles},
Mike McCormack3b955152005-09-28 18:10:44 +00004333 { szRemoveFolders, NULL},
4334 { szRemoveIniValues, ACTION_RemoveIniValues },
4335 { szRemoveODBC, NULL},
4336 { szRemoveRegistryValues, NULL},
4337 { szRemoveShortcuts, NULL},
4338 { szResolveSource, ACTION_ResolveSource},
Mike McCormackf24a9e22005-12-31 13:14:23 +01004339 { szRMCCPSearch, ACTION_RMCCPSearch},
Mike McCormack3b955152005-09-28 18:10:44 +00004340 { szScheduleReboot, NULL},
4341 { szSelfRegModules, ACTION_SelfRegModules },
4342 { szSelfUnregModules, ACTION_SelfUnregModules },
4343 { szSetODBCFolders, NULL},
4344 { szStartServices, ACTION_StartServices },
4345 { szStopServices, ACTION_StopServices },
4346 { szUnpublishComponents, NULL},
4347 { szUnpublishFeatures, NULL},
4348 { szUnregisterClassInfo, NULL},
Mike McCormack88603662006-03-22 23:01:56 +09004349 { szUnregisterComPlus, ACTION_UnregisterComPlus},
Mike McCormack3b955152005-09-28 18:10:44 +00004350 { szUnregisterExtensionInfo, NULL},
4351 { szUnregisterFonts, ACTION_UnregisterFonts },
4352 { szUnregisterMIMEInfo, NULL},
4353 { szUnregisterProgIdInfo, NULL},
4354 { szUnregisterTypeLibraries, NULL},
4355 { szValidateProductID, NULL},
4356 { szWriteEnvironmentStrings, ACTION_WriteEnvironmentStrings },
4357 { szWriteIniValues, ACTION_WriteIniValues },
4358 { szWriteRegistryValues, ACTION_WriteRegistryValues},
4359 { NULL, NULL},
4360};