configurationengine/source/cone/validation/builtinvalidators/confml.py
author m2lahtel
Tue, 10 Aug 2010 14:29:28 +0300
changeset 3 e7e0ae78773e
permissions -rw-r--r--
ConE 1.2.11 release
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
     1
#
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
     2
# Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
     3
# All rights reserved.
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
     4
# This component and the accompanying materials are made available
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
     5
# under the terms of "Eclipse Public License v1.0"
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
     6
# which accompanies this distribution, and is available
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
     7
# at the URL "http://www.eclipse.org/legal/epl-v10.html".
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
     8
#
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
     9
# Initial Contributors:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    10
# Nokia Corporation - initial contribution.
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    11
#
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    12
# Contributors:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    13
#
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    14
# Description: 
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    15
#
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    16
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    17
import logging
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    18
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    19
from cone.public import api, exceptions, container, utils
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    20
from cone.confml import model
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    21
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    22
from cone.validation.confmlvalidation import ValidatorBase, FixerBase
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    23
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    24
class SettingValidatorBase(ValidatorBase):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    25
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    26
    Base class for validators that validate ConfML settings
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    27
    (sub-classes of cone.confml.model.ConfmlSetting).
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    28
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    29
    def validate(self):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    30
        for ref, feature in self.context.feature_dict.iteritems():
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    31
            if isinstance(feature._obj, model.ConfmlSetting):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    32
                self.validate_setting(ref, feature)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    33
    
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    34
    def validate_setting(self, ref, setting):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    35
        raise NotImplementedError()
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    36
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    37
class LengthConstraintValidator(SettingValidatorBase):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    38
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    39
    Validator for validating xs:length, xs:minLength and xs:maxLength
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    40
    constraints on setting data values.
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    41
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    42
    PROBLEM_TYPES = ['model.confml.invalid_value.length',
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    43
                     'model.confml.invalid_value.minlength',
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    44
                     'model.confml.invalid_value.maxlength']
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    45
    
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    46
    def validate_setting(self, ref, setting):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    47
        if setting.length:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    48
            value = setting.get_value()
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    49
            if isinstance(value, basestring) and len(value) != setting.length:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    50
                self._add_problem(
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    51
                    setting = setting,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    52
                    msg = "Setting %s: Exact number of characters must be %s (value has %s)" % (ref, setting.length, len(value)),
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    53
                    prob_type = self.PROBLEM_TYPES[0])
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    54
                
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    55
        if setting.minLength:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    56
            value = setting.get_value()
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    57
            if isinstance(value, basestring) and len(value) < setting.minLength:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    58
                self._add_problem(
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    59
                    setting = setting,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    60
                    msg = "Setting %s: Minimum number of characters is %s (value has %s)" % (ref, setting.minLength, len(value)),
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    61
                    prob_type = self.PROBLEM_TYPES[1])
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    62
        
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    63
        if setting.maxLength:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    64
            value = setting.get_value()
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    65
            if isinstance(value, basestring) and len(value) > setting.maxLength:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    66
                self._add_problem(
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    67
                    setting = setting,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    68
                    msg = "Setting %s: Maximum number of characters is %s (value has %s)" % (ref, setting.maxLength, len(value)),
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    69
                    prob_type = self.PROBLEM_TYPES[2])
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    70
   
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    71
    def _add_problem(self, setting, msg, prob_type):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    72
        dataobj = setting.datas['data'][-1]
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    73
        prob = api.Problem(
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    74
            msg = msg,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    75
            type = prob_type,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    76
            line = dataobj.lineno,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    77
            file = dataobj.get_configuration_path())
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    78
        self.context.problems.append(prob)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    79
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    80
class MissingFeatureForDataValidator(ValidatorBase):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    81
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    82
    Validator for validating data elements that do not have a
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    83
    corresponding feature/setting in the configuration.
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    84
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    85
    PROBLEM_TYPES = ['model.confml.missing_feature_for_data']
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    86
    
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    87
    def validate(self):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    88
        for dataobj in self.context.configuration._traverse(type=api.Data):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    89
            try:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    90
                self.context.dview.get_feature(dataobj.fqr)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    91
            except exceptions.NotFound:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    92
                prob = api.Problem(
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    93
                    msg = "Feature '%s' not found" % dataobj.fqr,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    94
                    type = self.PROBLEM_TYPES[0],
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    95
                    line = dataobj.lineno,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    96
                    file = dataobj.get_configuration_path())
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    97
                self.context.problems.append(prob)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    98
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
    99
class MissingDescriptionValidator(SettingValidatorBase):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   100
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   101
    Validator for validating missing descriptions in feature/setting in the configuration.
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   102
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   103
    PROBLEM_TYPES = ['model.confml.missing_desc']
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   104
    
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   105
    def validate_setting(self, ref, setting):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   106
        print 'Validating missing desc!'
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   107
        if not setting.desc or setting.desc == '':
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   108
            prob = api.Problem(
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   109
                msg = "Setting/Feature %s: has no description" % (ref),
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   110
                type = self.PROBLEM_TYPES[0],
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   111
                line = setting.lineno,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   112
                file = setting.get_configuration_path(),
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   113
                severity = api.Problem.SEVERITY_WARNING)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   114
            self.context.problems.append(prob)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   115
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   116
class DuplicateSettingValidator(ValidatorBase):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   117
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   118
    Validator for validating that there are no settings with same ref in given 
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   119
    configuration.
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   120
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   121
    PROBLEM_TYPES = ['model.confml.duplicate.setting']
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   122
    
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   123
    def validate(self):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   124
        settings_container = container.DataContainer()
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   125
        # Traverse through the configuration model and store each feature to 
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   126
        # the settings_container. 
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   127
        for setting in self.context.configuration._traverse(type=model.ConfmlSetting):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   128
            settings_container.add_value(setting.fqr, setting)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   129
        # Go though the settings_container to see if any features have more than one 
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   130
        # definition and report those as problems
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   131
        for fqr in settings_container.list_keys():
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   132
            if len(settings_container.get_values(fqr)) > 1:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   133
                files = [setting.get_configuration_path() for setting in settings_container.get_values(fqr)]
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   134
                prob = api.Problem(
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   135
                    msg = "Feature %s has '%s' definitions in files %s" % (fqr, len(settings_container.get_values(fqr)), files),
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   136
                    type = self.PROBLEM_TYPES[0],
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   137
                    severity = api.Problem.SEVERITY_WARNING, 
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   138
                    line = settings_container.get_value(fqr).lineno,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   139
                    file = files[-1],
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   140
                    problem_data = settings_container.get_values(fqr))
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   141
                self.context.problems.append(prob)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   142
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   143
class DuplicateFeatureValidator(ValidatorBase):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   144
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   145
    Validator for validating that there are no features with same ref in given 
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   146
    configuration.
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   147
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   148
    PROBLEM_TYPES = ['model.confml.duplicate.feature']
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   149
    
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   150
    def validate(self):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   151
        settings_container = container.DataContainer()
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   152
        # Traverse through the configuration model and store each feature to 
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   153
        # the settings_container. 
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   154
        for setting in self.context.configuration._traverse(type=model.ConfmlFeature):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   155
            settings_container.add_value(setting.fqr, setting)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   156
        # Go though the settings_container to see if any features have more than one 
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   157
        # definition and report those as problems
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   158
        for fqr in settings_container.list_keys():
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   159
            if len(settings_container.get_values(fqr)) > 1:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   160
                files = [setting.get_configuration_path() for setting in settings_container.get_values(fqr)]
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   161
                prob = api.Problem(
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   162
                    msg = "Feature %s has '%s' definitions in files %s" % (fqr, len(settings_container.get_values(fqr)), files),
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   163
                    type = self.PROBLEM_TYPES[0],
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   164
                    severity = api.Problem.SEVERITY_INFO,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   165
                    line = settings_container.get_value(fqr).lineno,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   166
                    file = files[-1],
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   167
                    problem_data = settings_container.get_values(fqr))
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   168
                self.context.problems.append(prob)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   169
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   170
class DuplicateSettingFixer(FixerBase):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   171
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   172
    A Fix class for duplicate settings that removes all but the last definition of the element.
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   173
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   174
    PROBLEM_TYPES = ['model.confml.duplicate.setting']
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   175
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   176
    def fix(self, context):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   177
        for problem in self.filter_problems(context.problems, self.PROBLEM_TYPES[0]):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   178
            logging.getLogger('cone.validation').info("Fixing problem %s" % problem.msg)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   179
            context.fixes.append("Fixed problem: %s" % problem.msg)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   180
            # The problem data is expected to have those duplicate settings and the 
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   181
            # actual setting as a last element
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   182
            for setting in problem.problem_data[0:-1]:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   183
                parent_fea = setting.find_parent(type=api.Feature)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   184
                logging.getLogger('cone.validation').info("Remove setting %s from %s" % (setting.fqr, parent_fea.get_configuration_path()))
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   185
                try:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   186
                    parent_fea.remove_feature(setting.ref)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   187
                except exceptions.NotFound:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   188
                    logging.getLogger('cone.validation').info("Already removed %s from %s" % (setting.ref, parent_fea.get_configuration_path()))
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   189
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   190
class DuplicateFeatureFixer(FixerBase):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   191
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   192
    A Fix class for duplicate features that merges all setting under a duplicate feature 
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   193
    to the first instance of the feature and removes the duplicates.
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   194
    """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   195
    PROBLEM_TYPES = ['model.confml.duplicate.feature']
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   196
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   197
    def fix(self, context):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   198
        for problem in self.filter_problems(context.problems, self.PROBLEM_TYPES[0]):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   199
            logging.getLogger('cone.validation').info("Fixing problem %s" % problem.msg)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   200
            context.fixes.append("Fixed problem: %s" % problem.msg)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   201
            # The problem data is expected to have those duplicate settings and the 
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   202
            # actual setting as a last element
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   203
            target_feature = problem.problem_data[0]
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   204
            target_config = target_feature.find_parent(type=api.Configuration)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   205
            for feature in problem.problem_data[1:]:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   206
                logging.getLogger('cone.validation').info("Move settings from Feature %s in %s to %s" % \
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   207
                                                          (feature.fqr, feature.get_configuration_path(), target_feature.get_configuration_path()))
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   208
                for setting_ref in feature.list_features():
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   209
                    setting = feature.get_feature(setting_ref)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   210
                    # Get the path from feature to the parent of this setting
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   211
                    # (pathto,ref) = utils.dottedref.psplit_ref(setting_ref)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   212
                    if target_feature.has_ref(setting_ref):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   213
                        target_feature.remove_feature(setting_ref)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   214
                    target_feature.add_feature(setting)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   215
                    
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   216
                config = feature.find_parent(type=api.Configuration)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   217
                logging.getLogger('cone.validation').info("Remove feature %s from %s" % (feature.fqr, config.get_full_path()))
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   218
                config.remove_feature(feature.ref)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   219
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   220
                
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   221
#: List of all built-in ConfML validator classes
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   222
VALIDATOR_CLASSES = [
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   223
    MissingFeatureForDataValidator,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   224
    LengthConstraintValidator,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   225
    DuplicateSettingValidator,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   226
    DuplicateFeatureValidator,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   227
#    MissingDescriptionValidator,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   228
]
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   229
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   230
#: List of all built-in ConfML fixer classes
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   231
FIXER_CLASSES = [
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   232
    DuplicateFeatureFixer,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents:
diff changeset
   233
]