Загрузка файла в Angular выдает ошибку при отправке POST-запроса

#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.properties

2. @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