mirror of
https://github.com/oxen-io/session-android.git
synced 2024-11-25 11:05:25 +00:00
0ace469d74
1) In addition to the Recipient interface, there is now RecipientDevice. A Recipient can have multiple corresponding RecipientDevices. All addressing is done to a Recipient, but crypto sessions and transport delivery are done to RecipientDevice. 2) The Push transport handles the discovery and session setup of additional Recipient devices. 3) Some internal rejiggering of Groups.
182 lines
6.7 KiB
Java
182 lines
6.7 KiB
Java
/**
|
|
* Copyright (C) 2011 Whisper Systems
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
package org.thoughtcrime.securesms.recipients;
|
|
|
|
import android.content.Context;
|
|
import android.database.Cursor;
|
|
import android.graphics.Bitmap;
|
|
import android.graphics.BitmapFactory;
|
|
import android.net.Uri;
|
|
import android.provider.ContactsContract;
|
|
import android.provider.ContactsContract.Contacts;
|
|
import android.provider.ContactsContract.PhoneLookup;
|
|
import android.util.Log;
|
|
|
|
import org.thoughtcrime.securesms.contacts.ContactPhotoFactory;
|
|
import org.thoughtcrime.securesms.database.CanonicalAddressDatabase;
|
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
|
import org.thoughtcrime.securesms.database.GroupDatabase;
|
|
import org.thoughtcrime.securesms.util.GroupUtil;
|
|
import org.thoughtcrime.securesms.util.LRUCache;
|
|
import org.whispersystems.textsecure.util.ListenableFutureTask;
|
|
import org.thoughtcrime.securesms.util.Util;
|
|
|
|
import java.io.IOException;
|
|
import java.io.InputStream;
|
|
import java.util.Collections;
|
|
import java.util.Map;
|
|
import java.util.concurrent.Callable;
|
|
import java.util.concurrent.ExecutorService;
|
|
|
|
public class RecipientProvider {
|
|
|
|
private static final Map<Long,Recipient> recipientCache = Collections.synchronizedMap(new LRUCache<Long,Recipient>(1000));
|
|
private static final ExecutorService asyncRecipientResolver = Util.newSingleThreadedLifoExecutor();
|
|
|
|
private static final String[] CALLER_ID_PROJECTION = new String[] {
|
|
PhoneLookup.DISPLAY_NAME,
|
|
PhoneLookup.LOOKUP_KEY,
|
|
PhoneLookup._ID,
|
|
};
|
|
|
|
public Recipient getRecipient(Context context, long recipientId, boolean asynchronous) {
|
|
Recipient cachedRecipient = recipientCache.get(recipientId);
|
|
|
|
if (cachedRecipient != null) return cachedRecipient;
|
|
else if (asynchronous) return getAsynchronousRecipient(context, recipientId);
|
|
else return getSynchronousRecipient(context, recipientId);
|
|
}
|
|
|
|
private Recipient getSynchronousRecipient(Context context, long recipientId) {
|
|
Log.w("RecipientProvider", "Cache miss [SYNC]!");
|
|
|
|
Recipient recipient;
|
|
RecipientDetails details;
|
|
|
|
String number = CanonicalAddressDatabase.getInstance(context).getAddressFromId(String.valueOf(recipientId));
|
|
|
|
if (GroupUtil.isEncodedGroup(number)) details = getGroupRecipientDetails(context, number);
|
|
else details = getRecipientDetails(context, number);
|
|
|
|
if (details != null) {
|
|
recipient = new Recipient(details.name, number, recipientId, details.contactUri, details.avatar);
|
|
} else {
|
|
recipient = new Recipient(null, number, recipientId, null, ContactPhotoFactory.getDefaultContactPhoto(context));
|
|
}
|
|
|
|
recipientCache.put(recipientId, recipient);
|
|
return recipient;
|
|
}
|
|
|
|
private Recipient getAsynchronousRecipient(final Context context, final long recipientId) {
|
|
Log.w("RecipientProvider", "Cache miss [ASYNC]!");
|
|
|
|
final String number = CanonicalAddressDatabase.getInstance(context).getAddressFromId(String.valueOf(recipientId));
|
|
|
|
Callable<RecipientDetails> task = new Callable<RecipientDetails>() {
|
|
@Override
|
|
public RecipientDetails call() throws Exception {
|
|
if (GroupUtil.isEncodedGroup(number)) return getGroupRecipientDetails(context, number);
|
|
else return getRecipientDetails(context, number);
|
|
}
|
|
};
|
|
|
|
ListenableFutureTask<RecipientDetails> future = new ListenableFutureTask<RecipientDetails>(task, null);
|
|
|
|
asyncRecipientResolver.submit(future);
|
|
|
|
Recipient recipient = new Recipient(number, ContactPhotoFactory.getDefaultContactPhoto(context), recipientId, future);
|
|
recipientCache.put(recipientId, recipient);
|
|
|
|
return recipient;
|
|
}
|
|
|
|
public void clearCache() {
|
|
recipientCache.clear();
|
|
}
|
|
|
|
private RecipientDetails getRecipientDetails(Context context, String number) {
|
|
Uri uri = Uri.withAppendedPath(PhoneLookup.CONTENT_FILTER_URI, Uri.encode(number));
|
|
Cursor cursor = context.getContentResolver().query(uri, CALLER_ID_PROJECTION,
|
|
null, null, null);
|
|
|
|
try {
|
|
if (cursor != null && cursor.moveToFirst()) {
|
|
Uri contactUri = Contacts.getLookupUri(cursor.getLong(2), cursor.getString(1));
|
|
Bitmap contactPhoto = getContactPhoto(context, Uri.withAppendedPath(Contacts.CONTENT_URI,
|
|
cursor.getLong(2)+""));
|
|
|
|
return new RecipientDetails(cursor.getString(0), contactUri, contactPhoto);
|
|
}
|
|
} finally {
|
|
if (cursor != null)
|
|
cursor.close();
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
private RecipientDetails getGroupRecipientDetails(Context context, String groupId) {
|
|
try {
|
|
GroupDatabase.Reader reader = DatabaseFactory.getGroupDatabase(context)
|
|
.getGroup(GroupUtil.getDecodedId(groupId));
|
|
|
|
GroupDatabase.GroupRecord record;
|
|
|
|
try {
|
|
if ((record = reader.getNext()) != null) {
|
|
byte[] avatarBytes = record.getAvatar();
|
|
Bitmap avatar;
|
|
|
|
if (avatarBytes == null) avatar = ContactPhotoFactory.getDefaultContactPhoto(context);
|
|
else avatar = BitmapFactory.decodeByteArray(avatarBytes, 0, avatarBytes.length);
|
|
|
|
return new RecipientDetails(record.getTitle(), null, avatar);
|
|
}
|
|
} finally {
|
|
reader.close();
|
|
}
|
|
|
|
return null;
|
|
} catch (IOException e) {
|
|
Log.w("RecipientProvider", e);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
private Bitmap getContactPhoto(Context context, Uri uri) {
|
|
InputStream inputStream = ContactsContract.Contacts.openContactPhotoInputStream(context.getContentResolver(), uri);
|
|
|
|
if (inputStream == null)
|
|
return ContactPhotoFactory.getDefaultContactPhoto(context);
|
|
else
|
|
return BitmapFactory.decodeStream(inputStream);
|
|
}
|
|
|
|
public static class RecipientDetails {
|
|
public final String name;
|
|
public final Bitmap avatar;
|
|
public final Uri contactUri;
|
|
|
|
public RecipientDetails(String name, Uri contactUri, Bitmap avatar) {
|
|
this.name = name;
|
|
this.avatar = avatar;
|
|
this.contactUri = contactUri;
|
|
}
|
|
}
|
|
|
|
} |