mirror of
https://github.com/oxen-io/session-android.git
synced 2025-02-22 03:48:26 +00:00
Support for multi-device contact sync.
// FREEBIE
This commit is contained in:
parent
a397f34ec4
commit
3c41f27298
@ -67,7 +67,7 @@ dependencies {
|
|||||||
compile 'org.whispersystems:jobmanager:0.11.0'
|
compile 'org.whispersystems:jobmanager:0.11.0'
|
||||||
compile 'org.whispersystems:libpastelog:1.0.6'
|
compile 'org.whispersystems:libpastelog:1.0.6'
|
||||||
compile 'com.amulyakhare:com.amulyakhare.textdrawable:1.0.1'
|
compile 'com.amulyakhare:com.amulyakhare.textdrawable:1.0.1'
|
||||||
compile 'org.whispersystems:textsecure-android:1.5.0'
|
compile 'org.whispersystems:textsecure-android:1.6.0-RC13'
|
||||||
|
|
||||||
compile 'com.squareup.leakcanary:leakcanary-android:1.3.1'
|
compile 'com.squareup.leakcanary:leakcanary-android:1.3.1'
|
||||||
|
|
||||||
@ -114,12 +114,10 @@ dependencyVerification {
|
|||||||
'org.whispersystems:jobmanager:ea9cb943c4892fb90c1eea1be30efeb85cefca213d52c788419553b58d0ed70d',
|
'org.whispersystems:jobmanager:ea9cb943c4892fb90c1eea1be30efeb85cefca213d52c788419553b58d0ed70d',
|
||||||
'org.whispersystems:libpastelog:550d33c565380d90f4c671e7b8ed5f3a6da55a9fda468373177106b2eb5220b2',
|
'org.whispersystems:libpastelog:550d33c565380d90f4c671e7b8ed5f3a6da55a9fda468373177106b2eb5220b2',
|
||||||
'com.amulyakhare:com.amulyakhare.textdrawable:54c92b5fba38cfd316a07e5a30528068f45ce8515a6890f1297df4c401af5dcb',
|
'com.amulyakhare:com.amulyakhare.textdrawable:54c92b5fba38cfd316a07e5a30528068f45ce8515a6890f1297df4c401af5dcb',
|
||||||
'org.whispersystems:textsecure-android:b8001bd3c77fadcf84e0c3b21353f76d6db783bd17b28c5d7296d71198446d4b',
|
|
||||||
'com.nineoldandroids:library:68025a14e3e7673d6ad2f95e4b46d78d7d068343aa99256b686fe59de1b3163a',
|
'com.nineoldandroids:library:68025a14e3e7673d6ad2f95e4b46d78d7d068343aa99256b686fe59de1b3163a',
|
||||||
'javax.inject:javax.inject:91c77044a50c481636c32d916fd89c9118a72195390452c81065080f957de7ff',
|
'javax.inject:javax.inject:91c77044a50c481636c32d916fd89c9118a72195390452c81065080f957de7ff',
|
||||||
'com.madgag.spongycastle:core:8d6240b974b0aca4d3da9c7dd44d42339d8a374358aca5fc98e50a995764511f',
|
'com.madgag.spongycastle:core:8d6240b974b0aca4d3da9c7dd44d42339d8a374358aca5fc98e50a995764511f',
|
||||||
'org.whispersystems:axolotl-android:40d3db5004a84749a73f68d2f0d01b2ae35a73c54df96d8c6c6723b96efb6fc0',
|
'org.whispersystems:axolotl-android:40d3db5004a84749a73f68d2f0d01b2ae35a73c54df96d8c6c6723b96efb6fc0',
|
||||||
'org.whispersystems:textsecure-java:b879094961f9ef0c851206223051e8478e6bbba0f039ea9340be20f40384bb20',
|
|
||||||
'org.whispersystems:axolotl-java:6daee739b89d8d7101de6d98f77132fee48495c6ea647d880e77def842f999ea',
|
'org.whispersystems:axolotl-java:6daee739b89d8d7101de6d98f77132fee48495c6ea647d880e77def842f999ea',
|
||||||
'org.whispersystems:curve25519-android:3c29a4131a69b0d16baaa3d707678deb39602c3a3ffd75805ce7f9db252e5d0d',
|
'org.whispersystems:curve25519-android:3c29a4131a69b0d16baaa3d707678deb39602c3a3ffd75805ce7f9db252e5d0d',
|
||||||
'com.googlecode.libphonenumber:libphonenumber:eba17eae81dd622ea89a00a3a8c025b2f25d342e0d9644c5b62e16f15687c3ab',
|
'com.googlecode.libphonenumber:libphonenumber:eba17eae81dd622ea89a00a3a8c025b2f25d342e0d9644c5b62e16f15687c3ab',
|
||||||
|
@ -31,7 +31,7 @@ import org.thoughtcrime.securesms.recipients.Recipients;
|
|||||||
import org.thoughtcrime.securesms.sms.MessageSender;
|
import org.thoughtcrime.securesms.sms.MessageSender;
|
||||||
import org.thoughtcrime.securesms.util.Base64;
|
import org.thoughtcrime.securesms.util.Base64;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureEnvelope;
|
import org.whispersystems.textsecure.api.messages.TextSecureEnvelope;
|
||||||
import org.whispersystems.textsecure.internal.push.PushMessageProtos;
|
import org.whispersystems.textsecure.internal.push.TextSecureProtos;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
|
||||||
@ -162,11 +162,12 @@ public class ConfirmIdentityDialog extends AlertDialog {
|
|||||||
mismatch.getRecipientId(),
|
mismatch.getRecipientId(),
|
||||||
mismatch.getIdentityKey());
|
mismatch.getIdentityKey());
|
||||||
|
|
||||||
TextSecureEnvelope envelope = new TextSecureEnvelope(PushMessageProtos.IncomingPushMessageSignal.Type.PREKEY_BUNDLE_VALUE,
|
TextSecureEnvelope envelope = new TextSecureEnvelope(TextSecureProtos.Envelope.Type.PREKEY_BUNDLE_VALUE,
|
||||||
messageRecord.getIndividualRecipient().getNumber(),
|
messageRecord.getIndividualRecipient().getNumber(),
|
||||||
messageRecord.getRecipientDeviceId(), "",
|
messageRecord.getRecipientDeviceId(), "",
|
||||||
messageRecord.getDateSent(),
|
messageRecord.getDateSent(),
|
||||||
Base64.decode(messageRecord.getBody().getBody()));
|
Base64.decode(messageRecord.getBody().getBody()),
|
||||||
|
null);
|
||||||
|
|
||||||
long pushId = pushDatabase.insert(envelope);
|
long pushId = pushDatabase.insert(envelope);
|
||||||
|
|
||||||
|
@ -109,7 +109,7 @@ import java.util.List;
|
|||||||
|
|
||||||
import static org.thoughtcrime.securesms.TransportOption.Type;
|
import static org.thoughtcrime.securesms.TransportOption.Type;
|
||||||
import static org.thoughtcrime.securesms.database.GroupDatabase.GroupRecord;
|
import static org.thoughtcrime.securesms.database.GroupDatabase.GroupRecord;
|
||||||
import static org.whispersystems.textsecure.internal.push.PushMessageProtos.PushMessageContent.GroupContext;
|
import static org.whispersystems.textsecure.internal.push.TextSecureProtos.GroupContext;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Activity for displaying a message thread, as well as
|
* Activity for displaying a message thread, as well as
|
||||||
|
@ -44,10 +44,12 @@ import com.google.protobuf.ByteString;
|
|||||||
import com.soundcloud.android.crop.Crop;
|
import com.soundcloud.android.crop.Crop;
|
||||||
|
|
||||||
import org.thoughtcrime.securesms.components.PushRecipientsPanel;
|
import org.thoughtcrime.securesms.components.PushRecipientsPanel;
|
||||||
import org.thoughtcrime.securesms.contacts.ContactAccessor;
|
|
||||||
import org.thoughtcrime.securesms.contacts.RecipientsEditor;
|
import org.thoughtcrime.securesms.contacts.RecipientsEditor;
|
||||||
|
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
||||||
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
||||||
import org.thoughtcrime.securesms.database.GroupDatabase;
|
import org.thoughtcrime.securesms.database.GroupDatabase;
|
||||||
|
import org.thoughtcrime.securesms.database.NotInDirectoryException;
|
||||||
|
import org.thoughtcrime.securesms.database.TextSecureDirectory;
|
||||||
import org.thoughtcrime.securesms.database.ThreadDatabase;
|
import org.thoughtcrime.securesms.database.ThreadDatabase;
|
||||||
import org.thoughtcrime.securesms.mms.OutgoingGroupMediaMessage;
|
import org.thoughtcrime.securesms.mms.OutgoingGroupMediaMessage;
|
||||||
import org.thoughtcrime.securesms.recipients.Recipient;
|
import org.thoughtcrime.securesms.recipients.Recipient;
|
||||||
@ -63,10 +65,8 @@ import org.thoughtcrime.securesms.util.ProgressDialogAsyncTask;
|
|||||||
import org.thoughtcrime.securesms.util.SelectedRecipientsAdapter;
|
import org.thoughtcrime.securesms.util.SelectedRecipientsAdapter;
|
||||||
import org.thoughtcrime.securesms.util.TextSecurePreferences;
|
import org.thoughtcrime.securesms.util.TextSecurePreferences;
|
||||||
import org.thoughtcrime.securesms.util.Util;
|
import org.thoughtcrime.securesms.util.Util;
|
||||||
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
|
||||||
import org.thoughtcrime.securesms.database.TextSecureDirectory;
|
|
||||||
import org.thoughtcrime.securesms.database.NotInDirectoryException;
|
|
||||||
import org.whispersystems.textsecure.api.util.InvalidNumberException;
|
import org.whispersystems.textsecure.api.util.InvalidNumberException;
|
||||||
|
import org.whispersystems.textsecure.internal.push.TextSecureProtos.GroupContext;
|
||||||
|
|
||||||
import java.io.ByteArrayOutputStream;
|
import java.io.ByteArrayOutputStream;
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
@ -80,8 +80,6 @@ import java.util.Set;
|
|||||||
|
|
||||||
import ws.com.google.android.mms.MmsException;
|
import ws.com.google.android.mms.MmsException;
|
||||||
|
|
||||||
import static org.thoughtcrime.securesms.contacts.ContactAccessor.ContactData;
|
|
||||||
import static org.whispersystems.textsecure.internal.push.PushMessageProtos.PushMessageContent.GroupContext;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Activity to create and update groups
|
* Activity to create and update groups
|
||||||
|
@ -62,7 +62,8 @@ public class DatabaseFactory {
|
|||||||
private static final int INTRODUCED_IDENTITY_COLUMN_VERSION = 16;
|
private static final int INTRODUCED_IDENTITY_COLUMN_VERSION = 16;
|
||||||
private static final int INTRODUCED_UNIQUE_PART_IDS_VERSION = 17;
|
private static final int INTRODUCED_UNIQUE_PART_IDS_VERSION = 17;
|
||||||
private static final int INTRODUCED_RECIPIENT_PREFS_DB = 18;
|
private static final int INTRODUCED_RECIPIENT_PREFS_DB = 18;
|
||||||
private static final int DATABASE_VERSION = 18;
|
private static final int INTRODUCED_ENVELOPE_CONTENT_VERSION = 19;
|
||||||
|
private static final int DATABASE_VERSION = 19;
|
||||||
|
|
||||||
private static final String DATABASE_NAME = "messages.db";
|
private static final String DATABASE_NAME = "messages.db";
|
||||||
private static final Object lock = new Object();
|
private static final Object lock = new Object();
|
||||||
@ -737,6 +738,10 @@ public class DatabaseFactory {
|
|||||||
"notification TEXT DEFAULT NULL, vibrate INTEGER DEFAULT 0, mute_until INTEGER DEFAULT 0)");
|
"notification TEXT DEFAULT NULL, vibrate INTEGER DEFAULT 0, mute_until INTEGER DEFAULT 0)");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (oldVersion < INTRODUCED_ENVELOPE_CONTENT_VERSION) {
|
||||||
|
db.execSQL("ALTER TABLE push ADD COLUMN content TEXT");
|
||||||
|
}
|
||||||
|
|
||||||
db.setTransactionSuccessful();
|
db.setTransactionSuccessful();
|
||||||
db.endTransaction();
|
db.endTransaction();
|
||||||
}
|
}
|
||||||
|
@ -11,6 +11,7 @@ import android.util.Log;
|
|||||||
import org.thoughtcrime.securesms.util.Base64;
|
import org.thoughtcrime.securesms.util.Base64;
|
||||||
import org.whispersystems.libaxolotl.util.guava.Optional;
|
import org.whispersystems.libaxolotl.util.guava.Optional;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureEnvelope;
|
import org.whispersystems.textsecure.api.messages.TextSecureEnvelope;
|
||||||
|
import org.whispersystems.textsecure.internal.util.Util;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
|
||||||
@ -23,11 +24,12 @@ public class PushDatabase extends Database {
|
|||||||
public static final String TYPE = "type";
|
public static final String TYPE = "type";
|
||||||
public static final String SOURCE = "source";
|
public static final String SOURCE = "source";
|
||||||
public static final String DEVICE_ID = "device_id";
|
public static final String DEVICE_ID = "device_id";
|
||||||
public static final String BODY = "body";
|
public static final String LEGACY_MSG = "body";
|
||||||
|
public static final String CONTENT = "content";
|
||||||
public static final String TIMESTAMP = "timestamp";
|
public static final String TIMESTAMP = "timestamp";
|
||||||
|
|
||||||
public static final String CREATE_TABLE = "CREATE TABLE " + TABLE_NAME + " (" + ID + " INTEGER PRIMARY KEY, " +
|
public static final String CREATE_TABLE = "CREATE TABLE " + TABLE_NAME + " (" + ID + " INTEGER PRIMARY KEY, " +
|
||||||
TYPE + " INTEGER, " + SOURCE + " TEXT, " + DEVICE_ID + " INTEGER, " + BODY + " TEXT, " + TIMESTAMP + " INTEGER);";
|
TYPE + " INTEGER, " + SOURCE + " TEXT, " + DEVICE_ID + " INTEGER, " + LEGACY_MSG + " TEXT, " + CONTENT + " TEXT, " + TIMESTAMP + " INTEGER);";
|
||||||
|
|
||||||
public PushDatabase(Context context, SQLiteOpenHelper databaseHelper) {
|
public PushDatabase(Context context, SQLiteOpenHelper databaseHelper) {
|
||||||
super(context, databaseHelper);
|
super(context, databaseHelper);
|
||||||
@ -43,7 +45,8 @@ public class PushDatabase extends Database {
|
|||||||
values.put(TYPE, envelope.getType());
|
values.put(TYPE, envelope.getType());
|
||||||
values.put(SOURCE, envelope.getSource());
|
values.put(SOURCE, envelope.getSource());
|
||||||
values.put(DEVICE_ID, envelope.getSourceDevice());
|
values.put(DEVICE_ID, envelope.getSourceDevice());
|
||||||
values.put(BODY, Base64.encodeBytes(envelope.getMessage()));
|
values.put(LEGACY_MSG, envelope.hasLegacyMessage() ? Base64.encodeBytes(envelope.getLegacyMessage()) : "");
|
||||||
|
values.put(CONTENT, envelope.hasContent() ? Base64.encodeBytes(envelope.getContent()) : "");
|
||||||
values.put(TIMESTAMP, envelope.getTimestamp());
|
values.put(TIMESTAMP, envelope.getTimestamp());
|
||||||
|
|
||||||
return databaseHelper.getWritableDatabase().insert(TABLE_NAME, null, values);
|
return databaseHelper.getWritableDatabase().insert(TABLE_NAME, null, values);
|
||||||
@ -59,12 +62,16 @@ public class PushDatabase extends Database {
|
|||||||
null, null, null);
|
null, null, null);
|
||||||
|
|
||||||
if (cursor != null && cursor.moveToNext()) {
|
if (cursor != null && cursor.moveToNext()) {
|
||||||
|
String legacyMessage = cursor.getString(cursor.getColumnIndexOrThrow(LEGACY_MSG));
|
||||||
|
String content = cursor.getString(cursor.getColumnIndexOrThrow(CONTENT));
|
||||||
|
|
||||||
return new TextSecureEnvelope(cursor.getInt(cursor.getColumnIndexOrThrow(TYPE)),
|
return new TextSecureEnvelope(cursor.getInt(cursor.getColumnIndexOrThrow(TYPE)),
|
||||||
cursor.getString(cursor.getColumnIndexOrThrow(SOURCE)),
|
cursor.getString(cursor.getColumnIndexOrThrow(SOURCE)),
|
||||||
cursor.getInt(cursor.getColumnIndexOrThrow(DEVICE_ID)),
|
cursor.getInt(cursor.getColumnIndexOrThrow(DEVICE_ID)),
|
||||||
"",
|
"",
|
||||||
cursor.getLong(cursor.getColumnIndexOrThrow(TIMESTAMP)),
|
cursor.getLong(cursor.getColumnIndexOrThrow(TIMESTAMP)),
|
||||||
Base64.decode(cursor.getString(cursor.getColumnIndexOrThrow(BODY))));
|
Util.isEmpty(legacyMessage) ? null : Base64.decode(legacyMessage),
|
||||||
|
Util.isEmpty(content) ? null : Base64.decode(content));
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
Log.w(TAG, e);
|
Log.w(TAG, e);
|
||||||
@ -95,12 +102,13 @@ public class PushDatabase extends Database {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
cursor = database.query(TABLE_NAME, null, TYPE + " = ? AND " + SOURCE + " = ? AND " +
|
cursor = database.query(TABLE_NAME, null, TYPE + " = ? AND " + SOURCE + " = ? AND " +
|
||||||
DEVICE_ID + " = ? AND " + BODY + " = ? AND " +
|
DEVICE_ID + " = ? AND " + LEGACY_MSG + " = ? AND " +
|
||||||
TIMESTAMP + " = ?" ,
|
CONTENT + " = ? AND " + TIMESTAMP + " = ?" ,
|
||||||
new String[] {String.valueOf(envelope.getType()),
|
new String[] {String.valueOf(envelope.getType()),
|
||||||
envelope.getSource(),
|
envelope.getSource(),
|
||||||
String.valueOf(envelope.getSourceDevice()),
|
String.valueOf(envelope.getSourceDevice()),
|
||||||
Base64.encodeBytes(envelope.getMessage()),
|
envelope.hasLegacyMessage() ? Base64.encodeBytes(envelope.getLegacyMessage()) : "",
|
||||||
|
envelope.hasContent() ? Base64.encodeBytes(envelope.getContent()) : "",
|
||||||
String.valueOf(envelope.getTimestamp())},
|
String.valueOf(envelope.getTimestamp())},
|
||||||
null, null, null);
|
null, null, null);
|
||||||
|
|
||||||
@ -129,10 +137,13 @@ public class PushDatabase extends Database {
|
|||||||
int type = cursor.getInt(cursor.getColumnIndexOrThrow(TYPE));
|
int type = cursor.getInt(cursor.getColumnIndexOrThrow(TYPE));
|
||||||
String source = cursor.getString(cursor.getColumnIndexOrThrow(SOURCE));
|
String source = cursor.getString(cursor.getColumnIndexOrThrow(SOURCE));
|
||||||
int deviceId = cursor.getInt(cursor.getColumnIndexOrThrow(DEVICE_ID));
|
int deviceId = cursor.getInt(cursor.getColumnIndexOrThrow(DEVICE_ID));
|
||||||
byte[] body = Base64.decode(cursor.getString(cursor.getColumnIndexOrThrow(BODY)));
|
String legacyMessage = cursor.getString(cursor.getColumnIndexOrThrow(LEGACY_MSG));
|
||||||
|
String content = cursor.getString(cursor.getColumnIndexOrThrow(CONTENT));
|
||||||
long timestamp = cursor.getLong(cursor.getColumnIndexOrThrow(TIMESTAMP));
|
long timestamp = cursor.getLong(cursor.getColumnIndexOrThrow(TIMESTAMP));
|
||||||
|
|
||||||
return new TextSecureEnvelope(type, source, deviceId, "", timestamp, body);
|
return new TextSecureEnvelope(type, source, deviceId, "", timestamp,
|
||||||
|
legacyMessage != null ? Base64.decode(legacyMessage) : null,
|
||||||
|
content != null ? Base64.decode(content) : null);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw new AssertionError(e);
|
throw new AssertionError(e);
|
||||||
}
|
}
|
||||||
|
@ -9,6 +9,7 @@ import org.thoughtcrime.securesms.jobs.AttachmentDownloadJob;
|
|||||||
import org.thoughtcrime.securesms.jobs.CleanPreKeysJob;
|
import org.thoughtcrime.securesms.jobs.CleanPreKeysJob;
|
||||||
import org.thoughtcrime.securesms.jobs.CreateSignedPreKeyJob;
|
import org.thoughtcrime.securesms.jobs.CreateSignedPreKeyJob;
|
||||||
import org.thoughtcrime.securesms.jobs.DeliveryReceiptJob;
|
import org.thoughtcrime.securesms.jobs.DeliveryReceiptJob;
|
||||||
|
import org.thoughtcrime.securesms.jobs.MultiDeviceContactUpdateJob;
|
||||||
import org.thoughtcrime.securesms.jobs.PushGroupSendJob;
|
import org.thoughtcrime.securesms.jobs.PushGroupSendJob;
|
||||||
import org.thoughtcrime.securesms.jobs.PushMediaSendJob;
|
import org.thoughtcrime.securesms.jobs.PushMediaSendJob;
|
||||||
import org.thoughtcrime.securesms.jobs.PushContentReceiveJob;
|
import org.thoughtcrime.securesms.jobs.PushContentReceiveJob;
|
||||||
@ -37,7 +38,8 @@ import dagger.Provides;
|
|||||||
AttachmentDownloadJob.class,
|
AttachmentDownloadJob.class,
|
||||||
RefreshPreKeysJob.class,
|
RefreshPreKeysJob.class,
|
||||||
MessageRetrievalService.class,
|
MessageRetrievalService.class,
|
||||||
PushNotificationReceiveJob.class})
|
PushNotificationReceiveJob.class,
|
||||||
|
MultiDeviceContactUpdateJob.class})
|
||||||
public class TextSecureCommunicationModule {
|
public class TextSecureCommunicationModule {
|
||||||
|
|
||||||
private final Context context;
|
private final Context context;
|
||||||
|
@ -21,7 +21,7 @@ import org.whispersystems.libaxolotl.util.guava.Optional;
|
|||||||
import org.whispersystems.textsecure.api.messages.TextSecureAttachment;
|
import org.whispersystems.textsecure.api.messages.TextSecureAttachment;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureEnvelope;
|
import org.whispersystems.textsecure.api.messages.TextSecureEnvelope;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureGroup;
|
import org.whispersystems.textsecure.api.messages.TextSecureGroup;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureMessage;
|
import org.whispersystems.textsecure.api.messages.TextSecureDataMessage;
|
||||||
|
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.LinkedList;
|
import java.util.LinkedList;
|
||||||
@ -29,8 +29,8 @@ import java.util.List;
|
|||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
import static org.thoughtcrime.securesms.database.GroupDatabase.GroupRecord;
|
import static org.thoughtcrime.securesms.database.GroupDatabase.GroupRecord;
|
||||||
import static org.whispersystems.textsecure.internal.push.PushMessageProtos.PushMessageContent.AttachmentPointer;
|
import static org.whispersystems.textsecure.internal.push.TextSecureProtos.AttachmentPointer;
|
||||||
import static org.whispersystems.textsecure.internal.push.PushMessageProtos.PushMessageContent.GroupContext;
|
import static org.whispersystems.textsecure.internal.push.TextSecureProtos.GroupContext;
|
||||||
|
|
||||||
public class GroupMessageProcessor {
|
public class GroupMessageProcessor {
|
||||||
|
|
||||||
@ -39,7 +39,7 @@ public class GroupMessageProcessor {
|
|||||||
public static void process(Context context,
|
public static void process(Context context,
|
||||||
MasterSecret masterSecret,
|
MasterSecret masterSecret,
|
||||||
TextSecureEnvelope envelope,
|
TextSecureEnvelope envelope,
|
||||||
TextSecureMessage message)
|
TextSecureDataMessage message)
|
||||||
{
|
{
|
||||||
if (!message.getGroupInfo().isPresent() || message.getGroupInfo().get().getGroupId() == null) {
|
if (!message.getGroupInfo().isPresent() || message.getGroupInfo().get().getGroupId() == null) {
|
||||||
Log.w(TAG, "Received group message with no id! Ignoring...");
|
Log.w(TAG, "Received group message with no id! Ignoring...");
|
||||||
|
@ -0,0 +1,165 @@
|
|||||||
|
package org.thoughtcrime.securesms.jobs;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.res.AssetFileDescriptor;
|
||||||
|
import android.database.Cursor;
|
||||||
|
import android.net.Uri;
|
||||||
|
import android.os.Build;
|
||||||
|
import android.provider.ContactsContract;
|
||||||
|
import android.util.Log;
|
||||||
|
|
||||||
|
import org.thoughtcrime.securesms.contacts.ContactAccessor;
|
||||||
|
import org.thoughtcrime.securesms.contacts.ContactAccessor.ContactData;
|
||||||
|
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
||||||
|
import org.thoughtcrime.securesms.dependencies.InjectableType;
|
||||||
|
import org.thoughtcrime.securesms.dependencies.TextSecureCommunicationModule.TextSecureMessageSenderFactory;
|
||||||
|
import org.thoughtcrime.securesms.jobs.requirements.MasterSecretRequirement;
|
||||||
|
import org.thoughtcrime.securesms.util.Util;
|
||||||
|
import org.whispersystems.jobqueue.JobParameters;
|
||||||
|
import org.whispersystems.jobqueue.requirements.NetworkRequirement;
|
||||||
|
import org.whispersystems.libaxolotl.util.guava.Optional;
|
||||||
|
import org.whispersystems.textsecure.api.TextSecureMessageSender;
|
||||||
|
import org.whispersystems.textsecure.api.crypto.UntrustedIdentityException;
|
||||||
|
import org.whispersystems.textsecure.api.messages.TextSecureAttachmentStream;
|
||||||
|
import org.whispersystems.textsecure.api.messages.multidevice.DeviceContact;
|
||||||
|
import org.whispersystems.textsecure.api.messages.multidevice.DeviceContactsOutputStream;
|
||||||
|
|
||||||
|
import java.io.ByteArrayInputStream;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileInputStream;
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
|
import java.io.FileOutputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.util.Collection;
|
||||||
|
|
||||||
|
import javax.inject.Inject;
|
||||||
|
|
||||||
|
public class MultiDeviceContactUpdateJob extends MasterSecretJob implements InjectableType {
|
||||||
|
|
||||||
|
private static final String TAG = MultiDeviceContactUpdateJob.class.getSimpleName();
|
||||||
|
|
||||||
|
@Inject transient TextSecureMessageSenderFactory messageSenderFactory;
|
||||||
|
|
||||||
|
public MultiDeviceContactUpdateJob(Context context) {
|
||||||
|
super(context, JobParameters.newBuilder()
|
||||||
|
.withRequirement(new NetworkRequirement(context))
|
||||||
|
.withRequirement(new MasterSecretRequirement(context))
|
||||||
|
.withGroupId(MultiDeviceContactUpdateJob.class.getSimpleName())
|
||||||
|
.withPersistence()
|
||||||
|
.create());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onRun(MasterSecret masterSecret)
|
||||||
|
throws IOException, UntrustedIdentityException, NetworkException
|
||||||
|
{
|
||||||
|
TextSecureMessageSender messageSender = messageSenderFactory.create(masterSecret);
|
||||||
|
File contactDataFile = createTempFile("multidevice-contact-update");
|
||||||
|
|
||||||
|
try {
|
||||||
|
DeviceContactsOutputStream out = new DeviceContactsOutputStream(new FileOutputStream(contactDataFile));
|
||||||
|
Collection<ContactData> contacts = ContactAccessor.getInstance().getContactsWithPush(context);
|
||||||
|
|
||||||
|
for (ContactData contactData : contacts) {
|
||||||
|
Uri contactUri = Uri.withAppendedPath(ContactsContract.Contacts.CONTENT_URI, String.valueOf(contactData.id));
|
||||||
|
String number = contactData.numbers.get(0).number;
|
||||||
|
Optional<String> name = Optional.fromNullable(contactData.name);
|
||||||
|
|
||||||
|
out.write(new DeviceContact(number, name, getAvatar(contactUri)));
|
||||||
|
}
|
||||||
|
|
||||||
|
out.close();
|
||||||
|
sendUpdate(messageSender, contactDataFile);
|
||||||
|
|
||||||
|
} finally {
|
||||||
|
if (contactDataFile != null) contactDataFile.delete();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean onShouldRetryThrowable(Exception exception) {
|
||||||
|
if (exception instanceof NetworkException) return true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onAdded() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onCanceled() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private void sendUpdate(TextSecureMessageSender messageSender, File contactsFile)
|
||||||
|
throws IOException, UntrustedIdentityException, NetworkException
|
||||||
|
{
|
||||||
|
FileInputStream contactsFileStream = new FileInputStream(contactsFile);
|
||||||
|
TextSecureAttachmentStream attachmentStream = new TextSecureAttachmentStream(contactsFileStream,
|
||||||
|
"application/octet-stream",
|
||||||
|
contactsFile.length());
|
||||||
|
|
||||||
|
try {
|
||||||
|
messageSender.sendMultiDeviceContactsUpdate(attachmentStream);
|
||||||
|
} catch (IOException ioe) {
|
||||||
|
throw new NetworkException(ioe);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private Optional<TextSecureAttachmentStream> getAvatar(Uri uri) throws IOException {
|
||||||
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
|
||||||
|
try {
|
||||||
|
Uri displayPhotoUri = Uri.withAppendedPath(uri, ContactsContract.Contacts.Photo.DISPLAY_PHOTO);
|
||||||
|
AssetFileDescriptor fd = context.getContentResolver().openAssetFileDescriptor(displayPhotoUri, "r");
|
||||||
|
return Optional.of(new TextSecureAttachmentStream(fd.createInputStream(), "image/*", fd.getLength()));
|
||||||
|
} catch (IOException e) {
|
||||||
|
Log.w(TAG, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Uri photoUri = Uri.withAppendedPath(uri, ContactsContract.Contacts.Photo.CONTENT_DIRECTORY);
|
||||||
|
|
||||||
|
if (photoUri == null) {
|
||||||
|
return Optional.absent();
|
||||||
|
}
|
||||||
|
|
||||||
|
Cursor cursor = context.getContentResolver().query(photoUri,
|
||||||
|
new String[] {
|
||||||
|
ContactsContract.CommonDataKinds.Photo.PHOTO,
|
||||||
|
ContactsContract.CommonDataKinds.Phone.MIMETYPE
|
||||||
|
}, null, null, null);
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (cursor != null && cursor.moveToNext()) {
|
||||||
|
byte[] data = cursor.getBlob(0);
|
||||||
|
|
||||||
|
if (data != null) {
|
||||||
|
return Optional.of(new TextSecureAttachmentStream(new ByteArrayInputStream(data), "image/*", data.length));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Optional.absent();
|
||||||
|
} finally {
|
||||||
|
if (cursor != null) {
|
||||||
|
cursor.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private File createTempFile(String prefix) throws IOException {
|
||||||
|
File file = File.createTempFile(prefix, "tmp", context.getCacheDir());
|
||||||
|
file.deleteOnExit();
|
||||||
|
|
||||||
|
return file;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class NetworkException extends Exception {
|
||||||
|
|
||||||
|
public NetworkException(Exception ioe) {
|
||||||
|
super(ioe);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -46,10 +46,12 @@ import org.whispersystems.libaxolotl.state.AxolotlStore;
|
|||||||
import org.whispersystems.libaxolotl.state.SessionStore;
|
import org.whispersystems.libaxolotl.state.SessionStore;
|
||||||
import org.whispersystems.libaxolotl.util.guava.Optional;
|
import org.whispersystems.libaxolotl.util.guava.Optional;
|
||||||
import org.whispersystems.textsecure.api.crypto.TextSecureCipher;
|
import org.whispersystems.textsecure.api.crypto.TextSecureCipher;
|
||||||
|
import org.whispersystems.textsecure.api.messages.TextSecureContent;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureEnvelope;
|
import org.whispersystems.textsecure.api.messages.TextSecureEnvelope;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureGroup;
|
import org.whispersystems.textsecure.api.messages.TextSecureGroup;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureMessage;
|
import org.whispersystems.textsecure.api.messages.TextSecureDataMessage;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureSyncContext;
|
import org.whispersystems.textsecure.api.messages.multidevice.SentTranscriptMessage;
|
||||||
|
import org.whispersystems.textsecure.api.messages.multidevice.TextSecureSyncMessage;
|
||||||
import org.whispersystems.textsecure.api.push.TextSecureAddress;
|
import org.whispersystems.textsecure.api.push.TextSecureAddress;
|
||||||
|
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
@ -112,12 +114,20 @@ public class PushDecryptJob extends MasterSecretJob {
|
|||||||
TextSecureAddress localAddress = new TextSecureAddress(TextSecurePreferences.getLocalNumber(context));
|
TextSecureAddress localAddress = new TextSecureAddress(TextSecurePreferences.getLocalNumber(context));
|
||||||
TextSecureCipher cipher = new TextSecureCipher(localAddress, axolotlStore);
|
TextSecureCipher cipher = new TextSecureCipher(localAddress, axolotlStore);
|
||||||
|
|
||||||
TextSecureMessage message = cipher.decrypt(envelope);
|
TextSecureContent content = cipher.decrypt(envelope);
|
||||||
|
|
||||||
|
if (content.getDataMessage().isPresent()) {
|
||||||
|
TextSecureDataMessage message = content.getDataMessage().get();
|
||||||
|
|
||||||
if (message.isEndSession()) handleEndSessionMessage(masterSecret, envelope, message, smsMessageId);
|
if (message.isEndSession()) handleEndSessionMessage(masterSecret, envelope, message, smsMessageId);
|
||||||
else if (message.isGroupUpdate()) handleGroupMessage(masterSecret, envelope, message, smsMessageId);
|
else if (message.isGroupUpdate()) handleGroupMessage(masterSecret, envelope, message, smsMessageId);
|
||||||
else if (message.getAttachments().isPresent()) handleMediaMessage(masterSecret, envelope, message, smsMessageId);
|
else if (message.getAttachments().isPresent()) handleMediaMessage(masterSecret, envelope, message, smsMessageId);
|
||||||
else handleTextMessage(masterSecret, envelope, message, smsMessageId);
|
else handleTextMessage(masterSecret, envelope, message, smsMessageId);
|
||||||
|
} else if (content.getSyncMessage().isPresent()) {
|
||||||
|
TextSecureSyncMessage syncMessage = content.getSyncMessage().get();
|
||||||
|
|
||||||
|
if (syncMessage.getSent().isPresent()) handleSynchronizeSentMessage(masterSecret, syncMessage.getSent().get(), smsMessageId);
|
||||||
|
}
|
||||||
|
|
||||||
if (envelope.isPreKeyWhisperMessage()) {
|
if (envelope.isPreKeyWhisperMessage()) {
|
||||||
ApplicationContext.getInstance(context).getJobManager().add(new RefreshPreKeysJob(context));
|
ApplicationContext.getInstance(context).getJobManager().add(new RefreshPreKeysJob(context));
|
||||||
@ -144,7 +154,7 @@ public class PushDecryptJob extends MasterSecretJob {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void handleEndSessionMessage(MasterSecret masterSecret, TextSecureEnvelope envelope,
|
private void handleEndSessionMessage(MasterSecret masterSecret, TextSecureEnvelope envelope,
|
||||||
TextSecureMessage message, Optional<Long> smsMessageId)
|
TextSecureDataMessage message, Optional<Long> smsMessageId)
|
||||||
{
|
{
|
||||||
EncryptingSmsDatabase smsDatabase = DatabaseFactory.getEncryptingSmsDatabase(context);
|
EncryptingSmsDatabase smsDatabase = DatabaseFactory.getEncryptingSmsDatabase(context);
|
||||||
IncomingTextMessage incomingTextMessage = new IncomingTextMessage(envelope.getSource(),
|
IncomingTextMessage incomingTextMessage = new IncomingTextMessage(envelope.getSource(),
|
||||||
@ -170,7 +180,7 @@ public class PushDecryptJob extends MasterSecretJob {
|
|||||||
MessageNotifier.updateNotification(context, masterSecret, threadId);
|
MessageNotifier.updateNotification(context, masterSecret, threadId);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void handleGroupMessage(MasterSecret masterSecret, TextSecureEnvelope envelope, TextSecureMessage message, Optional<Long> smsMessageId) {
|
private void handleGroupMessage(MasterSecret masterSecret, TextSecureEnvelope envelope, TextSecureDataMessage message, Optional<Long> smsMessageId) {
|
||||||
GroupMessageProcessor.process(context, masterSecret, envelope, message);
|
GroupMessageProcessor.process(context, masterSecret, envelope, message);
|
||||||
|
|
||||||
if (smsMessageId.isPresent()) {
|
if (smsMessageId.isPresent()) {
|
||||||
@ -178,17 +188,30 @@ public class PushDecryptJob extends MasterSecretJob {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void handleMediaMessage(MasterSecret masterSecret, TextSecureEnvelope envelope,
|
private void handleSynchronizeSentMessage(MasterSecret masterSecret, SentTranscriptMessage message, Optional<Long> smsMessageId)
|
||||||
TextSecureMessage message, Optional<Long> smsMessageId)
|
|
||||||
throws MmsException
|
throws MmsException
|
||||||
{
|
{
|
||||||
Pair<Long, Long> messageAndThreadId;
|
if (message.getMessage().getAttachments().isPresent()) {
|
||||||
|
handleSynchronizeSentMediaMessage(masterSecret, message, smsMessageId);
|
||||||
if (message.getSyncContext().isPresent()) {
|
|
||||||
messageAndThreadId = insertSyncMediaMessage(masterSecret, envelope, message);
|
|
||||||
} else {
|
} else {
|
||||||
messageAndThreadId = insertStandardMediaMessage(masterSecret, envelope, message);
|
handleSynchronizeSentTextMessage(masterSecret, message, smsMessageId);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void handleMediaMessage(MasterSecret masterSecret, TextSecureEnvelope envelope,
|
||||||
|
TextSecureDataMessage message, Optional<Long> smsMessageId)
|
||||||
|
throws MmsException
|
||||||
|
{
|
||||||
|
MmsDatabase database = DatabaseFactory.getMmsDatabase(context);
|
||||||
|
String localNumber = TextSecurePreferences.getLocalNumber(context);
|
||||||
|
IncomingMediaMessage mediaMessage = new IncomingMediaMessage(masterSecret, envelope.getSource(),
|
||||||
|
localNumber, message.getTimestamp(),
|
||||||
|
Optional.fromNullable(envelope.getRelay()),
|
||||||
|
message.getBody(),
|
||||||
|
message.getGroupInfo(),
|
||||||
|
message.getAttachments());
|
||||||
|
|
||||||
|
Pair<Long, Long> messageAndThreadId = database.insertSecureDecryptedMessageInbox(masterSecret, mediaMessage, -1);
|
||||||
|
|
||||||
ApplicationContext.getInstance(context)
|
ApplicationContext.getInstance(context)
|
||||||
.getJobManager()
|
.getJobManager()
|
||||||
@ -201,20 +224,79 @@ public class PushDecryptJob extends MasterSecretJob {
|
|||||||
MessageNotifier.updateNotification(context, masterSecret, messageAndThreadId.second);
|
MessageNotifier.updateNotification(context, masterSecret, messageAndThreadId.second);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void handleTextMessage(MasterSecret masterSecret, TextSecureEnvelope envelope,
|
private void handleSynchronizeSentMediaMessage(MasterSecret masterSecret,
|
||||||
TextSecureMessage message, Optional<Long> smsMessageId)
|
SentTranscriptMessage message,
|
||||||
|
Optional<Long> smsMessageId)
|
||||||
|
throws MmsException
|
||||||
{
|
{
|
||||||
|
MmsDatabase database = DatabaseFactory.getMmsDatabase(context);
|
||||||
|
Recipients recipients = getSyncMessageDestination(message);
|
||||||
|
OutgoingMediaMessage mediaMessage = new OutgoingMediaMessage(context, masterSecret, recipients,
|
||||||
|
message.getMessage().getAttachments().get(),
|
||||||
|
message.getMessage().getBody().orNull());
|
||||||
|
|
||||||
|
mediaMessage = new OutgoingSecureMediaMessage(mediaMessage);
|
||||||
|
|
||||||
|
long threadId = DatabaseFactory.getThreadDatabase(context).getThreadIdFor(recipients);
|
||||||
|
long messageId = database.insertMessageOutbox(masterSecret, mediaMessage, threadId, false, message.getTimestamp());
|
||||||
|
|
||||||
|
database.markAsSent(messageId, "push".getBytes(), 0);
|
||||||
|
database.markAsPush(messageId);
|
||||||
|
|
||||||
|
ApplicationContext.getInstance(context)
|
||||||
|
.getJobManager()
|
||||||
|
.add(new AttachmentDownloadJob(context, messageId));
|
||||||
|
|
||||||
|
if (smsMessageId.isPresent()) {
|
||||||
|
DatabaseFactory.getSmsDatabase(context).deleteMessage(smsMessageId.get());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void handleTextMessage(MasterSecret masterSecret, TextSecureEnvelope envelope,
|
||||||
|
TextSecureDataMessage message, Optional<Long> smsMessageId)
|
||||||
|
{
|
||||||
|
EncryptingSmsDatabase database = DatabaseFactory.getEncryptingSmsDatabase(context);
|
||||||
|
String body = message.getBody().isPresent() ? message.getBody().get() : "";
|
||||||
|
|
||||||
Pair<Long, Long> messageAndThreadId;
|
Pair<Long, Long> messageAndThreadId;
|
||||||
|
|
||||||
if (message.getSyncContext().isPresent()) {
|
if (smsMessageId.isPresent()) {
|
||||||
messageAndThreadId = insertSyncTextMessage(masterSecret, envelope, message, smsMessageId);
|
messageAndThreadId = database.updateBundleMessageBody(masterSecret, smsMessageId.get(), body);
|
||||||
} else {
|
} else {
|
||||||
messageAndThreadId = insertStandardTextMessage(masterSecret, envelope, message, smsMessageId);
|
IncomingTextMessage textMessage = new IncomingTextMessage(envelope.getSource(),
|
||||||
|
envelope.getSourceDevice(),
|
||||||
|
message.getTimestamp(), body,
|
||||||
|
message.getGroupInfo());
|
||||||
|
|
||||||
|
textMessage = new IncomingEncryptedMessage(textMessage, body);
|
||||||
|
|
||||||
|
messageAndThreadId = database.insertMessageInbox(masterSecret, textMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
MessageNotifier.updateNotification(context, masterSecret, messageAndThreadId.second);
|
MessageNotifier.updateNotification(context, masterSecret, messageAndThreadId.second);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void handleSynchronizeSentTextMessage(MasterSecret masterSecret,
|
||||||
|
SentTranscriptMessage message,
|
||||||
|
Optional<Long> smsMessageId)
|
||||||
|
{
|
||||||
|
EncryptingSmsDatabase database = DatabaseFactory.getEncryptingSmsDatabase(context);
|
||||||
|
Recipients recipients = getSyncMessageDestination(message);
|
||||||
|
String body = message.getMessage().getBody().or("");
|
||||||
|
OutgoingTextMessage outgoingTextMessage = new OutgoingTextMessage(recipients, body);
|
||||||
|
|
||||||
|
long threadId = DatabaseFactory.getThreadDatabase(context).getThreadIdFor(recipients);
|
||||||
|
long messageId = database.insertMessageOutbox(masterSecret, threadId, outgoingTextMessage, false, message.getTimestamp());
|
||||||
|
|
||||||
|
database.markAsSent(messageId);
|
||||||
|
database.markAsPush(messageId);
|
||||||
|
database.markAsSecure(messageId);
|
||||||
|
|
||||||
|
if (smsMessageId.isPresent()) {
|
||||||
|
database.deleteMessage(smsMessageId.get());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void handleInvalidVersionMessage(MasterSecret masterSecret, TextSecureEnvelope envelope, Optional<Long> smsMessageId) {
|
private void handleInvalidVersionMessage(MasterSecret masterSecret, TextSecureEnvelope envelope, Optional<Long> smsMessageId) {
|
||||||
EncryptingSmsDatabase smsDatabase = DatabaseFactory.getEncryptingSmsDatabase(context);
|
EncryptingSmsDatabase smsDatabase = DatabaseFactory.getEncryptingSmsDatabase(context);
|
||||||
|
|
||||||
@ -281,9 +363,9 @@ public class PushDecryptJob extends MasterSecretJob {
|
|||||||
EncryptingSmsDatabase database = DatabaseFactory.getEncryptingSmsDatabase(context);
|
EncryptingSmsDatabase database = DatabaseFactory.getEncryptingSmsDatabase(context);
|
||||||
Recipients recipients = RecipientFactory.getRecipientsFromString(context, envelope.getSource(), false);
|
Recipients recipients = RecipientFactory.getRecipientsFromString(context, envelope.getSource(), false);
|
||||||
long recipientId = recipients.getPrimaryRecipient().getRecipientId();
|
long recipientId = recipients.getPrimaryRecipient().getRecipientId();
|
||||||
PreKeyWhisperMessage whisperMessage = new PreKeyWhisperMessage(envelope.getMessage());
|
PreKeyWhisperMessage whisperMessage = new PreKeyWhisperMessage(envelope.getLegacyMessage());
|
||||||
IdentityKey identityKey = whisperMessage.getIdentityKey();
|
IdentityKey identityKey = whisperMessage.getIdentityKey();
|
||||||
String encoded = Base64.encodeBytes(envelope.getMessage());
|
String encoded = Base64.encodeBytes(envelope.getLegacyMessage());
|
||||||
IncomingTextMessage textMessage = new IncomingTextMessage(envelope.getSource(), envelope.getSourceDevice(),
|
IncomingTextMessage textMessage = new IncomingTextMessage(envelope.getSource(), envelope.getSourceDevice(),
|
||||||
envelope.getTimestamp(), encoded,
|
envelope.getTimestamp(), encoded,
|
||||||
Optional.<TextSecureGroup>absent());
|
Optional.<TextSecureGroup>absent());
|
||||||
@ -316,97 +398,11 @@ public class PushDecryptJob extends MasterSecretJob {
|
|||||||
return database.insertMessageInbox(masterSecret, textMessage);
|
return database.insertMessageInbox(masterSecret, textMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
private Pair<Long, Long> insertSyncTextMessage(MasterSecret masterSecret,
|
private Recipients getSyncMessageDestination(SentTranscriptMessage message) {
|
||||||
TextSecureEnvelope envelope,
|
if (message.getMessage().getGroupInfo().isPresent()) {
|
||||||
TextSecureMessage message,
|
return RecipientFactory.getRecipientsFromString(context, GroupUtil.getEncodedId(message.getMessage().getGroupInfo().get().getGroupId()), false);
|
||||||
Optional<Long> smsMessageId)
|
|
||||||
{
|
|
||||||
EncryptingSmsDatabase database = DatabaseFactory.getEncryptingSmsDatabase(context);
|
|
||||||
Recipients recipients = getSyncMessageDestination(message);
|
|
||||||
String body = message.getBody().or("");
|
|
||||||
OutgoingTextMessage outgoingTextMessage = new OutgoingTextMessage(recipients, body);
|
|
||||||
|
|
||||||
long threadId = DatabaseFactory.getThreadDatabase(context).getThreadIdFor(recipients);
|
|
||||||
long messageId = database.insertMessageOutbox(masterSecret, threadId, outgoingTextMessage, false, message.getSyncContext().get().getTimestamp());
|
|
||||||
|
|
||||||
database.markAsSent(messageId);
|
|
||||||
database.markAsPush(messageId);
|
|
||||||
database.markAsSecure(messageId);
|
|
||||||
|
|
||||||
if (smsMessageId.isPresent()) {
|
|
||||||
database.deleteMessage(smsMessageId.get());
|
|
||||||
}
|
|
||||||
|
|
||||||
return new Pair<>(messageId, threadId);
|
|
||||||
}
|
|
||||||
|
|
||||||
private Pair<Long, Long> insertStandardTextMessage(MasterSecret masterSecret,
|
|
||||||
TextSecureEnvelope envelope,
|
|
||||||
TextSecureMessage message,
|
|
||||||
Optional<Long> smsMessageId)
|
|
||||||
{
|
|
||||||
EncryptingSmsDatabase database = DatabaseFactory.getEncryptingSmsDatabase(context);
|
|
||||||
String body = message.getBody().isPresent() ? message.getBody().get() : "";
|
|
||||||
|
|
||||||
if (smsMessageId.isPresent()) {
|
|
||||||
return database.updateBundleMessageBody(masterSecret, smsMessageId.get(), body);
|
|
||||||
} else {
|
} else {
|
||||||
IncomingTextMessage textMessage = new IncomingTextMessage(envelope.getSource(),
|
return RecipientFactory.getRecipientsFromString(context, message.getDestination().get(), false);
|
||||||
envelope.getSourceDevice(),
|
|
||||||
message.getTimestamp(), body,
|
|
||||||
message.getGroupInfo());
|
|
||||||
|
|
||||||
textMessage = new IncomingEncryptedMessage(textMessage, body);
|
|
||||||
|
|
||||||
return database.insertMessageInbox(masterSecret, textMessage);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private Pair<Long, Long> insertSyncMediaMessage(MasterSecret masterSecret,
|
|
||||||
TextSecureEnvelope envelope,
|
|
||||||
TextSecureMessage message)
|
|
||||||
throws MmsException
|
|
||||||
{
|
|
||||||
MmsDatabase database = DatabaseFactory.getMmsDatabase(context);
|
|
||||||
TextSecureSyncContext syncContext = message.getSyncContext().get();
|
|
||||||
Recipients recipients = getSyncMessageDestination(message);
|
|
||||||
OutgoingMediaMessage mediaMessage = new OutgoingMediaMessage(context, masterSecret, recipients,
|
|
||||||
message.getAttachments().get(),
|
|
||||||
message.getBody().orNull());
|
|
||||||
|
|
||||||
mediaMessage = new OutgoingSecureMediaMessage(mediaMessage);
|
|
||||||
|
|
||||||
long threadId = DatabaseFactory.getThreadDatabase(context).getThreadIdFor(recipients);
|
|
||||||
long messageId = database.insertMessageOutbox(masterSecret, mediaMessage, threadId, false, syncContext.getTimestamp());
|
|
||||||
|
|
||||||
database.markAsSent(messageId, "push".getBytes(), 0);
|
|
||||||
database.markAsPush(messageId);
|
|
||||||
|
|
||||||
return new Pair<>(messageId, threadId);
|
|
||||||
}
|
|
||||||
|
|
||||||
private Pair<Long, Long> insertStandardMediaMessage(MasterSecret masterSecret,
|
|
||||||
TextSecureEnvelope envelope,
|
|
||||||
TextSecureMessage message)
|
|
||||||
throws MmsException
|
|
||||||
{
|
|
||||||
MmsDatabase database = DatabaseFactory.getMmsDatabase(context);
|
|
||||||
String localNumber = TextSecurePreferences.getLocalNumber(context);
|
|
||||||
IncomingMediaMessage mediaMessage = new IncomingMediaMessage(masterSecret, envelope.getSource(),
|
|
||||||
localNumber, message.getTimestamp(),
|
|
||||||
Optional.fromNullable(envelope.getRelay()),
|
|
||||||
message.getBody(),
|
|
||||||
message.getGroupInfo(),
|
|
||||||
message.getAttachments());
|
|
||||||
|
|
||||||
return database.insertSecureDecryptedMessageInbox(masterSecret, mediaMessage, -1);
|
|
||||||
}
|
|
||||||
|
|
||||||
private Recipients getSyncMessageDestination(TextSecureMessage message) {
|
|
||||||
if (message.getGroupInfo().isPresent()) {
|
|
||||||
return RecipientFactory.getRecipientsFromString(context, GroupUtil.getEncodedId(message.getGroupInfo().get().getGroupId()), false);
|
|
||||||
} else {
|
|
||||||
return RecipientFactory.getRecipientsFromString(context, message.getSyncContext().get().getDestination(), false);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -26,12 +26,12 @@ import org.whispersystems.textsecure.api.TextSecureMessageSender;
|
|||||||
import org.whispersystems.textsecure.api.crypto.UntrustedIdentityException;
|
import org.whispersystems.textsecure.api.crypto.UntrustedIdentityException;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureAttachment;
|
import org.whispersystems.textsecure.api.messages.TextSecureAttachment;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureGroup;
|
import org.whispersystems.textsecure.api.messages.TextSecureGroup;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureMessage;
|
import org.whispersystems.textsecure.api.messages.TextSecureDataMessage;
|
||||||
import org.whispersystems.textsecure.api.push.TextSecureAddress;
|
import org.whispersystems.textsecure.api.push.TextSecureAddress;
|
||||||
import org.whispersystems.textsecure.api.push.exceptions.EncapsulatedExceptions;
|
import org.whispersystems.textsecure.api.push.exceptions.EncapsulatedExceptions;
|
||||||
import org.whispersystems.textsecure.api.push.exceptions.NetworkFailureException;
|
import org.whispersystems.textsecure.api.push.exceptions.NetworkFailureException;
|
||||||
import org.whispersystems.textsecure.api.util.InvalidNumberException;
|
import org.whispersystems.textsecure.api.util.InvalidNumberException;
|
||||||
import org.whispersystems.textsecure.internal.push.PushMessageProtos;
|
import org.whispersystems.textsecure.internal.push.TextSecureProtos.GroupContext;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.LinkedList;
|
import java.util.LinkedList;
|
||||||
@ -148,18 +148,18 @@ public class PushGroupSendJob extends PushSendJob implements InjectableType {
|
|||||||
String content = PartParser.getMessageText(message.getBody());
|
String content = PartParser.getMessageText(message.getBody());
|
||||||
|
|
||||||
if (content != null && !content.trim().isEmpty()) {
|
if (content != null && !content.trim().isEmpty()) {
|
||||||
PushMessageProtos.PushMessageContent.GroupContext groupContext = PushMessageProtos.PushMessageContent.GroupContext.parseFrom(Base64.decode(content));
|
GroupContext groupContext = GroupContext.parseFrom(Base64.decode(content));
|
||||||
TextSecureAttachment avatar = attachments.isEmpty() ? null : attachments.get(0);
|
TextSecureAttachment avatar = attachments.isEmpty() ? null : attachments.get(0);
|
||||||
TextSecureGroup.Type type = MmsSmsColumns.Types.isGroupQuit(message.getDatabaseMessageBox()) ? TextSecureGroup.Type.QUIT : TextSecureGroup.Type.UPDATE;
|
TextSecureGroup.Type type = MmsSmsColumns.Types.isGroupQuit(message.getDatabaseMessageBox()) ? TextSecureGroup.Type.QUIT : TextSecureGroup.Type.UPDATE;
|
||||||
TextSecureGroup group = new TextSecureGroup(type, groupId, groupContext.getName(), groupContext.getMembersList(), avatar);
|
TextSecureGroup group = new TextSecureGroup(type, groupId, groupContext.getName(), groupContext.getMembersList(), avatar);
|
||||||
TextSecureMessage groupMessage = new TextSecureMessage(message.getSentTimestamp(), group, null, null);
|
TextSecureDataMessage groupMessage = new TextSecureDataMessage(message.getSentTimestamp(), group, null, null);
|
||||||
|
|
||||||
messageSender.sendMessage(addresses, groupMessage);
|
messageSender.sendMessage(addresses, groupMessage);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
String body = PartParser.getMessageText(message.getBody());
|
String body = PartParser.getMessageText(message.getBody());
|
||||||
TextSecureGroup group = new TextSecureGroup(groupId);
|
TextSecureGroup group = new TextSecureGroup(groupId);
|
||||||
TextSecureMessage groupMessage = new TextSecureMessage(message.getSentTimestamp(), group, attachments, body);
|
TextSecureDataMessage groupMessage = new TextSecureDataMessage(message.getSentTimestamp(), group, attachments, body);
|
||||||
|
|
||||||
messageSender.sendMessage(addresses, groupMessage);
|
messageSender.sendMessage(addresses, groupMessage);
|
||||||
}
|
}
|
||||||
|
@ -19,7 +19,7 @@ import org.thoughtcrime.securesms.transport.UndeliverableMessageException;
|
|||||||
import org.whispersystems.textsecure.api.TextSecureMessageSender;
|
import org.whispersystems.textsecure.api.TextSecureMessageSender;
|
||||||
import org.whispersystems.textsecure.api.crypto.UntrustedIdentityException;
|
import org.whispersystems.textsecure.api.crypto.UntrustedIdentityException;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureAttachment;
|
import org.whispersystems.textsecure.api.messages.TextSecureAttachment;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureMessage;
|
import org.whispersystems.textsecure.api.messages.TextSecureDataMessage;
|
||||||
import org.whispersystems.textsecure.api.push.TextSecureAddress;
|
import org.whispersystems.textsecure.api.push.TextSecureAddress;
|
||||||
import org.whispersystems.textsecure.api.push.exceptions.UnregisteredUserException;
|
import org.whispersystems.textsecure.api.push.exceptions.UnregisteredUserException;
|
||||||
import org.whispersystems.textsecure.api.util.InvalidNumberException;
|
import org.whispersystems.textsecure.api.util.InvalidNumberException;
|
||||||
@ -109,7 +109,7 @@ public class PushMediaSendJob extends PushSendJob implements InjectableType {
|
|||||||
TextSecureAddress address = getPushAddress(destination);
|
TextSecureAddress address = getPushAddress(destination);
|
||||||
List<TextSecureAttachment> attachments = getAttachments(masterSecret, message);
|
List<TextSecureAttachment> attachments = getAttachments(masterSecret, message);
|
||||||
String body = PartParser.getMessageText(message.getBody());
|
String body = PartParser.getMessageText(message.getBody());
|
||||||
TextSecureMessage mediaMessage = TextSecureMessage.newBuilder()
|
TextSecureDataMessage mediaMessage = TextSecureDataMessage.newBuilder()
|
||||||
.withBody(body)
|
.withBody(body)
|
||||||
.withAttachments(attachments)
|
.withAttachments(attachments)
|
||||||
.withTimestamp(message.getSentTimestamp())
|
.withTimestamp(message.getSentTimestamp())
|
||||||
|
@ -18,7 +18,7 @@ import org.thoughtcrime.securesms.transport.InsecureFallbackApprovalException;
|
|||||||
import org.thoughtcrime.securesms.transport.RetryLaterException;
|
import org.thoughtcrime.securesms.transport.RetryLaterException;
|
||||||
import org.whispersystems.textsecure.api.TextSecureMessageSender;
|
import org.whispersystems.textsecure.api.TextSecureMessageSender;
|
||||||
import org.whispersystems.textsecure.api.crypto.UntrustedIdentityException;
|
import org.whispersystems.textsecure.api.crypto.UntrustedIdentityException;
|
||||||
import org.whispersystems.textsecure.api.messages.TextSecureMessage;
|
import org.whispersystems.textsecure.api.messages.TextSecureDataMessage;
|
||||||
import org.whispersystems.textsecure.api.push.TextSecureAddress;
|
import org.whispersystems.textsecure.api.push.TextSecureAddress;
|
||||||
import org.whispersystems.textsecure.api.push.exceptions.UnregisteredUserException;
|
import org.whispersystems.textsecure.api.push.exceptions.UnregisteredUserException;
|
||||||
import org.whispersystems.textsecure.api.util.InvalidNumberException;
|
import org.whispersystems.textsecure.api.util.InvalidNumberException;
|
||||||
@ -103,7 +103,7 @@ public class PushTextSendJob extends PushSendJob implements InjectableType {
|
|||||||
try {
|
try {
|
||||||
TextSecureAddress address = getPushAddress(message.getIndividualRecipient().getNumber());
|
TextSecureAddress address = getPushAddress(message.getIndividualRecipient().getNumber());
|
||||||
TextSecureMessageSender messageSender = messageSenderFactory.create(masterSecret);
|
TextSecureMessageSender messageSender = messageSenderFactory.create(masterSecret);
|
||||||
TextSecureMessage textSecureMessage = TextSecureMessage.newBuilder()
|
TextSecureDataMessage textSecureMessage = TextSecureDataMessage.newBuilder()
|
||||||
.withTimestamp(message.getDateSent())
|
.withTimestamp(message.getDateSent())
|
||||||
.withBody(message.getBody().getBody())
|
.withBody(message.getBody().getBody())
|
||||||
.asEndSessionMessage(message.isEndSession())
|
.asEndSessionMessage(message.isEndSession())
|
||||||
|
@ -5,13 +5,12 @@ import android.content.Context;
|
|||||||
import org.thoughtcrime.securesms.database.ThreadDatabase;
|
import org.thoughtcrime.securesms.database.ThreadDatabase;
|
||||||
import org.thoughtcrime.securesms.recipients.Recipients;
|
import org.thoughtcrime.securesms.recipients.Recipients;
|
||||||
import org.thoughtcrime.securesms.util.Base64;
|
import org.thoughtcrime.securesms.util.Base64;
|
||||||
|
import org.whispersystems.textsecure.internal.push.TextSecureProtos.GroupContext;
|
||||||
|
|
||||||
import ws.com.google.android.mms.ContentType;
|
import ws.com.google.android.mms.ContentType;
|
||||||
import ws.com.google.android.mms.pdu.PduBody;
|
import ws.com.google.android.mms.pdu.PduBody;
|
||||||
import ws.com.google.android.mms.pdu.PduPart;
|
import ws.com.google.android.mms.pdu.PduPart;
|
||||||
|
|
||||||
import static org.whispersystems.textsecure.internal.push.PushMessageProtos.PushMessageContent.GroupContext;
|
|
||||||
|
|
||||||
public class OutgoingGroupMediaMessage extends OutgoingSecureMediaMessage {
|
public class OutgoingGroupMediaMessage extends OutgoingSecureMediaMessage {
|
||||||
|
|
||||||
private final GroupContext group;
|
private final GroupContext group;
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
package org.thoughtcrime.securesms.sms;
|
package org.thoughtcrime.securesms.sms;
|
||||||
|
|
||||||
import static org.whispersystems.textsecure.internal.push.PushMessageProtos.PushMessageContent.GroupContext;
|
import static org.whispersystems.textsecure.internal.push.TextSecureProtos.GroupContext;
|
||||||
|
|
||||||
public class IncomingGroupMessage extends IncomingTextMessage {
|
public class IncomingGroupMessage extends IncomingTextMessage {
|
||||||
|
|
||||||
|
@ -9,7 +9,7 @@ import java.io.IOException;
|
|||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import org.thoughtcrime.securesms.R;
|
import org.thoughtcrime.securesms.R;
|
||||||
import static org.whispersystems.textsecure.internal.push.PushMessageProtos.PushMessageContent.GroupContext;
|
import static org.whispersystems.textsecure.internal.push.TextSecureProtos.GroupContext;
|
||||||
|
|
||||||
public class GroupUtil {
|
public class GroupUtil {
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user