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

org.osgi.service.resolver.ResolveContext Maven / Gradle / Ivy

There is a newer version: 7.0.0
Show newest version
/*
 * Copyright (c) OSGi Alliance (2011, 2017). All Rights Reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.osgi.service.resolver;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.CancellationException;

import org.osgi.annotation.versioning.ConsumerType;
import org.osgi.framework.namespace.BundleNamespace;
import org.osgi.framework.namespace.HostNamespace;
import org.osgi.framework.namespace.PackageNamespace;
import org.osgi.resource.Capability;
import org.osgi.resource.Requirement;
import org.osgi.resource.Resource;
import org.osgi.resource.Wire;
import org.osgi.resource.Wiring;

/**
 * A resolve context provides resources, options and constraints to the
 * potential solution of a {@link Resolver#resolve(ResolveContext) resolve}
 * operation.
 * 

* Resolve Contexts: *

    *
  • Specify the mandatory and optional resources to resolve. The mandatory * and optional resources must be consistent and correct. For example, they must * not violate the singleton policy of the implementer.
  • *
  • Provide {@link Capability capabilities} that the Resolver can use to * satisfy {@link Requirement requirements} via the * {@link #findProviders(Requirement)} method
  • *
  • Constrain solutions via the {@link #getWirings()} method. A wiring * consists of a map of existing {@link Resource resources} to {@link Wiring * wiring}.
  • *
  • Filter requirements that are part of a resolve operation via the * {@link #isEffective(Requirement)}.
  • *
*

* A resolver may call the methods on the resolve context any number of times * during a resolve operation using any thread. Implementors should ensure that * this class is properly thread safe. *

* Except for {@link #insertHostedCapability(List, HostedCapability)} and * {@link #onCancel(Runnable)}, the resolve context methods must be * idempotent. This means that resources must have constant capabilities * and requirements and the resolve context must return a consistent set of * capabilities, wires and effective requirements. * * @ThreadSafe * @author $Id: 5a3d32eb947b703bcca36f00d1ba6a86ae4a8e4b $ */ @ConsumerType public abstract class ResolveContext { /** * Return the resources that must be resolved for this resolve context. * *

* The default implementation returns an empty collection. * * @return A collection of the resources that must be resolved for this * resolve context. May be empty if there are no mandatory * resources. The returned collection may be unmodifiable. */ public Collection getMandatoryResources() { return emptyCollection(); } /** * Return the resources that the resolver should attempt to resolve for this * resolve context. Inability to resolve one of the specified resources will * not result in a resolution exception. * *

* The default implementation returns an empty collection. * * @return A collection of the resources that the resolver should attempt to * resolve for this resolve context. May be empty if there are no * optional resources. The returned collection may be unmodifiable. */ public Collection getOptionalResources() { return emptyCollection(); } @SuppressWarnings("unchecked") private static Collection emptyCollection() { return Collections.EMPTY_LIST; } /** * Find Capabilities that match the given Requirement. *

* The returned list contains {@link Capability} objects where the Resource * must be the declared Resource of the Capability. The Resolver can then * add additional {@link HostedCapability} objects with the * {@link #insertHostedCapability(List, HostedCapability)} method when it, * for example, attaches fragments. Those {@link HostedCapability} objects * will then use the host's Resource which likely differs from the declared * Resource of the corresponding Capability. * *

* The returned list is in priority order such that the Capabilities with a * lower index have a preference over those with a higher index. The * resolver must use the * {@link #insertHostedCapability(List, HostedCapability)} method to add * additional Capabilities to maintain priority order. In general, this is * necessary when the Resolver uses Capabilities declared in a Resource but * that must originate from an attached host. * *

* Each returned Capability must match the given Requirement. This means * that the filter in the Requirement must match as well as any namespace * specific directives. For example, the mandatory attributes for the * {@code osgi.wiring.package} namespace. * * @param requirement The requirement that a resolver is attempting to * satisfy. Must not be {@code null}. * @return A list of {@link Capability} objects that match the specified * requirement. */ public abstract List findProviders(Requirement requirement); /** * Add a {@link HostedCapability} to the list of capabilities returned from * {@link #findProviders(Requirement)}. * *

* This method is used by the {@link Resolver} to add Capabilities that are * hosted by another Resource to the list of Capabilities returned from * {@link #findProviders(Requirement)}. This function is necessary to allow * fragments to attach to hosts, thereby changing the origin of a * Capability. This method must insert the specified HostedCapability in a * place that makes the list maintain the preference order. It must return * the index in the list of the inserted {@link HostedCapability}. * * @param capabilities The list returned from * {@link #findProviders(Requirement)}. Must not be {@code null}. * @param hostedCapability The HostedCapability to insert in the specified * list. Must not be {@code null}. * @return The index in the list of the inserted HostedCapability. * */ public abstract int insertHostedCapability(List capabilities, HostedCapability hostedCapability); /** * Test if a given requirement should be wired in the resolve operation. If * this method returns {@code false}, then the resolver should ignore this * requirement during the resolve operation. * *

* The primary use case for this is to test the {@code effective} directive * on the requirement, though implementations are free to use any effective * test. * * @param requirement The Requirement to test. Must not be {@code null}. * @return {@code true} if the requirement should be considered as part of * the resolve operation. */ public abstract boolean isEffective(Requirement requirement); /** * Returns the wirings for existing resolved resources. * *

* For example, if this resolve context is for an OSGi framework, then the * result would contain all the currently resolved bundles with each * bundle's current wiring. * *

* Multiple calls to this method for this resolve context must return the * same result. * * @return The wirings for existing resolved resources. The returned map is * unmodifiable. */ public abstract Map getWirings(); /** * Find resources that are related to the given resource. *

* The resolver attempts to resolve related resources during the current * resolve operation. Failing to resolve one of the related resources will * not result in a resolution exception unless the related resource is also * a {@link #getMandatoryResources() mandatory} resource. *

* The resolve context is asked to return related resources for each * resource that is pulled into a resolve operation. This includes the * {@link #getMandatoryResources() mandatory} and * {@link #getOptionalResources() optional} resources and each related * resource returned by this method. *

* For example, a fragment can be considered a related resource for a host * bundle. When a host is being resolved the resolve context will be asked * if any related resources should be added to the resolve operation. The * resolve context may decide that the potential fragments of the host * should be resolved along with the host. * * @param resource The Resource that a resolver is attempting to find * related resources for. Must not be {@code null}. * @return A collection of the resources that the resolver should attempt to * resolve for this resolve context. May be empty if there are no * related resources. The returned collection may be unmodifiable. * @since 1.1 */ public Collection findRelatedResources(Resource resource) { return Collections.emptyList(); } /** * Registers a callback with the resolve context that is associated with the * currently running resolve operation. The callback can be executed in * order to cancel the currently running resolve operation. *

* When a resolve operation begins, the resolver must call this method once * and only once for the duration of the resolve operation and that call * must happen before calling any other method on this resolve context. If * the specified callback is executed then the resolver must cancel the * currently running resolve operation and throw a * {@link ResolutionException} with a cause of type * {@link CancellationException}. *

* The callback allows a resolve context to cancel a long running resolve * operation that appears to be running endlessly or at risk of running out * of resources. The resolve context may then decide to give up on resolve * operation or attempt to try another resolve operation with a smaller set * of resources which may allow the resolve operation to complete normally. * * @param callback the callback to execute in order to cancel the resolve * operation. Must not be {@code null}. * @throws IllegalStateException if the resolver attempts to register more * than one callback for a resolve operation * @since 1.1 */ public void onCancel(Runnable callback) { // do nothing by default } /** * Returns the subset of {@link Wiring#getRequiredResourceWires(String) * required} wires that provide wires to {@link Capability capabilities} * which substitute capabilities of the wiring. For example, when a * {@link PackageNamespace package} name is both provided and required by * the same resource. If the package requirement is resolved to a capability * provided by a different wiring then the package capability is considered * to be substituted. *

* The resolver asks the resolve context to return substitution wires for * each wiring that {@link Wiring#getResourceCapabilities(String) provides} * a {@link BundleNamespace bundle} namespace capability that is used to * resolve one or more bundle requirements. *

* Note that this method searches all the {@link PackageNamespace package} * capabilities declared as {@link Resource#getCapabilities(String) * provided} by the resource associated with the wiring and fragment * resources wired to the wiring with the {@link HostNamespace host} * namespace. The provided package names are compared against the * {@link Wiring#getRequiredResourceWires(String) required} package wires to * determine which wires are substitution wires. Subclasses of * ResolveContext should provide a more efficient * implementation of this method. * * @param wiring the wiring to get the substitution wires for. Must not be * {@code null}. * @return A list containing a snapshot of the substitution {@link Wire}s * for the {@link Requirement requirements} of the wiring, or an * empty list if the wiring has no substitution wires. The list * contains the wires in the order they are found in the * {@link Wiring#getRequiredResourceWires(String) required} wires of * the wiring. * @since 1.1 */ public List getSubstitutionWires(Wiring wiring) { // Keep track of the declared capability package names Set exportNames = new HashSet(); // Add packages declared as provided by the wiring host for (Capability cap : wiring.getResource().getCapabilities(null)) { if (PackageNamespace.PACKAGE_NAMESPACE.equals(cap.getNamespace())) { exportNames.add((String) cap.getAttributes() .get(PackageNamespace.PACKAGE_NAMESPACE)); } } // Add packages declared as provided by the attached fragments for (Wire wire : wiring.getProvidedResourceWires(null)) { if (HostNamespace.HOST_NAMESPACE .equals(wire.getCapability().getNamespace())) { Resource fragment = wire.getRequirement().getResource(); for (Capability cap : fragment.getCapabilities(null)) { if (PackageNamespace.PACKAGE_NAMESPACE .equals(cap.getNamespace())) { exportNames.add((String) cap.getAttributes() .get(PackageNamespace.PACKAGE_NAMESPACE)); } } } } // collect the package wires that substitute one of the declared // export package names List substitutionWires = new ArrayList(); for (Wire wire : wiring.getRequiredResourceWires(null)) { if (PackageNamespace.PACKAGE_NAMESPACE .equals(wire.getCapability().getNamespace())) { if (exportNames .contains(wire.getCapability().getAttributes().get( PackageNamespace.PACKAGE_NAMESPACE))) { substitutionWires.add(wire); } } } return substitutionWires; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy