LoggingConstraintValidationHandler.java
/*
* SPDX-FileCopyrightText: none
* SPDX-License-Identifier: CC0-1.0
*/
package gov.nist.secauto.metaschema.core.model.constraint;
import gov.nist.secauto.metaschema.core.datatype.IDataTypeAdapter;
import gov.nist.secauto.metaschema.core.metapath.DynamicContext;
import gov.nist.secauto.metaschema.core.metapath.ISequence;
import gov.nist.secauto.metaschema.core.metapath.MetapathException;
import gov.nist.secauto.metaschema.core.metapath.item.node.INodeItem;
import gov.nist.secauto.metaschema.core.model.constraint.IConstraint.Level;
import gov.nist.secauto.metaschema.core.util.ObjectUtils;
import org.apache.logging.log4j.LogBuilder;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import java.util.Comparator;
import java.util.List;
import java.util.regex.Pattern;
import edu.umd.cs.findbugs.annotations.NonNull;
import edu.umd.cs.findbugs.annotations.Nullable;
/**
* Supports logging constraint findings to the configured Log4J2 instance.
*/
public class LoggingConstraintValidationHandler
extends AbstractConstraintValidationHandler {
private static final Logger LOGGER = LogManager.getLogger(DefaultConstraintValidator.class);
private static final Throwable NO_EXCEPTION = null;
private static LogBuilder getLogBuilder(@NonNull Level level) {
LogBuilder retval;
switch (level) {
case CRITICAL:
retval = LOGGER.atFatal();
break;
case ERROR:
retval = LOGGER.atError();
break;
case WARNING:
retval = LOGGER.atWarn();
break;
case INFORMATIONAL:
retval = LOGGER.atInfo();
break;
default:
throw new UnsupportedOperationException(String.format("unsupported level '%s'", level));
}
return retval;
}
@Override
protected String toPath(@NonNull INodeItem nodeItem) {
return nodeItem.toPath(getPathFormatter());
}
/**
* Determine if a failure to validate a constraint at the given severity level
* should be logged.
*
* @param level
* the severity level to check
* @return {@code true} if the severity level should be logged, or {@code false}
* otherwise
*/
private static boolean isLogged(@NonNull Level level) {
boolean retval;
switch (level) {
case CRITICAL:
retval = LOGGER.isFatalEnabled();
break;
case ERROR:
retval = LOGGER.isErrorEnabled();
break;
case WARNING:
retval = LOGGER.isWarnEnabled();
break;
case INFORMATIONAL:
retval = LOGGER.isInfoEnabled();
break;
default:
throw new UnsupportedOperationException(String.format("unsupported level '%s'", level));
}
return retval;
}
private void logMessage(
@NonNull Level level,
@Nullable String identifier,
@NonNull INodeItem node,
@NonNull CharSequence message,
@Nullable Throwable cause) {
LogBuilder builder = getLogBuilder(level);
if (cause != null) {
builder.withThrowable(cause);
}
builder.log("{}{}: ({}) {}",
identifier == null ? "" : "[" + identifier + "] ",
level.name(),
toPath(node),
message);
}
@Override
public void handleCardinalityMinimumViolation(
@NonNull ICardinalityConstraint constraint,
@NonNull INodeItem node,
@NonNull ISequence<? extends INodeItem> targets,
@NonNull DynamicContext dynamicContext) {
Level level = constraint.getLevel();
if (isLogged(level)) {
logMessage(
level,
constraint.getId(),
node,
newCardinalityMinimumViolationMessage(
constraint,
node,
targets,
dynamicContext),
NO_EXCEPTION); // Null because there is no exeception, a Throwable cause.
}
}
@Override
public void handleCardinalityMaximumViolation(
@NonNull ICardinalityConstraint constraint,
@NonNull INodeItem node,
@NonNull ISequence<? extends INodeItem> targets,
@NonNull DynamicContext dynamicContext) {
Level level = constraint.getLevel();
if (isLogged(level)) {
logMessage(
level,
constraint.getId(),
node,
newCardinalityMaximumViolationMessage(
constraint,
node,
targets,
dynamicContext),
null);
}
}
@Override
public void handleIndexDuplicateKeyViolation(
@NonNull IIndexConstraint constraint,
@NonNull INodeItem node,
@NonNull INodeItem oldItem,
@NonNull INodeItem target,
@NonNull DynamicContext dynamicContext) {
Level level = constraint.getLevel();
if (isLogged(level)) {
logMessage(
level,
constraint.getId(),
target,
newIndexDuplicateKeyViolationMessage(
constraint,
node,
oldItem,
target,
dynamicContext),
NO_EXCEPTION);
}
}
@Override
public void handleUniqueKeyViolation(
@NonNull IUniqueConstraint constraint,
@NonNull INodeItem node,
@NonNull INodeItem oldItem,
@NonNull INodeItem target,
@NonNull DynamicContext dynamicContext) {
Level level = constraint.getLevel();
if (isLogged(level)) {
logMessage(
level,
constraint.getId(),
target,
newUniqueKeyViolationMessage(
constraint,
node,
oldItem,
target,
dynamicContext),
NO_EXCEPTION);
}
}
@SuppressWarnings("null")
@Override
public void handleKeyMatchError(
@NonNull IKeyConstraint constraint,
@NonNull INodeItem node,
@NonNull INodeItem target,
@NonNull MetapathException cause,
@NonNull DynamicContext dynamicContext) {
Level level = constraint.getLevel();
if (isLogged(level)) {
logMessage(
level,
constraint.getId(),
target,
cause.getLocalizedMessage(),
cause);
}
}
@Override
public void handleMatchPatternViolation(
@NonNull IMatchesConstraint constraint,
@NonNull INodeItem node,
@NonNull INodeItem target,
@NonNull String value,
@NonNull Pattern pattern,
@NonNull DynamicContext dynamicContext) {
Level level = constraint.getLevel();
if (isLogged(level)) {
logMessage(
level,
constraint.getId(),
target,
newMatchPatternViolationMessage(
constraint,
node,
target,
value,
pattern,
dynamicContext),
NO_EXCEPTION);
}
}
@Override
public void handleMatchDatatypeViolation(
@NonNull IMatchesConstraint constraint,
@NonNull INodeItem node,
@NonNull INodeItem target,
@NonNull String value,
@NonNull IDataTypeAdapter<?> adapter,
@NonNull IllegalArgumentException cause,
@NonNull DynamicContext dynamicContext) {
Level level = constraint.getLevel();
if (isLogged(level)) {
logMessage(
level,
constraint.getId(),
target,
newMatchDatatypeViolationMessage(
constraint,
node,
target,
value,
adapter,
dynamicContext),
cause);
}
}
@Override
public void handleExpectViolation(
@NonNull IExpectConstraint constraint,
@NonNull INodeItem node,
@NonNull INodeItem target,
@NonNull DynamicContext dynamicContext) {
Level level = constraint.getLevel();
if (isLogged(level)) {
logMessage(
level,
constraint.getId(),
target,
newExpectViolationMessage(
constraint,
node,
target,
dynamicContext),
NO_EXCEPTION);
}
}
@Override
public void handleAllowedValuesViolation(
List<IAllowedValuesConstraint> failedConstraints,
INodeItem target,
@NonNull DynamicContext dynamicContext) {
Level level = ObjectUtils.notNull(failedConstraints.stream()
.map(IConstraint::getLevel)
.max(Comparator.comparing(Level::ordinal))
.get());
if (isLogged(level)) {
logMessage(
level,
null,
target,
newAllowedValuesViolationMessage(
failedConstraints,
target),
null);
}
}
@Override
public void handleIndexDuplicateViolation(
@NonNull IIndexConstraint constraint,
@NonNull INodeItem node,
@NonNull DynamicContext dynamicContext) {
// always log at level critical
Level level = Level.CRITICAL;
if (isLogged(level)) {
logMessage(
level,
constraint.getId(),
node,
newIndexDuplicateViolationMessage(
constraint,
node),
NO_EXCEPTION);
}
}
@Override
public void handleIndexMiss(
@NonNull IIndexHasKeyConstraint constraint,
@NonNull INodeItem node,
@NonNull INodeItem target,
@NonNull List<String> key,
@NonNull DynamicContext dynamicContext) {
Level level = constraint.getLevel();
if (isLogged(level)) {
logMessage(
level,
constraint.getId(),
node,
newIndexMissMessage(
constraint,
node,
target,
key,
dynamicContext),
NO_EXCEPTION);
}
}
@Override
public void handleMissingIndexViolation(
@NonNull IIndexHasKeyConstraint constraint,
@NonNull INodeItem node,
@NonNull INodeItem target,
@NonNull String message,
@NonNull DynamicContext dynamicContext) {
Level level = constraint.getLevel();
if (isLogged(level)) {
logMessage(
level,
constraint.getId(),
node,
newMissingIndexViolationMessage(
constraint,
node,
target,
message,
dynamicContext),
NO_EXCEPTION);
}
}
@Override
public void handlePass(
@NonNull IConstraint constraint,
@NonNull INodeItem node,
@NonNull INodeItem target,
@NonNull DynamicContext dynamicContext) {
if (LOGGER.isDebugEnabled()) {
String identifier = constraint.getId();
LOGGER.atDebug().log("{}{}: ({}) {}",
identifier == null ? "" : "[" + identifier + "] ",
Level.INFORMATIONAL.name(),
toPath(node),
"Passed");
}
}
@Override
public void handleError(
@NonNull IConstraint constraint,
@NonNull INodeItem node,
@NonNull String message,
@NonNull Throwable exception,
@NonNull DynamicContext dynamicContext) {
Level level = Level.CRITICAL;
if (isLogged(level)) {
logMessage(
level,
constraint.getId(),
node,
message,
exception);
}
}
}