You are viewing a plain text version of this content. The canonical link for it is here.
Posted to commits@felix.apache.org by pd...@apache.org on 2015/09/19 00:04:51 UTC

svn commit: r1703921 - /felix/trunk/dependencymanager/org.apache.felix.dependencymanager.itest/src/org/apache/felix/dm/itest/api/FELIX5045_OptionalDependencyCBCalledBeforeStartTest.java

Author: pderop
Date: Fri Sep 18 22:04:50 2015
New Revision: 1703921

URL: http://svn.apache.org/viewvc?rev=1703921&view=rev
Log:
Added test case for the FELIX-5045 issue.

Added:
    felix/trunk/dependencymanager/org.apache.felix.dependencymanager.itest/src/org/apache/felix/dm/itest/api/FELIX5045_OptionalDependencyCBCalledBeforeStartTest.java

Added: felix/trunk/dependencymanager/org.apache.felix.dependencymanager.itest/src/org/apache/felix/dm/itest/api/FELIX5045_OptionalDependencyCBCalledBeforeStartTest.java
URL: http://svn.apache.org/viewvc/felix/trunk/dependencymanager/org.apache.felix.dependencymanager.itest/src/org/apache/felix/dm/itest/api/FELIX5045_OptionalDependencyCBCalledBeforeStartTest.java?rev=1703921&view=auto
==============================================================================
--- felix/trunk/dependencymanager/org.apache.felix.dependencymanager.itest/src/org/apache/felix/dm/itest/api/FELIX5045_OptionalDependencyCBCalledBeforeStartTest.java (added)
+++ felix/trunk/dependencymanager/org.apache.felix.dependencymanager.itest/src/org/apache/felix/dm/itest/api/FELIX5045_OptionalDependencyCBCalledBeforeStartTest.java Fri Sep 18 22:04:50 2015
@@ -0,0 +1,117 @@
+/*
+ * 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.felix.dm.itest.api;
+
+import org.apache.felix.dm.Component;
+import org.apache.felix.dm.Dependency;
+import org.apache.felix.dm.DependencyManager;
+import org.apache.felix.dm.itest.util.Ensure;
+import org.apache.felix.dm.itest.util.TestBase;
+import org.osgi.framework.BundleContext;
+import org.osgi.framework.ServiceRegistration;
+
+/**
+ * This test validates a corner case:
+ * 
+ * We have the following component players: A, BFactory, B.
+ * 
+ * component A defines from A.init() a required dependency on BFactory, and an optional dependency on B.
+ * component A has a "start" lifecycle callback.
+ * 
+ * when A.bind(BFactory factory) is called, the factory.create() method is then invoked, which triggers a registration of the B Service.
+ * At this point B is available, then A.start() should be called before A.bind(B).
+ * 
+ * @author <a href="mailto:dev@felix.apache.org">Felix Project Team</a>
+ */
+public class FELIX5045_OptionalDependencyCBCalledBeforeStartTest extends TestBase {
+    public void test_A_DependsOnBFactoryFromInit() throws Throwable {
+        final DependencyManager m = getDM();
+        Ensure e = new Ensure();
+        
+        Component bFactory = m.createComponent().setImplementation(new BFactory()).setInterface(BFactory.class.getName(), null);
+        Component a = m.createComponent().setImplementation(new A(e));            
+        
+        // Enable first bFactory.
+        m.add(bFactory);
+        
+        // Then enable A.
+        m.add(a);
+        
+        // A should get BFactory, then it should instantiate B, then A.start() should be called, then A.bind(B) should be called.
+        e.waitForStep(4, 5000);
+        
+        // Now, remove BFactory. A.unbind(B b) should be called, then 
+        m.remove(bFactory);
+        e.waitForStep(6, 5000);
+        e.ensure();
+    }
+             
+    public static class A {
+        final Ensure m_e;
+
+        public A(Ensure e) {
+            m_e = e;
+        }
+
+        void init(Component component) {
+            m_e.step(1);
+        	DependencyManager dm = component.getDependencyManager();
+            Dependency depBFactory = dm.createServiceDependency().setService(BFactory.class).setRequired(true).setCallbacks("bind", "unbind");
+            Dependency depB = dm.createServiceDependency().setService(B.class).setRequired(false).setCallbacks("bind", "unbind");
+            component.add(depBFactory, depB);
+        }
+        
+        void bind(BFactory bFactory) {
+            m_e.step(2);
+            bFactory.createB();
+        }
+
+        void start() {
+            m_e.step(3);
+        }
+            
+        void bind(B b) {  
+            m_e.step(4);
+        }
+        
+        void unbind(B b) {  
+            m_e.step(5);
+        }
+        
+        void unbind(BFactory bFactory) {
+        	m_e.step(6);
+        }
+    }
+    
+    public static class BFactory {
+        volatile BundleContext m_bctx;
+        ServiceRegistration m_registraiton;
+        
+        void createB() {
+            m_registraiton = m_bctx.registerService(B.class.getName(), new B(), null);
+        }
+        
+        void deleteB() {
+            m_registraiton.unregister();
+        }
+    }
+    
+    public static class B {
+    }
+}