msi: Apply feature selection to the whole feature subtree.
Signed-off-by: Hans Leidekker <hans@codeweavers.com> Signed-off-by: Alexandre Julliard <julliard@winehq.org>
This commit is contained in:
parent
14d81d9eb9
commit
d94653d0eb
|
@ -1772,6 +1772,45 @@ static BOOL process_overrides( MSIPACKAGE *package, int level )
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void disable_children( MSIFEATURE *feature, int level )
|
||||||
|
{
|
||||||
|
FeatureList *fl;
|
||||||
|
|
||||||
|
LIST_FOR_EACH_ENTRY( fl, &feature->Children, FeatureList, entry )
|
||||||
|
{
|
||||||
|
if (!is_feature_selected( feature, level ))
|
||||||
|
{
|
||||||
|
TRACE("child %s (level %d request %d) follows disabled parent %s (level %d request %d)\n",
|
||||||
|
debugstr_w(fl->feature->Feature), fl->feature->Level, fl->feature->ActionRequest,
|
||||||
|
debugstr_w(feature->Feature), feature->Level, feature->ActionRequest);
|
||||||
|
|
||||||
|
fl->feature->Level = feature->Level;
|
||||||
|
fl->feature->Action = INSTALLSTATE_UNKNOWN;
|
||||||
|
fl->feature->ActionRequest = INSTALLSTATE_UNKNOWN;
|
||||||
|
}
|
||||||
|
disable_children( fl->feature, level );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void follow_parent( MSIFEATURE *feature )
|
||||||
|
{
|
||||||
|
FeatureList *fl;
|
||||||
|
|
||||||
|
LIST_FOR_EACH_ENTRY( fl, &feature->Children, FeatureList, entry )
|
||||||
|
{
|
||||||
|
if (fl->feature->Attributes & msidbFeatureAttributesFollowParent)
|
||||||
|
{
|
||||||
|
TRACE("child %s (level %d request %d) follows parent %s (level %d request %d)\n",
|
||||||
|
debugstr_w(fl->feature->Feature), fl->feature->Level, fl->feature->ActionRequest,
|
||||||
|
debugstr_w(feature->Feature), feature->Level, feature->ActionRequest);
|
||||||
|
|
||||||
|
fl->feature->Action = feature->Action;
|
||||||
|
fl->feature->ActionRequest = feature->ActionRequest;
|
||||||
|
}
|
||||||
|
follow_parent( fl->feature );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
UINT MSI_SetFeatureStates(MSIPACKAGE *package)
|
UINT MSI_SetFeatureStates(MSIPACKAGE *package)
|
||||||
{
|
{
|
||||||
int level;
|
int level;
|
||||||
|
@ -1810,24 +1849,9 @@ UINT MSI_SetFeatureStates(MSIPACKAGE *package)
|
||||||
/* disable child features of unselected parent or follow parent */
|
/* disable child features of unselected parent or follow parent */
|
||||||
LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
|
LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
|
||||||
{
|
{
|
||||||
FeatureList *fl;
|
if (feature->Feature_Parent) continue;
|
||||||
|
disable_children( feature, level );
|
||||||
LIST_FOR_EACH_ENTRY( fl, &feature->Children, FeatureList, entry )
|
follow_parent( feature );
|
||||||
{
|
|
||||||
if (!is_feature_selected( feature, level ))
|
|
||||||
{
|
|
||||||
fl->feature->Action = INSTALLSTATE_UNKNOWN;
|
|
||||||
fl->feature->ActionRequest = INSTALLSTATE_UNKNOWN;
|
|
||||||
}
|
|
||||||
else if (fl->feature->Attributes & msidbFeatureAttributesFollowParent)
|
|
||||||
{
|
|
||||||
TRACE("feature %s (level %d request %d) follows parent %s (level %d request %d)\n",
|
|
||||||
debugstr_w(fl->feature->Feature), fl->feature->Level, fl->feature->ActionRequest,
|
|
||||||
debugstr_w(feature->Feature), feature->Level, feature->ActionRequest);
|
|
||||||
fl->feature->Action = feature->Action;
|
|
||||||
fl->feature->ActionRequest = feature->ActionRequest;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else /* preselected */
|
else /* preselected */
|
||||||
|
@ -1852,22 +1876,9 @@ UINT MSI_SetFeatureStates(MSIPACKAGE *package)
|
||||||
}
|
}
|
||||||
LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
|
LIST_FOR_EACH_ENTRY( feature, &package->features, MSIFEATURE, entry )
|
||||||
{
|
{
|
||||||
FeatureList *fl;
|
if (feature->Feature_Parent) continue;
|
||||||
|
disable_children( feature, level );
|
||||||
if (!is_feature_selected( feature, level )) continue;
|
follow_parent( feature );
|
||||||
|
|
||||||
LIST_FOR_EACH_ENTRY( fl, &feature->Children, FeatureList, entry )
|
|
||||||
{
|
|
||||||
if (fl->feature->Attributes & msidbFeatureAttributesFollowParent &&
|
|
||||||
(!(feature->Attributes & msidbFeatureAttributesFavorAdvertise)))
|
|
||||||
{
|
|
||||||
TRACE("feature %s (level %d request %d) follows parent %s (level %d request %d)\n",
|
|
||||||
debugstr_w(fl->feature->Feature), fl->feature->Level, fl->feature->ActionRequest,
|
|
||||||
debugstr_w(feature->Feature), feature->Level, feature->ActionRequest);
|
|
||||||
fl->feature->Action = feature->Action;
|
|
||||||
fl->feature->ActionRequest = feature->ActionRequest;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1222,6 +1222,69 @@ static const char shc_install_exec_seq_dat[] =
|
||||||
"PublishProduct\t\t1900\n"
|
"PublishProduct\t\t1900\n"
|
||||||
"InstallFinalize\t\t2000\n";
|
"InstallFinalize\t\t2000\n";
|
||||||
|
|
||||||
|
static const char ft_file_dat[] =
|
||||||
|
"File\tComponent_\tFileName\tFileSize\tVersion\tLanguage\tAttributes\tSequence\n"
|
||||||
|
"s72\ts72\tl255\ti4\tS72\tS20\tI2\ti2\n"
|
||||||
|
"File\tFile\n"
|
||||||
|
"featuretree\tcomp\tfeaturetree.txt\t1000\t\t\t8192\t1\n";
|
||||||
|
|
||||||
|
static const char ft_comp_dat[] =
|
||||||
|
"Component\tComponentId\tDirectory_\tAttributes\tCondition\tKeyPath\n"
|
||||||
|
"s72\tS38\ts72\ti2\tS255\tS72\n"
|
||||||
|
"Component\tComponent\n"
|
||||||
|
"comp\t{12345678-1234-1234-1234-222222222222}\tTARGETDIR\t0\t\t\n"
|
||||||
|
"comp2\t{12345678-1234-1234-1234-333333333333}\tTARGETDIR\t0\t\tfeaturetree\n";
|
||||||
|
|
||||||
|
static const char ft_feature_dat[] =
|
||||||
|
"Feature\tFeature_Parent\tTitle\tDescription\tDisplay\tLevel\tDirectory_\tAttributes\n"
|
||||||
|
"s38\tS38\tL64\tL255\tI2\ti2\tS72\ti2\n"
|
||||||
|
"Feature\tFeature\n"
|
||||||
|
"A\t\t\t\t2\t1\t\t0\n"
|
||||||
|
"C\tB\t\t\t2\t1\t\t0\n"
|
||||||
|
"B\tA\t\t\t4\t1\t\t0\n"
|
||||||
|
"D\t\t\t\t2\t1\t\t0\n";
|
||||||
|
|
||||||
|
static const char ft_feature_comp_dat[] =
|
||||||
|
"Feature_\tComponent_\n"
|
||||||
|
"s38\ts72\n"
|
||||||
|
"FeatureComponents\tFeature_\tComponent_\n"
|
||||||
|
"C\tcomp\n"
|
||||||
|
"D\tcomp2\n";
|
||||||
|
|
||||||
|
static const char ft_condition_dat[] =
|
||||||
|
"Feature_\tLevel\tCondition\n"
|
||||||
|
"s38\ti2\tS255\n"
|
||||||
|
"Condition\tFeature_\tLevel\n"
|
||||||
|
"A\t0\t\"0\"<>INSTALLTYPE\n";
|
||||||
|
|
||||||
|
static const char ft_custom_action_dat[] =
|
||||||
|
"Action\tType\tSource\tTarget\tISComments\n"
|
||||||
|
"s72\ti2\tS64\tS0\tS255\n"
|
||||||
|
"CustomAction\tAction\n"
|
||||||
|
"Run A\t19\t\tA\t\n"
|
||||||
|
"Run B\t19\t\tB\t\n"
|
||||||
|
"Run C\t19\t\tC\t\n";
|
||||||
|
|
||||||
|
static const char ft_install_exec_seq_dat[] =
|
||||||
|
"Action\tCondition\tSequence\n"
|
||||||
|
"s72\tS255\tI2\n"
|
||||||
|
"InstallExecuteSequence\tAction\n"
|
||||||
|
"CostInitialize\t\t100\n"
|
||||||
|
"FileCost\t\t200\n"
|
||||||
|
"CostFinalize\t\t300\n"
|
||||||
|
"InstallValidate\t\t400\n"
|
||||||
|
"InstallInitialize\t\t500\n"
|
||||||
|
"Run C\t3 = &C AND NOT Installed\t600\n"
|
||||||
|
"Run B\t3 = &B AND NOT Installed\t700\n"
|
||||||
|
"Run A\t3 = &A AND NOT Installed\t800\n"
|
||||||
|
"ProcessComponents\t\t900\n"
|
||||||
|
"RemoveFiles\t\t1000\n"
|
||||||
|
"InstallFiles\t\t1100\n"
|
||||||
|
"RegisterProduct\t\t1200\n"
|
||||||
|
"PublishFeatures\t\t1300\n"
|
||||||
|
"PublishProduct\t\t1400\n"
|
||||||
|
"InstallFinalize\t\t1500\n";
|
||||||
|
|
||||||
typedef struct _msi_table
|
typedef struct _msi_table
|
||||||
{
|
{
|
||||||
const CHAR *filename;
|
const CHAR *filename;
|
||||||
|
@ -1862,6 +1925,20 @@ static const msi_table shc2_tables[] =
|
||||||
ADD_TABLE(shc2_property)
|
ADD_TABLE(shc2_property)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const msi_table ft_tables[] =
|
||||||
|
{
|
||||||
|
ADD_TABLE(media),
|
||||||
|
ADD_TABLE(directory),
|
||||||
|
ADD_TABLE(ft_file),
|
||||||
|
ADD_TABLE(ft_comp),
|
||||||
|
ADD_TABLE(ft_feature),
|
||||||
|
ADD_TABLE(ft_feature_comp),
|
||||||
|
ADD_TABLE(ft_condition),
|
||||||
|
ADD_TABLE(ft_custom_action),
|
||||||
|
ADD_TABLE(ft_install_exec_seq),
|
||||||
|
ADD_TABLE(property)
|
||||||
|
};
|
||||||
|
|
||||||
/* cabinet definitions */
|
/* cabinet definitions */
|
||||||
|
|
||||||
/* make the max size large so there is only one cab file */
|
/* make the max size large so there is only one cab file */
|
||||||
|
@ -5939,6 +6016,35 @@ static void test_remove_upgrade_code(void)
|
||||||
DeleteFileA( msifile );
|
DeleteFileA( msifile );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void test_feature_tree(void)
|
||||||
|
{
|
||||||
|
UINT r;
|
||||||
|
|
||||||
|
if (is_process_limited())
|
||||||
|
{
|
||||||
|
skip( "process is limited\n" );
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
create_file( "msitest\\featuretree.txt", 1000 );
|
||||||
|
create_database( msifile, ft_tables, sizeof(ft_tables)/sizeof(ft_tables[0]) );
|
||||||
|
|
||||||
|
MsiSetInternalUI( INSTALLUILEVEL_NONE, NULL );
|
||||||
|
|
||||||
|
r = MsiInstallProductA( msifile, "INSTALLTYPE=\"0\"" );
|
||||||
|
ok( r == ERROR_INSTALL_FAILURE, "got %u\n", r );
|
||||||
|
|
||||||
|
r = MsiInstallProductA( msifile, "INSTALLTYPE=\"1\"" );
|
||||||
|
ok( r == ERROR_SUCCESS, "got %u\n", r );
|
||||||
|
|
||||||
|
r = MsiInstallProductA( msifile, "REMOVE=ALL" );
|
||||||
|
ok( r == ERROR_SUCCESS, "got %u\n", r );
|
||||||
|
|
||||||
|
DeleteFileA( "msitest\\featuretree.txt" );
|
||||||
|
RemoveDirectoryA( "msitest" );
|
||||||
|
DeleteFileA( msifile );
|
||||||
|
}
|
||||||
|
|
||||||
START_TEST(install)
|
START_TEST(install)
|
||||||
{
|
{
|
||||||
DWORD len;
|
DWORD len;
|
||||||
|
@ -6026,6 +6132,7 @@ START_TEST(install)
|
||||||
test_volume_props();
|
test_volume_props();
|
||||||
test_shared_component();
|
test_shared_component();
|
||||||
test_remove_upgrade_code();
|
test_remove_upgrade_code();
|
||||||
|
test_feature_tree();
|
||||||
|
|
||||||
DeleteFileA(log_file);
|
DeleteFileA(log_file);
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue