X-Git-Url: http://git.salome-platform.org/gitweb/?a=blobdiff_plain;f=src%2FModel%2FModel_Validator.cpp;h=f7a58ddb68058b0100fd161fb011e5892e04097e;hb=93d261c062c12388f4420043776e8ee19ddceaa0;hp=fe7110924634b04434eedb72f2e73e0f4de07ddb;hpb=0918097e4f9023a2d60866cbd272bc66a745d195;p=modules%2Fshaper.git diff --git a/src/Model/Model_Validator.cpp b/src/Model/Model_Validator.cpp index fe7110924..f7a58ddb6 100644 --- a/src/Model/Model_Validator.cpp +++ b/src/Model/Model_Validator.cpp @@ -1,94 +1,417 @@ +// Copyright (C) 2014-20xx CEA/DEN, EDF R&D + // File: Model_Validator.cpp // Created: 2 Jul 2014 // Author: Mikhail PONIKAROV -#include +#include "Model_Validator.h" + +#include "Model_AttributeValidator.h" +#include "Model_FeatureValidator.h" + +#include +#include +#include #include -#include +#include +#include -using namespace std; +#include -void Model_ValidatorsFactory::registerValidator( - const string& theID, ModelAPI_Validator* theValidator) +void Model_ValidatorsFactory::registerValidator(const std::string& theID, + ModelAPI_Validator* theValidator) { if (myIDs.find(theID) != myIDs.end()) { - Events_Error::send(string("Validator ") + theID + " is already registered"); + Events_InfoMessage("Model_Validator", "Validator %1 is already registered").arg(theID).send(); } else { myIDs[theID] = theValidator; } } -void Model_ValidatorsFactory::assignValidator(const string& theID, const string& theFeatureID) +void Model_ValidatorsFactory::assignValidator(const std::string& theID, + const std::string& theFeatureID) { - bool isError = false; - map::iterator aVal = myIDs.find(theID); - if (aVal == myIDs.end()) { - Events_Error::send( - string("Validator ") + theID + " for feature " + theFeatureID + " was not registered"); - isError = true; + if (myFeatures.find(theFeatureID) == myFeatures.end()) { + myFeatures[theFeatureID] = AttrValidators(); } - if (myFeatures.find(theFeatureID) != myFeatures.end()) { - Events_Error::send( - string("Validator for feature ") + theFeatureID + " is already registered"); - isError = true; + if (myFeatures[theFeatureID].find(theID) != myFeatures[theFeatureID].end()) { + //Events_Error::send(std::string("Validator ") + theID + + // " for feature " + theFeatureID + "is already registered"); + } else { + myFeatures[theFeatureID][theID] = std::list(); } - if (!isError) - myFeatures[theFeatureID] = aVal->second; } -void Model_ValidatorsFactory::assignValidator(const string& theID, - const string& theFeatureID, const string& theAttrID, const list& theArguments) +void Model_ValidatorsFactory::assignValidator(const std::string& theID, + const std::string& theFeatureID, + const std::list& theArguments) { - bool isError = false; - map::iterator aVal = myIDs.find(theID); - if (aVal == myIDs.end()) { - Events_Error::send( - string("Validator ") + theID + " for feature " + theFeatureID + " was not registered"); - isError = true; + if (myFeatures.find(theFeatureID) == myFeatures.end()) { + myFeatures[theFeatureID] = AttrValidators(); + } + + if (myFeatures[theFeatureID].find(theID) != myFeatures[theFeatureID].end()) { + //Events_Error::send(std::string("Validator ") + theID + + // " for feature " + theFeatureID + "is already registered"); + } else { + myFeatures[theFeatureID][theID] = theArguments; } +} + +void Model_ValidatorsFactory::assignValidator(const std::string& theID, + const std::string& theFeatureID, + const std::string& theAttrID, + const std::list& theArguments) +{ // create feature-structures if not exist - map > > >::iterator - aFeature = myAttrs.find(theFeatureID); + std::map >::iterator aFeature = myAttrs.find( + theFeatureID); if (aFeature == myAttrs.end()) { - myAttrs[theFeatureID] = map > >(); + myAttrs[theFeatureID] = std::map(); aFeature = myAttrs.find(theFeatureID); } // add attr-structure if not exist, or generate error if already exist - map > >::iterator - anAttr = aFeature->second.find(theAttrID); + std::map::iterator anAttr = aFeature->second.find(theAttrID); if (anAttr == aFeature->second.end()) { - if (!isError) { - aFeature->second[theAttrID] = - pair >(aVal->second, theArguments); + aFeature->second[theAttrID] = AttrValidators(); + } + aFeature->second[theAttrID][theID] = theArguments; +} + +void Model_ValidatorsFactory::validators(const std::string& theFeatureID, + Validators& theValidators) const +{ + std::map::const_iterator aFeatureIt = + myFeatures.find(theFeatureID); + if (aFeatureIt != myFeatures.cend()) { + AttrValidators::const_iterator aValidatorsIt = aFeatureIt->second.cbegin(); + for (; aValidatorsIt != aFeatureIt->second.cend(); aValidatorsIt++) { + if (!validator(aValidatorsIt->first)) { + Events_InfoMessage("Model_Validator", "Validator %1 was not registered") + .arg(aValidatorsIt->first).send(); + } else { + theValidators.push_back(std::make_pair(aValidatorsIt->first, aValidatorsIt->second)); + } + } + } + addDefaultValidators(theValidators); +} + +void Model_ValidatorsFactory::validators(const std::string& theFeatureID, const std::string& theAttrID, + Validators& theValidators) const +{ + std::map >::const_iterator aFeatureIt = + myAttrs.find(theFeatureID); + if (aFeatureIt != myAttrs.cend()) { + std::map::const_iterator anAttrIt = aFeatureIt->second.find(theAttrID); + if (anAttrIt != aFeatureIt->second.end()) { + AttrValidators::const_iterator aValidatorsIt = anAttrIt->second.cbegin(); + for (; aValidatorsIt != anAttrIt->second.cend(); aValidatorsIt++) { + if (!validator(aValidatorsIt->first)) { + Events_InfoMessage("Model_Validator", "Validator %1 was not registered") + .arg(aValidatorsIt->first).send(); + } else { + theValidators.push_back(std::make_pair(aValidatorsIt->first, aValidatorsIt->second)); + } + } + } + } + addDefaultAttributeValidators(theValidators); +} + +Model_ValidatorsFactory::Model_ValidatorsFactory() + : ModelAPI_ValidatorsFactory() +{ + registerValidator("Model_FeatureValidator", new Model_FeatureValidator); + registerValidator("Model_AttributeValidator", new Model_AttributeValidator); +} + +const ModelAPI_Validator* Model_ValidatorsFactory::validator(const std::string& theID) const +{ + std::map::const_iterator aIt = myIDs.find(theID); + if (aIt != myIDs.end()) { + return aIt->second; + } + return NULL; +} + +void Model_ValidatorsFactory::addDefaultValidators(Validators& theValidators) const +{ + const static std::string kDefaultId = "Model_FeatureValidator"; + if (!validator(kDefaultId)) + return; + theValidators.push_back(std::make_pair(kDefaultId, std::list())); +} + +void Model_ValidatorsFactory::addDefaultAttributeValidators(Validators& theValidators) const +{ + const static std::string kDefaultId = "Model_AttributeValidator"; + if (!validator(kDefaultId)) + return; + theValidators.push_back(std::make_pair(kDefaultId, std::list())); +} + +bool Model_ValidatorsFactory::validate(const std::shared_ptr& theFeature) const +{ + std::shared_ptr aData = std::dynamic_pointer_cast(theFeature->data()); + if (aData.get() && aData->isValid()) { + if (aData->execState() == ModelAPI_StateDone) + aData->eraseErrorString(); // no error => erase the information string + } else { + return false; // feature is broken, already not presented in the data structure + } + + // check feature validators first + Validators aValidators; + validators(theFeature->getKind(), aValidators); + + if (!aValidators.empty()) { + Validators::const_iterator aValidatorIt = aValidators.cbegin(); + for(; aValidatorIt != aValidators.cend(); aValidatorIt++) { + const std::string& aValidatorID = aValidatorIt->first; + const std::list& anArguments = aValidatorIt->second; + // validators() checks invalid validator names + //if (!aValidator) { + // Events_Error::send(std::string("Validator ") + aValidatorID + " was not registered"); + // continue; + //} + const ModelAPI_FeatureValidator* aFValidator = + dynamic_cast(validator(aValidatorID)); + if (aFValidator) { + Events_InfoMessage anError; + if (!aFValidator->isValid(theFeature, anArguments, anError)) { + if (anError.empty()) + anError = "Unknown error."; + anError = aValidatorID + ": " + anError.messageString(); + theFeature->setError(anError.messageString(), false); + theFeature->data()->execState(ModelAPI_StateInvalidArgument); + return false; + } + } + } + } + // The default validator was retrned by validators() and was checked in previous cycle + //// check default validator + //std::map::const_iterator aDefaultVal = myIDs.find(kDefaultId); + //if(aDefaultVal != myIDs.end()) { + // static const std::list anEmptyArgList; + // const ModelAPI_FeatureValidator* aFValidator = + // dynamic_cast(aDefaultVal->second); + // if (aFValidator) { + // std::string anError; + // if (!aFValidator->isValid(theFeature, anEmptyArgList, anError)) { + // if (anError.empty()) + // anError = "Unknown error."; + // anError = "Feature invalidated by \"" + kDefaultId + "\" with error: " + anError; + // theFeature->setError(anError, false); + // theFeature->data()->execState(ModelAPI_StateInvalidArgument); + // return false; + // } + // } + //} + + // check all attributes for validity + // Validity of data is checked by "Model_FeatureValidator" (kDefaultId) + // if (!aData || !aData->isValid()) + // return false; + static const std::string kAllTypes = ""; + std::list aLtAttributes = aData->attributesIDs(kAllTypes); + std::list::const_iterator anAttrIt = aLtAttributes.cbegin(); + for (; anAttrIt != aLtAttributes.cend(); anAttrIt++) { + const std::string& anAttributeID = *anAttrIt; + AttributePtr anAttribute = theFeature->data()->attribute(anAttributeID); + + std::string aValidatorID; + Events_InfoMessage anError; + if (!validate(anAttribute, aValidatorID, anError)) { + if (anError.empty()) + anError = "Unknown error."; + anError = anAttributeID + " - " + aValidatorID + ": " + anError.messageString(); + theFeature->setError(anError.messageString(), false); + theFeature->data()->execState(ModelAPI_StateInvalidArgument); + return false; + } + } + + return true; +} + +bool Model_ValidatorsFactory::validate(const std::shared_ptr& theAttribute, + std::string& theValidator, + Events_InfoMessage& theError) const +{ + FeaturePtr aFeature = ModelAPI_Feature::feature(theAttribute->owner()); + if (!aFeature.get()) { + theValidator = "Model_ValidatorsFactory"; + theError = "Attribute has no feature."; + return false; + } + + // skip not-case attributes, that really may be invalid (issue 671) + if (!const_cast(this)->isCase(aFeature, theAttribute->id())) + return true; + + Validators aValidators; + validators(aFeature->getKind(), theAttribute->id(), aValidators); + + Validators::iterator aValidatorIt = aValidators.begin(); + for (; aValidatorIt != aValidators.end(); ++aValidatorIt) { + const std::string& aValidatorID = aValidatorIt->first; + const std::list& anArguments = aValidatorIt->second; + const ModelAPI_AttributeValidator* anAttrValidator = + dynamic_cast(validator(aValidatorID)); + if (!anAttrValidator) + continue; + if (!anAttrValidator->isValid(theAttribute, anArguments, theError)) { + theValidator = aValidatorID; + return false; + } + } + + return true; +} + +void Model_ValidatorsFactory::registerNotObligatory(std::string theFeature, std::string theAttribute) +{ + const static std::string kDefaultId = "Model_FeatureValidator"; + std::map::const_iterator it = myIDs.find(kDefaultId); + if (it != myIDs.end()) { + Model_FeatureValidator* aValidator = dynamic_cast(it->second); + if (aValidator) { + aValidator->registerNotObligatory(theFeature, theAttribute); } + } +} + +bool Model_ValidatorsFactory::isNotObligatory(std::string theFeature, std::string theAttribute) +{ + const static std::string kDefaultId = "Model_FeatureValidator"; + std::map::const_iterator it = myIDs.find(kDefaultId); + if (it != myIDs.end()) { + Model_FeatureValidator* aValidator = dynamic_cast(it->second); + if (aValidator) { + return aValidator->isNotObligatory(theFeature, theAttribute); + } + } + return false; // default +} + +void Model_ValidatorsFactory::registerConcealment(std::string theFeature, std::string theAttribute) +{ + std::map >::iterator aFind = myConcealed.find(theFeature); + if (aFind == myConcealed.end()) { + std::set aNewSet; + aNewSet.insert(theAttribute); + myConcealed[theFeature] = aNewSet; } else { - Events_Error::send( - string("Validator ") + theID + " for feature " + theFeatureID + - "attribute " + theAttrID + " is already registered"); - isError = true; + aFind->second.insert(theAttribute); + } +} + +bool Model_ValidatorsFactory::isConcealed(std::string theFeature, std::string theAttribute) +{ + std::map >::iterator aFind = myConcealed.find(theFeature); + return aFind != myConcealed.end() && aFind->second.find(theAttribute) != aFind->second.end(); +} + +void Model_ValidatorsFactory::registerUnconcealment(std::shared_ptr theUnconcealed, + std::shared_ptr theCanceledFeat) +{ + if (myUnconcealed.find(theUnconcealed) == myUnconcealed.end()) { + myUnconcealed[theUnconcealed] = std::list >(); + } + myUnconcealed[theUnconcealed].push_back(theCanceledFeat); + std::dynamic_pointer_cast(theUnconcealed->data())->updateConcealmentFlag(); +} + +void Model_ValidatorsFactory::disableUnconcealment(std::shared_ptr theUnconcealed, + std::shared_ptr theCanceledFeat) +{ + std::map, std::list > > + ::iterator aResFound = myUnconcealed.find(theUnconcealed); + if (aResFound != myUnconcealed.end()) { + std::list >::iterator anIter = aResFound->second.begin(); + for(; anIter != aResFound->second.end(); anIter++) { + if (*anIter == theCanceledFeat) { + aResFound->second.erase(anIter); + std::dynamic_pointer_cast(theUnconcealed->data())->updateConcealmentFlag(); + break; + } + } } } -const ModelAPI_Validator* Model_ValidatorsFactory::validator(const string& theFeatureID) const +bool Model_ValidatorsFactory::isUnconcealed(std::shared_ptr theUnconcealed, + std::shared_ptr theCanceledFeat) { - map::const_iterator aFeature = myFeatures.find(theFeatureID); - if (aFeature != myFeatures.cend()) - return aFeature->second; - return NULL; // not found + std::map, std::list > > + ::iterator aResFound = myUnconcealed.find(theUnconcealed); + if (aResFound != myUnconcealed.end()) { + std::list >::iterator aFeatIter = aResFound->second.begin(); + for(; aFeatIter != aResFound->second.end(); aFeatIter++) { + if (aFeatIter->get()) { + if ((*aFeatIter)->isDisabled()) continue; + if (*aFeatIter == theCanceledFeat) + return true; // this is exactly canceled + if (theCanceledFeat->document()->isLater(*aFeatIter, theCanceledFeat)) + return true; // if unconcealed feature (from the list) is later than concealed + } else + return true; // empty attribute means that everything is canceled + } + } + return false; } -bool Model_ValidatorsFactory::validate( - const boost::shared_ptr& theFeature, const string& theAttrID ) const +void Model_ValidatorsFactory::registerCase(std::string theFeature, std::string theAttribute, + const std::list >& theCases) { - map > > >::const_iterator - aFeature = myAttrs.find(theFeature->getKind()); - if (aFeature == myAttrs.cend()) return true; // feature is not found - map > >::const_iterator - anAttr = aFeature->second.find(theAttrID); - if (anAttr == aFeature->second.cend()) return true; // attribute is not found - return anAttr->second.first->validate(theFeature, theAttrID, anAttr->second.second); + std::map > > > + ::iterator aFindFeature = myCases.find(theFeature); + if (aFindFeature == myCases.end()) { + myCases[theFeature] = std::map > >(); + aFindFeature = myCases.find(theFeature); + } + std::map > >::iterator aFindAttrID = + aFindFeature->second.find(theAttribute); + + if (aFindAttrID == aFindFeature->second.end()) { + aFindFeature->second[theAttribute] = + std::map >(); + aFindAttrID = aFindFeature->second.find(theAttribute); + } + std::list >::const_iterator aCasesIt = theCases.begin(), + aCasesLast = theCases.end(); + std::map > aFindCases = aFindAttrID->second; + for (; aCasesIt != aCasesLast; aCasesIt++) { + std::pair aCasePair = *aCasesIt; + std::string aSwitch = aCasePair.first; + if (aFindAttrID->second.find(aSwitch) == aFindAttrID->second.end()) { + aFindAttrID->second[aSwitch] = std::set(); + } + aFindAttrID->second[aSwitch].insert(aCasePair.second); + } } -Model_ValidatorsFactory::Model_ValidatorsFactory() : ModelAPI_ValidatorsFactory() +bool Model_ValidatorsFactory::isCase(FeaturePtr theFeature, std::string theAttribute) { + bool anInCase = true; + std::map > > > + ::iterator aFindFeature = myCases.find(theFeature->getKind()); + if (aFindFeature != myCases.end()) { + std::map > >::iterator + aFindAttrID = aFindFeature->second.find(theAttribute); + if (aFindAttrID != aFindFeature->second.end()) { + std::map >::iterator + aCasesIt = aFindAttrID->second.begin(), aCasesLast = aFindAttrID->second.end(); + for (; aCasesIt != aCasesLast && anInCase; aCasesIt++) { + // the the switch-attribute that contains the case value + AttributeStringPtr aSwitch = theFeature->string(aCasesIt->first); + if (aSwitch.get()) { + // the second has the case identifier + anInCase = aCasesIt->second.find(aSwitch->value()) != aCasesIt->second.end(); + } + } + } + } + return anInCase; // if no additional conditions, this attribute is the case to be validated }