DefaultMatchesConstraint.java

  1. /*
  2.  * Portions of this software was developed by employees of the National Institute
  3.  * of Standards and Technology (NIST), an agency of the Federal Government and is
  4.  * being made available as a public service. Pursuant to title 17 United States
  5.  * Code Section 105, works of NIST employees are not subject to copyright
  6.  * protection in the United States. This software may be subject to foreign
  7.  * copyright. Permission in the United States and in foreign countries, to the
  8.  * extent that NIST may hold copyright, to use, copy, modify, create derivative
  9.  * works, and distribute this software and its documentation without fee is hereby
  10.  * granted on a non-exclusive basis, provided that this notice and disclaimer
  11.  * of warranty appears in all copies.
  12.  *
  13.  * THE SOFTWARE IS PROVIDED 'AS IS' WITHOUT ANY WARRANTY OF ANY KIND, EITHER
  14.  * EXPRESSED, IMPLIED, OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, ANY WARRANTY
  15.  * THAT THE SOFTWARE WILL CONFORM TO SPECIFICATIONS, ANY IMPLIED WARRANTIES OF
  16.  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, AND FREEDOM FROM
  17.  * INFRINGEMENT, AND ANY WARRANTY THAT THE DOCUMENTATION WILL CONFORM TO THE
  18.  * SOFTWARE, OR ANY WARRANTY THAT THE SOFTWARE WILL BE ERROR FREE.  IN NO EVENT
  19.  * SHALL NIST BE LIABLE FOR ANY DAMAGES, INCLUDING, BUT NOT LIMITED TO, DIRECT,
  20.  * INDIRECT, SPECIAL OR CONSEQUENTIAL DAMAGES, ARISING OUT OF, RESULTING FROM,
  21.  * OR IN ANY WAY CONNECTED WITH THIS SOFTWARE, WHETHER OR NOT BASED UPON WARRANTY,
  22.  * CONTRACT, TORT, OR OTHERWISE, WHETHER OR NOT INJURY WAS SUSTAINED BY PERSONS OR
  23.  * PROPERTY OR OTHERWISE, AND WHETHER OR NOT LOSS WAS SUSTAINED FROM, OR AROSE OUT
  24.  * OF THE RESULTS OF, OR USE OF, THE SOFTWARE OR SERVICES PROVIDED HEREUNDER.
  25.  */

  26. package gov.nist.secauto.metaschema.core.model.constraint;

  27. import gov.nist.secauto.metaschema.core.datatype.IDataTypeAdapter;
  28. import gov.nist.secauto.metaschema.core.datatype.markup.MarkupLine;
  29. import gov.nist.secauto.metaschema.core.datatype.markup.MarkupMultiline;
  30. import gov.nist.secauto.metaschema.core.metapath.MetapathExpression;
  31. import gov.nist.secauto.metaschema.core.util.ObjectUtils;

  32. import java.util.Map;
  33. import java.util.Set;
  34. import java.util.regex.Pattern;

  35. import javax.xml.namespace.QName;

  36. import edu.umd.cs.findbugs.annotations.NonNull;
  37. import edu.umd.cs.findbugs.annotations.Nullable;

  38. public final class DefaultMatchesConstraint
  39.     extends AbstractConstraint
  40.     implements IMatchesConstraint {
  41.   private final Pattern pattern;
  42.   private final IDataTypeAdapter<?> dataType;

  43.   /**
  44.    * Create a new matches constraint, which enforces a value pattern and/or data
  45.    * type.
  46.    *
  47.    * @param id
  48.    *          the optional identifier for the constraint
  49.    * @param formalName
  50.    *          the constraint's formal name or {@code null} if not provided
  51.    * @param description
  52.    *          the constraint's semantic description or {@code null} if not
  53.    *          provided
  54.    * @param source
  55.    *          information about the constraint source
  56.    * @param level
  57.    *          the significance of a violation of this constraint
  58.    * @param target
  59.    *          the Metapath expression identifying the nodes the constraint targets
  60.    * @param properties
  61.    *          a collection of associated properties
  62.    * @param pattern
  63.    *          the value pattern to match or {@code null} if there is no match
  64.    *          pattern
  65.    * @param dataType
  66.    *          the value data type to match or {@code null} if there is no match
  67.    *          data type
  68.    * @param remarks
  69.    *          optional remarks describing the intent of the constraint
  70.    */
  71.   private DefaultMatchesConstraint(
  72.       @Nullable String id,
  73.       @Nullable String formalName,
  74.       @Nullable MarkupLine description,
  75.       @NonNull ISource source,
  76.       @NonNull Level level,
  77.       @NonNull MetapathExpression target,
  78.       @NonNull Map<QName, Set<String>> properties,
  79.       @Nullable Pattern pattern,
  80.       @Nullable IDataTypeAdapter<?> dataType,
  81.       @Nullable MarkupMultiline remarks) {
  82.     super(id, formalName, description, source, level, target, properties, remarks);
  83.     if (pattern == null && dataType == null) {
  84.       throw new IllegalArgumentException("a pattern or data type must be provided");
  85.     }
  86.     this.pattern = pattern;
  87.     this.dataType = dataType;
  88.   }

  89.   @Override
  90.   public Pattern getPattern() {
  91.     return pattern;
  92.   }

  93.   @Override
  94.   public IDataTypeAdapter<?> getDataType() {
  95.     return dataType;
  96.   }

  97.   @Override
  98.   public <T, R> R accept(IConstraintVisitor<T, R> visitor, T state) {
  99.     return visitor.visitMatchesConstraint(this, state);
  100.   }

  101.   @NonNull
  102.   public static Builder builder() {
  103.     return new Builder();
  104.   }

  105.   public static final class Builder
  106.       extends AbstractConstraintBuilder<Builder, DefaultMatchesConstraint> {
  107.     private Pattern pattern;
  108.     private IDataTypeAdapter<?> datatype;

  109.     private Builder() {
  110.       // disable construction
  111.     }

  112.     public Builder regex(@NonNull String pattern) {
  113.       return regex(ObjectUtils.notNull(Pattern.compile(pattern)));
  114.     }

  115.     public Builder regex(@NonNull Pattern pattern) {
  116.       this.pattern = pattern;
  117.       return this;
  118.     }

  119.     public Builder datatype(@NonNull IDataTypeAdapter<?> datatype) {
  120.       this.datatype = datatype;
  121.       return this;
  122.     }

  123.     @Override
  124.     protected Builder getThis() {
  125.       return this;
  126.     }

  127.     @Override
  128.     protected void validate() {
  129.       super.validate();

  130.       if (getPattern() == null && getDatatype() == null) {
  131.         throw new IllegalStateException("A pattern or data type must be provided at minimum.");
  132.       }
  133.     }

  134.     protected Pattern getPattern() {
  135.       return pattern;
  136.     }

  137.     protected IDataTypeAdapter<?> getDatatype() {
  138.       return datatype;
  139.     }

  140.     @Override
  141.     protected DefaultMatchesConstraint newInstance() {
  142.       return new DefaultMatchesConstraint(
  143.           getId(),
  144.           getFormalName(),
  145.           getDescription(),
  146.           ObjectUtils.notNull(getSource()),
  147.           getLevel(),
  148.           getTarget(),
  149.           getProperties(),
  150.           getPattern(),
  151.           getDatatype(),
  152.           getRemarks());
  153.     }
  154.   }
  155. }