com.gs.dmn.validation.DefaultDMNValidator Maven / Gradle / Ivy
/*
* Copyright 2016 Goldman Sachs.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License.
*
* You may obtain a copy of the License at
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the
* specific language governing permissions and limitations under the License.
*/
package com.gs.dmn.validation;
import com.gs.dmn.DMNModelRepository;
import com.gs.dmn.QualifiedName;
import com.gs.dmn.ast.*;
import com.gs.dmn.log.BuildLogger;
import com.gs.dmn.log.Slf4jBuildLogger;
import com.gs.dmn.transformation.AbstractDMNToNativeTransformer;
import org.apache.commons.lang3.StringUtils;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.function.Function;
public class DefaultDMNValidator extends SimpleDMNValidator {
public DefaultDMNValidator() {
super(new Slf4jBuildLogger(LOGGER));
}
public DefaultDMNValidator(BuildLogger logger) {
super(logger);
}
@Override
public List validate(DMNModelRepository repository) {
List errors = new ArrayList<>();
if (isEmpty(repository)) {
logger.warn("DMN repository is empty; validator will not run");
return errors;
}
for (TDefinitions definitions: repository.getAllDefinitions()) {
logger.debug("Validate unique 'DRGElement.id'");
validateUnique(
repository, definitions, new ArrayList<>(repository.findDRGElements(definitions)), "DRGElement", "id", false,
TDMNElement::getId, null, errors
);
logger.debug("Validate unique 'DRGElement.name'");
validateUnique(
repository, definitions, new ArrayList<>(repository.findDRGElements(definitions)), "DRGElement", "name", false,
e -> ((TNamedElement)e).getName(), null, errors
);
logger.debug("Validate unique 'ItemDefinition.name'");
validateUnique(
repository, definitions, new ArrayList<>(repository.findItemDefinitions(definitions)), "ItemDefinition", "name", false,
e -> ((TNamedElement)e).getName(), null, errors
);
for (TDRGElement element : repository.findDRGElements(definitions)) {
logger.debug(String.format("Validate element '%s'", element.getName()));
if (element instanceof TInputData) {
validateInputData(repository, definitions, (TInputData) element, errors);
} else if (element instanceof TBusinessKnowledgeModel) {
validateBusinessKnowledgeModel(repository, definitions, (TBusinessKnowledgeModel) element, errors);
} else if (element instanceof TDecision) {
validateDecision(repository, definitions, (TDecision) element, errors);
}
}
}
return errors;
}
private void validateUnique(DMNModelRepository repository, TDefinitions definitions, List extends TDMNElement> elements, String elementType, String property, boolean isOptionalProperty, Function accessor, String errorMessage, List errors) {
if (errorMessage == null) {
errorMessage = String.format("The '%s' of a '%s' must be unique.", property, elementType);
}
// Create a map
Map> map = new LinkedHashMap<>();
for (TDMNElement element : elements) {
String key = accessor.apply(element);
if (!isOptionalProperty || key != null) {
List list = map.get(key);
if (list == null) {
list = new ArrayList<>();
list.add(element);
map.put(key, list);
} else {
list.add(element);
}
}
}
// Find duplicates
List duplicates = new ArrayList<>();
for (Map.Entry> entry : map.entrySet()) {
String key = entry.getKey();
if(entry.getValue().size() > 1){
duplicates.add(key);
}
}
// Report error
if (!duplicates.isEmpty()) {
String message = String.join(", ", duplicates);
errors.add(makeError(repository, definitions, null, String.format("%s Found duplicates for '%s'.", errorMessage, message)));
}
}
private void validateInputData(DMNModelRepository repository, TDefinitions definitions, TInputData inputData, List errors) {
validateNamedElement(repository, definitions, inputData, errors);
validateVariable(repository, definitions, inputData, inputData.getVariable(), errors);
}
protected void validateBusinessKnowledgeModel(DMNModelRepository repository, TDefinitions definitions, TBusinessKnowledgeModel knowledgeModel, List errors) {
validateNamedElement(repository, definitions, knowledgeModel, errors);
validateVariable(repository, definitions, knowledgeModel, knowledgeModel.getVariable(), errors);
}
protected void validateDecision(DMNModelRepository repository, TDefinitions definitions, TDecision decision, List errors) {
validateNamedElement(repository, definitions, decision, errors);
TInformationItem variable = decision.getVariable();
validateVariable(repository, definitions, decision, variable, errors);
String decisionName = decision.getName();
if (variable != null) {
// decision/@name == decision/variable/@name
String variableName = variable.getName();
if (!decisionName.equals(variableName)) {
String errorMessage = String.format("Decision name and variable name should be the same. Found '%s' and '%s'", decisionName, variableName);
errors.add(makeError(repository, definitions, decision, errorMessage));
}
// decision/variable/@typeRef is not null
QualifiedName typeRef = QualifiedName.toQualifiedName(definitions, variable.getTypeRef());
if (repository.isNull(typeRef)) {
String errorMessage = "Missing typRef in variable";
errors.add(makeError(repository, definitions, decision, errorMessage));
}
} else {
String errorMessage = "Missing variable";
errors.add(makeError(repository, definitions, decision, errorMessage));
}
// Validate requirements
List informationRequirement = decision.getInformationRequirement();
Function accessor =
(TDMNElement e) -> {
TInformationRequirement ir = (TInformationRequirement) e;
if (ir.getRequiredInput() != null) {
return ir.getRequiredInput().getHref();
} else {
return ir.getRequiredDecision().getHref();
}
};
validateUnique(repository, definitions, informationRequirement, "TInformationRequirement", "href", false,
accessor, decision.getName(), errors);
// Validate expression
TExpression expression = decision.getExpression();
if (expression != null) {
validateExpression(repository, definitions, decision, expression, errors);
}
}
private void validateDecisionTable(DMNModelRepository repository, TDefinitions definitions, TDMNElement element, TDecisionTable decisionTable, List errors) {
List input = decisionTable.getInput();
if (input == null || input.isEmpty()) {
String errorMessage = "Missing input clauses";
errors.add(makeError(repository, definitions, element, errorMessage));
}
List output = decisionTable.getOutput();
if (output == null || output.isEmpty()) {
String errorMessage = "Missing output clauses";
errors.add(makeError(repository, definitions, element, errorMessage));
}
validateHitPolicy(repository, definitions, element, decisionTable, errors);
List ruleList = decisionTable.getRule();
if (ruleList == null || ruleList.isEmpty()) {
String errorMessage = "Missing rules in decision table";
errors.add(makeError(repository, definitions, element, errorMessage));
} else {
for (TDecisionRule rule : ruleList) {
validateRule(repository, definitions, element, rule, errors);
}
}
}
private void validateHitPolicy(DMNModelRepository repository, TDefinitions definitions, TDMNElement element, TDecisionTable decisionTable, List errors) {
List output = decisionTable.getOutput();
THitPolicy hitPolicy = decisionTable.getHitPolicy();
TBuiltinAggregator aggregation = decisionTable.getAggregation();
if (hitPolicy != THitPolicy.COLLECT && aggregation != null) {
String errorMessage = String.format("Aggregation '%s' not allowed for hit policy '%s'", aggregation, hitPolicy);
errors.add(makeError(repository, definitions, element, errorMessage));
}
if (output != null && output.size() > 1
&& hitPolicy == THitPolicy.COLLECT
&& aggregation != null) {
String errorMessage = String.format("Collect operator is not defined over multiple outputs for decision table '%s'", decisionTable.getId());
errors.add(makeError(repository, definitions, element, errorMessage));
}
}
private void validateNamedElement(DMNModelRepository repository, TDefinitions definitions, TNamedElement element, List errors) {
if (StringUtils.isBlank(element.getName())) {
String errorMessage = "Missing name";
errors.add(makeError(repository, definitions, element, errorMessage));
}
}
private void validateVariable(DMNModelRepository repository, TDefinitions definitions, TNamedElement element, TInformationItem variable, List errors) {
if (variable != null && variable.getName() == null) {
String errorMessage = "Missing variable name";
errors.add(makeError(repository, definitions, element, errorMessage));
}
}
private void validateExpression(DMNModelRepository repository, TDefinitions definitions, TDRGElement element, TExpression expression, List errors) {
if (expression == null) {
String errorMessage = "Missing expression";
errors.add(makeError(repository, definitions, element, errorMessage));
} else {
if (expression instanceof TDecisionTable) {
TDecisionTable decisionTable = (TDecisionTable) expression;
validateDecisionTable(repository, definitions, element, decisionTable, errors);
} else if (expression instanceof TInvocation) {
TInvocation invocation = (TInvocation) expression;
validateExpression(repository, definitions, element, invocation.getExpression(), errors);
} else if (expression instanceof TLiteralExpression) {
TLiteralExpression literalExpression = (TLiteralExpression) expression;
String expressionLanguage = ((TLiteralExpression) expression).getExpressionLanguage();
if (!isSupported(expressionLanguage)) {
String errorMessage = String.format("Not supported expression language '%s'", expressionLanguage);
errors.add(makeError(repository, definitions, element, errorMessage));
}
if (StringUtils.isBlank(literalExpression.getText())) {
String errorMessage = "Missing text in literal expressions";
errors.add(makeError(repository, definitions, element, errorMessage));
}
} else if (expression instanceof TContext) {
List contextEntryList = ((TContext) expression).getContextEntry();
if (contextEntryList.isEmpty()) {
String errorMessage = "Missing entries in context expression";
errors.add(makeError(repository, definitions, element, errorMessage));
}
} else if (expression instanceof TRelation) {
if (((TRelation) expression).getColumn() == null && ((TRelation) expression).getRow() == null) {
String errorMessage = "Empty relation";
errors.add(makeError(repository, definitions, element, errorMessage));
}
} else {
throw new UnsupportedOperationException("Not supported DMN expression type " + expression.getClass().getName());
}
}
}
private boolean isSupported(String expressionLanguage) {
return expressionLanguage == null || AbstractDMNToNativeTransformer.SUPPORTED_LANGUAGES.contains(expressionLanguage);
}
private void validateRule(DMNModelRepository repository, TDefinitions definitions, TDMNElement element, TDecisionRule rule, List errors) {
List inputEntry = rule.getInputEntry();
if (inputEntry == null || inputEntry.isEmpty()) {
String errorMessage = "No input entries for rule " + rule.getId();
errors.add(makeError(repository, definitions, element, errorMessage));
}
List outputEntry = rule.getOutputEntry();
if (outputEntry == null || outputEntry.isEmpty()) {
String errorMessage = "No outputEntry entries for rule " + rule.getId();
errors.add(makeError(repository, definitions, element, errorMessage));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy