blob: be4da629d4f3b5a56254c89ec4bcfd6da1734562 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2008 IBM Corporation and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
* Code 9 Corporation - ongoing enhancements
* Bartosz Michalik <bartosz.michalik@gmail.com> - bug 109440
* Benjamin Cabe <benjamin.cabe@anyware-tech.com> - bug 248852, bug 247553
*******************************************************************************/
package org.eclipse.pde.internal.ui.wizards.plugin;
import java.io.*;
import java.lang.reflect.InvocationTargetException;
import java.util.*;
import java.util.jar.JarFile;
import java.util.jar.Manifest;
import java.util.zip.ZipFile;
import org.eclipse.core.resources.*;
import org.eclipse.core.runtime.*;
import org.eclipse.jdt.core.*;
import org.eclipse.osgi.service.resolver.BundleDescription;
import org.eclipse.osgi.util.ManifestElement;
import org.eclipse.osgi.util.NLS;
import org.eclipse.pde.core.build.IBuildEntry;
import org.eclipse.pde.core.build.IBuildModelFactory;
import org.eclipse.pde.core.plugin.*;
import org.eclipse.pde.internal.core.ICoreConstants;
import org.eclipse.pde.internal.core.build.WorkspaceBuildModel;
import org.eclipse.pde.internal.core.bundle.BundlePluginBase;
import org.eclipse.pde.internal.core.converter.PluginConverter;
import org.eclipse.pde.internal.core.ibundle.IBundle;
import org.eclipse.pde.internal.core.ibundle.IBundlePluginModelBase;
import org.eclipse.pde.internal.core.natures.PDE;
import org.eclipse.pde.internal.core.plugin.WorkspacePluginModelBase;
import org.eclipse.pde.internal.ui.*;
import org.eclipse.pde.internal.ui.search.dependencies.AddNewBinaryDependenciesOperation;
import org.eclipse.pde.internal.ui.wizards.IProjectProvider;
import org.eclipse.pde.ui.IFieldData;
import org.eclipse.pde.ui.IPluginContentWizard;
import org.eclipse.ui.dialogs.IOverwriteQuery;
import org.eclipse.ui.wizards.datatransfer.ImportOperation;
import org.eclipse.ui.wizards.datatransfer.ZipFileStructureProvider;
import org.osgi.framework.BundleException;
import org.osgi.framework.Constants;
public class NewLibraryPluginCreationOperation extends NewProjectCreationOperation {
private LibraryPluginFieldData fData;
public NewLibraryPluginCreationOperation(LibraryPluginFieldData data, IProjectProvider provider, IPluginContentWizard contentWizard) {
super(data, provider, contentWizard);
fData = data;
}
private void addJar(File jarFile, IProject project, IProgressMonitor monitor) throws CoreException {
String jarName = jarFile.getName();
IFile file = project.getFile(jarName);
monitor.subTask(NLS.bind(PDEUIMessages.NewProjectCreationOperation_copyingJar, jarName));
InputStream in = null;
try {
in = new FileInputStream(jarFile);
file.create(in, true, monitor);
} catch (FileNotFoundException fnfe) {
} finally {
if (in != null) {
try {
in.close();
} catch (IOException ioe) {
}
}
}
}
private void adjustExportRoot(IProject project, IBundle bundle) throws CoreException {
IResource[] resources = project.members(false);
for (int j = 0; j < resources.length; j++) {
if (resources[j] instanceof IFile) {
if (".project".equals(resources[j].getName()) //$NON-NLS-1$
|| ".classpath".equals(resources[j] //$NON-NLS-1$
.getName()) || "plugin.xml".equals(resources[j] //$NON-NLS-1$
.getName()) || "build.properties".equals(resources[j] //$NON-NLS-1$
.getName())) {
continue;
}
// resource at the root, export root
return;
}
}
removeExportRoot(bundle);
}
protected void adjustManifests(IProgressMonitor monitor, IProject project, IPluginBase base) throws CoreException {
super.adjustManifests(monitor, project, base);
int units = fData.doFindDependencies() ? 4 : 2;
units += fData.isUpdateReferences() ? 1 : 0;
monitor.beginTask(new String(), units);
IBundle bundle = (base instanceof BundlePluginBase) ? ((BundlePluginBase) base).getBundle() : null;
if (bundle != null) {
adjustExportRoot(project, bundle);
monitor.worked(1);
addExportedPackages(project, bundle);
monitor.worked(1);
if (fData.doFindDependencies()) {
addDependencies(project, base.getModel(), new SubProgressMonitor(monitor, 2));
}
if (fData.isUpdateReferences()) {
updateReferences(monitor, project);
monitor.worked(1);
}
}
monitor.done();
}
protected void updateReferences(IProgressMonitor monitor, IProject project) throws JavaModelException {
IJavaProject currentProject = JavaCore.create(project);
IPluginModelBase[] pluginstoUpdate = fData.getPluginsToUpdate();
for (int i = 0; i < pluginstoUpdate.length; ++i) {
IProject proj = pluginstoUpdate[i].getUnderlyingResource().getProject();
if (currentProject.getProject().equals(proj))
continue;
IJavaProject javaProject = JavaCore.create(proj);
IClasspathEntry[] cp = javaProject.getRawClasspath();
IClasspathEntry[] updated = getUpdatedClasspath(cp, currentProject);
if (updated != null) {
javaProject.setRawClasspath(updated, monitor);
}
try {
updateRequiredPlugins(javaProject, monitor, pluginstoUpdate[i]);
} catch (CoreException e) {
PDEPlugin.log(e);
}
}
}
private static void updateRequiredPlugins(IJavaProject javaProject, IProgressMonitor monitor, IPluginModelBase model) throws CoreException {
IClasspathEntry[] entries = javaProject.getRawClasspath();
List classpath = new ArrayList();
List requiredProjects = new ArrayList();
for (int i = 0; i < entries.length; i++) {
if (isPluginProjectEntry(entries[i])) {
requiredProjects.add(entries[i]);
} else {
classpath.add(entries[i]);
}
}
if (requiredProjects.size() <= 0)
return;
IFile file = javaProject.getProject().getFile(ICoreConstants.MANIFEST_PATH);
try {
// TODO format manifest
Manifest manifest = new Manifest(file.getContents());
String value = manifest.getMainAttributes().getValue(Constants.REQUIRE_BUNDLE);
StringBuffer sb = value != null ? new StringBuffer(value) : new StringBuffer();
if (sb.length() > 0)
sb.append(","); //$NON-NLS-1$
for (int i = 0; i < requiredProjects.size(); i++) {
IClasspathEntry entry = (IClasspathEntry) requiredProjects.get(i);
if (i > 0)
sb.append(","); //$NON-NLS-1$
sb.append(entry.getPath().segment(0));
if (entry.isExported())
sb.append(";visibility:=reexport"); // TODO is there a //$NON-NLS-1$
// constant?
}
manifest.getMainAttributes().putValue(Constants.REQUIRE_BUNDLE, sb.toString());
ByteArrayOutputStream content = new ByteArrayOutputStream();
manifest.write(content);
file.setContents(new ByteArrayInputStream(content.toByteArray()), true, false, monitor);
// now update .classpath
javaProject.setRawClasspath((IClasspathEntry[]) classpath.toArray(new IClasspathEntry[classpath.size()]), monitor);
// ClasspathComputer.setClasspath(javaProject.getProject(), model);
} catch (IOException e) {
} catch (CoreException e) {
}
}
private static boolean isPluginProjectEntry(IClasspathEntry entry) {
if (IClasspathEntry.CPE_PROJECT != entry.getEntryKind())
return false;
IWorkspaceRoot workspaceRoot = ResourcesPlugin.getWorkspace().getRoot();
IProject other = workspaceRoot.getProject(entry.getPath().segment(0));
if (!PDE.hasPluginNature(other))
return false;
if (other.findMember(ICoreConstants.FRAGMENT_FILENAME_DESCRIPTOR) != null)
return false;
try {
InputStream is = other.getFile(ICoreConstants.MANIFEST_PATH).getContents();
try {
Manifest mf = new Manifest(is);
if (mf.getMainAttributes().getValue(Constants.FRAGMENT_HOST) != null)
return false;
} finally {
is.close();
}
} catch (IOException e) {
// assume "not a fragment"
} catch (CoreException e) {
// assume "not a fragment"
}
return true;
}
/**
* @return updated classpath or null if there were no changes
*/
private IClasspathEntry[] getUpdatedClasspath(IClasspathEntry[] cp, IJavaProject currentProject) {
boolean exposed = false;
int refIndex = -1;
List result = new ArrayList();
Set manifests = new HashSet();
for (int i = 0; i < fData.getLibraryPaths().length; ++i) {
try {
manifests.add(new JarFile(fData.getLibraryPaths()[i]).getManifest());
} catch (IOException e) {
PDEPlugin.log(e);
}
}
IWorkspaceRoot root = ResourcesPlugin.getWorkspace().getRoot();
for (int i = 0; i < cp.length; ++i) {
IClasspathEntry cpe = cp[i];
switch (cpe.getEntryKind()) {
case IClasspathEntry.CPE_LIBRARY :
String path = null;
IPath location = root.getFile(cpe.getPath()).getLocation();
if (location != null) {
path = location.toString();
}
//try maybe path is absolute
if (path == null) {
path = cpe.getPath().toString();
}
try {
JarFile jarFile = new JarFile(path);
if (manifests.contains(jarFile.getManifest())) {
if (refIndex < 0) {
// allocate slot
refIndex = result.size();
result.add(null);
}
exposed |= cpe.isExported();
} else {
result.add(cpe);
}
} catch (IOException e) {
PDEPlugin.log(e);
}
break;
default :
result.add(cpe);
break;
}
}
if (refIndex >= 0) {
result.set(refIndex, JavaCore.newProjectEntry(currentProject.getPath(), exposed));
return (IClasspathEntry[]) result.toArray(new IClasspathEntry[result.size()]);
}
return null;
}
protected void createContents(IProgressMonitor monitor, IProject project) throws CoreException, JavaModelException, InvocationTargetException, InterruptedException {
// copy jars
String[] paths = fData.getLibraryPaths();
for (int i = paths.length - 1; i >= 0; i--) {
File jarFile = new File(paths[i]);
if (fData.isUnzipLibraries()) {
importJar(jarFile, project, monitor);
} else {
addJar(jarFile, project, monitor);
}
monitor.worked(1);
}
// delete manifest.mf imported from libraries
IFile importedManifest = project.getFile(ICoreConstants.BUNDLE_FILENAME_DESCRIPTOR);
if (importedManifest.exists()) {
importedManifest.delete(true, false, monitor);
if (!fData.hasBundleStructure()) {
IFolder meta_inf = project.getFolder("META-INF"); //$NON-NLS-1$
if (meta_inf.members().length == 0) {
meta_inf.delete(true, false, monitor);
}
}
}
}
protected void fillBinIncludes(IProject project, IBuildEntry binEntry) throws CoreException {
if (fData.hasBundleStructure())
binEntry.addToken(ICoreConstants.MANIFEST_FOLDER_NAME);
else
binEntry.addToken(ICoreConstants.PLUGIN_FILENAME_DESCRIPTOR);
if (fData.isUnzipLibraries()) {
IResource[] resources = project.members(false);
for (int j = 0; j < resources.length; j++) {
String resourceName = resources[j].getName();
if (resources[j] instanceof IFolder) {
if (!".settings".equals(resourceName) && !binEntry.contains(resourceName + "/")) //$NON-NLS-1$ //$NON-NLS-2$
binEntry.addToken(resourceName + "/"); //$NON-NLS-1$
} else {
if (".project".equals(resourceName) //$NON-NLS-1$
|| ".classpath".equals(resourceName) //$NON-NLS-1$
|| ICoreConstants.BUILD_FILENAME_DESCRIPTOR.equals(resourceName)) {
continue;
}
if (!binEntry.contains(resourceName))
binEntry.addToken(resourceName);
}
}
} else {
String[] libraryPaths = fData.getLibraryPaths();
for (int j = 0; j < libraryPaths.length; j++) {
File jarFile = new File(libraryPaths[j]);
String name = jarFile.getName();
if (!binEntry.contains(name))
binEntry.addToken(name);
}
}
}
protected IClasspathEntry[] getInternalClassPathEntries(IJavaProject project, IFieldData data) {
String[] libraryPaths;
if (fData.isUnzipLibraries()) {
libraryPaths = new String[] {""}; //$NON-NLS-1$
} else {
libraryPaths = fData.getLibraryPaths();
}
IClasspathEntry[] entries = new IClasspathEntry[libraryPaths.length];
for (int j = 0; j < libraryPaths.length; j++) {
File jarFile = new File(libraryPaths[j]);
String jarName = jarFile.getName();
IPath path = project.getProject().getFullPath().append(jarName);
entries[j] = JavaCore.newLibraryEntry(path, null, null, true);
}
return entries;
}
protected int getNumberOfWorkUnits() {
int numUnits = super.getNumberOfWorkUnits();
numUnits += fData.getLibraryPaths().length;
return numUnits;
}
private void importJar(File jar, IResource destination, IProgressMonitor monitor) throws CoreException, InvocationTargetException, InterruptedException {
ZipFile input = null;
try {
try {
input = new ZipFile(jar);
ZipFileStructureProvider provider = new ZipFileStructureProvider(input);
ImportOperation op = new ImportOperation(destination.getFullPath(), provider.getRoot(), provider, new IOverwriteQuery() {
public String queryOverwrite(String pathString) {
return IOverwriteQuery.ALL;
}
});
op.run(monitor);
} finally {
if (input != null)
input.close();
}
} catch (IOException e) {
throw new CoreException(new Status(IStatus.ERROR, IPDEUIConstants.PLUGIN_ID, IStatus.OK, NLS.bind(PDEUIMessages.NewProjectCreationOperation_errorImportingJar, jar), e));
}
}
private void removeExportRoot(IBundle bundle) {
String value = bundle.getHeader(Constants.BUNDLE_CLASSPATH);
if (value == null)
value = "."; //$NON-NLS-1$
try {
ManifestElement[] elems = ManifestElement.parseHeader(Constants.BUNDLE_CLASSPATH, value);
StringBuffer buff = new StringBuffer(value.length());
for (int i = 0; i < elems.length; i++) {
if (!elems[i].getValue().equals(".")) //$NON-NLS-1$
buff.append(elems[i].getValue());
}
bundle.setHeader(Constants.BUNDLE_CLASSPATH, buff.toString());
} catch (BundleException e) {
}
}
protected void setPluginLibraries(WorkspacePluginModelBase model) throws CoreException {
IPluginBase pluginBase = model.getPluginBase();
if (fData.isUnzipLibraries()) {
IPluginLibrary library = model.getPluginFactory().createLibrary();
library.setName("."); //$NON-NLS-1$
library.setExported(true);
pluginBase.add(library);
} else {
String[] paths = fData.getLibraryPaths();
for (int i = 0; i < paths.length; i++) {
File jarFile = new File(paths[i]);
IPluginLibrary library = model.getPluginFactory().createLibrary();
library.setName(jarFile.getName());
library.setExported(true);
pluginBase.add(library);
}
}
}
protected void createSourceOutputBuildEntries(WorkspaceBuildModel model, IBuildModelFactory factory) throws CoreException {
if (fData.isUnzipLibraries()) {
// SOURCE.<LIBRARY_NAME>
IBuildEntry entry = factory.createEntry(IBuildEntry.JAR_PREFIX + "."); //$NON-NLS-1$
entry.addToken("."); //$NON-NLS-1$
model.getBuild().add(entry);
// OUTPUT.<LIBRARY_NAME>
entry = factory.createEntry(IBuildEntry.OUTPUT_PREFIX + "."); //$NON-NLS-1$
entry.addToken("."); //$NON-NLS-1$
model.getBuild().add(entry);
}
}
private void addExportedPackages(IProject project, IBundle bundle) {
String value = bundle.getHeader(Constants.BUNDLE_CLASSPATH);
if (value == null)
value = "."; //$NON-NLS-1$
try {
ManifestElement[] elems = ManifestElement.parseHeader(Constants.BUNDLE_CLASSPATH, value);
HashMap map = new HashMap();
for (int i = 0; i < elems.length; i++) {
ArrayList filter = new ArrayList();
filter.add("*"); //$NON-NLS-1$
map.put(elems[i].getValue(), filter);
}
Set packages = PluginConverter.getDefault().getExports(project, map);
String pkgValue = getCommaValuesFromPackagesSet(packages, fData.getVersion());
bundle.setHeader(Constants.EXPORT_PACKAGE, pkgValue);
} catch (BundleException e) {
}
}
private void addDependencies(IProject project, ISharedPluginModel model, IProgressMonitor monitor) {
if (!(model instanceof IBundlePluginModelBase)) {
monitor.done();
return;
}
final boolean unzip = fData.isUnzipLibraries();
try {
new AddNewBinaryDependenciesOperation(project, (IBundlePluginModelBase) model) {
// Need to override this function to include every bundle in the
// target platform as a possible dependency
protected String[] findSecondaryBundles(IBundle bundle, Set ignorePkgs) {
IPluginModelBase[] bases = PluginRegistry.getActiveModels();
String[] ids = new String[bases.length];
for (int i = 0; i < bases.length; i++) {
BundleDescription desc = bases[i].getBundleDescription();
if (desc == null)
ids[i] = bases[i].getPluginBase().getId();
else
ids[i] = desc.getSymbolicName();
}
return ids;
}
// Need to override this function because when jar is unzipped,
// build.properties does not contain entry for '.'.
// Therefore, the super.addProjectPackages will not find the
// project packages(it includes only things in bin.includes)
protected void addProjectPackages(IBundle bundle, Set ignorePkgs) {
if (!unzip)
super.addProjectPackages(bundle, ignorePkgs);
Stack stack = new Stack();
stack.push(fProject);
try {
while (!stack.isEmpty()) {
IContainer folder = (IContainer) stack.pop();
IResource[] children = folder.members();
for (int i = 0; i < children.length; i++) {
if (children[i] instanceof IContainer)
stack.push(children[i]);
else if ("class".equals(((IFile) children[i]).getFileExtension())) { //$NON-NLS-1$
String path = folder.getProjectRelativePath().toString();
ignorePkgs.add(path.replace('/', '.'));
}
}
}
} catch (CoreException e) {
}
}
}.run(monitor);
} catch (InvocationTargetException e) {
} catch (InterruptedException e) {
}
}
}