blob: fe977cf87fb27dab2b18775fc1ef1b0a7717e69b [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2005, 2006 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
*******************************************************************************/
/*
*/
package org.eclipse.jem.internal.proxy.core;
import java.io.*;
import java.net.URL;
import java.util.ArrayList;
import java.util.Properties;
import org.eclipse.core.runtime.*;
import org.osgi.framework.Bundle;
/**
* Utility to help finding files. It is used to find them within plugins of the current platform or to find them in the PDE target platform.
* Also if in the current platform and the current platform is in development mode it will find the bin directories for the jars instead since
* there are no jars in that case.
* It can also extract non-local files (such as files within a jar) and make a cache of them so that are a local file and not extracted everytime
* they are needed.
*
* TODO If we ever get back to continuing development we should add to this PDE Target support. And test the code here
* since it hasn't been tested yet. Also note that in configuration contributor,
* when we add the library path we expect it to be a folder. This would be a problem with jarred plugins because the directory containing the
* libraries may be the root of the plugin and localizing a directory would extract and cache the entire plugin. Need to change to figure out
* how to only extract the dll/so needed instead. Also, none of these handle the case where the bundle is actually non-local (e.g. "http:/").
* @since 1.2.0
*/
public class ProxyFindSupport {
/**
* Find the filename relative to the platform bundle. It will search in the bundle and in all fragments for the file name.
* It will return a URL for each found file.
* <p>
* This is useful for NLS purposes. In NLS situations the main bundle would contain the main non-library jar with the default properties files.
* Then a fragment for each set of languages will be installed. Within each fragment will be a jar with the same filepath, and it will
* contain the properties files for that language. This would be done only for plugin non-library jars.
* The NLS for library jars of a plugin (i.e. those jars that are the <runtime> jars for the plugin) are usually library jars themselves in
* the fragment, and they usually have a different name than the main library jar. So this should not be used for library jars.
* <p>
* If in development mode, then if the jar is not found it will look in the plugin project's build.properties to find the output directory
* of the jar. If not in the build.properties (meaning it isn't being built by PDE), it will look in the proxy.jars file (if there is one).
* This file is used for jars that are built in a separate project from the plugin but will be found in the plugin at execution.
*
* @param bundle bundle and its fragments to look in.
* @param filenameWithinBundle path to filename within the bundle and fragments. <b>This must be a file, not a directory.</b>
* For finding the bundle itself (only when the bundle is packaged as a jar), use <@link #platformFind(Bundle)>. This does not use
* the platform shortcuts like "$os$". It finds the files exactly as specified.
* @param asLocal <code>true</code> if the urls need to point to local filesystem files. This may cause caching of files locally. This
* could happen if the file is within a plugin that is jarred. <code>false</code> if the url can be a non-local filesystem url.
* @return array of URL's for found files.
*
* @since 1.2.0
*/
public static URL[] platformFindAll(Bundle bundle, IPath filenameWithinBundle, boolean asLocal) {
Bundle[] fragments = Platform.getFragments(bundle);
ArrayList urls = new ArrayList((fragments == null ? 0 : fragments.length) + 1);
String filename = filenameWithinBundle.toString();
URL url = internalFindFromBundleOnly(bundle, filename, asLocal);
if (url != null)
urls.add(url);
if (fragments != null) {
for (int i = 0; i < fragments.length; i++) {
Bundle fragment = fragments[i];
url = internalFindFromBundleOnly(fragment, filename, asLocal);
if (url != null)
urls.add(url);
}
}
return (URL[]) urls.toArray(new URL[urls.size()]);
}
/**
* @param bundle
* @param filenameWithinBundle
* @param asLocal
* @return
*
* @since 1.2.0
*/
private static URL internalFindFromBundleOnly(Bundle bundle, String filenameWithinBundle, boolean asLocal) {
try {
URL pvm = bundle.getEntry(filenameWithinBundle);
if (pvm != null)
return asLocal ? FileLocator.toFileURL(pvm) : pvm;
} catch (IOException e) {
}
if (ProxyPlugin.getPlugin().isDevMode()) {
return findDev(bundle, filenameWithinBundle, asLocal);
} else
return null;
}
/**
* Find the filename relative to the platform bundle. It will search in the bundle and in all fragments for the file name. It will return
* URL for the file if found.
* <p>
* If in development mode, then if the jar is not found it will look in the plugin project's build.properties to find the output directory
* of the jar. If not in the build.properties (meaning it isn't being built by PDE), it will look in the proxy.jars file (if there is one).
* This file is used for jars that are built in a separate project from the plugin but will be found in the plugin at execution.
*
* @param bundle bundle and its fragments to look in.
* @param filenameWithinBundle path to filename within the bundle and fragments. <b>This must be a file, not a directory.</b>
* For finding the bundle itself (only when the bundle is packaged as a jar), use <@link #platformFind(Bundle)>.
* @param asLocal <code>true</code> if the urls need to point to local filesystem files. This may cause caching of files locally. This
* could happen if the file is within a plugin that is jarred. <code>false</code> if the url can be a non-local filesystem url.
* @return the found url or <code>null</code> if not found.
*
* @since 1.2.0
*/
public static URL platformFind(Bundle bundle, IPath filenameWithinBundle, boolean asLocal) {
try {
URL pvm = FileLocator.find(bundle, filenameWithinBundle, null);
if (pvm != null)
return asLocal ? FileLocator.toFileURL(pvm) : pvm;
} catch (IOException e) {
}
if (ProxyPlugin.getPlugin().isDevMode()) {
return findDev(bundle, filenameWithinBundle.toString(), asLocal);
} else
return null;
}
private static final String BUILDPROPERTIES = "build.properties"; //$NON-NLS-1$
private static final String PROXYJARS = "proxy.jars"; //$NON-NLS-1$
private static URL findDev(Bundle bundle, String filenameWithinBundle, boolean asLocal) {
// Got this far and in dev mode means it wasn't found, so we'll try for development style.
// It is assumed that in dev mode, we are running with the IDE as local and any
// build outputs will be local so local file protocol will be returned
// from Platform.resolve(). We won't be running in dev mode with our entireplugin being in a jar,
// or on a separate system.
try {
URL bp = bundle.getEntry(BUILDPROPERTIES);
if (bp != null) {
InputStream ios = null;
try {
ios = bp.openStream();
Properties props = new Properties();
props.load(ios);
String pathString = props.getProperty("output."+filenameWithinBundle.toString()); //$NON-NLS-1$
if (pathString != null) {
URL pvm = bundle.getEntry(pathString);
if (pvm != null)
return asLocal ? FileLocator.toFileURL(pvm) : pvm;
}
} finally {
if (ios != null)
ios.close();
}
}
} catch (IOException e) {
}
try {
URL pvm = bundle.getEntry(PROXYJARS);
if (pvm != null) {
InputStream ios = null;
try {
ios = pvm.openStream();
Properties props = new Properties();
props.load(ios);
String pathString = props.getProperty(filenameWithinBundle.toString());
if (pathString != null) {
URL url = FileLocator.resolve(bundle.getEntry("/")); // It is assumed that if in debug mode, then this plugin is an imported plugin within the developement workspace. //$NON-NLS-1$
if (url.getProtocol().equals("file")) { //$NON-NLS-1$
File file = new File(url.getFile()).getParentFile(); // This gets us to workspace root of development workspace.
file = new File(file, pathString);
return file.toURL();
}
}
} finally {
if (ios != null)
ios.close();
}
}
} catch (IOException e) {
}
return null;
}
/**
* Find the bundle and all active fragments libraries. This should normally be used only when the bundle is packaged as a jar and not as a directory. It will
* return the urls to the jars of the bundle and fragments. If in development mode it will find the bundle's "." output directory.
* @param bundle
* @return array of URL's for the jarred bundle and fragments. If the bundle/fragments are not packaged as jars, then the array will be empty.
*
* @since 1.2.0
*/
public static URL[] platformFind(Bundle bundle) {
return null;
}
}