All Downloads are FREE. Search and download functionalities are using the official Maven repository.

aQute.bnd.build.Container Maven / Gradle / Ivy

There is a newer version: 7.0.0
Show newest version
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 final File			file;
	private final String		path;
	final TYPE					type;
	final String				bsn;
	final String				version;
	final String				error;
	final Project				project;
	final DownloadBlocker		db;
	volatile Map	attributes;
	private long				manifestTime;
	private Manifest			manifest;
	private 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())
			this.attributes = Collections.emptyMap();
		else
			this.attributes = attributes;
		this.db = db;

		if (!this.attributes.containsKey("expand-bcp")) {
			this.bundleClasspathExpansion = new File[] {
					this.file
			};
		}
	}

	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, null);
	}

	public File getFile() {
		if (db != null && db.getReason() != null) {
			return new File(db.getReason() + ": " + file);
		}
		return file;
	}

	/**
	 * Iterate over the containers and get the files they represent
	 * 
	 * @param files
	 * @throws Exception
	 */
	public boolean contributeFiles(List files, Processor reporter) throws Exception {
		switch (type) {
			case EXTERNAL :
			case REPO :
				for (File f : getBundleClasspathFiles()) {
					files.add(f);
				}
				return true;

			case PROJECT :
				File[] fs = project.build();
				reporter.getInfo(project);
				if (fs == null)
					return false;

				for (File f : fs)
					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(error);
				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 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
	 */

	File[] getBundleClasspathFiles() throws Exception {

		if (this.bundleClasspathExpansion != null)
			return bundleClasspathExpansion;

		File file = getFile();

		Manifest m = getManifest();
		String bundleClassPath;
		if (m == null || (bundleClassPath = m.getMainAttributes().getValue(Constants.BUNDLE_CLASSPATH)) == null) {
			this.bundleClasspathExpansion = new File[] {
					file
			};
		} else {

			File bundleClasspathDirectory = IO.getFile(file.getParentFile(), "." + file.getName() + "-bcp");
			Parameters header = new Parameters(bundleClassPath);
			this.bundleClasspathExpansion = new File[header.size()];
			bundleClasspathDirectory.mkdir();

			int n = 0;
			Jar jar = null;
			try {
				for (Map.Entry entry : header.entrySet()) {
					if (".".equals(entry.getKey())) {
						this.bundleClasspathExpansion[n] = 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());
							}

						}
						this.bundleClasspathExpansion[n] = member;
					}
					n++;
				}
			} finally {
				if (jar != null)
					jar.close();
			}
		}

		return this.bundleClasspathExpansion;
	}

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




© 2015 - 2024 Weber Informatics LLC | Privacy Policy