Audinaut-subsonic-app-android/app/src/main/java/net/nullsum/audinaut/util/BackgroundTask.java

285 lines
8.4 KiB
Java
Raw Normal View History

2016-12-18 18:41:30 +01:00
/*
This file is part of Subsonic.
Subsonic 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.
Subsonic 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 Subsonic. If not, see <http://www.gnu.org/licenses/>.
Copyright 2009 (C) Sindre Mehus
*/
2017-01-09 08:01:12 +01:00
package net.nullsum.audinaut.util;
2016-12-18 18:41:30 +01:00
2018-03-25 03:28:28 +02:00
import android.app.Activity;
import android.content.Context;
import android.os.Handler;
import android.os.Looper;
import android.util.Log;
import net.nullsum.audinaut.R;
import net.nullsum.audinaut.view.ErrorDialog;
import org.xmlpull.v1.XmlPullParserException;
2016-12-18 18:41:30 +01:00
import java.io.FileNotFoundException;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.atomic.AtomicBoolean;
/**
* @author Sindre Mehus
*/
public abstract class BackgroundTask<T> implements ProgressListener {
2018-03-25 03:28:28 +02:00
static final BlockingQueue<BackgroundTask.Task> queue = new LinkedBlockingQueue<>(10);
2016-12-18 18:41:30 +01:00
private static final String TAG = BackgroundTask.class.getSimpleName();
2018-03-24 20:25:12 +01:00
private static final int DEFAULT_CONCURRENCY = 8;
private static final Collection<Thread> threads = Collections.synchronizedCollection(new ArrayList<Thread>());
private static Handler handler = null;
2018-03-25 03:28:28 +02:00
2018-03-24 20:25:12 +01:00
static {
try {
handler = new Handler(Looper.getMainLooper());
2018-03-25 03:28:28 +02:00
} catch (Exception e) {
2018-03-24 20:25:12 +01:00
// Not called from main thread
}
}
2016-12-18 18:41:30 +01:00
2018-03-25 03:28:28 +02:00
final AtomicBoolean cancelled = new AtomicBoolean(false);
private final Context context;
private final Runnable onCompletionListener = null;
Task task;
BackgroundTask(Context context) {
2016-12-18 18:41:30 +01:00
this.context = context;
2018-03-25 03:28:28 +02:00
if (threads.size() < DEFAULT_CONCURRENCY) {
for (int i = threads.size(); i < DEFAULT_CONCURRENCY; i++) {
2018-03-24 20:25:12 +01:00
Thread thread = new Thread(new TaskRunnable(), String.format("BackgroundTask_%d", i));
threads.add(thread);
thread.start();
}
}
2018-03-25 03:28:28 +02:00
if (handler == null) {
2018-03-24 20:25:12 +01:00
try {
handler = new Handler(Looper.getMainLooper());
2018-03-25 03:28:28 +02:00
} catch (Exception e) {
2018-03-24 20:25:12 +01:00
// Not called from main thread
}
}
2016-12-18 18:41:30 +01:00
}
2018-03-25 03:28:28 +02:00
private Activity getActivity() {
2016-12-18 18:41:30 +01:00
return (context instanceof Activity) ? ((Activity) context) : null;
}
2018-03-25 03:28:28 +02:00
Handler getHandler() {
2016-12-18 18:41:30 +01:00
return handler;
}
2018-03-24 20:25:12 +01:00
public abstract void execute();
2016-12-18 18:41:30 +01:00
protected abstract T doInBackground() throws Throwable;
protected abstract void done(T result);
protected void error(Throwable error) {
Log.w(TAG, "Got exception: " + error, error);
2018-03-24 20:25:12 +01:00
Activity activity = getActivity();
2018-03-25 03:28:28 +02:00
if (activity != null) {
2018-03-24 20:25:12 +01:00
new ErrorDialog(activity, getErrorMessage(error), true);
}
2016-12-18 18:41:30 +01:00
}
protected String getErrorMessage(Throwable error) {
if (error instanceof IOException && !Util.isNetworkConnected(context)) {
return context.getResources().getString(R.string.background_task_no_network);
}
if (error instanceof FileNotFoundException) {
return context.getResources().getString(R.string.background_task_not_found);
}
if (error instanceof IOException) {
return context.getResources().getString(R.string.background_task_network_error);
}
if (error instanceof XmlPullParserException) {
return context.getResources().getString(R.string.background_task_parse_error);
}
String message = error.getMessage();
if (message != null) {
return message;
}
return error.getClass().getSimpleName();
}
2018-03-24 20:25:12 +01:00
public void cancel() {
2018-03-25 03:28:28 +02:00
if (cancelled.compareAndSet(false, true)) {
if (isRunning()) {
task.cancel();
2018-03-24 20:25:12 +01:00
}
task = null;
}
}
2018-03-25 03:28:28 +02:00
2018-03-24 20:25:12 +01:00
public boolean isCancelled() {
return cancelled.get();
}
public boolean isRunning() {
2018-03-25 03:28:28 +02:00
return task != null && task.isRunning();
2018-03-24 20:25:12 +01:00
}
2016-12-18 18:41:30 +01:00
@Override
public abstract void updateProgress(final String message);
2018-03-25 03:28:28 +02:00
public void updateProgress() {
updateProgress(context.getResources().getString(R.string.settings_testing_connection));
2016-12-18 18:41:30 +01:00
}
2018-03-24 20:25:12 +01:00
@Override
public void updateCache(int changeCode) {
}
2018-03-25 03:28:28 +02:00
class Task {
private final AtomicBoolean taskStart = new AtomicBoolean(false);
2018-03-24 20:25:12 +01:00
private Thread thread;
private void execute() throws Exception {
// Don't run if cancelled already
2018-03-25 03:28:28 +02:00
if (isCancelled()) {
2018-03-24 20:25:12 +01:00
return;
}
try {
thread = Thread.currentThread();
taskStart.set(true);
final T result = doInBackground();
2018-03-25 03:28:28 +02:00
if (isCancelled()) {
2018-03-24 20:25:12 +01:00
taskStart.set(false);
return;
}
2018-03-25 03:28:28 +02:00
if (handler != null) {
handler.post(() -> {
if (!isCancelled()) {
try {
onDone(result);
} catch (Throwable t) {
if (!isCancelled()) {
try {
onError(t);
} catch (Exception e) {
// Don't care
2018-03-24 20:25:12 +01:00
}
}
}
}
2018-03-25 03:28:28 +02:00
taskStart.set(false);
2018-03-24 20:25:12 +01:00
});
} else {
taskStart.set(false);
}
2018-03-25 03:28:28 +02:00
} catch (InterruptedException interrupt) {
if (taskStart.get()) {
2018-03-24 20:25:12 +01:00
// Don't exit root thread if task cancelled
throw interrupt;
}
2018-03-25 03:28:28 +02:00
} catch (final Throwable t) {
if (isCancelled()) {
2018-03-24 20:25:12 +01:00
taskStart.set(false);
return;
}
2018-03-25 03:28:28 +02:00
if (handler != null) {
handler.post(() -> {
if (!isCancelled()) {
try {
onError(t);
} catch (Exception e) {
// Don't care
2018-03-24 20:25:12 +01:00
}
}
2018-03-25 03:28:28 +02:00
taskStart.set(false);
2018-03-24 20:25:12 +01:00
});
} else {
taskStart.set(false);
}
} finally {
thread = null;
}
}
public void cancel() {
2018-03-25 03:28:28 +02:00
if (taskStart.compareAndSet(true, false)) {
2018-03-24 20:25:12 +01:00
if (thread != null) {
thread.interrupt();
}
}
}
2018-03-25 03:28:28 +02:00
2018-03-24 20:25:12 +01:00
public boolean isCancelled() {
2018-03-25 03:28:28 +02:00
return Thread.interrupted() || BackgroundTask.this.isCancelled();
2018-03-24 20:25:12 +01:00
}
2018-03-25 03:28:28 +02:00
2018-03-24 20:25:12 +01:00
public void onDone(T result) {
done(result);
2018-03-25 03:28:28 +02:00
if (onCompletionListener != null) {
2018-03-24 20:25:12 +01:00
onCompletionListener.run();
}
}
2018-03-25 03:28:28 +02:00
2018-03-24 20:25:12 +01:00
public void onError(Throwable t) {
error(t);
}
public boolean isRunning() {
return taskStart.get();
}
}
private class TaskRunnable implements Runnable {
private boolean running = true;
public TaskRunnable() {
}
@Override
public void run() {
Looper.prepare();
2018-03-25 03:28:28 +02:00
while (running) {
2018-03-24 20:25:12 +01:00
try {
Task task = queue.take();
task.execute();
2018-03-25 03:28:28 +02:00
} catch (InterruptedException stop) {
2018-03-24 20:25:12 +01:00
Log.e(TAG, "Thread died");
running = false;
threads.remove(Thread.currentThread());
2018-03-25 03:28:28 +02:00
} catch (Throwable t) {
2018-03-24 20:25:12 +01:00
Log.e(TAG, "Unexpected crash in BackgroundTask thread", t);
}
}
}
}
2016-12-18 18:41:30 +01:00
}