spring - 在不配置 CORS 的情况下将文件从 Angular 上传到 Spring
问题描述
我正在尝试使用 Angular 实现文件上传。Angular 托管在具有此代理模块配置的 Apache 服务器上:
<Directory /var/www/html/admin>
Options Indexes FollowSymLinks
AllowOverride All
Require all granted
</Directory>
<IfModule mod_proxy.c>
ProxyRequests Off
ProxyVia Off
ProxyPreserveHost On
ProxyStatus On
ProxyPass /api_admin http://localhost:8080/api_admin
ProxyPassReverse /api_admin http://localhost:8080/api_admin
ProxyPassReverseCookiePath /api_admin /
</IfModule>
我使用此配置将 Angular 应用程序托管在 Apache 的子目录下:
<IfModule mod_rewrite.c>
RewriteEngine on
RewriteBase /admin/
RewriteCond %{REQUEST_FILENAME} -s [OR]
RewriteCond %{REQUEST_FILENAME} -l [OR]
RewriteCond %{REQUEST_FILENAME} !-f [OR]
RewriteCond %{REQUEST_FILENAME} -d
RewriteRule ^.*$ - [NC,L]
RewriteRule ^(.*) index.html [NC,L]
</IfModule>
文件上传的打字稿代码:
imports (file: any) {
const url = `http://111.111.111/api_admin/merchants/upload`;
const formData = new FormData();
formData.append('file', file, file.name);
return this.http.post(url, formData);
}
弹簧端点:
@RestController
@RequestMapping("/merchants")
public class MerchantController {
@PostMapping("/upload")
public ResponseEntity<String> uploadData(@RequestParam("file") MultipartFile file) throws Exception {
.........
}
}
弹簧安全配置:
@Configuration
@EnableWebSecurity
@EnableGlobalMethodSecurity(prePostEnabled = true)
public class SecurityConfig extends WebSecurityConfigurerAdapter {
@Autowired
private DbUserDetailsService dbuserDetailsService;
@Autowired
private PasswordEncoder passwordEncoder;
@Override
@Bean
public AuthenticationManager authenticationManagerBean() throws Exception {
return super.authenticationManagerBean();
}
@Override
public UserDetailsService userDetailsServiceBean() throws Exception {
return dbuserDetailsService;
}
@Override
protected UserDetailsService userDetailsService() {
return dbuserDetailsService;
}
@Override
protected void configure(AuthenticationManagerBuilder auth) throws Exception {
auth
.userDetailsService(dbuserDetailsService)
.passwordEncoder(passwordEncoder);
}
@Autowired
public void globalUserDetails(AuthenticationManagerBuilder auth) throws Exception {
auth.userDetailsService(dbuserDetailsService)
.passwordEncoder(passwordEncoder);
}
}
路径:
@Configuration
@EnableResourceServer
public class ResourceSecurityConfig extends ResourceServerConfigurerAdapter {
@Override
public void configure(ResourceServerSecurityConfigurer resources) {
resources.resourceId("resource_id").stateless(true);
}
@Override
public void configure(HttpSecurity http) throws Exception {
http.authorizeRequests()
.antMatchers("/users/**").permitAll()
.anyRequest().authenticated();
}
}
但我收到 CORS 错误。通常我可以打开所有页面并向 BE 发出请求,但只有这个文件上传不起作用。我试过这个:
@CrossOrigin(origins = "*", methods = {RequestMethod.POST, RequestMethod.OPTIONS}, allowedHeaders = {"Content-Type", "X-Requested-With", "accept", "Origin", "Access-Control-Request-Method", "Access-Control-Request-Headers"}, exposedHeaders = {"Access-Control-Allow-Origin", "Access-Control-Allow-Credentials"})
@PostMapping("/upload")
public ResponseEntity<String> uploadData(@RequestParam("file") MultipartFile file) throws Exception {
.........
}
但是上传请求再次被 CORS 阻止。
您知道如何配置 Apache mod_rewrite 以正确重定向流量吗?
我尝试的第二种方法:
@Configuration
@EnableResourceServer
public class ResourceSecurityConfig extends ResourceServerConfigurerAdapter {
@Override
public void configure(ResourceServerSecurityConfigurer resources) {
resources.resourceId("resource_id").stateless(true);
}
@Override
public void configure(HttpSecurity http) throws Exception {
http.authorizeRequests()
.antMatchers("/users/**").permitAll()
.anyRequest().authenticated()
.and()
.cors().disable()
.authorizeRequests()
.antMatchers(HttpMethod.OPTIONS, "/**").permitAll()
.anyRequest()
.fullyAuthenticated()
.and()
.httpBasic()
.and()
.csrf().disable();
}
@Bean
public CorsConfigurationSource corsConfigurationSources() {
CorsConfiguration configuration = new CorsConfiguration();
configuration.setAllowedOrigins(Arrays.asList("*"));
configuration.setAllowedMethods(Arrays.asList("GET", "POST", "PUT", "PATCH", "DELETE", "OPTIONS"));
configuration.setAllowedHeaders(Arrays.asList("authorization", "content-type", "x-auth-token"));
configuration.setExposedHeaders(Arrays.asList("x-auth-token"));
UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource();
source.registerCorsConfiguration("/**", configuration);
return source;
}
}
使用第二种配置,我得到has been blocked by CORS policy: Response to preflight request doesn't pass access control check: It does not have HTTP ok status.
auth:1 Failed to load resource: the server responded with a status of 404 (Not Found)
实现此结果的最佳方法是什么?
有没有办法在不禁用 CORS 的情况下实现这一点?
解决方案
您已通过配置在 ResourceSecurityConfig 中禁用跨源资源共享.cors().disable()
。尝试移除disable()
零件并留下.cors()
零件。这将在应用程序的安全部分启用 CORS,并使其在预检请求中返回 OK 状态。
推荐阅读
- gstreamer - 为什么这两个 Gstreamer 管道分别可以正常工作,但从源头开球时却失败了?
- javascript - 在 code.org 上,我将如何过滤我的列表以显示每个相应的索引?
- r - 如何将长字符串值自动重命名为更易读的字符串?
- flask - 从不涉及 Flask 表单的 PUT 获取请求中获取 CSRF 错误
- discord.js - 尝试获取用户拥有的角色数量(Discord.js)
- c# - 如何编写 IAsyncEnumerable 函数以便始终执行清理代码
- node.js - 我要同时托管后端和前端吗?
- flutter - 点击时如何修改TextField的高亮颜色?(扑)
- javascript - JS:遍历 DOM 以将元素推入数组
- bash - 带有bash case语句的“选项不存在”