org.robolectric.shadows.ShadowAsyncTask Maven / Gradle / Ivy
package org.robolectric.shadows;
import android.os.AsyncTask;
import org.robolectric.annotation.Implementation;
import org.robolectric.annotation.Implements;
import org.robolectric.annotation.RealObject;
import org.robolectric.util.SimpleFuture;
import java.util.concurrent.Callable;
import java.util.concurrent.CancellationException;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executor;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
/**
* Shadow for {@link android.os.AsyncTask}.
*/
@Implements(AsyncTask.class)
public class ShadowAsyncTask {
@RealObject private AsyncTask realAsyncTask;
private final SimpleFuture future;
private final BackgroundWorker worker;
private AsyncTask.Status status = AsyncTask.Status.PENDING;
public ShadowAsyncTask() {
worker = new BackgroundWorker();
future = new SimpleFuture(worker) {
@Override
protected void done() {
status = AsyncTask.Status.FINISHED;
try {
final Result result = get();
try {
ShadowApplication.getInstance().getForegroundThreadScheduler().post(new Runnable() {
@Override
public void run() {
getBridge().onPostExecute(result);
}
});
} catch (Throwable t) {
throw new OnPostExecuteException(t);
}
} catch (CancellationException e) {
ShadowApplication.getInstance().getForegroundThreadScheduler().post(new Runnable() {
@Override
public void run() {
getBridge().onCancelled();
}
});
} catch (InterruptedException e) {
// Ignore.
} catch (OnPostExecuteException e) {
throw new RuntimeException(e.getCause());
} catch (Throwable t) {
throw new RuntimeException("An error occured while executing doInBackground()",
t.getCause());
}
}
};
}
@Implementation
public boolean isCancelled() {
return future.isCancelled();
}
@Implementation
public boolean cancel(boolean mayInterruptIfRunning) {
return future.cancel(mayInterruptIfRunning);
}
@Implementation
public Result get() throws InterruptedException, ExecutionException {
return future.get();
}
@Implementation
public Result get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
return future.get(timeout, unit);
}
@Implementation
public AsyncTask execute(final Params... params) {
status = AsyncTask.Status.RUNNING;
getBridge().onPreExecute();
worker.params = params;
ShadowApplication.getInstance().getBackgroundThreadScheduler().post(new Runnable() {
@Override
public void run() {
future.run();
}
});
return realAsyncTask;
}
@Implementation
public AsyncTask executeOnExecutor(Executor executor, Params... params) {
status = AsyncTask.Status.RUNNING;
getBridge().onPreExecute();
worker.params = params;
executor.execute(new Runnable() {
@Override
public void run() {
future.run();
}
});
return realAsyncTask;
}
@Implementation
public AsyncTask.Status getStatus() {
return status;
}
/**
* Enqueue a call to {@link AsyncTask#onProgressUpdate(Object[])} on UI looper (or run it immediately
* if the looper it is not paused).
*
* @param values The progress values to update the UI with.
* @see AsyncTask#publishProgress(Object[])
*/
@Implementation
public void publishProgress(final Progress... values) {
ShadowApplication.getInstance().getForegroundThreadScheduler().post(new Runnable() {
@Override
public void run() {
getBridge().onProgressUpdate(values);
}
});
}
private ShadowAsyncTaskBridge getBridge() {
return new ShadowAsyncTaskBridge<>(realAsyncTask);
}
private final class BackgroundWorker implements Callable {
Params[] params;
@Override
public Result call() throws Exception {
return getBridge().doInBackground(params);
}
}
private static class OnPostExecuteException extends Exception {
public OnPostExecuteException(Throwable throwable) {
super(throwable);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy