package com.fasterxml.jackson.dataformat.xml.util;
import javax.xml.namespace.QName;
import com.fasterxml.jackson.databind.*;
import com.fasterxml.jackson.databind.cfg.MapperConfig;
import com.fasterxml.jackson.databind.introspect.AnnotatedClass;
import com.fasterxml.jackson.databind.type.ClassKey;
import com.fasterxml.jackson.databind.util.LRUMap;
import com.fasterxml.jackson.dataformat.xml.XmlAnnotationIntrospector;
Helper class used for efficiently finding root element name used with
XML serializations.
/**
* Helper class used for efficiently finding root element name used with
* XML serializations.
*/
public class XmlRootNameLookup
implements java.io.Serializable
{
private static final long serialVersionUID = 1L;
For efficient operation, let's try to minimize number of times we
need to introspect root element name to use.
Note: changed to transient
for 2.3; no point in serializing such
state
/**
* For efficient operation, let's try to minimize number of times we
* need to introspect root element name to use.
*<p>
* Note: changed to <code>transient</code> for 2.3; no point in serializing such
* state
*/
protected final transient LRUMap<ClassKey,QName> _rootNames = new LRUMap<ClassKey,QName>(40, 200);
public XmlRootNameLookup() { }
protected Object readResolve() {
// just need to make 100% sure it gets set to non-null, that's all
if (_rootNames == null) {
return new XmlRootNameLookup();
}
return this;
}
public QName findRootName(JavaType rootType, MapperConfig<?> config) {
return findRootName(rootType.getRawClass(), config);
}
public QName findRootName(Class<?> rootType, MapperConfig<?> config)
{
ClassKey key = new ClassKey(rootType);
QName name;
synchronized (_rootNames) {
name = _rootNames.get(key);
}
if (name != null) {
return name;
}
name = _findRootName(rootType, config);
synchronized (_rootNames) {
_rootNames.put(key, name);
}
return name;
}
// NOTE: needed to be synchronized in 2.6.4, but 2.7.0 adds a proper fix
// for annotation introspection hence not needed any more
protected QName _findRootName(Class<?> rootType, MapperConfig<?> config)
{
BeanDescription beanDesc = config.introspectClassAnnotations(rootType);
AnnotationIntrospector intr = config.getAnnotationIntrospector();
AnnotatedClass ac = beanDesc.getClassInfo();
String localName = null;
String ns = null;
PropertyName root = intr.findRootName(ac);
if (root != null) {
localName = root.getSimpleName();
ns = root.getNamespace();
}
// No answer so far? Let's just default to using simple class name
if (localName == null || localName.length() == 0) {
// Should we strip out enclosing class tho? For now, nope:
// one caveat: array simple names end with "[]"; also, "$" needs replacing
localName = StaxUtil.sanitizeXmlTypeName(rootType.getSimpleName());
return new QName("", localName);
}
// Otherwise let's see if there's namespace, too (if we are missing it)
if (ns == null || ns.length() == 0) {
ns = findNamespace(intr, ac);
}
if (ns == null) { // some QName impls barf on nulls...
ns = "";
}
return new QName(ns, localName);
}
private String findNamespace(AnnotationIntrospector ai, AnnotatedClass ann)
{
for (AnnotationIntrospector intr : ai.allIntrospectors()) {
if (intr instanceof XmlAnnotationIntrospector) {
String ns = ((XmlAnnotationIntrospector) intr).findNamespace(ann);
if (ns != null) {
return ns;
}
}
}
return null;
}
}