我一直在尝试使用Axios将文件(图像或东西)上传到Spring启动REST服务器,但它根本无法正常工作。除文件外,其他有效负载均为find。例如,我发送字符串,整数和图像,但服务器只接收字符串和整数。奇怪的是,当我尝试使用Postman进行操作时,一切都很顺利,包括文件。
@PostMapping
@PreAuthorize("hasAnyRole('ROLE_ADMIN', 'ROLE_MASTER')")
public ResponseEntity<ApiResponse> uploadProduct(@ModelAttribute ProductRequest productRequest) {
System.out.println("-----------------");
System.out.println("name: " + productRequest.getName());
System.out.println("price: " + productRequest.getPrice());
System.out.println("is empty: " + productRequest.getImage().isEmpty());
productRequest.getImage().forEach(o-> System.out.println(o.getOriginalFilename()));
System.out.println("-----------------");
return null;
}
这是我的测试控制器。 .getImage()始终返回true。
onSubmit() {
const data = new FormData();
data.append('name', this.name);
data.append('price', this.price);
data.append('files', this.image);
const token = localStorage.getItem('token');
const auth = {
headers: { Authorization: 'Bearer '.concat(token) },
};
axios.post('http://192.168.0.2:8080/api/products/', data, auth).then((response) => {
if (response.status === 200) {
alert('okay');
} else if (response.status === 401) {
alert('not okay');
}
}).catch((error) => {
console.log(error);
});
}
这是上面的javascript代码。
另外,我在服务器上设置过滤器,允许CORS如下所示。
public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) {
HttpServletResponse response = (HttpServletResponse) res;
HttpServletRequest request = (HttpServletRequest) req;
response.setHeader("Access-Control-Allow-Origin", "*");
response.setHeader("Access-Control-Allow-Methods", "POST, PUT, GET, OPTIONS, DELETE");
response.setHeader("Access-Control-Allow-Headers", "*");
response.setHeader("Access-Control-Max-Age", "3600");
response.setHeader("Access-Control-Allow-Credentials", "true");
if (!(request.getMethod().equalsIgnoreCase("OPTIONS"))) {
try {
chain.doFilter(req, res);
} catch(Exception e) {
e.printStackTrace();
}
} else {
System.out.println("Pre-flight");
response.setHeader("Access-Control-Allow-Methods", "POST,GET,DELETE");
response.setHeader("Access-Control-Max-Age", "3600");
response.setHeader("Access-Control-Allow-Headers", "authorization, content-type," +
"access-control-request-headers,access-control-request-method,accept,origin,authorization,x-requested-with");
response.setStatus(HttpServletResponse.SC_OK);
}
}
Vue.js dev服务器端口:8081,Spring:8080,我附上了请求标头和有效负载的屏幕截图。
感谢您的阅读。
General
Request URL: http://192.168.0.2:8080/api/products/
Request Method: POST
Status Code: 200
Remote Address: 192.168.0.2:8080
Referrer Policy: no-referrer-when-downgrade
Response Headers
Access-Control-Allow-Credentials: true
Access-Control-Allow-Headers: *
Access-Control-Allow-Methods: POST, PUT, GET, OPTIONS, DELETE
Access-Control-Allow-Origin: *
Access-Control-Max-Age: 3600
Cache-Control: no-cache, no-store, max-age=0, must-revalidate
Content-Type: application/json;charset=UTF-8
Date: Tue, 29 May 2018 15:12:43 GMT
Expires: 0
Pragma: no-cache
Transfer-Encoding: chunked
X-Content-Type-Options: nosniff
X-Frame-Options: DENY
X-XSS-Protection: 1; mode=block
Request Headers
Provisional headers are shown
Accept: application/json, text/plain, */*
Authorization: Bearer eyJhbGciOiJIUzUxMiJ9.eyJzdWIiOiIxMiIsImlhdCI6MTUyNzYwNTg1MywiZXhwIjoxNTI3NjkyMjUzfQ.GJSdT0YgpKVhai9MQN6tPRDNgmmnCPchwNiLY_XUGlwQE-aMZcUpANEIvaz6bRlLSkHYnceJ63rx1fJIdxyE_Q
Content-Type: multipart/form-data; boundary=----WebKitFormBoundaryW5gjajdcI3pBL6q7
Origin: http://localhost:8081
Referer: http://localhost:8081/menu
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/66.0.3359.181 Safari/537.36
Request Payload
------WebKitFormBoundaryW5gjajdcI3pBL6q7
Content-Disposition: form-data; name="name"
114d
------WebKitFormBoundaryW5gjajdcI3pBL6q7
Content-Disposition: form-data; name="price"
1234
------WebKitFormBoundaryW5gjajdcI3pBL6q7
Content-Disposition: form-data; name="files"
 more
并且控制台说:
log.js?4244:23 [HMR] Waiting for update signal from WDS...
Menu.vue?64be:224 eyJhbGciOiJIUzUxMiJ9.eyJzdWIiOiIxMiIsImlhdCI6MTUyNzYwNTg1MywiZXhwIjoxNTI3NjkyMjUzfQ.GJSdT0YgpKVhai9MQN6tPRDNgmmnCPchwNiLY_XUGlwQE-aMZcUpANEIvaz6bRlLSkHYnceJ63rx1fJIdxyE_Q
Menu.vue?64be:228 {headers: {…}}
Menu.vue?64be:233 [__ob__: Observer]
Menu.vue?64be:230 {data: Array(22), status: 200, statusText: "", headers: {…}, config: {…}, …}
vue.esm.js?efeb:591 [Vue warn]: Invalid prop: type check failed for prop "src". Expected String, got File.
warn @ vue.esm.js?efeb:591
assertProp @ vue.esm.js?efeb:1632
validateProp @ vue.esm.js?efeb:1560
createFunctionalComponent @ vue.esm.js?efeb:4045
createComponent @ vue.esm.js?efeb:4250
_createElement @ vue.esm.js?efeb:4420
createElement @ vue.esm.js?efeb:4357
vm._c @ vue.esm.js?efeb:4489
render @ Menu.vue?0af2:380
Vue._render @ vue.esm.js?efeb:4544
updateComponent @ vue.esm.js?efeb:2788
get @ vue.esm.js?efeb:3142
run @ vue.esm.js?efeb:3219
flushSchedulerQueue @ vue.esm.js?efeb:2981
(anonymous) @ vue.esm.js?efeb:1837
flushCallbacks @ vue.esm.js?efeb:1758
vue.esm.js?efeb:591 [Vue warn]: Invalid prop: type check failed for prop "src". Expected String, got File.
warn @ vue.esm.js?efeb:591
assertProp @ vue.esm.js?efeb:1632
validateProp @ vue.esm.js?efeb:1560
createFunctionalComponent @ vue.esm.js?efeb:4045
createComponent @ vue.esm.js?efeb:4250
_createElement @ vue.esm.js?efeb:4420
createElement @ vue.esm.js?efeb:4357
vm._c @ vue.esm.js?efeb:4489
render @ Menu.vue?0af2:659
Vue._render @ vue.esm.js?efeb:4544
updateComponent @ vue.esm.js?efeb:2788
get @ vue.esm.js?efeb:3142
run @ vue.esm.js?efeb:3219
flushSchedulerQueue @ vue.esm.js?efeb:2981
(anonymous) @ vue.esm.js?efeb:1837
flushCallbacks @ vue.esm.js?efeb:1758
vue.esm.js?efeb:591 [Vue warn]: Invalid prop: type check failed for prop "src". Expected String, got File.
warn @ vue.esm.js?efeb:591
assertProp @ vue.esm.js?efeb:1632
validateProp @ vue.esm.js?efeb:1560
createFunctionalComponent @ vue.esm.js?efeb:4045
createComponent @ vue.esm.js?efeb:4250
_createElement @ vue.esm.js?efeb:4420
createElement @ vue.esm.js?efeb:4357
vm._c @ vue.esm.js?efeb:4489
render @ Menu.vue?0af2:839
Vue._render @ vue.esm.js?efeb:4544
updateComponent @ vue.esm.js?efeb:2788
get @ vue.esm.js?efeb:3142
run @ vue.esm.js?efeb:3219
flushSchedulerQueue @ vue.esm.js?efeb:2981
(anonymous) @ vue.esm.js?efeb:1837
flushCallbacks @ vue.esm.js?efeb:1758
Menu.vue?64be:348 114d
Menu.vue?64be:354 eyJhbGciOiJIUzUxMiJ9.eyJzdWIiOiIxMiIsImlhdCI6MTUyNzYwNTg1MywiZXhwIjoxNTI3NjkyMjUzfQ.GJSdT0YgpKVhai9MQN6tPRDNgmmnCPchwNiLY_XUGlwQE-aMZcUpANEIvaz6bRlLSkHYnceJ63rx1fJIdxyE_Q
Menu.vue?64be:359 {headers: {…}}headers: Authorization: "Bearer eyJhbGciOiJIUzUxMiJ9.eyJzdWIiOiIxMiIsImlhdCI6MTUyNzYwNTg1MywiZXhwIjoxNTI3NjkyMjUzfQ.GJSdT0YgpKVhai9MQN6tPRDNgmmnCPchwNiLY_XUGlwQE-aMZcUpANEIvaz6bRlLSkHYnceJ63rx1fJIdxyE_Q"__proto__: constructor: ƒ Object()hasOwnProperty: ƒ hasOwnProperty()isPrototypeOf: ƒ isPrototypeOf()propertyIsEnumerable: ƒ propertyIsEnumerable()toLocaleString: ƒ toLocaleString()toString: ƒ toString()valueOf: ƒ valueOf()__defineGetter__: ƒ __defineGetter__()__defineSetter__: ƒ __defineSetter__()__lookupGetter__: ƒ __lookupGetter__()__lookupSetter__: ƒ __lookupSetter__()get __proto__: ƒ __proto__()set __proto__: ƒ __proto__()__proto__: Object