org.simpleframework.xml.core.CacheLabel Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of simple-xml Show documentation
Show all versions of simple-xml Show documentation
Simple is a high performance XML serialization and configuration framework for Java
The newest version!
/*
* CacheLabel.java July 2007
*
* Copyright (C) 2006, Niall Gallagher
*
* Licensed 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
*
* http://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.simpleframework.xml.core;
import java.lang.annotation.Annotation;
import org.simpleframework.xml.strategy.Type;
/**
* The CacheLabel
object is used to acquire details from an
* inner label object so that details can be retrieved repeatedly without
* the need to perform any logic for extracting the values. This ensures
* that a class XML schema requires only initial processing the first
* time the class XML schema is required.
*
* @author Niall Gallagher
*/
class CacheLabel implements Label {
/**
* This is the annotation that this label instance represents.
*/
private final Annotation annotation;
/**
* This is the expression that is used to represent this label.
*/
private final Expression expression;
/**
* This is the decorator that is associated with the label.
*/
private final Decorator decorator;
/**
* This is the contact used to set and get the value for the node.
*/
private final Contact contact;
/**
* This represents the names that identify this label instance.
*/
private final String[] names;
/**
* This represents the paths that identify this label instance.
*/
private final String[] paths;
/**
* This is used to represent the label class that this will use.
*/
private final Class type;
/**
* This is used to represent the name of the entry item use.
*/
private final String entry;
/**
* This represents the path location within the XML for this.
*/
private final String path;
/**
* This is used to represent the name override for the annotation.
*/
private final String override;
/**
* This is used to represent the name of the annotated element.
*/
private final String name;
/**
* This is the label the this cache is wrapping the values for.
*/
private final Label label;
/**
* This is the key is used to represent this label object.
*/
private final Object key;
/**
* This is used to represent the dependent type to be used.
*/
private final Type depend;
/**
* This is used to represent whether the data is written as data.
*/
private final boolean data;
/**
* This is used to determine if this label represents text.
*/
private final boolean text;
/**
* This is used to determine if the label represents a union.
*/
private final boolean union;
/**
* This is used to determine the styling of the label name.
*/
private final boolean attribute;
/**
* This is used to represent whether the entity is required or not.
*/
private final boolean required;
/**
* This is used to determine if the label represents a collection.
*/
private final boolean collection;
/**
* This is used to determine whether the entity is inline or not.
*/
private final boolean inline;
/**
* This is used to determine if the label represents a text list.
*/
private final boolean list;
/**
* Constructor for the CacheLabel
object. This is used
* to create a Label
that acquires details from another
* label in such a way that any logic involved in acquiring details
* is performed only once.
*
* @param label this is the label to acquire the details from
*/
public CacheLabel(Label label) throws Exception {
this.annotation = label.getAnnotation();
this.expression = label.getExpression();
this.decorator = label.getDecorator();
this.attribute = label.isAttribute();
this.collection = label.isCollection();
this.contact = label.getContact();
this.depend = label.getDependent();
this.required = label.isRequired();
this.override = label.getOverride();
this.list = label.isTextList();
this.inline = label.isInline();
this.union = label.isUnion();
this.names = label.getNames();
this.paths = label.getPaths();
this.path = label.getPath();
this.type = label.getType();
this.name = label.getName();
this.entry = label.getEntry();
this.data = label.isData();
this.text = label.isText();
this.key = label.getKey();
this.label = label;
}
/**
* This is used to acquire the Type
that the type
* provided is represented by. Typically this will return the
* field or method represented by the label. However, in the
* case of unions this will provide an override type.
*
* @param type this is the class to acquire the type for
*
* @return this returns the type represented by this class
*/
public Type getType(Class type) throws Exception {
return label.getType(type);
}
/**
* This is used to acquire the Label
that the type
* provided is represented by. Typically this will return the
* same instance. However, in the case of unions this will
* look for an individual label to match the type provided.
*
* @param type this is the type to acquire the label for
*
* @return this returns the label represented by this type
*/
public Label getLabel(Class type) throws Exception {
return label.getLabel(type);
}
/**
* This returns a Collection
of element names. This
* will typically contain both the name and path of the label.
* However, if this is a union it can contain many names and
* paths. This method should never return null.
*
* @return this returns the names of each of the elements
*/
public String[] getNames() throws Exception {
return names;
}
/**
* This returns a Collection
of element paths. This
* will typically contain only the path of the label, which is
* composed using the Path
annotation and the name
* of the label. However, if this is a union it can contain many
* paths. This method should never return null.
*
* @return this returns the names of each of the elements
*/
public String[] getPaths() throws Exception {
return paths;
}
/**
* This acquires the annotation associated with this label. This
* is typically the annotation acquired from the field or method.
* However, in the case of unions this will return the actual
* annotation within the union group that this represents.
*
* @return this returns the annotation that this represents
*/
public Annotation getAnnotation(){
return annotation;
}
/**
* This is used to acquire the contact object for this label. The
* contact retrieved can be used to set any object or primitive that
* has been deserialized, and can also be used to acquire values to
* be serialized in the case of object persistence. All contacts
* that are retrieved from this method will be accessible.
*
* @return returns the field that this label is representing
*/
public Contact getContact() {
return contact;
}
/**
* This is used to acquire the Decorator
for this.
* A decorator is an object that adds various details to the
* node without changing the overall structure of the node. For
* example comments and namespaces can be added to the node with
* a decorator as they do not affect the deserialization.
*
* @return this returns the decorator associated with this
*/
public Decorator getDecorator() throws Exception {
return decorator;
}
/**
* This method returns a Converter
which can be used to
* convert an XML node into an object value and vice versa. The
* converter requires only the context object in order to perform
* serialization or deserialization of the provided XML node.
*
* @param context this is the context object for the serialization
*
* @return this returns an object that is used for conversion
*/
public Converter getConverter(Context context) throws Exception {
return label.getConverter(context);
}
/**
* This is used to provide a configured empty value used when the
* annotated value is null. This ensures that XML can be created
* with required details regardless of whether values are null or
* not. It also provides a means for sensible default values.
*
* @param context this is the context object for the serialization
*
* @return this returns the string to use for default values
*/
public Object getEmpty(Context context) throws Exception {
return label.getEmpty(context);
}
/**
* This returns the dependent type for the annotation. This type
* is the type other than the annotated field or method type that
* the label depends on. For the ElementList
and
* the ElementArray
this is the component type that
* is deserialized individually and inserted into the container.
*
* @return this is the type that the annotation depends on
*/
public Type getDependent() throws Exception {
return depend;
}
/**
* This is the key used to represent this label. The key is used
* to store the parameter in hash containers. Typically the
* key is generated from the paths associated with the label.
*
* @return this is the key used to represent the label
*/
public Object getKey() throws Exception {
return key;
}
/**
* This is used to either provide the entry value provided within
* the annotation or compute a entry value. If the entry string
* is not provided the the entry value is calculated as the type
* of primitive the object is as a simplified class name.
*
* @return this returns the name of the XML entry element used
*/
public String getEntry() throws Exception {
return entry;
}
/**
* This is used to acquire the name of the element or attribute
* that is used by the class schema. The name is determined by
* checking for an override within the annotation. If it contains
* a name then that is used, if however the annotation does not
* specify a name the the field or method name is used instead.
*
* @return returns the name that is used for the XML property
*/
public String getName() throws Exception {
return name;
}
/**
* This is used to acquire the path of the element or attribute
* that is used by the class schema. The path is determined by
* acquiring the XPath expression and appending the name of the
* label to form a fully qualified path.
*
* @return returns the path that is used for the XML property
*/
public String getPath() throws Exception {
return path;
}
/**
* This method is used to return an XPath expression that is
* used to represent the position of this label. If there is no
* XPath expression associated with this then an empty path is
* returned. This will never return a null expression.
*
* @return the XPath expression identifying the location
*/
public Expression getExpression() throws Exception {
return expression;
}
/**
* This is used to acquire the name of the element or attribute
* as taken from the annotation. If the element or attribute
* explicitly specifies a name then that name is used for the
* XML element or attribute used. If however no overriding name
* is provided then the method or field is used for the name.
*
* @return returns the name of the annotation for the contact
*/
public String getOverride() {
return override;
}
/**
* This acts as a convenience method used to determine the type of
* the field this represents. This is used when an object is written
* to XML. It determines whether a class
attribute
* is required within the serialized XML element, that is, if the
* class returned by this is different from the actual value of the
* object to be serialized then that type needs to be remembered.
*
* @return this returns the type of the field class
*/
public Class getType() {
return type;
}
/**
* This is used to determine whether the annotation requires it
* and its children to be written as a CDATA block. This is done
* when a primitive or other such element requires a text value
* and that value needs to be encapsulated within a CDATA block.
*
* @return this returns true if the element requires CDATA
*/
public boolean isData() {
return data;
}
/**
* This is used to determine if the label represents text. If
* a label represents text it typically does not have a name,
* instead the empty string represents the name. Also text
* labels can not exist with other text labels, or elements.
*
* @return this returns true if this label represents text
*/
public boolean isText() {
return text;
}
/**
* This is used to determine if an annotated list is a text
* list. A text list is a list of elements that also accepts
* free text. Typically this will be an element list union that
* will allow unstructured XML such as XHTML to be parsed.
*
* @return returns true if the label represents a text list
*/
public boolean isTextList() {
return list;
}
/**
* This is used to determine whether the label represents an
* inline XML entity. The ElementList
annotation
* and the Text
annotation represent inline
* items. This means that they contain no containing element
* and so can not specify overrides or special attributes.
*
* @return this returns true if the annotation is inline
*/
public boolean isInline() {
return inline;
}
/**
* This method is used to determine if the label represents an
* attribute. This is used to style the name so that elements
* are styled as elements and attributes are styled as required.
*
* @return this is used to determine if this is an attribute
*/
public boolean isAttribute() {
return attribute;
}
/**
* This is used to determine if the label is a collection. If the
* label represents a collection then any original assignment to
* the field or method can be written to without the need to
* create a new collection. This allows obscure collections to be
* used and also allows initial entries to be maintained.
*
* @return true if the label represents a collection value
*/
public boolean isCollection() {
return collection;
}
/**
* Determines whether the XML attribute or element is required.
* This ensures that if an XML element is missing from a document
* that deserialization can continue. Also, in the process of
* serialization, if a value is null it does not need to be
* written to the resulting XML document.
*
* @return true if the label represents a some required data
*/
public boolean isRequired() {
return required;
}
/**
* This is used to determine if this label is a union. If this
* is true then this label represents a number of labels and
* is simply a wrapper for these labels.
*
* @return this returns true if the label represents a union
*/
public boolean isUnion() {
return union;
}
/**
* This is used to describe the annotation and method or field
* that this label represents. This is used to provide error
* messages that can be used to debug issues that occur when
* processing a method. This should provide enough information
* such that the problem can be isolated correctly.
*
* @return this returns a string representation of the label
*/
public String toString() {
return label.toString();
}
}