425 lines
15 KiB
Java
Raw Normal View History

package org.whispersystems.textsecure.push;
2013-03-25 21:26:03 -07:00
import android.content.Context;
import android.util.Log;
import android.util.Pair;
2013-03-25 21:26:03 -07:00
import com.google.thoughtcrimegson.Gson;
2013-10-18 22:45:27 -07:00
2013-11-18 13:50:35 -08:00
import org.apache.http.conn.ssl.StrictHostnameVerifier;
import org.whispersystems.textsecure.R;
import org.whispersystems.textsecure.crypto.IdentityKey;
import org.whispersystems.textsecure.storage.PreKeyRecord;
import org.whispersystems.textsecure.util.Base64;
2013-07-09 19:48:33 -07:00
import org.whispersystems.textsecure.util.Util;
2013-03-25 21:26:03 -07:00
import java.io.File;
import java.io.FileOutputStream;
2013-03-25 21:26:03 -07:00
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
2013-07-08 16:29:28 -07:00
import java.net.HttpURLConnection;
2013-03-25 21:26:03 -07:00
import java.net.MalformedURLException;
import java.net.URL;
import java.security.KeyManagementException;
import java.security.KeyStore;
import java.security.KeyStoreException;
import java.security.NoSuchAlgorithmException;
import java.security.cert.CertificateException;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Set;
2013-03-25 21:26:03 -07:00
2013-10-18 22:45:27 -07:00
import javax.net.ssl.HttpsURLConnection;
import javax.net.ssl.SSLContext;
import javax.net.ssl.TrustManagerFactory;
2013-07-08 16:29:28 -07:00
public class PushServiceSocket {
2013-03-25 21:26:03 -07:00
2013-10-12 09:46:20 -07:00
private static final String CREATE_ACCOUNT_SMS_PATH = "/v1/accounts/sms/code/%s";
private static final String CREATE_ACCOUNT_VOICE_PATH = "/v1/accounts/voice/code/%s";
2013-07-08 16:29:28 -07:00
private static final String VERIFY_ACCOUNT_PATH = "/v1/accounts/code/%s";
private static final String REGISTER_GCM_PATH = "/v1/accounts/gcm/";
private static final String PREKEY_PATH = "/v1/keys/%s";
2013-07-08 16:29:28 -07:00
private static final String DIRECTORY_TOKENS_PATH = "/v1/directory/tokens";
private static final String DIRECTORY_VERIFY_PATH = "/v1/directory/%s";
2013-07-08 16:29:28 -07:00
private static final String MESSAGE_PATH = "/v1/messages/";
private static final String ATTACHMENT_PATH = "/v1/attachments/%s";
2013-03-25 21:26:03 -07:00
private static final boolean ENFORCE_SSL = true;
private final Context context;
private final String serviceUrl;
2013-03-25 21:26:03 -07:00
private final String localNumber;
private final String password;
private final TrustManagerFactory trustManagerFactory;
public PushServiceSocket(Context context, String serviceUrl, String localNumber, String password) {
this.context = context.getApplicationContext();
this.serviceUrl = serviceUrl;
2013-03-25 21:26:03 -07:00
this.localNumber = localNumber;
this.password = password;
this.trustManagerFactory = initializeTrustManagerFactory(context);
}
public PushServiceSocket(Context context, String serviceUrl, PushCredentials credentials) {
this(context, serviceUrl, credentials.getLocalNumber(context), credentials.getPassword(context));
2013-10-19 11:15:45 -07:00
}
public void createAccount(boolean voice) throws IOException {
2013-07-08 16:29:28 -07:00
String path = voice ? CREATE_ACCOUNT_VOICE_PATH : CREATE_ACCOUNT_SMS_PATH;
2013-10-12 09:46:20 -07:00
makeRequest(String.format(path, localNumber), "GET", null);
2013-03-25 21:26:03 -07:00
}
2013-08-28 15:35:30 -07:00
public void verifyAccount(String verificationCode, String signalingKey) throws IOException {
SignalingKey signalingKeyEntity = new SignalingKey(signalingKey);
makeRequest(String.format(VERIFY_ACCOUNT_PATH, verificationCode), "PUT", new Gson().toJson(signalingKeyEntity));
2013-03-25 21:26:03 -07:00
}
public void registerGcmId(String gcmRegistrationId) throws IOException {
2013-03-25 21:26:03 -07:00
GcmRegistrationId registration = new GcmRegistrationId(gcmRegistrationId);
2013-07-08 16:29:28 -07:00
makeRequest(REGISTER_GCM_PATH, "PUT", new Gson().toJson(registration));
2013-03-25 21:26:03 -07:00
}
public void unregisterGcmId() throws IOException {
makeRequest(REGISTER_GCM_PATH, "DELETE", null);
}
public void sendMessage(PushDestination recipient, PushBody pushBody)
throws IOException
2013-07-08 16:29:28 -07:00
{
OutgoingPushMessage message = new OutgoingPushMessage(recipient.getRelay(),
recipient.getNumber(),
pushBody.getBody(),
pushBody.getType());
sendMessage(new OutgoingPushMessageList(message));
}
public void sendMessage(List<PushDestination> recipients, List<PushBody> bodies)
throws IOException
{
List<OutgoingPushMessage> messages = new LinkedList<OutgoingPushMessage>();
Iterator<PushDestination> recipientsIterator = recipients.iterator();
Iterator<PushBody> bodiesIterator = bodies.iterator();
while (recipientsIterator.hasNext()) {
PushDestination recipient = recipientsIterator.next();
PushBody body = bodiesIterator.next();
messages.add(new OutgoingPushMessage(recipient.getRelay(), recipient.getNumber(),
body.getBody(), body.getType()));
}
sendMessage(new OutgoingPushMessageList(messages));
}
private void sendMessage(OutgoingPushMessageList messages) throws IOException {
String responseText = makeRequest(MESSAGE_PATH, "POST", new Gson().toJson(messages));
PushMessageResponse response = new Gson().fromJson(responseText, PushMessageResponse.class);
if (response.getFailure().size() != 0)
throw new IOException("Got send failure: " + response.getFailure().get(0));
}
2013-08-28 15:35:30 -07:00
public void registerPreKeys(IdentityKey identityKey,
PreKeyRecord lastResortKey,
List<PreKeyRecord> records)
throws IOException
{
List<PreKeyEntity> entities = new LinkedList<PreKeyEntity>();
for (PreKeyRecord record : records) {
PreKeyEntity entity = new PreKeyEntity(record.getId(),
record.getKeyPair().getPublicKey(),
identityKey);
entities.add(entity);
}
2013-08-28 15:35:30 -07:00
PreKeyEntity lastResortEntity = new PreKeyEntity(lastResortKey.getId(),
lastResortKey.getKeyPair().getPublicKey(),
identityKey);
makeRequest(String.format(PREKEY_PATH, ""), "PUT",
PreKeyList.toJson(new PreKeyList(lastResortEntity, entities)));
}
public PreKeyEntity getPreKey(PushDestination destination) throws IOException {
String path = String.format(PREKEY_PATH, destination.getNumber());
2013-10-18 22:45:27 -07:00
2013-11-19 10:02:07 -08:00
if (!Util.isEmpty(destination.getRelay())) {
path = path + "?relay=" + destination.getRelay();
2013-10-18 22:45:27 -07:00
}
String responseText = makeRequest(path, "GET", null);
Log.w("PushServiceSocket", "Got prekey: " + responseText);
return PreKeyEntity.fromJson(responseText);
}
public long sendAttachment(PushAttachmentData attachment) throws IOException {
Pair<String, String> response = makeRequestForResponseHeader(String.format(ATTACHMENT_PATH, ""),
"GET", null, "Content-Location");
String contentLocation = response.first;
Log.w("PushServiceSocket", "Got attachment content location: " + contentLocation);
if (contentLocation == null) {
throw new IOException("Server failed to allocate an attachment key!");
}
uploadExternalFile("PUT", contentLocation, attachment.getData());
return new Gson().fromJson(response.second, AttachmentKey.class).getId();
}
public File retrieveAttachment(String relay, long attachmentId) throws IOException {
String path = String.format(ATTACHMENT_PATH, String.valueOf(attachmentId));
2013-11-19 10:02:07 -08:00
if (!Util.isEmpty(relay)) {
path = path + "?relay=" + relay;
}
Pair<String, String> response = makeRequestForResponseHeader(path, "GET", null, "Content-Location");
Log.w("PushServiceSocket", "Attachment: " + attachmentId + " is at: " + response.first);
File attachment = File.createTempFile("attachment", ".tmp", context.getFilesDir());
attachment.deleteOnExit();
downloadExternalFile(response.first, attachment);
return attachment;
}
2013-10-18 22:45:27 -07:00
public List<ContactTokenDetails> retrieveDirectory(Set<String> contactTokens) {
2013-03-25 21:26:03 -07:00
try {
2013-10-18 22:45:27 -07:00
ContactTokenList contactTokenList = new ContactTokenList(new LinkedList(contactTokens));
String response = makeRequest(DIRECTORY_TOKENS_PATH, "PUT", new Gson().toJson(contactTokenList));
ContactTokenDetailsList activeTokens = new Gson().fromJson(response, ContactTokenDetailsList.class);
2013-03-25 21:26:03 -07:00
return activeTokens.getContacts();
2013-03-25 21:26:03 -07:00
} catch (IOException ioe) {
2013-07-08 16:29:28 -07:00
Log.w("PushServiceSocket", ioe);
return null;
}
}
2013-10-18 22:45:27 -07:00
public ContactTokenDetails getContactTokenDetails(String contactToken) throws IOException {
try {
2013-10-18 22:45:27 -07:00
String response = makeRequest(String.format(DIRECTORY_VERIFY_PATH, contactToken), "GET", null);
return new Gson().fromJson(response, ContactTokenDetails.class);
} catch (NotFoundException nfe) {
2013-10-18 22:45:27 -07:00
return null;
}
}
private void downloadExternalFile(String url, File localDestination)
throws IOException
{
URL downloadUrl = new URL(url);
HttpURLConnection connection = (HttpURLConnection) downloadUrl.openConnection();
connection.setRequestProperty("Content-Type", "application/octet-stream");
connection.setRequestMethod("GET");
connection.setDoInput(true);
try {
if (connection.getResponseCode() != 200) {
throw new IOException("Bad response: " + connection.getResponseCode());
}
OutputStream output = new FileOutputStream(localDestination);
InputStream input = connection.getInputStream();
byte[] buffer = new byte[4096];
int read;
while ((read = input.read(buffer)) != -1) {
output.write(buffer, 0, read);
}
output.close();
Log.w("PushServiceSocket", "Downloaded: " + url + " to: " + localDestination.getAbsolutePath());
} finally {
connection.disconnect();
}
}
private void uploadExternalFile(String method, String url, byte[] data)
throws IOException
{
URL uploadUrl = new URL(url);
HttpsURLConnection connection = (HttpsURLConnection) uploadUrl.openConnection();
connection.setDoOutput(true);
connection.setRequestMethod(method);
connection.setRequestProperty("Content-Type", "application/octet-stream");
connection.connect();
try {
OutputStream out = connection.getOutputStream();
out.write(data);
out.close();
if (connection.getResponseCode() != 200) {
throw new IOException("Bad response: " + connection.getResponseCode() + " " + connection.getResponseMessage());
}
} finally {
connection.disconnect();
2013-03-25 21:26:03 -07:00
}
}
private Pair<String, String> makeRequestForResponseHeader(String urlFragment, String method,
String body, String responseHeader)
throws IOException
{
HttpURLConnection connection = makeBaseRequest(urlFragment, method, body);
String response = Util.readFully(connection.getInputStream());
String headerValue = connection.getHeaderField(responseHeader);
connection.disconnect();
return new Pair<String, String>(headerValue, response);
2013-03-25 21:26:03 -07:00
}
2013-07-08 16:29:28 -07:00
private String makeRequest(String urlFragment, String method, String body)
throws IOException
{
HttpURLConnection connection = makeBaseRequest(urlFragment, method, body);
String response = Util.readFully(connection.getInputStream());
connection.disconnect();
return response;
}
private HttpURLConnection makeBaseRequest(String urlFragment, String method, String body)
throws IOException
2013-07-08 16:29:28 -07:00
{
HttpURLConnection connection = getConnection(urlFragment, method);
if (body != null) {
connection.setDoOutput(true);
}
connection.connect();
if (body != null) {
2013-07-08 16:29:28 -07:00
Log.w("PushServiceSocket", method + " -- " + body);
OutputStream out = connection.getOutputStream();
out.write(body.getBytes());
out.close();
}
2013-07-08 16:29:28 -07:00
if (connection.getResponseCode() == 413) {
throw new RateLimitException("Rate limit exceeded: " + connection.getResponseCode());
}
if (connection.getResponseCode() == 403) {
throw new AuthorizationFailedException("Authorization failed!");
}
if (connection.getResponseCode() == 404) {
throw new NotFoundException("Not found");
}
2013-10-31 12:30:22 -07:00
if (connection.getResponseCode() != 200 && connection.getResponseCode() != 204) {
throw new IOException("Bad response: " + connection.getResponseCode() + " " + connection.getResponseMessage());
}
return connection;
}
2013-07-08 16:29:28 -07:00
private HttpURLConnection getConnection(String urlFragment, String method) throws IOException {
2013-03-25 21:26:03 -07:00
try {
SSLContext context = SSLContext.getInstance("TLS");
context.init(null, trustManagerFactory.getTrustManagers(), null);
URL url = new URL(String.format("%s%s", serviceUrl, urlFragment));
Log.w("PushServiceSocket", "Push service URL: " + serviceUrl);
2013-10-18 22:45:27 -07:00
Log.w("PushServiceSocket", "Opening URL: " + url);
2013-07-08 16:29:28 -07:00
HttpURLConnection connection = (HttpURLConnection)url.openConnection();
if (ENFORCE_SSL) {
2013-07-08 16:29:28 -07:00
((HttpsURLConnection)connection).setSSLSocketFactory(context.getSocketFactory());
2013-11-18 13:50:35 -08:00
((HttpsURLConnection)connection).setHostnameVerifier(new StrictHostnameVerifier());
2013-07-08 16:29:28 -07:00
}
2013-03-25 21:26:03 -07:00
connection.setRequestMethod(method);
connection.setRequestProperty("Content-Type", "application/json");
if (password != null) {
connection.setRequestProperty("Authorization", getAuthorizationHeader());
}
return connection;
2013-03-25 21:26:03 -07:00
} catch (NoSuchAlgorithmException e) {
throw new AssertionError(e);
} catch (KeyManagementException e) {
throw new AssertionError(e);
} catch (MalformedURLException e) {
throw new AssertionError(e);
}
}
private String getAuthorizationHeader() {
try {
return "Basic " + Base64.encodeBytes((localNumber + ":" + password).getBytes("UTF-8"));
2013-03-25 21:26:03 -07:00
} catch (UnsupportedEncodingException e) {
throw new AssertionError(e);
}
}
private TrustManagerFactory initializeTrustManagerFactory(Context context) {
try {
InputStream keyStoreInputStream = context.getResources().openRawResource(R.raw.whisper);
KeyStore trustStore = KeyStore.getInstance("BKS");
trustStore.load(keyStoreInputStream, "whisper".toCharArray());
TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance("X509");
trustManagerFactory.init(trustStore);
return trustManagerFactory;
} catch (KeyStoreException kse) {
throw new AssertionError(kse);
} catch (CertificateException e) {
throw new AssertionError(e);
} catch (NoSuchAlgorithmException e) {
throw new AssertionError(e);
} catch (IOException ioe) {
throw new AssertionError(ioe);
}
}
private static class GcmRegistrationId {
2013-03-25 21:26:03 -07:00
private String gcmRegistrationId;
public GcmRegistrationId() {}
public GcmRegistrationId(String gcmRegistrationId) {
this.gcmRegistrationId = gcmRegistrationId;
}
}
private static class AttachmentKey {
private long id;
public AttachmentKey(long id) {
this.id = id;
}
public long getId() {
return id;
}
}
2013-10-19 11:15:45 -07:00
public interface PushCredentials {
public String getLocalNumber(Context context);
public String getPassword(Context context);
}
2013-03-25 21:26:03 -07:00
}