msi/tests: Test deferral of CreateFolders and RemoveFolders.
Signed-off-by: Zebediah Figura <z.figura12@gmail.com> Signed-off-by: Hans Leidekker <hans@codeweavers.com> Signed-off-by: Alexandre Julliard <julliard@winehq.org>
This commit is contained in:
parent
7fdb828abf
commit
0f7d50c02e
|
@ -1,7 +1,7 @@
|
||||||
TESTDLL = msi.dll
|
TESTDLL = msi.dll
|
||||||
IMPORTS = cabinet msi shell32 ole32 oleaut32 user32 advapi32 version
|
IMPORTS = cabinet msi shell32 ole32 oleaut32 user32 advapi32 version
|
||||||
|
|
||||||
custom_IMPORTS = uuid msi ole32
|
custom_IMPORTS = uuid msi ole32 shell32
|
||||||
|
|
||||||
SOURCES = \
|
SOURCES = \
|
||||||
action.c \
|
action.c \
|
||||||
|
|
|
@ -596,7 +596,11 @@ static const char cf_install_exec_seq_dat[] =
|
||||||
"FileCost\t\t900\n"
|
"FileCost\t\t900\n"
|
||||||
"RemoveFiles\t\t3500\n"
|
"RemoveFiles\t\t3500\n"
|
||||||
"CreateFolders\t\t3700\n"
|
"CreateFolders\t\t3700\n"
|
||||||
|
"cf_immediate\tNOT REMOVE\t3701\n"
|
||||||
|
"cf_deferred\tNOT REMOVE\t3702\n"
|
||||||
"RemoveFolders\t\t3800\n"
|
"RemoveFolders\t\t3800\n"
|
||||||
|
"rf_immediate\tREMOVE\t3801\n"
|
||||||
|
"rf_deferred\tREMOVE\t3802\n"
|
||||||
"InstallFiles\t\t4000\n"
|
"InstallFiles\t\t4000\n"
|
||||||
"RegisterUser\t\t6000\n"
|
"RegisterUser\t\t6000\n"
|
||||||
"RegisterProduct\t\t6100\n"
|
"RegisterProduct\t\t6100\n"
|
||||||
|
@ -609,6 +613,15 @@ static const char cf_install_exec_seq_dat[] =
|
||||||
"InstallValidate\t\t1400\n"
|
"InstallValidate\t\t1400\n"
|
||||||
"LaunchConditions\t\t100\n";
|
"LaunchConditions\t\t100\n";
|
||||||
|
|
||||||
|
static const char cf_custom_action_dat[] =
|
||||||
|
"Action\tType\tSource\tTarget\n"
|
||||||
|
"s72\ti2\tS64\tS0\n"
|
||||||
|
"CustomAction\tAction\n"
|
||||||
|
"cf_immediate\t1\tcustom.dll\tcf_absent\n"
|
||||||
|
"cf_deferred\t1025\tcustom.dll\tcf_present\n"
|
||||||
|
"rf_immediate\t1\tcustom.dll\tcf_present\n"
|
||||||
|
"rf_deferred\t1025\tcustom.dll\tcf_absent\n";
|
||||||
|
|
||||||
static const char sr_selfreg_dat[] =
|
static const char sr_selfreg_dat[] =
|
||||||
"File_\tCost\n"
|
"File_\tCost\n"
|
||||||
"s72\tI2\n"
|
"s72\tI2\n"
|
||||||
|
@ -1771,6 +1784,7 @@ static const msi_table cf_tables[] =
|
||||||
ADD_TABLE(cf_file),
|
ADD_TABLE(cf_file),
|
||||||
ADD_TABLE(cf_create_folders),
|
ADD_TABLE(cf_create_folders),
|
||||||
ADD_TABLE(cf_install_exec_seq),
|
ADD_TABLE(cf_install_exec_seq),
|
||||||
|
ADD_TABLE(cf_custom_action),
|
||||||
ADD_TABLE(media),
|
ADD_TABLE(media),
|
||||||
ADD_TABLE(property)
|
ADD_TABLE(property)
|
||||||
};
|
};
|
||||||
|
|
|
@ -24,10 +24,13 @@
|
||||||
#include <windef.h>
|
#include <windef.h>
|
||||||
#include <winbase.h>
|
#include <winbase.h>
|
||||||
#define COBJMACROS
|
#define COBJMACROS
|
||||||
|
#include <shlobj.h>
|
||||||
#include <msxml.h>
|
#include <msxml.h>
|
||||||
#include <msi.h>
|
#include <msi.h>
|
||||||
#include <msiquery.h>
|
#include <msiquery.h>
|
||||||
|
|
||||||
|
static int todo_level, todo_do_loop;
|
||||||
|
|
||||||
static void ok_(MSIHANDLE hinst, int todo, const char *file, int line, int condition, const char *msg, ...)
|
static void ok_(MSIHANDLE hinst, int todo, const char *file, int line, int condition, const char *msg, ...)
|
||||||
{
|
{
|
||||||
static char buffer[2000];
|
static char buffer[2000];
|
||||||
|
@ -47,8 +50,30 @@ static void ok_(MSIHANDLE hinst, int todo, const char *file, int line, int condi
|
||||||
MsiProcessMessage(hinst, INSTALLMESSAGE_USER, record);
|
MsiProcessMessage(hinst, INSTALLMESSAGE_USER, record);
|
||||||
MsiCloseHandle(record);
|
MsiCloseHandle(record);
|
||||||
}
|
}
|
||||||
#define ok(hinst, condition, ...) ok_(hinst, 0, __FILE__, __LINE__, condition, __VA_ARGS__)
|
|
||||||
#define todo_wine_ok(hinst, condition, ...) ok_(hinst, 1, __FILE__, __LINE__, condition, __VA_ARGS__)
|
static void winetest_start_todo( int is_todo )
|
||||||
|
{
|
||||||
|
todo_level = (todo_level << 1) | (is_todo != 0);
|
||||||
|
todo_do_loop=1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int winetest_loop_todo(void)
|
||||||
|
{
|
||||||
|
int do_loop=todo_do_loop;
|
||||||
|
todo_do_loop=0;
|
||||||
|
return do_loop;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void winetest_end_todo(void)
|
||||||
|
{
|
||||||
|
todo_level >>= 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define ok(hinst, condition, ...) ok_(hinst, todo_level, __FILE__, __LINE__, condition, __VA_ARGS__)
|
||||||
|
#define todo_wine_if(is_todo) for (winetest_start_todo(is_todo); \
|
||||||
|
winetest_loop_todo(); \
|
||||||
|
winetest_end_todo())
|
||||||
|
#define todo_wine todo_wine_if(1)
|
||||||
|
|
||||||
static const char *dbgstr_w(WCHAR *str)
|
static const char *dbgstr_w(WCHAR *str)
|
||||||
{
|
{
|
||||||
|
@ -886,7 +911,8 @@ static void test_costs(MSIHANDLE hinst)
|
||||||
cost = 0xdead;
|
cost = 0xdead;
|
||||||
r = MsiGetFeatureCostA(hinst, NULL, MSICOSTTREE_CHILDREN, INSTALLSTATE_LOCAL, &cost);
|
r = MsiGetFeatureCostA(hinst, NULL, MSICOSTTREE_CHILDREN, INSTALLSTATE_LOCAL, &cost);
|
||||||
ok(hinst, r == ERROR_INVALID_PARAMETER, "got %u\n", r);
|
ok(hinst, r == ERROR_INVALID_PARAMETER, "got %u\n", r);
|
||||||
todo_wine_ok(hinst, !cost, "got %d\n", cost);
|
todo_wine
|
||||||
|
ok(hinst, !cost, "got %d\n", cost);
|
||||||
|
|
||||||
r = MsiGetFeatureCostA(hinst, "One", MSICOSTTREE_CHILDREN, INSTALLSTATE_LOCAL, NULL);
|
r = MsiGetFeatureCostA(hinst, "One", MSICOSTTREE_CHILDREN, INSTALLSTATE_LOCAL, NULL);
|
||||||
ok(hinst, r == RPC_X_NULL_REF_POINTER, "got %u\n", r);
|
ok(hinst, r == RPC_X_NULL_REF_POINTER, "got %u\n", r);
|
||||||
|
@ -894,7 +920,8 @@ static void test_costs(MSIHANDLE hinst)
|
||||||
cost = 0xdead;
|
cost = 0xdead;
|
||||||
r = MsiGetFeatureCostA(hinst, "One", MSICOSTTREE_CHILDREN, INSTALLSTATE_LOCAL, &cost);
|
r = MsiGetFeatureCostA(hinst, "One", MSICOSTTREE_CHILDREN, INSTALLSTATE_LOCAL, &cost);
|
||||||
ok(hinst, !r, "got %u\n", r);
|
ok(hinst, !r, "got %u\n", r);
|
||||||
todo_wine_ok(hinst, cost == 8, "got %d\n", cost);
|
todo_wine
|
||||||
|
ok(hinst, cost == 8, "got %d\n", cost);
|
||||||
|
|
||||||
sz = cost = temp = 0xdead;
|
sz = cost = temp = 0xdead;
|
||||||
r = MsiEnumComponentCostsA(hinst, "One", 0, INSTALLSTATE_LOCAL, NULL, &sz, &cost, &temp);
|
r = MsiEnumComponentCostsA(hinst, "One", 0, INSTALLSTATE_LOCAL, NULL, &sz, &cost, &temp);
|
||||||
|
@ -946,7 +973,8 @@ static void test_costs(MSIHANDLE hinst)
|
||||||
r = MsiEnumComponentCostsA(hinst, "One", 0, INSTALLSTATE_LOCAL, buffer, &sz, &cost, &temp);
|
r = MsiEnumComponentCostsA(hinst, "One", 0, INSTALLSTATE_LOCAL, buffer, &sz, &cost, &temp);
|
||||||
ok(hinst, r == ERROR_MORE_DATA, "got %u\n", r);
|
ok(hinst, r == ERROR_MORE_DATA, "got %u\n", r);
|
||||||
ok(hinst, !strcmp(buffer, "q"), "got \"%s\"\n", buffer);
|
ok(hinst, !strcmp(buffer, "q"), "got \"%s\"\n", buffer);
|
||||||
todo_wine_ok(hinst, sz == 4, "got size %u\n", sz);
|
todo_wine
|
||||||
|
ok(hinst, sz == 4, "got size %u\n", sz);
|
||||||
ok(hinst, cost == 8, "got cost %d\n", cost);
|
ok(hinst, cost == 8, "got cost %d\n", cost);
|
||||||
ok(hinst, !temp, "got temp %d\n", temp);
|
ok(hinst, !temp, "got temp %d\n", temp);
|
||||||
|
|
||||||
|
@ -954,15 +982,19 @@ static void test_costs(MSIHANDLE hinst)
|
||||||
strcpy(buffer,"x");
|
strcpy(buffer,"x");
|
||||||
r = MsiEnumComponentCostsA(hinst, "One", 0, INSTALLSTATE_LOCAL, buffer, &sz, &cost, &temp);
|
r = MsiEnumComponentCostsA(hinst, "One", 0, INSTALLSTATE_LOCAL, buffer, &sz, &cost, &temp);
|
||||||
ok(hinst, r == ERROR_MORE_DATA, "got %u\n", r);
|
ok(hinst, r == ERROR_MORE_DATA, "got %u\n", r);
|
||||||
todo_wine_ok(hinst, !buffer[0], "got \"%s\"\n", buffer);
|
todo_wine {
|
||||||
todo_wine_ok(hinst, sz == 4, "got size %u\n", sz);
|
ok(hinst, !buffer[0], "got \"%s\"\n", buffer);
|
||||||
|
ok(hinst, sz == 4, "got size %u\n", sz);
|
||||||
|
}
|
||||||
|
|
||||||
sz = 2;
|
sz = 2;
|
||||||
strcpy(buffer,"x");
|
strcpy(buffer,"x");
|
||||||
r = MsiEnumComponentCostsA(hinst, "One", 0, INSTALLSTATE_LOCAL, buffer, &sz, &cost, &temp);
|
r = MsiEnumComponentCostsA(hinst, "One", 0, INSTALLSTATE_LOCAL, buffer, &sz, &cost, &temp);
|
||||||
ok(hinst, r == ERROR_MORE_DATA, "got %u\n", r);
|
ok(hinst, r == ERROR_MORE_DATA, "got %u\n", r);
|
||||||
todo_wine_ok(hinst, !strcmp(buffer, "C"), "got \"%s\"\n", buffer);
|
todo_wine {
|
||||||
todo_wine_ok(hinst, sz == 4, "got size %u\n", sz);
|
ok(hinst, !strcmp(buffer, "C"), "got \"%s\"\n", buffer);
|
||||||
|
ok(hinst, sz == 4, "got size %u\n", sz);
|
||||||
|
}
|
||||||
|
|
||||||
sz = 3;
|
sz = 3;
|
||||||
strcpy(buffer,"x");
|
strcpy(buffer,"x");
|
||||||
|
@ -1021,7 +1053,8 @@ UINT WINAPI main_test(MSIHANDLE hinst)
|
||||||
|
|
||||||
/* Test MsiGetDatabaseState() */
|
/* Test MsiGetDatabaseState() */
|
||||||
res = MsiGetDatabaseState(hinst);
|
res = MsiGetDatabaseState(hinst);
|
||||||
todo_wine_ok(hinst, res == MSIDBSTATE_ERROR, "expected MSIDBSTATE_ERROR, got %u\n", res);
|
todo_wine
|
||||||
|
ok(hinst, res == MSIDBSTATE_ERROR, "expected MSIDBSTATE_ERROR, got %u\n", res);
|
||||||
|
|
||||||
test_props(hinst);
|
test_props(hinst);
|
||||||
test_db(hinst);
|
test_db(hinst);
|
||||||
|
@ -1101,7 +1134,8 @@ UINT WINAPI da_deferred(MSIHANDLE hinst)
|
||||||
len = sizeof(prop);
|
len = sizeof(prop);
|
||||||
r = MsiGetPropertyA(hinst, "TESTPATH", prop, &len);
|
r = MsiGetPropertyA(hinst, "TESTPATH", prop, &len);
|
||||||
ok(hinst, r == ERROR_SUCCESS, "got %u\n", r);
|
ok(hinst, r == ERROR_SUCCESS, "got %u\n", r);
|
||||||
todo_wine_ok(hinst, !prop[0], "got %s\n", prop);
|
todo_wine
|
||||||
|
ok(hinst, !prop[0], "got %s\n", prop);
|
||||||
|
|
||||||
/* Test modes */
|
/* Test modes */
|
||||||
ok(hinst, MsiGetMode(hinst, MSIRUNMODE_SCHEDULED), "should be scheduled\n");
|
ok(hinst, MsiGetMode(hinst, MSIRUNMODE_SCHEDULED), "should be scheduled\n");
|
||||||
|
@ -1113,3 +1147,34 @@ UINT WINAPI da_deferred(MSIHANDLE hinst)
|
||||||
|
|
||||||
return ERROR_SUCCESS;
|
return ERROR_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static BOOL pf_exists(const char *file)
|
||||||
|
{
|
||||||
|
char path[MAX_PATH];
|
||||||
|
|
||||||
|
if (FAILED(SHGetFolderPathA(NULL, CSIDL_PROGRAM_FILESX86, NULL, 0, path)))
|
||||||
|
SHGetFolderPathA(NULL, CSIDL_PROGRAM_FILES, NULL, 0, path);
|
||||||
|
strcat(path, "\\");
|
||||||
|
strcat(path, file);
|
||||||
|
return GetFileAttributesA(path) != INVALID_FILE_ATTRIBUTES;
|
||||||
|
}
|
||||||
|
|
||||||
|
UINT WINAPI cf_present(MSIHANDLE hinst)
|
||||||
|
{
|
||||||
|
todo_wine_if(!MsiGetMode(hinst, MSIRUNMODE_SCHEDULED)) {
|
||||||
|
ok(hinst, pf_exists("msitest\\first"), "folder absent\n");
|
||||||
|
ok(hinst, pf_exists("msitest\\second"), "folder absent\n");
|
||||||
|
ok(hinst, pf_exists("msitest\\third"), "folder absent\n");
|
||||||
|
}
|
||||||
|
return ERROR_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
UINT WINAPI cf_absent(MSIHANDLE hinst)
|
||||||
|
{
|
||||||
|
todo_wine_if(!MsiGetMode(hinst, MSIRUNMODE_SCHEDULED)) {
|
||||||
|
ok(hinst, !pf_exists("msitest\\first"), "folder present\n");
|
||||||
|
ok(hinst, !pf_exists("msitest\\second"), "folder present\n");
|
||||||
|
ok(hinst, !pf_exists("msitest\\third"), "folder present\n");
|
||||||
|
}
|
||||||
|
return ERROR_SUCCESS;
|
||||||
|
}
|
||||||
|
|
|
@ -3,3 +3,5 @@
|
||||||
@ stdcall da_immediate(long)
|
@ stdcall da_immediate(long)
|
||||||
@ stdcall da_deferred(long)
|
@ stdcall da_deferred(long)
|
||||||
@ stdcall nested(long)
|
@ stdcall nested(long)
|
||||||
|
@ stdcall cf_present(long)
|
||||||
|
@ stdcall cf_absent(long)
|
||||||
|
|
Loading…
Reference in New Issue