2017-08-15 03:11:13 +02:00
|
|
|
package org.thoughtcrime.securesms.jobs;
|
|
|
|
|
2019-03-28 16:56:35 +01:00
|
|
|
import android.app.Application;
|
2017-08-15 03:11:13 +02:00
|
|
|
import android.text.TextUtils;
|
2018-08-09 16:15:43 +02:00
|
|
|
|
2021-03-09 01:56:24 +01:00
|
|
|
import androidx.annotation.NonNull;
|
|
|
|
|
2021-05-18 08:03:47 +02:00
|
|
|
import org.session.libsession.avatars.AvatarHelper;
|
2021-05-13 02:31:06 +02:00
|
|
|
import org.session.libsession.messaging.utilities.Data;
|
2021-05-18 08:11:38 +02:00
|
|
|
import org.session.libsession.utilities.Address;
|
2021-04-27 06:48:57 +02:00
|
|
|
import org.session.libsession.utilities.DownloadUtilities;
|
2021-03-09 01:56:24 +01:00
|
|
|
import org.session.libsession.utilities.TextSecurePreferences;
|
2021-01-13 07:11:30 +01:00
|
|
|
import org.session.libsession.utilities.Util;
|
2021-10-04 09:51:19 +02:00
|
|
|
import org.session.libsession.utilities.recipients.Recipient;
|
2021-05-18 01:50:16 +02:00
|
|
|
import org.session.libsignal.exceptions.PushNetworkException;
|
2021-10-04 09:51:19 +02:00
|
|
|
import org.session.libsignal.streams.ProfileCipherInputStream;
|
2021-05-18 01:12:33 +02:00
|
|
|
import org.session.libsignal.utilities.Log;
|
2017-08-22 03:37:39 +02:00
|
|
|
import org.thoughtcrime.securesms.database.RecipientDatabase;
|
2021-10-04 09:51:19 +02:00
|
|
|
import org.thoughtcrime.securesms.dependencies.DatabaseComponent;
|
2021-03-09 01:56:24 +01:00
|
|
|
import org.thoughtcrime.securesms.jobmanager.Job;
|
|
|
|
import org.thoughtcrime.securesms.jobmanager.impl.NetworkConstraint;
|
2017-08-15 03:11:13 +02:00
|
|
|
|
|
|
|
import java.io.File;
|
2021-04-27 06:48:57 +02:00
|
|
|
import java.io.FileInputStream;
|
2017-08-15 03:11:13 +02:00
|
|
|
import java.io.FileOutputStream;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.InputStream;
|
2021-03-09 01:56:24 +01:00
|
|
|
import java.security.SecureRandom;
|
2019-03-28 16:56:35 +01:00
|
|
|
import java.util.concurrent.TimeUnit;
|
2017-08-15 03:11:13 +02:00
|
|
|
|
2021-10-04 09:51:19 +02:00
|
|
|
public class RetrieveProfileAvatarJob extends BaseJob {
|
2018-08-09 16:15:43 +02:00
|
|
|
|
2019-03-28 16:56:35 +01:00
|
|
|
public static final String KEY = "RetrieveProfileAvatarJob";
|
2017-08-15 03:11:13 +02:00
|
|
|
|
|
|
|
private static final String TAG = RetrieveProfileAvatarJob.class.getSimpleName();
|
|
|
|
|
2021-05-14 02:56:47 +02:00
|
|
|
private static final int MAX_PROFILE_SIZE_BYTES = 10 * 1024 * 1024;
|
2017-08-15 03:11:13 +02:00
|
|
|
|
2018-08-09 16:15:43 +02:00
|
|
|
private static final String KEY_PROFILE_AVATAR = "profile_avatar";
|
|
|
|
private static final String KEY_ADDRESS = "address";
|
|
|
|
|
2017-08-15 03:11:13 +02:00
|
|
|
|
2018-08-09 16:15:43 +02:00
|
|
|
private String profileAvatar;
|
|
|
|
private Recipient recipient;
|
|
|
|
|
2019-03-28 16:56:35 +01:00
|
|
|
public RetrieveProfileAvatarJob(Recipient recipient, String profileAvatar) {
|
|
|
|
this(new Job.Parameters.Builder()
|
2021-05-14 02:56:47 +02:00
|
|
|
.setQueue("RetrieveProfileAvatarJob" + recipient.getAddress().serialize())
|
|
|
|
.addConstraint(NetworkConstraint.KEY)
|
|
|
|
.setLifespan(TimeUnit.HOURS.toMillis(1))
|
Performance improvements and bug fixes (#869)
* refactor: fail on testSnode instead of recursively using up snode list. add call timeout on http client
* refactor: refactoring batch message receives and pollers
* refactor: reduce thread utils pool count to a 2 thread fixed pool. Do a check against pubkey instead of room names for oxenHostedOpenGroup
* refactor: caching lib with potential loader fixes and no-cache for giphy
* refactor: remove store and instead use ConcurrentHashMap with a backing update coroutine
* refactor: queue trim thread jobs instead of add every message processed
* fix: wrapping auth token and initial sync for open groups in a threadutils queued runnable, getting initial sync times down
* fix: fixing the user contacts cache in ConversationAdapter.kt
* refactor: improve polling and initial sync, move group joins from config messages into a background job fetching image.
* refactor: improving the job queuing for open groups, replacing placeholder avatar generation with a custom glide loader and archiving initial sync of open groups
* feat: add OpenGroupDeleteJob.kt
* feat: add open group delete job to process deletions after batch adding
* feat: add vacuum and fix job queue re-adding jobs forever, only try to set message hash values in DB if they have changed
* refactor: remove redundant inflation for profile image views throughout app
* refactor(wip): reducing layout inflation and starting to refactor the open group deletion issues taking a long time
* refactor(wip): refactoring group deletion to not iterate through and delete messages individually
* refactor(wip): refactoring group deletion to not iterate through and delete messages individually
* fix: group deletion optimisation
* build: bump build number
* build: bump build number and fix batch message receive retry logic
* fix: clear out open group deletes
* fix: update visible ConversationAdapter.kt binding for initial contact fetching and better traces for debugging background jobs
* fix: add in check for / force sync latest encryption key pair from linked devices if we already have that closed group
* Rename .java to .kt
* refactor: change MmsDatabase to kotlin to make list operations easier
* fix: nullable type
* fix: compilation issues and constants in .kt instead of .java
* fix: bug fix expiration timer on closed group recipient
* feat: use the job queue properly across executors
* feat: start on open group dispatcher-specific logic, probably a queue factory based on openGroupId if that is the same across new message and deletion jobs to ensure consistent entry and removal
* refactor: removing redundant code and fixing jobqueue per opengroup
* fix: allow attachments in note to self
* fix: make the minWidth in quote view bind max of text / title and body, wrapped ?
* fix: fixing up layouts and code view layouts
* fix: remove TODO, remove timestamp binding
* feat: fix view logic, avatars and padding, downloading attachments lazily (on bind), fixing potential crash, add WindowDebouncer.kt
* fix: NPE on viewModel recipient from removed thread while tearing down the Recipient observer in ConversationActivityV2.kt
* refactor: replace conversation notification debouncer handler with handlerthread, same as conversation list debouncer
* refactor: UI for groups and poller improvements
* fix: revert some changes in poller
* feat: add header back in for message requests
* refactor: remove Trace calls, add more conditions to the HomeDiffUtil for updating more efficiently
* feat: try update the home adapter if we get a profile picture modified event
* feat: bump build numbers
* fix: try to start with list in homeViewModel if we don't have already, render quotes to be width of attachment slide view instead of fixed
* fix: set channel to be conflated instead of no buffer
* fix: set unreads based off last local user message vs incrementing unreads to be all amount
* feat: add profile update flag, update build number
* fix: link preview thumbnails download on bind
* fix: centercrop placeholder in glide request
* feat: recycle the contact selection list and profile image in unbind
* fix: try to prevent user KP crash at weird times
* fix: remove additional log, improve attachment download success rate, fix share logs dialog issue
2022-06-08 09:12:34 +02:00
|
|
|
.setMaxAttempts(2)
|
|
|
|
.setMaxInstances(1)
|
2021-05-14 02:56:47 +02:00
|
|
|
.build(),
|
2019-03-28 16:56:35 +01:00
|
|
|
recipient,
|
|
|
|
profileAvatar);
|
2018-08-09 16:15:43 +02:00
|
|
|
}
|
2017-08-15 03:11:13 +02:00
|
|
|
|
2019-03-28 16:56:35 +01:00
|
|
|
private RetrieveProfileAvatarJob(@NonNull Job.Parameters parameters, @NonNull Recipient recipient, String profileAvatar) {
|
|
|
|
super(parameters);
|
2017-08-15 03:11:13 +02:00
|
|
|
this.recipient = recipient;
|
|
|
|
this.profileAvatar = profileAvatar;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2021-01-22 05:16:50 +01:00
|
|
|
public @NonNull
|
|
|
|
Data serialize() {
|
2021-05-14 02:56:47 +02:00
|
|
|
return new Data.Builder()
|
|
|
|
.putString(KEY_PROFILE_AVATAR, profileAvatar)
|
|
|
|
.putString(KEY_ADDRESS, recipient.getAddress().serialize())
|
|
|
|
.build();
|
2018-08-09 16:15:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2019-03-28 16:56:35 +01:00
|
|
|
public @NonNull String getFactoryKey() {
|
|
|
|
return KEY;
|
2018-08-09 16:15:43 +02:00
|
|
|
}
|
2017-08-15 03:11:13 +02:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onRun() throws IOException {
|
2021-10-04 09:51:19 +02:00
|
|
|
RecipientDatabase database = DatabaseComponent.get(context).recipientDatabase();
|
2017-08-22 19:44:04 +02:00
|
|
|
byte[] profileKey = recipient.resolve().getProfileKey();
|
2017-08-15 03:11:13 +02:00
|
|
|
|
2022-01-22 14:24:34 +01:00
|
|
|
if (profileKey == null || (profileKey.length != 32 && profileKey.length != 16)) {
|
2017-08-15 03:11:13 +02:00
|
|
|
Log.w(TAG, "Recipient profile key is gone!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-03-14 23:24:15 +01:00
|
|
|
if (AvatarHelper.avatarFileExists(context, recipient.resolve().getAddress()) && Util.equals(profileAvatar, recipient.resolve().getProfileAvatar())) {
|
2017-08-15 03:11:13 +02:00
|
|
|
Log.w(TAG, "Already retrieved profile avatar: " + profileAvatar);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (TextUtils.isEmpty(profileAvatar)) {
|
|
|
|
Log.w(TAG, "Removing profile avatar for: " + recipient.getAddress().serialize());
|
2017-08-16 06:03:31 +02:00
|
|
|
AvatarHelper.delete(context, recipient.getAddress());
|
2017-08-22 19:44:04 +02:00
|
|
|
database.setProfileAvatar(recipient, profileAvatar);
|
2017-08-15 03:11:13 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-09-04 09:27:57 +02:00
|
|
|
File downloadDestination = File.createTempFile("avatar", ".jpg", context.getCacheDir());
|
2017-08-15 03:11:13 +02:00
|
|
|
|
|
|
|
try {
|
2021-05-21 07:02:34 +02:00
|
|
|
DownloadUtilities.downloadFile(downloadDestination, profileAvatar);
|
2021-04-27 06:48:57 +02:00
|
|
|
InputStream avatarStream = new ProfileCipherInputStream(new FileInputStream(downloadDestination), profileKey);
|
2020-09-04 09:27:57 +02:00
|
|
|
File decryptDestination = File.createTempFile("avatar", ".jpg", context.getCacheDir());
|
2017-08-15 03:11:13 +02:00
|
|
|
|
|
|
|
Util.copy(avatarStream, new FileOutputStream(decryptDestination));
|
2017-08-16 06:03:31 +02:00
|
|
|
decryptDestination.renameTo(AvatarHelper.getAvatarFile(context, recipient.getAddress()));
|
2017-08-15 03:11:13 +02:00
|
|
|
} finally {
|
|
|
|
if (downloadDestination != null) downloadDestination.delete();
|
|
|
|
}
|
|
|
|
|
2021-03-09 01:56:24 +01:00
|
|
|
if (recipient.isLocalNumber()) {
|
|
|
|
TextSecurePreferences.setProfileAvatarId(context, new SecureRandom().nextInt());
|
|
|
|
}
|
2017-08-22 19:44:04 +02:00
|
|
|
database.setProfileAvatar(recipient, profileAvatar);
|
2017-08-15 03:11:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2019-05-22 18:51:56 +02:00
|
|
|
public boolean onShouldRetry(@NonNull Exception e) {
|
2017-08-15 03:11:13 +02:00
|
|
|
if (e instanceof PushNetworkException) return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onCanceled() {
|
2019-03-28 16:56:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public static final class Factory implements Job.Factory<RetrieveProfileAvatarJob> {
|
|
|
|
|
|
|
|
private final Application application;
|
2017-08-15 03:11:13 +02:00
|
|
|
|
2019-03-28 16:56:35 +01:00
|
|
|
public Factory(Application application) {
|
|
|
|
this.application = application;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public @NonNull RetrieveProfileAvatarJob create(@NonNull Parameters parameters, @NonNull Data data) {
|
|
|
|
return new RetrieveProfileAvatarJob(parameters,
|
2021-02-17 06:09:36 +01:00
|
|
|
Recipient.from(application, Address.fromSerialized(data.getString(KEY_ADDRESS)), true),
|
2019-03-28 16:56:35 +01:00
|
|
|
data.getString(KEY_PROFILE_AVATAR));
|
|
|
|
}
|
2017-08-15 03:11:13 +02:00
|
|
|
}
|
|
|
|
}
|