我正在尝试使用Angular2和SpringBoot完成文件上传功能。我可以证明我的文件上传的java代码工作正常,因为我已经使用Postman成功测试了它。
但是,当从Angular2前端发送文件时,我收到的是HTTP 400响应Required request part 'file' is not present
。
这是我从Angular2发送POST请求的方式。
savePhoto(photoToSave: File) {
let formData: FormData = new FormData();
formData.append('file', photoToSave);
// this will be used to add headers to the requests conditionally later using Custom Request Options
this._globals.setRequestFrom("save-photo");
let savedPath = this._http
.post(this._endpointUrl + "save-photo", formData)
.map(
res => {
return res.json();
}
)
.catch(handleError);
return savedPath;
}
请注意,我编写了一个CustomRequestOptions
类,其扩展BaseRequestOptions
以附加Authorization标头和Content Type标头。内容类型标题将有条件地添加。
以下是该代码。
@Injectable()
export class CustomRequestOptions extends BaseRequestOptions {
constructor(private _globals: Globals) {
super();
this.headers.set('X-Requested-By', 'Angular 2');
this.headers.append('virglk', "vigamage");
}
merge(options?: RequestOptionsArgs): RequestOptions {
var newOptions = super.merge(options);
let hdr = this._globals.getAuthorization();
newOptions.headers.set("Authorization", hdr);
if(this._globals.getRequestFrom() != "save-photo"){
newOptions.headers.set('Content-Type', 'application/json');
}else{
//request coming from save photo
console.log("request coming from save photo");
}
return newOptions;
}
}
此条件标头追加工作正常。这样做的目的是,如果我为每个请求添加'Content-Type', 'application/json'
标头,Spring控制器中的文件上载方法将不接受它。 (返回http 415)
一切似乎都很好。但我得到Required request part 'file' is not present
错误响应。这是为什么?我将该参数添加到数据表单。
let formData: FormData = new FormData();
formData.append('file', photoToSave);
这是Spring Controller方法供您参考。
@RequestMapping(method = RequestMethod.POST, value = "/tender/save-new/save-photo", consumes = {"multipart/form-data"})
public ResponseEntity<?> uploadPhoto(@RequestParam("file") MultipartFile file){
if (file.isEmpty()) {
ErrorResponse errorResponse = new ErrorResponse();
errorResponse.setMessage("DEBUG: Attached file is empty");
return new ResponseEntity<ErrorResponse>(errorResponse, HttpStatus.NOT_FOUND);
}
String returnPath = null;
try {
// upload stuff
} catch (IOException e) {
ErrorResponse errorResponse = new ErrorResponse();
errorResponse.setMessage(e.getMessage());
return new ResponseEntity<ErrorResponse> (errorResponse, HttpStatus.INTERNAL_SERVER_ERROR);
}
return new ResponseEntity<String>(returnPath, HttpStatus.OK);
}
编辑 - 添加浏览器捕获的请求的有效负载
如您所见,那里有param“file”。
答案 0 :(得分:0)
尝试添加
headers: {
'Content-Type': 'multipart/form-data'
},
到你的
.post(this._endpointUrl + "save-photo", formData)
答案 1 :(得分:0)
更改formData.append('file', photoToSave);
formData.append('file', this.photoToSave, this.photoToSave.name);
并添加标头,指定您传递给 API 的数据类型,在您的情况下,它将是
'Content-Type': 'multipart/form-data'
。如果更改后仍然失败,请在此处发布输出。
答案 2 :(得分:0)
您是否有可能在转发请求的辅助应用中使用 zuul?我在一个更新中看到了这一点,其中在转发多部分上传时去除了标题。我有一个看门人应用程序,它通过从尤里卡的观察使用 zuul 将请求转发到实际服务。我通过像这样修改 url 来修复它:
http://myserver.com/service/upload
到
http://myserver.com/zuul/service/upload
突然,上传标头的“文件”部分不再被剥离和丢弃。
原因,我怀疑是缓存请求的重试机制。失败时,它会重新提交请求,但不知何故对于文件上传,它无法正常工作。