A Day In The Life

とあるプログラマの備忘録

Android でライブラリの追加無しでマルチパート POST する方法

Android SDK には Apache Commons の HttpClient など HTTP 通信するためのライブラリがあらかじめ搭載されているのですがマルチパート POST するためのライブラリがありません。Apache のライブラリをプロジェクトに自前で追加してもいいのですが必要のない物まで入ってしまってアプリの容量がムダに増えてしまうので自分で作ってみました。
マルチパート POST って普段あまり使わないのですが画像データとコメントをサーバに一緒に送信したい時なんかにあると便利です。
作成するにあたってこちらのサイトのソースコードを参考にしました。

import java.io.BufferedReader;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.net.HttpURLConnection;
import java.net.URL;
import java.net.URLConnection;
import java.util.List;
import org.apache.http.NameValuePair;
import android.util.Log;

public class HttpMultipartRequest {
  private static final String BOUNDARY = "----------V2ymHFg03ehbqgZCaKO6jy";
  private String url;
  private List<NameValuePair> postData;
  private String imageName;
  private String imagePath;
	
  public HttpMultipartRequest(
      String url, List<NameValuePair> postData,
      String imageName, String imagePath) {
    this.url = url;
    this.postData = postData;
    this.imageName = imageName;
    this.imagePath = imagePath;
  }

  public String send() {
    URLConnection conn = null;
    String res = null;
    try {
      conn = new URL(url).openConnection();
      conn.setRequestProperty("Content-Type", "multipart/form-data; boundary=" + BOUNDARY);
      // User Agentの設定はAndroid1.6の場合のみ必要
      conn.setRequestProperty("User-Agent", "Android");
      // HTTP POSTのための設定
      ((HttpURLConnection)conn).setRequestMethod("POST");
      conn.setDoOutput(true);
      // HTTP接続開始
      conn.connect();
      // send post data
      File file =new File(imagePath);
      OutputStream os = conn.getOutputStream();
      os.write(createBoundaryMessage(file.getName()).getBytes());
      os.write(getImageBytes(file));
      String endBoundary = "\r\n--" + BOUNDARY + "--\r\n";
      os.write(endBoundary.getBytes());
      os.close();
      // get response
      InputStream is = conn.getInputStream();
      res = convertToString(is);
    } catch(Exception e) {
      Log.d("HttpMultipartRequest:", e.getMessage());
    } finally {
      if (conn != null ) 
        ((HttpURLConnection)conn).disconnect();
    }
    return res;
  }

  private String createBoundaryMessage(String fileName) {
    StringBuffer res = new StringBuffer("--").append(BOUNDARY).append("\r\n");
    for (NameValuePair nv : postData) {
      res.append("Content-Disposition: form-data; name=\"").append(nv.getName()).append("\"\r\n")    
        .append("\r\n").append(nv.getValue()).append("\r\n")
        .append("--").append(BOUNDARY).append("\r\n");
    }
    String[] fileChunks = fileName.split("\\.");
    String fileType = "image/" + fileChunks[fileChunks.length - 1];
    res.append("Content-Disposition: form-data; name=\"")
      .append(imageName)
      .append("\"; filename=\"").append(fileName).append("\"\r\n") 
      .append("Content-Type: ").append(fileType).append("\r\n\r\n");
    return res.toString();
  }

  private byte[] getImageBytes(File file) {
    byte[] b = new byte[10];
    FileInputStream fis = null;
    ByteArrayOutputStream bo = new ByteArrayOutputStream();
    try {
      fis = new FileInputStream(file);
      while(fis.read(b) > 0) {
        bo.write(b);
      }
    } catch (FileNotFoundException e) {
      Log.d("HttpMultipartRequest:", e.getMessage());
    } catch (IOException e) {
      Log.d("HttpMultipartRequest:", e.getMessage());
    } finally {
      try { bo.close(); } catch (IOException e) {}
      if (fis != null) try { fis.close(); } catch (IOException e) {}
    }
    return bo.toByteArray();
  }

  private String convertToString(InputStream stream) {
    InputStreamReader streamReader = null;
    BufferedReader bufferReader = null;
    try {
      streamReader = new InputStreamReader(stream, "UTF-8");
      bufferReader = new BufferedReader(streamReader);
      StringBuilder builder = new StringBuilder();
      for (String line = null; (line = bufferReader.readLine()) != null;) {
        builder.append(line).append("\n");
      }
      return builder.toString();
    } catch(UnsupportedEncodingException e) {
      Log.e("HttpMultipartRequest:", e.getMessage());
    } catch(IOException e) {
      Log.e("HttpMultipartRequest:", e.toString());
    } finally {
      try {
        stream.close();
        if (bufferReader != null) bufferReader.close();
      } catch(IOException e) {
        //
      }
    }
    return null;
  }
}

使い方

以下の HTML に相当する処理をするためには

<form method="post" action="http://www.hogehoge.com" enctype="multipart/form-data">
<p>コメント:<input type="text" name="comment" size="200"></p>
<p>ファイル:<input type="file" name="upload-file" size="100"></p>
<p><input type="submit" value="画像を投稿"></p>
</form>

POST 先のアドレスと通常の POST データに加えデータファイルの名前とデータのファイルパスを引数に渡してやります。

List<NameValuePair> postData = new ArrayList<NameValuePair>();
NameValuePair nv = new NameValuePair("comment", "この写真はxxで撮りました。");
postData.add(nv);
HttpMultipartRequest request = new HttpMultipartRequest(
    "http://www.hogehoge.com",
    postData,
    "upload-file",
    "file:///hoge/hoge.jpeg");
String response = request.send();
:
: response に応じた処理
:

参考

マルチパート POST って何?という方は以下のページが参考になります。

参考にしたソースコードです。