From be634dba71bc5c2ebfe85e388cf64cf5c43eae04 Mon Sep 17 00:00:00 2001 From: Hannes Wellmann Date: Wed, 14 Feb 2024 00:14:28 +0100 Subject: [PATCH] Add utility to derive the BREE of a resource from its EE requirements In general the strategy is to consider all osgi.ee requirements of a resource and parse their filter. For the case of a simple filter like '(&(osgi.ee=JavaSE)(version=17)', probably the majority of cases, a fast-path is provided where the filter is just looked up in a map. Otherwise each registered execution-environment is matched against the filter and the list of matching EEs is returned. The new method is intended to support the migration from the Equinox resolver to the OSGi Resources/Wiring API. --- .../pde/internal/core/util/ManifestUtils.java | 117 +++++++++++++++++- 1 file changed, 115 insertions(+), 2 deletions(-) diff --git a/ui/org.eclipse.pde.core/src/org/eclipse/pde/internal/core/util/ManifestUtils.java b/ui/org.eclipse.pde.core/src/org/eclipse/pde/internal/core/util/ManifestUtils.java index f2b7db192e..dc2b5cceb2 100644 --- a/ui/org.eclipse.pde.core/src/org/eclipse/pde/internal/core/util/ManifestUtils.java +++ b/ui/org.eclipse.pde.core/src/org/eclipse/pde/internal/core/util/ManifestUtils.java @@ -1,5 +1,5 @@ /******************************************************************************* - * Copyright (c) 2006, 2018 IBM Corporation and others. + * Copyright (c) 2006, 2024 IBM Corporation and others. * * This program and the accompanying materials * are made available under the terms of the Eclipse Public License 2.0 @@ -10,14 +10,20 @@ * * Contributors: * IBM Corporation - initial API and implementation + * Hannes Wellmann - Add utility method to derive the BREE of a resource from its EE requirements *******************************************************************************/ package org.eclipse.pde.internal.core.util; +import static org.osgi.framework.namespace.ExecutionEnvironmentNamespace.CAPABILITY_VERSION_ATTRIBUTE; +import static org.osgi.framework.namespace.ExecutionEnvironmentNamespace.EXECUTION_ENVIRONMENT_NAMESPACE; + import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; import java.io.Writer; +import java.util.ArrayList; +import java.util.Collection; import java.util.HashMap; import java.util.Hashtable; import java.util.LinkedList; @@ -25,6 +31,7 @@ import java.util.Map; import java.util.Map.Entry; import java.util.jar.JarFile; +import java.util.regex.Pattern; import java.util.zip.ZipEntry; import java.util.zip.ZipFile; @@ -32,12 +39,16 @@ import org.eclipse.core.resources.IProject; import org.eclipse.core.resources.IResource; import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.ILog; import org.eclipse.core.runtime.IStatus; import org.eclipse.core.runtime.Status; import org.eclipse.jdt.core.IJavaProject; import org.eclipse.jdt.core.IPackageFragmentRoot; import org.eclipse.jdt.core.JavaCore; import org.eclipse.jdt.core.JavaModelException; +import org.eclipse.jdt.launching.JavaRuntime; +import org.eclipse.jdt.launching.environments.IExecutionEnvironment; +import org.eclipse.jdt.launching.environments.IExecutionEnvironmentsManager; import org.eclipse.osgi.util.ManifestElement; import org.eclipse.osgi.util.NLS; import org.eclipse.pde.core.build.IBuild; @@ -50,6 +61,13 @@ import org.eclipse.pde.internal.core.project.PDEProject; import org.osgi.framework.BundleException; import org.osgi.framework.Constants; +import org.osgi.framework.Filter; +import org.osgi.framework.FrameworkUtil; +import org.osgi.framework.InvalidSyntaxException; +import org.osgi.framework.namespace.ExecutionEnvironmentNamespace; +import org.osgi.resource.Namespace; +import org.osgi.resource.Requirement; +import org.osgi.resource.Resource; public class ManifestUtils { @@ -77,7 +95,7 @@ public class ManifestUtils { *

* If this method is being called from a dev mode workspace, the returned map * should be passed to {@link TargetWeaver#weaveManifest(Map, File)} so that the - * bundle classpath can be corrected. + * bundle classpath can be corrected. *

*

* This method is called by @@ -285,4 +303,99 @@ private static String splitOnComma(String value) { sb.append(values[values.length - 1]); return sb.toString(); } + + /** + * Returns the list all execution-environments required by the given OSGi + * resource. Only registered EEs are considered. + * + * @param resource + * the osgi resource + * @return a list containing the id's of all required EEs + * @see ExecutionEnvironmentNamespace#EXECUTION_ENVIRONMENT_NAMESPACE + * @see IExecutionEnvironmentsManager#getExecutionEnvironments() + */ + public static List getRequiredExecutionEnvironments(Resource resource) { + List requirements = resource.getRequirements(EXECUTION_ENVIRONMENT_NAMESPACE); + List requiredEEs = new ArrayList<>(); + for (Requirement requirement : requirements) { + String eeFilter = requirement.getDirectives().get(Namespace.REQUIREMENT_FILTER_DIRECTIVE); + if (eeFilter != null) { + parseRequiredEEsFromFilter(eeFilter, requiredEEs); + } + } + return List.copyOf(requiredEEs); + } + + // provide fast-path for simple filters like: (&(osgi.ee=JavaSE)(version=17) + private static final Map SIMPLE_EE_FILTERS = new HashMap<>(); + private static final Map> AVAILABLE_EE_ATTRIBUTES = new HashMap<>(); + + static { + // Manually add ancient EE id that are difficult to parse + AVAILABLE_EE_ATTRIBUTES.put("CDC-1.0/Foundation-1.0", //$NON-NLS-1$ + Map.of(EXECUTION_ENVIRONMENT_NAMESPACE, "CDC/Foundation", CAPABILITY_VERSION_ATTRIBUTE, "1.0")); //$NON-NLS-1$ //$NON-NLS-2$ + AVAILABLE_EE_ATTRIBUTES.put("CDC-1.1/Foundation-1.1", //$NON-NLS-1$ + Map.of(EXECUTION_ENVIRONMENT_NAMESPACE, "CDC/Foundation", CAPABILITY_VERSION_ATTRIBUTE, "1.1")); //$NON-NLS-1$ //$NON-NLS-2$ + + IExecutionEnvironment[] availableEEs = JavaRuntime.getExecutionEnvironmentsManager().getExecutionEnvironments(); + for (IExecutionEnvironment ee : availableEEs) { + String id = ee.getId(); + String eeName; + String eeVersion; + // Extract the osgi.ee name and version attribute from the EE id + Map predefinedAttributes = AVAILABLE_EE_ATTRIBUTES.get(id); + if (predefinedAttributes == null) { + if (id.indexOf('/') >= id.indexOf('-')) { + ILog.get().error("Cannot reliably parse filter attributes from BREE with id: " + id); //$NON-NLS-1$ + continue; + } + int versionSeparator = id.lastIndexOf('-'); + if (versionSeparator < 0) { + throw new IllegalArgumentException("Missing version-separator in EE Id"); //$NON-NLS-1$ + } + eeName = id.substring(0, versionSeparator); + // OSGi spec chapter 3.4.1 Bundle-RequiredExecutionEnvironment + if ("J2SE".equals(eeName)) { //$NON-NLS-1$ + eeName = "JavaSE"; //$NON-NLS-1$ + } + eeVersion = id.substring(versionSeparator + 1); + + AVAILABLE_EE_ATTRIBUTES.put(id, + Map.of(EXECUTION_ENVIRONMENT_NAMESPACE, eeName, CAPABILITY_VERSION_ATTRIBUTE, eeVersion)); + } else { + eeName = predefinedAttributes.get(EXECUTION_ENVIRONMENT_NAMESPACE); + eeVersion = predefinedAttributes.get(CAPABILITY_VERSION_ATTRIBUTE); + } + String eeNamespace = EXECUTION_ENVIRONMENT_NAMESPACE; + String versionAttribute = CAPABILITY_VERSION_ATTRIBUTE; + String filter1 = "(&(" + eeNamespace + "=" + eeName + ")(" + versionAttribute + "=" + eeVersion + "))"; //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$ //$NON-NLS-4$ //$NON-NLS-5$ + String filter2 = "(&(" + versionAttribute + "=" + eeVersion + ")(" + eeNamespace + "=" + eeName + "))"; //$NON-NLS-1$//$NON-NLS-2$//$NON-NLS-3$ //$NON-NLS-4$ //$NON-NLS-5$ + SIMPLE_EE_FILTERS.put(filter1, id); // add both variants + SIMPLE_EE_FILTERS.put(filter2, id); + } + } + + private static final Pattern WHITESPACE = Pattern.compile("\\s*"); //$NON-NLS-1$ + + private static void parseRequiredEEsFromFilter(String eeFilter, Collection collector) { + if (eeFilter.chars().anyMatch(Character::isWhitespace)) { + eeFilter = WHITESPACE.matcher(eeFilter).replaceAll(""); //$NON-NLS-1$ + } + String ee = SIMPLE_EE_FILTERS.get(eeFilter); + if (ee != null) { + collector.add(ee); + } else { + try { // complex filter. Collect all matching EEs + Filter filter = FrameworkUtil.createFilter(eeFilter); + AVAILABLE_EE_ATTRIBUTES.forEach((id, eeAttributes) -> { + if (filter.matches(eeAttributes)) { + collector.add(id); + } + }); + } catch (InvalidSyntaxException e) { // should not happen + throw new IllegalArgumentException("Invalid execution environment filter", e); //$NON-NLS-1$ + } + } + } + }