285 lines
8.4 KiB
Java
285 lines
8.4 KiB
Java
/*
|
|
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
|
|
*/
|
|
package net.nullsum.audinaut.util;
|
|
|
|
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;
|
|
|
|
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 {
|
|
static final BlockingQueue<BackgroundTask.Task> queue = new LinkedBlockingQueue<>(10);
|
|
private static final String TAG = BackgroundTask.class.getSimpleName();
|
|
private static final int DEFAULT_CONCURRENCY = 8;
|
|
private static final Collection<Thread> threads = Collections.synchronizedCollection(new ArrayList<Thread>());
|
|
private static Handler handler = null;
|
|
|
|
static {
|
|
try {
|
|
handler = new Handler(Looper.getMainLooper());
|
|
} catch (Exception e) {
|
|
// Not called from main thread
|
|
}
|
|
}
|
|
|
|
final AtomicBoolean cancelled = new AtomicBoolean(false);
|
|
private final Context context;
|
|
private final Runnable onCompletionListener = null;
|
|
Task task;
|
|
|
|
BackgroundTask(Context context) {
|
|
this.context = context;
|
|
|
|
if (threads.size() < DEFAULT_CONCURRENCY) {
|
|
for (int i = threads.size(); i < DEFAULT_CONCURRENCY; i++) {
|
|
Thread thread = new Thread(new TaskRunnable(), String.format("BackgroundTask_%d", i));
|
|
threads.add(thread);
|
|
thread.start();
|
|
}
|
|
}
|
|
if (handler == null) {
|
|
try {
|
|
handler = new Handler(Looper.getMainLooper());
|
|
} catch (Exception e) {
|
|
// Not called from main thread
|
|
}
|
|
}
|
|
}
|
|
|
|
private Activity getActivity() {
|
|
return (context instanceof Activity) ? ((Activity) context) : null;
|
|
}
|
|
|
|
Handler getHandler() {
|
|
return handler;
|
|
}
|
|
|
|
public abstract void execute();
|
|
|
|
protected abstract T doInBackground() throws Throwable;
|
|
|
|
protected abstract void done(T result);
|
|
|
|
protected void error(Throwable error) {
|
|
Log.w(TAG, "Got exception: " + error, error);
|
|
Activity activity = getActivity();
|
|
if (activity != null) {
|
|
new ErrorDialog(activity, getErrorMessage(error), true);
|
|
}
|
|
}
|
|
|
|
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();
|
|
}
|
|
|
|
public void cancel() {
|
|
if (cancelled.compareAndSet(false, true)) {
|
|
if (isRunning()) {
|
|
task.cancel();
|
|
}
|
|
task = null;
|
|
}
|
|
}
|
|
|
|
public boolean isCancelled() {
|
|
return cancelled.get();
|
|
}
|
|
|
|
public boolean isRunning() {
|
|
return task != null && task.isRunning();
|
|
}
|
|
|
|
@Override
|
|
public abstract void updateProgress(final String message);
|
|
|
|
public void updateProgress() {
|
|
updateProgress(context.getResources().getString(R.string.settings_testing_connection));
|
|
}
|
|
|
|
@Override
|
|
public void updateCache(int changeCode) {
|
|
|
|
}
|
|
|
|
class Task {
|
|
private final AtomicBoolean taskStart = new AtomicBoolean(false);
|
|
private Thread thread;
|
|
|
|
private void execute() throws Exception {
|
|
// Don't run if cancelled already
|
|
if (isCancelled()) {
|
|
return;
|
|
}
|
|
|
|
try {
|
|
thread = Thread.currentThread();
|
|
taskStart.set(true);
|
|
|
|
final T result = doInBackground();
|
|
if (isCancelled()) {
|
|
taskStart.set(false);
|
|
return;
|
|
}
|
|
|
|
if (handler != null) {
|
|
handler.post(() -> {
|
|
if (!isCancelled()) {
|
|
try {
|
|
onDone(result);
|
|
} catch (Throwable t) {
|
|
if (!isCancelled()) {
|
|
try {
|
|
onError(t);
|
|
} catch (Exception e) {
|
|
// Don't care
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
taskStart.set(false);
|
|
});
|
|
} else {
|
|
taskStart.set(false);
|
|
}
|
|
} catch (InterruptedException interrupt) {
|
|
if (taskStart.get()) {
|
|
// Don't exit root thread if task cancelled
|
|
throw interrupt;
|
|
}
|
|
} catch (final Throwable t) {
|
|
if (isCancelled()) {
|
|
taskStart.set(false);
|
|
return;
|
|
}
|
|
|
|
if (handler != null) {
|
|
handler.post(() -> {
|
|
if (!isCancelled()) {
|
|
try {
|
|
onError(t);
|
|
} catch (Exception e) {
|
|
// Don't care
|
|
}
|
|
}
|
|
|
|
taskStart.set(false);
|
|
});
|
|
} else {
|
|
taskStart.set(false);
|
|
}
|
|
} finally {
|
|
thread = null;
|
|
}
|
|
}
|
|
|
|
public void cancel() {
|
|
if (taskStart.compareAndSet(true, false)) {
|
|
if (thread != null) {
|
|
thread.interrupt();
|
|
}
|
|
}
|
|
}
|
|
|
|
public boolean isCancelled() {
|
|
return Thread.interrupted() || BackgroundTask.this.isCancelled();
|
|
}
|
|
|
|
public void onDone(T result) {
|
|
done(result);
|
|
|
|
if (onCompletionListener != null) {
|
|
onCompletionListener.run();
|
|
}
|
|
}
|
|
|
|
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();
|
|
while (running) {
|
|
try {
|
|
Task task = queue.take();
|
|
task.execute();
|
|
} catch (InterruptedException stop) {
|
|
Log.e(TAG, "Thread died");
|
|
running = false;
|
|
threads.remove(Thread.currentThread());
|
|
} catch (Throwable t) {
|
|
Log.e(TAG, "Unexpected crash in BackgroundTask thread", t);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|