/******************************************************************************* * Copyright (c) 2003, 2012 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 * Danail Nachev - ProSyst - bug 218625 * Rob Harrop - SpringSource Inc. (bug 247522) *******************************************************************************/ package org.eclipse.osgi.internal.resolver; import java.util.Collections; import java.util.Map; import org.eclipse.osgi.internal.framework.EquinoxContainer; import org.eclipse.osgi.internal.resolver.BaseDescriptionImpl.BaseCapability; import org.eclipse.osgi.service.resolver.*; import org.eclipse.osgi.service.resolver.VersionRange; import org.eclipse.osgi.util.ManifestElement; import org.osgi.framework.*; import org.osgi.framework.namespace.AbstractWiringNamespace; import org.osgi.framework.wiring.*; import org.osgi.resource.Capability; import org.osgi.resource.Namespace; abstract class VersionConstraintImpl implements VersionConstraint { protected final Object monitor = new Object(); private String name; private VersionRange versionRange; private BundleDescription bundle; private BaseDescription supplier; private volatile Object userObject; public String getName() { synchronized (this.monitor) { if (Constants.SYSTEM_BUNDLE_SYMBOLICNAME.equals(name)) { StateImpl state = (StateImpl) getBundle().getContainingState(); return state == null ? EquinoxContainer.NAME : state.getSystemBundle(); } return name; } } public VersionRange getVersionRange() { synchronized (this.monitor) { if (versionRange == null) return VersionRange.emptyRange; return versionRange; } } public BundleDescription getBundle() { synchronized (this.monitor) { return bundle; } } public boolean isResolved() { synchronized (this.monitor) { return supplier != null; } } public BaseDescription getSupplier() { synchronized (this.monitor) { return supplier; } } public boolean isSatisfiedBy(BaseDescription candidate) { synchronized (this.monitor) { return false; } } protected void setName(String name) { synchronized (this.monitor) { this.name = name; } } protected void setVersionRange(VersionRange versionRange) { synchronized (this.monitor) { this.versionRange = versionRange; } } protected void setBundle(BundleDescription bundle) { synchronized (this.monitor) { this.bundle = bundle; } } protected void setSupplier(BaseDescription supplier) { synchronized (this.monitor) { this.supplier = supplier; } } protected abstract String getInternalNameSpace(); protected abstract Map<String, String> getInternalDirectives(); protected abstract Map<String, Object> getInteralAttributes(); protected abstract boolean hasMandatoryAttributes(String[] mandatory); public BundleRequirement getRequirement() { String namespace = getInternalNameSpace(); if (namespace == null) return null; return new BundleRequirementImpl(namespace); } public Object getUserObject() { return userObject; } public void setUserObject(Object userObject) { this.userObject = userObject; } class BundleRequirementImpl implements BundleRequirement { private final String namespace; public BundleRequirementImpl(String namespace) { this.namespace = namespace; } public String getNamespace() { return namespace; } public Map<String, String> getDirectives() { return Collections.unmodifiableMap(getInternalDirectives()); } public Map<String, Object> getAttributes() { return Collections.unmodifiableMap(getInteralAttributes()); } public BundleRevision getRevision() { return getBundle(); } public boolean matches(BundleCapability capability) { return isSatisfiedBy(((BaseCapability) capability).getBaseDescription()); } public int hashCode() { return System.identityHashCode(VersionConstraintImpl.this); } private VersionConstraintImpl getVersionConstraint() { return VersionConstraintImpl.this; } public boolean equals(Object obj) { if (this == obj) return true; if (!(obj instanceof BundleRequirementImpl)) return false; return ((BundleRequirementImpl) obj).getVersionConstraint() == VersionConstraintImpl.this; } public String toString() { return getNamespace() + BaseDescriptionImpl.toString(getAttributes(), false) + BaseDescriptionImpl.toString(getDirectives(), true); } public boolean matches(Capability capability) { if (capability instanceof BundleCapability) return matches((BundleCapability) capability); // now we must do the generic thing if (!namespace.equals(capability.getNamespace())) return false; String filterSpec = getDirectives().get(Namespace.REQUIREMENT_FILTER_DIRECTIVE); try { if (filterSpec != null && !FrameworkUtil.createFilter(filterSpec).matches(capability.getAttributes())) return false; } catch (InvalidSyntaxException e) { return false; } return hasMandatoryAttributes(ManifestElement.getArrayFromList(capability.getDirectives().get(AbstractWiringNamespace.CAPABILITY_MANDATORY_DIRECTIVE))); } public BundleRevision getResource() { return getRevision(); } } static StringBuffer addFilterAttributes(StringBuffer filter, Map<String, ?> attributes) { for (Map.Entry<String, ?> entry : attributes.entrySet()) { addFilterAttribute(filter, entry.getKey(), entry.getValue()); } return filter; } static StringBuffer addFilterAttribute(StringBuffer filter, String attr, Object value) { return addFilterAttribute(filter, attr, value, true); } static StringBuffer addFilterAttribute(StringBuffer filter, String attr, Object value, boolean escapeWildCard) { if (value instanceof VersionRange) { VersionRange range = (VersionRange) value; filter.append(range.toFilterString(attr)); } else { filter.append('(').append(attr).append('=').append(escapeValue(value, escapeWildCard)).append(')'); } return filter; } private static String escapeValue(Object o, boolean escapeWildCard) { String value = o.toString(); boolean escaped = false; int inlen = value.length(); int outlen = inlen << 1; /* inlen * 2 */ char[] output = new char[outlen]; value.getChars(0, inlen, output, inlen); int cursor = 0; for (int i = inlen; i < outlen; i++) { char c = output[i]; switch (c) { case '*' : if (!escapeWildCard) break; case '\\' : case '(' : case ')' : output[cursor] = '\\'; cursor++; escaped = true; break; } output[cursor] = c; cursor++; } return escaped ? new String(output, 0, cursor) : value; } }