/* * 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 . */ package org.thoughtcrime.securesms; import android.annotation.SuppressLint; import androidx.appcompat.app.AppCompatDelegate; import androidx.camera.camera2.Camera2AppConfig; import androidx.camera.core.CameraX; import androidx.lifecycle.DefaultLifecycleObserver; import androidx.lifecycle.LifecycleOwner; import androidx.lifecycle.ProcessLifecycleOwner; import android.content.Context; import android.os.AsyncTask; import android.os.Build; import androidx.annotation.NonNull; import androidx.multidex.MultiDexApplication; import com.google.android.gms.security.ProviderInstaller; import org.conscrypt.Conscrypt; import org.signal.aesgcmprovider.AesGcmProvider; import org.signal.ringrtc.CallConnectionFactory; import org.thoughtcrime.securesms.components.TypingStatusRepository; import org.thoughtcrime.securesms.components.TypingStatusSender; import org.thoughtcrime.securesms.database.DatabaseFactory; import org.thoughtcrime.securesms.database.helpers.SQLCipherOpenHelper; import org.thoughtcrime.securesms.dependencies.ApplicationDependencies; import org.thoughtcrime.securesms.dependencies.ApplicationDependencyProvider; import org.thoughtcrime.securesms.gcm.FcmJobService; import org.thoughtcrime.securesms.insights.InsightsOptOut; import org.thoughtcrime.securesms.jobmanager.JobManager; import org.thoughtcrime.securesms.jobs.CreateSignedPreKeyJob; import org.thoughtcrime.securesms.jobs.FcmRefreshJob; import org.thoughtcrime.securesms.jobs.MultiDeviceContactUpdateJob; import org.thoughtcrime.securesms.jobs.PushNotificationReceiveJob; import org.thoughtcrime.securesms.jobs.StickerPackDownloadJob; import org.thoughtcrime.securesms.keyvalue.SignalStore; import org.thoughtcrime.securesms.megaphone.MegaphoneRepository; import org.thoughtcrime.securesms.logging.AndroidLogger; import org.thoughtcrime.securesms.logging.CustomSignalProtocolLogger; import org.thoughtcrime.securesms.logging.Log; import org.thoughtcrime.securesms.logging.PersistentLogger; import org.thoughtcrime.securesms.logging.SignalUncaughtExceptionHandler; import org.thoughtcrime.securesms.mediasend.camerax.CameraXUtil; import org.thoughtcrime.securesms.migrations.ApplicationMigrations; import org.thoughtcrime.securesms.notifications.MessageNotifier; import org.thoughtcrime.securesms.notifications.NotificationChannels; import org.thoughtcrime.securesms.providers.BlobProvider; import org.thoughtcrime.securesms.push.SignalServiceNetworkAccess; import org.thoughtcrime.securesms.ringrtc.RingRtcLogger; import org.thoughtcrime.securesms.service.DirectoryRefreshListener; import org.thoughtcrime.securesms.service.ExpiringMessageManager; import org.thoughtcrime.securesms.service.IncomingMessageObserver; import org.thoughtcrime.securesms.service.KeyCachingService; import org.thoughtcrime.securesms.service.LocalBackupListener; import org.thoughtcrime.securesms.revealable.ViewOnceMessageManager; import org.thoughtcrime.securesms.service.RotateSenderCertificateListener; import org.thoughtcrime.securesms.service.RotateSignedPreKeyListener; import org.thoughtcrime.securesms.service.UpdateApkRefreshListener; import org.thoughtcrime.securesms.stickers.BlessedPacks; import org.thoughtcrime.securesms.util.FeatureFlags; import org.thoughtcrime.securesms.util.TextSecurePreferences; import org.thoughtcrime.securesms.util.Util; import org.thoughtcrime.securesms.util.concurrent.SignalExecutors; import org.thoughtcrime.securesms.util.dynamiclanguage.DynamicLanguageContextWrapper; import org.webrtc.voiceengine.WebRtcAudioManager; import org.webrtc.voiceengine.WebRtcAudioUtils; import org.whispersystems.libsignal.logging.SignalProtocolLoggerProvider; import java.security.Security; import java.util.HashSet; import java.util.Set; import java.util.concurrent.TimeUnit; /** * 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 */ public class ApplicationContext extends MultiDexApplication implements DefaultLifecycleObserver { private static final String TAG = ApplicationContext.class.getSimpleName(); private ExpiringMessageManager expiringMessageManager; private ViewOnceMessageManager viewOnceMessageManager; private TypingStatusRepository typingStatusRepository; private TypingStatusSender typingStatusSender; private IncomingMessageObserver incomingMessageObserver; private PersistentLogger persistentLogger; private volatile boolean isAppVisible; public static ApplicationContext getInstance(Context context) { return (ApplicationContext)context.getApplicationContext(); } @Override public void onCreate() { super.onCreate(); Log.i(TAG, "onCreate()"); initializeSecurityProvider(); initializeLogging(); initializeCrashHandling(); initializeAppDependencies(); initializeFirstEverAppLaunch(); initializeApplicationMigrations(); initializeMessageRetrieval(); initializeExpiringMessageManager(); initializeRevealableMessageManager(); initializeTypingStatusRepository(); initializeTypingStatusSender(); initializeGcmCheck(); initializeSignedPreKeyCheck(); initializePeriodicTasks(); initializeCircumvention(); initializeRingRtc(); initializePendingMessages(); initializeBlobProvider(); initializeCleanup(); initializeCameraX(); FeatureFlags.init(); NotificationChannels.create(this); ProcessLifecycleOwner.get().getLifecycle().addObserver(this); if (Build.VERSION.SDK_INT < 21) { AppCompatDelegate.setCompatVectorFromResourcesEnabled(true); } ApplicationDependencies.getJobManager().beginJobLoop(); } @Override public void onStart(@NonNull LifecycleOwner owner) { isAppVisible = true; Log.i(TAG, "App is now visible."); FeatureFlags.refresh(); ApplicationDependencies.getRecipientCache().warmUp(); executePendingContactSync(); KeyCachingService.onAppForegrounded(this); ApplicationDependencies.getFrameRateTracker().begin(); ApplicationDependencies.getMegaphoneRepository().onAppForegrounded(); } @Override public void onStop(@NonNull LifecycleOwner owner) { isAppVisible = false; Log.i(TAG, "App is no longer visible."); KeyCachingService.onAppBackgrounded(this); MessageNotifier.setVisibleThread(-1); ApplicationDependencies.getFrameRateTracker().end(); } public ExpiringMessageManager getExpiringMessageManager() { return expiringMessageManager; } public ViewOnceMessageManager getViewOnceMessageManager() { return viewOnceMessageManager; } public TypingStatusRepository getTypingStatusRepository() { return typingStatusRepository; } public TypingStatusSender getTypingStatusSender() { return typingStatusSender; } public boolean isAppVisible() { return isAppVisible; } public PersistentLogger getPersistentLogger() { return persistentLogger; } 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."); } } private void initializeLogging() { 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(); Thread.setDefaultUncaughtExceptionHandler(new SignalUncaughtExceptionHandler(originalHandler)); } private void initializeApplicationMigrations() { ApplicationMigrations.onApplicationCreate(this, ApplicationDependencies.getJobManager()); } public void initializeMessageRetrieval() { this.incomingMessageObserver = new IncomingMessageObserver(this); } private void initializeAppDependencies() { ApplicationDependencies.init(this, new ApplicationDependencyProvider(this, new SignalServiceNetworkAccess(this))); } private void initializeFirstEverAppLaunch() { if (TextSecurePreferences.getFirstInstallVersion(this) == -1) { if (!SQLCipherOpenHelper.databaseFileExists(this)) { Log.i(TAG, "First ever app launch!"); InsightsOptOut.userRequestedOptOut(this); TextSecurePreferences.setAppMigrationVersion(this, ApplicationMigrations.CURRENT_VERSION); TextSecurePreferences.setJobManagerVersion(this, JobManager.CURRENT_VERSION); TextSecurePreferences.setLastExperienceVersionCode(this, Util.getCanonicalVersionCode()); TextSecurePreferences.setHasSeenStickerIntroTooltip(this, true); ApplicationDependencies.getMegaphoneRepository().onFirstEverAppLaunch(); SignalStore.registrationValues().onNewInstall(); ApplicationDependencies.getJobManager().add(StickerPackDownloadJob.forInstall(BlessedPacks.ZOZO.getPackId(), BlessedPacks.ZOZO.getPackKey(), false)); ApplicationDependencies.getJobManager().add(StickerPackDownloadJob.forInstall(BlessedPacks.BANDIT.getPackId(), BlessedPacks.BANDIT.getPackKey(), false)); ApplicationDependencies.getJobManager().add(StickerPackDownloadJob.forReference(BlessedPacks.SWOON_HANDS.getPackId(), BlessedPacks.SWOON_HANDS.getPackKey())); ApplicationDependencies.getJobManager().add(StickerPackDownloadJob.forReference(BlessedPacks.SWOON_FACES.getPackId(), BlessedPacks.SWOON_FACES.getPackKey())); } Log.i(TAG, "Setting first install version to " + BuildConfig.CANONICAL_VERSION_CODE); TextSecurePreferences.setFirstInstallVersion(this, BuildConfig.CANONICAL_VERSION_CODE); } } private void initializeGcmCheck() { if (TextSecurePreferences.isPushRegistered(this)) { long nextSetTime = TextSecurePreferences.getFcmTokenLastSetTime(this) + TimeUnit.HOURS.toMillis(6); if (TextSecurePreferences.getFcmToken(this) == null || nextSetTime <= System.currentTimeMillis()) { ApplicationDependencies.getJobManager().add(new FcmRefreshJob()); } } } private void initializeSignedPreKeyCheck() { if (!TextSecurePreferences.isSignedPreKeyRegistered(this)) { ApplicationDependencies.getJobManager().add(new CreateSignedPreKeyJob(this)); } } private void initializeExpiringMessageManager() { this.expiringMessageManager = new ExpiringMessageManager(this); } private void initializeRevealableMessageManager() { this.viewOnceMessageManager = new ViewOnceMessageManager(this); } private void initializeTypingStatusRepository() { this.typingStatusRepository = new TypingStatusRepository(); } private void initializeTypingStatusSender() { this.typingStatusSender = new TypingStatusSender(this); } private void initializePeriodicTasks() { RotateSignedPreKeyListener.schedule(this); DirectoryRefreshListener.schedule(this); LocalBackupListener.schedule(this); RotateSenderCertificateListener.schedule(this); if (BuildConfig.PLAY_STORE_DISABLED) { UpdateApkRefreshListener.schedule(this); } } private void initializeRingRtc() { try { Set HARDWARE_AEC_BLACKLIST = new HashSet() {{ add("Pixel"); add("Pixel XL"); add("Moto G5"); add("Moto G (5S) Plus"); add("Moto G4"); add("TA-1053"); add("Mi A1"); add("Mi A2"); add("E5823"); // Sony z5 compact add("Redmi Note 5"); add("FP2"); // Fairphone FP2 add("MI 5"); }}; Set OPEN_SL_ES_WHITELIST = new HashSet() {{ add("Pixel"); add("Pixel XL"); }}; if (HARDWARE_AEC_BLACKLIST.contains(Build.MODEL)) { WebRtcAudioUtils.setWebRtcBasedAcousticEchoCanceler(true); } if (!OPEN_SL_ES_WHITELIST.contains(Build.MODEL)) { WebRtcAudioManager.setBlacklistDeviceForOpenSLESUsage(true); } CallConnectionFactory.initialize(this, new RingRtcLogger()); } catch (UnsatisfiedLinkError e) { Log.w(TAG, e); } } @SuppressLint("StaticFieldLeak") private void initializeCircumvention() { AsyncTask task = new AsyncTask() { @Override protected Void doInBackground(Void... params) { if (new SignalServiceNetworkAccess(ApplicationContext.this).isCensored(ApplicationContext.this)) { try { ProviderInstaller.installIfNeeded(ApplicationContext.this); } catch (Throwable t) { Log.w(TAG, t); } } return null; } }; task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); } private void executePendingContactSync() { if (TextSecurePreferences.needsFullContactSync(this)) { ApplicationDependencies.getJobManager().add(new MultiDeviceContactUpdateJob(true)); } } private void initializePendingMessages() { if (TextSecurePreferences.getNeedsMessagePull(this)) { Log.i(TAG, "Scheduling a message fetch."); if (Build.VERSION.SDK_INT >= 26) { FcmJobService.schedule(this); } else { ApplicationDependencies.getJobManager().add(new PushNotificationReceiveJob(this)); } TextSecurePreferences.setNeedsMessagePull(this, false); } } private void initializeBlobProvider() { SignalExecutors.BOUNDED.execute(() -> { BlobProvider.getInstance().onSessionStart(this); }); } private void initializeCleanup() { SignalExecutors.BOUNDED.execute(() -> { int deleted = DatabaseFactory.getAttachmentDatabase(this).deleteAbandonedPreuploadedAttachments(); Log.i(TAG, "Deleted " + deleted + " abandoned attachments."); }); } @SuppressLint("RestrictedApi") private void initializeCameraX() { if (CameraXUtil.isSupported()) { new Thread(() -> { try { CameraX.init(this, Camera2AppConfig.create(this)); } catch (Throwable t) { Log.w(TAG, "Failed to initialize CameraX."); } }, "signal-camerax-initialization").start(); } } @Override protected void attachBaseContext(Context base) { super.attachBaseContext(DynamicLanguageContextWrapper.updateContext(base, TextSecurePreferences.getLanguage(base))); } private static class ProviderInitializationException extends RuntimeException { } }