/* See LICENSE for licensing and NOTICE for copyright. */ package org.ldaptive.schema; import java.text.ParseException; import java.util.Arrays; import java.util.regex.Matcher; import java.util.regex.Pattern; import org.ldaptive.LdapUtils; /** * Bean for an matching rule schema element. * * <pre> MatchingRuleDescription = LPAREN WSP numericoid ; object identifier [ SP "NAME" SP qdescrs ] ; short names (descriptors) [ SP "DESC" SP qdstring ] ; description [ SP "OBSOLETE" ] ; not active SP "SYNTAX" SP numericoid ; assertion syntax extensions WSP RPAREN ; extensions * </pre> * * @author Middleware Services */ public class MatchingRule extends AbstractNamedSchemaElement { /** hash code seed. */ private static final int HASH_CODE_SEED = 1117; /** Pattern to match definitions. */ private static final Pattern DEFINITION_PATTERN = Pattern.compile( WSP_REGEX + "\\(" + WSP_REGEX + "(" + NO_WSP_REGEX + ")" + WSP_REGEX + "(?:NAME (?:'([^']+)'|\\(([^\\)]+)\\)))?" + WSP_REGEX + "(?:DESC '([^']*)')?" + WSP_REGEX + "(OBSOLETE)?" + WSP_REGEX + "(?:SYNTAX (" + NO_WSP_REGEX + "))?" + WSP_REGEX + "(?:(X-[^ ]+.*))?" + WSP_REGEX + "\\)" + WSP_REGEX); /** OID. */ private final String oid; /** Syntax OID. */ private String syntaxOID; /** * Creates a new matching rule. * * @param s oid */ public MatchingRule(final String s) { oid = s; } /** * Creates a new matching rule. * * @param oid oid * @param names names * @param description description * @param obsolete obsolete * @param syntaxOID syntax OID * @param extensions extensions */ // CheckStyle:ParameterNumber|HiddenField OFF public MatchingRule( final String oid, final String[] names, final String description, final boolean obsolete, final String syntaxOID, final Extensions extensions) { this(oid); setNames(names); setDescription(description); setObsolete(obsolete); setSyntaxOID(syntaxOID); setExtensions(extensions); } // CheckStyle:ParameterNumber|HiddenField ON /** * Returns the oid. * * @return oid */ public String getOID() { return oid; } /** * Returns the syntax oid. * * @return syntax oid */ public String getSyntaxOID() { return syntaxOID; } /** * Sets the syntax oid. * * @param s syntax oid */ public void setSyntaxOID(final String s) { syntaxOID = s; } /** * Parses the supplied definition string and creates an initialized matching rule. * * @param definition to parse * * @return matching rule * * @throws ParseException if the supplied definition is invalid */ public static MatchingRule parse(final String definition) throws ParseException { final Matcher m = DEFINITION_PATTERN.matcher(definition); if (!m.matches()) { throw new ParseException("Invalid matching rule definition: " + definition, definition.length()); } final MatchingRule mrd = new MatchingRule(m.group(1).trim()); // CheckStyle:MagicNumber OFF // parse names if (m.group(2) != null) { mrd.setNames(SchemaUtils.parseDescriptors(m.group(2).trim())); } else if (m.group(3) != null) { mrd.setNames(SchemaUtils.parseDescriptors(m.group(3).trim())); } mrd.setDescription(m.group(4) != null ? m.group(4).trim() : null); mrd.setObsolete(m.group(5) != null); mrd.setSyntaxOID(m.group(6) != null ? m.group(6).trim() : null); // parse extensions if (m.group(7) != null) { mrd.setExtensions(Extensions.parse(m.group(7).trim())); } return mrd; // CheckStyle:MagicNumber ON } @Override public String format() { final StringBuilder sb = new StringBuilder("( "); sb.append(oid).append(" "); if (getNames() != null && getNames().length > 0) { sb.append("NAME "); sb.append(SchemaUtils.formatDescriptors(getNames())); } if (getDescription() != null) { sb.append("DESC "); sb.append(SchemaUtils.formatDescriptors(getDescription())); } if (isObsolete()) { sb.append("OBSOLETE "); } if (syntaxOID != null) { sb.append("SYNTAX ").append(syntaxOID).append(" "); } if (getExtensions() != null) { sb.append(getExtensions().format()); } sb.append(")"); return sb.toString(); } @Override public boolean equals(final Object o) { if (o == this) { return true; } if (o instanceof MatchingRule) { final MatchingRule v = (MatchingRule) o; return LdapUtils.areEqual(oid, v.oid) && LdapUtils.areEqual(getNames(), v.getNames()) && LdapUtils.areEqual(getDescription(), v.getDescription()) && LdapUtils.areEqual(isObsolete(), v.isObsolete()) && LdapUtils.areEqual(syntaxOID, v.syntaxOID) && LdapUtils.areEqual(getExtensions(), v.getExtensions()); } return false; } @Override public int hashCode() { return LdapUtils.computeHashCode( HASH_CODE_SEED, oid, getNames(), getDescription(), isObsolete(), syntaxOID, getExtensions()); } @Override public String toString() { return String.format( "[%s@%d::oid=%s, names=%s, description=%s, obsolete=%s, " + "synxtaxOID=%s, extensions=%s]", getClass().getName(), hashCode(), oid, Arrays.toString(getNames()), getDescription(), isObsolete(), syntaxOID, getExtensions()); } }