
com.imsweb.validation.ValidatingContext Maven / Gradle / Ivy
/*
* Copyright (C) 2014 Information Management Services, Inc.
*/
package com.imsweb.validation;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import com.imsweb.validation.entities.Rule;
/**
* This class is used as a context when processing a validatable. Note that it has noting to do with user-defined contexts that are made available to the rules...
*
* By overwriting the conditionFailed() and atLeastOneDependencyFailed() methods, the conditions and dependencies mechanism can be customized to allow more complex features
* (for example, to allow cross-validator conditions)...
*
* This class is not thread-safe. Contexts are created by the engine in the different "validate" methods. While those methods can be called in a multi-threaded environment,
* what happens inside the method is single-threaded. In other words, the validating context are not shared between threads and they do not need to be thread-safe.
*/
public class ValidatingContext {
// rule IDs to dynamically ignore (if both a collection of rule to execute and ignore are provided, the execute takes precedence)
protected Collection _toIgnore;
// rule IDs to dynamically execute (if both a collection of rule to execute and ignore are provided, the execute takes precedence)
protected Collection _toExecute;
// single rule to force, useful for unit tests (this takes precedence on both the execute and ignore collections of IDs)
protected Rule _toForce;
// rule IDs that have failed so far, mapped by validatable path.
protected Map> _failedRuleIds;
// condition IDs that have failed so far, mapped by validatable path.
protected Map> _failedConditionIds;
// the edits duration, per validatable path, then per edit ID (done that way to ensure thread-safety)
private final Map> _editsStats;
// whether or not the edits statistics should be computed
private boolean _computeEditsStats;
/**
* Constructor.
*/
public ValidatingContext() {
_failedRuleIds = new HashMap<>();
_failedConditionIds = new HashMap<>();
_editsStats = new HashMap<>();
_computeEditsStats = false;
}
public Collection getToIgnore() {
return _toIgnore;
}
public void setToIgnore(Collection toIgnore) {
_toIgnore = toIgnore;
}
public Collection getToExecute() {
return _toExecute;
}
public void setToExecute(Collection toExecute) {
_toExecute = toExecute;
}
public Rule getToForce() {
return _toForce;
}
public void setToForce(Rule toForce) {
_toForce = toForce;
}
public Map> getFailedRuleIds() {
return _failedRuleIds;
}
public Map> getFailedConditionIds() {
return _failedConditionIds;
}
public boolean computeEditsStats() {
return _computeEditsStats;
}
/**
* Sets whether or not the edits statistics should be computed; this method should only be called by the engine.
*/
public void setComputeEditsStats(boolean computeEditsStats) {
_computeEditsStats = computeEditsStats;
}
public void reportEditDuration(String path, String id, long duration) {
_editsStats.computeIfAbsent(path, k -> new HashMap<>()).put(id, duration);
}
public Map getEditDurations() {
Map stats = new HashMap<>();
_editsStats.values().forEach(stats::putAll);
return stats;
}
/**
* Resets the rule and condition failures. This method should only be used for testing purposes.
*/
public void resetFailures() {
_failedRuleIds.clear();
_failedConditionIds.clear();
}
/**
* Returns true if the provided condition ID has failed, false otherwise.
*
* The default behavior of this method is to check the failed conditions for the current validatable level, and any of it's parent (so no cross-paths in the validatable tree).
*
* Implementation: this method uses the _failedConditionsIds property to determine its result.
* @param validatablePaths the currently processed validatable path, already split for convenience
* @param conditionId the condition ID to check
* @return true if the condition has failed, false otherwise
*/
public boolean conditionFailed(List validatablePaths, String conditionId) {
if (validatablePaths == null || conditionId == null)
return false;
for (String validatablePath : validatablePaths) {
Set failedIds = _failedConditionIds.get(validatablePath);
if (failedIds != null && failedIds.contains(conditionId))
return true;
}
return false;
}
/**
* Returns true if at least one rule ID in the provided list of dependencies has failed, false otherwise.
*
* The default behavior of this method is to check the failed rules for the current validatable level, and any of it's parent (so no cross-paths in the validatable tree).
*
* Implementation: this method uses the _failedRuleIds property to determine its result.
* @param validatablePaths the currently processed validatable path, already split for convenience
* @param dependencies the rule IDs to check
* @return true if the at least one "depends-on" rule has failed, false otherwise
*/
public boolean atLeastOneDependencyFailed(List validatablePaths, Set dependencies) {
if (validatablePaths == null || dependencies == null)
return false;
for (String validatablePath : validatablePaths) {
Set failedIds = _failedRuleIds.get(validatablePath);
if (failedIds != null && !Collections.disjoint(failedIds, dependencies))
return true;
}
return false;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy