mirror of
				https://github.com/oxen-io/session-android.git
				synced 2025-10-25 12:28:47 +00:00 
			
		
		
		
	Support for multi-device group sync and group requests.
// FREEBIE
This commit is contained in:
		
							
								
								
									
										11
									
								
								build.gradle
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								build.gradle
									
									
									
									
									
								
							| @@ -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.6.0-RC19' |     compile 'org.whispersystems:textsecure-android:1.6.0' | ||||||
|  |  | ||||||
|     compile 'com.squareup.leakcanary:leakcanary-android:1.3.1' |     compile 'com.squareup.leakcanary:leakcanary-android:1.3.1' | ||||||
|  |  | ||||||
| @@ -114,25 +114,28 @@ 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:b5786690a2603ca78eed8a4f829737c41e2b5099695ce02bd44d0a9af3392318', | ||||||
|             '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:textsecure-java:dd32ab5fbb232116e7e533a78dce7b8be168bf561c5774772406aea54a677c0a', | ||||||
|             'org.whispersystems:axolotl-android:40d3db5004a84749a73f68d2f0d01b2ae35a73c54df96d8c6c6723b96efb6fc0', |             'org.whispersystems:axolotl-android:40d3db5004a84749a73f68d2f0d01b2ae35a73c54df96d8c6c6723b96efb6fc0', | ||||||
|             'org.whispersystems:axolotl-java:6daee739b89d8d7101de6d98f77132fee48495c6ea647d880e77def842f999ea', |  | ||||||
|             'org.whispersystems:curve25519-android:3c29a4131a69b0d16baaa3d707678deb39602c3a3ffd75805ce7f9db252e5d0d', |  | ||||||
|             'com.googlecode.libphonenumber:libphonenumber:eba17eae81dd622ea89a00a3a8c025b2f25d342e0d9644c5b62e16f15687c3ab', |             'com.googlecode.libphonenumber:libphonenumber:eba17eae81dd622ea89a00a3a8c025b2f25d342e0d9644c5b62e16f15687c3ab', | ||||||
|             'com.google.protobuf:protobuf-java:e0c1c64575c005601725e7c6a02cebf9e1285e888f756b2a1d73ffa8d725cc74', |             'com.google.protobuf:protobuf-java:e0c1c64575c005601725e7c6a02cebf9e1285e888f756b2a1d73ffa8d725cc74', | ||||||
|             'com.squareup.okhttp:okhttp:89b7f63e2e5b6c410266abc14f50fe52ea8d2d8a57260829e499b1cd9f0e61af', |             'com.squareup.okhttp:okhttp:89b7f63e2e5b6c410266abc14f50fe52ea8d2d8a57260829e499b1cd9f0e61af', | ||||||
|             'com.fasterxml.jackson.core:jackson-databind:835097bcdd11f5bc8a08378c70d4c8054dfa4b911691cc2752063c75534d198d', |             'com.fasterxml.jackson.core:jackson-databind:835097bcdd11f5bc8a08378c70d4c8054dfa4b911691cc2752063c75534d198d', | ||||||
|             'org.whispersystems:curve25519-java:9ccef8f5aba05d9942336f023c589d6278b4f9135bdc34a7bade1f4e7ad65fa3', |             'org.whispersystems:axolotl-java:6daee739b89d8d7101de6d98f77132fee48495c6ea647d880e77def842f999ea', | ||||||
|  |             'org.whispersystems:curve25519-android:3c29a4131a69b0d16baaa3d707678deb39602c3a3ffd75805ce7f9db252e5d0d', | ||||||
|             'com.squareup.okio:okio:5e1098bd3fdee4c3347f5ab815b40ba851e4ab1b348c5e49a5b0362f0ce6e978', |             'com.squareup.okio:okio:5e1098bd3fdee4c3347f5ab815b40ba851e4ab1b348c5e49a5b0362f0ce6e978', | ||||||
|             'com.fasterxml.jackson.core:jackson-annotations:0ca408c24202a7626ec8b861e99d85eca5e38b73311dd6dd12e3e9deecc3fe94', |             'com.fasterxml.jackson.core:jackson-annotations:0ca408c24202a7626ec8b861e99d85eca5e38b73311dd6dd12e3e9deecc3fe94', | ||||||
|             'com.fasterxml.jackson.core:jackson-core:cbf4604784b4de226262845447a1ad3bb38a6728cebe86562e2c5afada8be2c0', |             'com.fasterxml.jackson.core:jackson-core:cbf4604784b4de226262845447a1ad3bb38a6728cebe86562e2c5afada8be2c0', | ||||||
|  |             'org.whispersystems:curve25519-java:9ccef8f5aba05d9942336f023c589d6278b4f9135bdc34a7bade1f4e7ad65fa3', | ||||||
|             'com.android.support:support-v4:1e2e4d35ac7fd30db5ce3bc177b92e4d5af86acef2ef93e9221599d733346f56', |             'com.android.support:support-v4:1e2e4d35ac7fd30db5ce3bc177b92e4d5af86acef2ef93e9221599d733346f56', | ||||||
|             'com.android.support:support-annotations:7bc07519aa613b186001160403bcfd68260fa82c61cc7e83adeedc9b862b94ae', |             'com.android.support:support-annotations:7bc07519aa613b186001160403bcfd68260fa82c61cc7e83adeedc9b862b94ae', | ||||||
|     ] |     ] | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| android { | android { | ||||||
|     compileSdkVersion 22 |     compileSdkVersion 22 | ||||||
|     buildToolsVersion '22.0.1' |     buildToolsVersion '22.0.1' | ||||||
|   | |||||||
| @@ -87,6 +87,11 @@ public class GroupDatabase extends Database { | |||||||
|     return new Reader(cursor); |     return new Reader(cursor); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   public Reader getGroups() { | ||||||
|  |     Cursor cursor = databaseHelper.getReadableDatabase().query(TABLE_NAME, null, null, null, null, null, null); | ||||||
|  |     return new Reader(cursor); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   public Recipients getGroupMembers(byte[] groupId, boolean includeSelf) { |   public Recipients getGroupMembers(byte[] groupId, boolean includeSelf) { | ||||||
|     String          localNumber = TextSecurePreferences.getLocalNumber(context); |     String          localNumber = TextSecurePreferences.getLocalNumber(context); | ||||||
|     List<String>    members     = getCurrentMembers(groupId); |     List<String>    members     = getCurrentMembers(groupId); | ||||||
|   | |||||||
| @@ -11,6 +11,7 @@ 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.MultiDeviceContactUpdateJob; | ||||||
|  | import org.thoughtcrime.securesms.jobs.MultiDeviceGroupUpdateJob; | ||||||
| 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.PushNotificationReceiveJob; | import org.thoughtcrime.securesms.jobs.PushNotificationReceiveJob; | ||||||
| @@ -40,6 +41,7 @@ import dagger.Provides; | |||||||
|                                      MessageRetrievalService.class, |                                      MessageRetrievalService.class, | ||||||
|                                      PushNotificationReceiveJob.class, |                                      PushNotificationReceiveJob.class, | ||||||
|                                      MultiDeviceContactUpdateJob.class, |                                      MultiDeviceContactUpdateJob.class, | ||||||
|  |                                      MultiDeviceGroupUpdateJob.class, | ||||||
|                                      DeviceListActivity.DeviceListFragment.class}) |                                      DeviceListActivity.DeviceListFragment.class}) | ||||||
| public class TextSecureCommunicationModule { | public class TextSecureCommunicationModule { | ||||||
|  |  | ||||||
|   | |||||||
| @@ -23,6 +23,8 @@ import org.whispersystems.textsecure.api.crypto.UntrustedIdentityException; | |||||||
| import org.whispersystems.textsecure.api.messages.TextSecureAttachmentStream; | import org.whispersystems.textsecure.api.messages.TextSecureAttachmentStream; | ||||||
| import org.whispersystems.textsecure.api.messages.multidevice.DeviceContact; | import org.whispersystems.textsecure.api.messages.multidevice.DeviceContact; | ||||||
| import org.whispersystems.textsecure.api.messages.multidevice.DeviceContactsOutputStream; | import org.whispersystems.textsecure.api.messages.multidevice.DeviceContactsOutputStream; | ||||||
|  | import org.whispersystems.textsecure.api.messages.multidevice.TextSecureSyncMessage; | ||||||
|  | import org.whispersystems.textsecure.api.push.exceptions.PushNetworkException; | ||||||
|  |  | ||||||
| import java.io.ByteArrayInputStream; | import java.io.ByteArrayInputStream; | ||||||
| import java.io.File; | import java.io.File; | ||||||
| @@ -79,7 +81,7 @@ public class MultiDeviceContactUpdateJob extends MasterSecretJob implements Inje | |||||||
|  |  | ||||||
|   @Override |   @Override | ||||||
|   public boolean onShouldRetryThrowable(Exception exception) { |   public boolean onShouldRetryThrowable(Exception exception) { | ||||||
|     if (exception instanceof NetworkException) return true; |     if (exception instanceof PushNetworkException) return true; | ||||||
|     return false; |     return false; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -102,7 +104,7 @@ public class MultiDeviceContactUpdateJob extends MasterSecretJob implements Inje | |||||||
|                                                                                    contactsFile.length()); |                                                                                    contactsFile.length()); | ||||||
|  |  | ||||||
|     try { |     try { | ||||||
|       messageSender.sendMultiDeviceContactsUpdate(attachmentStream); |       messageSender.sendMessage(TextSecureSyncMessage.forContacts(attachmentStream)); | ||||||
|     } catch (IOException ioe) { |     } catch (IOException ioe) { | ||||||
|       throw new NetworkException(ioe); |       throw new NetworkException(ioe); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -0,0 +1,117 @@ | |||||||
|  | package org.thoughtcrime.securesms.jobs; | ||||||
|  |  | ||||||
|  | import android.content.Context; | ||||||
|  | import android.support.annotation.Nullable; | ||||||
|  |  | ||||||
|  | import org.thoughtcrime.securesms.crypto.MasterSecret; | ||||||
|  | import org.thoughtcrime.securesms.database.DatabaseFactory; | ||||||
|  | import org.thoughtcrime.securesms.database.GroupDatabase; | ||||||
|  | import org.thoughtcrime.securesms.dependencies.InjectableType; | ||||||
|  | import org.thoughtcrime.securesms.dependencies.TextSecureCommunicationModule; | ||||||
|  | import org.thoughtcrime.securesms.jobs.requirements.MasterSecretRequirement; | ||||||
|  | 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.DeviceGroup; | ||||||
|  | import org.whispersystems.textsecure.api.messages.multidevice.DeviceGroupsOutputStream; | ||||||
|  | import org.whispersystems.textsecure.api.messages.multidevice.TextSecureSyncMessage; | ||||||
|  | import org.whispersystems.textsecure.api.push.exceptions.PushNetworkException; | ||||||
|  |  | ||||||
|  | import java.io.ByteArrayInputStream; | ||||||
|  | import java.io.File; | ||||||
|  | import java.io.FileInputStream; | ||||||
|  | import java.io.FileOutputStream; | ||||||
|  | import java.io.IOException; | ||||||
|  |  | ||||||
|  | import javax.inject.Inject; | ||||||
|  |  | ||||||
|  | public class MultiDeviceGroupUpdateJob extends MasterSecretJob implements InjectableType { | ||||||
|  |  | ||||||
|  |   @Inject | ||||||
|  |   transient TextSecureCommunicationModule.TextSecureMessageSenderFactory messageSenderFactory; | ||||||
|  |  | ||||||
|  |   public MultiDeviceGroupUpdateJob(Context context) { | ||||||
|  |     super(context, JobParameters.newBuilder() | ||||||
|  |                                 .withRequirement(new NetworkRequirement(context)) | ||||||
|  |                                 .withRequirement(new MasterSecretRequirement(context)) | ||||||
|  |                                 .withGroupId(MultiDeviceGroupUpdateJob.class.getSimpleName()) | ||||||
|  |                                 .withPersistence() | ||||||
|  |                                 .create()); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @Override | ||||||
|  |   public void onRun(MasterSecret masterSecret) throws Exception { | ||||||
|  |     TextSecureMessageSender messageSender   = messageSenderFactory.create(masterSecret); | ||||||
|  |     File                    contactDataFile = createTempFile("multidevice-contact-update"); | ||||||
|  |     GroupDatabase.Reader    reader          = null; | ||||||
|  |  | ||||||
|  |     GroupDatabase.GroupRecord record; | ||||||
|  |  | ||||||
|  |     try { | ||||||
|  |       DeviceGroupsOutputStream out = new DeviceGroupsOutputStream(new FileOutputStream(contactDataFile)); | ||||||
|  |  | ||||||
|  |       reader = DatabaseFactory.getGroupDatabase(context).getGroups(); | ||||||
|  |  | ||||||
|  |       while ((record = reader.getNext()) != null) { | ||||||
|  |         out.write(new DeviceGroup(record.getId(), Optional.fromNullable(record.getTitle()), | ||||||
|  |                                   record.getMembers(), getAvatar(record.getAvatar()))); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       out.close(); | ||||||
|  |  | ||||||
|  |       sendUpdate(messageSender, contactDataFile); | ||||||
|  |  | ||||||
|  |     } finally { | ||||||
|  |       if (contactDataFile != null) contactDataFile.delete(); | ||||||
|  |       if (reader != null)          reader.close(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @Override | ||||||
|  |   public boolean onShouldRetryThrowable(Exception exception) { | ||||||
|  |     if (exception instanceof PushNetworkException) return true; | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @Override | ||||||
|  |   public void onAdded() { | ||||||
|  |  | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @Override | ||||||
|  |   public void onCanceled() { | ||||||
|  |  | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private void sendUpdate(TextSecureMessageSender messageSender, File contactsFile) | ||||||
|  |       throws IOException, UntrustedIdentityException | ||||||
|  |   { | ||||||
|  |     FileInputStream            contactsFileStream = new FileInputStream(contactsFile); | ||||||
|  |     TextSecureAttachmentStream attachmentStream   = new TextSecureAttachmentStream(contactsFileStream, | ||||||
|  |                                                                                    "application/octet-stream", | ||||||
|  |                                                                                    contactsFile.length()); | ||||||
|  |  | ||||||
|  |     messageSender.sendMessage(TextSecureSyncMessage.forGroups(attachmentStream)); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   private Optional<TextSecureAttachmentStream> getAvatar(@Nullable byte[] avatar) { | ||||||
|  |     if (avatar == null) return Optional.absent(); | ||||||
|  |  | ||||||
|  |     return Optional.of(new TextSecureAttachmentStream(new ByteArrayInputStream(avatar), | ||||||
|  |                                                       "image/*", avatar.length)); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private File createTempFile(String prefix) throws IOException { | ||||||
|  |     File file = File.createTempFile(prefix, "tmp", context.getCacheDir()); | ||||||
|  |     file.deleteOnExit(); | ||||||
|  |  | ||||||
|  |     return file; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -206,6 +206,12 @@ public class PushDecryptJob extends MasterSecretJob { | |||||||
|                         .getJobManager() |                         .getJobManager() | ||||||
|                         .add(new MultiDeviceContactUpdateJob(getContext())); |                         .add(new MultiDeviceContactUpdateJob(getContext())); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     if (message.isGroupsRequest()) { | ||||||
|  |       ApplicationContext.getInstance(context) | ||||||
|  |                         .getJobManager() | ||||||
|  |                         .add(new MultiDeviceGroupUpdateJob(getContext())); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private void handleMediaMessage(MasterSecret masterSecret, TextSecureEnvelope envelope, |   private void handleMediaMessage(MasterSecret masterSecret, TextSecureEnvelope envelope, | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Moxie Marlinspike
					Moxie Marlinspike