blob: 6fe7140eb33e75637549e81c152e6cea8c28aa41 [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.dm.annotation.plugin.bnd;
import java.util.Map;
import java.util.Set;
import aQute.bnd.osgi.Analyzer;
import aQute.bnd.osgi.Resource;
import aQute.bnd.service.AnalyzerPlugin;
import aQute.bnd.service.Plugin;
import aQute.service.reporter.Reporter;
/**
* This class is a BND plugin. It scans the target bundle and look for DependencyManager annotations.
* It can be directly used when using ant and can be referenced inside the ".bnd" descriptor, using
* the "-plugin" parameter.
*
* @author <a href="mailto:dev@felix.apache.org">Felix Project Team</a>
*/
public class AnnotationPlugin implements AnalyzerPlugin, Plugin {
private static final String IMPORT_SERVICE = "Import-Service";
private static final String EXPORT_SERVICE = "Export-Service";
private static final String REQUIRE_CAPABILITY = "Require-Capability";
private static final String LOGLEVEL = "log";
private static final String BUILD_IMPEXT = "build-import-export-service";
private static final String ADD_REQUIRE_CAPABILITY = "add-require-capability";
private static final String DM_RUNTIME_CAPABILITY = "osgi.extender; filter:=\"(&(osgi.extender=org.apache.felix.dependencymanager.runtime)(version>=4.0.0))\"";
private BndLogger m_logger;
private Reporter m_reporter;
private boolean m_buildImportExportService;
private boolean m_addRequireCapability;
private Map<String, String> m_properties;
public void setReporter(Reporter reporter) {
m_reporter = reporter;
}
public void setProperties(Map<String, String> map) {
m_properties = map;
}
/**
* This plugin is called after analysis of the JAR but before manifest
* generation. When some DM annotations are found, the plugin will add the corresponding
* DM component descriptors under META-INF/ directory. It will also set the
* "DependencyManager-Component" manifest header (which references the descriptor paths).
*
* @param analyzer the object that is used to retrieve classes containing DM annotations.
* @return true if the classpath has been modified so that the bundle classpath must be reanalyzed
* @throws Exception on any errors.
*/
public boolean analyzeJar(Analyzer analyzer) throws Exception {
try {
init(analyzer);
// We'll do the actual parsing using a DescriptorGenerator object.
DescriptorGenerator generator = new DescriptorGenerator(analyzer, m_logger);
if (generator.execute()) {
// We have parsed some annotations: set the OSGi "DependencyManager-Component" header in the target bundle.
analyzer.setProperty("DependencyManager-Component", generator.getDescriptorPaths());
if (m_addRequireCapability) {
// Add our Require-Capability header
buildRequireCapability(analyzer);
}
// Possibly set the Import-Service/Export-Service header
if (m_buildImportExportService) {
// Don't override Import-Service header, if it is found from the bnd directives.
if (analyzer.getProperty(IMPORT_SERVICE) == null) {
buildImportExportService(analyzer, IMPORT_SERVICE, generator.getImportService());
}
// Don't override Export-Service header, if already defined
if (analyzer.getProperty(EXPORT_SERVICE) == null) {
buildImportExportService(analyzer, EXPORT_SERVICE, generator.getExportService());
}
}
// And insert the generated descriptors into the target bundle.
Map<String, Resource> resources = generator.getDescriptors();
for (Map.Entry<String, Resource> entry : resources.entrySet()) {
analyzer.getJar().putResource(entry.getKey(), entry.getValue());
}
// Insert the metatype resource, if any.
Resource metaType = generator.getMetaTypeResource();
if (metaType != null) {
analyzer.getJar().putResource("OSGI-INF/metatype/metatype.xml", metaType);
}
}
}
catch (Throwable t) {
m_logger.error("error: " + t.toString(), t);
}
finally {
m_logger.close();
}
return false; // do not reanalyze bundle classpath because our plugin has not changed it.
}
private void init(Analyzer analyzer) {
m_logger = new BndLogger(m_reporter, analyzer.getBsn(), parseOption(m_properties, LOGLEVEL, null));
m_buildImportExportService = parseOption(m_properties, BUILD_IMPEXT, false);
m_addRequireCapability = parseOption(m_properties, ADD_REQUIRE_CAPABILITY, false);
analyzer.setExceptions(true);
m_logger.info("Initialized Bnd DependencyManager plugin: buildImportExport=%b", m_buildImportExportService);
}
private String parseOption(Map<String, String> opts, String name, String def) {
String value = opts.get(name);
return value == null ? def : value;
}
private boolean parseOption(Map<String, String> opts, String name, boolean def) {
String value = opts.get(name);
return value == null ? def : Boolean.valueOf(value);
}
private void buildImportExportService(Analyzer analyzer, String header, Set<String> services) {
m_logger.info("building %s header with the following services: %s", header, services);
if (services.size() > 0) {
StringBuilder sb = new StringBuilder();
for (String service : services) {
sb.append(service);
sb.append(",");
}
sb.setLength(sb.length() - 1); // skip last comma
analyzer.setProperty(header, sb.toString());
}
}
private void buildRequireCapability(Analyzer analyzer) {
String requireCapability = analyzer.getProperty(REQUIRE_CAPABILITY);
if (requireCapability == null) {
analyzer.setProperty(REQUIRE_CAPABILITY, DM_RUNTIME_CAPABILITY);
} else {
StringBuilder sb = new StringBuilder(requireCapability).append(",").append(DM_RUNTIME_CAPABILITY);
analyzer.setProperty(REQUIRE_CAPABILITY, sb.toString());
}
}
}