blob: c4ce4176e84bdeeb8ac0c02151b333d7cbec2fbf [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
*
* 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.cocoon.spring.configurator.impl;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
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.BeanDefinitionReaderUtils;
import org.springframework.beans.factory.support.BeanDefinitionRegistry;
import org.springframework.beans.factory.support.RootBeanDefinition;
import org.springframework.beans.factory.xml.BeanDefinitionParser;
import org.springframework.beans.factory.xml.BeanDefinitionParserDelegate;
import org.springframework.util.StringUtils;
import org.apache.cocoon.configuration.Settings;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
/**
* This is a base class for all bean definition parsers used in Cocoon. It
* provides some utility methods.
*
* @since 1.0
* @version $Id$
*/
public abstract class AbstractElementParser implements BeanDefinitionParser {
/** Logger (we use the same logging mechanism as Spring!) */
protected final Log logger = LogFactory.getLog(getClass());
/**
* Get the value of an attribute or if the attribute is not present return
* the default value. As Element#getAttribute(String) returns an empty string
* and not null, we have to check this.
*/
protected String getAttributeValue(Element element, String attributeName, String defaultValue) {
String value = element.getAttribute(attributeName);
if (value == null || value.trim().length() == 0) {
value = defaultValue;
}
return value;
}
/**
* Returns all Element children of an Element that have the given local name.
*/
protected Element[] getChildElements(Element element, String localName) {
final ArrayList elements = new ArrayList();
final NodeList nodeList = element.getChildNodes();
for (int i = 0; i < nodeList.getLength(); i++) {
final Node node = nodeList.item(i);
if (node instanceof Element && localName.equals(node.getLocalName())) {
elements.add(node);
}
}
return (Element[]) elements.toArray(new Element[elements.size()]);
}
/**
* Register a bean definition.
*
* @param beanDef The bean definition.
* @param beanName The name of the bean.
* @param registry The registry.
*/
protected void register(BeanDefinition beanDef, String beanName, BeanDefinitionRegistry registry) {
register(beanDef, beanName, null, registry);
}
/**
* Register a bean definition.
*
* @param beanDef The bean definition.
* @param beanName The name of the bean.
* @param alias Optional alias.
* @param registry The registry.
*/
protected void register(BeanDefinition beanDef, String beanName, String alias, BeanDefinitionRegistry registry) {
if (this.logger.isDebugEnabled()) {
this.logger.debug("Registering bean with name " + beanName +
(alias != null ? " (alias=" + alias + ") " : " ") + beanDef);
}
final BeanDefinitionHolder holder;
if (alias != null) {
holder = new BeanDefinitionHolder(beanDef, beanName, new String[] { alias });
} else {
holder = new BeanDefinitionHolder(beanDef, beanName);
}
BeanDefinitionReaderUtils.registerBeanDefinition(holder, registry);
}
/**
* Register a global bean definition.
* The provided xml element is searched for an id and/or name attribute to register
* the bean. This implementation works the same as the default spring implementation.
*
* @param beanDef The bean definition.
* @param element The xml element defining the bean.
* @param registry The registry.
* @since 1.0.1
*/
protected void register(BeanDefinition beanDef, Element element, BeanDefinitionRegistry registry) {
String id = element.getAttribute(BeanDefinitionParserDelegate.ID_ATTRIBUTE);
String nameAttr = element.getAttribute(BeanDefinitionParserDelegate.NAME_ATTRIBUTE);
final List aliases = new ArrayList();
if (StringUtils.hasLength(nameAttr)) {
String[] nameArr = StringUtils.tokenizeToStringArray(nameAttr, BeanDefinitionParserDelegate.BEAN_NAME_DELIMITERS);
aliases.addAll(Arrays.asList(nameArr));
}
String beanName = id;
if (!StringUtils.hasText(beanName) && !aliases.isEmpty()) {
beanName = (String) aliases.remove(0);
if (this.logger.isDebugEnabled()) {
this.logger.debug("No XML 'id' specified - using '" + beanName +
"' as bean name and " + aliases + " as aliases");
}
}
final BeanDefinitionHolder holder = new BeanDefinitionHolder(beanDef, beanName, StringUtils.toStringArray(aliases));
BeanDefinitionReaderUtils.registerBeanDefinition(holder, registry);
}
/**
* Helper method to create a new bean definition.
*
* @param componentClass The class of the implementation.
* @param initMethod Optional initialization method.
* @param requiresSettings If set to true, this bean has a property "settings" for the
* settings object.
* @return A new root bean definition.
*/
protected RootBeanDefinition createBeanDefinition(Class componentClass, String initMethod, boolean requiresSettings) {
final RootBeanDefinition beanDef = new RootBeanDefinition();
beanDef.setBeanClass(componentClass);
fillBeanDefinition(beanDef, initMethod, requiresSettings);
return beanDef;
}
/**
* Helper method to create a new bean definition.
*
* @param componentClass The class of the implementation.
* @param initMethod Optional initialization method.
* @param requiresSettings If set to true, this bean has a property "settings" for the
* settings object.
* @return A new root bean definition.
*/
protected RootBeanDefinition createBeanDefinition(String componentClass, String initMethod, boolean requiresSettings) {
final RootBeanDefinition beanDef = new RootBeanDefinition();
beanDef.setBeanClassName(componentClass);
fillBeanDefinition(beanDef, initMethod, requiresSettings);
return beanDef;
}
private void fillBeanDefinition(RootBeanDefinition beanDef, String initMethod, boolean requiresSettings) {
beanDef.setScope(BeanDefinition.SCOPE_SINGLETON);
beanDef.setLazyInit(false);
if (initMethod != null) {
beanDef.setInitMethodName(initMethod);
}
if (requiresSettings) {
beanDef.getPropertyValues().addPropertyValue("settings", new RuntimeBeanReference(Settings.ROLE));
}
}
/**
* Add a new bean definition to the registry.
*
* @param componentClass The class of the implementation.
* @param beanName The name of the bean.
* @param initMethod Optional initialization method.
* @param requiresSettings If set to true, this bean has a property "settings" for the
* settings object.
* @param registry The bean registry.
*/
protected void addComponent(Class componentClass, String beanName,
String initMethod, boolean requiresSettings,
BeanDefinitionRegistry registry) {
final RootBeanDefinition beanDef = createBeanDefinition(componentClass, initMethod, requiresSettings);
register(beanDef, beanName, registry);
}
/**
* Add a new bean definition to the registry.
*
* @param componentClass The class of the implementation.
* @param beanName The name of the bean.
* @param initMethod Optional initialization method.
* @param requiresSettings If set to true, this bean has a property "settings" for the
* settings object.
* @param registry The bean registry.
*/
protected void addComponent(String componentClass, String beanName,
String initMethod, boolean requiresSettings,
BeanDefinitionRegistry registry) {
final RootBeanDefinition beanDef = createBeanDefinition(componentClass, initMethod, requiresSettings);
register(beanDef, beanName, registry);
}
}