web-dev-qa-db-ja.com

レトロフィットを使用してサーバーにファイルをアップロードする

私のプロジェクトでは、Androidデバイスから取得した画像をサーバーサイトに送信する必要があります。サーバーサイトでは、ディスクに保存する必要があります。残念ながら、出会ったデバイスサイトでメソッドを呼び出すとこのエラー:

DEBUG/Retrofit(4429): Java.lang.RuntimeException: 
    Unable to write multipart request.
at retrofit.mime.MultipartTypedOutput.buildPart(MultipartTypedOutput.Java:86)
at retrofit.mime.MultipartTypedOutput.addPart(MultipartTypedOutput.Java:49)
at retrofit.RequestBuilder.setArguments(RequestBuilder.Java:211)
at retrofit.RestAdapter$RestHandler.invokeRequest(RestAdapter.Java:264)
at retrofit.RestAdapter$RestHandler.access$500(RestAdapter.Java:197)
at retrofit.RestAdapter$RestHandler$1.obtainResponse(RestAdapter.Java:243)
at retrofit.CallbackRunnable.run(CallbackRunnable.Java:38)
at Java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.Java:1076)
at Java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.Java:569)
at retrofit.Platform$Android$2$1.run(Platform.Java:134)
at Java.lang.Thread.run(Thread.Java:856)

Caused by: Java.io.FileNotFoundException: /external/images/media/1270: 
    open failed: ENOENT (No such file or directory)
at libcore.io.IoBridge.open(IoBridge.Java:416)
at Java.io.FileInputStream.<init>(FileInputStream.Java:78)
at retrofit.mime.TypedFile.writeTo(TypedFile.Java:74)
at retrofit.mime.MultipartTypedOutput.buildPart(MultipartTypedOutput.Java:83)
... 10 more

Caused by: libcore.io.ErrnoException: 
    open failed: ENOENT (No such file or directory)
at libcore.io.Posix.open(Native Method)
at libcore.io.BlockGuardOs.open(BlockGuardOs.Java:110)
at libcore.io.IoBridge.open(IoBridge.Java:400)
... 13 more

それがメソッドの宣言です

@Multipart
@POST("/monument/photo/upload")
void addMonumentPhoto(@Part("MonumentID") int monumentId,
                      @Part("name") String name,
                      @Part("subscript") String subscript,
                      @Part("photo") TypedFile photo,
                      Callback<Photo> callback);

...そしてそれは私がそれを呼ぶ方法です

photo = new File(selectedImageUri.getPath());
typedFile = new TypedFile("application/octet-stream", photo);
MonumentsUtil.getApi().addMonumentPhoto(monument.getIdZabytek(),
      "podpis",
      "Main photo",
      typedFile,
      new Callback<Photo>() {
    @Override
    public void success(Photo aPhoto, Response response) {
    }

    @Override
    public void failure(RetrofitError retrofitError) {
         Log.e(TAG, retrofitError.getMessage());
    }
});

サーバーのサイトには、次のような方法があります。

@RequestMapping(value="/monument/photo/upload", method=RequestMethod.POST)
public @ResponseBody Photo requestMonumentPhotoAdd(
       @RequestParam("MonumentID") int monumentId,
       @RequestParam("name") String name ,
       @RequestParam("subscript") String subscript,
       @RequestParam("photo") org.springframework.web.multipart.MultipartFile file) {

    Photo photo = new Photo();
    photo.setIdZabytek(monumentId);
    photo.setUri(URL+ "/images/" + name);
    photo.setPodpis(subscript);
    photo = monumentsRepo.addPhoto(photo);
    String filePath = "D:\\Projekty\\Images\\" + monumentId + ":" + photo.getIdZjecia();

    if (!file.isEmpty()) {
        try {
            byte[] bytes = file.getBytes();
            BufferedOutputStream stream = new BufferedOutputStream(
                new FileOutputStream(new File(filePath)));
            stream.write(bytes);
            stream.close();
            photo.setUri(filePath);
            monumentsRepo.updatePhoto(photo);
            return photo;
        } catch (Exception e) {
            return null;
        }
    } else {
        return null;
    }
}

プログラムが失敗ブロックに到達しましたが、その理由はわかりません。誰かが間違いを指摘して何が悪いのか説明できますか?

編集

コードを修正した後、修正できない次のエラーが発生しました。

Java.net.ProtocolException: content-length promised 1431984 bytes, but received 0      
    com.squareup.okhttp.internal.http.RetryableOutputStream.close(RetryableOutputStream.Java:52),
    com.squareup.okhttp.internal.http.HttpEngine.readResponse(HttpEngine.Java:629),
    com.squareup.okhttp.internal.http.HttpURLConnectionImpl.execute(HttpURLConnectionImpl.Java: 346),
    com.squareup.okhttp.internal.http.HttpURLConnectionImpl.getResponse(HttpURLConnectionImpl.Java:295),
    com.squareup.okhttp.internal.http.HttpURLConnectionImpl.getResponseCode(HttpURLConnectionImpl.Java:489), 
retrofit.client.UrlConnectionClient.readResponse(UrlConnectionClient.Java:90), 
retrofit.client.UrlConnectionClient.execute(UrlConnectionClient.Java:48), 
retrofit.RestAdapter$RestHandler.invokeRequest(RestAdapter.Java:287), 
retrofit.RestAdapter$RestHandler.access$500(RestAdapter.Java:197), 
retrofit.RestAdapter$RestHandler$1.obtainResponse(RestAdapter.Java:243), 
retrofit.CallbackRunnable.run(CallbackRunnable.Java:38), 
Java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.Java:1076), 
Java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.Java:569), 
retrofit.Platform$Android$2$1.run(Platform.Java:134), 
Java.lang.Thread.run(Thread.Java:856)

およびHTTP応答

HTTP/1.1 400 Bad Request, 
Connection: close, 
Content-Length: 1068, 
Content-Type:     text/html;charset=utf-8, 
Date: Wed, 11 Dec 2013 16:45:39 GMT,  
OkHttp-Received-Millis: 1386780339873, 
OkHttp-Response-Source: NETWORK 400,
OkHttp-Selected-Transport: http/1.1, 
OkHttp-Sent-Millis: 1386780339799, Server: Apache-Coyote/1.1

サーバーのメソッドが間違って書かれている可能性があると思いますが、よくわかりません。誰かがそれを手伝ってくれる?

14

Uri#getPathは、ファイルシステム上のパスではなく、Uriのパスコンポーネントを返します。

実際の画像については、コンテンツプロバイダーに問い合わせる必要があります。このようなもの: https://stackoverflow.com/a/20186918/132047

9
Jake Wharton