问题
I have services that need to be created on a per-configuration base, each of which relies on an external resource and thus should manage it's own lifcycle (i.e. (de)register the service). Thus implementing these as DS and let SCR spawn multiple instances does not work.
One can implement a bundle that registers a ManagedServiceFactory to accomplish this task perfectly (see my previous post). But as a consequence, if the factory depends on several other services, you need to start tracking those services and write a lot of glue code to get everything running. Instead I'd like to implement the factory as (singleton) declarative service, for which the SCR registers a ManagedServiceFactory
at the Service Registry.
Here's my attempt:
import org.osgi.framework.BundleContext;
import org.osgi.framework.ServiceRegistration;
import org.osgi.service.cm.ConfigurationException;
import org.osgi.service.cm.ManagedServiceFactory;
import org.osgi.service.component.ComponentContext;
@SuppressWarnings({ "rawtypes", "unchecked" })
public class Factory implements ManagedServiceFactory {
private BundleContext bundleCtxt;
private Map<String, ServiceRegistration> services;
public void activate(ComponentContext context) throws Exception {
System.out.println("actiavting...");
this.bundleCtxt = context.getBundleContext();
services = new HashMap<String, ServiceRegistration>();
}
public void deactivate(ComponentContext context) {
for(ServiceRegistration reg : services.values()) {
System.out.println("deregister " + reg);
reg.unregister();
}
services.clear();
}
@Override
public String getName() {
System.out.println("returning factory name");
return "my.project.servicefactory";
}
@Override
public void updated(String pid, Dictionary properties)
throws ConfigurationException {
System.out.println("retrieved update for pid " + pid);
ServiceRegistration reg = services.get(pid);
if (reg == null) {
services.put(pid, bundleCtxt.registerService(ServiceInterface.class,
new Service(), properties));
} else {
// i should to some update here
}
}
@Override
public void deleted(String pid) {
ServiceRegistration reg = services.get(pid);
if (reg != null) {
reg.unregister();
services.remove(pid);
}
}
}
and the service description:
<?xml version="1.0" encoding="UTF-8"?>
<scr:component xmlns:scr="http://www.osgi.org/xmlns/scr/v1.1.0" configuration-policy="ignore" name="my.project.servicefactory">
<implementation class="my.project.factory.Factory"/>
<service>
<provide interface="org.osgi.service.cm.ManagedServiceFactory"/>
</service>
<property name="service.pid" type="String" value="my.project.servicefactory"/>
</scr:component>
I already found out that the "factory" property in the service description is the wrong path, because this way the component is never registerd as ManagedServiceFactory
in the Service Registry, instead it becomes a ComponentFactory
.
As a kind of hack, I just added a component property, namely
<property name="service.pid" type="String" value="my.project.servicefactory"/>
and added configuration-policy="ignore"
. This works: configurations named my.project.servicefactory-foobar.cfg
are handed to my service, which registers them as in the Service Registry, everything fine.
But theres two things about it i do not like:
- manually setting the property
service.pid
feels like a dirty hack to me - setting
configuration-policy="ignore"
prevents me from configuring theManagedServiceFactory
itsself. If I escape this property or set it to require, I would get oneManagedServiceFactory
for a configuration namedmy.project.servicefactory.cfg
and then two services for each configuration named with the patternmy.project.servicefactory-foobar.cfg
: oneManagedServiceFactory
that the SCR spawns and oneServiceInterface
that my firstManagedServiceFactory
registers when it gets notified about this new configuration. (At least this is not growing exponetially because SCR overwrites theservice.pid
property for factory configurations)
So how should I set this up properly?
PS: For those wondering about my reference to configurations on their filenames: I use Felix Fileinstall for the configurations, thus foo.cfg
is put to the ConfigAdmin for PID foo
, and foo-bar.cfg
is put there for factory-pid foo
.
回答1:
Just use your DS instance headless, the properties, and register the service yourself:
@Component(immedate=true, provide={}, serviceFactory=true, configurationPolicy=require)
public class Mine {
BundleContext context;
volatile ServiceRegistration r;
@Activate
void activate(BundleContext context, Map<String,Object> map) {
this.context = context;
track(map);
}
@Deactivate
void deactivate() {
if ( r != null)
r.unregisterService();
}
void track(Map<String,Object> map) {
... // do your remote stuff
r = context.registerService(...);
...
}
}
回答2:
Why doesn't the support in DS for this not work for you? See 112.6:
Factory Configuration – If a factory PID exists, with zero or more Configurations, that is equal to the configuration PID, then for each Configuration, a component configuration must be created that will obtain additional component properties from Configuration Admin.
This says that if the configuration pid of your component is the same as the factory pid in CM, then DS will create an instance of your component for each configuration under the factory pid.
来源:https://stackoverflow.com/questions/11860240/how-do-you-properly-implement-a-managedservicefactory-as-decalarative-service-in