Jesery-SpringServlet を使用しているジャージー アプリケーションがあります。さらに、Spring セキュリティを使用して webapp を保護しようとしています。具体的には、@Secured および @PreAuthorize アノテーションを使用して Web リソース メソッドを保護していますが、これまたはその他のスプリング セキュリティ アノテーションを適用すると、自動配線されたフィールドが自動配線されません。
私はJersey-Springサーブレット+ spring 3.1.2セキュリティでJersey 1.7を使用しています。spring-security custom-filter も使用しています。userDetailsService を実装し、必要な Spring Bean を設定して preAuthProvider に接続しました。以下のコードからわかるように、ダミーのユーザーとロールをいくつか用意しました。
すべてが正常にロードされ、すべての Bean が期待どおりに自動配線されます。ただし、以下の JerseyServiceTest クラスでは、 @PreAuthorize アノテーションが getInfo() メソッドに適用されると、 myService フィールドは null になります。@PreAuthorize アノテーションを削除すると正常に動作します。
これを解決するのを手伝ってください。以下は私のファイルです:
security-config.xml:
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:security="http://www.springframework.org/schema/security"
xsi:schemaLocation="
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.1.xsd
http://activemq.apache.org/schema/core
http://activemq.apache.org/schema/core/spring-core-3.1.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-3.1.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-3.1.xsd
http://www.springframework.org/schema/security
http://www.springframework.org/schema/security/spring-security-3.1.xsd">
<context:annotation-config/>
<context:component-scan base-package="com.*" />
<security:global-method-security
pre-post-annotations="enabled">
<security:expression-handler ref="expressionHandler" />
</security:global-method-security>
<bean id="expressionHandler" class="org.springframework.security.access.expression.method.DefaultMethodSecurityExpressionHandler">
</bean>
<security:http use-expressions="true" auto-config="true" create-session="stateless">
<security:custom-filter position="PRE_AUTH_FILTER" ref="siteminderFilter" />
<security:intercept-url pattern="/resources/batchstatus/criteria" access="ROLE_App_user"/> -->
</security:http>
<bean id="siteminderFilter" class="org.springframework.security.web.authentication.preauth.RequestHeaderAuthenticationFilter">
<property name="principalRequestHeader" value="SM_USER"/>
<property name="authenticationManager" ref="authenticationManager" />
<property name="exceptionIfHeaderMissing" value="false" />
</bean>
<bean id="preauthAuthProvider"
class="org.springframework.security.web.authentication.preauth.PreAuthenticatedAuthenticationProvider">
<property name="preAuthenticatedUserDetailsService">
<bean id="userDetailsServiceWrapper"
class="org.springframework.security.core.userdetails.UserDetailsByNameServiceWrapper">
<property name="userDetailsService" ref="userDetailsService"/>
</bean>
</property>
</bean>
<security:authentication-manager alias="authenticationManager">
<security:authentication-provider ref="preauthAuthProvider" />
</security:authentication-manager>
</beans>
私の contextInitializer クラスのスニペット:
context.setConfigLocations(new String[] {
"/WEB-INF/spring/Application-config.xml",
"/WEB-INF/spring/security-config.xml"});
}
.....
JerseyServiceTest:
...
@Component
@Path("/test")
@Component
public class JerseyServiceTest {
@Autowired
private MyService myService;
@GET
@Produces(MediaType.APPLICATION_JSON)
@PreAuthorize("hasRole('ROLE_App_user')")
public final getInfo(){
... do something
}
...
ユーザー詳細サービス:
@Service("userDetailsService")
public class UserRoleFetchService implements UserDetailsService {
GrantedAuthority ga = new GrantedAuthorityImpl("ROLE_App_user");
GrantedAuthority ga1 = new GrantedAuthorityImpl("ROLE_App_user1");
Collection<GrantedAuthority> gas = new ArrayList<GrantedAuthority>();
@Override
public UserDetails loadUserByUsername(String userName)
throws UsernameNotFoundException, DataAccessException {
gas.add(ga);
gas.add(ga1);
UserDetails user = new User(userName, "password", gas);
return user;
}
}
web.xml :
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5">
<display-name>Jersey-Test</display-name>
<welcome-file-list>
<welcome-file>index.html</welcome-file>
</welcome-file-list>
<context-param>
<param-name>contextInitializerClasses</param-name>
<param-value>mypackage.myContextInitializerClass</param-value>
</context-param>
<filter>
<filter-name>springSecurityFilterChain</filter-name>
<filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class>
</filter>
<filter-mapping>
<filter-name>springSecurityFilterChain</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<filter-mapping>
<filter-name>springSecurityFilterChain</filter-name>
<url-pattern>/resources/*</url-pattern>
</filter-mapping>
<listener>
<listener-class>org.springframework.web.util.Log4jConfigListener</listener-class>
</listener>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<servlet>
<servlet-name>jersey-servlet</servlet-name>
<servlet-class>com.sun.jersey.spi.spring.container.servlet.SpringServlet</servlet-class>
<init-param>
<param-name>com.sun.jersey.config.property.packages</param-name>
<param-value>managed</param-value>
</init-param>
<init-param>
<param-name>com.sun.jersey.api.json.POJOMappingFeature</param-name>
<param-value>true</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>jersey-servlet</servlet-name>
<url-pattern>/resources/*</url-pattern>
</servlet-mapping>
</web-app>
会社の都合であまり詳しくは書けません。