2014-07-25 22:14:29 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2013 Open 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;
|
|
|
|
|
2018-07-07 00:28:58 +00:00
|
|
|
import android.arch.lifecycle.DefaultLifecycleObserver;
|
|
|
|
import android.arch.lifecycle.LifecycleOwner;
|
|
|
|
import android.arch.lifecycle.ProcessLifecycleOwner;
|
2014-07-25 22:14:29 +00:00
|
|
|
import android.content.Context;
|
2019-11-21 05:31:01 +00:00
|
|
|
import android.content.Intent;
|
2017-01-11 23:37:51 +00:00
|
|
|
import android.os.AsyncTask;
|
2017-02-14 06:55:06 +00:00
|
|
|
import android.os.Build;
|
2019-11-21 05:31:01 +00:00
|
|
|
import android.os.Handler;
|
2018-07-07 00:28:58 +00:00
|
|
|
import android.support.annotation.NonNull;
|
2019-10-10 00:38:43 +00:00
|
|
|
import android.support.annotation.Nullable;
|
2017-01-08 17:43:43 +00:00
|
|
|
import android.support.multidex.MultiDexApplication;
|
2017-01-11 23:37:51 +00:00
|
|
|
|
2020-04-15 00:24:30 +00:00
|
|
|
import com.google.firebase.iid.FirebaseInstanceId;
|
|
|
|
|
2019-05-01 16:00:26 +00:00
|
|
|
import org.conscrypt.Conscrypt;
|
2019-06-07 05:21:25 +00:00
|
|
|
import org.jetbrains.annotations.NotNull;
|
2019-05-01 16:00:26 +00:00
|
|
|
import org.signal.aesgcmprovider.AesGcmProvider;
|
2018-10-29 22:14:31 +00:00
|
|
|
import org.thoughtcrime.securesms.components.TypingStatusRepository;
|
|
|
|
import org.thoughtcrime.securesms.components.TypingStatusSender;
|
2019-09-24 03:33:42 +00:00
|
|
|
import org.thoughtcrime.securesms.crypto.IdentityKeyUtil;
|
2019-11-21 05:31:01 +00:00
|
|
|
import org.thoughtcrime.securesms.crypto.MasterSecretUtil;
|
2019-11-28 01:49:33 +00:00
|
|
|
import org.thoughtcrime.securesms.crypto.ProfileKeyUtil;
|
2020-07-09 23:46:14 +00:00
|
|
|
import org.thoughtcrime.securesms.crypto.storage.TextSecureSessionStore;
|
2019-11-28 01:49:33 +00:00
|
|
|
import org.thoughtcrime.securesms.database.Address;
|
2019-04-03 20:30:29 +00:00
|
|
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
2014-11-12 03:57:53 +00:00
|
|
|
import org.thoughtcrime.securesms.dependencies.AxolotlStorageModule;
|
|
|
|
import org.thoughtcrime.securesms.dependencies.InjectableType;
|
2016-12-20 17:55:52 +00:00
|
|
|
import org.thoughtcrime.securesms.dependencies.SignalCommunicationModule;
|
2019-03-28 15:56:35 +00:00
|
|
|
import org.thoughtcrime.securesms.jobmanager.DependencyInjector;
|
2018-06-18 19:27:04 +00:00
|
|
|
import org.thoughtcrime.securesms.jobmanager.JobManager;
|
2019-03-28 15:56:35 +00:00
|
|
|
import org.thoughtcrime.securesms.jobmanager.impl.JsonDataSerializer;
|
2016-03-12 01:07:22 +00:00
|
|
|
import org.thoughtcrime.securesms.jobs.CreateSignedPreKeyJob;
|
2019-06-18 03:23:58 +00:00
|
|
|
import org.thoughtcrime.securesms.jobs.FastJobStorage;
|
|
|
|
import org.thoughtcrime.securesms.jobs.JobManagerFactories;
|
2018-07-07 00:28:58 +00:00
|
|
|
import org.thoughtcrime.securesms.jobs.MultiDeviceContactUpdateJob;
|
2019-07-01 06:49:42 +00:00
|
|
|
import org.thoughtcrime.securesms.jobs.PushContentReceiveJob;
|
2018-10-05 22:23:33 +00:00
|
|
|
import org.thoughtcrime.securesms.jobs.PushNotificationReceiveJob;
|
2018-10-11 23:45:22 +00:00
|
|
|
import org.thoughtcrime.securesms.jobs.RefreshUnidentifiedDeliveryAbilityJob;
|
2018-07-26 14:10:46 +00:00
|
|
|
import org.thoughtcrime.securesms.logging.AndroidLogger;
|
|
|
|
import org.thoughtcrime.securesms.logging.CustomSignalProtocolLogger;
|
2018-08-01 15:09:24 +00:00
|
|
|
import org.thoughtcrime.securesms.logging.Log;
|
2018-07-26 14:10:46 +00:00
|
|
|
import org.thoughtcrime.securesms.logging.PersistentLogger;
|
|
|
|
import org.thoughtcrime.securesms.logging.UncaughtExceptionLogger;
|
2020-05-11 06:19:26 +00:00
|
|
|
import org.thoughtcrime.securesms.loki.activities.HomeActivity;
|
|
|
|
import org.thoughtcrime.securesms.loki.api.BackgroundPollWorker;
|
2020-05-13 23:35:34 +00:00
|
|
|
import org.thoughtcrime.securesms.loki.api.LokiPushNotificationManager;
|
2020-07-15 05:14:37 +00:00
|
|
|
import org.thoughtcrime.securesms.loki.api.PublicChatManager;
|
2020-05-11 06:19:26 +00:00
|
|
|
import org.thoughtcrime.securesms.loki.database.LokiAPIDatabase;
|
2020-05-13 23:35:34 +00:00
|
|
|
import org.thoughtcrime.securesms.loki.database.LokiThreadDatabase;
|
2020-05-11 06:19:26 +00:00
|
|
|
import org.thoughtcrime.securesms.loki.database.LokiUserDatabase;
|
2020-07-15 05:14:37 +00:00
|
|
|
import org.thoughtcrime.securesms.loki.protocol.PushSessionRequestMessageSendJob;
|
2020-07-15 04:26:20 +00:00
|
|
|
import org.thoughtcrime.securesms.loki.protocol.SessionResetImplementation;
|
2020-05-11 06:19:26 +00:00
|
|
|
import org.thoughtcrime.securesms.loki.utilities.Broadcaster;
|
2020-06-26 06:18:19 +00:00
|
|
|
import org.thoughtcrime.securesms.notifications.DefaultMessageNotifier;
|
2019-04-17 14:21:30 +00:00
|
|
|
import org.thoughtcrime.securesms.notifications.MessageNotifier;
|
2018-08-06 16:20:24 +00:00
|
|
|
import org.thoughtcrime.securesms.notifications.NotificationChannels;
|
2020-06-26 06:18:19 +00:00
|
|
|
import org.thoughtcrime.securesms.notifications.OptimizedMessageNotifier;
|
2020-02-24 03:57:51 +00:00
|
|
|
import org.thoughtcrime.securesms.profiles.AvatarHelper;
|
2019-02-26 01:47:30 +00:00
|
|
|
import org.thoughtcrime.securesms.providers.BlobProvider;
|
2016-12-30 04:54:05 +00:00
|
|
|
import org.thoughtcrime.securesms.push.SignalServiceNetworkAccess;
|
2019-11-28 01:49:33 +00:00
|
|
|
import org.thoughtcrime.securesms.recipients.Recipient;
|
2016-08-16 03:23:56 +00:00
|
|
|
import org.thoughtcrime.securesms.service.ExpiringMessageManager;
|
2018-10-15 20:27:21 +00:00
|
|
|
import org.thoughtcrime.securesms.service.IncomingMessageObserver;
|
|
|
|
import org.thoughtcrime.securesms.service.KeyCachingService;
|
2018-02-26 17:58:18 +00:00
|
|
|
import org.thoughtcrime.securesms.service.LocalBackupListener;
|
2018-05-22 09:13:10 +00:00
|
|
|
import org.thoughtcrime.securesms.service.RotateSenderCertificateListener;
|
2017-01-06 17:19:58 +00:00
|
|
|
import org.thoughtcrime.securesms.service.RotateSignedPreKeyListener;
|
2017-02-26 22:36:43 +00:00
|
|
|
import org.thoughtcrime.securesms.service.UpdateApkRefreshListener;
|
2014-07-25 22:14:29 +00:00
|
|
|
import org.thoughtcrime.securesms.util.TextSecurePreferences;
|
2020-05-25 00:20:18 +00:00
|
|
|
import org.thoughtcrime.securesms.util.Util;
|
2019-03-25 20:23:38 +00:00
|
|
|
import org.thoughtcrime.securesms.util.dynamiclanguage.DynamicLanguageContextWrapper;
|
2016-11-09 17:37:40 +00:00
|
|
|
import org.webrtc.PeerConnectionFactory;
|
2017-12-22 00:37:07 +00:00
|
|
|
import org.webrtc.PeerConnectionFactory.InitializationOptions;
|
2017-03-21 17:23:16 +00:00
|
|
|
import org.webrtc.voiceengine.WebRtcAudioManager;
|
|
|
|
import org.webrtc.voiceengine.WebRtcAudioUtils;
|
2020-07-09 23:46:14 +00:00
|
|
|
import org.whispersystems.libsignal.SignalProtocolAddress;
|
2016-03-23 17:34:41 +00:00
|
|
|
import org.whispersystems.libsignal.logging.SignalProtocolLoggerProvider;
|
2019-06-25 03:51:24 +00:00
|
|
|
import org.whispersystems.signalservice.api.messages.SignalServiceEnvelope;
|
2020-07-09 23:46:14 +00:00
|
|
|
import org.whispersystems.signalservice.api.push.SignalServiceAddress;
|
2020-02-24 03:57:51 +00:00
|
|
|
import org.whispersystems.signalservice.api.util.StreamDetails;
|
2019-06-25 03:51:24 +00:00
|
|
|
import org.whispersystems.signalservice.internal.push.SignalServiceProtos;
|
2020-07-08 07:05:26 +00:00
|
|
|
import org.whispersystems.signalservice.loki.api.Poller;
|
|
|
|
import org.whispersystems.signalservice.loki.api.PushNotificationAcknowledgement;
|
|
|
|
import org.whispersystems.signalservice.loki.api.SnodeAPI;
|
|
|
|
import org.whispersystems.signalservice.loki.api.SwarmAPI;
|
2020-07-15 02:24:43 +00:00
|
|
|
import org.whispersystems.signalservice.loki.api.fileserver.FileServerAPI;
|
|
|
|
import org.whispersystems.signalservice.loki.api.opengroups.PublicChatAPI;
|
2020-07-08 07:05:26 +00:00
|
|
|
import org.whispersystems.signalservice.loki.api.shelved.p2p.LokiP2PAPI;
|
|
|
|
import org.whispersystems.signalservice.loki.api.shelved.p2p.LokiP2PAPIDelegate;
|
2020-05-07 07:59:41 +00:00
|
|
|
import org.whispersystems.signalservice.loki.database.LokiAPIDatabaseProtocol;
|
2020-05-13 23:35:34 +00:00
|
|
|
import org.whispersystems.signalservice.loki.protocol.mentions.MentionsManager;
|
|
|
|
import org.whispersystems.signalservice.loki.protocol.meta.SessionMetaProtocol;
|
2020-05-14 03:52:20 +00:00
|
|
|
import org.whispersystems.signalservice.loki.protocol.multidevice.DeviceLink;
|
2020-05-13 23:35:34 +00:00
|
|
|
import org.whispersystems.signalservice.loki.protocol.multidevice.MultiDeviceProtocol;
|
|
|
|
import org.whispersystems.signalservice.loki.protocol.sessionmanagement.SessionManagementProtocol;
|
|
|
|
import org.whispersystems.signalservice.loki.protocol.sessionmanagement.SessionManagementProtocolDelegate;
|
|
|
|
import org.whispersystems.signalservice.loki.protocol.syncmessages.SyncMessagesProtocol;
|
2014-07-25 22:14:29 +00:00
|
|
|
|
2020-02-24 03:57:51 +00:00
|
|
|
import java.io.File;
|
|
|
|
import java.io.FileInputStream;
|
|
|
|
import java.security.SecureRandom;
|
2019-05-01 16:00:26 +00:00
|
|
|
import java.security.Security;
|
2020-02-24 03:57:51 +00:00
|
|
|
import java.util.Date;
|
2017-03-21 17:23:16 +00:00
|
|
|
import java.util.HashSet;
|
|
|
|
import java.util.Set;
|
2017-03-17 17:28:06 +00:00
|
|
|
|
2014-11-12 03:57:53 +00:00
|
|
|
import dagger.ObjectGraph;
|
2019-06-25 03:51:24 +00:00
|
|
|
import kotlin.Unit;
|
2019-07-24 02:30:23 +00:00
|
|
|
import network.loki.messenger.BuildConfig;
|
2014-11-12 03:57:53 +00:00
|
|
|
|
2019-09-19 05:00:52 +00:00
|
|
|
import static nl.komponents.kovenant.android.KovenantAndroid.startKovenant;
|
|
|
|
import static nl.komponents.kovenant.android.KovenantAndroid.stopKovenant;
|
|
|
|
|
2014-07-25 22:14:29 +00:00
|
|
|
/**
|
|
|
|
* Will be called once when the TextSecure process is created.
|
|
|
|
*
|
|
|
|
* We're using this as an insertion point to patch up the Android PRNG disaster,
|
|
|
|
* to initialize the job manager, and to check for GCM registration freshness.
|
|
|
|
*
|
|
|
|
* @author Moxie Marlinspike
|
|
|
|
*/
|
2020-05-13 23:35:34 +00:00
|
|
|
public class ApplicationContext extends MultiDexApplication implements DependencyInjector, DefaultLifecycleObserver, LokiP2PAPIDelegate, SessionManagementProtocolDelegate {
|
2014-07-25 22:14:29 +00:00
|
|
|
|
2018-07-26 14:10:46 +00:00
|
|
|
private static final String TAG = ApplicationContext.class.getSimpleName();
|
2019-11-15 05:24:58 +00:00
|
|
|
private final static int OK_HTTP_CACHE_SIZE = 10 * 1024 * 1024; // 10 MB
|
2017-01-11 23:37:51 +00:00
|
|
|
|
2018-10-15 20:27:21 +00:00
|
|
|
private ExpiringMessageManager expiringMessageManager;
|
2018-10-29 22:14:31 +00:00
|
|
|
private TypingStatusRepository typingStatusRepository;
|
|
|
|
private TypingStatusSender typingStatusSender;
|
2019-04-17 14:21:30 +00:00
|
|
|
private JobManager jobManager;
|
2018-10-15 20:27:21 +00:00
|
|
|
private IncomingMessageObserver incomingMessageObserver;
|
|
|
|
private ObjectGraph objectGraph;
|
|
|
|
private PersistentLogger persistentLogger;
|
2019-06-28 06:08:27 +00:00
|
|
|
|
|
|
|
// Loki
|
2020-06-26 06:18:19 +00:00
|
|
|
public MessageNotifier messageNotifier = null;
|
2020-07-15 22:55:29 +00:00
|
|
|
public Poller poller = null;
|
2020-07-15 04:26:20 +00:00
|
|
|
public PublicChatManager publicChatManager = null;
|
2020-07-15 02:24:43 +00:00
|
|
|
private PublicChatAPI publicChatAPI = null;
|
2020-01-16 03:35:51 +00:00
|
|
|
public Broadcaster broadcaster = null;
|
2019-06-28 06:08:27 +00:00
|
|
|
public SignalCommunicationModule communicationModule;
|
2019-06-18 02:49:21 +00:00
|
|
|
|
2018-07-07 00:28:58 +00:00
|
|
|
private volatile boolean isAppVisible;
|
|
|
|
|
2014-07-25 22:14:29 +00:00
|
|
|
public static ApplicationContext getInstance(Context context) {
|
|
|
|
return (ApplicationContext)context.getApplicationContext();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onCreate() {
|
2018-10-04 16:09:04 +00:00
|
|
|
super.onCreate();
|
2018-10-05 22:23:33 +00:00
|
|
|
Log.i(TAG, "onCreate()");
|
2019-11-21 05:31:01 +00:00
|
|
|
startKovenant();
|
2019-05-01 16:00:26 +00:00
|
|
|
initializeSecurityProvider();
|
2018-10-04 16:09:04 +00:00
|
|
|
initializeLogging();
|
|
|
|
initializeCrashHandling();
|
|
|
|
initializeDependencyInjection();
|
|
|
|
NotificationChannels.create(this);
|
|
|
|
ProcessLifecycleOwner.get().getLifecycle().addObserver(this);
|
2020-05-13 23:35:34 +00:00
|
|
|
// Loki
|
2020-05-25 23:03:41 +00:00
|
|
|
// ========
|
2020-06-26 06:18:19 +00:00
|
|
|
messageNotifier = new OptimizedMessageNotifier(new DefaultMessageNotifier());
|
2020-05-13 23:35:34 +00:00
|
|
|
broadcaster = new Broadcaster(this);
|
|
|
|
LokiAPIDatabase apiDB = DatabaseFactory.getLokiAPIDatabase(this);
|
|
|
|
LokiThreadDatabase threadDB = DatabaseFactory.getLokiThreadDatabase(this);
|
|
|
|
LokiUserDatabase userDB = DatabaseFactory.getLokiUserDatabase(this);
|
|
|
|
String userPublicKey = TextSecurePreferences.getLocalNumber(this);
|
2020-07-15 04:26:20 +00:00
|
|
|
SessionResetImplementation sessionResetImpl = new SessionResetImplementation(this);
|
2020-05-13 23:35:34 +00:00
|
|
|
if (userPublicKey != null) {
|
2020-07-08 07:05:26 +00:00
|
|
|
SwarmAPI.Companion.configureIfNeeded(apiDB);
|
|
|
|
SnodeAPI.Companion.configureIfNeeded(userPublicKey, apiDB, broadcaster);
|
2020-05-13 23:35:34 +00:00
|
|
|
MentionsManager.Companion.configureIfNeeded(userPublicKey, threadDB, userDB);
|
|
|
|
SessionMetaProtocol.Companion.configureIfNeeded(apiDB, userPublicKey);
|
|
|
|
SyncMessagesProtocol.Companion.configureIfNeeded(apiDB, userPublicKey);
|
|
|
|
}
|
2020-05-25 08:01:21 +00:00
|
|
|
MultiDeviceProtocol.Companion.configureIfNeeded(apiDB);
|
|
|
|
SessionManagementProtocol.Companion.configureIfNeeded(sessionResetImpl, threadDB, this);
|
2020-05-13 23:35:34 +00:00
|
|
|
setUpP2PAPIIfNeeded();
|
2020-07-08 07:05:26 +00:00
|
|
|
PushNotificationAcknowledgement.Companion.configureIfNeeded(BuildConfig.DEBUG);
|
2019-10-24 02:35:14 +00:00
|
|
|
if (setUpStorageAPIIfNeeded()) {
|
2020-05-13 23:35:34 +00:00
|
|
|
if (userPublicKey != null) {
|
2020-05-14 03:52:20 +00:00
|
|
|
Set<DeviceLink> deviceLinks = DatabaseFactory.getLokiAPIDatabase(this).getDeviceLinks(userPublicKey);
|
2020-07-15 02:24:43 +00:00
|
|
|
FileServerAPI.shared.setDeviceLinks(deviceLinks);
|
2019-11-21 22:35:15 +00:00
|
|
|
}
|
2019-10-23 04:29:56 +00:00
|
|
|
}
|
2020-02-24 03:57:51 +00:00
|
|
|
resubmitProfilePictureIfNeeded();
|
2020-07-15 04:26:20 +00:00
|
|
|
publicChatManager = new PublicChatManager(this);
|
2020-05-13 23:35:34 +00:00
|
|
|
updateOpenGroupProfilePicturesIfNeeded();
|
2020-04-17 03:26:11 +00:00
|
|
|
registerForFCMIfNeeded(false);
|
2020-05-25 23:03:41 +00:00
|
|
|
// ========
|
|
|
|
initializeJobManager();
|
|
|
|
initializeMessageRetrieval();
|
|
|
|
initializeExpiringMessageManager();
|
|
|
|
initializeTypingStatusRepository();
|
|
|
|
initializeTypingStatusSender();
|
|
|
|
initializeSignedPreKeyCheck();
|
|
|
|
initializePeriodicTasks();
|
|
|
|
initializeWebRtc();
|
|
|
|
initializePendingMessages();
|
|
|
|
initializeUnidentifiedDeliveryAbilityRefresh();
|
|
|
|
initializeBlobProvider();
|
2018-07-07 00:28:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onStart(@NonNull LifecycleOwner owner) {
|
|
|
|
isAppVisible = true;
|
|
|
|
Log.i(TAG, "App is now visible.");
|
|
|
|
executePendingContactSync();
|
2018-10-15 20:27:21 +00:00
|
|
|
KeyCachingService.onAppForegrounded(this);
|
2020-05-13 23:35:34 +00:00
|
|
|
// Loki
|
2020-07-15 22:55:29 +00:00
|
|
|
if (poller != null) { poller.setCaughtUp(false); }
|
2020-03-24 02:48:23 +00:00
|
|
|
startPollingIfNeeded();
|
2020-07-15 04:26:20 +00:00
|
|
|
publicChatManager.markAllAsNotCaughtUp();
|
|
|
|
publicChatManager.startPollersIfNeeded();
|
2018-07-07 00:28:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onStop(@NonNull LifecycleOwner owner) {
|
|
|
|
isAppVisible = false;
|
|
|
|
Log.i(TAG, "App is no longer visible.");
|
2018-10-15 20:27:21 +00:00
|
|
|
KeyCachingService.onAppBackgrounded(this);
|
2020-06-26 06:18:19 +00:00
|
|
|
messageNotifier.setVisibleThread(-1);
|
2020-05-13 23:35:34 +00:00
|
|
|
// Loki
|
2020-07-15 22:55:29 +00:00
|
|
|
if (poller != null) { poller.stopIfNeeded(); }
|
2020-07-15 04:26:20 +00:00
|
|
|
if (publicChatManager != null) { publicChatManager.stopPollers(); }
|
2014-07-25 22:14:29 +00:00
|
|
|
}
|
|
|
|
|
2019-09-19 05:00:52 +00:00
|
|
|
@Override
|
|
|
|
public void onTerminate() {
|
2020-05-13 23:35:34 +00:00
|
|
|
stopKovenant(); // Loki
|
2019-09-19 05:00:52 +00:00
|
|
|
super.onTerminate();
|
|
|
|
}
|
|
|
|
|
2014-11-12 03:57:53 +00:00
|
|
|
@Override
|
|
|
|
public void injectDependencies(Object object) {
|
|
|
|
if (object instanceof InjectableType) {
|
|
|
|
objectGraph.inject(object);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-25 22:14:29 +00:00
|
|
|
public JobManager getJobManager() {
|
|
|
|
return jobManager;
|
|
|
|
}
|
|
|
|
|
2016-08-16 03:23:56 +00:00
|
|
|
public ExpiringMessageManager getExpiringMessageManager() {
|
|
|
|
return expiringMessageManager;
|
|
|
|
}
|
|
|
|
|
2018-10-29 22:14:31 +00:00
|
|
|
public TypingStatusRepository getTypingStatusRepository() {
|
|
|
|
return typingStatusRepository;
|
|
|
|
}
|
|
|
|
|
|
|
|
public TypingStatusSender getTypingStatusSender() {
|
|
|
|
return typingStatusSender;
|
|
|
|
}
|
|
|
|
|
2018-07-07 00:28:58 +00:00
|
|
|
public boolean isAppVisible() {
|
|
|
|
return isAppVisible;
|
|
|
|
}
|
|
|
|
|
2018-07-26 14:10:46 +00:00
|
|
|
public PersistentLogger getPersistentLogger() {
|
|
|
|
return persistentLogger;
|
|
|
|
}
|
|
|
|
|
2020-05-13 23:35:34 +00:00
|
|
|
// Loki
|
2020-07-15 02:24:43 +00:00
|
|
|
public @Nullable PublicChatAPI getPublicChatAPI() {
|
|
|
|
if (publicChatAPI != null || !IdentityKeyUtil.hasIdentityKey(this)) { return publicChatAPI; }
|
2020-05-13 23:35:34 +00:00
|
|
|
String userPublicKey = TextSecurePreferences.getLocalNumber(this);
|
2020-07-15 02:24:43 +00:00
|
|
|
if (userPublicKey== null) { return publicChatAPI; }
|
2020-05-13 23:35:34 +00:00
|
|
|
byte[] userPrivateKey = IdentityKeyUtil.getIdentityKeyPair(this).getPrivateKey().serialize();
|
|
|
|
LokiAPIDatabase apiDB = DatabaseFactory.getLokiAPIDatabase(this);
|
|
|
|
LokiUserDatabase userDB = DatabaseFactory.getLokiUserDatabase(this);
|
2020-07-15 02:24:43 +00:00
|
|
|
publicChatAPI = new PublicChatAPI(userPublicKey, userPrivateKey, apiDB, userDB);
|
|
|
|
return publicChatAPI;
|
2019-10-10 00:38:43 +00:00
|
|
|
}
|
|
|
|
|
2019-05-01 16:00:26 +00:00
|
|
|
private void initializeSecurityProvider() {
|
|
|
|
try {
|
|
|
|
Class.forName("org.signal.aesgcmprovider.AesGcmCipher");
|
|
|
|
} catch (ClassNotFoundException e) {
|
|
|
|
Log.e(TAG, "Failed to find AesGcmCipher class");
|
|
|
|
throw new ProviderInitializationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
int aesPosition = Security.insertProviderAt(new AesGcmProvider(), 1);
|
|
|
|
Log.i(TAG, "Installed AesGcmProvider: " + aesPosition);
|
|
|
|
|
|
|
|
if (aesPosition < 0) {
|
|
|
|
Log.e(TAG, "Failed to install AesGcmProvider()");
|
|
|
|
throw new ProviderInitializationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
int conscryptPosition = Security.insertProviderAt(Conscrypt.newProvider(), 2);
|
|
|
|
Log.i(TAG, "Installed Conscrypt provider: " + conscryptPosition);
|
|
|
|
|
|
|
|
if (conscryptPosition < 0) {
|
|
|
|
Log.w(TAG, "Did not install Conscrypt provider. May already be present.");
|
|
|
|
}
|
2014-07-25 22:14:29 +00:00
|
|
|
}
|
|
|
|
|
2015-03-02 16:25:19 +00:00
|
|
|
private void initializeLogging() {
|
2018-07-26 14:10:46 +00:00
|
|
|
persistentLogger = new PersistentLogger(this);
|
|
|
|
org.thoughtcrime.securesms.logging.Log.initialize(new AndroidLogger(), persistentLogger);
|
|
|
|
|
|
|
|
SignalProtocolLoggerProvider.setProvider(new CustomSignalProtocolLogger());
|
|
|
|
}
|
|
|
|
|
|
|
|
private void initializeCrashHandling() {
|
|
|
|
final Thread.UncaughtExceptionHandler originalHandler = Thread.getDefaultUncaughtExceptionHandler();
|
2019-04-10 16:53:55 +00:00
|
|
|
Thread.setDefaultUncaughtExceptionHandler(new UncaughtExceptionLogger(originalHandler));
|
2015-03-02 16:25:19 +00:00
|
|
|
}
|
|
|
|
|
2014-07-25 22:14:29 +00:00
|
|
|
private void initializeJobManager() {
|
2019-03-28 15:56:35 +00:00
|
|
|
this.jobManager = new JobManager(this, new JobManager.Configuration.Builder()
|
|
|
|
.setDataSerializer(new JsonDataSerializer())
|
|
|
|
.setJobFactories(JobManagerFactories.getJobFactories(this))
|
|
|
|
.setConstraintFactories(JobManagerFactories.getConstraintFactories(this))
|
|
|
|
.setConstraintObservers(JobManagerFactories.getConstraintObservers(this))
|
|
|
|
.setJobStorage(new FastJobStorage(DatabaseFactory.getJobDatabase(this)))
|
|
|
|
.setDependencyInjector(this)
|
|
|
|
.build());
|
2014-07-25 22:14:29 +00:00
|
|
|
}
|
|
|
|
|
2018-10-15 20:27:21 +00:00
|
|
|
public void initializeMessageRetrieval() {
|
|
|
|
this.incomingMessageObserver = new IncomingMessageObserver(this);
|
|
|
|
}
|
|
|
|
|
2014-11-12 03:57:53 +00:00
|
|
|
private void initializeDependencyInjection() {
|
2019-06-28 06:08:27 +00:00
|
|
|
communicationModule = new SignalCommunicationModule(this, new SignalServiceNetworkAccess(this));
|
|
|
|
this.objectGraph = ObjectGraph.create(communicationModule, new AxolotlStorageModule(this));
|
2014-11-12 03:57:53 +00:00
|
|
|
}
|
|
|
|
|
2016-03-12 01:07:22 +00:00
|
|
|
private void initializeSignedPreKeyCheck() {
|
|
|
|
if (!TextSecurePreferences.isSignedPreKeyRegistered(this)) {
|
|
|
|
jobManager.add(new CreateSignedPreKeyJob(this));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-16 03:23:56 +00:00
|
|
|
private void initializeExpiringMessageManager() {
|
|
|
|
this.expiringMessageManager = new ExpiringMessageManager(this);
|
|
|
|
}
|
|
|
|
|
2018-10-29 22:14:31 +00:00
|
|
|
private void initializeTypingStatusRepository() {
|
|
|
|
this.typingStatusRepository = new TypingStatusRepository();
|
|
|
|
}
|
|
|
|
|
|
|
|
private void initializeTypingStatusSender() {
|
|
|
|
this.typingStatusSender = new TypingStatusSender(this);
|
|
|
|
}
|
|
|
|
|
2017-01-06 17:19:58 +00:00
|
|
|
private void initializePeriodicTasks() {
|
|
|
|
RotateSignedPreKeyListener.schedule(this);
|
2018-02-26 17:58:18 +00:00
|
|
|
LocalBackupListener.schedule(this);
|
2018-05-22 09:13:10 +00:00
|
|
|
RotateSenderCertificateListener.schedule(this);
|
2020-05-13 23:35:34 +00:00
|
|
|
BackgroundPollWorker.schedule(this); // Loki
|
2017-02-26 22:36:43 +00:00
|
|
|
|
|
|
|
if (BuildConfig.PLAY_STORE_DISABLED) {
|
|
|
|
UpdateApkRefreshListener.schedule(this);
|
|
|
|
}
|
2017-01-06 17:19:58 +00:00
|
|
|
}
|
|
|
|
|
2017-03-21 17:23:16 +00:00
|
|
|
private void initializeWebRtc() {
|
2017-07-21 23:29:03 +00:00
|
|
|
try {
|
|
|
|
Set<String> HARDWARE_AEC_BLACKLIST = new HashSet<String>() {{
|
|
|
|
add("Pixel");
|
|
|
|
add("Pixel XL");
|
|
|
|
add("Moto G5");
|
2018-04-19 17:58:39 +00:00
|
|
|
add("Moto G (5S) Plus");
|
2018-06-05 16:09:20 +00:00
|
|
|
add("Moto G4");
|
|
|
|
add("TA-1053");
|
2018-06-29 10:07:58 +00:00
|
|
|
add("Mi A1");
|
2018-07-03 11:16:01 +00:00
|
|
|
add("E5823"); // Sony z5 compact
|
2019-02-14 05:07:34 +00:00
|
|
|
add("Redmi Note 5");
|
2018-12-11 21:23:01 +00:00
|
|
|
add("FP2"); // Fairphone FP2
|
2019-03-10 13:29:02 +00:00
|
|
|
add("MI 5");
|
2017-07-21 23:29:03 +00:00
|
|
|
}};
|
|
|
|
|
|
|
|
Set<String> OPEN_SL_ES_WHITELIST = new HashSet<String>() {{
|
|
|
|
add("Pixel");
|
|
|
|
add("Pixel XL");
|
|
|
|
}};
|
|
|
|
|
2017-12-22 00:37:07 +00:00
|
|
|
if (HARDWARE_AEC_BLACKLIST.contains(Build.MODEL)) {
|
|
|
|
WebRtcAudioUtils.setWebRtcBasedAcousticEchoCanceler(true);
|
|
|
|
}
|
2017-03-21 17:23:16 +00:00
|
|
|
|
2017-12-22 00:37:07 +00:00
|
|
|
if (!OPEN_SL_ES_WHITELIST.contains(Build.MODEL)) {
|
|
|
|
WebRtcAudioManager.setBlacklistDeviceForOpenSLESUsage(true);
|
2017-07-21 23:29:03 +00:00
|
|
|
}
|
2017-12-22 00:37:07 +00:00
|
|
|
|
2019-01-29 22:25:49 +00:00
|
|
|
PeerConnectionFactory.initialize(InitializationOptions.builder(this).createInitializationOptions());
|
2017-07-21 23:29:03 +00:00
|
|
|
} catch (UnsatisfiedLinkError e) {
|
|
|
|
Log.w(TAG, e);
|
2017-03-21 17:23:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-07 00:28:58 +00:00
|
|
|
private void executePendingContactSync() {
|
|
|
|
if (TextSecurePreferences.needsFullContactSync(this)) {
|
|
|
|
ApplicationContext.getInstance(this).getJobManager().add(new MultiDeviceContactUpdateJob(this, true));
|
|
|
|
}
|
|
|
|
}
|
2018-10-05 22:23:33 +00:00
|
|
|
|
|
|
|
private void initializePendingMessages() {
|
|
|
|
if (TextSecurePreferences.getNeedsMessagePull(this)) {
|
|
|
|
Log.i(TAG, "Scheduling a message fetch.");
|
2018-10-10 16:00:14 +00:00
|
|
|
ApplicationContext.getInstance(this).getJobManager().add(new PushNotificationReceiveJob(this));
|
2018-10-05 22:23:33 +00:00
|
|
|
TextSecurePreferences.setNeedsMessagePull(this, false);
|
|
|
|
}
|
|
|
|
}
|
2018-10-11 23:45:22 +00:00
|
|
|
|
|
|
|
private void initializeUnidentifiedDeliveryAbilityRefresh() {
|
|
|
|
if (TextSecurePreferences.isMultiDevice(this) && !TextSecurePreferences.isUnidentifiedDeliveryEnabled(this)) {
|
2019-03-28 15:56:35 +00:00
|
|
|
jobManager.add(new RefreshUnidentifiedDeliveryAbilityJob());
|
2018-10-11 23:45:22 +00:00
|
|
|
}
|
|
|
|
}
|
2019-02-26 01:47:30 +00:00
|
|
|
|
|
|
|
private void initializeBlobProvider() {
|
|
|
|
AsyncTask.THREAD_POOL_EXECUTOR.execute(() -> {
|
|
|
|
BlobProvider.getInstance().onSessionStart(this);
|
|
|
|
});
|
|
|
|
}
|
2019-03-25 20:23:38 +00:00
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void attachBaseContext(Context base) {
|
|
|
|
super.attachBaseContext(DynamicLanguageContextWrapper.updateContext(base, TextSecurePreferences.getLanguage(base)));
|
|
|
|
}
|
2019-05-01 16:00:26 +00:00
|
|
|
|
2020-07-15 02:24:43 +00:00
|
|
|
private static class ProviderInitializationException extends RuntimeException { }
|
2019-06-07 05:21:25 +00:00
|
|
|
|
2019-06-18 03:23:58 +00:00
|
|
|
// region Loki
|
2019-10-24 02:35:14 +00:00
|
|
|
public boolean setUpStorageAPIIfNeeded() {
|
2020-05-13 23:35:34 +00:00
|
|
|
String userPublicKey = TextSecurePreferences.getLocalNumber(this);
|
|
|
|
if (userPublicKey == null || !IdentityKeyUtil.hasIdentityKey(this)) { return false; }
|
|
|
|
boolean isDebugMode = BuildConfig.DEBUG;
|
|
|
|
byte[] userPrivateKey = IdentityKeyUtil.getIdentityKeyPair(this).getPrivateKey().serialize();
|
|
|
|
LokiAPIDatabaseProtocol apiDB = DatabaseFactory.getLokiAPIDatabase(this);
|
2020-07-15 02:24:43 +00:00
|
|
|
FileServerAPI.Companion.configure(userPublicKey, userPrivateKey, apiDB);
|
2020-05-13 23:35:34 +00:00
|
|
|
return true;
|
2019-09-23 01:28:17 +00:00
|
|
|
}
|
|
|
|
|
2020-05-13 23:35:34 +00:00
|
|
|
public void setUpP2PAPIIfNeeded() {
|
2019-06-18 03:23:58 +00:00
|
|
|
String hexEncodedPublicKey = TextSecurePreferences.getLocalNumber(this);
|
|
|
|
if (hexEncodedPublicKey == null) { return; }
|
|
|
|
LokiP2PAPI.Companion.configure(hexEncodedPublicKey, (isOnline, contactPublicKey) -> {
|
|
|
|
// TODO: Implement
|
|
|
|
return null;
|
|
|
|
}, this);
|
|
|
|
}
|
|
|
|
|
2020-04-17 03:26:11 +00:00
|
|
|
public void registerForFCMIfNeeded(Boolean force) {
|
2020-04-15 00:24:30 +00:00
|
|
|
Context context = this;
|
2020-04-16 06:56:12 +00:00
|
|
|
FirebaseInstanceId.getInstance().getInstanceId().addOnCompleteListener(task -> {
|
|
|
|
if (!task.isSuccessful()) {
|
|
|
|
Log.w(TAG, "getInstanceId failed", task.getException());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
String token = task.getResult().getToken();
|
2020-05-13 23:35:34 +00:00
|
|
|
String userPublicKey = TextSecurePreferences.getLocalNumber(context);
|
|
|
|
if (userPublicKey == null) return;
|
2020-04-17 02:11:27 +00:00
|
|
|
if (TextSecurePreferences.isUsingFCM(this)) {
|
2020-05-13 23:35:34 +00:00
|
|
|
LokiPushNotificationManager.register(token, userPublicKey, context, force);
|
2020-04-17 02:11:27 +00:00
|
|
|
} else {
|
|
|
|
LokiPushNotificationManager.unregister(token, context);
|
|
|
|
}
|
2020-04-16 06:56:12 +00:00
|
|
|
});
|
2020-04-15 00:24:30 +00:00
|
|
|
}
|
|
|
|
|
2019-06-07 05:21:25 +00:00
|
|
|
@Override
|
|
|
|
public void ping(@NotNull String s) {
|
|
|
|
// TODO: Implement
|
|
|
|
}
|
2019-06-18 03:23:58 +00:00
|
|
|
|
2020-03-24 02:48:23 +00:00
|
|
|
private void setUpPollingIfNeeded() {
|
2020-05-13 23:35:34 +00:00
|
|
|
String userPublicKey = TextSecurePreferences.getLocalNumber(this);
|
|
|
|
if (userPublicKey == null) return;
|
2020-07-15 22:55:29 +00:00
|
|
|
if (poller != null) {
|
2020-07-08 07:05:26 +00:00
|
|
|
SnodeAPI.shared.setUserPublicKey(userPublicKey);
|
2020-07-15 22:55:29 +00:00
|
|
|
poller.setUserPublicKey(userPublicKey);
|
2020-06-24 23:33:44 +00:00
|
|
|
return;
|
|
|
|
}
|
2020-05-13 23:35:34 +00:00
|
|
|
LokiAPIDatabase apiDB = DatabaseFactory.getLokiAPIDatabase(this);
|
2019-06-25 03:51:24 +00:00
|
|
|
Context context = this;
|
2020-07-08 07:05:26 +00:00
|
|
|
SwarmAPI.Companion.configureIfNeeded(apiDB);
|
|
|
|
SnodeAPI.Companion.configureIfNeeded(userPublicKey, apiDB, broadcaster);
|
2020-07-15 22:55:29 +00:00
|
|
|
poller = new Poller(userPublicKey, apiDB, protos -> {
|
2019-08-28 04:19:08 +00:00
|
|
|
for (SignalServiceProtos.Envelope proto : protos) {
|
2020-05-22 03:41:36 +00:00
|
|
|
new PushContentReceiveJob(context).processEnvelope(new SignalServiceEnvelope(proto), false);
|
2019-06-25 03:51:24 +00:00
|
|
|
}
|
2019-08-28 04:19:08 +00:00
|
|
|
return Unit.INSTANCE;
|
2019-06-25 03:51:24 +00:00
|
|
|
});
|
2019-06-18 03:23:58 +00:00
|
|
|
}
|
2019-08-05 02:08:23 +00:00
|
|
|
|
2020-03-24 02:48:23 +00:00
|
|
|
public void startPollingIfNeeded() {
|
|
|
|
setUpPollingIfNeeded();
|
2020-07-15 22:55:29 +00:00
|
|
|
if (poller != null) { poller.startIfNeeded(); }
|
2019-08-05 02:08:23 +00:00
|
|
|
}
|
|
|
|
|
2020-07-08 00:48:09 +00:00
|
|
|
public void stopPolling() {
|
2020-07-15 22:55:29 +00:00
|
|
|
if (poller == null) { return; }
|
|
|
|
poller.stopIfNeeded();
|
2020-06-24 23:33:44 +00:00
|
|
|
}
|
|
|
|
|
2020-02-24 03:57:51 +00:00
|
|
|
private void resubmitProfilePictureIfNeeded() {
|
2020-05-13 23:35:34 +00:00
|
|
|
String userPublicKey = TextSecurePreferences.getLocalNumber(this);
|
|
|
|
if (userPublicKey == null) return;
|
2020-02-24 03:57:51 +00:00
|
|
|
long now = new Date().getTime();
|
|
|
|
long lastProfilePictureUpload = TextSecurePreferences.getLastProfilePictureUpload(this);
|
|
|
|
if (now - lastProfilePictureUpload <= 14 * 24 * 60 * 60 * 1000) return;
|
|
|
|
AsyncTask.execute(() -> {
|
|
|
|
String encodedProfileKey = ProfileKeyUtil.generateEncodedProfileKey(this);
|
|
|
|
byte[] profileKey = ProfileKeyUtil.getProfileKeyFromEncodedString(encodedProfileKey);
|
|
|
|
try {
|
2020-05-13 23:35:34 +00:00
|
|
|
File profilePicture = AvatarHelper.getAvatarFile(this, Address.fromSerialized(userPublicKey));
|
2020-02-24 03:57:51 +00:00
|
|
|
StreamDetails stream = new StreamDetails(new FileInputStream(profilePicture), "image/jpeg", profilePicture.length());
|
2020-07-15 02:24:43 +00:00
|
|
|
FileServerAPI.shared.uploadProfilePicture(FileServerAPI.shared.getServer(), profileKey, stream, () -> {
|
2020-02-24 03:57:51 +00:00
|
|
|
TextSecurePreferences.setLastProfilePictureUpload(this, new Date().getTime());
|
|
|
|
TextSecurePreferences.setProfileAvatarId(this, new SecureRandom().nextInt());
|
|
|
|
ProfileKeyUtil.setEncodedProfileKey(this, encodedProfileKey);
|
|
|
|
return Unit.INSTANCE;
|
|
|
|
});
|
|
|
|
} catch (Exception exception) {
|
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-05-13 23:35:34 +00:00
|
|
|
public void updateOpenGroupProfilePicturesIfNeeded() {
|
2019-11-28 01:49:33 +00:00
|
|
|
AsyncTask.execute(() -> {
|
2020-07-15 02:24:43 +00:00
|
|
|
PublicChatAPI publicChatAPI = null;
|
2019-12-16 10:43:08 +00:00
|
|
|
try {
|
2020-07-15 02:24:43 +00:00
|
|
|
publicChatAPI = getPublicChatAPI();
|
2019-12-16 10:43:08 +00:00
|
|
|
} catch (Exception e) {
|
|
|
|
// Do nothing
|
|
|
|
}
|
2020-05-13 23:35:34 +00:00
|
|
|
if (publicChatAPI == null) { return; }
|
|
|
|
byte[] profileKey = ProfileKeyUtil.getProfileKey(this);
|
|
|
|
String url = TextSecurePreferences.getProfilePictureURL(this);
|
|
|
|
String userMasterDevice = TextSecurePreferences.getMasterHexEncodedPublicKey(this);
|
|
|
|
if (userMasterDevice != null) {
|
|
|
|
Recipient userMasterDeviceAsRecipient = Recipient.from(this, Address.fromSerialized(userMasterDevice), false).resolve();
|
|
|
|
profileKey = userMasterDeviceAsRecipient.getProfileKey();
|
|
|
|
url = userMasterDeviceAsRecipient.getProfileAvatar();
|
|
|
|
}
|
|
|
|
Set<String> servers = DatabaseFactory.getLokiThreadDatabase(this).getAllPublicChatServers();
|
|
|
|
for (String server : servers) {
|
|
|
|
if (profileKey != null) {
|
|
|
|
publicChatAPI.setProfilePicture(server, profileKey, url);
|
2019-11-28 01:49:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2019-11-21 22:35:15 +00:00
|
|
|
|
2019-11-21 05:31:01 +00:00
|
|
|
public void clearData() {
|
2020-05-13 23:35:34 +00:00
|
|
|
boolean wasUnlinked = TextSecurePreferences.getWasUnlinked(this);
|
|
|
|
TextSecurePreferences.clearAll(this);
|
|
|
|
TextSecurePreferences.setWasUnlinked(this, wasUnlinked);
|
|
|
|
MasterSecretUtil.clear(this);
|
2020-05-22 03:47:51 +00:00
|
|
|
if (!deleteDatabase("signal.db")) {
|
|
|
|
Log.d("Loki", "Failed to delete database.");
|
|
|
|
}
|
2020-05-25 00:20:18 +00:00
|
|
|
Util.runOnMain(() -> new Handler().postDelayed(ApplicationContext.this::restartApplication, 200));
|
2019-11-21 05:31:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void restartApplication() {
|
2019-12-17 13:27:59 +00:00
|
|
|
Intent intent = new Intent(this, HomeActivity.class);
|
2020-05-13 23:35:34 +00:00
|
|
|
startActivity(Intent.makeRestartActivityTask(intent.getComponent()));
|
2019-11-21 05:31:01 +00:00
|
|
|
Runtime.getRuntime().exit(0);
|
|
|
|
}
|
2020-05-13 23:35:34 +00:00
|
|
|
|
|
|
|
@Override
|
2020-07-15 02:24:43 +00:00
|
|
|
public void sendSessionRequestIfNeeded(@NotNull String publicKey) {
|
2020-07-09 23:46:14 +00:00
|
|
|
// It's never necessary to establish a session with self
|
|
|
|
String userPublicKey = TextSecurePreferences.getLocalNumber(this);
|
|
|
|
if (publicKey.equals(userPublicKey)) { return; }
|
|
|
|
// Check that we don't already have a session
|
|
|
|
SignalProtocolAddress address = new SignalProtocolAddress(publicKey, SignalServiceAddress.DEFAULT_DEVICE_ID);
|
|
|
|
boolean hasSession = new TextSecureSessionStore(this).containsSession(address);
|
|
|
|
if (hasSession) { return; }
|
|
|
|
// Check that we didn't already send or process a session request
|
|
|
|
LokiAPIDatabase apiDB = DatabaseFactory.getLokiAPIDatabase(this);
|
|
|
|
boolean hasSentOrProcessedSessionRequest = (apiDB.getSessionRequestTimestamp(publicKey) != null);
|
|
|
|
if (hasSentOrProcessedSessionRequest) { return; }
|
|
|
|
// Send the session request
|
2020-05-22 00:41:31 +00:00
|
|
|
DatabaseFactory.getLokiAPIDatabase(this).setSessionRequestTimestamp(publicKey, new Date().getTime());
|
2020-07-15 05:14:37 +00:00
|
|
|
PushSessionRequestMessageSendJob job = new PushSessionRequestMessageSendJob(publicKey);
|
|
|
|
jobManager.add(job);
|
2020-05-13 23:35:34 +00:00
|
|
|
}
|
2020-02-13 03:39:29 +00:00
|
|
|
// endregion
|
2014-07-25 22:14:29 +00:00
|
|
|
}
|