2016-01-01 22:16:41 +01:00
|
|
|
package org.schabi.newpipe;
|
|
|
|
|
|
|
|
import android.app.Application;
|
2017-08-18 14:05:31 +02:00
|
|
|
import android.app.NotificationChannel;
|
|
|
|
import android.app.NotificationManager;
|
2016-01-01 23:04:29 +01:00
|
|
|
import android.content.Context;
|
2017-08-18 14:05:31 +02:00
|
|
|
import android.os.Build;
|
2018-02-10 20:07:17 +01:00
|
|
|
import android.support.annotation.Nullable;
|
2017-09-03 08:04:18 +02:00
|
|
|
import android.util.Log;
|
2016-01-01 22:16:41 +01:00
|
|
|
|
2018-01-30 05:42:52 +01:00
|
|
|
import com.nostra13.universalimageloader.cache.memory.impl.LRULimitedMemoryCache;
|
2016-02-05 17:09:29 +01:00
|
|
|
import com.nostra13.universalimageloader.core.ImageLoader;
|
|
|
|
import com.nostra13.universalimageloader.core.ImageLoaderConfiguration;
|
2018-02-08 21:46:54 +01:00
|
|
|
import com.squareup.leakcanary.LeakCanary;
|
|
|
|
import com.squareup.leakcanary.RefWatcher;
|
2016-02-05 17:09:29 +01:00
|
|
|
|
2016-09-13 22:36:47 +02:00
|
|
|
import org.acra.ACRA;
|
|
|
|
import org.acra.config.ACRAConfiguration;
|
|
|
|
import org.acra.config.ACRAConfigurationException;
|
|
|
|
import org.acra.config.ConfigurationBuilder;
|
|
|
|
import org.acra.sender.ReportSenderFactory;
|
2018-02-21 12:05:23 +01:00
|
|
|
import org.schabi.newpipe.extractor.Downloader;
|
2016-09-27 22:59:04 +02:00
|
|
|
import org.schabi.newpipe.extractor.NewPipe;
|
2016-09-13 23:39:32 +02:00
|
|
|
import org.schabi.newpipe.report.AcraReportSenderFactory;
|
|
|
|
import org.schabi.newpipe.report.ErrorActivity;
|
2017-06-28 07:27:32 +02:00
|
|
|
import org.schabi.newpipe.report.UserAction;
|
2016-08-02 21:17:54 +02:00
|
|
|
import org.schabi.newpipe.settings.SettingsActivity;
|
2017-09-03 08:04:18 +02:00
|
|
|
import org.schabi.newpipe.util.ExtractorHelper;
|
|
|
|
import org.schabi.newpipe.util.StateSaver;
|
2016-08-02 21:17:54 +02:00
|
|
|
|
2017-09-03 08:04:18 +02:00
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.InterruptedIOException;
|
|
|
|
import java.net.SocketException;
|
2018-02-21 07:35:25 +01:00
|
|
|
import java.util.Collections;
|
|
|
|
import java.util.List;
|
2016-01-01 22:16:41 +01:00
|
|
|
|
2017-09-03 08:04:18 +02:00
|
|
|
import io.reactivex.annotations.NonNull;
|
|
|
|
import io.reactivex.exceptions.CompositeException;
|
2018-02-21 07:35:25 +01:00
|
|
|
import io.reactivex.exceptions.MissingBackpressureException;
|
|
|
|
import io.reactivex.exceptions.OnErrorNotImplementedException;
|
2017-09-03 08:04:18 +02:00
|
|
|
import io.reactivex.exceptions.UndeliverableException;
|
|
|
|
import io.reactivex.functions.Consumer;
|
|
|
|
import io.reactivex.plugins.RxJavaPlugins;
|
|
|
|
|
|
|
|
/*
|
2016-01-05 21:41:55 +01:00
|
|
|
* Copyright (C) Hans-Christoph Steiner 2016 <hans@eds.org>
|
|
|
|
* App.java is part of NewPipe.
|
|
|
|
*
|
|
|
|
* NewPipe 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.
|
|
|
|
*
|
|
|
|
* NewPipe 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 NewPipe. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2016-01-01 22:16:41 +01:00
|
|
|
public class App extends Application {
|
2017-09-03 08:04:18 +02:00
|
|
|
protected static final String TAG = App.class.toString();
|
2018-02-08 21:46:54 +01:00
|
|
|
private RefWatcher refWatcher;
|
2017-09-03 08:04:18 +02:00
|
|
|
|
|
|
|
@SuppressWarnings("unchecked")
|
|
|
|
private static final Class<? extends ReportSenderFactory>[] reportSenderFactoryClasses = new Class[]{AcraReportSenderFactory.class};
|
2016-01-01 22:16:41 +01:00
|
|
|
|
2017-09-03 08:04:18 +02:00
|
|
|
@Override
|
|
|
|
protected void attachBaseContext(Context base) {
|
|
|
|
super.attachBaseContext(base);
|
2016-01-01 23:04:29 +01:00
|
|
|
|
2017-09-03 08:04:18 +02:00
|
|
|
initACRA();
|
|
|
|
}
|
2016-09-13 22:36:47 +02:00
|
|
|
|
2016-01-01 22:16:41 +01:00
|
|
|
@Override
|
|
|
|
public void onCreate() {
|
|
|
|
super.onCreate();
|
2016-12-29 20:19:39 +01:00
|
|
|
|
2018-02-08 21:46:54 +01:00
|
|
|
if (LeakCanary.isInAnalyzerProcess(this)) {
|
|
|
|
// This process is dedicated to LeakCanary for heap analysis.
|
|
|
|
// You should not init your app in this process.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
refWatcher = installLeakCanary();
|
|
|
|
|
2017-09-03 08:04:18 +02:00
|
|
|
// Initialize settings first because others inits can use its values
|
|
|
|
SettingsActivity.initSettings(this);
|
2017-08-07 15:02:30 +02:00
|
|
|
|
2018-02-21 12:05:23 +01:00
|
|
|
NewPipe.init(getDownloader());
|
2017-09-03 08:04:18 +02:00
|
|
|
StateSaver.init(this);
|
|
|
|
initNotificationChannel();
|
2016-09-27 22:59:04 +02:00
|
|
|
|
2016-02-05 17:09:29 +01:00
|
|
|
// Initialize image loader
|
2018-02-03 18:36:40 +01:00
|
|
|
ImageLoader.getInstance().init(getImageLoaderConfigurations(10, 50));
|
2016-02-05 17:09:29 +01:00
|
|
|
|
2017-09-03 08:04:18 +02:00
|
|
|
configureRxJavaErrorHandler();
|
2016-01-01 22:16:41 +01:00
|
|
|
}
|
|
|
|
|
2018-02-21 12:05:23 +01:00
|
|
|
protected Downloader getDownloader() {
|
|
|
|
return org.schabi.newpipe.Downloader.init(null);
|
|
|
|
}
|
|
|
|
|
2017-09-03 08:04:18 +02:00
|
|
|
private void configureRxJavaErrorHandler() {
|
|
|
|
// https://github.com/ReactiveX/RxJava/wiki/What's-different-in-2.0#error-handling
|
|
|
|
RxJavaPlugins.setErrorHandler(new Consumer<Throwable>() {
|
|
|
|
@Override
|
|
|
|
public void accept(@NonNull Throwable throwable) throws Exception {
|
2018-02-21 07:35:25 +01:00
|
|
|
Log.e(TAG, "RxJavaPlugins.ErrorHandler called with -> : " +
|
|
|
|
"throwable = [" + throwable.getClass().getName() + "]");
|
2017-09-03 08:04:18 +02:00
|
|
|
|
|
|
|
if (throwable instanceof UndeliverableException) {
|
|
|
|
// As UndeliverableException is a wrapper, get the cause of it to get the "real" exception
|
|
|
|
throwable = throwable.getCause();
|
|
|
|
}
|
|
|
|
|
2018-02-21 07:35:25 +01:00
|
|
|
final List<Throwable> errors;
|
2017-09-03 08:04:18 +02:00
|
|
|
if (throwable instanceof CompositeException) {
|
2018-02-21 07:35:25 +01:00
|
|
|
errors = ((CompositeException) throwable).getExceptions();
|
|
|
|
} else {
|
|
|
|
errors = Collections.singletonList(throwable);
|
2017-09-03 08:04:18 +02:00
|
|
|
}
|
|
|
|
|
2018-02-21 07:35:25 +01:00
|
|
|
for (final Throwable error : errors) {
|
|
|
|
if (isThrowableIgnored(error)) return;
|
|
|
|
if (isThrowableCritical(error)) {
|
|
|
|
reportException(error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2017-09-03 08:04:18 +02:00
|
|
|
|
2018-02-21 07:35:25 +01:00
|
|
|
// Out-of-lifecycle exceptions should only be reported if a debug user wishes so,
|
|
|
|
// When exception is not reported, log it
|
|
|
|
if (isDisposedRxExceptionsReported()) {
|
|
|
|
reportException(throwable);
|
|
|
|
} else {
|
|
|
|
Log.e(TAG, "RxJavaPlugin: Undeliverable Exception received: ", throwable);
|
|
|
|
}
|
2017-09-03 08:04:18 +02:00
|
|
|
}
|
|
|
|
|
2018-02-21 07:35:25 +01:00
|
|
|
private boolean isThrowableIgnored(@NonNull final Throwable throwable) {
|
2017-09-03 08:04:18 +02:00
|
|
|
// Don't crash the application over a simple network problem
|
|
|
|
return ExtractorHelper.hasAssignableCauseThrowable(throwable,
|
2018-02-21 07:35:25 +01:00
|
|
|
IOException.class, SocketException.class, // network api cancellation
|
|
|
|
InterruptedException.class, InterruptedIOException.class); // blocking code disposed
|
|
|
|
}
|
|
|
|
|
|
|
|
private boolean isThrowableCritical(@NonNull final Throwable throwable) {
|
|
|
|
// Though these exceptions cannot be ignored
|
|
|
|
return ExtractorHelper.hasAssignableCauseThrowable(throwable,
|
|
|
|
NullPointerException.class, IllegalArgumentException.class, // bug in app
|
|
|
|
OnErrorNotImplementedException.class, MissingBackpressureException.class,
|
|
|
|
IllegalStateException.class); // bug in operator
|
|
|
|
}
|
|
|
|
|
|
|
|
private void reportException(@NonNull final Throwable throwable) {
|
|
|
|
// Throw uncaught exception that will trigger the report system
|
|
|
|
Thread.currentThread().getUncaughtExceptionHandler()
|
|
|
|
.uncaughtException(Thread.currentThread(), throwable);
|
2017-09-03 08:04:18 +02:00
|
|
|
}
|
|
|
|
});
|
2016-01-01 22:16:41 +01:00
|
|
|
}
|
2016-01-01 23:04:29 +01:00
|
|
|
|
2018-02-03 18:36:40 +01:00
|
|
|
private ImageLoaderConfiguration getImageLoaderConfigurations(final int memoryCacheSizeMb,
|
|
|
|
final int diskCacheSizeMb) {
|
2018-01-30 05:42:52 +01:00
|
|
|
return new ImageLoaderConfiguration.Builder(this)
|
|
|
|
.memoryCache(new LRULimitedMemoryCache(memoryCacheSizeMb * 1024 * 1024))
|
2018-02-03 18:36:40 +01:00
|
|
|
.diskCacheSize(diskCacheSizeMb * 1024 * 1024)
|
2018-02-22 13:25:56 +01:00
|
|
|
.imageDownloader(new ImageDownloader(getApplicationContext()))
|
2018-01-30 05:42:52 +01:00
|
|
|
.build();
|
|
|
|
}
|
|
|
|
|
2017-09-03 08:04:18 +02:00
|
|
|
private void initACRA() {
|
|
|
|
try {
|
|
|
|
final ACRAConfiguration acraConfig = new ConfigurationBuilder(this)
|
|
|
|
.setReportSenderFactoryClasses(reportSenderFactoryClasses)
|
|
|
|
.setBuildConfigClass(BuildConfig.class)
|
|
|
|
.build();
|
|
|
|
ACRA.init(this, acraConfig);
|
|
|
|
} catch (ACRAConfigurationException ace) {
|
|
|
|
ace.printStackTrace();
|
|
|
|
ErrorActivity.reportError(this, ace, null, null, ErrorActivity.ErrorInfo.make(UserAction.SOMETHING_ELSE, "none",
|
|
|
|
"Could not initialize ACRA crash report", R.string.app_ui_crash));
|
|
|
|
}
|
2016-01-02 22:47:21 +01:00
|
|
|
}
|
2017-08-18 14:05:31 +02:00
|
|
|
|
|
|
|
public void initNotificationChannel() {
|
|
|
|
if (Build.VERSION.SDK_INT < android.os.Build.VERSION_CODES.O) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
final String id = getString(R.string.notification_channel_id);
|
|
|
|
final CharSequence name = getString(R.string.notification_channel_name);
|
|
|
|
final String description = getString(R.string.notification_channel_description);
|
|
|
|
|
|
|
|
// Keep this below DEFAULT to avoid making noise on every notification update
|
|
|
|
final int importance = NotificationManager.IMPORTANCE_LOW;
|
|
|
|
|
|
|
|
NotificationChannel mChannel = new NotificationChannel(id, name, importance);
|
|
|
|
mChannel.setDescription(description);
|
|
|
|
|
|
|
|
NotificationManager mNotificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
|
|
|
|
mNotificationManager.createNotificationChannel(mChannel);
|
|
|
|
}
|
2018-01-28 21:03:01 +01:00
|
|
|
|
2018-02-10 20:07:17 +01:00
|
|
|
@Nullable
|
2018-02-08 21:46:54 +01:00
|
|
|
public static RefWatcher getRefWatcher(Context context) {
|
|
|
|
final App application = (App) context.getApplicationContext();
|
|
|
|
return application.refWatcher;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected RefWatcher installLeakCanary() {
|
|
|
|
return RefWatcher.DISABLED;
|
|
|
|
}
|
2018-02-21 07:35:25 +01:00
|
|
|
|
|
|
|
protected boolean isDisposedRxExceptionsReported() {
|
2018-02-21 19:42:54 +01:00
|
|
|
return false;
|
2018-02-21 07:35:25 +01:00
|
|
|
}
|
2016-01-01 22:16:41 +01:00
|
|
|
}
|