#java #html #angular #spring-boot #file-upload
#java #HTML #angular #весенняя загрузка #загрузка файла
Вопрос:
У меня есть такая HTML-форма для загрузки файлов из интерфейса в серверную часть и выполнения некоторых операций:
<button mat-raised-button color="primary" type="button" style='margin-right:20px' (click)="selectFile()">Select File To Upload</button>
<input #fileUploadInput type="file" id="fileUpload" hidden name="avatar" (change)="fileChangeEvent($event)">
<button mat-raised-button color="primary" type="button" style='margin-right:20px' enctype="multipart/form-data" (click)="uploadFile()">Submit</button>
<br><br>
<a class="generate-full-width" style="color: darkred;" *ngIf="fileName"><strong>{{fileName}}</strong></a>
Component.ts является:
export class uploadFileDialog {
constructor(
public dialogRef: MatDialogRef<AddProductDialog>,
private uploadService: UploadService,
private builder: FormBuilder, public dialog: MatDialog,
@Inject(MAT_DIALOG_DATA) public data) {
}
@ViewChild('fileUploadInput', {static: false})
fileUploadVariable: ElementRef;
fileName;
currentFile: File;
filesToUpload = [];
resetFile(){
this.fileUploadVariable.nativeElement.value = "";
}
selectFile(){
this.resetFile();
let el: HTMLElement = this.fileUploadVariable.nativeElement as HTMLElement;
el.click();
}
fileChangeEvent(fileInput: any) {
let file = fileInput.target.files[0]
console.log(file)
//console.log(file.data.toString());
this.filesToUpload = [];
this.filesToUpload.push(file);
this.fileName = file['name'];
}
uploadFile(){
this.currentFile = this.fileName;
console.log(this.currentFile);
this.uploadService.uploadFile(this.currentFile)
.subscribe((data) => {
console.log(data)
},
error => {
console.log(error)
});
}
}
Service.ts является:
uploadFile(file: File): Observable<any> {
let headers = new HttpHeaders({
'Access-Control-Allow-Origin': '*',
'Access-Control-Allow-Methods': 'GET, POST, PUT, DELETE, PATCH, OPTIONS',
//'Access-Control-Allow-Headers': 'X-Requested-With, content-type, Authorization'
'Access-Control-Allow-Headers': 'Content-Type,Accept,X-Access-Token,X-Key,Authorization,X-Requested-With,Origin,Access-Control-Allow-Origin,Access-Control-Allow-Credentials,content-type=multipart/*'
})
let options = {headers:headers, observer: 'response'};
const formData: FormData = new FormData();
formData.append('file', file);
//return this.http.post(this.url '/fileUpload/upload', formData,options)
const req = new HttpRequest('POST', this.url '/fileUpload/upload', formData, {
reportProgress: true,
responseType: 'json'
});
return this.http.request(req);
}
Файл контроллера в серверной части Java:
@RestController
@CrossOrigin(origins = "*", allowedHeaders="*", exposedHeaders="Access-Control-Allow-Origin")
@RequestMapping("/fileUpload")
public class FileController {
private final FileService fileService;
@Autowired
public FileController(FileService fileService) {
this.fileService = fileService;
}
@PostMapping(value = "/upload")
public void handleFileUpload(@RequestParam("file") MultipartFile file) throws IOException {
fileService.storeFile(file);
}}
и служебный файл в серверной части Java:
@Service
public class FileService {
private static final String FILE_DIRECTORY = "D:\temp";
public void storeFile(MultipartFile file) throws IOException {
Path filePath = Paths.get(FILE_DIRECTORY "" file.getOriginalFilename());
Files.copy(file.getInputStream(), filePath, StandardCopyOption.REPLACE_EXISTING);
}
}
Я могу видеть имя файла при загрузке в консоль. Кроме того, в теле запроса FormData отображает XML-файл как содержимое на вкладке Сети. Я консоль Java, я получаю сообщение об ошибке:
2020-12-15 12:26:53.144 WARN 9688 --- [nio-8080-exec-8] .w.s.m.s.DefaultHandlerExceptionResolver : Resolved [org.springframework.web.multipart.support.MissingServletRequestPartException: Required request part 'file' is not present]
Ошибка во интерфейсной консоли:
HttpHeaderResponse {headers: HttpHeaders, status: 400, statusText: "OK", url: "http://localhost:8080/fileUpload/upload", ok: false, …}
headers: HttpHeaders {normalizedNames: Map(0), lazyUpdate: null, lazyInit: ƒ}
ok: false
status: 400
statusText: "OK"
type: 2
url: "http://localhost:8080/fileUpload/upload"
__proto__: HttpResponseBase
Что я делаю не так?
Комментарии:
1. Некоторое время назад я столкнулся с аналогичной проблемой на серверной части. Тогда я использовал чистый JS во внешнем интерфейсе, но я помню, что исправил это, добавив поддержку нескольких частей для диспетчерского сервлета. Если вы используете Spring MVC, попробуйте добавить следующие составные свойства между XML-тегами сервлета в web.xml :
code
<составная конфигурация> <максимальный размер файла>10485760</максимальный размер файла> <максимальный размер запроса>20971520</максимальный размер запроса> <размер файла-threshold>5242880</file-size-threshold> </multipart-config>code
Если вы используете Spring Boot, определите эти значения в application.properties2. @Przemek на самом деле размер файла очень мал. Так понадобится ли это и в этом случае?
3. Я думаю, что при загрузке Spring он предварительно настроен, поскольку в него встроен tomcat, но я думаю, что при использовании pure Spring поддержка нескольких компонентов по умолчанию не добавляется. По крайней мере, это решение сработало в моем случае.
4. Если вы можете проверить, подходят ли вам запросы и другие части кода?
Ответ №1:
Вы отправляете только имя файла, но не фактический большой двоичный объект файла.
Попробуйте внести следующие изменения,
component.ts :
uploadFile() {
this.currentFile = this.fileName;
console.log(this.currentFile, this.filesToUpload[0]);
this.uploadService.uploadFile(this.currentFile, this.filesToUpload[0])
.subscribe((data) => {
console.log(data)
},
error => {
console.log(error)
});
}
service.ts
uploadFile(fileName: string, file: File): Observable<any> {
let headers = new HttpHeaders({
'Access-Control-Allow-Origin': '*',
'Access-Control-Allow-Methods': 'GET, POST, PUT, DELETE, PATCH, OPTIONS',
'Access-Control-Allow-Headers': 'Content-Type,Accept,X-Access-Token,X-Key,Authorization,X-Requested-With,Origin,Access-Control-Allow-Origin,Access-Control-Allow-Credentials,content-type=multipart/*'
})
let options = {headers:headers, observer: 'response'};
const formData: FormData = new FormData();
formData.append('fileName', fileName);
formData.append('file', file);
const req = new HttpRequest('POST', this.url '/fileUpload/upload', formData, {
reportProgress: true,
responseType: 'json'
});
return this.http.request(req);
}
Пожалуйста, обратитесь по этой ссылке, чтобы узнать больше о formData