/******************************************************************************* * Copyright (c) 2011, 2013 VMware Inc. * 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: * VMware Inc. - initial contribution *******************************************************************************/ package org.eclipse.equinox.region.internal.tests.hook; import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertFalse; import static org.junit.Assert.assertTrue; import java.util.*; import org.eclipse.equinox.region.*; import org.eclipse.equinox.region.internal.tests.RegionReflectionUtils; import org.eclipse.virgo.teststubs.osgi.framework.StubBundle; import org.eclipse.virgo.teststubs.osgi.framework.StubBundleContext; import org.junit.*; import org.osgi.framework.*; import org.osgi.framework.hooks.bundle.CollisionHook; public class RegionBundleCollisionHookTests { private static final String BUNDLE_X = "X"; private static final Version BUNDLE_VERSION = new Version("0"); private long bundleId; private static final String REGION_A = "RegionA"; private static final String BUNDLE_A = "BundleA"; private static final String REGION_B = "RegionB"; private static final String BUNDLE_B = "BundleB"; private static final String REGION_C = "RegionC"; private static final String BUNDLE_C = "BundleC"; private static final String REGION_D = "RegionD"; private static final String BUNDLE_D = "BundleD"; private RegionDigraph digraph; private CollisionHook collisionFindHook; private Map<String, Region> regions; private Map<String, Bundle> bundles; private Collection<Bundle> candidates; private ThreadLocal<Region> threadLocal; @Before public void setUp() throws Exception { this.bundleId = 1L; this.regions = new HashMap<String, Region>(); this.bundles = new HashMap<String, Bundle>(); StubBundle stubSystemBundle = new StubBundle(0L, "osgi.framework", new Version("0"), "loc"); StubBundleContext stubBundleContext = new StubBundleContext(); stubBundleContext.addInstalledBundle(stubSystemBundle); this.threadLocal = new ThreadLocal<Region>(); this.digraph = RegionReflectionUtils.newStandardRegionDigraph(stubBundleContext, this.threadLocal); this.collisionFindHook = RegionReflectionUtils.newRegionBundleCollisionHook(this.digraph, this.threadLocal); this.candidates = new HashSet<Bundle>(); // Create regions A, B, C, D containing bundles A, B, C, D, respectively. createRegion(REGION_A, BUNDLE_A); createRegion(REGION_B, BUNDLE_B); createRegion(REGION_C, BUNDLE_C); createRegion(REGION_D, BUNDLE_D); createBundle(BUNDLE_X); } @After public void tearDown() throws Exception { // nothing } @Test public void testCollisionInSameRegion() { this.candidates.add(bundle(BUNDLE_A)); this.collisionFindHook.filterCollisions(CollisionHook.INSTALLING, bundle(BUNDLE_A), this.candidates); assertTrue(this.candidates.contains(bundle(BUNDLE_A))); this.collisionFindHook.filterCollisions(CollisionHook.UPDATING, bundle(BUNDLE_A), this.candidates); assertTrue(this.candidates.contains(bundle(BUNDLE_A))); } @Test public void testCollsionInDisconnectedRegion() { this.candidates.add(bundle(BUNDLE_B)); this.collisionFindHook.filterCollisions(CollisionHook.INSTALLING, bundle(BUNDLE_A), this.candidates); assertFalse(this.candidates.contains(bundle(BUNDLE_B))); this.candidates.add(bundle(BUNDLE_B)); this.collisionFindHook.filterCollisions(CollisionHook.UPDATING, bundle(BUNDLE_A), this.candidates); assertFalse(this.candidates.contains(bundle(BUNDLE_B))); } @Test public void testCollisionConnectedRegionAllowed() throws BundleException, InvalidSyntaxException { RegionFilter filter = createFilter(BUNDLE_B); region(REGION_A).connectRegion(region(REGION_B), filter); this.candidates.add(bundle(BUNDLE_B)); this.collisionFindHook.filterCollisions(CollisionHook.INSTALLING, bundle(BUNDLE_A), this.candidates); assertTrue(this.candidates.contains(bundle(BUNDLE_B))); this.candidates.add(bundle(BUNDLE_B)); this.collisionFindHook.filterCollisions(CollisionHook.UPDATING, bundle(BUNDLE_A), this.candidates); assertTrue(this.candidates.contains(bundle(BUNDLE_B))); } @Test public void testCollisionConnectedRegionFiltering() throws BundleException, InvalidSyntaxException { region(REGION_A).connectRegion(region(REGION_B), createFilter(BUNDLE_A)); Bundle aDuplicate = createBundle(BUNDLE_A, false); region(REGION_B).addBundle(aDuplicate); this.candidates.add(bundle(BUNDLE_B)); this.candidates.add(aDuplicate); this.collisionFindHook.filterCollisions(CollisionHook.INSTALLING, bundle(BUNDLE_A), this.candidates); assertFalse(this.candidates.contains(bundle(BUNDLE_B))); assertTrue(this.candidates.contains(aDuplicate)); this.candidates.add(bundle(BUNDLE_B)); this.collisionFindHook.filterCollisions(CollisionHook.UPDATING, bundle(BUNDLE_A), this.candidates); assertFalse(this.candidates.contains(bundle(BUNDLE_B))); assertTrue(this.candidates.contains(aDuplicate)); } @Test public void testCollsionTransitive() throws BundleException, InvalidSyntaxException { region(REGION_A).connectRegion(region(REGION_B), createFilter(BUNDLE_C)); region(REGION_B).connectRegion(region(REGION_C), createFilter(BUNDLE_C)); region(REGION_C).addBundle(bundle(BUNDLE_X)); this.candidates.add(bundle(BUNDLE_B)); this.candidates.add(bundle(BUNDLE_C)); this.candidates.add(bundle(BUNDLE_X)); this.collisionFindHook.filterCollisions(CollisionHook.INSTALLING, bundle(BUNDLE_A), this.candidates); assertTrue(this.candidates.contains(bundle(BUNDLE_C))); assertFalse(this.candidates.contains(bundle(BUNDLE_B))); assertFalse(this.candidates.contains(bundle(BUNDLE_X))); } @Test public void testCollisionInCyclicGraph() throws BundleException, InvalidSyntaxException { region(REGION_D).addBundle(bundle(BUNDLE_X)); region(REGION_A).connectRegion(region(REGION_B), createFilter(BUNDLE_D, BUNDLE_X)); region(REGION_B).connectRegion(region(REGION_A), createFilter()); region(REGION_B).connectRegion(region(REGION_D), createFilter(BUNDLE_D)); region(REGION_D).connectRegion(region(REGION_B), createFilter()); region(REGION_B).connectRegion(region(REGION_C), createFilter(BUNDLE_X)); region(REGION_C).connectRegion(region(REGION_B), createFilter()); region(REGION_C).connectRegion(region(REGION_D), createFilter(BUNDLE_X)); region(REGION_D).connectRegion(region(REGION_C), createFilter()); region(REGION_A).connectRegion(region(REGION_C), createFilter()); region(REGION_C).connectRegion(region(REGION_A), createFilter()); region(REGION_D).connectRegion(region(REGION_A), createFilter()); region(REGION_A).connectRegion(region(REGION_D), createFilter()); // Collide from region A. this.candidates.add(bundle(BUNDLE_B)); this.candidates.add(bundle(BUNDLE_C)); this.candidates.add(bundle(BUNDLE_D)); this.candidates.add(bundle(BUNDLE_X)); this.collisionFindHook.filterCollisions(CollisionHook.INSTALLING, bundle(BUNDLE_A), this.candidates); assertEquals(2, this.candidates.size()); assertTrue(this.candidates.contains(bundle(BUNDLE_D))); assertTrue(this.candidates.contains(bundle(BUNDLE_X))); this.candidates.add(bundle(BUNDLE_B)); this.candidates.add(bundle(BUNDLE_C)); this.collisionFindHook.filterCollisions(CollisionHook.UPDATING, bundle(BUNDLE_A), this.candidates); assertEquals(2, this.candidates.size()); assertTrue(this.candidates.contains(bundle(BUNDLE_D))); assertTrue(this.candidates.contains(bundle(BUNDLE_X))); // Collide from region B this.candidates.add(bundle(BUNDLE_B)); this.candidates.add(bundle(BUNDLE_C)); this.candidates.add(bundle(BUNDLE_D)); this.candidates.add(bundle(BUNDLE_X)); this.collisionFindHook.filterCollisions(CollisionHook.INSTALLING, bundle(BUNDLE_B), this.candidates); assertEquals(3, this.candidates.size()); assertTrue(this.candidates.contains(bundle(BUNDLE_B))); assertTrue(this.candidates.contains(bundle(BUNDLE_D))); assertTrue(this.candidates.contains(bundle(BUNDLE_X))); this.candidates.add(bundle(BUNDLE_C)); this.collisionFindHook.filterCollisions(CollisionHook.UPDATING, bundle(BUNDLE_B), this.candidates); assertEquals(3, this.candidates.size()); assertTrue(this.candidates.contains(bundle(BUNDLE_B))); assertTrue(this.candidates.contains(bundle(BUNDLE_D))); assertTrue(this.candidates.contains(bundle(BUNDLE_X))); } @Test public void testCollisionFromBundleInNoRegion() { this.candidates.add(bundle(BUNDLE_A)); Bundle stranger = createBundle("stranger"); this.collisionFindHook.filterCollisions(CollisionHook.INSTALLING, stranger, this.candidates); assertTrue(this.candidates.contains(bundle(BUNDLE_A))); this.collisionFindHook.filterCollisions(CollisionHook.UPDATING, stranger, this.candidates); assertTrue(this.candidates.contains(bundle(BUNDLE_A))); } private Region createRegion(String regionName, String... bundleSymbolicNames) throws BundleException { Region region = this.digraph.createRegion(regionName); for (String bundleSymbolicName : bundleSymbolicNames) { Bundle stubBundle = createBundle(bundleSymbolicName); region.addBundle(stubBundle); } this.regions.put(regionName, region); return region; } private Region region(String regionName) { return this.regions.get(regionName); } private RegionFilter createFilter(String... bundleSymbolicNames) throws InvalidSyntaxException { Collection<String> filters = new ArrayList<String>(bundleSymbolicNames.length); for (String bundleSymbolicName : bundleSymbolicNames) { filters.add('(' + RegionFilter.VISIBLE_BUNDLE_NAMESPACE + '=' + bundleSymbolicName + ')'); } RegionFilterBuilder builder = digraph.createRegionFilterBuilder(); for (String filter : filters) { builder.allow(RegionFilter.VISIBLE_BUNDLE_NAMESPACE, filter); } return builder.build(); } private Bundle createBundle(String bundleSymbolicName) { return createBundle(bundleSymbolicName, true); } private Bundle createBundle(String bundleSymbolicName, boolean cache) { Bundle stubBundle = new StubBundle(this.bundleId++, bundleSymbolicName, BUNDLE_VERSION, "loc:" + bundleSymbolicName); if (cache) this.bundles.put(bundleSymbolicName, stubBundle); return stubBundle; } private Bundle bundle(String bundleSymbolicName) { Bundle bundleA = this.bundles.get(bundleSymbolicName); return bundleA; } }