|
|
@ -15,6 +15,7 @@ import org.thoughtcrime.securesms.contacts.ContactAccessor.ContactData;
|
|
|
|
import org.thoughtcrime.securesms.crypto.ProfileKeyUtil;
|
|
|
|
import org.thoughtcrime.securesms.crypto.ProfileKeyUtil;
|
|
|
|
import org.thoughtcrime.securesms.crypto.UnidentifiedAccessUtil;
|
|
|
|
import org.thoughtcrime.securesms.crypto.UnidentifiedAccessUtil;
|
|
|
|
import org.thoughtcrime.securesms.database.Address;
|
|
|
|
import org.thoughtcrime.securesms.database.Address;
|
|
|
|
|
|
|
|
import org.thoughtcrime.securesms.database.Database;
|
|
|
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
|
|
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
|
|
|
import org.thoughtcrime.securesms.database.IdentityDatabase;
|
|
|
|
import org.thoughtcrime.securesms.database.IdentityDatabase;
|
|
|
|
import org.thoughtcrime.securesms.dependencies.InjectableType;
|
|
|
|
import org.thoughtcrime.securesms.dependencies.InjectableType;
|
|
|
@ -38,6 +39,7 @@ import org.whispersystems.signalservice.api.messages.multidevice.SignalServiceSy
|
|
|
|
import org.whispersystems.signalservice.api.messages.multidevice.VerifiedMessage;
|
|
|
|
import org.whispersystems.signalservice.api.messages.multidevice.VerifiedMessage;
|
|
|
|
import org.whispersystems.signalservice.api.push.exceptions.PushNetworkException;
|
|
|
|
import org.whispersystems.signalservice.api.push.exceptions.PushNetworkException;
|
|
|
|
import org.whispersystems.signalservice.api.util.InvalidNumberException;
|
|
|
|
import org.whispersystems.signalservice.api.util.InvalidNumberException;
|
|
|
|
|
|
|
|
import org.whispersystems.signalservice.loki.messaging.LokiThreadFriendRequestStatus;
|
|
|
|
|
|
|
|
|
|
|
|
import java.io.ByteArrayInputStream;
|
|
|
|
import java.io.ByteArrayInputStream;
|
|
|
|
import java.io.File;
|
|
|
|
import java.io.File;
|
|
|
@ -83,7 +85,7 @@ public class MultiDeviceContactUpdateJob extends BaseJob implements InjectableTy
|
|
|
|
.addConstraint(NetworkConstraint.KEY)
|
|
|
|
.addConstraint(NetworkConstraint.KEY)
|
|
|
|
.setQueue("MultiDeviceContactUpdateJob")
|
|
|
|
.setQueue("MultiDeviceContactUpdateJob")
|
|
|
|
.setLifespan(TimeUnit.DAYS.toMillis(1))
|
|
|
|
.setLifespan(TimeUnit.DAYS.toMillis(1))
|
|
|
|
.setMaxAttempts(Parameters.UNLIMITED)
|
|
|
|
.setMaxAttempts(3)
|
|
|
|
.build(),
|
|
|
|
.build(),
|
|
|
|
address,
|
|
|
|
address,
|
|
|
|
forceSync);
|
|
|
|
forceSync);
|
|
|
@ -126,6 +128,9 @@ public class MultiDeviceContactUpdateJob extends BaseJob implements InjectableTy
|
|
|
|
private void generateSingleContactUpdate(@NonNull Address address)
|
|
|
|
private void generateSingleContactUpdate(@NonNull Address address)
|
|
|
|
throws IOException, UntrustedIdentityException, NetworkException
|
|
|
|
throws IOException, UntrustedIdentityException, NetworkException
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
// Loki - Only sync regular contacts
|
|
|
|
|
|
|
|
if (!address.isPhone()) { return; }
|
|
|
|
|
|
|
|
|
|
|
|
File contactDataFile = createTempFile("multidevice-contact-update");
|
|
|
|
File contactDataFile = createTempFile("multidevice-contact-update");
|
|
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
try {
|
|
|
@ -134,16 +139,19 @@ public class MultiDeviceContactUpdateJob extends BaseJob implements InjectableTy
|
|
|
|
Optional<IdentityDatabase.IdentityRecord> identityRecord = DatabaseFactory.getIdentityDatabase(context).getIdentity(address);
|
|
|
|
Optional<IdentityDatabase.IdentityRecord> identityRecord = DatabaseFactory.getIdentityDatabase(context).getIdentity(address);
|
|
|
|
Optional<VerifiedMessage> verifiedMessage = getVerifiedMessage(recipient, identityRecord);
|
|
|
|
Optional<VerifiedMessage> verifiedMessage = getVerifiedMessage(recipient, identityRecord);
|
|
|
|
|
|
|
|
|
|
|
|
out.write(new DeviceContact(address.toPhoneString(),
|
|
|
|
// Loki - Only sync contacts we are friends with
|
|
|
|
Optional.fromNullable(recipient.getName()),
|
|
|
|
if (getFriendRequestStatus(recipient) == LokiThreadFriendRequestStatus.FRIENDS) {
|
|
|
|
getAvatar(recipient.getContactUri()),
|
|
|
|
out.write(new DeviceContact(address.toPhoneString(),
|
|
|
|
Optional.fromNullable(recipient.getColor().serialize()),
|
|
|
|
Optional.fromNullable(recipient.getName()),
|
|
|
|
verifiedMessage,
|
|
|
|
getAvatar(recipient.getContactUri()),
|
|
|
|
Optional.fromNullable(recipient.getProfileKey()),
|
|
|
|
Optional.fromNullable(recipient.getColor().serialize()),
|
|
|
|
recipient.isBlocked(),
|
|
|
|
verifiedMessage,
|
|
|
|
recipient.getExpireMessages() > 0 ?
|
|
|
|
Optional.fromNullable(recipient.getProfileKey()),
|
|
|
|
Optional.of(recipient.getExpireMessages()) :
|
|
|
|
recipient.isBlocked(),
|
|
|
|
Optional.absent()));
|
|
|
|
recipient.getExpireMessages() > 0 ?
|
|
|
|
|
|
|
|
Optional.of(recipient.getExpireMessages()) :
|
|
|
|
|
|
|
|
Optional.absent()));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
out.close();
|
|
|
|
out.close();
|
|
|
|
sendUpdate(messageSender, contactDataFile, false);
|
|
|
|
sendUpdate(messageSender, contactDataFile, false);
|
|
|
@ -158,11 +166,6 @@ public class MultiDeviceContactUpdateJob extends BaseJob implements InjectableTy
|
|
|
|
private void generateFullContactUpdate()
|
|
|
|
private void generateFullContactUpdate()
|
|
|
|
throws IOException, UntrustedIdentityException, NetworkException
|
|
|
|
throws IOException, UntrustedIdentityException, NetworkException
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (!Permissions.hasAny(context, Manifest.permission.READ_CONTACTS, Manifest.permission.WRITE_CONTACTS)) {
|
|
|
|
|
|
|
|
Log.w(TAG, "No contact permissions, skipping multi-device contact update...");
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
boolean isAppVisible = ApplicationContext.getInstance(context).isAppVisible();
|
|
|
|
boolean isAppVisible = ApplicationContext.getInstance(context).isAppVisible();
|
|
|
|
long timeSinceLastSync = System.currentTimeMillis() - TextSecurePreferences.getLastFullContactSyncTime(context);
|
|
|
|
long timeSinceLastSync = System.currentTimeMillis() - TextSecurePreferences.getLastFullContactSyncTime(context);
|
|
|
|
|
|
|
|
|
|
|
@ -189,13 +192,16 @@ public class MultiDeviceContactUpdateJob extends BaseJob implements InjectableTy
|
|
|
|
Recipient recipient = Recipient.from(context, address, false);
|
|
|
|
Recipient recipient = Recipient.from(context, address, false);
|
|
|
|
Optional<IdentityDatabase.IdentityRecord> identity = DatabaseFactory.getIdentityDatabase(context).getIdentity(address);
|
|
|
|
Optional<IdentityDatabase.IdentityRecord> identity = DatabaseFactory.getIdentityDatabase(context).getIdentity(address);
|
|
|
|
Optional<VerifiedMessage> verified = getVerifiedMessage(recipient, identity);
|
|
|
|
Optional<VerifiedMessage> verified = getVerifiedMessage(recipient, identity);
|
|
|
|
Optional<String> name = Optional.fromNullable(contactData.name);
|
|
|
|
Optional<String> name = Optional.fromNullable(DatabaseFactory.getLokiUserDatabase(context).getDisplayName(address.serialize()));
|
|
|
|
Optional<String> color = Optional.of(recipient.getColor().serialize());
|
|
|
|
Optional<String> color = Optional.of(recipient.getColor().serialize());
|
|
|
|
Optional<byte[]> profileKey = Optional.fromNullable(recipient.getProfileKey());
|
|
|
|
Optional<byte[]> profileKey = Optional.fromNullable(recipient.getProfileKey());
|
|
|
|
boolean blocked = recipient.isBlocked();
|
|
|
|
boolean blocked = recipient.isBlocked();
|
|
|
|
Optional<Integer> expireTimer = recipient.getExpireMessages() > 0 ? Optional.of(recipient.getExpireMessages()) : Optional.absent();
|
|
|
|
Optional<Integer> expireTimer = recipient.getExpireMessages() > 0 ? Optional.of(recipient.getExpireMessages()) : Optional.absent();
|
|
|
|
|
|
|
|
|
|
|
|
out.write(new DeviceContact(address.toPhoneString(), name, getAvatar(contactUri), color, verified, profileKey, blocked, expireTimer));
|
|
|
|
// Loki - Only sync contacts we are friends with
|
|
|
|
|
|
|
|
if (getFriendRequestStatus(recipient) == LokiThreadFriendRequestStatus.FRIENDS) {
|
|
|
|
|
|
|
|
out.write(new DeviceContact(address.toPhoneString(), name, getAvatar(contactUri), color, verified, profileKey, blocked, expireTimer));
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (ProfileKeyUtil.hasProfileKey(context)) {
|
|
|
|
if (ProfileKeyUtil.hasProfileKey(context)) {
|
|
|
@ -216,6 +222,11 @@ public class MultiDeviceContactUpdateJob extends BaseJob implements InjectableTy
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private LokiThreadFriendRequestStatus getFriendRequestStatus(Recipient recipient) {
|
|
|
|
|
|
|
|
long threadId = DatabaseFactory.getThreadDatabase(context).getThreadIdIfExistsFor(recipient);
|
|
|
|
|
|
|
|
return DatabaseFactory.getLokiThreadDatabase(context).getFriendRequestStatus(threadId);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
@Override
|
|
|
|
public boolean onShouldRetry(@NonNull Exception exception) {
|
|
|
|
public boolean onShouldRetry(@NonNull Exception exception) {
|
|
|
|
if (exception instanceof PushNetworkException) return true;
|
|
|
|
if (exception instanceof PushNetworkException) return true;
|
|
|
@ -239,7 +250,6 @@ public class MultiDeviceContactUpdateJob extends BaseJob implements InjectableTy
|
|
|
|
.build();
|
|
|
|
.build();
|
|
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
try {
|
|
|
|
// TODO: Message ID
|
|
|
|
|
|
|
|
messageSender.sendMessage(0, SignalServiceSyncMessage.forContacts(new ContactsMessage(attachmentStream, complete)),
|
|
|
|
messageSender.sendMessage(0, SignalServiceSyncMessage.forContacts(new ContactsMessage(attachmentStream, complete)),
|
|
|
|
UnidentifiedAccessUtil.getAccessForSync(context));
|
|
|
|
UnidentifiedAccessUtil.getAccessForSync(context));
|
|
|
|
} catch (IOException ioe) {
|
|
|
|
} catch (IOException ioe) {
|
|
|
@ -319,6 +329,9 @@ public class MultiDeviceContactUpdateJob extends BaseJob implements InjectableTy
|
|
|
|
default: throw new AssertionError("Unknown state: " + identity.get().getVerifiedStatus());
|
|
|
|
default: throw new AssertionError("Unknown state: " + identity.get().getVerifiedStatus());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Loki - For now always set to verified
|
|
|
|
|
|
|
|
state = VerifiedMessage.VerifiedState.VERIFIED;
|
|
|
|
|
|
|
|
|
|
|
|
return Optional.of(new VerifiedMessage(destination, identityKey, state, System.currentTimeMillis()));
|
|
|
|
return Optional.of(new VerifiedMessage(destination, identityKey, state, System.currentTimeMillis()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|