blob: 393ab3e8bd7bf10b6d57212514294c893fc7ed64 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*
*/
package org.apache.directory.api.ldap.model.schema;
import java.util.ArrayList;
import java.util.List;
import org.apache.directory.api.i18n.I18n;
/**
* An objectClass definition.
* <p>
* According to ldapbis [MODELS]:
* </p>
*
* <pre>
* Object Class definitions are written according to the ABNF:
*
* ObjectClassDescription = LPAREN WSP
* numericoid ; object identifier
* [ SP &quot;NAME&quot; SP qdescrs ] ; short names (descriptors)
* [ SP &quot;DESC&quot; SP qdstring ] ; description
* [ SP &quot;OBSOLETE&quot; ] ; not active
* [ SP &quot;SUP&quot; SP oids ] ; superior object classes
* [ SP kind ] ; kind of class
* [ SP &quot;MUST&quot; SP oids ] ; attribute types
* [ SP &quot;MAY&quot; SP oids ] ; attribute types
* extensions WSP RPAREN
*
* kind = &quot;ABSTRACT&quot; / &quot;STRUCTURAL&quot; / &quot;AUXILIARY&quot;
*
* where:
* [numericoid] is object identifier assigned to this object class;
* NAME [qdescrs] are short names (descriptors) identifying this object
* class;
* DESC [qdstring] is a short descriptive string;
* OBSOLETE indicates this object class is not active;
* SUP [oids] specifies the direct superclasses of this object class;
* the kind of object class is indicated by one of ABSTRACT,
* STRUCTURAL, or AUXILIARY, default is STRUCTURAL;
* MUST and MAY specify the sets of required and allowed attribute
* types, respectively; and
* [extensions] describe extensions.
* </pre>
*
* @see <a href="http://www.faqs.org/rfcs/rfc2252.html">RFC2252 Section 4.4</a>
* @see <a
* href="http://www.ietf.org/internet-drafts/draft-ietf-ldapbis-models-11.txt">ldapbis
* [MODELS]</a>
* @author <a href="mailto:dev@directory.apache.org">Apache Directory Project</a>
*/
public class ObjectClass extends AbstractSchemaObject
{
/** The mandatory serialVersionUID */
public static final long serialVersionUID = 1L;
/** The ObjectClass type : ABSTRACT, AUXILIARY or STRUCTURAL */
protected ObjectClassTypeEnum objectClassType = ObjectClassTypeEnum.STRUCTURAL;
/** The ObjectClass superior OIDs */
protected transient List<String> superiorOids;
/** The ObjectClass superiors */
protected transient List<ObjectClass> superiors;
/** The list of allowed AttributeType OIDs */
protected transient List<String> mayAttributeTypeOids;
/** The list of allowed AttributeTypes */
protected transient List<AttributeType> mayAttributeTypes;
/** The list of required AttributeType OIDs */
protected transient List<String> mustAttributeTypeOids;
/** The list of required AttributeTypes */
protected transient List<AttributeType> mustAttributeTypes;
/**
* Creates a new instance of MatchingRuleUseDescription
* @param oid the OID for this objectClass
*/
public ObjectClass( String oid )
{
super( SchemaObjectType.OBJECT_CLASS, oid );
mayAttributeTypeOids = new ArrayList<>();
mustAttributeTypeOids = new ArrayList<>();
superiorOids = new ArrayList<>();
mayAttributeTypes = new ArrayList<>();
mustAttributeTypes = new ArrayList<>();
superiors = new ArrayList<>();
objectClassType = ObjectClassTypeEnum.STRUCTURAL;
}
/**
* @return the mayAttributeTypeOids
*/
public List<String> getMayAttributeTypeOids()
{
return mayAttributeTypeOids;
}
/**
* @return the mayAttributeTypes
*/
public List<AttributeType> getMayAttributeTypes()
{
return mayAttributeTypes;
}
/**
* Add some allowed AttributeType
*
* @param oids The attributeType oids
*/
public void addMayAttributeTypeOids( String... oids )
{
if ( locked )
{
throw new UnsupportedOperationException( I18n.err( I18n.ERR_13700_CANNOT_MODIFY_LOCKED_SCHEMA_OBJECT, getName() ) );
}
for ( String oid : oids )
{
mayAttributeTypeOids.add( oid );
}
}
/**
* Add some allowed AttributeTypes
*
* @param attributeTypes The attributeTypes
*/
public void addMayAttributeTypes( AttributeType... attributeTypes )
{
if ( locked )
{
throw new UnsupportedOperationException( I18n.err( I18n.ERR_13700_CANNOT_MODIFY_LOCKED_SCHEMA_OBJECT, getName() ) );
}
for ( AttributeType attributeType : attributeTypes )
{
if ( !mayAttributeTypeOids.contains( attributeType.getOid() ) )
{
mayAttributeTypes.add( attributeType );
mayAttributeTypeOids.add( attributeType.getOid() );
}
}
}
/**
* @param mayAttributeTypeOids the mayAttributeTypeOids to set
*/
public void setMayAttributeTypeOids( List<String> mayAttributeTypeOids )
{
if ( locked )
{
throw new UnsupportedOperationException( I18n.err( I18n.ERR_13700_CANNOT_MODIFY_LOCKED_SCHEMA_OBJECT, getName() ) );
}
this.mayAttributeTypeOids = mayAttributeTypeOids;
}
/**
* Sets the list of allowed AttributeTypes
*
* @param mayAttributeTypes the list of allowed AttributeTypes
*/
public void setMayAttributeTypes( List<AttributeType> mayAttributeTypes )
{
if ( locked )
{
throw new UnsupportedOperationException( I18n.err( I18n.ERR_13700_CANNOT_MODIFY_LOCKED_SCHEMA_OBJECT, getName() ) );
}
this.mayAttributeTypes = mayAttributeTypes;
// update the OIDS now
mayAttributeTypeOids.clear();
for ( AttributeType may : mayAttributeTypes )
{
mayAttributeTypeOids.add( may.getOid() );
}
}
/**
* @return the mustAttributeTypeOids
*/
public List<String> getMustAttributeTypeOids()
{
return mustAttributeTypeOids;
}
/**
* @return the mustAttributeTypes
*/
public List<AttributeType> getMustAttributeTypes()
{
return mustAttributeTypes;
}
/**
* Add some required AttributeType OIDs
*
* @param oids The attributeType OIDs
*/
public void addMustAttributeTypeOids( String... oids )
{
if ( locked )
{
throw new UnsupportedOperationException( I18n.err( I18n.ERR_13700_CANNOT_MODIFY_LOCKED_SCHEMA_OBJECT, getName() ) );
}
for ( String oid : oids )
{
mustAttributeTypeOids.add( oid );
}
}
/**
* Add some required AttributeTypes
*
* @param attributeTypes The attributeTypse
*/
public void addMustAttributeTypes( AttributeType... attributeTypes )
{
if ( locked )
{
throw new UnsupportedOperationException( I18n.err( I18n.ERR_13700_CANNOT_MODIFY_LOCKED_SCHEMA_OBJECT, getName() ) );
}
for ( AttributeType attributeType : attributeTypes )
{
if ( !mustAttributeTypeOids.contains( attributeType.getOid() ) )
{
mustAttributeTypes.add( attributeType );
mustAttributeTypeOids.add( attributeType.getOid() );
}
}
}
/**
* @param mustAttributeTypeOids the mustAttributeTypeOids to set
*/
public void setMustAttributeTypeOids( List<String> mustAttributeTypeOids )
{
if ( locked )
{
throw new UnsupportedOperationException( I18n.err( I18n.ERR_13700_CANNOT_MODIFY_LOCKED_SCHEMA_OBJECT, getName() ) );
}
this.mustAttributeTypeOids = mustAttributeTypeOids;
}
/**
* Sets the list of required AttributeTypes
*
* @param mustAttributeTypes the list of required AttributeTypes
*/
public void setMustAttributeTypes( List<AttributeType> mustAttributeTypes )
{
if ( locked )
{
throw new UnsupportedOperationException( I18n.err( I18n.ERR_13700_CANNOT_MODIFY_LOCKED_SCHEMA_OBJECT, getName() ) );
}
this.mustAttributeTypes = mustAttributeTypes;
// update the OIDS now
mustAttributeTypeOids.clear();
for ( AttributeType may : mustAttributeTypes )
{
mustAttributeTypeOids.add( may.getOid() );
}
}
/**
* Gets the superclasses of this ObjectClass.
*
* @return the superclasses
*/
public List<ObjectClass> getSuperiors()
{
return superiors;
}
/**
* Gets the superclasses OIDsof this ObjectClass.
*
* @return the superclasses OIDs
*/
public List<String> getSuperiorOids()
{
return superiorOids;
}
/**
* Add some superior ObjectClass OIDs
*
* @param oids The superior ObjectClass OIDs
*/
public void addSuperiorOids( String... oids )
{
if ( locked )
{
throw new UnsupportedOperationException( I18n.err( I18n.ERR_13700_CANNOT_MODIFY_LOCKED_SCHEMA_OBJECT, getName() ) );
}
for ( String oid : oids )
{
if ( !superiorOids.contains( oid ) )
{
superiorOids.add( oid );
}
}
}
/**
* Add some superior ObjectClasses
*
* @param objectClasses The superior ObjectClasses
*/
public void addSuperior( ObjectClass... objectClasses )
{
if ( locked )
{
throw new UnsupportedOperationException( I18n.err( I18n.ERR_13700_CANNOT_MODIFY_LOCKED_SCHEMA_OBJECT, getName() ) );
}
for ( ObjectClass objectClass : objectClasses )
{
if ( !superiorOids.contains( objectClass.getOid() ) )
{
superiorOids.add( objectClass.getOid() );
superiors.add( objectClass );
}
}
}
/**
* Sets the superior object classes
*
* @param superiors the object classes to set
*/
public void setSuperiors( List<ObjectClass> superiors )
{
if ( locked )
{
throw new UnsupportedOperationException( I18n.err( I18n.ERR_13700_CANNOT_MODIFY_LOCKED_SCHEMA_OBJECT, getName() ) );
}
this.superiors = superiors;
// update the OIDS now
superiorOids.clear();
for ( ObjectClass oc : superiors )
{
superiorOids.add( oc.getOid() );
}
}
/**
* Sets the superior object class OIDs
*
* @param superiorOids the object class OIDs to set
*/
public void setSuperiorOids( List<String> superiorOids )
{
if ( locked )
{
throw new UnsupportedOperationException( I18n.err( I18n.ERR_13700_CANNOT_MODIFY_LOCKED_SCHEMA_OBJECT, getName() ) );
}
this.superiorOids = superiorOids;
}
/**
* Gets the type of this ObjectClass as a type safe enum.
*
* @return the ObjectClass type as an enum
*/
public ObjectClassTypeEnum getType()
{
return objectClassType;
}
/**
* Set the ObjectClass type, one of ABSTRACT, AUXILIARY or STRUCTURAL.
*
* @param objectClassType The ObjectClassType value
*/
public void setType( ObjectClassTypeEnum objectClassType )
{
if ( locked )
{
throw new UnsupportedOperationException( I18n.err( I18n.ERR_13700_CANNOT_MODIFY_LOCKED_SCHEMA_OBJECT, getName() ) );
}
this.objectClassType = objectClassType;
}
/**
* Tells if the current ObjectClass is STRUCTURAL
*
* @return <code>true</code> if the ObjectClass is STRUCTURAL
*/
public boolean isStructural()
{
return objectClassType == ObjectClassTypeEnum.STRUCTURAL;
}
/**
* Tells if the current ObjectClass is ABSTRACT
*
* @return <code>true</code> if the ObjectClass is ABSTRACT
*/
public boolean isAbstract()
{
return objectClassType == ObjectClassTypeEnum.ABSTRACT;
}
/**
* Tells if the current ObjectClass is AUXILIARY
*
* @return <code>true</code> if the ObjectClass is AUXILIARY
*/
public boolean isAuxiliary()
{
return objectClassType == ObjectClassTypeEnum.AUXILIARY;
}
/**
* {@inheritDoc}
*/
@Override
public String toString()
{
return SchemaObjectRenderer.OPEN_LDAP_SCHEMA_RENDERER.render( this );
}
/**
* Copy an ObjectClass
*/
@Override
public ObjectClass copy()
{
ObjectClass copy = new ObjectClass( oid );
// Copy the SchemaObject common data
copy.copy( this );
// Copy the ObjectClass type
copy.objectClassType = objectClassType;
// Copy the Superiors ObjectClasses OIDs
copy.superiorOids = new ArrayList<>();
for ( String oid : superiorOids )
{
copy.superiorOids.add( oid );
}
// Copy the Superiors ObjectClasses ( will be empty )
copy.superiors = new ArrayList<>();
// Copy the MAY AttributeTypes OIDs
copy.mayAttributeTypeOids = new ArrayList<>();
for ( String oid : mayAttributeTypeOids )
{
copy.mayAttributeTypeOids.add( oid );
}
// Copy the MAY AttributeTypes ( will be empty )
copy.mayAttributeTypes = new ArrayList<>();
// Copy the MUST AttributeTypes OIDs
copy.mustAttributeTypeOids = new ArrayList<>();
for ( String oid : mustAttributeTypeOids )
{
copy.mustAttributeTypeOids.add( oid );
}
// Copy the MUST AttributeTypes ( will be empty )
copy.mustAttributeTypes = new ArrayList<>();
return copy;
}
/**
* @see Object#equals(Object)
*/
@Override
public int hashCode()
{
int hash = h;
hash = hash * 17 + objectClassType.getValue();
// As the elements aren't sorted, we don't iterate on
// them and add the computed hashcode to the global hashcode
int tempHash = 0;
for ( String oid : superiorOids )
{
tempHash += oid.hashCode();
}
hash = hash * 17 + tempHash;
tempHash = 0;
for ( String may : mayAttributeTypeOids )
{
tempHash += may.hashCode();
}
hash = hash * 17 + tempHash;
tempHash = 0;
for ( String must : mustAttributeTypeOids )
{
tempHash += must.hashCode();
}
hash = hash * 17 + tempHash;
return hash;
}
/**
* @see Object#equals(Object)
*/
@Override
public boolean equals( Object o )
{
if ( !super.equals( o ) )
{
return false;
}
if ( !( o instanceof ObjectClass ) )
{
return false;
}
ObjectClass that = ( ObjectClass ) o;
// The ObjectClassType
if ( objectClassType != that.objectClassType )
{
return false;
}
// The Superiors OIDs
if ( superiorOids.size() != that.superiorOids.size() )
{
return false;
}
// One way
for ( String oid : superiorOids )
{
if ( !that.superiorOids.contains( oid ) )
{
return false;
}
}
// The other way
for ( String oid : that.superiorOids )
{
if ( !superiorOids.contains( oid ) )
{
return false;
}
}
// The Superiors
if ( superiors.size() != that.superiors.size() )
{
return false;
}
// One way
for ( ObjectClass superior : superiors )
{
if ( !that.superiors.contains( superior ) )
{
return false;
}
}
// The other way
for ( ObjectClass superior : that.superiors )
{
if ( !superiors.contains( superior ) )
{
return false;
}
}
// The MAY OIDs
if ( mayAttributeTypeOids.size() != that.mayAttributeTypeOids.size() )
{
return false;
}
// One way
for ( String oid : mayAttributeTypeOids )
{
if ( !that.mayAttributeTypeOids.contains( oid ) )
{
return false;
}
}
// The other way
for ( String oid : that.mayAttributeTypeOids )
{
if ( !mayAttributeTypeOids.contains( oid ) )
{
return false;
}
}
// The MAY
if ( mayAttributeTypes.size() != that.mayAttributeTypes.size() )
{
return false;
}
// One way
for ( AttributeType oid : mayAttributeTypes )
{
if ( !that.mayAttributeTypes.contains( oid ) )
{
return false;
}
}
// The other way
for ( AttributeType oid : that.mayAttributeTypes )
{
if ( !mayAttributeTypes.contains( oid ) )
{
return false;
}
}
// The MUST OIDs
if ( mustAttributeTypeOids.size() != that.mustAttributeTypeOids.size() )
{
return false;
}
// One way
for ( String oid : mustAttributeTypeOids )
{
if ( !that.mustAttributeTypeOids.contains( oid ) )
{
return false;
}
}
// The other way
for ( String oid : that.mustAttributeTypeOids )
{
if ( !mustAttributeTypeOids.contains( oid ) )
{
return false;
}
}
// The MUST
if ( mustAttributeTypes.size() != that.mustAttributeTypes.size() )
{
return false;
}
// One way
for ( AttributeType oid : mustAttributeTypes )
{
if ( !that.mustAttributeTypes.contains( oid ) )
{
return false;
}
}
// The other way
for ( AttributeType oid : that.mustAttributeTypes )
{
if ( !mustAttributeTypes.contains( oid ) )
{
return false;
}
}
return true;
}
/**
* {@inheritDoc}
*/
@Override
public void clear()
{
// Clear the common elements
super.clear();
// Clear the references
mayAttributeTypes.clear();
mayAttributeTypeOids.clear();
mustAttributeTypes.clear();
mustAttributeTypeOids.clear();
superiors.clear();
superiorOids.clear();
}
}