繁体   English   中英

如何干净地覆盖Jersey使用的默认ServiceLocator?

[英]How can I cleanly override the default ServiceLocator used by Jersey?

我正在开发一个应用程序,它使用Jersey(2.5)作为其REST前端,Jetty作为嵌入式HTTP(S)服务器,两者都采用所谓的“嵌入式”方式,例如。 不依靠制作.war并部署它,而是通过程序化配置处理程序,资源,注入......

我想以某种方式覆盖Jersey在服务器端使用的HK2 ServiceLocator ,或者可能为父服务定位器提供一个父服务器来解析在应用程序的REST部分之外定义的依赖关系。 从我看到的代码来看,这似乎是不可能的:ServiceLocator通过调用InjectionsApplicationHandler实例化:

if (customBinder == null) {
        this.locator = Injections.createLocator(new ServerBinder(application.getProperties()), new ApplicationBinder());
    } else {
        this.locator = Injections.createLocator(new ServerBinder(application.getProperties()), new ApplicationBinder(),
                                                customBinder);
    }

注射中的代码告诉我以下内容:

 public static ServiceLocator createLocator(Binder... binders) {
    return _createLocator(null, null, binders);
 }

这意味着新创建的服务定位器具有一些任意生成的名称,并且没有父级。

有没有(干净的)方法来改变这种行为,以便我自己注入ServiceLocator作为应用程序的父类?

我知道这个答案有点晚了。 我在同一个问题上挣扎,但在Dropwizard框架中。 经过一些调试,我看到了一些令我开心的代码!

final ServiceLocator locator = (ServiceLocator) webConfig.getServletContext()
            .getAttribute(ServletProperties.SERVICE_LOCATOR);

这段代码在jerseyes WebComponent构造器中。 所以解决方案是为ServletContext提供ServletProperties.SERVICE_LOCATOR 在Dropwizard环境中,我实现了它

environment.getApplicationContext().getAttributes().setAttribute(ServletProperties.SERVICE_LOCATOR, locator);

我们有类似的设置,我已经设法使用jwells131313的新bridgeServiceLocator API来使用我们的架构。

编辑:请注意,当前的HK2桥接实现意味着Singleton实例仅对其创建的ServiceLocator是本地的,这意味着桥接体系结构可以包含多个Singleton服务的实例。 请参阅此问题以获取更多信息和可能的解决方法/替代方法。

编辑#2:修复了ServiceLocator桥中的上述错误。 修复程序将在hk2 2.5.0-b07或更高版本中

基本上我创建了一个Feature实现来设置桥接并将其注册到Jersey(在我们的实例中通过ServletContainer )。

public class InjectionBridge implements Feature
{
  private static ServiceLocator _applicationServiceLocator;

  private final ServiceLocator _serviceLocator;

  @Inject
  private InjectionBridge(ServiceLocator serviceLocator)
  {
    _serviceLocator = serviceLocator;
  }

  @Override
  public boolean configure(FeatureContext context)
  {
    if (_applicationServiceLocator != null)
      ExtrasUtilities.bridgeServiceLocator(_serviceLocator, _applicationServiceLocator);
    return true;
  }

  public static void setApplicationServiceLocator(ServiceLocator applicationServiceLocator)
  {
    _applicationServiceLocator = applicationServiceLocator;
  }
}

然后,从应用程序代码调用setApplicationServiceLocator并使用应用程序创建的ServiceLocator来管理应用程序资源。

这意味着Jersey RESTful服务实现现在可以简单地使用@Inject批注声明字段,并访问那些注入资源以服务请求。

在创建ServiceLocator之后,您无法使一个ServiceLocator成为另一个ServiceLocator的父级。

但是,从hk2 2.4.0-b11开始,将能够拥有ServiceLocator - > ServiceLocator桥。 因此,ServiceLocator的所有服务都可以放入Jersey(或任何其他ServiceLocator)的ServiceLocator中。 这是API: bridgeServiceLocator 它位于hk2的hk2-extras模块中。

在hk2 2.4.0-b10中有一个版本可以使用,完全测试和记录的功能将在hk2 2.4.0-b11

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM