blob: 996279ad432a7b0cfeff0d3b424587eee6631011 [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.felix.scrplugin.xml;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;
import java.util.StringTokenizer;
import javax.xml.transform.TransformerException;
import org.apache.felix.scrplugin.Log;
import org.apache.felix.scrplugin.Options;
import org.apache.felix.scrplugin.SCRDescriptorException;
import org.apache.felix.scrplugin.SCRDescriptorFailureException;
import org.apache.felix.scrplugin.SpecVersion;
import org.apache.felix.scrplugin.description.AbstractDescription;
import org.apache.felix.scrplugin.description.ClassDescription;
import org.apache.felix.scrplugin.description.ComponentConfigurationPolicy;
import org.apache.felix.scrplugin.description.ComponentDescription;
import org.apache.felix.scrplugin.description.PropertyDescription;
import org.apache.felix.scrplugin.description.PropertyType;
import org.apache.felix.scrplugin.description.PropertyUnbounded;
import org.apache.felix.scrplugin.description.ReferenceCardinality;
import org.apache.felix.scrplugin.description.ReferenceDescription;
import org.apache.felix.scrplugin.description.ReferencePolicy;
import org.apache.felix.scrplugin.description.ReferencePolicyOption;
import org.apache.felix.scrplugin.description.ReferenceStrategy;
import org.apache.felix.scrplugin.description.ServiceDescription;
import org.apache.felix.scrplugin.helper.ComponentContainer;
import org.apache.felix.scrplugin.helper.ComponentContainerUtil;
import org.apache.felix.scrplugin.helper.ComponentContainerUtil.ComponentContainerContainer;
import org.apache.felix.scrplugin.helper.DescriptionContainer;
import org.apache.felix.scrplugin.helper.IssueLog;
import org.xml.sax.Attributes;
import org.xml.sax.ContentHandler;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.AttributesImpl;
import org.xml.sax.helpers.DefaultHandler;
/**
* <code>ComponentDescriptorIO</code>
*
* is a helper class to read and write component descriptor files.
*
*/
public class ComponentDescriptorIO {
private static final String PROPERTY_ATTR_TYPE = "type";
/** General attribute for the name (component, reference, property) */
private static final String ATTR_NAME = "name";
private static final String ATTR_CARDINALITY = "cardinality";
private static final String ATTR_DESCRIPTION = "description";
private static final String ATTR_LABEL = "label";
/** The inner namespace - used for all inner elements. */
public static final String INNER_NAMESPACE_URI = "";
/** The prefix used for the namespace. */
private static final String PREFIX = "scr";
/** The root element. */
private static final String COMPONENTS = "components";
/** The component element. */
private static final String COMPONENT = "component";
/** The qualified component element. */
private static final String COMPONENT_QNAME = PREFIX + ':' + COMPONENT;
/** The enabled attribute. */
private static final String COMPONENT_ATTR_ENABLED = "enabled";
/** Component: The policy attribute. */
private static final String COMPONENT_ATTR_POLICY = "configuration-policy";
/** Component: The factory attribute. */
private static final String COMPONENT_ATTR_FACTORY = "factory";
/** Component: The immediate attribute. */
private static final String COMPONENT_ATTR_IMMEDIATE = "immediate";
/** Component: The activate attribute. */
private static final String COMPONENT_ATTR_ACTIVATE = "activate";
/** Component: The deactivate attribute. */
private static final String COMPONENT_ATTR_DEACTIVATE = "deactivate";
/** Component: The modified attribute. */
private static final String COMPONENT_ATTR_MODIFIED = "modified";
/** Component: The configuration pid attribute. */
private static final String COMPONENT_ATTR_CONFIGURATION_PID = "configuration-pid";
private static final String IMPLEMENTATION = "implementation";
private static final String IMPLEMENTATION_QNAME = IMPLEMENTATION;
private static final String IMPLEMENTATION_ATTR_CLASS = "class";
private static final String SERVICE = "service";
private static final String SERVICE_QNAME = SERVICE;
private static final String SERVICE_ATTR_FACTORY = "servicefactory";
private static final String PROPERTY = "property";
private static final String PROPERTY_QNAME = PROPERTY;
private static final String PROPERTY_ATTR_VALUE = "value";
private static final String PROPERTY_ATTR_PRIVATE = "private";
private static final String REFERENCE = "reference";
private static final String REFERENCE_QNAME = REFERENCE;
private static final String REFERENCE_ATTR_POLICY = "policy";
private static final String REFERENCE_ATTR_POLICY_OPTION = "policy-option";
private static final String REFERENCE_ATTR_UPDATED = "updated";
private static final String REFERENCE_ATTR_UNBIND = "unbind";
private static final String REFERENCE_ATTR_BIND = "bind";
private static final String REFERENCE_ATTR_TARGET = "target";
private static final String REFERENCE_ATTR_STRATEGY = "strategy";
private static final String INTERFACE = "provide";
private static final String INTERFACE_QNAME = INTERFACE;
private static final String INTERFACE_ATTR_NAME = "interface";
private static final String PROPERTIES = "properties";
public static List<ClassDescription> read(final InputStream file,
final ClassLoader classLoader,
final IssueLog iLog, final String location) throws SCRDescriptorException {
try {
final XmlHandler xmlHandler = new XmlHandler(classLoader, iLog, location);
IOUtils.parse(file, xmlHandler);
return xmlHandler.components;
} catch (final TransformerException e) {
throw new SCRDescriptorException("Unable to read xml", location, e);
}
}
/**
* Generate the xml top level element and start streaming
* the components.
*
* @param components
* @param contentHandler
* @throws SAXException
*/
private static void generateXML(final DescriptionContainer module,
final List<ComponentContainer> components,
final File descriptorFile,
final Log logger) throws SAXException, IOException, TransformerException {
logger.info("Writing " + components.size() + " Service Component Descriptors to "
+ descriptorFile);
FileOutputStream fos = new FileOutputStream(descriptorFile);
try {
final ContentHandler contentHandler = IOUtils.getSerializer(fos);
// detect namespace to use
final String namespace = module.getOptions().getSpecVersion().getNamespaceUrl();
contentHandler.startDocument();
contentHandler.startPrefixMapping(PREFIX, namespace);
IOUtils.newline(contentHandler);
// wrapper element to generate well formed xml if 0 or more than 1 component
int startIndent = 0;
if ( components.size() != 1 ) {
contentHandler.startElement("", ComponentDescriptorIO.COMPONENTS, ComponentDescriptorIO.COMPONENTS, new AttributesImpl());
IOUtils.newline(contentHandler);
startIndent = 1;
}
for (final ComponentContainer component : components) {
generateXML(namespace, module, component, contentHandler, startIndent);
}
// end wrapper element
if ( components.size() != 1 ) {
contentHandler.endElement("", ComponentDescriptorIO.COMPONENTS, ComponentDescriptorIO.COMPONENTS);
IOUtils.newline(contentHandler);
}
contentHandler.endPrefixMapping(PREFIX);
contentHandler.endDocument();
}
finally {
fos.close();
}
}
/**
* Write the xml for a Component
*
* @param component
* @param contentHandler
* @throws SAXException
*/
private static void generateXML(final String namespace,
final DescriptionContainer module,
final ComponentContainer container,
final ContentHandler contentHandler,
final int indent)
throws SAXException {
final ComponentDescription component = container.getComponentDescription();
final AttributesImpl ai = new AttributesImpl();
IOUtils.addAttribute(ai, COMPONENT_ATTR_ENABLED, component.getEnabled());
IOUtils.addAttribute(ai, COMPONENT_ATTR_IMMEDIATE, component.getImmediate());
IOUtils.addAttribute(ai, ATTR_NAME, component.getName());
IOUtils.addAttribute(ai, COMPONENT_ATTR_FACTORY, component.getFactory());
// attributes new in 1.1
if (module.getOptions().getSpecVersion().ordinal() >= SpecVersion.VERSION_1_1.ordinal() ) {
if ( component.getConfigurationPolicy() != null
&& component.getConfigurationPolicy() != ComponentConfigurationPolicy.OPTIONAL ) {
IOUtils.addAttribute(ai, COMPONENT_ATTR_POLICY, component.getConfigurationPolicy().name().toLowerCase());
}
IOUtils.addAttribute(ai, COMPONENT_ATTR_ACTIVATE, component.getActivate());
IOUtils.addAttribute(ai, COMPONENT_ATTR_DEACTIVATE, component.getDeactivate());
IOUtils.addAttribute(ai, COMPONENT_ATTR_MODIFIED, component.getModified());
}
// attributes new in 1.2
if ( module.getOptions().getSpecVersion().ordinal() >= SpecVersion.VERSION_1_2.ordinal() ) {
if ( component.getConfigurationPid() != null && !component.getConfigurationPid().equals(component.getName())) {
IOUtils.addAttribute(ai, COMPONENT_ATTR_CONFIGURATION_PID, component.getConfigurationPid());
}
}
IOUtils.indent(contentHandler, indent);
contentHandler.startElement(namespace, ComponentDescriptorIO.COMPONENT, ComponentDescriptorIO.COMPONENT_QNAME, ai);
IOUtils.newline(contentHandler);
for (final PropertyDescription property : container.getProperties().values()) {
generatePropertyXML(property, contentHandler, indent+1);
}
if (container.getServiceDescription() != null) {
generateServiceXML(container.getServiceDescription(), contentHandler, indent+1);
}
for (final ReferenceDescription reference : container.getReferences().values()) {
generateReferenceXML(component, module, reference, contentHandler, indent+1);
}
generateImplementationXML(container, contentHandler, indent+1);
IOUtils.indent(contentHandler, indent);
contentHandler.endElement(namespace, ComponentDescriptorIO.COMPONENT, ComponentDescriptorIO.COMPONENT_QNAME);
IOUtils.newline(contentHandler);
}
/**
* Write the xml for an Implementation.
*
* @param implementation
* @param contentHandler
* @throws SAXException
*/
private static void generateImplementationXML(final ComponentContainer component,
final ContentHandler contentHandler,
final int indent)
throws SAXException {
final AttributesImpl ai = new AttributesImpl();
IOUtils.addAttribute(ai, IMPLEMENTATION_ATTR_CLASS, component.getClassDescription().getDescribedClass().getName());
IOUtils.indent(contentHandler, indent);
contentHandler.startElement(INNER_NAMESPACE_URI, ComponentDescriptorIO.IMPLEMENTATION,
ComponentDescriptorIO.IMPLEMENTATION_QNAME, ai);
contentHandler.endElement(INNER_NAMESPACE_URI, ComponentDescriptorIO.IMPLEMENTATION,
ComponentDescriptorIO.IMPLEMENTATION_QNAME);
IOUtils.newline(contentHandler);
}
/**
* Write the xml for a service.
*
* @param service
* @param contentHandler
* @throws SAXException
*/
private static void generateServiceXML(
final ServiceDescription service,
final ContentHandler contentHandler,
final int indent)
throws SAXException {
final AttributesImpl ai = new AttributesImpl();
IOUtils.addAttribute(ai, SERVICE_ATTR_FACTORY, String.valueOf(service.isServiceFactory()));
IOUtils.indent(contentHandler, indent);
contentHandler.startElement(INNER_NAMESPACE_URI, ComponentDescriptorIO.SERVICE, ComponentDescriptorIO.SERVICE_QNAME, ai);
if (service.getInterfaces() != null && service.getInterfaces().size() > 0) {
IOUtils.newline(contentHandler);
for (final String interf : service.getInterfaces()) {
generateServiceXML(interf, contentHandler, indent+1);
}
IOUtils.indent(contentHandler, indent);
}
contentHandler.endElement(INNER_NAMESPACE_URI, ComponentDescriptorIO.SERVICE, ComponentDescriptorIO.SERVICE_QNAME);
IOUtils.newline(contentHandler);
}
/**
* Write the xml for a interface
*
* @param interf
* @param contentHandler
* @throws SAXException
*/
private static void generateServiceXML(final String interfaceName,
final ContentHandler contentHandler,
final int indent)
throws SAXException {
final AttributesImpl ai = new AttributesImpl();
IOUtils.addAttribute(ai, INTERFACE_ATTR_NAME, interfaceName);
IOUtils.indent(contentHandler, indent);
contentHandler.startElement(INNER_NAMESPACE_URI, ComponentDescriptorIO.INTERFACE, ComponentDescriptorIO.INTERFACE_QNAME,
ai);
contentHandler.endElement(INNER_NAMESPACE_URI, ComponentDescriptorIO.INTERFACE, ComponentDescriptorIO.INTERFACE_QNAME);
IOUtils.newline(contentHandler);
}
/**
* Write the xml for a property.
*
* @param property
* @param contentHandler
* @throws SAXException
*/
private static void generatePropertyXML(final PropertyDescription property,
final ContentHandler contentHandler,
final int indent)
throws SAXException {
final AttributesImpl ai = new AttributesImpl();
IOUtils.addAttribute(ai, ATTR_NAME, property.getName());
if ( property.getType() != PropertyType.String && property.getType() != PropertyType.Password) {
IOUtils.addAttribute(ai, PROPERTY_ATTR_TYPE, property.getType());
}
String value = property.getValue();
if ( value != null ) {
if ( property.getType() == PropertyType.Character || property.getType() == PropertyType.Char ) {
value = String.valueOf((int)value.charAt(0));
}
IOUtils.addAttribute(ai, PROPERTY_ATTR_VALUE, value);
}
IOUtils.indent(contentHandler, indent);
contentHandler.startElement(INNER_NAMESPACE_URI, ComponentDescriptorIO.PROPERTY, ComponentDescriptorIO.PROPERTY_QNAME, ai);
if (property.getMultiValue() != null && property.getMultiValue().length > 0) {
// generate a new line first
IOUtils.text(contentHandler, "\n");
for (int i = 0; i < property.getMultiValue().length; i++) {
IOUtils.indent(contentHandler, indent + 1);
value = property.getMultiValue()[i];
if ( property.getType() == PropertyType.Character || property.getType() == PropertyType.Char ) {
value = String.valueOf((int)value.charAt(0));
}
IOUtils.text(contentHandler, value);
IOUtils.newline(contentHandler);
}
IOUtils.indent(contentHandler, indent);
}
contentHandler.endElement(INNER_NAMESPACE_URI, ComponentDescriptorIO.PROPERTY, ComponentDescriptorIO.PROPERTY_QNAME);
IOUtils.newline(contentHandler);
}
/**
* Write the xml for a Reference.
*
* @param reference
* @param contentHandler
* @throws SAXException
*/
private static void generateReferenceXML(final ComponentDescription component,
final DescriptionContainer module,
final ReferenceDescription reference,
final ContentHandler contentHandler,
final int indent)
throws SAXException {
final AttributesImpl ai = new AttributesImpl();
IOUtils.addAttribute(ai, ATTR_NAME, reference.getName());
IOUtils.addAttribute(ai, INTERFACE_ATTR_NAME, reference.getInterfaceName());
IOUtils.addAttribute(ai, ATTR_CARDINALITY, reference.getCardinality().getCardinalityString());
IOUtils.addAttribute(ai, REFERENCE_ATTR_POLICY, reference.getPolicy().name().toLowerCase());
IOUtils.addAttribute(ai, REFERENCE_ATTR_TARGET, reference.getTarget());
IOUtils.addAttribute(ai, REFERENCE_ATTR_BIND, reference.getBind());
IOUtils.addAttribute(ai, REFERENCE_ATTR_UNBIND, reference.getUnbind());
// attributes new in 1.1-felix (FELIX-1893)
if (module.getOptions().getSpecVersion().ordinal() >= SpecVersion.VERSION_1_1_FELIX.ordinal() ) {
IOUtils.addAttribute(ai, REFERENCE_ATTR_UPDATED, reference.getUpdated());
}
// attributes new in 1.2
if (module.getOptions().getSpecVersion().ordinal() >= SpecVersion.VERSION_1_2.ordinal() ) {
if ( reference.getPolicyOption() != ReferencePolicyOption.RELUCTANT ) {
IOUtils.addAttribute(ai, REFERENCE_ATTR_POLICY_OPTION, reference.getPolicyOption().name().toLowerCase());
}
}
IOUtils.indent(contentHandler, indent);
contentHandler.startElement(INNER_NAMESPACE_URI, ComponentDescriptorIO.REFERENCE, ComponentDescriptorIO.REFERENCE_QNAME,
ai);
contentHandler.endElement(INNER_NAMESPACE_URI, ComponentDescriptorIO.REFERENCE, ComponentDescriptorIO.REFERENCE_QNAME);
IOUtils.newline(contentHandler);
}
/**
* A content handler for parsing the component descriptions.
*
*/
protected static final class XmlHandler extends DefaultHandler {
/** The components container. */
private final List<ClassDescription> components = new ArrayList<ClassDescription>();
/** Spec version. */
private SpecVersion specVersion;
/** A reference to the current class. */
private ClassDescription currentClass;
/** A reference to the current component. */
private ComponentDescription currentComponent;
/** The current service. */
private ServiceDescription currentService;
/** Pending property. */
private PropertyDescription pendingProperty;
/** Flag for detecting the first element. */
private boolean firstElement = true;
/** Flag for elements inside a component element */
private boolean isComponent = false;
/** Override namespace. */
private String overrideNamespace;
/** The issue log. */
private final IssueLog iLog;
/** XML file location. */
private final String location;
/** Properties. */
private final List<AbstractDescription> propsAndRefs = new ArrayList<AbstractDescription>();
/** Classloader. */
private final ClassLoader classLoader;
public XmlHandler(final ClassLoader classLoader, final IssueLog iLog, final String loc) {
this.iLog = iLog;
this.location = loc;
this.classLoader = classLoader;
}
/**
* @see org.xml.sax.helpers.DefaultHandler#startElement(java.lang.String, java.lang.String, java.lang.String, org.xml.sax.Attributes)
*/
@Override
public void startElement(String uri, final String localName, final String name, final Attributes attributes)
throws SAXException {
// according to the spec, the elements should have the namespace,
// except when the root element is the "component" element
// So we check this for the first element, we receive.
if (this.firstElement) {
this.firstElement = false;
if (localName.equals(COMPONENT) && "".equals(uri)) {
this.overrideNamespace = SpecVersion.VERSION_1_0.getNamespaceUrl();
}
}
if (this.overrideNamespace != null && "".equals(uri)) {
uri = this.overrideNamespace;
}
// however the spec also states that the inner elements
// of a component are unqualified, so they don't have
// the namespace - we allow both: with or without namespace!
if (this.isComponent && "".equals(uri)) {
uri = SpecVersion.VERSION_1_0.getNamespaceUrl();
}
// from here on, uri has the namespace regardless of the used xml format
specVersion = SpecVersion.fromNamespaceUrl(uri);
if (specVersion != null) {
if (localName.equals(COMPONENT)) {
this.isComponent = true;
final ComponentDescription desc = new ComponentDescription(null);
desc.setName(attributes.getValue(ATTR_NAME));
// enabled attribute is optional
if (attributes.getValue(COMPONENT_ATTR_ENABLED) != null) {
desc.setEnabled(Boolean.valueOf(attributes.getValue(COMPONENT_ATTR_ENABLED)));
}
// immediate attribute is optional
if (attributes.getValue(COMPONENT_ATTR_IMMEDIATE) != null) {
desc.setImmediate(Boolean.valueOf(attributes.getValue(COMPONENT_ATTR_IMMEDIATE)));
}
desc.setFactory(attributes.getValue(COMPONENT_ATTR_FACTORY));
desc.setConfigurationPolicy(ComponentConfigurationPolicy.OPTIONAL);
// check for version 1.1 attributes
if (specVersion.ordinal() >= SpecVersion.VERSION_1_1.ordinal()) {
final String policy = attributes.getValue(COMPONENT_ATTR_POLICY);
if ( policy != null ) {
try {
desc.setConfigurationPolicy(ComponentConfigurationPolicy.valueOf(policy.toUpperCase()));
} catch (final IllegalArgumentException iae) {
iLog.addWarning("Invalid value for attribute " + COMPONENT_ATTR_POLICY + " : " + policy, this.location);
}
}
if ( attributes.getValue(COMPONENT_ATTR_ACTIVATE) != null ) {
desc.setActivate(attributes.getValue(COMPONENT_ATTR_ACTIVATE));
}
if ( attributes.getValue(COMPONENT_ATTR_DEACTIVATE) != null ) {
desc.setDeactivate(attributes.getValue(COMPONENT_ATTR_DEACTIVATE));
}
if ( attributes.getValue(COMPONENT_ATTR_MODIFIED) != null ) {
desc.setModified(attributes.getValue(COMPONENT_ATTR_MODIFIED));
}
}
this.currentComponent = desc;
} else if (localName.equals(IMPLEMENTATION)) {
// now we can create the class description and attach the component description
// Set the implementation class name (mandatory)
final String className = attributes.getValue(IMPLEMENTATION_ATTR_CLASS);
Class<?> cl = null;
try {
cl = this.classLoader.loadClass(className);
} catch (final Throwable e) {
// this doesn't have an effect as the classes we processed are loaded
// anyway.
}
this.currentClass = new ClassDescription(cl, "classpath:" + className);
this.currentClass.add(this.currentComponent);
this.components.add(this.currentClass);
} else if (localName.equals(PROPERTY)) {
// read the property, unless it is the service.pid
// property which must not be inherited
final String propName = attributes.getValue(ATTR_NAME);
if (!org.osgi.framework.Constants.SERVICE_PID.equals(propName)) {
final PropertyDescription prop = new PropertyDescription(null);
prop.setName(propName);
final String type = attributes.getValue(PROPERTY_ATTR_TYPE);
if ( type != null ) {
try {
prop.setType(PropertyType.valueOf(type));
} catch (final IllegalArgumentException iae) {
iLog.addWarning("Invalid value for attribute type : " + type, this.location);
}
}
if ( prop.getType() == null ) {
prop.setType(PropertyType.String);
}
if (attributes.getValue(PROPERTY_ATTR_VALUE) != null) {
if ( prop.getType() == PropertyType.Char || prop.getType() == PropertyType.Character ) {
final int val = Integer.valueOf(attributes.getValue(PROPERTY_ATTR_VALUE));
final Character c = Character.valueOf((char)val);
prop.setValue(c.toString());
} else {
prop.setValue(attributes.getValue(PROPERTY_ATTR_VALUE));
}
propsAndRefs.add(prop);
} else {
// hold the property pending as we have a multi value
this.pendingProperty = prop;
}
// check for abstract properties
prop.setLabel(attributes.getValue(ATTR_LABEL));
prop.setDescription(attributes.getValue(ATTR_DESCRIPTION));
final String cardinality = attributes.getValue(ATTR_CARDINALITY);
prop.setUnbounded(PropertyUnbounded.DEFAULT);
if ( cardinality != null ) {
prop.setCardinality(Integer.valueOf(cardinality));
if ( prop.getCardinality() == Integer.MAX_VALUE ) {
prop.setCardinality(0);
prop.setUnbounded(PropertyUnbounded.ARRAY);
} else if ( prop.getCardinality() == Integer.MIN_VALUE ) {
prop.setCardinality(0);
prop.setUnbounded(PropertyUnbounded.VECTOR);
}
}
final String pValue = attributes.getValue(PROPERTY_ATTR_PRIVATE);
if (pValue != null) {
prop.setPrivate(Boolean.valueOf(pValue));
}
}
} else if (localName.equals(PROPERTIES)) {
// TODO: implement the properties tag
} else if (localName.equals(SERVICE)) {
this.currentService = new ServiceDescription(null);
if (attributes.getValue(SERVICE_ATTR_FACTORY) != null) {
this.currentService.setServiceFactory(Boolean.valueOf(attributes.getValue(SERVICE_ATTR_FACTORY)));
}
} else if (localName.equals(INTERFACE)) {
this.currentService.addInterface(attributes.getValue(INTERFACE_ATTR_NAME));
} else if (localName.equals(REFERENCE)) {
final ReferenceDescription ref = new ReferenceDescription(null);
ref.setName(attributes.getValue(ATTR_NAME));
ref.setInterfaceName(attributes.getValue(INTERFACE_ATTR_NAME));
final String cardinality = attributes.getValue(ATTR_CARDINALITY);
if ( cardinality != null ) {
ref.setCardinality(ReferenceCardinality.fromValue(cardinality));
if ( ref.getCardinality() == null ) {
iLog.addWarning("Invalid value for attribute cardinality : " + cardinality, this.location);
}
}
ref.setPolicy(ReferencePolicy.STATIC);
final String policy = attributes.getValue(REFERENCE_ATTR_POLICY);
if ( policy != null ) {
try {
ref.setPolicy(ReferencePolicy.valueOf(policy.toUpperCase()));
} catch (final IllegalArgumentException iae) {
iLog.addWarning("Invalid value for attribute policy : " + policy, this.location);
}
}
ref.setPolicyOption(ReferencePolicyOption.RELUCTANT);
final String policyOption = attributes.getValue(REFERENCE_ATTR_POLICY_OPTION);
if ( policyOption != null ) {
try {
ref.setPolicyOption(ReferencePolicyOption.valueOf(policyOption.toUpperCase()));
} catch (final IllegalArgumentException iae) {
iLog.addWarning("Invalid value for attribute policy-option : " + policyOption, this.location);
}
}
ref.setTarget(attributes.getValue(REFERENCE_ATTR_TARGET));
if ( attributes.getValue(REFERENCE_ATTR_BIND) != null ) {
ref.setBind(attributes.getValue(REFERENCE_ATTR_BIND));
}
if ( attributes.getValue(REFERENCE_ATTR_UNBIND) != null ) {
ref.setUnbind(attributes.getValue(REFERENCE_ATTR_UNBIND));
}
if ( attributes.getValue(REFERENCE_ATTR_UPDATED) != null ) {
ref.setUnbind(attributes.getValue(REFERENCE_ATTR_UPDATED));
}
final String strategy = attributes.getValue(REFERENCE_ATTR_STRATEGY);
if ( strategy != null ) {
try {
ref.setStrategy(ReferenceStrategy.valueOf(strategy.toUpperCase()));
} catch (final IllegalArgumentException iae) {
throw new SAXException("Invalid value for attribute strategy : " + strategy);
}
}
propsAndRefs.add(ref);
}
}
}
/**
* @see org.xml.sax.helpers.DefaultHandler#endElement(java.lang.String, java.lang.String, java.lang.String)
*/
@Override
public void endElement(String uri, String localName, String name) throws SAXException {
if (this.overrideNamespace != null && "".equals(uri)) {
uri = this.overrideNamespace;
}
if (this.isComponent && "".equals(uri)) {
uri = SpecVersion.VERSION_1_0.getNamespaceUrl();
}
if (SpecVersion.fromNamespaceUrl(uri) != null) {
if (localName.equals(COMPONENT)) {
if ( this.currentClass != null && this.currentService != null ) {
this.currentClass.add(currentService);
}
if ( this.currentClass != null ) {
for(final AbstractDescription d : propsAndRefs) {
this.currentClass.add(d);
}
}
this.propsAndRefs.clear();
this.currentClass = null;
this.currentComponent = null;
this.currentService = null;
this.isComponent = false;
} else if (localName.equals(PROPERTY) && this.pendingProperty != null) {
// now split the value
final String text = this.pendingProperty.getValue();
if (text != null) {
final StringTokenizer st = new StringTokenizer(text);
final String[] values = new String[st.countTokens()];
int index = 0;
while (st.hasMoreTokens()) {
values[index] = st.nextToken();
if ( this.pendingProperty.getType() == PropertyType.Char || this.pendingProperty.getType() == PropertyType.Character ) {
final int val = Integer.valueOf(values[index]);
final Character c = Character.valueOf((char)val);
values[index] = c.toString();
}
index++;
}
this.pendingProperty.setMultiValue(values);
}
propsAndRefs.add(this.pendingProperty);
this.pendingProperty = null;
}
}
}
/**
* @see org.xml.sax.helpers.DefaultHandler#characters(char[], int, int)
*/
@Override
public void characters(char[] ch, int start, int length) throws SAXException {
if (this.pendingProperty != null) {
final String text = new String(ch, start, length);
if (this.pendingProperty.getValue() == null) {
this.pendingProperty.setValue(text);
} else {
this.pendingProperty.setValue(this.pendingProperty.getValue() + text);
}
}
}
}
private static final String PARENT_NAME = "OSGI-INF";
/**
* Generate descriptor file(s)
*/
public static List<String> generateDescriptorFiles(final DescriptionContainer module, final Options options, final Log logger)
throws SCRDescriptorException, SCRDescriptorFailureException {
// get the list of all relevant containers
final List<ComponentContainer> components = new ArrayList<ComponentContainer>();
for(final ComponentContainer container : module.getComponents()) {
if (!container.getComponentDescription().isCreateDs()) {
logger.debug("Ignoring descriptor for DS : " + container);
} else if (!container.getComponentDescription().isAbstract()) {
logger.debug("Adding descriptor for DS : " + container);
components.add(container);
}
}
// check descriptor file
final File descriptorDir = options.getComponentDescriptorDirectory();
// terminate if there is nothing else to write
if (components.isEmpty()) {
logger.debug("No Service Component Descriptors found in project.");
// remove files if it exists
if ( descriptorDir.exists() && !options.isIncremental()) {
for(final File f : descriptorDir.listFiles()) {
if ( f.isFile() ) {
logger.debug("Removing obsolete service descriptor " + f);
f.delete();
}
}
}
return null;
}
// finally the descriptors have to be written ....
descriptorDir.mkdirs(); // ensure parent dir
final List<String> fileNames = new ArrayList<String>();
final List<ComponentContainerContainer> containers = ComponentContainerUtil.split(components);
for(final ComponentContainerContainer ccc : containers) {
final SpecVersion globalVersion = module.getOptions().getSpecVersion();
SpecVersion sv = null;
for(final ComponentContainer cc : ccc.components ) {
if ( sv == null || sv.ordinal() < cc.getComponentDescription().getSpecVersion().ordinal() ) {
sv = cc.getComponentDescription().getSpecVersion();
}
}
module.getOptions().setSpecVersion(sv);
final File useFile = new File(descriptorDir, ccc.className + ".xml");
try {
ComponentDescriptorIO.generateXML(module, ccc.components, useFile, logger);
} catch (final IOException e) {
throw new SCRDescriptorException("Unable to generate xml", useFile.toString(), e);
} catch (final TransformerException e) {
throw new SCRDescriptorException("Unable to generate xml", useFile.toString(), e);
} catch (final SAXException e) {
throw new SCRDescriptorException("Unable to generate xml", useFile.toString(), e);
}
fileNames.add(PARENT_NAME + '/' + useFile.getName());
module.getOptions().setSpecVersion(globalVersion);
}
return fileNames;
}
}