| /* |
| * 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; |
| } |
| } |