/*
* 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.tuscany.sca.contribution.osgi.impl;
import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Enumeration;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.jar.JarEntry;
import java.util.jar.JarInputStream;
import org.apache.tuscany.sca.contribution.Artifact;
import org.apache.tuscany.sca.contribution.Contribution;
import org.apache.tuscany.sca.contribution.ContributionFactory;
import org.apache.tuscany.sca.contribution.PackageType;
import org.apache.tuscany.sca.contribution.processor.ContributionException;
import org.apache.tuscany.sca.contribution.processor.ContributionReadException;
import org.apache.tuscany.sca.contribution.scanner.ContributionScanner;
import org.osgi.framework.Bundle;
import org.osgi.framework.Constants;
/**
* Bundle Contribution package processor.
*
* @version $Rev$ $Date$
*/
public class OSGiBundleContributionScanner implements ContributionScanner {
private ContributionFactory contributionFactory;
public OSGiBundleContributionScanner(ContributionFactory contributionFactory) {
this.contributionFactory = contributionFactory;
}
public String getContributionType() {
return PackageType.BUNDLE;
}
public List<Artifact> scan(Contribution contribution) throws ContributionReadException {
Bundle bundle = OSGiBundleActivator.findBundle(contribution.getLocation());
if (bundle == null) {
throw new IllegalArgumentException("Could not find OSGi bundle " + contribution.getLocation());
}
List<Artifact> artifacts = new ArrayList<Artifact>();
Set<String> bundleClassPath = new HashSet<String>();
String cp = (String)bundle.getHeaders().get(Constants.BUNDLE_CLASSPATH);
if (cp != null) {
String[] paths = cp.split(",");
for (String path : paths) {
bundleClassPath.add(path.trim());
}
}
try {
// Test if the bundle is an Eclipse project
boolean devMode = (bundle.getEntry("/.project") != null);
// FIXME: The entries can come from fragments. Do we need to have a way to differentiate the entries?
Enumeration<?> entries = bundle.findEntries("/", "*", true);
while (entries.hasMoreElements()) {
URL entry = (URL)entries.nextElement();
String entryName = entry.getPath();
if (devMode && entryName.contains("/.svn/")
|| entryName.startsWith("/.")
|| entryName.startsWith("/target/")
|| entryName.startsWith("/src/")) {
// Ignore .svn files
// Ignore .classpath, .project, src, and target
continue;
}
if (entryName.startsWith("/")) {
entryName = entryName.substring(1);
}
//Add artifact to list
Artifact artifact = contributionFactory.createArtifact();
artifact.setURI(entryName);
artifact.setLocation(entry.toString());
artifacts.add(artifact);
//if Artifact is a JAR, add jar artifacts as well
/*
if (entryName.endsWith(".jar") && bundleClassPath.contains(entryName)) {
List<String> jarArtifactURIs = getJarArtifacts(entry, entry.openStream());
for( String uri : jarArtifactURIs) {
Artifact jarArtifact = contributionFactory.createArtifact();
jarArtifact.setURI(uri);
URL url = bundle.getResource(uri);
String location = null;
if(url!=null) {
location = url.toString();
}
jarArtifact.setLocation(location);
artifacts.add(jarArtifact);
}
}
*/
}
} catch (Exception e) {
throw new RuntimeException(e);
}
contribution.getExtensions().add(bundle);
contribution.getTypes().add(getContributionType());
contribution.setClassLoader(new BundleClassLoader(bundle));
return artifacts;
}
/**
* Retrieve a list of Artifact URIs for a given JAR
* @param packageSourceURL
* @param inputStream
* @return
* @throws ContributionException
* @throws IOException
*/
private List<String> getJarArtifacts(URL packageSourceURL, InputStream inputStream) throws ContributionException, IOException {
if (packageSourceURL == null) {
throw new IllegalArgumentException("Invalid null package source URL.");
}
if (inputStream == null) {
throw new IllegalArgumentException("Invalid null source inputstream.");
}
// Assume the root is a jar file
JarInputStream jar = new JarInputStream(inputStream);
try {
Set<String> names = new HashSet<String>();
while (true) {
JarEntry entry = jar.getNextJarEntry();
if (entry == null) {
// EOF
break;
}
// FIXME: Maybe we should externalize the filter as a property
String name = entry.getName();
if (!name.startsWith(".") && !entry.isDirectory()) {
// Trim trailing /
if (name.endsWith("/")) {
name = name.substring(0, name.length() - 1);
}
// Add the entry name
if (!names.contains(name) && name.length() > 0) {
names.add(name);
}
}
}
// Return list of URIs
List<String> artifacts = new ArrayList<String>();
for (String name : names) {
artifacts.add(name);
}
return artifacts;
} finally {
jar.close();
}
}
private static class BundleClassLoader extends ClassLoader {
private Bundle bundle;
public BundleClassLoader(Bundle bundle) {
super(null);
this.bundle = bundle;
}
@Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
return bundle.loadClass(name);
}
@Override
protected URL findResource(String name) {
return bundle.getResource(name);
}
@SuppressWarnings("unchecked")
@Override
protected Enumeration<URL> findResources(String name) throws IOException {
Enumeration<URL> urls = bundle.getResources(name);
if (urls == null) {
List<URL> list = Collections.emptyList();
return Collections.enumeration(list);
} else {
return urls;
}
}
}
}