blob: 58f64ec4abc7286ae99bfff5ce3867e9d2f073df [file] [log] [blame]
/*
* 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 optionally depends on BFactory.
* component A optionally depends on B
*
* 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 A.bind(B) should be called back.
*
* @author <a href="mailto:dev@felix.apache.org">Felix Project Team</a>
*/
public class FELIX4913_OptionalCallbackInvokedTwiceTest extends TestBase {
public void test_A_Defines_BDependency_FromInitMethod() throws Throwable {
final DependencyManager m = getDM();
Ensure e = new Ensure();
Component bFactory = m.createComponent().setImplementation(new BFactory()).setInterface(BFactory.class.getName(), null);
Dependency dep = m.createServiceDependency().setService(B.class).setRequired(false).setCallbacks("bind", "unbind");
Component a = m.createComponent()
.setImplementation(new A(e, dep))
.add(m.createServiceDependency().setService(BFactory.class).setRequired(false).setCallbacks("bind", "unbind"));
// Enable first bFactory.
m.add(bFactory);
// Then Enable A.
m.add(a);
// A should get BFactory, then it should instantiate B, and B should then be bound to A.
e.waitForStep(4, 5000);
// Now, remove BFactory. The AComponent should then call bFactory.removeB(), abd A.unbind(B) should be called.
m.remove(bFactory);
e.waitForStep(6, 5000);
e.ensure();
clearComponents();
}
public void test_A_Defines_BDependency_BeforeActivation() 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, null))
.add(m.createServiceDependency().setService(B.class).setRequired(false).setCallbacks("bind", "unbind"))
.add(m.createServiceDependency().setService(BFactory.class).setRequired(false).setCallbacks("bind", "unbind"));
// Enable first bFactory.
m.add(bFactory);
// Then Enable A.
m.add(a);
// A should get BFactory, then it should instantiate B, and B should then be bound to A.
e.waitForStep(4, 5000);
// Now, remove BFactory. The AComponent should then call bFactory.removeB(), abd A.unbind(B) should be called.
m.remove(bFactory);
e.waitForStep(6, 5000);
e.ensure();
clearComponents();
}
public static class A {
final Ensure m_e;
final Dependency m_BDependency;
public A(Ensure e, Dependency bfactoryDependency) {
m_e = e;
m_BDependency = bfactoryDependency;
}
void init(Component component) {
m_e.step(1);
if (m_BDependency != null) {
component.add(m_BDependency);
}
}
void start() {
m_e.step(2);
}
void bind(BFactory bFactory) {
m_e.step(3);
bFactory.createB();
}
void unbind(BFactory bFactory) {
m_e.step(5);
bFactory.deleteB();
}
void bind(B b) {
m_e.step(4);
}
void unbind(B b) {
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 {
}
}