blob: bb1ff013688e32823ee1d260e8273067526c363e [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.junit.Assert;
import org.apache.felix.dm.Component;
import org.apache.felix.dm.DependencyManager;
import org.apache.felix.dm.ServiceDependency;
import org.apache.felix.dm.itest.util.Ensure;
import org.apache.felix.dm.itest.util.TestBase;
/**
* This testcase verify that a Service is not started if one of its extra required dependencies
* is unavailable.
*
* @author <a href="mailto:dev@felix.apache.org">Felix Project Team</a>
*/
public class FELIX2369_ExtraDependencyTest extends TestBase
{
public void testExtraDependencies() {
DependencyManager m = getDM();
// helper class that ensures certain steps get executed in sequence
Ensure e = new Ensure();
// create a service consumer and provider
Component sp1 = m.createComponent().setInterface(MyService1.class.getName(), null).setImplementation(new MyService1Impl());
Component sc = m.createComponent().setImplementation(new MyClient(e, 1));
// provides the MyService1 service (but not the MyService2, which is required by MyClient).
m.add(sp1);
// add MyClient (it should not be invoked in its start() method because MyService2 is not there
m.add(sc);
// remove MyClient (it should not be invoked in its stop() method because it should not be active, since MyService2 is not there.
m.remove(sc);
e.waitForStep(2, 5000);
m.clear();
}
public interface MyService1 {
}
public interface MyService2 {
}
public static class MyService1Impl implements MyService1 {
}
public static class MyService2Impl implements MyService2 {
}
// This client is not using callbacks, but instead, it uses auto config.
public static class MyClient {
MyService1 m_myService2; // required/unavailable
private Ensure m_ensure;
private final int m_startStep;
public MyClient(Ensure e, int startStep) {
m_ensure = e;
m_startStep = startStep;
}
public void init(Component s) {
DependencyManager dm = s.getDependencyManager();
m_ensure.step(m_startStep);
ServiceDependency d1 =
dm.createServiceDependency() // this dependency is available at this point
.setService(MyService1.class)
.setRequired(false)
.setCallbacks("bind", null);
ServiceDependency d2 =
dm.createServiceDependency() // not available: we should not be started
.setService(MyService2.class)
.setRequired(true)
.setAutoConfig("m_myService2");
s.add(d1, d2); // atomically add these two dependencies
}
public void start() {
Assert.fail("start should not be called since MyService2 is unavailable");
}
void bind(MyService1 s1) { // optional/available
System.out.println("bound MyService1");
}
public void stop() {
Assert.fail("stop should not be called since we should not be active at this point");
}
public void destroy() {
m_ensure.step(m_startStep+1);
}
}
}