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

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

The newest version!
package aQute.bnd.build;

import java.io.BufferedReader;
import java.io.File;
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.build.DownloadBlocker.Stage;
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 = IO.absolutePath(file);

		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) {
			File f = blocker.getFile();
			if (blocker.getStage() == Stage.FAILURE) {
				String r = blocker.getReason();
				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 IO.absolutePath(getFile());
	}

	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.
			try (BufferedReader rd = IO.reader(getFile(), Constants.DEFAULT_CHARSET)) {
				String line;
				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);
					}
				}
			}
		} 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()) {
			try (JarInputStream jin = new JarInputStream(IO.stream(getFile()))) {
				manifest = jin.getManifest();
			}
			manifestTime = getFile().lastModified();
		}
		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, project);
		List files = new ArrayList<>(header.size());
		IO.mkdirs(bundleClasspathDirectory);

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




© 2015 - 2024 Weber Informatics LLC | Privacy Policy