spring Angular客户端未使用JSessionID设置Cookie

nukf8bse  于 2023-04-10  发布在  Spring
关注(0)|答案(1)|浏览(193)

我使用的是一个Angular 前端和一个带有Spring Security 的spring后端。

  1. http
  2. .cors().and()
  3. // .csrf().disable()
  4. .csrf().csrfTokenRepository(CookieCsrfTokenRepository.withHttpOnlyFalse()).and()
  5. .authorizeRequests()
  6. // .antMatchers(HttpMethod.OPTIONS, "/**").permitAll()
  7. .antMatchers("/xrtool/*").permitAll()
  8. .antMatchers("/advanced_search").hasRole("USER")
  9. .antMatchers("/creation").hasRole("USER")
  10. .antMatchers("/edit").hasRole("USER")
  11. .antMatchers("/basicAuth").permitAll()
  12. .antMatchers("/**").permitAll()
  13. .anyRequest().authenticated().and()
  14. .httpBasic();

我在访问没有保护(permitAll)的API时没有问题,但是那些需要特定角色的API给予了我一个未经授权的状态401。我相信这是由于我的客户端在初始身份验证请求后没有自动设置cookie。
我的授权请求代码:

  1. authenticate(username : string, password : string){
  2. let authHeader = new HttpHeaders({
  3. authorization : 'Basic ' + btoa(username + ':' + password),
  4. withCredentials: 'true'
  5. });
  6. return this.http.get(`${API_URL}/basicAuth`, {headers: authHeader})
  7. }

我的后台代码:

  1. @Override
  2. protected void configure(AuthenticationManagerBuilder auth) throws Exception {
  3. auth.userDetailsService(userService);
  4. }
  5. @Override
  6. protected void configure(HttpSecurity http) {
  7. try {
  8. http
  9. .cors().and()
  10. .csrf().csrfTokenRepository(CookieCsrfTokenRepository.withHttpOnlyFalse()).and()
  11. .authorizeRequests()
  12. .antMatchers("/xrtool/*").permitAll()
  13. .antMatchers("/advanced_search").hasRole("USER")
  14. .antMatchers("/creation").hasRole("USER")
  15. .antMatchers("/edit").hasRole("USER")
  16. .antMatchers("/basicAuth").permitAll()
  17. .antMatchers("/**").permitAll()
  18. .anyRequest().authenticated().and()
  19. .httpBasic();
  20. } catch (Exception e){
  21. System.out.println(e);
  22. }
  23. }
  24. @Bean
  25. CorsConfigurationSource corsConfigurationSource() {
  26. CorsConfiguration configuration = new CorsConfiguration();
  27. configuration.setAllowedOrigins(Arrays.asList("http://localhost:4200"));
  28. configuration.setAllowedMethods(Arrays.asList("GET","POST"));
  29. configuration.setAllowCredentials(true);
  30. configuration.setAllowedHeaders(Arrays.asList("*"));
  31. UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource();
  32. source.registerCorsConfiguration("/**", configuration);
  33. return source;
  34. }

我的初始请求的响应头:

  1. HTTP/1.1 200
  2. Vary: Origin
  3. Vary: Access-Control-Request-Method
  4. Vary: Access-Control-Request-Headers
  5. Access-Control-Allow-Origin: http://localhost:4200
  6. Access-Control-Allow-Credentials: true
  7. Set-Cookie: XSRF-TOKEN=475b11b6-39a4-4a46-a58d-e096a3bc07c6; Path=/
  8. Set-Cookie: JSESSIONID=3E8382E29CB7C0AD9BBC168EA08ACF30; Path=/; HttpOnly
  9. X-Content-Type-Options: nosniff
  10. X-XSS-Protection: 1; mode=block
  11. Cache-Control: no-cache, no-store, max-age=0, must-revalidate
  12. Pragma: no-cache
  13. Expires: 0
  14. X-Frame-Options: DENY
  15. Content-Type: application/json
  16. Transfer-Encoding: chunked
  17. Date: Sat, 21 Jan 2023 17:56:43 GMT
  18. Keep-Alive: timeout=60
  19. Connection: keep-alive

以及后续的请求头:

  1. GET /xrtool/Test HTTP/1.1
  2. Host: localhost:8080
  3. User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:109.0) Gecko/20100101 Firefox/109.0
  4. Accept: application/json, text/plain, */*
  5. Accept-Language: en-US,en;q=0.5
  6. Accept-Encoding: gzip, deflate, br
  7. withCredentials: true
  8. Origin: http://localhost:4200
  9. DNT: 1
  10. Connection: keep-alive
  11. Referer: http://localhost:4200/
  12. Sec-Fetch-Dest: empty
  13. Sec-Fetch-Mode: cors
  14. Sec-Fetch-Site: same-site

网络客户端的存储也不包含Cookie。
我已经用不同的浏览器(Chrome,Firefox,Edge)试过了,但结果是一致的。
我试着直接访问API(localhost:8080/api/request),一切正常。我还可以看到webclient存储中的cookie。
Postman也工作得很好。我可以看到cookie被保存在标头中,在初始授权请求之后,后续请求被授权。
我尝试通过拦截器在每个请求中添加授权头,并且我能够访问API。
编辑:
感谢Tr 1 Monsters的观察,我能够找到这个问题。withCredentials头在错误的位置。似乎它不需要在初始身份验证请求中,但每个后续请求。我已经将它添加到我的Http-Intercept中,如下所示:

  1. intercept(request: HttpRequest<any>, next: HttpHandler): Observable<HttpEvent<any>> {
  2. let username = "b"
  3. let password = "b"
  4. let basicAuthHeaderString = "Basic " + window.btoa(username + ':' + password)
  5. request = request.clone({
  6. setHeaders: {
  7. // Authorization: basicAuthHeaderString
  8. }
  9. , withCredentials: true
  10. })
  11. return next.handle(request);
  12. }

我保留了原来的方式,添加了头,以显示withCredentials头被放置在setHeaders部分之外。

iibxawm4

iibxawm41#

感谢Tr 1 Monsters的观察,我能够找到这个问题。withCredentials头在错误的位置。似乎它不需要在初始身份验证请求中,但每个后续请求。我已经像这样将它添加到我的Http-Interceptor:

  1. intercept(request: HttpRequest<any>, next: HttpHandler): Observable<HttpEvent<any>> {
  2. let username = "b"
  3. let password = "b"
  4. let basicAuthHeaderString = "Basic " + window.btoa(username + ':' + password)
  5. request = request.clone({
  6. setHeaders: {
  7. // Authorization: basicAuthHeaderString
  8. }
  9. , withCredentials: true
  10. })
  11. return next.handle(request);
  12. }

我保留了添加头的原始方式,以显示withCredentials头放置在setHeaders部分之外。

展开查看全部

相关问题