aQute.bnd.build.Container Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of biz.aQute.bndlib Show documentation
Show all versions of biz.aQute.bndlib Show documentation
bndlib: A Swiss Army Knife for OSGi
package aQute.bnd.build;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.jar.JarInputStream;
import java.util.jar.Manifest;
import aQute.bnd.header.Attrs;
import aQute.bnd.header.Parameters;
import aQute.bnd.osgi.Constants;
import aQute.bnd.osgi.Jar;
import aQute.bnd.osgi.Processor;
import aQute.bnd.osgi.Resource;
import aQute.bnd.service.Strategy;
import aQute.lib.io.IO;
public class Container {
public enum TYPE {
REPO, PROJECT, EXTERNAL, LIBRARY, ERROR
}
private volatile File file;
private final String path;
private final TYPE type;
private final String bsn;
private final String version;
private volatile String error;
private final Project project;
private volatile DownloadBlocker db;
private volatile Map attributes;
private long manifestTime;
private Manifest manifest;
private volatile File[] bundleClasspathExpansion;
public String warning = "";
Container(Project project, String bsn, String version, TYPE type, File source, String error,
Map attributes, DownloadBlocker db) {
this.bsn = bsn;
this.version = version;
this.type = type;
this.file = source != null ? source : new File("/" + bsn + ":" + version + ":" + type);
this.path = file.getAbsolutePath();
this.project = project;
this.error = error;
if (attributes == null || attributes.isEmpty()) {
attributes = Collections.emptyMap();
} else if (attributes.containsKey("expand-bcp")) {
this.bundleClasspathExpansion = new File[0];
}
this.attributes = attributes;
this.db = db;
}
public Container(Project project, File file, Map attributes) {
this(project, file.getName(), "project", TYPE.PROJECT, file, null, attributes, null);
}
public Container(Project project, File file) {
this(project, file, null);
}
public Container(File file, DownloadBlocker db) {
this(null, file.getName(), "project", TYPE.EXTERNAL, file, null, null, db);
}
public Container(File file, DownloadBlocker db, Attrs attributes) {
this(null, file.getName(), "project", TYPE.EXTERNAL, file, null, attributes, db);
}
public File getFile() {
DownloadBlocker blocker = db;
if (blocker != null) {
String r = blocker.getReason();
File f = blocker.getFile();
if (r != null) {
if (error == null) {
error = r;
}
return new File(r + ": " + f);
}
this.file = f;
this.db = null;
}
return file;
}
/**
* Iterate over the containers and get the files they represent. If a file
* is already in the list, it is not added again.
*
* @param files
* @throws Exception
*/
public boolean contributeFiles(List files, Processor reporter) throws Exception {
switch (type) {
case EXTERNAL :
case REPO :
for (File f : getBundleClasspathFiles()) {
if (!files.contains(f)) {
files.add(f);
}
}
return true;
case PROJECT :
File[] fs = project.build();
reporter.getInfo(project);
if (fs == null)
return false;
for (File f : fs) {
if (!files.contains(f)) {
files.add(f);
}
}
return true;
case LIBRARY :
List containers = getMembers();
for (Container container : containers) {
if (!container.contributeFiles(files, reporter))
return false;
}
return true;
case ERROR :
reporter.error("%s", getError());
return false;
}
return false;
}
public String getBundleSymbolicName() {
return bsn;
}
public String getVersion() {
return version;
}
public TYPE getType() {
return type;
}
public String getError() {
return error;
}
@Override
public boolean equals(Object other) {
if (other instanceof Container)
return path.equals(((Container) other).path);
return false;
}
@Override
public int hashCode() {
return path.hashCode();
}
public Project getProject() {
return project;
}
/**
* Must show the file name or the error formatted as a file name
*
*/
@Override
public String toString() {
if (getError() != null)
return "/error/" + getError();
return getFile().getAbsolutePath().replace(File.separatorChar, '/');
}
public Map getAttributes() {
return attributes;
}
public synchronized void putAttribute(String name, String value) {
if (attributes == Collections. emptyMap())
attributes = new HashMap(1);
attributes.put(name, value);
}
/**
* Return the this if this is anything else but a library. If it is a
* library, return the members. This could work recursively, e.g., libraries
* can point to libraries.
*
* @throws Exception
*/
public List getMembers() throws Exception {
List result = project.newList();
// Are ww a library? If no, we are the result
if (getType() == TYPE.LIBRARY) {
// We are a library, parse the file. This is
// basically a specification clause per line.
// I.e. you can do bsn; version, bsn2; version. But also
// spread it out over lines.
InputStream in = null;
BufferedReader rd = null;
String line;
try {
in = new FileInputStream(getFile());
rd = new BufferedReader(new InputStreamReader(in, Constants.DEFAULT_CHARSET));
while ((line = rd.readLine()) != null) {
line = line.trim();
if (!line.startsWith("#") && line.length() > 0) {
List list = project.getBundles(Strategy.HIGHEST, line, null);
result.addAll(list);
}
}
} finally {
if (rd != null) {
rd.close();
}
if (in != null) {
in.close();
}
}
} else
result.add(this);
return result;
}
/**
* Flatten a container in the output list. (e.g. expand any libraries).
*
* @param container the container to flatten
* @param list the result list
*/
public static void flatten(Container container, List list) throws Exception {
if (container.getType() == TYPE.LIBRARY) {
flatten(container.getMembers(), list);
} else
list.add(container);
}
/**
* Take a container list and flatten it (e.g. expand any libraries).
*
* @param containers The containers to flatten, can be null
* @return a list of containers guaranteed to contain no libraries
*/
public static List flatten(Collection containers) throws Exception {
List list = new ArrayList();
flatten(containers, list);
return list;
}
/**
* Take a container list and flatten it (e.g. expand any libraries).
*
* @param containers The containers to flatten, can be null
* @param list of containers guaranteed to contain no libraries
*/
public static void flatten(Collection containers, List list) throws Exception {
if (containers == null)
return;
for (Container container : containers) {
flatten(container, list);
}
}
/**
* Answer the manifest for this container (if possible). Manifest is cached
* until the file is renewed.
*/
public Manifest getManifest() throws Exception {
if (getError() != null || getFile() == null)
return null;
if (manifestTime < getFile().lastModified()) {
InputStream in = new FileInputStream(getFile());
try {
JarInputStream jin = new JarInputStream(in);
manifest = jin.getManifest();
jin.close();
manifestTime = getFile().lastModified();
} finally {
in.close();
}
}
return manifest;
}
/**
* @throws Exception
*/
private File[] getBundleClasspathFiles() throws Exception {
File[] bce = bundleClasspathExpansion;
if (bce == null) {
return bundleClasspathExpansion = new File[] {
getFile()
};
}
if (bce.length != 0) {
return bce;
}
File file = getFile();
Manifest m = getManifest();
String bundleClassPath;
if (m == null || (bundleClassPath = m.getMainAttributes().getValue(Constants.BUNDLE_CLASSPATH)) == null) {
return bundleClasspathExpansion = new File[] {
file
};
}
File bundleClasspathDirectory = IO.getFile(file.getParentFile(), "." + file.getName() + "-bcp");
Parameters header = new Parameters(bundleClassPath);
List files = new ArrayList<>(header.size());
bundleClasspathDirectory.mkdir();
int n = 0;
Jar jar = null;
try {
for (Map.Entry entry : header.entrySet()) {
if (".".equals(entry.getKey())) {
files.add(file);
} else {
File member = new File(bundleClasspathDirectory, n + "-" + toName(entry.getKey()));
if (!isCurrent(file, member)) {
if (jar == null) {
jar = new Jar(file);
}
Resource resource = jar.getResource(entry.getKey());
if (resource == null) {
warning += "Invalid bcp entry: " + entry.getKey() + "\n";
} else {
IO.copy(resource.openInputStream(), member);
member.setLastModified(file.lastModified());
}
}
files.add(member);
}
n++;
}
} finally {
if (jar != null)
jar.close();
}
return bundleClasspathExpansion = files.toArray(bce);
}
private boolean isCurrent(File file, File member) {
return member.isFile() && member.lastModified() == file.lastModified();
}
private String toName(String key) {
int n = key.lastIndexOf('/');
return key.substring(n + 1);
}
public String getWarning() {
return warning;
}
}