/* 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 a name form schema element.
*
* <pre>
NameFormDescription = LPAREN WSP
numericoid ; object identifier
[ SP "NAME" SP qdescrs ] ; short names (descriptors)
[ SP "DESC" SP qdstring ] ; description
[ SP "OBSOLETE" ] ; not active
SP "OC" SP oid ; structural object class
SP "MUST" SP oids ; attribute types
[ SP "MAY" SP oids ] ; attribute types
extensions WSP RPAREN ; extensions
* </pre>
*
* @author Middleware Services
*/
public class NameForm extends AbstractNamedSchemaElement
{
/** hash code seed. */
private static final int HASH_CODE_SEED = 1163;
/** 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 + "(?:OC (" + NO_WSP_REGEX + "))?" +
WSP_REGEX + "(?:MUST (?:(" + NO_WSP_REGEX + ")|\\(([^\\)]+)\\)))?" +
WSP_REGEX + "(?:MAY (?:(" + NO_WSP_REGEX + ")|\\(([^\\)]+)\\)))?" +
WSP_REGEX + "(?:(X-[^ ]+.*))?" +
WSP_REGEX + "\\)" + WSP_REGEX);
/** OID. */
private final String oid;
/** Structural object class. */
private String structuralClass;
/** Required attributes. */
private String[] requiredAttributes;
/** Optional attributes. */
private String[] optionalAttributes;
/**
* Creates a new name form.
*
* @param s oid
*/
public NameForm(final String s)
{
oid = s;
}
/**
* Creates a new name form.
*
* @param oid oid
* @param names names
* @param description description
* @param obsolete obsolete
* @param structuralClass structural object class
* @param requiredAttributes required attributes
* @param optionalAttributes optional attributes
* @param extensions extensions
*/
// CheckStyle:ParameterNumber|HiddenField OFF
public NameForm(
final String oid,
final String[] names,
final String description,
final boolean obsolete,
final String structuralClass,
final String[] requiredAttributes,
final String[] optionalAttributes,
final Extensions extensions)
{
this(oid);
setNames(names);
setDescription(description);
setObsolete(obsolete);
setStructuralClass(structuralClass);
setRequiredAttributes(requiredAttributes);
setOptionalAttributes(optionalAttributes);
setExtensions(extensions);
}
// CheckStyle:ParameterNumber|HiddenField ON
/**
* Returns the oid.
*
* @return oid
*/
public String getOID()
{
return oid;
}
/**
* Returns the structural object class.
*
* @return structural object class
*/
public String getStructuralClass()
{
return structuralClass;
}
/**
* Sets the structural object class.
*
* @param s structural object class
*/
public void setStructuralClass(final String s)
{
structuralClass = s;
}
/**
* Returns the required attributes.
*
* @return required attributes
*/
public String[] getRequiredAttributes()
{
return requiredAttributes;
}
/**
* Sets the required attributes.
*
* @param s required attributes
*/
public void setRequiredAttributes(final String[] s)
{
requiredAttributes = s;
}
/**
* Returns the optional attributes.
*
* @return optional attributes
*/
public String[] getOptionalAttributes()
{
return optionalAttributes;
}
/**
* Sets the optional attributes.
*
* @param s optional attributes
*/
public void setOptionalAttributes(final String[] s)
{
optionalAttributes = s;
}
/**
* Parses the supplied definition string and creates an initialized name form.
*
* @param definition to parse
*
* @return name form
*
* @throws ParseException if the supplied definition is invalid
*/
public static NameForm parse(final String definition)
throws ParseException
{
final Matcher m = DEFINITION_PATTERN.matcher(definition);
if (!m.matches()) {
throw new ParseException("Invalid name form definition: " + definition, definition.length());
}
final NameForm nfd = new NameForm(m.group(1).trim());
// CheckStyle:MagicNumber OFF
// parse names
if (m.group(2) != null) {
nfd.setNames(SchemaUtils.parseDescriptors(m.group(2).trim()));
} else if (m.group(3) != null) {
nfd.setNames(SchemaUtils.parseDescriptors(m.group(3).trim()));
}
nfd.setDescription(m.group(4) != null ? m.group(4).trim() : null);
nfd.setObsolete(m.group(5) != null);
nfd.setStructuralClass(m.group(6) != null ? m.group(6).trim() : null);
// parse required attributes
if (m.group(7) != null) {
nfd.setRequiredAttributes(SchemaUtils.parseOIDs(m.group(7).trim()));
} else if (m.group(8) != null) {
nfd.setRequiredAttributes(SchemaUtils.parseOIDs(m.group(8).trim()));
}
// parse optional attributes
if (m.group(9) != null) {
nfd.setOptionalAttributes(SchemaUtils.parseOIDs(m.group(9).trim()));
} else if (m.group(10) != null) {
nfd.setOptionalAttributes(SchemaUtils.parseOIDs(m.group(10).trim()));
}
// parse extensions
if (m.group(11) != null) {
nfd.setExtensions(Extensions.parse(m.group(11).trim()));
}
return nfd;
// 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 (structuralClass != null) {
sb.append("OC ").append(structuralClass).append(" ");
}
if (requiredAttributes != null && requiredAttributes.length > 0) {
sb.append("MUST ");
sb.append(SchemaUtils.formatOids(requiredAttributes));
}
if (optionalAttributes != null && optionalAttributes.length > 0) {
sb.append("MAY ");
sb.append(SchemaUtils.formatOids(optionalAttributes));
}
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 NameForm) {
final NameForm v = (NameForm) o;
return LdapUtils.areEqual(oid, v.oid) &&
LdapUtils.areEqual(getNames(), v.getNames()) &&
LdapUtils.areEqual(getDescription(), v.getDescription()) &&
LdapUtils.areEqual(isObsolete(), v.isObsolete()) &&
LdapUtils.areEqual(structuralClass, v.structuralClass) &&
LdapUtils.areEqual(requiredAttributes, v.requiredAttributes) &&
LdapUtils.areEqual(optionalAttributes, v.optionalAttributes) &&
LdapUtils.areEqual(getExtensions(), v.getExtensions());
}
return false;
}
@Override
public int hashCode()
{
return
LdapUtils.computeHashCode(
HASH_CODE_SEED,
oid,
getNames(),
getDescription(),
isObsolete(),
structuralClass,
requiredAttributes,
optionalAttributes,
getExtensions());
}
@Override
public String toString()
{
return
String.format(
"[%s@%d::oid=%s, names=%s, description=%s, obsolete=%s, " +
"structuralClass=%s, requiredAttributes=%s, " +
"optionalAttributes=%s, extensions=%s]",
getClass().getName(),
hashCode(),
oid,
Arrays.toString(getNames()),
getDescription(),
isObsolete(),
structuralClass,
Arrays.toString(requiredAttributes),
Arrays.toString(optionalAttributes),
getExtensions());
}
}