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

org.apache.karaf.features.internal.region.Subsystem Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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.apache.karaf.features.internal.region;

import java.io.IOException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.jar.Attributes;
import java.util.jar.Manifest;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;

import org.apache.felix.utils.collections.StringArrayMap;
import org.apache.felix.utils.manifest.Clause;
import org.apache.felix.utils.manifest.Parser;
import org.apache.felix.utils.repository.BaseRepository;
import org.apache.felix.utils.resource.CapabilityImpl;
import org.apache.felix.utils.resource.RequirementImpl;
import org.apache.felix.utils.resource.ResourceBuilder;
import org.apache.felix.utils.resource.ResourceImpl;
import org.apache.felix.utils.resource.SimpleFilter;
import org.apache.felix.utils.version.VersionRange;
import org.apache.felix.utils.version.VersionTable;
import org.apache.karaf.features.BundleInfo;
import org.apache.karaf.features.Conditional;
import org.apache.karaf.features.Dependency;
import org.apache.karaf.features.Feature;
import org.apache.karaf.features.FeaturesNamespaces;
import org.apache.karaf.features.FeaturesService;
import org.apache.karaf.features.Library;
import org.apache.karaf.features.ScopeFilter;
import org.apache.karaf.features.internal.download.DownloadManager;
import org.apache.karaf.features.internal.download.Downloader;
import org.apache.karaf.features.internal.download.StreamProvider;
import org.apache.karaf.features.internal.resolver.FeatureResource;
import org.apache.karaf.features.internal.resolver.ResolverUtil;
import org.apache.karaf.features.internal.resolver.ResourceUtils;
import org.apache.karaf.features.internal.service.Overrides;
import org.osgi.framework.BundleException;
import org.osgi.framework.Version;
import org.osgi.resource.Capability;
import org.osgi.resource.Requirement;
import org.osgi.resource.Resource;

import static java.util.jar.JarFile.MANIFEST_NAME;
import static org.apache.karaf.features.internal.resolver.ResourceUtils.TYPE_FEATURE;
import static org.apache.karaf.features.internal.resolver.ResourceUtils.TYPE_SUBSYSTEM;
import static org.apache.karaf.features.internal.resolver.ResourceUtils.addIdentityRequirement;
import static org.apache.karaf.features.internal.resolver.ResourceUtils.getUri;
import static org.apache.karaf.features.internal.resolver.ResourceUtils.toFeatureRequirement;
import static org.apache.karaf.features.internal.util.MapUtils.addToMapSet;
import static org.eclipse.equinox.region.RegionFilter.VISIBLE_ALL_NAMESPACE;
import static org.osgi.framework.namespace.IdentityNamespace.CAPABILITY_TYPE_ATTRIBUTE;
import static org.osgi.framework.namespace.IdentityNamespace.CAPABILITY_VERSION_ATTRIBUTE;
import static org.osgi.framework.namespace.IdentityNamespace.IDENTITY_NAMESPACE;
import static org.osgi.resource.Namespace.REQUIREMENT_FILTER_DIRECTIVE;

/**
 * A {@link Resource} representing ...
 */
public class Subsystem extends ResourceImpl {

    private static final String ALL_FILTER = "(|(!(all=*))(all=*))";

    private static final String SUBSYSTEM_FILTER = String.format("(%s=%s)", CAPABILITY_TYPE_ATTRIBUTE, TYPE_SUBSYSTEM);

    private static final String FEATURE_FILTER = String.format("(%s=%s)", CAPABILITY_TYPE_ATTRIBUTE, TYPE_FEATURE);

    private static final String SUBSYSTEM_OR_FEATURE_FILTER = String.format("(|%s%s)", SUBSYSTEM_FILTER, FEATURE_FILTER);

    // Everything is visible
    private static final Map> SHARE_ALL_POLICY =
            Collections.singletonMap(
                    VISIBLE_ALL_NAMESPACE,
                    Collections.singleton(ALL_FILTER));

    // Nothing (but systems) is visible
    private static final Map> SHARE_NONE_POLICY =
            Collections.singletonMap(
                    IDENTITY_NAMESPACE,
                    Collections.singleton(SUBSYSTEM_FILTER));

    // name of the subsystem: region or region#feature[-version]
    private final String name;
    // works only with feature scoping. region subsystems by default accept deps
    private final boolean acceptDependencies;
    // parent Subsystem for child subsystems representing child regions or regions' features
    private final Subsystem parent;
    // feature for Subsystem representing a feature
    private final Feature feature;

    private final boolean mandatory;

    private final List children = new ArrayList<>();

    // a set of filters applied when child subsystem needs capabilities from parent subsystem
    private final Map> importPolicy;
    // a set of filters applied when parent subsystem needs capabilities from child subsystem
    private final Map> exportPolicy;

    // contains subsystems representing features of this region, child subsystems for child regions, system resources(?),
    // bundle resources added explicitly as reqs for this Subsystem, feature resources for subsystems representing
    // features, ...
    private final List installable = new ArrayList<>();

    // mapping from "symbolic-name|version" to a DependencyInfo wrapping a Resource
    //  are collected directly in feature's subsystem
    //  are collected in first parent subsystem of feature or in subsystem of scoped feature
    private final Map dependencies = new HashMap<>();
    // non-mandatory dependant features (/) collected from current and child subsystems representing
    // features (unless some subsystem for feature has )
    private final List dependentFeatures = new ArrayList<>();

    // direct bundle URI dependencies - not added by FeaturesService, but used in startup stage of assembly builder
    // these bundles will be downloaded
    private final List bundles = new ArrayList<>();

    /**
     * Constructs root subsystem {@link Resource} for {@link FeaturesService#ROOT_REGION} that imports/exports only
     * caps/reqs with (type=karaf.subsystem).
     * Root subsystem by default accepts dependencies - will gather dependant features of child feature subsystems,
     * effectively _flattening_ the set of features within single region's subsystem.
     *
     * @param name The name of the subsystem.
     */
    public Subsystem(String name) {
        super(name, TYPE_SUBSYSTEM, Version.emptyVersion);
        this.name = name;
        this.parent = null;
        this.acceptDependencies = true;
        this.feature = null;
        this.importPolicy = SHARE_NONE_POLICY;
        this.exportPolicy = SHARE_NONE_POLICY;
        this.mandatory = true;
    }

    /**
     * Constructs subsystem for a feature that either imports/exports all caps or (see {@link Feature#getScoping()})
     * has configurable import/export policy + (|(type=karaf.subsystem)(type=karaf.feature)) filter in
     * {@link org.osgi.framework.namespace.IdentityNamespace#IDENTITY_NAMESPACE}.
     * Such subsystem requires type=karaf.feature; osgi.identity=feature-name[; version=feature-version].
     *
     * @param name The subsystem name.
     * @param feature The feature.
     * @param parent The parent subsystem.
     * @param mandatory True if mandatory, false else.
     */
    public Subsystem(String name, Feature feature, Subsystem parent, boolean mandatory) {
        super(name, TYPE_SUBSYSTEM, Version.emptyVersion);
        this.name = name;
        this.parent = parent;
        this.acceptDependencies = feature.getScoping() != null && feature.getScoping().acceptDependencies();
        this.feature = feature;
        this.mandatory = mandatory;
        if (feature.getScoping() != null) {
            this.importPolicy = createPolicy(feature.getScoping().getImports());
            this.importPolicy.put(IDENTITY_NAMESPACE, Collections.singleton(SUBSYSTEM_OR_FEATURE_FILTER));
            this.exportPolicy = createPolicy(feature.getScoping().getExports());
            this.exportPolicy.put(IDENTITY_NAMESPACE, Collections.singleton(SUBSYSTEM_OR_FEATURE_FILTER));
        } else {
            this.importPolicy = SHARE_ALL_POLICY;
            this.exportPolicy = SHARE_ALL_POLICY;
        }

        addIdentityRequirement(this,
                feature.getName(),
                TYPE_FEATURE,
                new VersionRange(VersionTable.getVersion(feature.getVersion()), true));
    }

    /**
     * Constructs child subsystem {@link Resource} for {@link FeaturesService#ROOT_REGION}'s child
     * that imports all caps and exports only caps with (type=karaf.subsystem).
     *
     * @param name The subsystem name.
     * @param parent The parent subsystem.
     * @param acceptDependencies True to accept dependencies, false else.
     * @param mandatory True to mandatory, false else.
     */
    public Subsystem(String name, Subsystem parent, boolean acceptDependencies, boolean mandatory) {
        super(name, TYPE_SUBSYSTEM, Version.emptyVersion);
        this.name = name;
        this.parent = parent;
        this.acceptDependencies = acceptDependencies;
        this.feature = null;
        this.mandatory = mandatory;
        this.importPolicy = SHARE_ALL_POLICY;
        this.exportPolicy = SHARE_NONE_POLICY;
    }

    public List getInstallable() {
        return installable;
    }

    public String getName() {
        return name;
    }

    public Subsystem getParent() {
        return parent;
    }

    public Collection getChildren() {
        return children;
    }

    public Subsystem getChild(String name) {
        for (Subsystem child : children) {
            if (child.getName().equals(name)) {
                return child;
            }
        }
        return null;
    }

    public boolean isAcceptDependencies() {
        return acceptDependencies;
    }

    public Map> getImportPolicy() {
        return importPolicy;
    }

    public Map> getExportPolicy() {
        return exportPolicy;
    }

    public Feature getFeature() {
        return feature;
    }

    /**
     * Create child subsystem for this subsystem. Child will become parent's mandatory requirement to force its resolution.
     *
     * @param name The subsystem name.
     * @param acceptDependencies True to accept dependencies, false else.
     */
    public Subsystem createSubsystem(String name, boolean acceptDependencies) {
        if (feature != null) {
            throw new UnsupportedOperationException("Can not create application subsystems inside a feature subsystem");
        }
        // Create subsystem
        String childName = getName() + "/" + name;
        Subsystem as = new Subsystem(childName, this, acceptDependencies, true);
        children.add(as);
        // Add a requirement to force its resolution
        ResourceUtils.addIdentityRequirement(this, childName, TYPE_SUBSYSTEM, (VersionRange) null);
        // Add it to repo
        installable.add(as);
        return as;
    }

    public void addSystemResource(Resource resource) {
        installable.add(resource);
    }

    public void requireFeature(String name, String range, boolean mandatory) {
        if (mandatory) {
            ResourceUtils.addIdentityRequirement(this, name, TYPE_FEATURE, range);
        } else {
            ResourceImpl res = new ResourceImpl();
            ResourceUtils.addIdentityRequirement(res, name, TYPE_FEATURE, range, false);
            dependentFeatures.addAll(res.getRequirements(null));
        }
    }

    public void require(String requirement) throws BundleException {
        int idx = requirement.indexOf(':');
        String type, req;
        if (idx >= 0) {
            type = requirement.substring(0, idx);
            req = requirement.substring(idx + 1);
        } else {
            type = "feature";
            req = requirement;
        }
        switch (type) {
        case "feature":
            addRequirement(toFeatureRequirement(req));
            break;
        case "requirement":
            addRequirement(req);
            break;
        case "bundle":
            bundles.add(req);
            break;
        }
    }

    protected void addRequirement(String requirement) throws BundleException {
        for (Requirement req : ResourceBuilder.parseRequirement(this, requirement)) {
            Object range = req.getAttributes().get(CAPABILITY_VERSION_ATTRIBUTE);
            if (range instanceof String) {
                req.getAttributes().put(CAPABILITY_VERSION_ATTRIBUTE, new VersionRange((String) range));
            }
            addRequirement(req);
        }
    }

    public Map getBundleInfos() {
        Map infos = new HashMap<>();
        for (DependencyInfo di : dependencies.values()) {
            infos.put(di.getLocation(), di);
        }
        return infos;
    }

    @SuppressWarnings("InfiniteLoopStatement")
    public void build(Map> allFeatures) throws Exception {
        doBuild(allFeatures, true);
    }

    private void doBuild(Map> allFeatures, boolean mandatory) throws Exception {
        for (Subsystem child : children) {
            child.doBuild(allFeatures, true);
        }
        if (feature != null) {
            // each dependant feature becomes a non-mandatory (why?) requirement of first parent that
            // accepts dependencies
            for (Dependency dep : feature.getDependencies()) {
                if (dep.isBlacklisted()) {
                    continue;
                }
                Subsystem ss = this;
                while (!ss.isAcceptDependencies()) {
                    ss = ss.getParent();
                }
                ss.requireFeature(dep.getName(), dep.getVersion(), false);
            }
            // each conditional feature becomes a child subsystem of this feature's subsystem
            for (Conditional cond : feature.getConditional()) {
                if (cond.isBlacklisted()) {
                    continue;
                }
                Feature fcond = cond.asFeature();
                String ssName = this.name + "#" + (fcond.hasVersion() ? fcond.getName() + "-" + fcond.getVersion() : fcond.getName());
                Subsystem fs = getChild(ssName);
                if (fs == null) {
                    fs = new Subsystem(ssName, fcond, this, true);
                    fs.doBuild(allFeatures, false);
                    installable.add(fs);
                    children.add(fs);
                }
            }
        }
        List processed = new ArrayList<>();
        while (true) {
            List requirements = getRequirements(IDENTITY_NAMESPACE);
            requirements.addAll(dependentFeatures);
            requirements.removeAll(processed);
            if (requirements.isEmpty()) {
                break;
            }
            // for each feature requirement on this subsystem (osgi.identity;type=karaf.feature), we create a
            // Subsystem representing mandatory feature.
            for (Requirement requirement : requirements) {
                String name = (String) requirement.getAttributes().get(IDENTITY_NAMESPACE);
                String type = (String) requirement.getAttributes().get(CAPABILITY_TYPE_ATTRIBUTE);
                VersionRange range = (VersionRange) requirement.getAttributes().get(CAPABILITY_VERSION_ATTRIBUTE);
                if (TYPE_FEATURE.equals(type) && allFeatures.containsKey(name)) {
                    for (Feature feature : allFeatures.get(name)) {
                        if (range == null || range.contains(VersionTable.getVersion(feature.getVersion()))) {
                            if (feature != this.feature && !feature.isBlacklisted()) {
                                String ssName = this.name + "#" + (feature.hasVersion() ? feature.getName() + "-" + feature.getVersion() : feature.getName());
                                Subsystem fs = getChild(ssName);
                                if (fs == null) {
                                    fs = new Subsystem(ssName, feature, this, mandatory && !SubsystemResolveContext.isOptional(requirement));
                                    fs.build(allFeatures);
                                    installable.add(fs);
                                    children.add(fs);
                                }
                            }
                        }
                    }
                }
                processed.add(requirement);
            }
        }
    }

    public Set collectPrerequisites() {
        Set prereqs = new HashSet<>();
        doCollectPrerequisites(prereqs);
        return prereqs;
    }

    private void doCollectPrerequisites(Set prereqs) {
        for (Subsystem child : children) {
            child.doCollectPrerequisites(prereqs);
        }
        if (feature != null) {
            boolean match = false;
            for (String prereq : prereqs) {
                String[] p = prereq.split("/");
                if (feature.getName().equals(p[0])
                        && VersionRange.parseVersionRange(p[1]).contains(Version.parseVersion(feature.getVersion()))) {
                    // our feature is already among prerequisites, so ...
                    match = true;
                    break;
                }
            }
            // ... we won't be adding its prerequisites - they'll be handled after another PartialDeploymentException
            if (!match) {
                for (Dependency dep : feature.getDependencies()) {
                    if (dep.isPrerequisite()) {
                        prereqs.add(dep.toString());
                    }
                }
            }
        }
    }

    /**
     * Downloads bundles for all the features in current and child subsystems. But also collects bundles
     * as {@link DependencyInfo}.
     *
     * @param manager The {@link DownloadManager} to use.
     * @param featureResolutionRange The feature resolution range to use.
     * @param serviceRequirements The {@link FeaturesService.ServiceRequirementsBehavior} behavior to use.
     * @param repos The {@link RepositoryManager} to use.
     * @param callback The {@link SubsystemResolverCallback} to use.
     */
    @SuppressWarnings("InfiniteLoopStatement")
    public void downloadBundles(DownloadManager manager,
                                String featureResolutionRange,
                                final FeaturesService.ServiceRequirementsBehavior serviceRequirements,
                                RepositoryManager repos,
                                SubsystemResolverCallback callback) throws Exception {
        for (Subsystem child : children) {
            child.downloadBundles(manager, featureResolutionRange, serviceRequirements, repos, callback);
        }

        // collect BundleInfos for given feature - both direct /s and //s
        final Map infos = new HashMap<>();
        final Downloader downloader = manager.createDownloader();
        if (feature != null) {
            for (Conditional cond : feature.getConditional()) {
                if (!cond.isBlacklisted()) {
                    for (final BundleInfo bi : cond.getBundles()) {
                        // bundles from conditional features will be added as non-mandatory requirements
                        infos.put(bi, cond);
                    }
                }
            }
            for (BundleInfo bi : feature.getBundles()) {
                infos.put(bi, null);
            }
        }

        // features model doesn't have blacklisted entries removed, but marked as blacklisted - we now don't have
        // to download them
        //infos.keySet().removeIf(Blacklisting::isBlacklisted);
        for (Iterator iterator = infos.keySet().iterator(); iterator.hasNext(); ) {
            BundleInfo bi = iterator.next();
            if (bi.isBlacklisted()) {
                iterator.remove();
                if (callback != null) {
                    callback.bundleBlacklisted(bi);
                }
            }
        }

        // all downloaded bundles
        final Map bundles = new ConcurrentHashMap<>();
        // resources for locations that were overriden in OSGi mode - to check whether the override should actually
        // take place, by checking resource's headers
        final Map overrides = new ConcurrentHashMap<>();

        boolean removeServiceRequirements = serviceRequirementsBehavior(feature, serviceRequirements);

        // download collected BundleInfo locations
        for (Map.Entry entry : infos.entrySet()) {
            final BundleInfo bi = entry.getKey();
            final String loc = bi.getLocation();
            downloader.download(loc, provider -> {
                // always download location (could be overriden)
                ResourceImpl resource = createResource(loc, getMetadata(provider), removeServiceRequirements);
                bundles.put(loc, resource);

                if (bi.isOverriden() == BundleInfo.BundleOverrideMode.OSGI) {
                    // also download original from original bundle URI to check if we should override by comparing
                    // symbolic name - requires MANIFEST.MF header access. If there should be no override, we'll get
                    // back to original URI
                    downloader.download(bi.getOriginalLocation(), provider2 -> {
                        ResourceImpl originalResource = createResource(bi.getOriginalLocation(),
                                getMetadata(provider2), removeServiceRequirements);
                        bundles.put(bi.getOriginalLocation(), originalResource);
                        // an entry in overrides map means that given location was overriden
                        overrides.put(loc, originalResource);
                    });
                }
            });
        }
        // download direct bundle: requirements - without consulting overrides
        for (Clause bundle : Parser.parseClauses(this.bundles.toArray(new String[this.bundles.size()]))) {
            final String loc = bundle.getName();
            downloader.download(loc, provider -> bundles.put(loc, createResource(loc, getMetadata(provider), removeServiceRequirements)));
        }
        // we *don't* have to download overrides separately - they're already taken into account from processed model

        // download additional libraries - only exported, so they're capabilities are taken into account during
        // resolution process
        if (feature != null) {
            for (Library library : feature.getLibraries()) {
                if (library.isExport()) {
                    final String loc = library.getLocation();
                    downloader.download(loc, provider -> bundles.put(loc, createResource(loc, getMetadata(provider), removeServiceRequirements)));
                }
            }
        }
        downloader.await();

        // opposite to what we had before. Currently bundles are already overriden at model level, but
        // as we finally have access to headers, we can compare symbolic names and if override mode is OSGi, then
        // we can restore original resource if there should be no override.
        Overrides.override(bundles, overrides);

        if (feature != null) {
            // Add conditionals
            Map resConds = new HashMap<>();
            for (Conditional cond : feature.getConditional()) {
                if (cond.isBlacklisted()) {
                    continue;
                }
                FeatureResource resCond = FeatureResource.build(feature, cond, featureResolutionRange, bundles);
                // feature's subsystem will optionally require conditional feature resource
                addIdentityRequirement(this, resCond, false);
                // but it's a mandatory requirement in other way
                addIdentityRequirement(resCond, this, true);
                installable.add(resCond);
                resConds.put(cond, resCond);
            }
            // Add features and make it require given subsystem that represents logical feature requirement
            FeatureResource resFeature = FeatureResource.build(feature, featureResolutionRange, bundles);
            addIdentityRequirement(resFeature, this);
            installable.add(resFeature);
            // Add dependencies
            for (Map.Entry entry : infos.entrySet()) {
                final BundleInfo bi = entry.getKey();
                final String loc = bi.getLocation();
                final Conditional cond = entry.getValue();
                ResourceImpl res = bundles.get(loc);
                int sl = bi.getStartLevel() <= 0 ? feature.getStartLevel() : bi.getStartLevel();
                if (cond != null) {
                    // bundle of conditional feature will have mandatory requirement on it
                    addIdentityRequirement(res, resConds.get(cond), true);
                }
                boolean mandatory = !bi.isDependency() && cond == null;
                if (bi.isDependency()) {
                    addDependency(res, mandatory, bi.isStart(), sl, bi.isBlacklisted());
                } else {
                    doAddDependency(res, mandatory, bi.isStart(), sl, bi.isBlacklisted());
                }
            }
            for (Library library : feature.getLibraries()) {
                if (library.isExport()) {
                    final String loc = library.getLocation();
                    ResourceImpl res = bundles.get(loc);
                    addDependency(res, false, false, 0, false);
                }
            }
            for (String uri : feature.getResourceRepositories()) {
                BaseRepository repo = repos.getRepository(feature.getRepositoryUrl(), uri);
                for (Resource resource : repo.getResources()) {
                    ResourceImpl res = cloneResource(resource);
                    addDependency(res, false, true, 0, false);
                }
            }
        }
        for (Clause bundle : Parser.parseClauses(this.bundles.toArray(new String[this.bundles.size()]))) {
            final String loc = bundle.getName();
            boolean dependency = Boolean.parseBoolean(bundle.getAttribute("dependency"));
            boolean start = bundle.getAttribute("start") == null || Boolean.parseBoolean(bundle.getAttribute("start"));
            boolean blacklisted = bundle.getAttribute("blacklisted") != null && Boolean.parseBoolean(bundle.getAttribute("blacklisted"));
            int startLevel = 0;
            try {
                startLevel = Integer.parseInt(bundle.getAttribute("start-level"));
            } catch (NumberFormatException e) {
                // Ignore
            }
            if (dependency) {
                addDependency(bundles.get(loc), false, start, startLevel, blacklisted);
            } else {
                doAddDependency(bundles.get(loc), true, start, startLevel, blacklisted);
                // non dependency bundle will be added as osgi.identity req on type=osgi.bundle
                addIdentityRequirement(this, bundles.get(loc));
            }
        }
        // Compute dependencies
        for (DependencyInfo info : dependencies.values()) {
            installable.add(info.resource);
            // bundle resource will have a requirement on its feature's subsystem too
            // when bundle is declared with dependency="true", it will have a requirement on its region's subsystem
            addIdentityRequirement(info.resource, this, info.mandatory);
        }
    }

    /**
     * How to handle requirements from {@link org.osgi.namespace.service.ServiceNamespace#SERVICE_NAMESPACE} for
     * given feature.
     */
    private boolean serviceRequirementsBehavior(Feature feature, FeaturesService.ServiceRequirementsBehavior serviceRequirements) {
        if (FeaturesService.ServiceRequirementsBehavior.Disable == serviceRequirements) {
            return true;
        } else if (feature != null && FeaturesService.ServiceRequirementsBehavior.Default == serviceRequirements) {
            return FeaturesNamespaces.URI_1_0_0.equals(feature.getNamespace())
                    || FeaturesNamespaces.URI_1_1_0.equals(feature.getNamespace())
                    || FeaturesNamespaces.URI_1_2_0.equals(feature.getNamespace())
                    || FeaturesNamespaces.URI_1_2_1.equals(feature.getNamespace());
        } else {
            return false;
        }
    }

    ResourceImpl cloneResource(Resource resource) {
        ResourceImpl res = new ResourceImpl();
        for (Capability cap : resource.getCapabilities(null)) {
            res.addCapability(new CapabilityImpl(res, cap.getNamespace(),
                    new StringArrayMap<>(cap.getDirectives()), new StringArrayMap<>(cap.getAttributes())));
        }
        for (Requirement req : resource.getRequirements(null)) {
            SimpleFilter sf;
            if (req instanceof RequirementImpl) {
                sf = ((RequirementImpl) req).getFilter();
            } else if (req.getDirectives().containsKey(REQUIREMENT_FILTER_DIRECTIVE)) {
                sf = SimpleFilter.parse(req.getDirectives().get(REQUIREMENT_FILTER_DIRECTIVE));
            } else {
                sf = SimpleFilter.convert(req.getAttributes());
            }
            res.addRequirement(new RequirementImpl(res, req.getNamespace(),
                    new StringArrayMap<>(req.getDirectives()), new StringArrayMap<>(req.getAttributes()), sf));
        }
        return res;
    }

    Map getMetadata(StreamProvider provider) throws IOException {
        try (
                ZipInputStream zis = new ZipInputStream(provider.open())
        ) {
            ZipEntry entry;
            while ((entry = zis.getNextEntry()) != null) {
                if (MANIFEST_NAME.equals(entry.getName())) {
                    Attributes attributes = new Manifest(zis).getMainAttributes();
                    Map headers = new HashMap<>();
                    for (Map.Entry attr : attributes.entrySet()) {
                        headers.put(attr.getKey().toString(), attr.getValue().toString());
                    }
                    return headers;
                }
            }
        }
        throw new IllegalArgumentException("Resource " + provider.getUrl() + " does not contain a manifest");
    }

    /**
     * Adds a {@link Resource} as dependency if this subsystem {@link Subsystem#isAcceptDependencies() accepts dependencies},
     * otherwise, the dependency is added to parent subsystem, effectively searching for first parent subsystem representing
     * region or scoped feature.
     */
    void addDependency(ResourceImpl resource, boolean mandatory, boolean start, int startLevel, boolean blacklisted) {
        if (isAcceptDependencies()) {
            doAddDependency(resource, mandatory, start, startLevel, blacklisted);
        } else {
            parent.addDependency(resource, mandatory, start, startLevel, blacklisted);
        }
    }

    /**
     * Adds a {@link Resource} to this subsystem.
     */
    private void doAddDependency(ResourceImpl resource, boolean mandatory, boolean start, int startLevel, boolean blacklisted) {
        String id = ResolverUtil.getSymbolicName(resource) + "|" + ResolverUtil.getVersion(resource);
        DependencyInfo info = new DependencyInfo(resource, mandatory, start, startLevel, blacklisted);
        dependencies.merge(id, info, this::merge);
    }

    /**
     * Merges two dependencies by taking lower start level, stronger mandatory option and stronger
     * start option.
     */
    private DependencyInfo merge(DependencyInfo di1, DependencyInfo di2) {
        DependencyInfo info = new DependencyInfo();
        if (di1.resource != di2.resource) {
            Requirement r1 = getFirstIdentityReq(di1.resource);
            Requirement r2 = getFirstIdentityReq(di2.resource);
            if (r1 == null) {
                info.resource = di1.resource;
            } else if (r2 == null) {
                info.resource = di2.resource;
            } else {
                String id = ResolverUtil.getSymbolicName(di1.resource) + "/" + ResolverUtil.getVersion(di1.resource);
                throw new IllegalStateException("Resource " + id + " is duplicated on subsystem " + this.toString() + ". First resource requires " + r1 + " while the second requires " + r2);
            }
        } else {
            info.resource = di1.resource;
        }
        info.mandatory = di1.mandatory | di2.mandatory;
        info.start = di1.start | di2.start;
        if (di1.startLevel > 0 && di2.startLevel > 0) {
            info.startLevel = Math.min(di1.startLevel, di2.startLevel);
        } else {
            info.startLevel = Math.max(di1.startLevel, di2.startLevel);
        }
        return info;
    }

    private RequirementImpl getFirstIdentityReq(ResourceImpl resource) {
        for (Requirement r : resource.getRequirements(null)) {
            if (IDENTITY_NAMESPACE.equals(r.getNamespace())) {
                return (RequirementImpl) r;
            }
        }
        return null;
    }

    /**
     * TODO DOCUMENT: More generic than just {@link BundleInfo}
     */
    class DependencyInfo implements BundleInfo {
        ResourceImpl resource;
        boolean mandatory;
        boolean start;
        int startLevel;
        boolean blacklisted;
        BundleInfo.BundleOverrideMode overriden;

        public DependencyInfo() {
        }

        public DependencyInfo(ResourceImpl resource, boolean mandatory, boolean start, int startLevel, boolean blacklisted) {
            this.resource = resource;
            this.mandatory = mandatory;
            this.start = start;
            this.startLevel = startLevel;
            this.blacklisted = blacklisted;
        }

        @Override
        public boolean isStart() {
            return start;
        }

        @Override
        public int getStartLevel() {
            return startLevel;
        }

        @Override
        public String getLocation() {
            return getUri(resource);
        }

        @Override
        public String getOriginalLocation() {
            // resource is already overriden
            return getUri(resource);
        }

        @Override
        public boolean isDependency() {
            return !mandatory;
        }

        @Override
        public boolean isBlacklisted() {
            return blacklisted;
        }

        @Override
        public BundleInfo.BundleOverrideMode isOverriden() {
            return overriden;
        }

        public void setOverriden(BundleInfo.BundleOverrideMode overriden) {
            this.overriden = overriden;
        }

        @Override
        public String toString() {
            return "DependencyInfo{" +
                    "resource=" + resource +
                    '}';
        }
    }

    Map> createPolicy(List filters) {
        Map> policy = new HashMap<>();
        for (ScopeFilter filter : filters) {
            addToMapSet(policy, filter.getNamespace(), filter.getFilter());
        }
        return policy;
    }

    ResourceImpl createResource(String uri, Map headers, boolean removeServiceRequirements) throws Exception {
        try {
            return ResourceBuilder.build(uri, headers, removeServiceRequirements);
        } catch (BundleException e) {
            throw new Exception("Unable to create resource for bundle " + uri, e);
        }
    }

    @Override
    public String toString() {
        return getName();
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy