mirror of
https://github.com/oxen-io/session-android.git
synced 2024-12-25 01:07:47 +00:00
Move "directory" information into RecipientPreferencesDatabase
// FREEBIE
This commit is contained in:
parent
a02f223a96
commit
5a5e47f2df
@ -54,8 +54,7 @@ import org.thoughtcrime.securesms.database.Address;
|
|||||||
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.GroupDatabase.GroupRecord;
|
import org.thoughtcrime.securesms.database.GroupDatabase.GroupRecord;
|
||||||
import org.thoughtcrime.securesms.database.NotInDirectoryException;
|
import org.thoughtcrime.securesms.database.RecipientPreferenceDatabase.RecipientsPreferences;
|
||||||
import org.thoughtcrime.securesms.database.TextSecureDirectory;
|
|
||||||
import org.thoughtcrime.securesms.database.ThreadDatabase;
|
import org.thoughtcrime.securesms.database.ThreadDatabase;
|
||||||
import org.thoughtcrime.securesms.groups.GroupManager;
|
import org.thoughtcrime.securesms.groups.GroupManager;
|
||||||
import org.thoughtcrime.securesms.groups.GroupManager.GroupActionResult;
|
import org.thoughtcrime.securesms.groups.GroupManager.GroupActionResult;
|
||||||
@ -170,11 +169,8 @@ public class GroupCreateActivity extends PassphraseRequiredActionBarActivity
|
|||||||
}
|
}
|
||||||
|
|
||||||
private static boolean isActiveInDirectory(Context context, Recipient recipient) {
|
private static boolean isActiveInDirectory(Context context, Recipient recipient) {
|
||||||
try {
|
Optional<RecipientsPreferences> preferences = DatabaseFactory.getRecipientPreferenceDatabase(context).getRecipientsPreferences(recipient.getAddress());
|
||||||
return TextSecureDirectory.getInstance(context).isSecureTextSupported(recipient.getAddress());
|
return preferences.isPresent() && preferences.get().isRegistered();
|
||||||
} catch (NotInDirectoryException e) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void addSelectedContacts(@NonNull Recipient... recipients) {
|
private void addSelectedContacts(@NonNull Recipient... recipients) {
|
||||||
|
@ -27,15 +27,18 @@ import android.provider.ContactsContract.CommonDataKinds.Phone;
|
|||||||
import android.provider.ContactsContract.Contacts;
|
import android.provider.ContactsContract.Contacts;
|
||||||
import android.provider.ContactsContract.PhoneLookup;
|
import android.provider.ContactsContract.PhoneLookup;
|
||||||
import android.telephony.PhoneNumberUtils;
|
import android.telephony.PhoneNumberUtils;
|
||||||
|
import android.text.TextUtils;
|
||||||
|
|
||||||
|
import org.thoughtcrime.securesms.database.Address;
|
||||||
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.TextSecureDirectory;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
|
import java.util.HashSet;
|
||||||
import java.util.LinkedList;
|
import java.util.LinkedList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
import static org.thoughtcrime.securesms.database.GroupDatabase.GroupRecord;
|
import static org.thoughtcrime.securesms.database.GroupDatabase.GroupRecord;
|
||||||
|
|
||||||
@ -61,6 +64,20 @@ public class ContactAccessor {
|
|||||||
return instance;
|
return instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Set<Address> getAllContactsWithNumbers(Context context) {
|
||||||
|
Set<Address> results = new HashSet<>();
|
||||||
|
|
||||||
|
try (Cursor cursor = context.getContentResolver().query(Phone.CONTENT_URI, new String[] {Phone.NUMBER}, null ,null, null)) {
|
||||||
|
while (cursor != null && cursor.moveToNext()) {
|
||||||
|
if (!TextUtils.isEmpty(cursor.getString(0))) {
|
||||||
|
results.add(Address.fromExternal(context, cursor.getString(0)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
|
||||||
public boolean isSystemContact(Context context, String number) {
|
public boolean isSystemContact(Context context, String number) {
|
||||||
Uri uri = Uri.withAppendedPath(PhoneLookup.CONTENT_FILTER_URI, Uri.encode(number));
|
Uri uri = Uri.withAppendedPath(PhoneLookup.CONTENT_FILTER_URI, Uri.encode(number));
|
||||||
String[] projection = new String[]{PhoneLookup.DISPLAY_NAME, PhoneLookup.LOOKUP_KEY,
|
String[] projection = new String[]{PhoneLookup.DISPLAY_NAME, PhoneLookup.LOOKUP_KEY,
|
||||||
@ -82,16 +99,17 @@ public class ContactAccessor {
|
|||||||
final ContentResolver resolver = context.getContentResolver();
|
final ContentResolver resolver = context.getContentResolver();
|
||||||
final String[] inProjection = new String[]{PhoneLookup._ID, PhoneLookup.DISPLAY_NAME};
|
final String[] inProjection = new String[]{PhoneLookup._ID, PhoneLookup.DISPLAY_NAME};
|
||||||
|
|
||||||
List<String> pushNumbers = TextSecureDirectory.getInstance(context).getActiveNumbers();
|
final List<Address> registeredAddresses = DatabaseFactory.getRecipientPreferenceDatabase(context).getRegistered();
|
||||||
final Collection<ContactData> lookupData = new ArrayList<>(pushNumbers.size());
|
final Collection<ContactData> lookupData = new ArrayList<>(registeredAddresses.size());
|
||||||
|
|
||||||
for (String pushNumber : pushNumbers) {
|
for (Address registeredAddress : registeredAddresses) {
|
||||||
Uri uri = Uri.withAppendedPath(PhoneLookup.CONTENT_FILTER_URI, Uri.encode(pushNumber));
|
Uri uri = Uri.withAppendedPath(PhoneLookup.CONTENT_FILTER_URI, Uri.encode(registeredAddress.serialize()));
|
||||||
Cursor lookupCursor = resolver.query(uri, inProjection, null, null, null);
|
Cursor lookupCursor = resolver.query(uri, inProjection, null, null, null);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if (lookupCursor != null && lookupCursor.moveToFirst()) {
|
if (lookupCursor != null && lookupCursor.moveToFirst()) {
|
||||||
final ContactData contactData = new ContactData(lookupCursor.getLong(0), lookupCursor.getString(1));
|
final ContactData contactData = new ContactData(lookupCursor.getLong(0), lookupCursor.getString(1));
|
||||||
contactData.numbers.add(new NumberData("TextSecure", pushNumber));
|
contactData.numbers.add(new NumberData("TextSecure", registeredAddress.serialize()));
|
||||||
lookupData.add(contactData);
|
lookupData.add(contactData);
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
@ -99,6 +117,7 @@ public class ContactAccessor {
|
|||||||
lookupCursor.close();
|
lookupCursor.close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return lookupData;
|
return lookupData;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -101,7 +101,8 @@ public class DatabaseFactory {
|
|||||||
private static final int SANIFY_ATTACHMENT_DOWNLOAD = 36;
|
private static final int SANIFY_ATTACHMENT_DOWNLOAD = 36;
|
||||||
private static final int NO_MORE_CANONICAL_ADDRESS_DATABASE = 37;
|
private static final int NO_MORE_CANONICAL_ADDRESS_DATABASE = 37;
|
||||||
private static final int NO_MORE_RECIPIENTS_PLURAL = 38;
|
private static final int NO_MORE_RECIPIENTS_PLURAL = 38;
|
||||||
private static final int DATABASE_VERSION = 38;
|
private static final int INTERNAL_DIRECTORY = 39;
|
||||||
|
private static final int DATABASE_VERSION = 39;
|
||||||
|
|
||||||
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();
|
||||||
@ -1273,6 +1274,24 @@ public class DatabaseFactory {
|
|||||||
if (cursor != null) cursor.close();
|
if (cursor != null) cursor.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (oldVersion < INTERNAL_DIRECTORY) {
|
||||||
|
db.execSQL("ALTER TABLE recipient_preferences ADD COLUMN registered INTEGER DEFAULT 0");
|
||||||
|
|
||||||
|
DatabaseHelper directoryDatabaseHelper = new DatabaseHelper(context, "whisper_directory.db", null, 5);
|
||||||
|
SQLiteDatabase directoryDatabase = directoryDatabaseHelper.getReadableDatabase();
|
||||||
|
|
||||||
|
Cursor cursor = directoryDatabase.query("directory", new String[] {"number", "registered"}, null, null, null, null, null);
|
||||||
|
|
||||||
|
while (cursor != null && cursor.moveToNext()) {
|
||||||
|
String address = new NumberMigrator(TextSecurePreferences.getLocalNumber(context)).migrate(cursor.getString(0));
|
||||||
|
ContentValues contentValues = new ContentValues(1);
|
||||||
|
|
||||||
|
contentValues.put("recipient_ids", address);
|
||||||
|
contentValues.put("registered", cursor.getInt(1) == 1);
|
||||||
|
db.replace("recipient_preferences", null, contentValues);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
db.setTransactionSuccessful();
|
db.setTransactionSuccessful();
|
||||||
db.endTransaction();
|
db.endTransaction();
|
||||||
}
|
}
|
||||||
|
@ -18,7 +18,10 @@ import org.thoughtcrime.securesms.recipients.Recipient;
|
|||||||
import org.thoughtcrime.securesms.recipients.RecipientFactory;
|
import org.thoughtcrime.securesms.recipients.RecipientFactory;
|
||||||
import org.whispersystems.libsignal.util.guava.Optional;
|
import org.whispersystems.libsignal.util.guava.Optional;
|
||||||
|
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.LinkedList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
public class RecipientPreferenceDatabase extends Database {
|
public class RecipientPreferenceDatabase extends Database {
|
||||||
|
|
||||||
@ -36,9 +39,10 @@ public class RecipientPreferenceDatabase extends Database {
|
|||||||
private static final String SEEN_INVITE_REMINDER = "seen_invite_reminder";
|
private static final String SEEN_INVITE_REMINDER = "seen_invite_reminder";
|
||||||
private static final String DEFAULT_SUBSCRIPTION_ID = "default_subscription_id";
|
private static final String DEFAULT_SUBSCRIPTION_ID = "default_subscription_id";
|
||||||
private static final String EXPIRE_MESSAGES = "expire_messages";
|
private static final String EXPIRE_MESSAGES = "expire_messages";
|
||||||
|
private static final String REGISTERED = "registered";
|
||||||
|
|
||||||
private static final String[] RECIPIENT_PROJECTION = new String[] {
|
private static final String[] RECIPIENT_PROJECTION = new String[] {
|
||||||
BLOCK, NOTIFICATION, VIBRATE, MUTE_UNTIL, COLOR, SEEN_INVITE_REMINDER, DEFAULT_SUBSCRIPTION_ID, EXPIRE_MESSAGES
|
BLOCK, NOTIFICATION, VIBRATE, MUTE_UNTIL, COLOR, SEEN_INVITE_REMINDER, DEFAULT_SUBSCRIPTION_ID, EXPIRE_MESSAGES, REGISTERED
|
||||||
};
|
};
|
||||||
|
|
||||||
static final List<String> TYPED_RECIPIENT_PROJECTION = Stream.of(RECIPIENT_PROJECTION)
|
static final List<String> TYPED_RECIPIENT_PROJECTION = Stream.of(RECIPIENT_PROJECTION)
|
||||||
@ -74,7 +78,8 @@ public class RecipientPreferenceDatabase extends Database {
|
|||||||
COLOR + " TEXT DEFAULT NULL, " +
|
COLOR + " TEXT DEFAULT NULL, " +
|
||||||
SEEN_INVITE_REMINDER + " INTEGER DEFAULT 0, " +
|
SEEN_INVITE_REMINDER + " INTEGER DEFAULT 0, " +
|
||||||
DEFAULT_SUBSCRIPTION_ID + " INTEGER DEFAULT -1, " +
|
DEFAULT_SUBSCRIPTION_ID + " INTEGER DEFAULT -1, " +
|
||||||
EXPIRE_MESSAGES + " INTEGER DEFAULT 0);";
|
EXPIRE_MESSAGES + " INTEGER DEFAULT 0, " +
|
||||||
|
REGISTERED + " INTEGER DEFAULT 0);";
|
||||||
|
|
||||||
public RecipientPreferenceDatabase(Context context, SQLiteOpenHelper databaseHelper) {
|
public RecipientPreferenceDatabase(Context context, SQLiteOpenHelper databaseHelper) {
|
||||||
super(context, databaseHelper);
|
super(context, databaseHelper);
|
||||||
@ -122,6 +127,7 @@ public class RecipientPreferenceDatabase extends Database {
|
|||||||
boolean seenInviteReminder = cursor.getInt(cursor.getColumnIndexOrThrow(SEEN_INVITE_REMINDER)) == 1;
|
boolean seenInviteReminder = cursor.getInt(cursor.getColumnIndexOrThrow(SEEN_INVITE_REMINDER)) == 1;
|
||||||
int defaultSubscriptionId = cursor.getInt(cursor.getColumnIndexOrThrow(DEFAULT_SUBSCRIPTION_ID));
|
int defaultSubscriptionId = cursor.getInt(cursor.getColumnIndexOrThrow(DEFAULT_SUBSCRIPTION_ID));
|
||||||
int expireMessages = cursor.getInt(cursor.getColumnIndexOrThrow(EXPIRE_MESSAGES));
|
int expireMessages = cursor.getInt(cursor.getColumnIndexOrThrow(EXPIRE_MESSAGES));
|
||||||
|
boolean registered = cursor.getInt(cursor.getColumnIndexOrThrow(REGISTERED)) == 1;
|
||||||
|
|
||||||
MaterialColor color;
|
MaterialColor color;
|
||||||
|
|
||||||
@ -135,7 +141,7 @@ public class RecipientPreferenceDatabase extends Database {
|
|||||||
return new RecipientsPreferences(blocked, muteUntil,
|
return new RecipientsPreferences(blocked, muteUntil,
|
||||||
VibrateState.fromId(vibrateState),
|
VibrateState.fromId(vibrateState),
|
||||||
notificationUri, color, seenInviteReminder,
|
notificationUri, color, seenInviteReminder,
|
||||||
defaultSubscriptionId, expireMessages);
|
defaultSubscriptionId, expireMessages, registered);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setColor(Recipient recipient, MaterialColor color) {
|
public void setColor(Recipient recipient, MaterialColor color) {
|
||||||
@ -190,6 +196,56 @@ public class RecipientPreferenceDatabase extends Database {
|
|||||||
updateOrInsert(recipient, values);
|
updateOrInsert(recipient, values);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Set<Address> getAllRecipients() {
|
||||||
|
SQLiteDatabase db = databaseHelper.getReadableDatabase();
|
||||||
|
Set<Address> results = new HashSet<>();
|
||||||
|
|
||||||
|
try (Cursor cursor = db.query(TABLE_NAME, new String[] {ADDRESS}, null, null, null, null, null)) {
|
||||||
|
while (cursor != null && cursor.moveToNext()) {
|
||||||
|
results.add(Address.fromExternal(context, cursor.getString(0)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setRegistered(@NonNull List<Address> activeAddresses,
|
||||||
|
@NonNull List<Address> inactiveAddresses)
|
||||||
|
{
|
||||||
|
SQLiteDatabase db = databaseHelper.getWritableDatabase();
|
||||||
|
|
||||||
|
for (Address activeAddress : activeAddresses) {
|
||||||
|
ContentValues contentValues = new ContentValues(2);
|
||||||
|
contentValues.put(ADDRESS, activeAddress.serialize());
|
||||||
|
contentValues.put(REGISTERED, 1);
|
||||||
|
|
||||||
|
db.replace(TABLE_NAME, null, contentValues);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (Address inactiveAddress : inactiveAddresses) {
|
||||||
|
ContentValues contentValues = new ContentValues(2);
|
||||||
|
contentValues.put(ADDRESS, inactiveAddress.serialize());
|
||||||
|
contentValues.put(REGISTERED, 0);
|
||||||
|
|
||||||
|
db.replace(TABLE_NAME, null, contentValues);
|
||||||
|
}
|
||||||
|
|
||||||
|
context.getContentResolver().notifyChange(Uri.parse(RECIPIENT_PREFERENCES_URI), null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<Address> getRegistered() {
|
||||||
|
SQLiteDatabase db = databaseHelper.getReadableDatabase();
|
||||||
|
List<Address> results = new LinkedList<>();
|
||||||
|
|
||||||
|
try (Cursor cursor = db.query(TABLE_NAME, new String[] {ADDRESS}, REGISTERED + " = ?", new String[] {"1"}, null, null, null)) {
|
||||||
|
while (cursor != null && cursor.moveToNext()) {
|
||||||
|
results.add(Address.fromSerialized(cursor.getString(0)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
|
||||||
private void updateOrInsert(Recipient recipient, ContentValues contentValues) {
|
private void updateOrInsert(Recipient recipient, ContentValues contentValues) {
|
||||||
SQLiteDatabase database = databaseHelper.getWritableDatabase();
|
SQLiteDatabase database = databaseHelper.getWritableDatabase();
|
||||||
|
|
||||||
@ -218,6 +274,7 @@ public class RecipientPreferenceDatabase extends Database {
|
|||||||
private final boolean seenInviteReminder;
|
private final boolean seenInviteReminder;
|
||||||
private final int defaultSubscriptionId;
|
private final int defaultSubscriptionId;
|
||||||
private final int expireMessages;
|
private final int expireMessages;
|
||||||
|
private final boolean registered;
|
||||||
|
|
||||||
RecipientsPreferences(boolean blocked, long muteUntil,
|
RecipientsPreferences(boolean blocked, long muteUntil,
|
||||||
@NonNull VibrateState vibrateState,
|
@NonNull VibrateState vibrateState,
|
||||||
@ -225,7 +282,8 @@ public class RecipientPreferenceDatabase extends Database {
|
|||||||
@Nullable MaterialColor color,
|
@Nullable MaterialColor color,
|
||||||
boolean seenInviteReminder,
|
boolean seenInviteReminder,
|
||||||
int defaultSubscriptionId,
|
int defaultSubscriptionId,
|
||||||
int expireMessages)
|
int expireMessages,
|
||||||
|
boolean registered)
|
||||||
{
|
{
|
||||||
this.blocked = blocked;
|
this.blocked = blocked;
|
||||||
this.muteUntil = muteUntil;
|
this.muteUntil = muteUntil;
|
||||||
@ -235,6 +293,7 @@ public class RecipientPreferenceDatabase extends Database {
|
|||||||
this.seenInviteReminder = seenInviteReminder;
|
this.seenInviteReminder = seenInviteReminder;
|
||||||
this.defaultSubscriptionId = defaultSubscriptionId;
|
this.defaultSubscriptionId = defaultSubscriptionId;
|
||||||
this.expireMessages = expireMessages;
|
this.expireMessages = expireMessages;
|
||||||
|
this.registered = registered;
|
||||||
}
|
}
|
||||||
|
|
||||||
public @Nullable MaterialColor getColor() {
|
public @Nullable MaterialColor getColor() {
|
||||||
@ -268,6 +327,10 @@ public class RecipientPreferenceDatabase extends Database {
|
|||||||
public int getExpireMessages() {
|
public int getExpireMessages() {
|
||||||
return expireMessages;
|
return expireMessages;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean isRegistered() {
|
||||||
|
return registered;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class BlockedReader {
|
public static class BlockedReader {
|
||||||
|
@ -1,295 +0,0 @@
|
|||||||
package org.thoughtcrime.securesms.database;
|
|
||||||
|
|
||||||
import android.content.ContentValues;
|
|
||||||
import android.content.Context;
|
|
||||||
import android.database.Cursor;
|
|
||||||
import android.database.sqlite.SQLiteDatabase;
|
|
||||||
import android.database.sqlite.SQLiteOpenHelper;
|
|
||||||
import android.net.Uri;
|
|
||||||
import android.provider.ContactsContract.CommonDataKinds.Phone;
|
|
||||||
import android.support.annotation.NonNull;
|
|
||||||
import android.text.TextUtils;
|
|
||||||
import android.util.Log;
|
|
||||||
|
|
||||||
import org.whispersystems.signalservice.api.push.ContactTokenDetails;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Collection;
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Set;
|
|
||||||
|
|
||||||
public class TextSecureDirectory {
|
|
||||||
|
|
||||||
private static final int INTRODUCED_CHANGE_FROM_TOKEN_TO_E164_NUMBER = 2;
|
|
||||||
private static final int INTRODUCED_VOICE_COLUMN = 4;
|
|
||||||
private static final int INTRODUCED_VIDEO_COLUMN = 5;
|
|
||||||
|
|
||||||
private static final String DATABASE_NAME = "whisper_directory.db";
|
|
||||||
private static final int DATABASE_VERSION = 5;
|
|
||||||
|
|
||||||
private static final String TABLE_NAME = "directory";
|
|
||||||
private static final String ID = "_id";
|
|
||||||
private static final String NUMBER = "number";
|
|
||||||
private static final String REGISTERED = "registered";
|
|
||||||
private static final String RELAY = "relay";
|
|
||||||
private static final String TIMESTAMP = "timestamp";
|
|
||||||
private static final String VOICE = "voice";
|
|
||||||
private static final String VIDEO = "video";
|
|
||||||
|
|
||||||
private static final String CREATE_TABLE = "CREATE TABLE " + TABLE_NAME + "(" + ID + " INTEGER PRIMARY KEY, " +
|
|
||||||
NUMBER + " TEXT UNIQUE, " +
|
|
||||||
REGISTERED + " INTEGER, " +
|
|
||||||
RELAY + " TEXT, " +
|
|
||||||
TIMESTAMP + " INTEGER, " +
|
|
||||||
VOICE + " INTEGER, " +
|
|
||||||
VIDEO + " INTEGER);";
|
|
||||||
|
|
||||||
private static final Object instanceLock = new Object();
|
|
||||||
private static volatile TextSecureDirectory instance;
|
|
||||||
|
|
||||||
public static TextSecureDirectory getInstance(Context context) {
|
|
||||||
if (instance == null) {
|
|
||||||
synchronized (instanceLock) {
|
|
||||||
if (instance == null) {
|
|
||||||
instance = new TextSecureDirectory(context.getApplicationContext());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return instance;
|
|
||||||
}
|
|
||||||
|
|
||||||
private final DatabaseHelper databaseHelper;
|
|
||||||
private final Context context;
|
|
||||||
|
|
||||||
private TextSecureDirectory(Context context) {
|
|
||||||
this.context = context;
|
|
||||||
this.databaseHelper = new DatabaseHelper(context, DATABASE_NAME, null, DATABASE_VERSION);
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isSecureTextSupported(@NonNull Address address) throws NotInDirectoryException {
|
|
||||||
if (address.isEmail()) return false;
|
|
||||||
if (address.isGroup()) return true;
|
|
||||||
|
|
||||||
SQLiteDatabase db = databaseHelper.getReadableDatabase();
|
|
||||||
Cursor cursor = null;
|
|
||||||
|
|
||||||
try {
|
|
||||||
cursor = db.query(TABLE_NAME,
|
|
||||||
new String[]{REGISTERED}, NUMBER + " = ?",
|
|
||||||
new String[] {address.serialize()}, null, null, null);
|
|
||||||
|
|
||||||
if (cursor != null && cursor.moveToFirst()) {
|
|
||||||
return cursor.getInt(0) == 1;
|
|
||||||
} else {
|
|
||||||
throw new NotInDirectoryException();
|
|
||||||
}
|
|
||||||
|
|
||||||
} finally {
|
|
||||||
if (cursor != null)
|
|
||||||
cursor.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// public boolean isSecureVoiceSupported(String e164number) throws NotInDirectoryException {
|
|
||||||
// if (TextUtils.isEmpty(e164number)) {
|
|
||||||
// return false;
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// SQLiteDatabase db = databaseHelper.getReadableDatabase();
|
|
||||||
// Cursor cursor = null;
|
|
||||||
//
|
|
||||||
// try {
|
|
||||||
// cursor = db.query(TABLE_NAME,
|
|
||||||
// new String[]{VOICE}, NUMBER + " = ?",
|
|
||||||
// new String[] {e164number}, null, null, null);
|
|
||||||
//
|
|
||||||
// if (cursor != null && cursor.moveToFirst()) {
|
|
||||||
// return cursor.getInt(0) == 1;
|
|
||||||
// } else {
|
|
||||||
// throw new NotInDirectoryException();
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// } finally {
|
|
||||||
// if (cursor != null)
|
|
||||||
// cursor.close();
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
// public boolean isSecureVideoSupported(String e164number) throws NotInDirectoryException {
|
|
||||||
// if (TextUtils.isEmpty(e164number)) {
|
|
||||||
// return false;
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// SQLiteDatabase db = databaseHelper.getReadableDatabase();
|
|
||||||
// Cursor cursor = null;
|
|
||||||
//
|
|
||||||
// try {
|
|
||||||
// cursor = db.query(TABLE_NAME,
|
|
||||||
// new String[]{VIDEO}, NUMBER + " = ?",
|
|
||||||
// new String[] {e164number}, null, null, null);
|
|
||||||
//
|
|
||||||
// if (cursor != null && cursor.moveToFirst()) {
|
|
||||||
// return cursor.getInt(0) == 1;
|
|
||||||
// } else {
|
|
||||||
// throw new NotInDirectoryException();
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// } finally {
|
|
||||||
// if (cursor != null)
|
|
||||||
// cursor.close();
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
public String getRelay(String e164number) {
|
|
||||||
SQLiteDatabase database = databaseHelper.getReadableDatabase();
|
|
||||||
Cursor cursor = null;
|
|
||||||
|
|
||||||
try {
|
|
||||||
cursor = database.query(TABLE_NAME, null, NUMBER + " = ?", new String[]{e164number}, null, null, null);
|
|
||||||
|
|
||||||
if (cursor != null && cursor.moveToFirst()) {
|
|
||||||
return cursor.getString(cursor.getColumnIndexOrThrow(RELAY));
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
} finally {
|
|
||||||
if (cursor != null)
|
|
||||||
cursor.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setNumber(ContactTokenDetails token, boolean active) {
|
|
||||||
SQLiteDatabase db = databaseHelper.getWritableDatabase();
|
|
||||||
ContentValues values = new ContentValues();
|
|
||||||
values.put(NUMBER, token.getNumber());
|
|
||||||
values.put(REGISTERED, active ? 1 : 0);
|
|
||||||
values.put(TIMESTAMP, System.currentTimeMillis());
|
|
||||||
values.put(RELAY, token.getRelay());
|
|
||||||
values.put(VOICE, token.isVoice());
|
|
||||||
values.put(VIDEO, token.isVideo());
|
|
||||||
db.replace(TABLE_NAME, null, values);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setNumbers(List<ContactTokenDetails> activeTokens, Collection<Address> inactiveAddresses) {
|
|
||||||
long timestamp = System.currentTimeMillis();
|
|
||||||
SQLiteDatabase db = databaseHelper.getWritableDatabase();
|
|
||||||
db.beginTransaction();
|
|
||||||
|
|
||||||
try {
|
|
||||||
for (ContactTokenDetails token : activeTokens) {
|
|
||||||
Log.w("Directory", "Adding active token: " + token.getNumber() + ", " + token.getToken() + ", video: " + token.isVideo());
|
|
||||||
ContentValues values = new ContentValues();
|
|
||||||
values.put(NUMBER, token.getNumber());
|
|
||||||
values.put(REGISTERED, 1);
|
|
||||||
values.put(TIMESTAMP, timestamp);
|
|
||||||
values.put(RELAY, token.getRelay());
|
|
||||||
values.put(VOICE, token.isVoice());
|
|
||||||
values.put(VIDEO, token.isVideo());
|
|
||||||
db.replace(TABLE_NAME, null, values);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (Address address : inactiveAddresses) {
|
|
||||||
ContentValues values = new ContentValues();
|
|
||||||
values.put(NUMBER, address.serialize());
|
|
||||||
values.put(REGISTERED, 0);
|
|
||||||
values.put(TIMESTAMP, timestamp);
|
|
||||||
db.replace(TABLE_NAME, null, values);
|
|
||||||
}
|
|
||||||
|
|
||||||
db.setTransactionSuccessful();
|
|
||||||
} finally {
|
|
||||||
db.endTransaction();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public Set<Address> getPushEligibleContactNumbers() {
|
|
||||||
final Uri uri = Phone.CONTENT_URI;
|
|
||||||
final Set<Address> results = new HashSet<>();
|
|
||||||
Cursor cursor = null;
|
|
||||||
|
|
||||||
try {
|
|
||||||
cursor = context.getContentResolver().query(uri, new String[] {Phone.NUMBER}, null, null, null);
|
|
||||||
|
|
||||||
while (cursor != null && cursor.moveToNext()) {
|
|
||||||
final String rawNumber = cursor.getString(0);
|
|
||||||
if (!TextUtils.isEmpty(rawNumber)) {
|
|
||||||
results.add(Address.fromExternal(context, rawNumber));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cursor != null)
|
|
||||||
cursor.close();
|
|
||||||
|
|
||||||
final SQLiteDatabase readableDb = databaseHelper.getReadableDatabase();
|
|
||||||
if (readableDb != null) {
|
|
||||||
cursor = readableDb.query(TABLE_NAME, new String[]{NUMBER},
|
|
||||||
null, null, null, null, null);
|
|
||||||
|
|
||||||
while (cursor != null && cursor.moveToNext()) {
|
|
||||||
results.add(Address.fromSerialized(cursor.getString(0)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return results;
|
|
||||||
} finally {
|
|
||||||
if (cursor != null)
|
|
||||||
cursor.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<String> getActiveNumbers() {
|
|
||||||
final List<String> results = new ArrayList<>();
|
|
||||||
Cursor cursor = null;
|
|
||||||
try {
|
|
||||||
cursor = databaseHelper.getReadableDatabase().query(TABLE_NAME, new String[]{NUMBER},
|
|
||||||
REGISTERED + " = 1", null, null, null, null);
|
|
||||||
|
|
||||||
while (cursor != null && cursor.moveToNext()) {
|
|
||||||
results.add(cursor.getString(0));
|
|
||||||
}
|
|
||||||
return results;
|
|
||||||
} finally {
|
|
||||||
if (cursor != null)
|
|
||||||
cursor.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static class DatabaseHelper extends SQLiteOpenHelper {
|
|
||||||
|
|
||||||
public DatabaseHelper(Context context, String name,
|
|
||||||
SQLiteDatabase.CursorFactory factory,
|
|
||||||
int version)
|
|
||||||
{
|
|
||||||
super(context, name, factory, version);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onCreate(SQLiteDatabase db) {
|
|
||||||
db.execSQL(CREATE_TABLE);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
|
|
||||||
if (oldVersion < INTRODUCED_CHANGE_FROM_TOKEN_TO_E164_NUMBER) {
|
|
||||||
db.execSQL("DROP TABLE directory;");
|
|
||||||
db.execSQL("CREATE TABLE directory ( _id INTEGER PRIMARY KEY, " +
|
|
||||||
"number TEXT UNIQUE, " +
|
|
||||||
"registered INTEGER, " +
|
|
||||||
"relay TEXT, " +
|
|
||||||
"supports_sms INTEGER, " +
|
|
||||||
"timestamp INTEGER);");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (oldVersion < INTRODUCED_VOICE_COLUMN) {
|
|
||||||
db.execSQL("ALTER TABLE directory ADD COLUMN voice INTEGER;");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (oldVersion < INTRODUCED_VIDEO_COLUMN) {
|
|
||||||
db.execSQL("ALTER TABLE directory ADD COLUMN video INTEGER;");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -3,16 +3,10 @@ package org.thoughtcrime.securesms.jobs;
|
|||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
|
|
||||||
import org.thoughtcrime.securesms.ApplicationContext;
|
|
||||||
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
|
||||||
import org.thoughtcrime.securesms.util.TextSecurePreferences;
|
import org.thoughtcrime.securesms.util.TextSecurePreferences;
|
||||||
import org.whispersystems.jobqueue.JobManager;
|
|
||||||
import org.whispersystems.jobqueue.JobParameters;
|
import org.whispersystems.jobqueue.JobParameters;
|
||||||
import org.whispersystems.libsignal.InvalidVersionException;
|
import org.whispersystems.libsignal.InvalidVersionException;
|
||||||
import org.whispersystems.signalservice.api.messages.SignalServiceEnvelope;
|
import org.whispersystems.signalservice.api.messages.SignalServiceEnvelope;
|
||||||
import org.thoughtcrime.securesms.database.TextSecureDirectory;
|
|
||||||
import org.thoughtcrime.securesms.database.NotInDirectoryException;
|
|
||||||
import org.whispersystems.signalservice.api.push.ContactTokenDetails;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
|
||||||
|
@ -7,15 +7,17 @@ import org.thoughtcrime.securesms.ApplicationContext;
|
|||||||
import org.thoughtcrime.securesms.database.Address;
|
import org.thoughtcrime.securesms.database.Address;
|
||||||
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
||||||
import org.thoughtcrime.securesms.database.MessagingDatabase.SyncMessageId;
|
import org.thoughtcrime.securesms.database.MessagingDatabase.SyncMessageId;
|
||||||
import org.thoughtcrime.securesms.database.NotInDirectoryException;
|
import org.thoughtcrime.securesms.database.RecipientPreferenceDatabase.RecipientsPreferences;
|
||||||
import org.thoughtcrime.securesms.database.TextSecureDirectory;
|
|
||||||
import org.thoughtcrime.securesms.recipients.Recipient;
|
import org.thoughtcrime.securesms.recipients.Recipient;
|
||||||
import org.thoughtcrime.securesms.recipients.RecipientFactory;
|
import org.thoughtcrime.securesms.recipients.RecipientFactory;
|
||||||
import org.thoughtcrime.securesms.service.KeyCachingService;
|
import org.thoughtcrime.securesms.service.KeyCachingService;
|
||||||
|
import org.thoughtcrime.securesms.util.Util;
|
||||||
import org.whispersystems.jobqueue.JobManager;
|
import org.whispersystems.jobqueue.JobManager;
|
||||||
import org.whispersystems.jobqueue.JobParameters;
|
import org.whispersystems.jobqueue.JobParameters;
|
||||||
|
import org.whispersystems.libsignal.util.guava.Optional;
|
||||||
import org.whispersystems.signalservice.api.messages.SignalServiceEnvelope;
|
import org.whispersystems.signalservice.api.messages.SignalServiceEnvelope;
|
||||||
import org.whispersystems.signalservice.api.push.ContactTokenDetails;
|
|
||||||
|
import java.util.LinkedList;
|
||||||
|
|
||||||
public abstract class PushReceivedJob extends ContextJob {
|
public abstract class PushReceivedJob extends ContextJob {
|
||||||
|
|
||||||
@ -29,12 +31,7 @@ public abstract class PushReceivedJob extends ContextJob {
|
|||||||
Address source = Address.fromExternal(context, envelope.getSource());
|
Address source = Address.fromExternal(context, envelope.getSource());
|
||||||
|
|
||||||
if (!isActiveNumber(context, source)) {
|
if (!isActiveNumber(context, source)) {
|
||||||
TextSecureDirectory directory = TextSecureDirectory.getInstance(context);
|
DatabaseFactory.getRecipientPreferenceDatabase(context).setRegistered(Util.asList(source), new LinkedList<>());
|
||||||
ContactTokenDetails contactTokenDetails = new ContactTokenDetails();
|
|
||||||
contactTokenDetails.setNumber(envelope.getSource());
|
|
||||||
|
|
||||||
directory.setNumber(contactTokenDetails, true);
|
|
||||||
|
|
||||||
Recipient recipient = RecipientFactory.getRecipientFor(context, source, false);
|
Recipient recipient = RecipientFactory.getRecipientFor(context, source, false);
|
||||||
ApplicationContext.getInstance(context).getJobManager().add(new DirectoryRefreshJob(context, KeyCachingService.getMasterSecret(context), recipient));
|
ApplicationContext.getInstance(context).getJobManager().add(new DirectoryRefreshJob(context, KeyCachingService.getMasterSecret(context), recipient));
|
||||||
}
|
}
|
||||||
@ -73,15 +70,8 @@ public abstract class PushReceivedJob extends ContextJob {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private boolean isActiveNumber(Context context, Address address) {
|
private boolean isActiveNumber(Context context, Address address) {
|
||||||
boolean isActiveNumber;
|
Optional<RecipientsPreferences> preferences = DatabaseFactory.getRecipientPreferenceDatabase(context).getRecipientsPreferences(address);
|
||||||
|
return preferences.isPresent() && preferences.get().isRegistered();
|
||||||
try {
|
|
||||||
isActiveNumber = TextSecureDirectory.getInstance(context).isSecureTextSupported(address);
|
|
||||||
} catch (NotInDirectoryException e) {
|
|
||||||
isActiveNumber = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return isActiveNumber;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -10,7 +10,6 @@ import org.thoughtcrime.securesms.attachments.Attachment;
|
|||||||
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
||||||
import org.thoughtcrime.securesms.database.Address;
|
import org.thoughtcrime.securesms.database.Address;
|
||||||
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
||||||
import org.thoughtcrime.securesms.database.TextSecureDirectory;
|
|
||||||
import org.thoughtcrime.securesms.events.PartProgressEvent;
|
import org.thoughtcrime.securesms.events.PartProgressEvent;
|
||||||
import org.thoughtcrime.securesms.jobs.requirements.MasterSecretRequirement;
|
import org.thoughtcrime.securesms.jobs.requirements.MasterSecretRequirement;
|
||||||
import org.thoughtcrime.securesms.mms.PartAuthority;
|
import org.thoughtcrime.securesms.mms.PartAuthority;
|
||||||
@ -62,7 +61,8 @@ public abstract class PushSendJob extends SendJob {
|
|||||||
}
|
}
|
||||||
|
|
||||||
protected SignalServiceAddress getPushAddress(Address address) {
|
protected SignalServiceAddress getPushAddress(Address address) {
|
||||||
String relay = TextSecureDirectory.getInstance(context).getRelay(address.toPhoneString());
|
// String relay = TextSecureDirectory.getInstance(context).getRelay(address.toPhoneString());
|
||||||
|
String relay = null;
|
||||||
return new SignalServiceAddress(address.toPhoneString(), Optional.fromNullable(relay));
|
return new SignalServiceAddress(address.toPhoneString(), Optional.fromNullable(relay));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -27,9 +27,9 @@ import org.thoughtcrime.securesms.database.Address;
|
|||||||
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
||||||
import org.thoughtcrime.securesms.database.EncryptingSmsDatabase;
|
import org.thoughtcrime.securesms.database.EncryptingSmsDatabase;
|
||||||
import org.thoughtcrime.securesms.database.MmsDatabase;
|
import org.thoughtcrime.securesms.database.MmsDatabase;
|
||||||
import org.thoughtcrime.securesms.database.NotInDirectoryException;
|
import org.thoughtcrime.securesms.database.RecipientPreferenceDatabase;
|
||||||
|
import org.thoughtcrime.securesms.database.RecipientPreferenceDatabase.RecipientsPreferences;
|
||||||
import org.thoughtcrime.securesms.database.SmsDatabase;
|
import org.thoughtcrime.securesms.database.SmsDatabase;
|
||||||
import org.thoughtcrime.securesms.database.TextSecureDirectory;
|
|
||||||
import org.thoughtcrime.securesms.database.ThreadDatabase;
|
import org.thoughtcrime.securesms.database.ThreadDatabase;
|
||||||
import org.thoughtcrime.securesms.database.model.MessageRecord;
|
import org.thoughtcrime.securesms.database.model.MessageRecord;
|
||||||
import org.thoughtcrime.securesms.jobs.MmsSendJob;
|
import org.thoughtcrime.securesms.jobs.MmsSendJob;
|
||||||
@ -50,6 +50,7 @@ import org.whispersystems.signalservice.api.SignalServiceAccountManager;
|
|||||||
import org.whispersystems.signalservice.api.push.ContactTokenDetails;
|
import org.whispersystems.signalservice.api.push.ContactTokenDetails;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.util.LinkedList;
|
||||||
|
|
||||||
public class MessageSender {
|
public class MessageSender {
|
||||||
|
|
||||||
@ -264,23 +265,21 @@ public class MessageSender {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private static boolean isPushDestination(Context context, Address destination) {
|
private static boolean isPushDestination(Context context, Address destination) {
|
||||||
TextSecureDirectory directory = TextSecureDirectory.getInstance(context);
|
RecipientPreferenceDatabase recipientsDatabase = DatabaseFactory.getRecipientPreferenceDatabase(context);
|
||||||
|
Optional<RecipientsPreferences> recipientPreferences = recipientsDatabase.getRecipientsPreferences(destination);
|
||||||
|
|
||||||
try {
|
if (recipientPreferences.isPresent()) {
|
||||||
return directory.isSecureTextSupported(destination);
|
return recipientPreferences.get().isRegistered();
|
||||||
} catch (NotInDirectoryException e) {
|
} else {
|
||||||
try {
|
try {
|
||||||
SignalServiceAccountManager accountManager = AccountManagerFactory.createManager(context);
|
SignalServiceAccountManager accountManager = AccountManagerFactory.createManager(context);
|
||||||
Optional<ContactTokenDetails> registeredUser = accountManager.getContact(destination.serialize());
|
Optional<ContactTokenDetails> registeredUser = accountManager.getContact(destination.serialize());
|
||||||
|
|
||||||
if (!registeredUser.isPresent()) {
|
if (!registeredUser.isPresent()) {
|
||||||
registeredUser = Optional.of(new ContactTokenDetails());
|
recipientsDatabase.setRegistered(new LinkedList<>(), Util.asList(destination));
|
||||||
registeredUser.get().setNumber(destination.serialize());
|
|
||||||
directory.setNumber(registeredUser.get(), false);
|
|
||||||
return false;
|
return false;
|
||||||
} else {
|
} else {
|
||||||
registeredUser.get().setNumber(destination.toPhoneString());
|
recipientsDatabase.setRegistered(Util.asList(destination), new LinkedList<>());
|
||||||
directory.setNumber(registeredUser.get(), true);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
} catch (IOException e1) {
|
} catch (IOException e1) {
|
||||||
|
@ -12,15 +12,19 @@ import android.support.annotation.Nullable;
|
|||||||
import android.text.TextUtils;
|
import android.text.TextUtils;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
|
|
||||||
|
import com.annimon.stream.Collectors;
|
||||||
|
import com.annimon.stream.Stream;
|
||||||
|
|
||||||
import org.thoughtcrime.securesms.ApplicationContext;
|
import org.thoughtcrime.securesms.ApplicationContext;
|
||||||
import org.thoughtcrime.securesms.R;
|
import org.thoughtcrime.securesms.R;
|
||||||
|
import org.thoughtcrime.securesms.contacts.ContactAccessor;
|
||||||
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
||||||
import org.thoughtcrime.securesms.crypto.SessionUtil;
|
import org.thoughtcrime.securesms.crypto.SessionUtil;
|
||||||
import org.thoughtcrime.securesms.database.Address;
|
import org.thoughtcrime.securesms.database.Address;
|
||||||
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
||||||
import org.thoughtcrime.securesms.database.MessagingDatabase.InsertResult;
|
import org.thoughtcrime.securesms.database.MessagingDatabase.InsertResult;
|
||||||
import org.thoughtcrime.securesms.database.NotInDirectoryException;
|
import org.thoughtcrime.securesms.database.RecipientPreferenceDatabase;
|
||||||
import org.thoughtcrime.securesms.database.TextSecureDirectory;
|
import org.thoughtcrime.securesms.database.RecipientPreferenceDatabase.RecipientsPreferences;
|
||||||
import org.thoughtcrime.securesms.jobs.MultiDeviceContactUpdateJob;
|
import org.thoughtcrime.securesms.jobs.MultiDeviceContactUpdateJob;
|
||||||
import org.thoughtcrime.securesms.notifications.MessageNotifier;
|
import org.thoughtcrime.securesms.notifications.MessageNotifier;
|
||||||
import org.thoughtcrime.securesms.push.AccountManagerFactory;
|
import org.thoughtcrime.securesms.push.AccountManagerFactory;
|
||||||
@ -44,6 +48,7 @@ public class DirectoryHelper {
|
|||||||
|
|
||||||
public static final UserCapabilities UNKNOWN = new UserCapabilities(Capability.UNKNOWN, Capability.UNKNOWN, Capability.UNKNOWN);
|
public static final UserCapabilities UNKNOWN = new UserCapabilities(Capability.UNKNOWN, Capability.UNKNOWN, Capability.UNKNOWN);
|
||||||
public static final UserCapabilities UNSUPPORTED = new UserCapabilities(Capability.UNSUPPORTED, Capability.UNSUPPORTED, Capability.UNSUPPORTED);
|
public static final UserCapabilities UNSUPPORTED = new UserCapabilities(Capability.UNSUPPORTED, Capability.UNSUPPORTED, Capability.UNSUPPORTED);
|
||||||
|
public static final UserCapabilities SUPPORTED = new UserCapabilities(Capability.SUPPORTED, Capability.SUPPORTED, Capability.SUPPORTED);
|
||||||
|
|
||||||
public enum Capability {
|
public enum Capability {
|
||||||
UNKNOWN, SUPPORTED, UNSUPPORTED
|
UNKNOWN, SUPPORTED, UNSUPPORTED
|
||||||
@ -96,29 +101,31 @@ public class DirectoryHelper {
|
|||||||
throws IOException
|
throws IOException
|
||||||
{
|
{
|
||||||
if (TextUtils.isEmpty(TextSecurePreferences.getLocalNumber(context))) {
|
if (TextUtils.isEmpty(TextSecurePreferences.getLocalNumber(context))) {
|
||||||
return new RefreshResult(new LinkedList<Address>(), false);
|
return new RefreshResult(new LinkedList<>(), false);
|
||||||
}
|
}
|
||||||
|
|
||||||
TextSecureDirectory directory = TextSecureDirectory.getInstance(context);
|
RecipientPreferenceDatabase recipientPreferenceDatabase = DatabaseFactory.getRecipientPreferenceDatabase(context);
|
||||||
Set<Address> eligibleContactNumbers = directory.getPushEligibleContactNumbers();
|
Set<Address> eligibleContactNumbers = recipientPreferenceDatabase.getAllRecipients();
|
||||||
Set<String> serializedAddresses = new HashSet<>();
|
eligibleContactNumbers.addAll(ContactAccessor.getInstance().getAllContactsWithNumbers(context));
|
||||||
|
|
||||||
for (Address address : eligibleContactNumbers) {
|
Set<String> serializedAddress = Stream.of(eligibleContactNumbers).map(Address::serialize).collect(Collectors.toSet());
|
||||||
serializedAddresses.add(address.serialize());
|
List<ContactTokenDetails> activeTokens = accountManager.getContacts(serializedAddress);
|
||||||
}
|
|
||||||
|
|
||||||
List<ContactTokenDetails> activeTokens = accountManager.getContacts(serializedAddresses);
|
|
||||||
|
|
||||||
if (activeTokens != null) {
|
if (activeTokens != null) {
|
||||||
|
List<Address> activeAddresses = new LinkedList<>();
|
||||||
|
Set<Address> inactiveAddresses = new HashSet<>(eligibleContactNumbers);
|
||||||
|
|
||||||
for (ContactTokenDetails activeToken : activeTokens) {
|
for (ContactTokenDetails activeToken : activeTokens) {
|
||||||
eligibleContactNumbers.remove(Address.fromSerialized(activeToken.getNumber()));
|
Address activeAddress = Address.fromSerialized(activeToken.getNumber());
|
||||||
|
activeAddresses.add(activeAddress);
|
||||||
|
inactiveAddresses.remove(activeAddress);
|
||||||
}
|
}
|
||||||
|
|
||||||
directory.setNumbers(activeTokens, eligibleContactNumbers);
|
recipientPreferenceDatabase.setRegistered(activeAddresses, new LinkedList<>(inactiveAddresses));
|
||||||
return updateContactsDatabase(context, activeTokens, true);
|
return updateContactsDatabase(context, activeTokens, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
return new RefreshResult(new LinkedList<Address>(), false);
|
return new RefreshResult(new LinkedList<>(), false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UserCapabilities refreshDirectoryFor(@NonNull Context context,
|
public static UserCapabilities refreshDirectoryFor(@NonNull Context context,
|
||||||
@ -126,13 +133,13 @@ public class DirectoryHelper {
|
|||||||
@NonNull Recipient recipient)
|
@NonNull Recipient recipient)
|
||||||
throws IOException
|
throws IOException
|
||||||
{
|
{
|
||||||
TextSecureDirectory directory = TextSecureDirectory.getInstance(context);
|
RecipientPreferenceDatabase recipientDatabase = DatabaseFactory.getRecipientPreferenceDatabase(context);
|
||||||
SignalServiceAccountManager accountManager = AccountManagerFactory.createManager(context);
|
SignalServiceAccountManager accountManager = AccountManagerFactory.createManager(context);
|
||||||
String number = recipient.getAddress().serialize();
|
String number = recipient.getAddress().serialize();
|
||||||
Optional<ContactTokenDetails> details = accountManager.getContact(number);
|
Optional<ContactTokenDetails> details = accountManager.getContact(number);
|
||||||
|
|
||||||
if (details.isPresent()) {
|
if (details.isPresent()) {
|
||||||
directory.setNumber(details.get(), true);
|
recipientDatabase.setRegistered(Util.asList(recipient.getAddress()), new LinkedList<>());
|
||||||
|
|
||||||
RefreshResult result = updateContactsDatabase(context, details.get());
|
RefreshResult result = updateContactsDatabase(context, details.get());
|
||||||
|
|
||||||
@ -148,9 +155,7 @@ public class DirectoryHelper {
|
|||||||
details.get().isVoice() ? Capability.SUPPORTED : Capability.UNSUPPORTED,
|
details.get().isVoice() ? Capability.SUPPORTED : Capability.UNSUPPORTED,
|
||||||
details.get().isVideo() ? Capability.SUPPORTED : Capability.UNSUPPORTED);
|
details.get().isVideo() ? Capability.SUPPORTED : Capability.UNSUPPORTED);
|
||||||
} else {
|
} else {
|
||||||
ContactTokenDetails absent = new ContactTokenDetails();
|
recipientDatabase.setRegistered(new LinkedList<>(), Util.asList(recipient.getAddress()));
|
||||||
absent.setNumber(number);
|
|
||||||
directory.setNumber(absent, false);
|
|
||||||
return UserCapabilities.UNSUPPORTED;
|
return UserCapabilities.UNSUPPORTED;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -158,34 +163,28 @@ public class DirectoryHelper {
|
|||||||
public static @NonNull UserCapabilities getUserCapabilities(@NonNull Context context,
|
public static @NonNull UserCapabilities getUserCapabilities(@NonNull Context context,
|
||||||
@Nullable Recipient recipient)
|
@Nullable Recipient recipient)
|
||||||
{
|
{
|
||||||
try {
|
if (recipient == null) {
|
||||||
if (recipient == null) {
|
return UserCapabilities.UNSUPPORTED;
|
||||||
return UserCapabilities.UNSUPPORTED;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!TextSecurePreferences.isPushRegistered(context)) {
|
|
||||||
return UserCapabilities.UNSUPPORTED;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (recipient.isMmsGroupRecipient()) {
|
|
||||||
return UserCapabilities.UNSUPPORTED;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (recipient.isPushGroupRecipient()) {
|
|
||||||
return new UserCapabilities(Capability.SUPPORTED, Capability.UNSUPPORTED, Capability.UNSUPPORTED);
|
|
||||||
}
|
|
||||||
|
|
||||||
final Address address = recipient.getAddress();
|
|
||||||
|
|
||||||
boolean secureText = TextSecureDirectory.getInstance(context).isSecureTextSupported(address);
|
|
||||||
|
|
||||||
return new UserCapabilities(secureText ? Capability.SUPPORTED : Capability.UNSUPPORTED,
|
|
||||||
secureText ? Capability.SUPPORTED : Capability.UNSUPPORTED,
|
|
||||||
secureText ? Capability.SUPPORTED : Capability.UNSUPPORTED);
|
|
||||||
|
|
||||||
} catch (NotInDirectoryException e) {
|
|
||||||
return UserCapabilities.UNKNOWN;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!TextSecurePreferences.isPushRegistered(context)) {
|
||||||
|
return UserCapabilities.UNSUPPORTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (recipient.isMmsGroupRecipient()) {
|
||||||
|
return UserCapabilities.UNSUPPORTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (recipient.isPushGroupRecipient()) {
|
||||||
|
return new UserCapabilities(Capability.SUPPORTED, Capability.UNSUPPORTED, Capability.UNSUPPORTED);
|
||||||
|
}
|
||||||
|
|
||||||
|
final RecipientPreferenceDatabase recipientDatabase = DatabaseFactory.getRecipientPreferenceDatabase(context);
|
||||||
|
final Optional<RecipientsPreferences> recipientPreferences = recipientDatabase.getRecipientsPreferences(recipient.getAddress());
|
||||||
|
|
||||||
|
if (recipientPreferences.isPresent() && recipientPreferences.get().isRegistered()) return UserCapabilities.SUPPORTED;
|
||||||
|
else if (recipientPreferences.isPresent()) return UserCapabilities.UNSUPPORTED;
|
||||||
|
else return UserCapabilities.UNKNOWN;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static @NonNull RefreshResult updateContactsDatabase(@NonNull Context context,
|
private static @NonNull RefreshResult updateContactsDatabase(@NonNull Context context,
|
||||||
|
@ -75,6 +75,12 @@ public class Util {
|
|||||||
|
|
||||||
public static Handler handler = new Handler(Looper.getMainLooper());
|
public static Handler handler = new Handler(Looper.getMainLooper());
|
||||||
|
|
||||||
|
public static <T> List<T> asList(T... elements) {
|
||||||
|
List<T> result = new LinkedList<>();
|
||||||
|
Collections.addAll(result, elements);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
public static String join(String[] list, String delimiter) {
|
public static String join(String[] list, String delimiter) {
|
||||||
return join(Arrays.asList(list), delimiter);
|
return join(Arrays.asList(list), delimiter);
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user