java - HK2 与 Jersey 2 和 Apache Shiro 的依赖注入(inject)

标签 java dependency-injection shiro jersey-2.0 hk2

我正在使用 Jersey 2.5.1 创建一个 REST API。我使用 HK2 进行依赖注入(inject)。后来我决定使用 Apache Shiro 进行身份验证和授权。

在创建我自己的自定义 Shiro Realm 时,我遇到了一些问题。在我的领域中,我想注入(inject)一个依赖项。但是,当我运行我的应用程序时,依赖性并未解决。

这是我的设置:

web.xml

<web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
    version="3.0">

<listener>
    <listener-class>org.apache.shiro.web.env.EnvironmentLoaderListener</listener-class>
</listener>

<servlet>
    <servlet-name>MyApplication</servlet-name>
    <servlet-class>org.glassfish.jersey.servlet.ServletContainer</servlet-class>
    <init-param>
        <param-name>javax.ws.rs.Application</param-name>
        <param-value>my.app.api.MyApplication</param-value>
    </init-param>
    <load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
    <servlet-name>MyApplication</servlet-name>
    <url-pattern>/*</url-pattern>
</servlet-mapping>

<filter>
    <filter-name>ShiroFilter</filter-name>
    <filter-class>org.apache.shiro.web.servlet.ShiroFilter</filter-class>
</filter>

<filter-mapping>
    <filter-name>ShiroFilter</filter-name>
    <url-pattern>/*</url-pattern>
    <dispatcher>REQUEST</dispatcher>
    <dispatcher>FORWARD</dispatcher>
    <dispatcher>INCLUDE</dispatcher>
    <dispatcher>ERROR</dispatcher>
</filter-mapping>

shiro.ini

[main]

authcBasicRealm = my.app.api.MyCustomRealm
matcher = my.app.api.MyCustomCredentialsMatcher
authcBasicRealm.credentialsMatcher = $matcher
cacheManager = org.apache.shiro.cache.MemoryConstrainedCacheManager
securityManager.cacheManager = $cacheManager

[urls]

/** = authcBasic

MyApplication.java

public class MyApplication extends ResourceConfig {
   public MyApplication() {
      register(new ApplicationBinder());
      packages(true, "my.app.api");
   }
}

ApplicationBinder.java

public class ApplicationBinder extends AbstractBinder {
   @Override
   protected void configure() {
      bind(UserDAO.class).to(new TypeLiteral<Dao<User>>(){});
      bind(RealDatasource.class).to(DataSource.class);
   }
}

MyCustomRealm.java

public class MyCustomRealm extends JdbcRealm {

   @Inject DataSource source;

   public MyCustomRealm() {
      super();
   }

   @PostConstruct
   private void postConstruct() {
      // postConstruct is never executed
      setDataSource(source);
   }
}




所以,问题是源没有注入(inject)到 MyCustomRealm 中。所有其他不是由 Shiro 创建的类都会注入(inject)其依赖项。 问题可能是 Shiro 正在通过 ini 文件创建我的 CustomRealm 吗?

最佳答案

我遇到了类似的问题,虽然这对您来说可能不再是问题,但我想提供我使用的解决方法。

问题在于 MyCustomRealm 的所有权。它是由 shiro 在 org.apache.shiro.web.env.EnvironmentLoaderListener 中通过读取 ini 文件来创建的,该文件超出了 Jersey servlet 中 hk2 提供程序的范围。

仅当对象由 hk2 的 ServiceLocator 提供时才会完成依赖注入(inject)——shiro 不知道此定位器,仅使用其默认构造函数构造 MyCustomRealm 实例。

我通过实现一个 org.glassfish.jersey.server.spi.ContainerLifecycleListener 来解决这个问题,它获取 ServiceLocator 和 shiro 的 SecurityManager 的句柄(通过 ServletContext已向 ServiceLocator 注册)。然后它手动将数据注入(inject)到 shiro 创建的领域中。

如果您有兴趣,我可以将代码作为要点发布。

关于java - HK2 与 Jersey 2 和 Apache Shiro 的依赖注入(inject),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21661645/

相关文章:

android - 尝试使用 Dagger2 了解 Android 上的依赖注入(inject)

shiro - 身份验证后,Apache Shiro 带我返回登录页面

java - Apache Shiro,没有 GUI?

java - Java 中的服务器套接字 (Android)

JavaFX 分页,在 setPageFactory 上泄漏?

java - 如何使用maven项目作为后端?

java - 摆脱Guice依赖

c# - 为什么不将所有服务类都集中到一个工厂方法中(而不是注入(inject)接口(interface))?

java - 如何从 WEB-INF 目录加载文件/属性?

java - 如何循环和存储物理方程中的值