前言
我已经给Facebook报bug了…
最近实习的任务是做个类似facebook第三方客户端, 要求用graph api. 调用graph api就是普通的http请求, 但是facebook在这方面挖了不少坑, 特别恶心. 写篇文章记录一下, 顺便介绍一下思路和方法, 有好几个方法, 有的成功了, 有的失败了, 会具体分析, 最后有一个最佳的方法.
图片上传
sdk文档的错误
查询facebook的graph api文档可以知道如果使用sdk上传图片, 需要使用下面的代码
Bundle params = new Bundle();
params.putString("source", "{image-data}");//这句错了
/* make the API call */
new Request(
session,
"/me/photos",
params,
HttpMethod.POST,
new Request.Callback() {
public void onCompleted(Response response) {
/* handle the result */
}
}
).executeAsync();
我要用sdk, 当然会照着文档来调用, 这个api描述一看就知道是个错的, 图片再怎么也不会用String
来传, 我一开始还以为facebook有什么黑科技, 结果是他们写错了, 应该是params.putByteArray
.
使用graph api
我们要用graph api完成上传, 所以重点看graph api如何调用, facebook是这样写的.
Capture a photo via file upload as multipart/form-data then >use the source parameter
POST /v2.2/me/photos HTTP/1.1
Host: graph.facebook.com
multipart/form-data是什么
接下来我们就要搞清楚这个multipart/form-data是什么. 这其实是POST方法中一种用于传送多个文件或数据的协议, 比较古老了. 我第一个想到的就是用HttpClient
来实现, 因为它帮我们封装了很多东西.
HttpClient实现
支持HTTPS的HttpClient
由于graph api的调用都是使用的https, 所以我先实现了一个SSLHttpClient类, 封装了信任facebook证书的逻辑, 用起来也比较简单. 代码如下, 可以不看.
public class SSLHttpClient {
private final static boolean DEBUG = true;
private final String TAG = "SSLHttpClient";
//特么的下个证书都下不了chrome真垃圾
private final static String CA_NAME = "fb_ca.cer";
public HttpClient mHttpClient;
public SSLHttpClient(Context context) {
InputStream inputStream = null;
try {
inputStream = context.getAssets().open(CA_NAME);
CertificateFactory certificateFactory = CertificateFactory.getInstance("X.509");
Certificate certificate = certificateFactory.generateCertificate(inputStream);
KeyStore keyStore = KeyStore.getInstance("PKCS12", "BC");
keyStore.load(null, null);
keyStore.setCertificateEntry("trust", certificate);
SSLSocketFactory socketFactory = new SSLSocketFactory(keyStore);
Scheme scheme = new Scheme("https", socketFactory, 443);
mHttpClient = new DefaultHttpClient();
mHttpClient.getConnectionManager().getSchemeRegistry().register(scheme);
} catch (IOException e) {
if (DEBUG == true) Log.d(TAG, "IOException");
e.printStackTrace();
} catch (CertificateException e) {
if (DEBUG == true) Log.d(TAG, "CertificateException");
e.printStackTrace();
} catch (KeyStoreException e) {
if (DEBUG == true) Log.d(TAG, "KeyStoreException");
e.printStackTrace();
} catch (NoSuchProviderException e) {
if (DEBUG == true) Log.d(TAG, "NoSuchProviderException");
e.printStackTrace();
} catch (NoSuchAlgorithmException e) {
if (DEBUG == true) Log.d(TAG, "NoSuchAlgorithmException");
e.printStackTrace();
} catch (KeyManagementException e) {
if (DEBUG == true) Log.d(TAG, "KeyManagementException");
e.printStackTrace();
} catch (UnrecoverableKeyException e) {
if (DEBUG == true) Log.d(TAG, "UnrecoverableKeyException");
e.printStackTrace();
} finally {
if (inputStream != null) {
try {