OSGi-DS:DS 组件中对所需服务的可选引用,并不是真正可动态更新的
OSGi-DS: Optional reference to required service in DS Component, is not really dynamically updatable
我想启动有许多服务可选的 DS 组件,如果某些服务未启动,则这些组件不会被激活。所以我将这些可选服务的基数设置为可选,这样如果这些可选服务不可用,那么也可以激活目标组件。
但现在的问题是,如果我将任何这些可选服务的配置更改为有效,这样服务就可用,但它不会反映在该目标组件中
@Component(immediate = true, configurationPolicy = ConfigurationPolicy.IGNORE,name="directory.comp")
public class DirectoryControllers
{
@Reference(policy=ReferencePolicy.DYNAMIC)
private volatile IZimbra zimbra;
@Reference(policy=ReferencePolicy.DYNAMIC,cardinality=ReferenceCardinality.OPTIONAL)
private volatile IOpenDJ opendj;
@Reference(policy=ReferencePolicy.DYNAMIC,cardinality=ReferenceCardinality.OPTIONAL)
private volatile IOpenIDM openidm;
private ServletRegistration _registration;
@Activate void activate(BundleContext bc) throws ServletException, NamespaceException
{
AppProvisioners provisioners=new AppProvisioners(zimbra,openidm,opendj);
_registration = ServletRegistration.register(
bc, _httpService, "/middleware",
new ProvisioningController(_db,provisioners),
new UserEnrollmentController(_db,provisioners)
);
}
}
在这里,zimbra 组件不是可选的,所以每当我更改配置时,它都会在 DirectoryControllers 组件中受到影响,但对于 openidm、opendj 组件则不是这样,当然它是可选的,但在更改它们的配置时有效,它是在 DirectoryControllers 组件中不受影响
我也试过了bind/unbind:
private volatile IOpenIDM openidm;
@Reference(name = "openidm.service", service = IOpenIDM.class, cardinality = ReferenceCardinality.OPTIONAL, policy = ReferencePolicy.DYNAMIC, unbind = "unbindOpenIDMService")
public void bindOpenIDMService(IOpenIDM openidm) {
this.openidm = openidm;
}
public void unbindOpenIDMService(IOpenIDM openidm) {
this.openidm = null;
}
我是否遗漏了导致此问题的任何内容?
谢谢。
我认为问题在于,因为这些组件是可选的,它们可能在激活时存在,也可能不存在,如果可选引用发生变化,它将不会被重新激活。
我会说尝试在组件的绑定/解除绑定方法中实现注册代码,在这种情况下,这是唯一可以 'notice' 更改的地方。
我认为您需要设置 policyOption
以引用 ReferencePolicyOption.GREEDY
。 policyOption
的默认值是 ReferencePolicyOption.RELUCTANT
,当可选服务可用时不会重新激活组件。
我想启动有许多服务可选的 DS 组件,如果某些服务未启动,则这些组件不会被激活。所以我将这些可选服务的基数设置为可选,这样如果这些可选服务不可用,那么也可以激活目标组件。
但现在的问题是,如果我将任何这些可选服务的配置更改为有效,这样服务就可用,但它不会反映在该目标组件中
@Component(immediate = true, configurationPolicy = ConfigurationPolicy.IGNORE,name="directory.comp")
public class DirectoryControllers
{
@Reference(policy=ReferencePolicy.DYNAMIC)
private volatile IZimbra zimbra;
@Reference(policy=ReferencePolicy.DYNAMIC,cardinality=ReferenceCardinality.OPTIONAL)
private volatile IOpenDJ opendj;
@Reference(policy=ReferencePolicy.DYNAMIC,cardinality=ReferenceCardinality.OPTIONAL)
private volatile IOpenIDM openidm;
private ServletRegistration _registration;
@Activate void activate(BundleContext bc) throws ServletException, NamespaceException
{
AppProvisioners provisioners=new AppProvisioners(zimbra,openidm,opendj);
_registration = ServletRegistration.register(
bc, _httpService, "/middleware",
new ProvisioningController(_db,provisioners),
new UserEnrollmentController(_db,provisioners)
);
}
}
在这里,zimbra 组件不是可选的,所以每当我更改配置时,它都会在 DirectoryControllers 组件中受到影响,但对于 openidm、opendj 组件则不是这样,当然它是可选的,但在更改它们的配置时有效,它是在 DirectoryControllers 组件中不受影响
我也试过了bind/unbind:
private volatile IOpenIDM openidm;
@Reference(name = "openidm.service", service = IOpenIDM.class, cardinality = ReferenceCardinality.OPTIONAL, policy = ReferencePolicy.DYNAMIC, unbind = "unbindOpenIDMService")
public void bindOpenIDMService(IOpenIDM openidm) {
this.openidm = openidm;
}
public void unbindOpenIDMService(IOpenIDM openidm) {
this.openidm = null;
}
我是否遗漏了导致此问题的任何内容?
谢谢。
我认为问题在于,因为这些组件是可选的,它们可能在激活时存在,也可能不存在,如果可选引用发生变化,它将不会被重新激活。
我会说尝试在组件的绑定/解除绑定方法中实现注册代码,在这种情况下,这是唯一可以 'notice' 更改的地方。
我认为您需要设置 policyOption
以引用 ReferencePolicyOption.GREEDY
。 policyOption
的默认值是 ReferencePolicyOption.RELUCTANT
,当可选服务可用时不会重新激活组件。