Feature Models
v1.3.9-alpha-54
- groupID
- at.tugraz.ist.ase
- artifactID
- fm-v2
fm-v2 provides the management functionalities for basic feature models.
Table of Contents
What the package provides
Generic FeatureModel
class
The package provides the generic FeatureModel
class to construct a feature model with any type of features,
relationships, and cross-tree constraints. The class accepts three bounded type parameters:
Feature
and all its subclassesAbstractRelationship
and all its subclassesCTConstraint
and all its subclasses
Since the class is generic, i.e., it doesn’t know how to build features, relationships, and constraints,
it requires builders to create features, relationships, and constraints.
The package includes built-in builders for basic features, relationships and constraints.
These builders conform to the interfaces IFeatureBuildable
, IRelationshipBuildable
, and IConstraintBuildable
.
Besides, the class provides the following data structures:
- A list of features - features are stored in the breadth-first order - access via
getBfFeatures()
- A list of relationships - access via
getRelationships()
- A list of cross-tree constraints - access via
getConstraints()
- A feature model tree - access via the method
getRoot()
, which returns the tree’s root.
Feature
class
Besides the basic information of a feature, e.g., name
, id
, the Feature
class contains a connection to the parent feature,
and a list of relationships starting from it.
The structure allows for the construction of a feature model tree.
Generic classes for relationships
The package provides an abstract class AbstractRelationship
and four inherited classes MandatoryRelationship
,
OptionalRelationship
, OrRelationship
, and AlternativeRelationship
to represent relationships of a basic feature model.
The classes are generic and accept features with the type of Feature
or Feature
’s subclasses.
Besides, each relationship contains a connection to the parent feature and a list of child features.
CTConstraint
class
The CTConstraint
class represents a cross-tree constraint of a feature model.
By supporting Abstract Syntax Trees (ASTs), the class allows the representation of complex constraints,
which are supported by the common feature model formats.
Configuration rule translator
When creating/initializing a relationship or a cross-tree constraint, a configuration rule translator
is required to translate the relationship or constraint into a text-based configuration rule.
For example, the built-in translator ConfRuleTranslator
translates a mandatory relationship
into the configuration rule mandatory(parent, child)
.
The AbstractRelationship
and CTConstraint
classes accept custom translators that conform to the interface IConfRuleTranslatable
.
Feature, Relationship and Constraint Builders
The FeatureModel
class is generic and doesn’t know how to build features, relationships, and constraints.
Therefore, it requires builders that are able to create features, relationships, and constraints.
These builders need to conform to the interfaces IFeatureBuildable
, IRelationshipBuildable
, and IConstraintBuildable
.
The package includes the following built-in builders:
FeatureBuilder
can create a root feature or a not-root feature with a givenname
andid
RelationshipBuilder
allows to create four types of relationships, including mandatory, optional, or and alternative.ConstraintBuilder
is able to create a complex cross-tree constraint
Parsers
fm-v2 currently provides five parsers corresponding to five following feature model formats:
SXFMParser
supports [SPLOT] feature models. The file extension could be “.sxfm” or “.splx”.FeatureIDEParser
supports the feature model format of the [FeatureIDE] tool. The file extension should be “xml”.XMIParser
can parser the feature model format of the v.control tool. The file extension should be “xmi”.GLENCOEParser
supports the feature model format of the Glencoe tool. The file extension should be “json.”DescriptiveFormatParser
supports our feature model format. The file extension should be “fm4conf”.
You can find examples of the above feature model formats in the resources folder.
The package also provides the FMParserFactory
class that can create a parser for a given feature model file on the basis of the file extension as well as the file content.
Descriptive format
Our feature model format is a text-based format that allows us to represent a basic feature model in the human-readable fashion.
The feature model file should contain the following sections:
- A header line that mentions the version of the format
- A model name section starting with a “MODEL:” line that contains the name of the feature model
- A features section starting with a “FEATURES:” line that contains the list of features, each feature in a new line
- A relationships section starting with a “RELATIONSHIPS:” line that contains the list of relationships, each relationship in a new line
- A constraints section starting with a “CONSTRAINTS:” line that contains the list of cross-tree constraints, each constraint in a new line
An example of the feature model file in the descriptive format is available in the resources folder.
How tos
Reading a feature model from a file
The following code shows the simplest way to read a feature model from a file. This way, the parser will use the built-in builders to create the feature model.
1
2
3
4
5
6
7
8
9
10
11
// create a parser for the given file
// getIntance returns a parser that uses the built-in builders
FMParserFactory<Feature, AbstractRelationship<Feature>, CTConstraint>
factory = FMParserFactory.getInstance();
@Cleanup("dispose")
FeatureModelParser<Feature, AbstractRelationship<Feature>, CTConstraint>
parser = factory.getParser("filename");
// parse the feature model file
FeatureModel<Feature, AbstractRelationship<Feature>, CTConstraint>
fm = parser.parse("path/to/file");
Another example shows a complete way to read a feature model from a given file.
First, we need to create builders on the basis of the built-in builders.
Then, we can use the builders to create an FMParserFactory
.
We can create a parser for a given feature model format with the factory.
Finally, we can use the parser to read the feature model from a file.
In this way, if you want to construct a feature model with your custom features, relationships, and constraints,
you only need to provide your custom builders at the first step.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
// create builders for features, relationships, and constraints using built-in builders
IFeatureBuildable featureBuilder = new FeatureBuilder();
IConfRuleTranslatable ruleTranslator = new ConfRuleTranslator();
IRelationshipBuildable relationshipBuilder = new RelationshipBuilder(ruleTranslator);
IConstraintBuildable constraintBuilder = new ConstraintBuilder(ruleTranslator);
// create a parser for the given file
FMParserFactory<Feature, AbstractRelationship<Feature>, CTConstraint>
factory = FMParserFactory.getInstance(featureBuilder, relationshipBuilder, constraintBuilder);
@Cleanup("dispose")
FeatureModelParser<Feature, AbstractRelationship<Feature>, CTConstraint>
parser = factory.getParser("filename");
// parse the feature model file
FeatureModel<Feature, AbstractRelationship<Feature>, CTConstraint>
fm = parser.parse("path/to/file");
The below code shows an example in which we read an anomaly-aware feature model from a file. Particularly, the feature builder is customized to create anomaly-aware features.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
// CREATE A CUSTOM BUILDER FOR FEATURES
IFeatureBuildable featureBuilder = new AnomalyAwareFeatureBuilder();
// create built-in builders for relationships and constraints
IConfRuleTranslatable ruleTranslator = new ConfRuleTranslator();
IRelationshipBuildable relationshipBuilder = new RelationshipBuilder(ruleTranslator);
IConstraintBuildable constraintBuilder = new ConstraintBuilder(ruleTranslator);
// create a parser for the given file
FMParserFactory<AnomalyAwareFeature, AbstractRelationship<AnomalyAwareFeature>, CTConstraint>
factory = FMParserFactory.getInstance(featureBuilder, relationshipBuilder, constraintBuilder);
@Cleanup("dispose")
FeatureModelParser<AnomalyAwareFeature, AbstractRelationship<AnomalyAwareFeature>, CTConstraint>
parser = factory.getParser("filename");
// parse the feature model file
FeatureModel<AnomalyAwareFeature, AbstractRelationship<AnomalyAwareFeature>, CTConstraint>
fm = parser.parse("path/to/file");
For more details related to anomaly-aware feature model and AnomalyAwareFeatureBuilder
, please refer to CECore’s Feature Model Analysis.
Example
Unit tests of 5 parsers, function processFM in
KBStatistics
, andFMAnalyzerTest
Encoding a feature model
The following code shows how to build a feature model from scratch.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
// create builders for features, relationships, and constraints using built-in builders
IFeatureBuildable featureBuilder = new FeatureBuilder();
IConfRuleTranslatable ruleTranslator = new ConfRuleTranslator();
IRelationshipBuildable relationshipBuilder = new RelationshipBuilder(ruleTranslator);
IConstraintBuildable constraintBuilder = new ConstraintBuilder(ruleTranslator);
// create an object for the feature model
FeatureModel<Feature, AbstractRelationship<Feature>, CTConstraint> fm = new FeatureModel<>("test", featureBuilder, relationshipBuilder, constraintBuilder);
// create the root feature
Feature root = fm.addRoot("survey", "survey");
// create other features
// the order of adding features should be in the breadth-first order
Feature pay = fm.addFeature("pay", "pay");
Feature ABtesting = fm.addFeature("ABtesting", "ABtesting");
Feature statistics = fm.addFeature("statistics", "statistics");
Feature qa = fm.addFeature("qa", "qa");
Feature license = fm.addFeature("license", "license");
Feature nonlicense = fm.addFeature("nonlicense", "nonlicense");
Feature multiplechoice = fm.addFeature("multiplechoice", "multiplechoice");
Feature singlechoice = fm.addFeature("singlechoice", "singlechoice");
// create relationships
fm.addMandatoryRelationship(root, pay);
fm.addOptionalRelationship(root, ABtesting);
fm.addMandatoryRelationship(root, statistics);
fm.addMandatoryRelationship(root, qa);
fm.addAlternativeRelationship(pay, List.of(license, nonlicense));
fm.addOrRelationship(qa, List.of(multiplechoice, singlechoice));
fm.addOptionalRelationship(ABtesting, statistics);
// create constraints
fm.addRequires(ABtesting, statistics);
fm.addExcludes(ABtesting, nonlicense);
fm.addExcludes(ABtesting, root);
Using the generic feature model with your custom features, relationships, and constraints
There are two requirements to use the generic feature model:
- The custom features, relationships, and cross-tree constraints must be inherited from the base classes, i.e.,
Feature
,AbstractRelationship
, andCTConstraint
. - Custom feature, relationship, and constraint builders must be provided to the feature model.
These classes are inherited from the base classes, i.e.,
IFeatureBuildable
,IRelationshipBuildable
, andIConstraintBuildable
.
The following example shows how to use the generic feature model to support anomaly-aware features.
1
2
3
4
5
6
7
8
9
10
11
12
File fileFM = new File("src/test/resources/bamboobike_featureide_deadfeature2.xml");
// create the factory for anomaly feature models
IFeatureBuildable featureBuilder = new AnomalyAwareFeatureBuilder();
FMParserFactory<AnomalyAwareFeature, AbstractRelationship<AnomalyAwareFeature>, CTConstraint>
factory = FMParserFactory.getInstance(featureBuilder);
@Cleanup("dispose")
FeatureModelParser<AnomalyAwareFeature, AbstractRelationship<AnomalyAwareFeature>, CTConstraint>
parser = factory.getParser(fileFM.getName());
FeatureModel<AnomalyAwareFeature, AbstractRelationship<AnomalyAwareFeature>, CTConstraint>
featureModel = parser.parse(fileFM);
In this context, two new classes are defined.
The class AnomalyAwareFeature
is inherited from the base class Feature
,
and the class AnomalyAwareFeatureBuilder
is inherited from the base class IFeatureBuildable
.
For more details related to anomaly-aware feature model, please refer to CECore’s Feature Model Analysis.