/*
Copyright (c) 2006-2008, Dennis M. Sosnoski
All rights reserved.
Redistribution and use in source and binary forms, with or without modification,
are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright notice, this
list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
* Neither the name of JiBX nor the names of its contributors may be used
to endorse or promote products derived from this software without specific
prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package org.jibx.schema.elements;
import org.jibx.binding.util.StringArray;
import org.jibx.runtime.EnumSet;
import org.jibx.runtime.IUnmarshallingContext;
import org.jibx.runtime.JiBXException;
import org.jibx.runtime.QName;
import org.jibx.schema.IArity;
import org.jibx.schema.INamed;
import org.jibx.schema.attributes.DefRefAttributeGroup;
import org.jibx.schema.attributes.FormChoiceAttribute;
import org.jibx.schema.attributes.OccursAttributeGroup;
import org.jibx.schema.types.AllEnumSet;
import org.jibx.schema.types.Count;
import org.jibx.schema.validation.ValidationContext;
/**
* <element> element definition. The same code is used for both global and
* local element definitions, with the differences checked during validation.
*
* TODO: implement common base class for attribute and element?
*
* @author Dennis M. Sosnoski
*/
public class ElementElement extends AnnotatedBase implements IArity, INamed
{
/** List of allowed attribute names. */
public static final StringArray s_allowedAttributes =
new StringArray(new String[] { "abstract", "block", "default", "final", "fixed", "nillable", "substitutionGroup", "type" },
DefRefAttributeGroup.s_allowedAttributes,
FormChoiceAttribute.s_allowedAttributes,
OccursAttributeGroup.s_allowedAttributes,
AnnotatedBase.s_allowedAttributes);
/** Mask bits for inline type definition child. */
private long INLINE_TYPE_MASK = ELEMENT_MASKS[COMPLEXTYPE_TYPE] |
ELEMENT_MASKS[SIMPLETYPE_TYPE];
//
// Value set information
public static final int EXTENSION_BLOCK = 0;
public static final int RESTRICTION_BLOCK = 1;
public static final int SUBSTITUTION_BLOCK = 2;
public static final EnumSet s_blockValues = new EnumSet(EXTENSION_BLOCK,
new String[] { "extension", "restriction", "substitution"});
public static final int EXTENSION_FINAL = 0;
public static final int RESTRICTION_FINAL = 1;
public static final EnumSet s_derivationValues = new EnumSet(EXTENSION_FINAL,
new String[] { "extension", "restriction"});
//
// Instance data
/** Filtered list of inline type definition elements (zero or one only). */
private final FilteredSegmentList m_inlineTypeList;
/** Name or reference. */
private DefRefAttributeGroup m_defRef;
/** Form of name. */
private FormChoiceAttribute m_formChoice;
/** Occurs attribute group. */
private OccursAttributeGroup m_occurs;
/** 'type' attribute value. */
private QName m_type;
/** 'default' attribute value. */
private String m_default;
/** 'fixed' attribute value. */
private String m_fixed;
/** 'abstract' attribute value. */
private boolean m_abstract;
/** 'nillable' attribute value. */
private boolean m_nillable;
/** 'block' attribute value. */
private AllEnumSet m_block;
/** 'final' attribute value. */
private AllEnumSet m_final;
/** 'substitutionGroup' attribute information. */
private QName m_substitutionGroup;
/** Element definition (from 'ref' attribute - null
if none). */
private ElementElement m_refElement;
/** Complex or simple type definition (from 'type' attribute, or inline
definition - null
if none). */
private CommonTypeDefinition m_typeDefinition;
/** Qualified name (only defined after validation). */
private QName m_qname;
/**
* Constructor.
*/
public ElementElement() {
super(ELEMENT_TYPE);
m_inlineTypeList = new FilteredSegmentList(getChildrenWritable(),
INLINE_TYPE_MASK, this);
m_defRef = new DefRefAttributeGroup(this);
m_formChoice = new FormChoiceAttribute(this);
m_occurs = new OccursAttributeGroup(this);
m_block = new AllEnumSet(s_blockValues, "block");
m_final = new AllEnumSet(s_derivationValues, "final");
}
/**
* Clear any type information. This method is only visible for internal use,
* to be called in the process of setting new type information.
*/
private void clearType() {
m_inlineTypeList.clear();
m_defRef.setRef(null);
m_refElement = null;
m_type = null;
}
//
// Base class overrides
/* (non-Javadoc)
* @see org.jibx.schema.ElementBase#preset(org.jibx.runtime.IUnmarshallingContext)
*/
protected void preset(IUnmarshallingContext ictx) throws JiBXException {
validateAttributes(ictx, s_allowedAttributes);
super.preset(ictx);
}
//
// Access methods
/**
* Get 'type' attribute value.
*
* @return type (null
if not set)
*/
public QName getType() {
return m_type;
}
/**
* Set 'type' attribute value. Note that this method should only be used
* prior to validation, since it will only set the type name and not link
* the actual type information.
*
* @param type (null
if not set)
*/
public void setType(QName type) {
clearType();
m_type = type;
}
/**
* Get 'default' attribute value.
*
* @return default (null
if not set)
*/
public String getDefault() {
return m_default;
}
/**
* Set the 'default' attribute value.
*
* @param dflt (null
if not set)
*/
public void setDefault(String dflt) {
m_default = dflt;
}
/**
* Get 'fixed' attribute value.
*
* @return fixed (null
if not set)
*/
public String getFixed() {
return m_fixed;
}
/**
* Set 'fixed' attribute value.
*
* @param fixed (null
if not set)
*/
public void setFixed(String fixed) {
m_fixed = fixed;
}
/**
* Check 'abstract' attribute value.
*
* @return abstract attribute value (null
if not set)
*/
public boolean isAbstract() {
return m_abstract;
}
/**
* Set 'abstract' attribute value.
*
* @param abs abstract attribute value (null
if not set)
*/
public void setAbstract(boolean abs) {
m_abstract = abs;
}
/**
* Check 'nillable' attribute value.
*
* @return nillable attribute value (null
if not set)
*/
public boolean isNillable() {
return m_nillable;
}
/**
* Set 'nillable' attribute value.
*
* @param nil nillable attribute value (null
if not set)
*/
public void setNillable(boolean nil) {
m_nillable = nil;
}
/**
* Get 'final' attribute.
*
* @return final
*/
public AllEnumSet getFinal() {
return m_final;
}
/**
* Get 'block' attribute.
*
* @return block
*/
public AllEnumSet getBlock() {
return m_block;
}
/**
* Get 'substitutionGroup' attribute value.
*
* @return substitutionGroup (null
if not set)
*/
public QName getSubstitutionGroup() {
return m_substitutionGroup;
}
/**
* Set 'substitutionGroup' attribute value.
*
* @param qname (null
if not set)
*/
public void setSubstitutionGroup(QName qname) {
m_substitutionGroup = qname;
}
/**
* Get the referenced element declaration. This method is only usable after
* validation.
*
* @return referenced element definition, or null
if not a
* reference
*/
public ElementElement getReference() {
return m_refElement;
}
/**
* Get qualified name for element. This method is only usable after
* prevalidation.
*
* @return qname (null
if a reference)
*/
public QName getQName() {
return m_qname;
}
/**
* Check if the element uses an inline type definition.
*
* @return true
if inline, false
if not
*/
public boolean isInlineType() {
return m_inlineTypeList.size() == 1;
}
/**
* Get type definition. This returns the actual type definition for the
* element, irrespective of whether the element uses an element reference, a
* type reference, or an inline type definition. It is only usable after
* validation.
*
* @return type definition (null
if empty type definition)
*/
public CommonTypeDefinition getTypeDefinition() {
if (m_defRef.getRef() != null) {
return m_refElement.getTypeDefinition();
} else {
return m_typeDefinition;
}
}
/**
* Set type definition (either inline, or as reference).
*
* @param def inline type definition
*/
public void setTypeDefinition(CommonTypeDefinition def) {
// clear existing type information and set new
clearType();
m_typeDefinition = def;
// check form of type definition
if (def != null) {
if (def.getQName() == null) {
// embed inline type definition
m_inlineTypeList.add(def);
def.setParent(this);
} else {
// set type reference
m_type = def.getQName();
}
}
}
//
// Delegated methods
/**
* Get 'name' attribute value.
*
* @return name
* @see org.jibx.schema.attributes.DefRefAttributeGroup#getName()
*/
public String getName() {
return m_defRef.getName();
}
/**
* Get 'ref' attribute value.
*
* @return ref
* @see org.jibx.schema.attributes.DefRefAttributeGroup#getRef()
*/
public QName getRef() {
return m_defRef.getRef();
}
/**
* Set 'name' attribute value.
*
* @param name
* @see org.jibx.schema.attributes.DefRefAttributeGroup#setName(java.lang.String)
*/
public void setName(String name) {
m_defRef.setName(name);
}
/**
* Set 'ref' attribute value.
*
* @param ref
* @see org.jibx.schema.attributes.DefRefAttributeGroup#setRef(org.jibx.runtime.QName)
*/
public void setRef(QName ref) {
clearType();
m_defRef.setRef(ref);
}
/**
* Get 'form' attribute type code.
*
* @return form
* @see org.jibx.schema.attributes.FormChoiceAttribute#getForm()
*/
public int getForm() {
return m_formChoice.getForm();
}
/**
* Get 'form' attribute name value.
*
* @return form
* @see org.jibx.schema.attributes.FormChoiceAttribute#getFormText()
*/
public String getFormText() {
return m_formChoice.getFormText();
}
/**
* Set 'form' attribute type code.
*
* @param type
* @see org.jibx.schema.attributes.FormChoiceAttribute#setForm(int)
*/
public void setForm(int type) {
m_formChoice.setForm(type);
}
/**
* Get 'maxOccurs' attribute value.
*
* @return count
* @see org.jibx.schema.attributes.OccursAttributeGroup#getMaxOccurs()
*/
public Count getMaxOccurs() {
return m_occurs.getMaxOccurs();
}
/**
* Get 'minOccurs' attribute value.
*
* @return count
* @see org.jibx.schema.attributes.OccursAttributeGroup#getMinOccurs()
*/
public Count getMinOccurs() {
return m_occurs.getMinOccurs();
}
/**
* Set 'maxOccurs' attribute value.
*
* @param count
* @see org.jibx.schema.attributes.OccursAttributeGroup#setMaxOccurs(org.jibx.schema.types.Count)
*/
public void setMaxOccurs(Count count) {
m_occurs.setMaxOccurs(count);
}
/**
* Get 'maxOccurs' attribute value.
*
* @param count
* @see org.jibx.schema.attributes.OccursAttributeGroup#setMinOccurs(org.jibx.schema.types.Count)
*/
public void setMinOccurs(Count count) {
m_occurs.setMinOccurs(count);
}
//
// Validation methods
/* (non-Javadoc)
* @see org.jibx.schema.ComponentBase#prevalidate(org.jibx.schema.ValidationContext)
*/
public void prevalidate(ValidationContext vctx) {
// prevalidate the attributes
m_defRef.prevalidate(vctx);
m_formChoice.prevalidate(vctx);
m_occurs.prevalidate(vctx);
// check whether global or local definition
if (isGlobal()) {
// make sure name is supplied for global element
if (getName() == null) {
vctx.addError("The 'name' attribute is required for a global definition", this);
} else {
m_qname = new QName(vctx.getCurrentSchema().getTargetNamespace(),
getName());
}
// make sure prohibited attributes are not present
if (getRef() != null || getForm() != -1 || getMinOccurs() != null || getMaxOccurs() != null) {
vctx.addError("The 'ref', 'form', 'minOccurs' and 'maxOccurs' attributes are prohibited for a global element definition", this);
}
} else {
// make sure name or reference is supplied for local element
if (getName() == null && getRef() == null) {
vctx.addError("Either a 'name' attribute or a 'ref' attribute is required for a local element definition", this);
}
// generate qname if name supplied
if (getName() != null) {
SchemaElement schema = vctx.getCurrentSchema();
boolean def = schema.isElementQualifiedDefault();
String uri = null;
if (m_formChoice.isQualified(def)) {
uri = vctx.getCurrentSchema().getTargetNamespace();
}
m_qname = new QName(uri, getName());
}
// make sure prohibited attributes are not present
if (getSubstitutionGroup() != null || getBlock().isPresent() || getFinal().isPresent()) {
vctx.addError("The 'substitutionGroup', 'block', and 'final' attributes are prohibited for a local element definition", this);
}
}
// continue with parent class prevalidation
super.prevalidate(vctx);
}
/* (non-Javadoc)
* @see org.jibx.schema.ComponentBase#validate(org.jibx.schema.ValidationContext)
*/
public void validate(ValidationContext vctx) {
// start with validating the attributes
m_defRef.validate(vctx);
m_formChoice.validate(vctx);
// check type of definition
QName ref = getRef();
if (ref != null) {
// make sure element reference is defined
m_refElement = vctx.findElement(ref);
if (m_refElement == null) {
vctx.addFatal("Referenced element '" + ref + "' is not defined", this);
}
// check for any conflicting attributes
if (m_type != null) {
vctx.addError("'type' attribute not allowed with 'ref' attribute", this);
}
} else {
// set the type definition
if (m_type == null) {
// verify inline type definition
if (m_inlineTypeList.size() == 1) {
m_typeDefinition = (CommonTypeDefinition)m_inlineTypeList.get(0);
} else if (m_inlineTypeList.size() == 0) {
vctx.addWarning("No type defined", this);
} else {
vctx.addFatal("Only one inline type definition allowed", this);
}
} else {
// look up referenced type definition
m_typeDefinition = vctx.findType(m_type);
if (m_typeDefinition == null) {
vctx.addFatal("Referenced type '" + m_type + "' is not defined", this);
}
}
}
// handle base class validation if still going
if (!vctx.isSkipped(this)) {
super.validate(vctx);
}
}
}