gpt4 book ai didi

java - 使用 Angular JS 在 Spring Security 中登录过程后缺少 CSFR_TOKEN session 属性

转载 作者:行者123 更新时间:2023-11-30 10:46:07 25 4
gpt4 key购买 nike

我对使用 CSRF 的 AngularJS 的 Spring Security 有疑问。

我的实现基于此文档:

http://spring.io/guides/tutorials/spring-security-and-angular-js/

我的问题是未正确管理登录/注销过程。

登录阶段似乎进展顺利:响应正常,Java session 已创建,并且有一个属性“SPRING_SECURITY_CONTEXT”,其中记录了 Principal。但是在登录过程之后,我注意到在 session 对象中缺少 CSFR_TOKEN 的属性

这导致了这样的效果:当我尝试注销时,我传递了带有 Spring Security 所需的所有 header 的请求。但是 org.springframework.security.web.csrf.CsrfFilter 类的“doFilterInternal”方法无法检索属性:

org.springframework.security.web.csrf.HttpSessionCsrfTokenRepository.CSRF_TOKEN

这会导致 token 重新生成,因为“ token 丢失”(在 session 对象中,而不是在请求 header 中),所以当“doFilterInternal”进行匹配控制时使用请求中传递的 token ,匹配失败并在 Spring 日志中打印:“为...找到无效的 CSRF token ”

这个问题停止了过滤器链:在自定义“csrfHeaderFilter”中创建的过滤器没有被调用,因为它是在标准过滤器之后调用的,所以我有这个错误页面作为返回:

HTTP 状态 403 - 未找到预期的 CSRF token 。您的 session 是否已过期?

session 未过期。事实上,如果我在浏览器上刷新页面,我可以看到我仍然处于登录状态。在日志中我可以看到用户名仍然存在,并且在 session 中完美保存了 Principal 对象。

如果我在页面刷新之后重试注销,注销不会再失败,因为在 session 中现在存在属性"CSRF_TOKEN"。并且注销过程得到了完美的管理,因为在那之后我在请求中有一个空 session 。

这有什么问题吗?

这是我的实际安全配置:

@Configuration
@EnableWebSecurity
@EnableGlobalMethodSecurity(prePostEnabled = true, securedEnabled = true, proxyTargetClass = true)
public class SecurityContextConfig extends WebSecurityConfigurerAdapter{

@Resource(name = "myUserDetailService")
private UserDetailsService userDetailsService;

@Bean
public static StandardPasswordEncoder passwordEncoder() throws Exception {
return new StandardPasswordEncoder();
}

@Override
protected void configure(AuthenticationManagerBuilder auth) throws Exception {
DaoAuthenticationProvider provider = new DaoAuthenticationProvider();
provider.setPasswordEncoder(passwordEncoder());
provider.setUserDetailsService(userDetailsService);
auth.authenticationProvider(provider);
}

@Override
protected void configure(HttpSecurity http) throws Exception {
http
.authorizeRequests()
.antMatchers("/main.html","/pages/**").permitAll()
.anyRequest().authenticated().and()
.httpBasic()
.and()
.csrf()
.csrfTokenRepository(csrfTokenRepository()).and()
.addFilterAfter(csrfHeaderFilter(), CsrfFilter.class)
.logout()
.logoutRequestMatcher(new AntPathRequestMatcher("/logout")).permitAll()
.logoutSuccessUrl("/main.html");
}

private Filter csrfHeaderFilter() {
return new OncePerRequestFilter() {
@Override
protected void doFilterInternal(HttpServletRequest request,
HttpServletResponse response, FilterChain filterChain)
throws ServletException, IOException {
CsrfToken csrf = (CsrfToken) request.getAttribute(CsrfToken.class
.getName());
if (csrf != null) {
Cookie cookie = WebUtils.getCookie(request, "XSRF-TOKEN");
String token = csrf.getToken();
if (cookie == null || token != null
&& !token.equals(cookie.getValue())) {
cookie = new Cookie("XSRF-TOKEN", token);
cookie.setPath("/");
response.addCookie(cookie);
}
}
filterChain.doFilter(request, response);
}
};
}

private CsrfTokenRepository csrfTokenRepository() {
HttpSessionCsrfTokenRepository repository = new HttpSessionCsrfTokenRepository();
repository.setHeaderName("X-XSRF-TOKEN");
return repository;
}

@Override
public void configure(WebSecurity web) throws Exception {
web
.ignoring()
.antMatchers("/resources/**");
}

@Bean(name="authenticationManager")
@Override
public AuthenticationManager authenticationManagerBean() throws Exception {
return super.authenticationManagerBean();
}

@Override
@Autowired
public void setObjectPostProcessor(ObjectPostProcessor<Object> objectPostProcessor) {
super.setObjectPostProcessor(objectPostProcessor);
}
}

这些是第一次注销尝试(不刷新)时的请求 header

HTTP Status 403 - Expected CSRF token not found. Has your session expired?

Request URL:https://localhost:8080/myApp/logout
Request Method:POST
Status Code:403 Forbidden
Remote Address:[::1]:8080

Request Headers
view source
Accept:application/json, text/plain, */*
Accept-Encoding:gzip, deflate
Accept-Language:it-IT,it;q=0.8,en-US;q=0.6,en;q=0.4
Connection:keep-alive
Content-Length:2
Content-Type:application/json;charset=UTF-8
Cookie:JSESSIONID=DFE1A9492F421EDBAEC0DAE6726BFDC4; XSRF-TOKEN=70e2a706-db6d-4f53-b39f-01f6f10b6af1
Host:localhost:8080
Origin:https://localhost:8080
Referer:https://localhost:8080/myApp/main.html
User-Agent:Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/49.0.2623.112 Safari/537.36
X-Requested-With:XMLHttpRequest
X-XSRF-TOKEN:70e2a706-db6d-4f53-b39f-01f6f10b6af1

Response Headers
view source
Cache-Control:no-cache, no-store, max-age=0, must-revalidate
Content-Language:en
Content-Length:1116
Content-Type:text/html;charset=utf-8
Date:Tue, 19 Apr 2016 12:51:09 GMT
Expires:0
Pragma:no-cache
Server:Apache-Coyote/1.1
Strict-Transport-Security:max-age=31536000 ; includeSubDomains
X-Content-Type-Options:nosniff
X-Frame-Options:DENY
X-XSS-Protection:1; mode=block

最佳答案

我自己解决的。

问题出在客户端,我在登录后省略了对 url 'user/' 的 http 访问。如果没有这个调用,CSRF_TOKEN 不会在 session 中重新保存,导致服务器端在注销过程中出现问题。

此修复后,行为正确且运行良好。

因此,按照 Spring-AngularJS 教程,调用 url 'user',如果 esit OK,然后调用 url 'user/'。

关于java - 使用 Angular JS 在 Spring Security 中登录过程后缺少 CSFR_TOKEN session 属性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36722657/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com