aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--jdisc_core/src/main/java/com/yahoo/jdisc/core/BundleCollisionHook.java107
-rw-r--r--jdisc_core/src/main/java/com/yahoo/jdisc/core/FelixFramework.java21
-rw-r--r--jdisc_core/src/main/java/com/yahoo/jdisc/core/FelixParams.java1
-rw-r--r--jdisc_core_test/integration_test/pom.xml12
-rw-r--r--jdisc_core_test/integration_test/src/test/java/com/yahoo/jdisc/core/BundleCollisionHookIntegrationTest.java88
-rw-r--r--jdisc_core_test/integration_test/src/test/java/com/yahoo/jdisc/core/FelixFrameworkIntegrationTest.java5
-rw-r--r--jdisc_core_test/test_bundles/cert-l1-dup/.gitignore2
-rw-r--r--jdisc_core_test/test_bundles/cert-l1-dup/pom.xml36
-rw-r--r--jdisc_core_test/test_bundles/cert-l1-dup/src/main/java/com/yahoo/jdisc/bundle/l/CertificateL.java15
-rw-r--r--jdisc_core_test/test_bundles/cert-ml-dup/.gitignore2
-rw-r--r--jdisc_core_test/test_bundles/cert-ml-dup/pom.xml39
-rw-r--r--jdisc_core_test/test_bundles/cert-ml-dup/src/main/java/com/yahoo/jdisc/bundle/m/CertificateM.java18
-rw-r--r--jdisc_core_test/test_bundles/pom.xml2
13 files changed, 342 insertions, 6 deletions
diff --git a/jdisc_core/src/main/java/com/yahoo/jdisc/core/BundleCollisionHook.java b/jdisc_core/src/main/java/com/yahoo/jdisc/core/BundleCollisionHook.java
new file mode 100644
index 00000000000..6fc7c368ad1
--- /dev/null
+++ b/jdisc_core/src/main/java/com/yahoo/jdisc/core/BundleCollisionHook.java
@@ -0,0 +1,107 @@
+package com.yahoo.jdisc.core;
+
+import org.osgi.framework.Bundle;
+import org.osgi.framework.BundleContext;
+import org.osgi.framework.BundleEvent;
+import org.osgi.framework.ServiceRegistration;
+import org.osgi.framework.Version;
+import org.osgi.framework.hooks.bundle.CollisionHook;
+import org.osgi.framework.hooks.bundle.EventHook;
+
+import java.util.Collection;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.Map;
+import java.util.Objects;
+import java.util.Set;
+
+/**
+ * A bundle {@link CollisionHook} that contains a set of bundles that are allowed to collide with
+ * bundles that are about to be installed. In order to clean up when bundles are uninstalled, this
+ * is also a bundle {@link EventHook}.
+ *
+ * Thread safe
+ *
+ * @author gjoranv
+ */
+public class BundleCollisionHook implements CollisionHook, EventHook {
+
+ private ServiceRegistration<?> registration;
+ private Map<Bundle, BsnVersion> allowedDuplicates = new HashMap<>(5);
+
+ public void start(BundleContext context) {
+ if (registration != null) {
+ throw new IllegalStateException();
+ }
+ registration = context.registerService(new String[]{CollisionHook.class.getName(), EventHook.class.getName()},
+ this, null);
+ }
+
+ public void stop() {
+ registration.unregister();
+ registration = null;
+ }
+
+ /**
+ * Adds a collection of bundles to the allowed duplicates.
+ */
+ synchronized void allowDuplicateBundles(Collection<Bundle> bundles) {
+ for (var bundle : bundles) {
+ allowedDuplicates.put(bundle, new BsnVersion(bundle.getSymbolicName(), bundle.getVersion()));
+ }
+ }
+
+ /**
+ * Cleans up the allowed duplicates when a bundle is uninstalled.
+ */
+ @Override
+ public void event(BundleEvent event, Collection<BundleContext> contexts) {
+ if (event.getType() != BundleEvent.UNINSTALLED) return;
+
+ synchronized (this) {
+ allowedDuplicates.remove(event.getBundle());
+ }
+ }
+
+ @Override
+ public synchronized void filterCollisions(int operationType, Bundle target, Collection<Bundle> collisionCandidates) {
+ Set<Bundle> whitelistedCandidates = new HashSet<>();
+ for (var bundle : collisionCandidates) {
+ var bsnVersion = new BsnVersion(bundle.getSymbolicName(), bundle.getVersion());
+
+ // This is O(n), but n should be small here, plus this is only called when bundles collide.
+ if (allowedDuplicates.containsValue(bsnVersion)) {
+ whitelistedCandidates.add(bundle);
+ }
+ }
+ collisionCandidates.removeAll(whitelistedCandidates);
+ }
+
+
+ static class BsnVersion {
+
+ private final String symbolicName;
+ private final Version version;
+
+ BsnVersion(String symbolicName, Version version) {
+ this.symbolicName = symbolicName;
+ this.version = version;
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) return true;
+ if (o == null || getClass() != o.getClass()) return false;
+ BsnVersion that = (BsnVersion) o;
+ return Objects.equals(symbolicName, that.symbolicName) &&
+ version.equals(that.version);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(symbolicName, version);
+ }
+
+ }
+
+}
diff --git a/jdisc_core/src/main/java/com/yahoo/jdisc/core/FelixFramework.java b/jdisc_core/src/main/java/com/yahoo/jdisc/core/FelixFramework.java
index 96fc0c91d2d..6c382153944 100644
--- a/jdisc_core/src/main/java/com/yahoo/jdisc/core/FelixFramework.java
+++ b/jdisc_core/src/main/java/com/yahoo/jdisc/core/FelixFramework.java
@@ -15,6 +15,7 @@ import org.osgi.framework.wiring.FrameworkWiring;
import java.io.File;
import java.util.Arrays;
+import java.util.Collection;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
@@ -26,6 +27,7 @@ import java.util.logging.Logger;
/**
* @author Simon Thoresen Hult
+ * @author gjoranv
*/
public class FelixFramework implements OsgiFramework {
@@ -35,11 +37,14 @@ public class FelixFramework implements OsgiFramework {
private final ConsoleLogManager logListener;
private final Felix felix;
+ private final BundleCollisionHook collisionHook;
+
@Inject
public FelixFramework(FelixParams params) {
deleteDirContents(new File(params.getCachePath()));
felix = new Felix(params.toConfig());
logListener = params.isLoggerEnabled() ? new ConsoleLogManager() : null;
+ collisionHook = new BundleCollisionHook();
}
@Override
@@ -48,6 +53,7 @@ public class FelixFramework implements OsgiFramework {
felix.start();
BundleContext ctx = felix.getBundleContext();
+ collisionHook.start(ctx);
logService.start(ctx);
logHandler.install(ctx);
if (logListener != null) {
@@ -65,6 +71,7 @@ public class FelixFramework implements OsgiFramework {
}
logHandler.uninstall();
logService.stop();
+ collisionHook.stop();
}
felix.stop();
try {
@@ -90,8 +97,8 @@ public class FelixFramework implements OsgiFramework {
for (Bundle bundle : bundles) {
if (!privileged && OsgiHeader.isSet(bundle, OsgiHeader.PRIVILEGED_ACTIVATOR)) {
log.log(Level.INFO, "OSGi bundle '" + bundle.getSymbolicName() + "' " +
- "states that it requires privileged " +
- "initialization, but privileges are not available. YMMV.");
+ "states that it requires privileged " +
+ "initialization, but privileges are not available. YMMV.");
}
if (bundle.getHeaders().get(Constants.FRAGMENT_HOST) != null) {
continue; // fragments can not be started
@@ -102,7 +109,7 @@ public class FelixFramework implements OsgiFramework {
}
private String startedBundlesMessage(List<Bundle> bundles) {
- StringBuilder sb = new StringBuilder("Started bundles: {" );
+ StringBuilder sb = new StringBuilder("Started bundles: {");
for (Bundle b : bundles)
sb.append("[" + b.getBundleId() + "]" + b.getSymbolicName() + ":" + b.getVersion() + ", ");
sb.setLength(sb.length() - 2);
@@ -127,9 +134,9 @@ public class FelixFramework implements OsgiFramework {
});
try {
long TIMEOUT_SECONDS = 60L;
- if ( ! latch.await(TIMEOUT_SECONDS, TimeUnit.SECONDS)) {
+ if (!latch.await(TIMEOUT_SECONDS, TimeUnit.SECONDS)) {
log.warning("No PACKAGES_REFRESHED FrameworkEvent received within " + TIMEOUT_SECONDS +
- " seconds of calling FrameworkWiring.refreshBundles()");
+ " seconds of calling FrameworkWiring.refreshBundles()");
}
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
@@ -146,6 +153,10 @@ public class FelixFramework implements OsgiFramework {
return Arrays.asList(felix.getBundleContext().getBundles());
}
+ public void allowDuplicateBundles(Collection<Bundle> bundles) {
+ collisionHook.allowDuplicateBundles(bundles);
+ }
+
private void installBundle(String bundleLocation, Set<String> mask, List<Bundle> out) throws BundleException {
bundleLocation = BundleLocationResolver.resolve(bundleLocation);
if (mask.contains(bundleLocation)) {
diff --git a/jdisc_core/src/main/java/com/yahoo/jdisc/core/FelixParams.java b/jdisc_core/src/main/java/com/yahoo/jdisc/core/FelixParams.java
index 9b877f68efd..0bca9388885 100644
--- a/jdisc_core/src/main/java/com/yahoo/jdisc/core/FelixParams.java
+++ b/jdisc_core/src/main/java/com/yahoo/jdisc/core/FelixParams.java
@@ -56,6 +56,7 @@ public class FelixParams {
ret.put(Constants.FRAMEWORK_SYSTEMPACKAGES, exportPackages.toString());
ret.put(Constants.SUPPORTS_BOOTCLASSPATH_EXTENSION, "true");
ret.put(Constants.FRAMEWORK_BOOTDELEGATION, "com.yourkit.runtime,com.yourkit.probes,com.yourkit.probes.builtin,com.singularity.*");
+ ret.put(Constants.FRAMEWORK_BSNVERSION, Constants.FRAMEWORK_BSNVERSION_MANAGED);
return ret;
}
}
diff --git a/jdisc_core_test/integration_test/pom.xml b/jdisc_core_test/integration_test/pom.xml
index 670d812c9e9..0b0a1ce81ef 100644
--- a/jdisc_core_test/integration_test/pom.xml
+++ b/jdisc_core_test/integration_test/pom.xml
@@ -153,6 +153,12 @@
</dependency>
<dependency>
<groupId>com.yahoo.vespa.jdisc_core</groupId>
+ <artifactId>cert-l1-dup</artifactId>
+ <version>${project.version}</version>
+ <scope>test</scope>
+ </dependency>
+ <dependency>
+ <groupId>com.yahoo.vespa.jdisc_core</groupId>
<artifactId>cert-l2</artifactId>
<version>${project.version}</version>
<scope>test</scope>
@@ -165,6 +171,12 @@
</dependency>
<dependency>
<groupId>com.yahoo.vespa.jdisc_core</groupId>
+ <artifactId>cert-ml-dup</artifactId>
+ <version>${project.version}</version>
+ <scope>test</scope>
+ </dependency>
+ <dependency>
+ <groupId>com.yahoo.vespa.jdisc_core</groupId>
<artifactId>cert-nac</artifactId>
<version>${project.version}</version>
<scope>test</scope>
diff --git a/jdisc_core_test/integration_test/src/test/java/com/yahoo/jdisc/core/BundleCollisionHookIntegrationTest.java b/jdisc_core_test/integration_test/src/test/java/com/yahoo/jdisc/core/BundleCollisionHookIntegrationTest.java
new file mode 100644
index 00000000000..58c6c1ba8d6
--- /dev/null
+++ b/jdisc_core_test/integration_test/src/test/java/com/yahoo/jdisc/core/BundleCollisionHookIntegrationTest.java
@@ -0,0 +1,88 @@
+package com.yahoo.jdisc.core;
+
+import com.yahoo.jdisc.test.TestDriver;
+import org.junit.After;
+import org.junit.Before;
+import org.junit.Test;
+import org.osgi.framework.Bundle;
+import org.osgi.framework.BundleException;
+
+import java.util.Collections;
+
+import static com.yahoo.jdisc.core.FelixFrameworkIntegrationTest.startBundle;
+import static org.junit.Assert.assertNotEquals;
+import static org.junit.Assert.assertTrue;
+import static org.junit.Assert.fail;
+
+/**
+ * Note that the '-dup' bundles are necessary, because felix ignores duplicate bundles with the same location.
+ *
+ * @author gjoranv
+ */
+public class BundleCollisionHookIntegrationTest {
+
+ private FelixFramework felix;
+
+ @Before
+ public void startFelix() throws Exception {
+ felix = TestDriver.newOsgiFramework();
+ felix.start();
+ }
+
+ @After
+ public void stopFelix() throws Exception {
+ felix.stop();
+ }
+
+ @Test
+ public void duplicate_bundles_cannot_be_installed_in_general() throws Exception {
+ startBundle(felix, "cert-l1.jar");
+ try {
+ startBundle(felix, "cert-l1-dup.jar");
+ fail("Expected exception due to duplicate bundles");
+ } catch (BundleException e) {
+ assertTrue(e.getMessage().contains("Bundle symbolic name and version are not unique"));
+ }
+ }
+
+ @Test
+ public void duplicate_bundles_can_be_installed_if_bsn_version_is_whitelisted() throws Exception {
+ Bundle bundleL = startBundle(felix, "cert-l1.jar");
+ felix.allowDuplicateBundles(Collections.singleton(bundleL));
+
+ Bundle bundleLdup = startBundle(felix, "cert-l1-dup.jar");
+ assertNotEquals(bundleL.getBundleId(), bundleLdup.getBundleId());
+ }
+
+ @Test
+ public void duplicates_whitelist_is_updated_when_bundles_are_uninstalled() throws Exception {
+ Bundle bundleL = startBundle(felix, "cert-l1.jar");
+ felix.allowDuplicateBundles(Collections.singleton(bundleL));
+
+ startBundle(felix, "cert-l1-dup.jar");
+
+ // Remove the original bundle -> will also remove the allowed duplicate
+ bundleL.uninstall();
+
+ // Trigger error by trying to re-install the bundle while the duplicate remains
+ try {
+ startBundle(felix, "cert-l1.jar");
+ fail("Expected exception due to duplicate bundles");
+ } catch (BundleException e) {
+ assertTrue(e.getMessage().contains("Bundle symbolic name and version are not unique"));
+ }
+
+ }
+
+ @Test
+ public void duplicates_whitelist_can_be_updated_with_additional_bundles() throws Exception {
+ Bundle bundleL = startBundle(felix, "cert-l1.jar");
+ Bundle bundleMl = startBundle(felix, "cert-ml.jar");
+ felix.allowDuplicateBundles(Collections.singleton(bundleL));
+ felix.allowDuplicateBundles(Collections.singleton(bundleMl));
+
+ startBundle(felix, "cert-l1-dup.jar");
+ startBundle(felix, "cert-ml-dup.jar");
+ }
+
+}
diff --git a/jdisc_core_test/integration_test/src/test/java/com/yahoo/jdisc/core/FelixFrameworkIntegrationTest.java b/jdisc_core_test/integration_test/src/test/java/com/yahoo/jdisc/core/FelixFrameworkIntegrationTest.java
index e8551b44a3c..9b7453d2d6b 100644
--- a/jdisc_core_test/integration_test/src/test/java/com/yahoo/jdisc/core/FelixFrameworkIntegrationTest.java
+++ b/jdisc_core_test/integration_test/src/test/java/com/yahoo/jdisc/core/FelixFrameworkIntegrationTest.java
@@ -111,8 +111,11 @@ public class FelixFrameworkIntegrationTest {
Bundle bundleL = startBundle(felix, "cert-l1.jar");
Bundle bundleM = startBundle(felix, "cert-ml.jar");
assertEquals(1, callClass(bundleM, "com.yahoo.jdisc.bundle.m.CertificateM"));
+
+ // Switch from l1 to l2 (identical bundles, except for bsn)
bundleL.uninstall();
startBundle(felix, "cert-l2.jar");
+
felix.refreshPackages();
assertEquals(2, callClass(bundleM, "com.yahoo.jdisc.bundle.m.CertificateM"));
felix.stop();
@@ -185,7 +188,7 @@ public class FelixFrameworkIntegrationTest {
"com.yahoo.vespa.jdisc_core.cert-q-frag");
}
- private static Bundle startBundle(FelixFramework felix, String bundleLocation) throws BundleException {
+ static Bundle startBundle(FelixFramework felix, String bundleLocation) throws BundleException {
List<Bundle> lst = felix.installBundle(bundleLocation);
assertEquals(1, lst.size());
felix.startBundles(lst, false);
diff --git a/jdisc_core_test/test_bundles/cert-l1-dup/.gitignore b/jdisc_core_test/test_bundles/cert-l1-dup/.gitignore
new file mode 100644
index 00000000000..3cc25b51fc4
--- /dev/null
+++ b/jdisc_core_test/test_bundles/cert-l1-dup/.gitignore
@@ -0,0 +1,2 @@
+/pom.xml.build
+/target
diff --git a/jdisc_core_test/test_bundles/cert-l1-dup/pom.xml b/jdisc_core_test/test_bundles/cert-l1-dup/pom.xml
new file mode 100644
index 00000000000..b1a61bc4145
--- /dev/null
+++ b/jdisc_core_test/test_bundles/cert-l1-dup/pom.xml
@@ -0,0 +1,36 @@
+<?xml version="1.0"?>
+<!-- Copyright 2017 Yahoo Holdings. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root. -->
+<project xmlns="http://maven.apache.org/POM/4.0.0"
+ xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+ xsi:schemaLocation="http://maven.apache.org/POM/4.0.0
+ http://maven.apache.org/xsd/maven-4.0.0.xsd">
+ <modelVersion>4.0.0</modelVersion>
+ <parent>
+ <groupId>com.yahoo.vespa.jdisc_core</groupId>
+ <artifactId>test_bundles</artifactId>
+ <version>7-SNAPSHOT</version>
+ </parent>
+ <artifactId>cert-l1-dup</artifactId>
+ <version>7-SNAPSHOT</version>
+ <packaging>bundle</packaging>
+ <name>${project.artifactId}</name>
+ <build>
+ <plugins>
+ <plugin>
+ <groupId>org.apache.felix</groupId>
+ <artifactId>maven-bundle-plugin</artifactId>
+ <extensions>true</extensions>
+ <configuration>
+ <instructions>
+ <Bundle-SymbolicName>
+ ${project.groupId}.cert-l1
+ </Bundle-SymbolicName>
+ <Export-Package>
+ com.yahoo.jdisc.bundle.l
+ </Export-Package>
+ </instructions>
+ </configuration>
+ </plugin>
+ </plugins>
+ </build>
+</project>
diff --git a/jdisc_core_test/test_bundles/cert-l1-dup/src/main/java/com/yahoo/jdisc/bundle/l/CertificateL.java b/jdisc_core_test/test_bundles/cert-l1-dup/src/main/java/com/yahoo/jdisc/bundle/l/CertificateL.java
new file mode 100644
index 00000000000..2060686ee4f
--- /dev/null
+++ b/jdisc_core_test/test_bundles/cert-l1-dup/src/main/java/com/yahoo/jdisc/bundle/l/CertificateL.java
@@ -0,0 +1,15 @@
+// Copyright 2017 Yahoo Holdings. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
+package com.yahoo.jdisc.bundle.l;
+
+import java.util.concurrent.Callable;
+
+/**
+ * @author Simon Thoresen Hult
+ */
+public class CertificateL implements Callable<Integer> {
+
+ @Override
+ public Integer call() throws Exception {
+ return 11;
+ }
+}
diff --git a/jdisc_core_test/test_bundles/cert-ml-dup/.gitignore b/jdisc_core_test/test_bundles/cert-ml-dup/.gitignore
new file mode 100644
index 00000000000..3cc25b51fc4
--- /dev/null
+++ b/jdisc_core_test/test_bundles/cert-ml-dup/.gitignore
@@ -0,0 +1,2 @@
+/pom.xml.build
+/target
diff --git a/jdisc_core_test/test_bundles/cert-ml-dup/pom.xml b/jdisc_core_test/test_bundles/cert-ml-dup/pom.xml
new file mode 100644
index 00000000000..4382664c923
--- /dev/null
+++ b/jdisc_core_test/test_bundles/cert-ml-dup/pom.xml
@@ -0,0 +1,39 @@
+<?xml version="1.0"?>
+<!-- Copyright 2017 Yahoo Holdings. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root. -->
+<project xmlns="http://maven.apache.org/POM/4.0.0"
+ xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+ xsi:schemaLocation="http://maven.apache.org/POM/4.0.0
+ http://maven.apache.org/xsd/maven-4.0.0.xsd">
+ <modelVersion>4.0.0</modelVersion>
+ <parent>
+ <groupId>com.yahoo.vespa.jdisc_core</groupId>
+ <artifactId>test_bundles</artifactId>
+ <version>7-SNAPSHOT</version>
+ </parent>
+ <artifactId>cert-ml-dup</artifactId>
+ <version>7-SNAPSHOT</version>
+ <packaging>bundle</packaging>
+ <name>${project.artifactId}</name>
+ <build>
+ <plugins>
+ <plugin>
+ <groupId>org.apache.felix</groupId>
+ <artifactId>maven-bundle-plugin</artifactId>
+ <extensions>true</extensions>
+ <configuration>
+ <instructions>
+ <Bundle-SymbolicName>
+ ${project.groupId}.cert-ml
+ </Bundle-SymbolicName>
+ <Import-Package>
+ com.yahoo.jdisc.bundle.l
+ </Import-Package>
+ <Export-Package>
+ com.yahoo.jdisc.bundle.m
+ </Export-Package>
+ </instructions>
+ </configuration>
+ </plugin>
+ </plugins>
+ </build>
+</project>
diff --git a/jdisc_core_test/test_bundles/cert-ml-dup/src/main/java/com/yahoo/jdisc/bundle/m/CertificateM.java b/jdisc_core_test/test_bundles/cert-ml-dup/src/main/java/com/yahoo/jdisc/bundle/m/CertificateM.java
new file mode 100644
index 00000000000..131c9e41a1e
--- /dev/null
+++ b/jdisc_core_test/test_bundles/cert-ml-dup/src/main/java/com/yahoo/jdisc/bundle/m/CertificateM.java
@@ -0,0 +1,18 @@
+// Copyright 2017 Yahoo Holdings. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
+package com.yahoo.jdisc.bundle.m;
+
+import java.util.concurrent.Callable;
+
+/**
+ * @author Simon Thoresen Hult
+ */
+public class CertificateM implements Callable<Integer> {
+
+ @Override
+ @SuppressWarnings({ "unchecked" })
+ public Integer call() throws Exception {
+ Class<?> certClass = Class.forName("com.yahoo.jdisc.bundle.l.CertificateL");
+ Callable<Integer> cert = (Callable<Integer>)certClass.getDeclaredConstructor().newInstance();
+ return cert.call();
+ }
+}
diff --git a/jdisc_core_test/test_bundles/pom.xml b/jdisc_core_test/test_bundles/pom.xml
index c11a94eacc5..d62d22c67f9 100644
--- a/jdisc_core_test/test_bundles/pom.xml
+++ b/jdisc_core_test/test_bundles/pom.xml
@@ -65,8 +65,10 @@
<module>cert-j-priv</module>
<module>cert-k-pkgs</module>
<module>cert-l1</module>
+ <module>cert-l1-dup</module>
<module>cert-l2</module>
<module>cert-ml</module>
+ <module>cert-ml-dup</module>
<module>cert-nac</module>
<module>cert-oa-path</module>
<module>cert-p-jar</module>