web-dev-qa-db-ja.com

Spring RESTfulサービスでファイルとJSONオブジェクトで構成されるマルチパートリクエストを処理するにはどうすればよいですか?

ファイルとJSONオブジェクトを受け入れる次のリソース(Spring 4.05.RELEASEを使用して実装)があります。

(P.S. activityTemplateはシリアル化可能なエンティティクラスです)

...
@RequestMapping(value="/create", method=RequestMethod.POST)
public @ResponseBody ActivityTemplate createActivityTemplate(
        @RequestPart ActivityTemplate activityTemplate, @RequestPart MultipartFile jarFile)
{
   //process the file and JSON
}
...

これは私がテストしているフォームです:

<form method="POST" enctype="multipart/form-data"
    action="http://localhost:8080/activityTemplates/create">
    JSON: <input type="text" name="activityTemplate" value='/* the JSON object*/'><br />

    File to upload: <input type="file" name="file">
    <input type="submit" value="Upload">
</form>

これは私が得るエラーです:

 There was an unexpected error (type=Unsupported Media Type, status=415).
 Content type 'application/octet-stream' not supported

では、マルチパートリクエストの一部としてJSONオブジェクトをリソースに許可する方法、または別の方法でフォームを送信する方法を教えてください。

19
Sami

これは私のために働くのに2日かかりました!

クライアント(角度):

$scope.saveForm = function () {
      var formData = new FormData();
      var file = $scope.myFile;
      var json = $scope.myJson;
      formData.append("file", file);
      formData.append("ad",JSON.stringify(json));//important: convert to string JSON!
      var req = {
        url: '/upload',
        method: 'POST',
        headers: {'Content-Type': undefined},
        data: formData,
        transformRequest: function (data, headersGetterFunction) {
          return data;
        }
      };

春(ブート):

@RequestMapping(value = "/upload", method = RequestMethod.POST)
    public @ResponseBody
    Advertisement storeAd(@RequestPart("ad") String adString, @RequestPart("file") MultipartFile file) throws IOException {

        Advertisement jsonAd = new ObjectMapper().readValue(adString, Advertisement.class);
//do whatever you want with your file and jsonAd
25
mohi

これがあなたを助けることを願っています。 HTTPリクエストに通知するには、リクエストに境界を設定する必要があります。簡単です; multipart formatの簡単な紹介は以下のリンクにあります

マルチパートのHTML 4.01仕様

次の例は、「multipart/form-data」エンコーディングを示しています。 Json Objectが "MyJsonObj"で、送信する必要があるファイルが "myfileである場合.txt "、ユーザーエージェントは次のデータを送り返す可能性があります。

Content-Type: multipart/form-data; boundary=MyBoundary

--MyBoundary
Content-Disposition: form-data; name="myJsonString"
Content-Type: application/json

MyJsonObj //Your Json Object goes here
--MyBoundary
Content-Disposition: form-data; name="files"; filename="myfile.txt"
Content-Type: text/plain

... contents of myfile.txt ...
--MyBoundary--

または、ファイルが「image.gif」という名前の画像タイプの場合、

--MyBoundary
Content-Disposition: file; filename="image.gif"
Content-Type: image/gif
Content-Transfer-Encoding: binary

...contents of image.gif...
--MyBoundary--

boundaryContent-Type headerに指定して、送信されたデータの分割方法をサーバーが認識できるようにします。 。

そのため、基本的に境界値を選択して以下を行う必要があります。

  • 'AaB03x'など、サーバーに送信されるHTTPデータには表示されない値を使用します。
  • 一貫性を保ち、リクエスト全体で同じ値を使用します。
2
Sridhar DD

@ RequestPart sにパラメーター名を指定していませんか?

public @ResponseBody ActivityTemplate createActivityTemplate(
    @RequestPart("activityTemplate") ActivityTemplate activityTemplate, @RequestPart("file") MultipartFile jarFile)
{
   //process the file and JSON
}

注:クラスパスにjackson mapper .jar(jsonをActivityTemplate)ファイルに含めることを忘れないでください。

1
codeMan

変えられませんか

@RequestMapping(value="/create", method=RequestMethod.POST)

@RequestMapping(value="/create",
                method=RequestMethod.POST, consumes ={"multipart/form-data"})
1
Vogel

Org.springframework.web.bind.annotation.RequestPartから@RequestPartを使用できます。 @RequestBodyとファイルアップロードの組み合わせとして使用されます。

@RequestParam( "file")MultipartFileファイルのような@RequestParamを使用すると、ファイルと複数の単一データ(キー値)のみをアップロードできます。

        @RequestMapping(value = "/uploadFile", method = RequestMethod.POST,  consumes = { MediaType.MULTIPART_FORM_DATA_VALUE }, produces = { MediaType.APPLICATION_JSON_VALUE })
    public void saveFile(
                         @RequestParam("userid") String userid,
                         @RequestParam("file") MultipartFile file) {

    }

jSONオブジェクトデータとファイルの両方を@RequestPartを使用して投稿できます

    @RequestMapping(value = "/patientp", method = RequestMethod.POST,  consumes = { MediaType.MULTIPART_FORM_DATA_VALUE }, produces = { MediaType.APPLICATION_JSON_VALUE })
public ResponseEntity<?> insertPatientInfo(
                                            @RequestPart PatientInfoDTO patientInfoDTO,
                                            @RequestPart("file") MultipartFile file) {
}

コントローラーメソッドのパラメーターとしてマルチパートファイルのアップロードを直接使用することに限定されません。フォームオブジェクトにはPartフィールドまたはMultipartFileフィールドを含めることができ、Springはファイルパーツから値を取得する必要があることを自動的に認識し、値を適切に変換します。

上記の方法は、単一のファイルを含む前述のマルチパートリクエストに応答できます。これは、Springにファイルパーツを認識するHTTPメッセージコンバーターが組み込まれているためです。 javax.servlet.http.Partタイプに加えて、ファイルのアップロードをorg.springframework.web.multipart.MultipartFileに変換することもできます。 2番目のマルチパートリクエストで示されているように、ファイルフィールドで複数のファイルのアップロードが許可されている場合は、単純に配列またはパーツのコレクションまたはMultipartFilesを使用します。

        @RequestMapping(value = "/patientp", method = RequestMethod.POST,  consumes = { MediaType.MULTIPART_FORM_DATA_VALUE }, produces = { MediaType.APPLICATION_JSON_VALUE })
    public ResponseEntity<?> insertPatientInfo(
                                                @RequestPart PatientInfoDTO patientInfoDTO,
                                                @RequestPart("files") List<MultipartFile> files) {
    }

お力になれて、嬉しいです...

0
Pramod Wayabase

Multipart/form-dataリクエストを処理するための適切なHttpMessageConverterがないため、例外がスローされます。 回避策

0
Maksim

これは役立つ場合があります。MultipartFileを受信して​​いる間、リクエストヘッダーのcontent-typeを "multipart/form-data"に設定し、コントローラーでconsumers = "multipart/form-data"を使用する必要があります。コントローラー内。

JSONデータを受信したい場合は、JSONStringの形式でリクエストを送信し、そのjsonstringを受信し、後でjson Object形式に変換してから、そのオブジェクトを操作に使用します。

以下のコードを確認してください:

@RequestMapping(value="/savingImg", method=RequestMethod.POST, 
        consumes="multipart/form-data", produces="application/json")
public ResponseEntity<?> savingAppIMgDtlss(
        @RequestParam(value="f1", required = false) MultipartFile f1 , 
        @RequestParam(value="f2", required = false) MultipartFile f2 ,
        @RequestParam(value="f3", required = false) MultipartFile f3 ,
        @RequestParam(value="f4", required = false) MultipartFile f4 ,
        @RequestParam(value="f5", required = false) MultipartFile f5 ,
        @RequestParam(value="f6", required = false) MultipartFile f6 ,
        @RequestParam(value="f7", required = false) MultipartFile f7 ,
        @RequestParam(value="f8", required = false) MultipartFile f8 ,@RequestParam("data") String jsonString) 
                throws Exception , ParseException {
    try{
        JSONObject gstcTranObj = new JSONObject();
                //converting JSONString to JSON
        net.sf.json.JSONObject jsonDtls = net.sf.json.JSONObject.fromObject(jsonString);
        System.out.println("f1::"+f1.getOriginalFilename());
        System.out.println("f2::"+f2.getOriginalFilename());
        System.out.println("f3::"+f3.getOriginalFilename());
        System.out.println("f4::"+f4.getOriginalFilename());
        System.out.println("f5::"+f5.getOriginalFilename());
        System.out.println("f6::"+f6.getOriginalFilename());
        System.out.println("f7::"+f7.getOriginalFilename());
        System.out.println("f8::"+f8.getOriginalFilename());
} catch (Exception e) {
        e.printStackTrace();

        return new ResponseEntity<>("Failed",HttpStatus.NOT_FOUND);
    }finally{

    }
return new ResponseEntity<>("Success", HttpStatus.OK);

  }
}

エラーメッセージは、コンテンツタイプapplication/octet-streamのマルチパート/ MIMEパートに登録されているHttpMessageConverterがないことを示しています。それでも、jarFileパラメーターはapplication/octet-streamとして正しく識別される可能性が最も高いため、パラメーターマッピングに不一致があると想定しています。

そのため、最初にパラメーターとフォームの入力要素に同じ名前を設定してみてください。

もう1つの問題は、JSONがフォームのテキスト入力の(通常の)値としてアップロードされ、マルチパート/ MIMEの個別のパートとしてアップロードされないことです。したがって、SpringがJSONデシリアライザーを使用する必要があることを確認するために関連付けられたコンテンツタイプヘッダーはありません。代わりに@RequestParamを使用して、この回答のように特定のコンバーターを登録できます。 spring MVCコントローラーのJSONパラメーター

0
miw

デフォルトのコンテンツタイプは「application/octet-stream」です。 jarファイルとJSONをアップロードしているため、コンテンツタイプは@RequestMapping注釈は次のとおりです。

@RequestMapping(value="/create", method=RequestMethod.POST, headers="content-type=application/json,application/Java-archive")
0
Mithun