You are viewing a plain text version of this content. The canonical link for it is here.
Posted to scm@geronimo.apache.org by gn...@apache.org on 2006/08/30 16:14:55 UTC

svn commit: r438491 [2/2] - in /geronimo/xbean/trunk: ./ xbean-spring-itests/ xbean-spring-itests/2.0-rc3/ xbean-spring-itests/2.0-rc3/src/ xbean-spring-itests/2.0-rc3/src/main/ xbean-spring-itests/2.0-rc3/src/main/resources/ xbean-spring-itests/2.0-rc...

Added: geronimo/xbean/trunk/xbean-spring-v2a/src/main/java/org/apache/xbean/spring/context/v2a/XBeanNamespaceHandler.java
URL: http://svn.apache.org/viewvc/geronimo/xbean/trunk/xbean-spring-v2a/src/main/java/org/apache/xbean/spring/context/v2a/XBeanNamespaceHandler.java?rev=438491&view=auto
==============================================================================
--- geronimo/xbean/trunk/xbean-spring-v2a/src/main/java/org/apache/xbean/spring/context/v2a/XBeanNamespaceHandler.java (added)
+++ geronimo/xbean/trunk/xbean-spring-v2a/src/main/java/org/apache/xbean/spring/context/v2a/XBeanNamespaceHandler.java Wed Aug 30 07:14:53 2006
@@ -0,0 +1,861 @@
+/**
+ * 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
+ *
+ *      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.apache.xbean.spring.context.v2a;
+
+import java.beans.BeanInfo;
+import java.beans.PropertyDescriptor;
+import java.beans.PropertyEditor;
+import java.io.ByteArrayInputStream;
+import java.io.File;
+import java.io.FileInputStream;
+import java.io.FileNotFoundException;
+import java.io.IOException;
+import java.io.InputStream;
+import java.util.Arrays;
+import java.util.Collection;
+import java.util.Enumeration;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.Map;
+import java.util.Properties;
+import java.util.Set;
+
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+import org.apache.xbean.spring.context.impl.MappingMetaData;
+import org.apache.xbean.spring.context.impl.NamedConstructorArgs;
+import org.apache.xbean.spring.context.impl.NamespaceHelper;
+import org.apache.xbean.spring.context.impl.PropertyEditorHelper;
+import org.apache.xbean.spring.context.v2.XBeanQNameHelper;
+import org.springframework.beans.PropertyValue;
+import org.springframework.beans.factory.BeanDefinitionStoreException;
+import org.springframework.beans.factory.config.BeanDefinition;
+import org.springframework.beans.factory.config.BeanDefinitionHolder;
+import org.springframework.beans.factory.config.RuntimeBeanReference;
+import org.springframework.beans.factory.support.AbstractBeanDefinition;
+import org.springframework.beans.factory.support.BeanComponentDefinition;
+import org.springframework.beans.factory.support.BeanDefinitionReaderUtils;
+import org.springframework.beans.factory.support.DefaultListableBeanFactory;
+import org.springframework.beans.factory.support.ManagedList;
+import org.springframework.beans.factory.xml.AbstractBeanDefinitionParser;
+import org.springframework.beans.factory.xml.BeanDefinitionParserDelegate;
+import org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader;
+import org.springframework.beans.factory.xml.NamespaceHandler;
+import org.springframework.beans.factory.xml.ParserContext;
+import org.springframework.beans.factory.xml.XmlBeanDefinitionReader;
+import org.springframework.context.support.AbstractApplicationContext;
+import org.w3c.dom.Attr;
+import org.w3c.dom.Element;
+import org.w3c.dom.NamedNodeMap;
+import org.w3c.dom.Node;
+import org.w3c.dom.NodeList;
+import org.w3c.dom.Text;
+
+/**
+ * An enhanced XML parser capable of handling custom XML schemas.
+ *
+ * @author James Strachan
+ * @version $Id$
+ * @since 2.0
+ */
+public class XBeanNamespaceHandler implements NamespaceHandler {
+
+    public static final String SPRING_SCHEMA = "http://xbean.apache.org/schemas/spring/1.0";
+    public static final String SPRING_SCHEMA_COMPAT = "http://xbean.org/schemas/spring/1.0";
+
+    static {
+        PropertyEditorHelper.registerCustomEditors();
+    }
+
+    private static final Log log = LogFactory.getLog(XBeanNamespaceHandler.class);
+
+    private static final String QNAME_ELEMENT = "qname";
+    
+    private static final String DESCRIPTION_ELEMENT = "description";
+
+    /**
+     * All the reserved Spring XML element names which cannot be overloaded by
+     * an XML extension
+     */
+    protected static final String[] RESERVED_ELEMENT_NAMES = { 
+            "beans", 
+            DESCRIPTION_ELEMENT, 
+            DefaultBeanDefinitionDocumentReader.IMPORT_ELEMENT,
+            DefaultBeanDefinitionDocumentReader.ALIAS_ELEMENT, 
+            DefaultBeanDefinitionDocumentReader.BEAN_ELEMENT, 
+            BeanDefinitionParserDelegate.CONSTRUCTOR_ARG_ELEMENT, 
+            BeanDefinitionParserDelegate.PROPERTY_ELEMENT, 
+            BeanDefinitionParserDelegate.LOOKUP_METHOD_ELEMENT,
+            BeanDefinitionParserDelegate.REPLACED_METHOD_ELEMENT, 
+            BeanDefinitionParserDelegate.ARG_TYPE_ELEMENT, 
+            BeanDefinitionParserDelegate.REF_ELEMENT, 
+            BeanDefinitionParserDelegate.IDREF_ELEMENT, 
+            BeanDefinitionParserDelegate.VALUE_ELEMENT, 
+            BeanDefinitionParserDelegate.NULL_ELEMENT,
+            BeanDefinitionParserDelegate.LIST_ELEMENT, 
+            BeanDefinitionParserDelegate.SET_ELEMENT, 
+            BeanDefinitionParserDelegate.MAP_ELEMENT, 
+            BeanDefinitionParserDelegate.ENTRY_ELEMENT, 
+            BeanDefinitionParserDelegate.KEY_ELEMENT, 
+            BeanDefinitionParserDelegate.PROPS_ELEMENT, 
+            BeanDefinitionParserDelegate.PROP_ELEMENT,
+            QNAME_ELEMENT };
+
+    protected static final String[] RESERVED_BEAN_ATTRIBUTE_NAMES = { 
+            AbstractBeanDefinitionParser.ID_ATTRIBUTE, 
+            BeanDefinitionParserDelegate.NAME_ATTRIBUTE, 
+            BeanDefinitionParserDelegate.CLASS_ATTRIBUTE,
+            BeanDefinitionParserDelegate.PARENT_ATTRIBUTE, 
+            BeanDefinitionParserDelegate.DEPENDS_ON_ATTRIBUTE, 
+            BeanDefinitionParserDelegate.FACTORY_METHOD_ATTRIBUTE, 
+            BeanDefinitionParserDelegate.FACTORY_BEAN_ATTRIBUTE,
+            BeanDefinitionParserDelegate.DEPENDENCY_CHECK_ATTRIBUTE, 
+            BeanDefinitionParserDelegate.AUTOWIRE_ATTRIBUTE, 
+            BeanDefinitionParserDelegate.INIT_METHOD_ATTRIBUTE, 
+            BeanDefinitionParserDelegate.DESTROY_METHOD_ATTRIBUTE,
+            BeanDefinitionParserDelegate.ABSTRACT_ATTRIBUTE, 
+            BeanDefinitionParserDelegate.SINGLETON_ATTRIBUTE, 
+            BeanDefinitionParserDelegate.LAZY_INIT_ATTRIBUTE };
+
+    private static final String JAVA_PACKAGE_PREFIX = "java://";
+
+    private static final String BEAN_REFERENCE_PREFIX = "#";
+    private static final String NULL_REFERENCE = "#null";
+
+    private Set reservedElementNames = new HashSet(Arrays.asList(RESERVED_ELEMENT_NAMES));
+    private Set reservedBeanAttributeNames = new HashSet(Arrays.asList(RESERVED_BEAN_ATTRIBUTE_NAMES));
+    protected final NamedConstructorArgs namedConstructorArgs = new NamedConstructorArgs();
+
+    private ParserContext parserContext;
+    
+    private XBeanQNameHelper qnameHelper;
+
+    public void init() {
+    }
+
+    public BeanDefinition parse(Element element, ParserContext parserContext) {
+        this.parserContext = parserContext;
+        this.qnameHelper = new XBeanQNameHelper(parserContext.getReaderContext());
+        BeanDefinitionHolder holder = parseBeanFromExtensionElement(element);
+        BeanDefinitionReaderUtils.registerBeanDefinition(holder, parserContext.getRegistry());
+        BeanComponentDefinition componentDefinition = new BeanComponentDefinition(holder);
+        parserContext.getReaderContext().fireComponentRegistered(componentDefinition);
+        return holder.getBeanDefinition();
+    }
+
+    public BeanDefinitionHolder decorate(Node node, BeanDefinitionHolder definition, ParserContext parserContext) {
+        throw new IllegalArgumentException("Cannot locate BeanDefinitionDecorator for "
+                        + (node instanceof Element ? "element" : "attribute") + " [" +
+                        node.getLocalName() + "].");
+    }
+
+    /**
+     * Configures the XmlBeanDefinitionReader to work nicely with extensible XML
+     * using this reader implementation.
+     */
+    public static void configure(AbstractApplicationContext context, XmlBeanDefinitionReader reader) {
+        reader.setNamespaceAware(true);
+        reader.setValidationMode(XmlBeanDefinitionReader.VALIDATION_XSD);
+    }
+
+    /**
+     * Registers whatever custom editors we need
+     */
+    public static void registerCustomEditors(DefaultListableBeanFactory beanFactory) {
+        PropertyEditorHelper.registerCustomEditors();
+    }
+
+    /**
+     * Parses the non-standard XML element as a Spring bean definition
+     */
+    protected BeanDefinitionHolder parseBeanFromExtensionElement(Element element, String parentClass, String property) {
+        String uri = element.getNamespaceURI();
+        String localName = getLocalName(element);
+
+        MappingMetaData metadata = findNamespaceProperties(uri, localName);
+        if (metadata != null) {
+            // lets see if we configured the localName to a bean class
+            String className = getPropertyDescriptor(parentClass, property).getPropertyType().getName();
+            if (className != null) {
+                return parseBeanFromExtensionElement(element, metadata, className);
+            }
+        }
+        return null;
+    }
+
+    private BeanDefinitionHolder parseBeanFromExtensionElement(Element element, MappingMetaData metadata, String className) {
+        Element original = cloneElement(element);
+        // lets assume the class name == the package name plus the
+        element.setAttributeNS(null, "class", className);
+        addSpringAttributeValues(className, element);
+        BeanDefinitionHolder definition = parserContext.getDelegate().parseBeanDefinitionElement(element, false);
+        addAttributeProperties(definition, metadata, className, original);
+        addContentProperty(definition, metadata, element);
+        addNestedPropertyElements(definition, metadata, className, element);
+        qnameHelper.coerceNamespaceAwarePropertyValues(definition.getBeanDefinition(), element);
+        declareLifecycleMethods(definition, metadata, element);
+        resolveBeanClass((AbstractBeanDefinition) definition.getBeanDefinition(), definition.getBeanName());
+        namedConstructorArgs.processParameters(definition, metadata);
+        return definition;
+    }
+
+    protected Class resolveBeanClass(AbstractBeanDefinition bd, String beanName) {
+        if (bd.hasBeanClass()) {
+            return bd.getBeanClass();
+        }
+        try {
+            ClassLoader cl = parserContext.getReaderContext().getReader().getBeanClassLoader();
+            if (cl == null) {
+                cl = Thread.currentThread().getContextClassLoader();
+            }
+            if (cl == null) {
+                cl = getClass().getClassLoader();
+            }
+            return bd.resolveBeanClass(cl);
+        }
+        catch (ClassNotFoundException ex) {
+            throw new BeanDefinitionStoreException(bd.getResourceDescription(),
+                    beanName, "Bean class [" + bd.getBeanClassName() + "] not found", ex);
+        }
+        catch (NoClassDefFoundError err) {
+            throw new BeanDefinitionStoreException(bd.getResourceDescription(),
+                    beanName, "Class that bean class [" + bd.getBeanClassName() + "] depends on not found", err);
+        }
+    }
+
+    
+    /**
+     * Parses the non-standard XML element as a Spring bean definition
+     */
+    protected BeanDefinitionHolder parseBeanFromExtensionElement(Element element) {
+        String uri = element.getNamespaceURI();
+        String localName = getLocalName(element);
+
+        MappingMetaData metadata = findNamespaceProperties(uri, localName);
+        if (metadata != null) {
+            // lets see if we configured the localName to a bean class
+            String className = metadata.getClassName(localName);
+            if (className != null) {
+                return parseBeanFromExtensionElement(element, metadata, className);
+            } else {
+                throw new BeanDefinitionStoreException("Unrecognized xbean element mapping: " + localName + " in namespace " + uri);
+            }
+        } else {
+            if (uri == null) throw new BeanDefinitionStoreException("Unrecognized Spring element: " + localName);
+            else throw new BeanDefinitionStoreException("Unrecognized xbean namespace mapping: " + uri);
+        }
+    }
+
+    protected void addSpringAttributeValues(String className, Element element) {
+        NamedNodeMap attributes = element.getAttributes();
+        for (int i = 0, size = attributes.getLength(); i < size; i++) {
+            Attr attribute = (Attr) attributes.item(i);
+            String uri = attribute.getNamespaceURI();
+            String localName = attribute.getLocalName();
+
+            if (uri != null && (uri.equals(SPRING_SCHEMA) || uri.equals(SPRING_SCHEMA_COMPAT))) {
+                element.setAttributeNS(null, localName, attribute.getNodeValue());
+            }
+        }
+    }
+
+    /**
+     * Creates a clone of the element and its attribute (though not its content)
+     */
+    protected Element cloneElement(Element element) {
+        Element answer = element.getOwnerDocument().createElementNS(element.getNamespaceURI(), element.getNodeName());
+        NamedNodeMap attributes = element.getAttributes();
+        for (int i = 0, size = attributes.getLength(); i < size; i++) {
+            Attr attribute = (Attr) attributes.item(i);
+            String uri = attribute.getNamespaceURI();
+            answer.setAttributeNS(uri, attribute.getName(), attribute.getNodeValue());
+        }
+        return answer;
+    }
+
+    /**
+     * Parses attribute names and values as being bean property expressions
+     */
+    protected void addAttributeProperties(BeanDefinitionHolder definition, MappingMetaData metadata, String className,
+            Element element) {
+        NamedNodeMap attributes = element.getAttributes();
+        // First pass on attributes with no namespaces
+        for (int i = 0, size = attributes.getLength(); i < size; i++) {
+            Attr attribute = (Attr) attributes.item(i);
+            String uri = attribute.getNamespaceURI();
+            String localName = attribute.getLocalName();
+            // Skip namespaces
+            if (localName == null || localName.equals("xmlns") || localName.startsWith("xmlns:")) {
+                continue;
+            }
+            // Add attributes with no namespaces
+            if (isEmpty(uri) && !localName.equals("class")) {
+                boolean addProperty = true;
+                if (reservedBeanAttributeNames.contains(localName)) {
+                    // should we allow the property to shine through?
+                    PropertyDescriptor descriptor = getPropertyDescriptor(className, localName);
+                    addProperty = descriptor != null;
+                }
+                if (addProperty) {
+                    addAttributeProperty(definition, metadata, element, attribute);
+                }
+            }
+        }
+        // Second pass on attributes with namespaces
+        for (int i = 0, size = attributes.getLength(); i < size; i++) {
+            Attr attribute = (Attr) attributes.item(i);
+            String uri = attribute.getNamespaceURI();
+            String localName = attribute.getLocalName();
+            // Skip namespaces
+            if (localName == null || localName.equals("xmlns") || localName.startsWith("xmlns:")) {
+                continue;
+            }
+            // Add attributs with namespaces matching the element ns
+            if (!isEmpty(uri) && uri.equals(element.getNamespaceURI())) {
+                boolean addProperty = true;
+                if (reservedBeanAttributeNames.contains(localName)) {
+                    // should we allow the property to shine through?
+                    PropertyDescriptor descriptor = getPropertyDescriptor(className, localName);
+                    addProperty = descriptor != null;
+                }
+                if (addProperty) {
+                    addAttributeProperty(definition, metadata, element, attribute);
+                }
+            }
+        }
+    }
+
+    protected void addContentProperty(BeanDefinitionHolder definition, MappingMetaData metadata, Element element) {
+        String name = metadata.getContentProperty(getLocalName(element));
+        if (name != null) {
+            String value = getElementText(element);
+            addProperty(definition, metadata, element, name, value);
+        }
+        else {
+            StringBuffer buffer = new StringBuffer();
+            NodeList childNodes = element.getChildNodes();
+            for (int i = 0, size = childNodes.getLength(); i < size; i++) {
+                Node node = childNodes.item(i);
+                if (node instanceof Text) {
+                    buffer.append(((Text) node).getData());
+                }
+            }
+
+            ByteArrayInputStream in = new ByteArrayInputStream(buffer.toString().getBytes());
+            Properties properties = new Properties();
+            try {
+                properties.load(in);
+            }
+            catch (IOException e) {
+                return;
+            }
+            Enumeration enumeration = properties.propertyNames();
+            while (enumeration.hasMoreElements()) {
+                String propertyName = (String) enumeration.nextElement();
+                String propertyEditor = metadata.getPropertyEditor(getLocalName(element), propertyName);
+                
+                Object value = getValue(properties.getProperty(propertyName), propertyEditor);
+                definition.getBeanDefinition().getPropertyValues().addPropertyValue(propertyName, value);
+            }
+        }
+    }
+
+    protected void addAttributeProperty(BeanDefinitionHolder definition, MappingMetaData metadata, Element element,
+            Attr attribute) {
+        String localName = attribute.getLocalName();
+        String value = attribute.getValue();
+        addProperty(definition, metadata, element, localName, value);
+    }
+
+    /**
+     * Add a property onto the current BeanDefinition.
+     */
+    protected void addProperty(BeanDefinitionHolder definition, MappingMetaData metadata, Element element,
+            String localName, String value) {
+        String propertyName = metadata.getPropertyName(getLocalName(element), localName);
+        String propertyEditor = metadata.getPropertyEditor(getLocalName(element), propertyName);
+        if (propertyName != null) {
+            definition.getBeanDefinition().getPropertyValues().addPropertyValue(
+                            propertyName, getValue(value,propertyEditor));
+        }
+    }
+
+    protected Object getValue(String value, String propertyEditor) {
+        if (value == null)  return null;
+
+        //
+        // If value is #null then we are explicitly setting the value null instead of an empty string
+        //
+        if (NULL_REFERENCE.equals(value)) {
+            return null;
+        }
+
+        //
+        // If value starts with # then we have a ref
+        //
+        if (value.startsWith(BEAN_REFERENCE_PREFIX)) {
+            // strip off the #
+            value = value.substring(BEAN_REFERENCE_PREFIX.length());
+
+            // if the new value starts with a #, then we had an excaped value (e.g. ##value)
+            if (!value.startsWith(BEAN_REFERENCE_PREFIX)) {
+                return new RuntimeBeanReference(value);
+            }
+        }
+
+        if( propertyEditor!=null ) {
+        	PropertyEditor p = createPropertyEditor(propertyEditor);
+        	p.setAsText(value);
+        	return p.getValue();
+        }
+        
+        //
+        // Neither null nor a reference
+        //
+        return value;
+    }
+
+    protected PropertyEditor createPropertyEditor(String propertyEditor) {    	
+    	ClassLoader cl = Thread.currentThread().getContextClassLoader();
+    	if( cl==null ) {
+    		cl = XBeanNamespaceHandler.class.getClassLoader();
+    	}
+    	
+    	try {
+    		return (PropertyEditor)cl.loadClass(propertyEditor).newInstance();
+    	} catch (Throwable e){
+    		throw (IllegalArgumentException)new IllegalArgumentException("Could not load property editor: "+propertyEditor).initCause(e);
+    	}
+	}
+
+    protected String getLocalName(Element element) {
+        String localName = element.getLocalName();
+        if (localName == null) {
+            localName = element.getNodeName();
+        }
+        return localName;
+    }
+
+    /**
+     * Lets iterate through the children of this element and create any nested
+     * child properties
+     */
+    protected void addNestedPropertyElements(BeanDefinitionHolder definition, MappingMetaData metadata,
+            String className, Element element) {
+        NodeList nl = element.getChildNodes();
+
+        for (int i = 0; i < nl.getLength(); i++) {
+            Node node = nl.item(i);
+            if (node instanceof Element) {
+                Element childElement = (Element) node;
+                String uri = childElement.getNamespaceURI();
+                String localName = childElement.getLocalName();
+
+                if (!isEmpty(uri) || !reservedElementNames.contains(localName)) {
+                    // we could be one of the following
+                    // * the child element maps to a <property> tag with inner
+                    // tags being the bean
+                    // * the child element maps to a <property><list> tag with
+                    // inner tags being the contents of the list
+                    // * the child element maps to a <property> tag and is the
+                    // bean tag too
+                    // * the child element maps to a <property> tag and is a simple
+                    // type (String, Class, int, etc).
+                    Object value = null;
+                    String propertyName = metadata.getNestedListProperty(getLocalName(element), localName);
+                    if (propertyName != null) {
+                        value = parseListElement(childElement, propertyName);
+                    }
+                    else {
+                        propertyName = metadata.getFlatCollectionProperty(getLocalName(element), localName);
+                        if (propertyName != null) {
+                            Object def = parseBeanFromExtensionElement(childElement);
+                            PropertyValue pv = definition.getBeanDefinition().getPropertyValues().getPropertyValue(propertyName);
+                            if (pv != null) {
+                                Collection l = (Collection) pv.getValue();
+                                l.add(def);
+                                continue;
+                            } else {
+                                ManagedList l = new ManagedList();
+                                l.add(def);
+                                value = l;
+                            }
+                        } else {
+                            propertyName = metadata.getNestedProperty(getLocalName(element), localName);
+                            if (propertyName != null) {
+                                // lets find the first child bean that parses fine
+                                value = parseChildExtensionBean(childElement);
+                            }
+                        }
+                    }
+
+                    if (propertyName == null && metadata.isFlatProperty(getLocalName(element), localName)) {
+                       value = parseBeanFromExtensionElement(childElement, className, localName);
+                       propertyName = localName;
+                    }
+
+                    if (propertyName == null) {
+                        value = tryParseNestedPropertyViaIntrospection(metadata, className, childElement);
+                        propertyName = localName;
+                    }
+
+                    if (value != null) {
+                        definition.getBeanDefinition().getPropertyValues().addPropertyValue(propertyName, value);
+                    }
+                    else
+                    {
+                        /**
+                         * In this case there is no nested property, so just do a normal
+                         * addProperty like we do with attributes.
+                         */
+                        String text = getElementText(childElement);
+
+                        if (text != null) {
+                            addProperty(definition, metadata, element, localName, text);
+                        }
+                    }
+                }
+            }
+        }
+    }
+
+    /**
+     * Attempts to use introspection to parse the nested property element.
+     */
+    protected Object tryParseNestedPropertyViaIntrospection(MappingMetaData metadata, String className, Element element) {
+        String localName = getLocalName(element);
+        PropertyDescriptor descriptor = getPropertyDescriptor(className, localName);
+        if (descriptor != null) {
+            return parseNestedPropertyViaIntrospection(metadata, element, descriptor.getName(), descriptor.getPropertyType());
+        } else {
+            return parseNestedPropertyViaIntrospection(metadata, element, localName, Object.class);
+        }
+    }
+
+    /**
+     * Looks up the property decriptor for the given class and property name
+     */
+    protected PropertyDescriptor getPropertyDescriptor(String className, String localName) {
+        BeanInfo beanInfo = qnameHelper.getBeanInfo(className);
+        if (beanInfo != null) {
+            PropertyDescriptor[] descriptors = beanInfo.getPropertyDescriptors();
+            for (int i = 0; i < descriptors.length; i++) {
+                PropertyDescriptor descriptor = descriptors[i];
+                String name = descriptor.getName();
+                if (name.equals(localName)) {
+                    return descriptor;
+                }
+            }
+        }
+        return null;
+    }
+
+    /**
+     * Attempts to use introspection to parse the nested property element.
+     */
+    private Object parseNestedPropertyViaIntrospection(MappingMetaData metadata, Element element, String propertyName, Class propertyType) {
+        if (isMap(propertyType)) {
+            return parseCustomMapElement(metadata, element, propertyName);
+        } else if (isCollection(propertyType)) {
+            return parseListElement(element, propertyName);
+        } else {
+            return parseChildExtensionBean(element);
+        }
+    }
+
+    protected Object parseListElement(Element element, String name) {
+        return parserContext.getDelegate().parseListElement(element);
+    }
+
+    protected Object parseCustomMapElement(MappingMetaData metadata, Element element, String name) {
+        Map map = new HashMap();
+
+        Element parent = (Element) element.getParentNode();
+        String entryName = metadata.getMapEntryName(getLocalName(parent), name);
+        String keyName = metadata.getMapKeyName(getLocalName(parent), name);
+
+        if (entryName == null) entryName = "property";
+        if (keyName == null) keyName = "key";
+
+        // TODO : support further customizations
+        //String valueName = "value";
+        //boolean keyIsAttr = true;
+        //boolean valueIsAttr = false;
+        NodeList nl = element.getChildNodes();
+        for (int i = 0; i < nl.getLength(); i++) {
+            Node node = nl.item(i);
+            if (node instanceof Element) {
+                Element childElement = (Element) node;
+
+                String localName = childElement.getLocalName();
+                String uri = childElement.getNamespaceURI();
+                if (localName == null || localName.equals("xmlns") || localName.startsWith("xmlns:")) {
+                    continue;
+                }
+
+                // we could use namespaced attributes to differentiate real spring
+                // attributes from namespace-specific attributes
+                if (!isEmpty(uri) && localName.equals(entryName)) {
+                    String key = childElement.getAttribute(keyName);
+                    if (key == null) throw new RuntimeException("No key defined for map " + entryName);
+
+                    Object keyValue = getValue(key, null);
+
+                    Object value = getValue(getElementText(childElement), null);
+
+                    map.put(keyValue, value);
+                }
+            }
+        }
+        return map;
+    }
+
+    protected boolean isMap(Class type) {
+        return Map.class.isAssignableFrom(type);
+    }
+
+    /**
+     * Returns true if the given type is a collection type or an array
+     */
+    protected boolean isCollection(Class type) {
+        return type.isArray() || Collection.class.isAssignableFrom(type);
+    }
+
+    /**
+     * Iterates the children of this element to find the first nested bean
+     */
+    protected Object parseChildExtensionBean(Element element) {
+        NodeList nl = element.getChildNodes();
+        for (int i = 0; i < nl.getLength(); i++) {
+            Node node = nl.item(i);
+            if (node instanceof Element) {
+                Element childElement = (Element) node;
+                String uri = childElement.getNamespaceURI();
+                String localName = childElement.getLocalName();
+
+                if (uri == null || 
+                    uri.equals(SPRING_SCHEMA) || 
+                    uri.equals(SPRING_SCHEMA_COMPAT) ||
+                    uri.equals(BeanDefinitionParserDelegate.BEANS_NAMESPACE_URI)) {
+                    if (BeanDefinitionParserDelegate.BEAN_ELEMENT.equals(localName)) {
+                        return parserContext.getDelegate().parseBeanDefinitionElement(childElement, true);
+                    } else {
+                        return parserContext.getDelegate().parsePropertySubElement(childElement);
+                    }
+                } else {
+                    Object value = parseBeanFromExtensionElement(childElement);
+                    if (value != null) {
+                        return value;
+                    }
+                }
+            }
+        }
+        return null;
+    }
+
+    /**
+     * Uses META-INF/services discovery to find a Properties file with the XML
+     * marshaling configuration
+     *
+     * @param namespaceURI
+     *            the namespace URI of the element
+     * @param localName
+     *            the local name of the element
+     * @return the properties configuration of the namespace or null if none
+     *         could be found
+     */
+    protected MappingMetaData findNamespaceProperties(String namespaceURI, String localName) {
+        // lets look for the magic prefix
+        if (namespaceURI != null && namespaceURI.startsWith(JAVA_PACKAGE_PREFIX)) {
+            String packageName = namespaceURI.substring(JAVA_PACKAGE_PREFIX.length());
+            return new MappingMetaData(packageName);
+        }
+
+        String uri = NamespaceHelper.createDiscoveryPathName(namespaceURI, localName);
+        InputStream in = loadResource(uri);
+        if (in == null) {
+            if (namespaceURI != null && namespaceURI.length() > 0) {
+                uri = NamespaceHelper.createDiscoveryPathName(namespaceURI);
+                in = loadResource(uri);
+                if (in == null) {
+                    uri = NamespaceHelper.createDiscoveryOldPathName(namespaceURI);
+                    in = loadResource(uri);
+                }
+            }
+        }
+
+        if (in != null) {
+            try {
+                Properties properties = new Properties();
+                properties.load(in);
+                return new MappingMetaData(properties);
+            }
+            catch (IOException e) {
+                log.warn("Failed to load resource from uri: " + uri, e);
+            }
+        }
+        return null;
+    }
+
+    /**
+     * Loads the resource from the given URI
+     */
+    protected InputStream loadResource(String uri) {
+        if (System.getProperty("xbean.dir") != null) {
+            File f = new File(System.getProperty("xbean.dir") + uri);
+            try {
+                return new FileInputStream(f);
+            } catch (FileNotFoundException e) {
+                // Ignore
+            }
+        }
+        // lets try the thread context class loader first
+        InputStream in = Thread.currentThread().getContextClassLoader().getResourceAsStream(uri);
+        if (in == null) {
+            in = getClass().getClassLoader().getResourceAsStream(uri);
+            if (in == null) {
+                log.debug("Could not find resource: " + uri);
+            }
+        }
+        return in;
+    }
+
+    protected boolean isEmpty(String uri) {
+        return uri == null || uri.length() == 0;
+    }
+
+    protected void declareLifecycleMethods(BeanDefinitionHolder definitionHolder, MappingMetaData metaData,
+            Element element) {
+        BeanDefinition definition = definitionHolder.getBeanDefinition();
+        if (definition instanceof AbstractBeanDefinition) {
+            AbstractBeanDefinition beanDefinition = (AbstractBeanDefinition) definition;
+            if (beanDefinition.getInitMethodName() == null) {
+                beanDefinition.setInitMethodName(metaData.getInitMethodName(getLocalName(element)));
+            }
+            if (beanDefinition.getDestroyMethodName() == null) {
+                beanDefinition.setDestroyMethodName(metaData.getDestroyMethodName(getLocalName(element)));
+            }
+            if (beanDefinition.getFactoryMethodName() == null) {
+                beanDefinition.setFactoryMethodName(metaData.getFactoryMethodName(getLocalName(element)));
+            }
+        }
+    }
+
+    // -------------------------------------------------------------------------
+    //
+    // TODO we could apply the following patches into the Spring code -
+    // though who knows if it'll ever make it into a release! :)
+    //
+    // -------------------------------------------------------------------------
+    /*
+    protected int parseBeanDefinitions(Element root) throws BeanDefinitionStoreException {
+        int beanDefinitionCount = 0;
+        if (isEmpty(root.getNamespaceURI()) || root.getLocalName().equals("beans")) {
+            NodeList nl = root.getChildNodes();
+            for (int i = 0; i < nl.getLength(); i++) {
+                Node node = nl.item(i);
+                if (node instanceof Element) {
+                    Element ele = (Element) node;
+                    if (IMPORT_ELEMENT.equals(node.getNodeName())) {
+                        importBeanDefinitionResource(ele);
+                    }
+                    else if (ALIAS_ELEMENT.equals(node.getNodeName())) {
+                        String name = ele.getAttribute(NAME_ATTRIBUTE);
+                        String alias = ele.getAttribute(ALIAS_ATTRIBUTE);
+                        getBeanDefinitionReader().getBeanFactory().registerAlias(name, alias);
+                    }
+                    else if (BEAN_ELEMENT.equals(node.getNodeName())) {
+                        beanDefinitionCount++;
+                        BeanDefinitionHolder bdHolder = parseBeanDefinitionElement(ele, false);
+                        BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getBeanDefinitionReader()
+                                .getBeanFactory());
+                    }
+                    else {
+                        BeanDefinitionHolder bdHolder = parseBeanFromExtensionElement(ele);
+                        if (bdHolder != null) {
+                            beanDefinitionCount++;
+                            BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getBeanDefinitionReader()
+                                    .getBeanFactory());
+                        }
+                        else {
+                            log.debug("Ignoring unknown element namespace: " + ele.getNamespaceURI() + " localName: "
+                                    + ele.getLocalName());
+                        }
+                    }
+                }
+            }
+        } else {
+            BeanDefinitionHolder bdHolder = parseBeanFromExtensionElement(root);
+            if (bdHolder != null) {
+                beanDefinitionCount++;
+                BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getBeanDefinitionReader()
+                        .getBeanFactory());
+            }
+            else {
+                log.debug("Ignoring unknown element namespace: " + root.getNamespaceURI() + " localName: " + root.getLocalName());
+            }
+        }
+        return beanDefinitionCount;
+    }
+
+    protected BeanDefinitionHolder parseBeanDefinitionElement(Element ele, boolean isInnerBean) throws BeanDefinitionStoreException {
+        
+        BeanDefinitionHolder bdh = super.parseBeanDefinitionElement(ele, isInnerBean);
+        coerceNamespaceAwarePropertyValues(bdh, ele);
+        return bdh;
+    }
+
+    protected Object parsePropertySubElement(Element element, String beanName) throws BeanDefinitionStoreException {
+        String uri = element.getNamespaceURI();
+        String localName = getLocalName(element);
+
+        if ((!isEmpty(uri) && !(uri.equals(SPRING_SCHEMA) || uri.equals(SPRING_SCHEMA_COMPAT)))
+                || !reservedElementNames.contains(localName)) {
+            Object answer = parseBeanFromExtensionElement(element);
+            if (answer != null) {
+                return answer;
+            }
+        }
+        if (QNAME_ELEMENT.equals(localName) && isQnameIsOnClassPath()) {
+            Object answer = parseQNameElement(element);
+            if (answer != null) {
+                return answer;
+            }
+        }
+        return super.parsePropertySubElement(element, beanName);
+    }
+
+    protected Object parseQNameElement(Element element) {
+        return QNameReflectionHelper.createQName(element, getElementText(element));
+    }
+    */
+
+    /**
+     * Returns the text of the element
+     */
+    protected String getElementText(Element element) {
+        StringBuffer buffer = new StringBuffer();
+        NodeList nodeList = element.getChildNodes();
+        for (int i = 0, size = nodeList.getLength(); i < size; i++) {
+            Node node = nodeList.item(i);
+            if (node.getNodeType() == Node.TEXT_NODE) {
+                buffer.append(node.getNodeValue());
+            }
+        }
+        return buffer.toString();
+    }
+}

Modified: geronimo/xbean/trunk/xbean-spring-v2b/pom.xml
URL: http://svn.apache.org/viewvc/geronimo/xbean/trunk/xbean-spring-v2b/pom.xml?rev=438491&r1=438490&r2=438491&view=diff
==============================================================================
--- geronimo/xbean/trunk/xbean-spring-v2b/pom.xml (original)
+++ geronimo/xbean/trunk/xbean-spring-v2b/pom.xml Wed Aug 30 07:14:53 2006
@@ -27,23 +27,23 @@
     <parent>
         <artifactId>xbean</artifactId>
         <groupId>org.apache.xbean</groupId>
-        <version>2.6</version>
+        <version>2.6-SNAPSHOT</version>
     </parent>
 
-    <artifactId>xbean-spring-v2</artifactId>
-    <name>XBean :: Spring :: V2</name>
+    <artifactId>xbean-spring-v2b</artifactId>
+    <name>XBean :: Spring :: V2B</name>
 
     <dependencies>
     
         <dependency>
             <groupId>org.springframework</groupId>
             <artifactId>spring</artifactId>
-            <version>2.0-m5</version>
+            <version>2.0-rc3</version>
         </dependency>
         
         <dependency>
             <groupId>org.apache.xbean</groupId>
-            <artifactId>xbean-spring-common</artifactId>
+            <artifactId>xbean-spring-v2</artifactId>
             <version>${pom.version}</version>
             <exclusions>
                 <exclusion>
@@ -96,6 +96,15 @@
         </testResources>
 
         <plugins>
+            <plugin>
+                <groupId>org.apache.maven.plugins</groupId>
+                <artifactId>maven-surefire-plugin</artifactId>
+                <configuration>
+                    <excludes>
+                        <exclude>**/BeerUsingXBeanNSTest.java</exclude>
+                    </excludes>
+                </configuration>
+            </plugin>
             <plugin>
                 <artifactId>maven-antrun-plugin</artifactId>
                 <executions>

Modified: geronimo/xbean/trunk/xbean-spring-v2b/src/main/java/org/apache/xbean/spring/context/v2b/XBeanNamespaceHandler.java
URL: http://svn.apache.org/viewvc/geronimo/xbean/trunk/xbean-spring-v2b/src/main/java/org/apache/xbean/spring/context/v2b/XBeanNamespaceHandler.java?rev=438491&r1=438432&r2=438491&view=diff
==============================================================================
--- geronimo/xbean/trunk/xbean-spring-v2b/src/main/java/org/apache/xbean/spring/context/v2b/XBeanNamespaceHandler.java (original)
+++ geronimo/xbean/trunk/xbean-spring-v2b/src/main/java/org/apache/xbean/spring/context/v2b/XBeanNamespaceHandler.java Wed Aug 30 07:14:53 2006
@@ -14,7 +14,7 @@
  * See the License for the specific language governing permissions and
  * limitations under the License.
  */
-package org.apache.xbean.spring.context.v2;
+package org.apache.xbean.spring.context.v2b;
 
 import java.beans.BeanInfo;
 import java.beans.PropertyDescriptor;
@@ -25,7 +25,6 @@
 import java.io.FileNotFoundException;
 import java.io.IOException;
 import java.io.InputStream;
-import java.lang.reflect.Method;
 import java.util.Arrays;
 import java.util.Collection;
 import java.util.Enumeration;
@@ -41,6 +40,7 @@
 import org.apache.xbean.spring.context.impl.NamedConstructorArgs;
 import org.apache.xbean.spring.context.impl.NamespaceHelper;
 import org.apache.xbean.spring.context.impl.PropertyEditorHelper;
+import org.apache.xbean.spring.context.v2.XBeanQNameHelper;
 import org.springframework.beans.PropertyValue;
 import org.springframework.beans.factory.BeanDefinitionStoreException;
 import org.springframework.beans.factory.config.BeanDefinition;
@@ -202,7 +202,7 @@
         // lets assume the class name == the package name plus the
         element.setAttributeNS(null, "class", className);
         addSpringAttributeValues(className, element);
-        BeanDefinitionHolder definition = parserContext.getDelegate().parseBeanDefinitionElement(element, false);
+        BeanDefinitionHolder definition = parserContext.getDelegate().parseBeanDefinitionElement(element, null);
         addAttributeProperties(definition, metadata, className, original);
         addContentProperty(definition, metadata, element);
         addNestedPropertyElements(definition, metadata, className, element);
@@ -578,7 +578,7 @@
     }
 
     protected Object parseListElement(Element element, String name) {
-        return parserContext.getDelegate().parseListElement(element);
+        return parserContext.getDelegate().parseListElement(element, null);
     }
 
     protected Object parseCustomMapElement(MappingMetaData metadata, Element element, String name) {
@@ -652,9 +652,9 @@
                     uri.equals(SPRING_SCHEMA_COMPAT) ||
                     uri.equals(BeanDefinitionParserDelegate.BEANS_NAMESPACE_URI)) {
                     if (BeanDefinitionParserDelegate.BEAN_ELEMENT.equals(localName)) {
-                        return parserContext.getDelegate().parseBeanDefinitionElement(childElement, true);
+                        return parserContext.getDelegate().parseBeanDefinitionElement(childElement, null);
                     } else {
-                        return parserContext.getDelegate().parsePropertySubElement(childElement);
+                        return parserContext.getDelegate().parsePropertySubElement(childElement, null);
                     }
                 } else {
                     Object value = parseBeanFromExtensionElement(childElement);

Modified: geronimo/xbean/trunk/xbean-spring/pom.xml
URL: http://svn.apache.org/viewvc/geronimo/xbean/trunk/xbean-spring/pom.xml?rev=438491&r1=438490&r2=438491&view=diff
==============================================================================
--- geronimo/xbean/trunk/xbean-spring/pom.xml (original)
+++ geronimo/xbean/trunk/xbean-spring/pom.xml Wed Aug 30 07:14:53 2006
@@ -37,6 +37,13 @@
 
         <dependency>
             <groupId>org.apache.xbean</groupId>
+            <artifactId>xbean-spring-common</artifactId>
+            <version>${pom.version}</version>
+            <optional>true</optional>
+        </dependency>
+
+        <dependency>
+            <groupId>org.apache.xbean</groupId>
             <artifactId>xbean-spring-v1</artifactId>
             <version>${pom.version}</version>
             <optional>true</optional>
@@ -50,6 +57,20 @@
         </dependency>
 
         <dependency>
+            <groupId>org.apache.xbean</groupId>
+            <artifactId>xbean-spring-v2a</artifactId>
+            <version>${pom.version}</version>
+            <optional>true</optional>
+        </dependency>
+
+        <dependency>
+            <groupId>org.apache.xbean</groupId>
+            <artifactId>xbean-spring-v2b</artifactId>
+            <version>${pom.version}</version>
+            <optional>true</optional>
+        </dependency>
+
+        <dependency>
             <groupId>commons-logging</groupId>
             <artifactId>commons-logging</artifactId>
         </dependency>
@@ -89,6 +110,16 @@
                                 <artifactItem>
                                     <groupId>org.apache.xbean</groupId>
                                     <artifactId>xbean-spring-v2</artifactId>
+                                    <version>${pom.version}</version>
+                                </artifactItem>
+                                <artifactItem>
+                                    <groupId>org.apache.xbean</groupId>
+                                    <artifactId>xbean-spring-v2a</artifactId>
+                                    <version>${pom.version}</version>
+                                </artifactItem>
+                                <artifactItem>
+                                    <groupId>org.apache.xbean</groupId>
+                                    <artifactId>xbean-spring-v2b</artifactId>
                                     <version>${pom.version}</version>
                                 </artifactItem>
                             </artifactItems>