2018-06-18 12:27:04 -07:00
|
|
|
package org.thoughtcrime.securesms.jobmanager;
|
|
|
|
|
2018-11-27 12:34:42 -08:00
|
|
|
import android.annotation.SuppressLint;
|
2018-08-09 10:15:43 -04:00
|
|
|
import android.content.Context;
|
|
|
|
import android.support.annotation.NonNull;
|
|
|
|
import android.support.annotation.Nullable;
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
import org.thoughtcrime.securesms.ApplicationContext;
|
2018-10-03 09:42:35 -07:00
|
|
|
import org.thoughtcrime.securesms.R;
|
2018-08-09 10:15:43 -04:00
|
|
|
import org.thoughtcrime.securesms.jobmanager.dependencies.ContextDependent;
|
2018-10-03 09:42:35 -07:00
|
|
|
import org.thoughtcrime.securesms.jobmanager.requirements.NetworkRequirement;
|
2018-08-09 10:15:43 -04:00
|
|
|
import org.thoughtcrime.securesms.jobs.requirements.SqlCipherMigrationRequirement;
|
|
|
|
import org.thoughtcrime.securesms.logging.Log;
|
2018-10-03 09:42:35 -07:00
|
|
|
import org.thoughtcrime.securesms.service.GenericForegroundService;
|
2018-06-18 12:27:04 -07:00
|
|
|
|
|
|
|
import java.io.Serializable;
|
2018-11-27 12:34:42 -08:00
|
|
|
import java.util.Collections;
|
2018-08-09 10:15:43 -04:00
|
|
|
import java.util.UUID;
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
import androidx.work.Data;
|
2019-01-10 16:27:01 -08:00
|
|
|
import androidx.work.ListenableWorker.Result;
|
2018-08-09 10:15:43 -04:00
|
|
|
import androidx.work.Worker;
|
|
|
|
import androidx.work.WorkerParameters;
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
public abstract class Job extends Worker implements Serializable {
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
private static final long serialVersionUID = -4658540468214421276L;
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
private static final String TAG = Job.class.getSimpleName();
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-10-28 22:13:51 -07:00
|
|
|
private static final WorkLockManager WORK_LOCK_MANAGER = new WorkLockManager();
|
|
|
|
|
2018-11-15 12:05:08 -08:00
|
|
|
static final String KEY_RETRY_COUNT = "Job_retry_count";
|
|
|
|
static final String KEY_RETRY_UNTIL = "Job_retry_until";
|
|
|
|
static final String KEY_SUBMIT_TIME = "Job_submit_time";
|
2018-12-07 12:16:37 -08:00
|
|
|
static final String KEY_FAILED = "Job_failed";
|
2018-11-15 12:05:08 -08:00
|
|
|
static final String KEY_REQUIRES_NETWORK = "Job_requires_network";
|
|
|
|
static final String KEY_REQUIRES_SQLCIPHER = "Job_requires_sqlcipher";
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
private JobParameters parameters;
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
public Job(@NonNull Context context, @NonNull WorkerParameters workerParams) {
|
|
|
|
super(context, workerParams);
|
2018-06-18 12:27:04 -07:00
|
|
|
}
|
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
/**
|
|
|
|
* Invoked when a job is first created in our own codebase.
|
|
|
|
*/
|
2018-11-27 12:34:42 -08:00
|
|
|
@SuppressLint("RestrictedApi")
|
|
|
|
protected Job(@NonNull Context context, @Nullable JobParameters parameters) {
|
|
|
|
//noinspection ConstantConditions
|
|
|
|
super(context, new WorkerParameters(null, null, Collections.emptySet(), null, 0, null, null, null));
|
2018-08-09 10:15:43 -04:00
|
|
|
this.parameters = parameters;
|
2018-06-19 19:22:39 -07:00
|
|
|
}
|
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
@Override
|
2018-10-28 22:13:51 -07:00
|
|
|
public @NonNull Result doWork() {
|
|
|
|
log("doWork()" + logSuffix());
|
|
|
|
|
|
|
|
try (WorkLockManager.WorkLock workLock = WORK_LOCK_MANAGER.acquire(getId())) {
|
|
|
|
Result result = workLock.getResult();
|
|
|
|
|
|
|
|
if (result == null) {
|
|
|
|
result = doWorkInternal();
|
|
|
|
workLock.setResult(result);
|
|
|
|
} else {
|
|
|
|
log("Using result from preempted run (" + result + ")." + logSuffix());
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private @NonNull Result doWorkInternal() {
|
2018-08-09 10:15:43 -04:00
|
|
|
Data data = getInputData();
|
2018-06-19 19:22:39 -07:00
|
|
|
|
2018-10-28 22:13:51 -07:00
|
|
|
log("doWorkInternal()" + logSuffix());
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
ApplicationContext.getInstance(getApplicationContext()).injectDependencies(this);
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
if (this instanceof ContextDependent) {
|
|
|
|
((ContextDependent)this).setContext(getApplicationContext());
|
|
|
|
}
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
try {
|
2018-11-17 09:56:52 -08:00
|
|
|
initialize(new SafeData(data));
|
|
|
|
|
2018-12-07 12:16:37 -08:00
|
|
|
if (data.getBoolean(KEY_FAILED, false)) {
|
|
|
|
warn("Failing due to a failure earlier in the chain." + logSuffix());
|
|
|
|
return cancel();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!withinRetryLimits(data)) {
|
2018-08-09 10:15:43 -04:00
|
|
|
warn("Failing after hitting the retry limit." + logSuffix());
|
|
|
|
return cancel();
|
|
|
|
}
|
2018-12-07 12:16:37 -08:00
|
|
|
|
|
|
|
if (!requirementsMet(data)) {
|
|
|
|
log("Retrying due to unmet requirements." + logSuffix());
|
|
|
|
return retry();
|
|
|
|
}
|
|
|
|
|
|
|
|
onRun();
|
|
|
|
|
|
|
|
log("Successfully completed." + logSuffix());
|
|
|
|
return success();
|
2018-08-09 10:15:43 -04:00
|
|
|
} catch (Exception e) {
|
|
|
|
if (onShouldRetry(e)) {
|
2018-10-01 16:14:45 -07:00
|
|
|
log("Retrying after a retryable exception." + logSuffix(), e);
|
2018-08-09 10:15:43 -04:00
|
|
|
return retry();
|
|
|
|
}
|
|
|
|
warn("Failing due to an exception." + logSuffix(), e);
|
|
|
|
return cancel();
|
|
|
|
}
|
2018-06-18 12:27:04 -07:00
|
|
|
}
|
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
@Override
|
2018-11-27 12:34:42 -08:00
|
|
|
public void onStopped() {
|
|
|
|
log("onStopped()" + logSuffix());
|
2018-06-18 12:27:04 -07:00
|
|
|
}
|
|
|
|
|
2018-10-20 22:52:14 -07:00
|
|
|
final void onSubmit(@NonNull Context context, @NonNull UUID id) {
|
2018-12-06 12:14:20 -08:00
|
|
|
Log.i(TAG, buildLog(id, "onSubmit() network: " + (new NetworkRequirement(getApplicationContext()).isPresent())));
|
2018-10-20 22:52:14 -07:00
|
|
|
|
|
|
|
if (this instanceof ContextDependent) {
|
|
|
|
((ContextDependent) this).setContext(context);
|
|
|
|
}
|
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
onAdded();
|
2018-06-18 12:27:04 -07:00
|
|
|
}
|
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
/**
|
|
|
|
* Called after a run has finished and we've determined a retry is required, but before the next
|
|
|
|
* attempt is run.
|
|
|
|
*/
|
|
|
|
protected void onRetry() { }
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
/**
|
|
|
|
* Called after a job has been added to the JobManager queue. Invoked off the main thread, so its
|
|
|
|
* safe to do longer-running work. However, work should finish relatively quickly, as it will
|
|
|
|
* block the submission of future tasks.
|
|
|
|
*/
|
|
|
|
protected void onAdded() { }
|
2018-06-19 19:22:39 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
/**
|
|
|
|
* All instance state needs to be persisted in the provided {@link Data.Builder} so that it can
|
|
|
|
* be restored in {@link #initialize(SafeData)}.
|
|
|
|
* @param dataBuilder The builder where you put your state.
|
|
|
|
* @return The result of {@code dataBuilder.build()}.
|
|
|
|
*/
|
|
|
|
protected abstract @NonNull Data serialize(@NonNull Data.Builder dataBuilder);
|
2018-06-19 19:22:39 -07:00
|
|
|
|
2018-06-18 12:27:04 -07:00
|
|
|
/**
|
2018-08-09 10:15:43 -04:00
|
|
|
* Restore all of your instance state from the provided {@link Data}. It should contain all of
|
|
|
|
* the data put in during {@link #serialize(Data.Builder)}.
|
|
|
|
* @param data Where your data is stored.
|
2018-06-18 12:27:04 -07:00
|
|
|
*/
|
2018-08-09 10:15:43 -04:00
|
|
|
protected abstract void initialize(@NonNull SafeData data);
|
2018-06-18 12:27:04 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Called to actually execute the job.
|
|
|
|
* @throws Exception
|
|
|
|
*/
|
2018-08-09 10:15:43 -04:00
|
|
|
public abstract void onRun() throws Exception;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Called if a job fails to run (onShouldRetry returned false, or the number of retries exceeded
|
|
|
|
* the job's configured retry count.
|
|
|
|
*/
|
|
|
|
protected abstract void onCanceled();
|
2018-06-18 12:27:04 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* If onRun() throws an exception, this method will be called to determine whether the
|
|
|
|
* job should be retried.
|
|
|
|
*
|
|
|
|
* @param exception The exception onRun() threw.
|
|
|
|
* @return true if onRun() should be called again, false otherwise.
|
|
|
|
*/
|
2018-08-09 10:15:43 -04:00
|
|
|
protected abstract boolean onShouldRetry(Exception exception);
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
@Nullable JobParameters getJobParameters() {
|
|
|
|
return parameters;
|
|
|
|
}
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-12-07 12:16:37 -08:00
|
|
|
private Result success() {
|
|
|
|
return Result.success();
|
|
|
|
}
|
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
private Result retry() {
|
|
|
|
onRetry();
|
2018-12-07 09:39:08 -08:00
|
|
|
return Result.retry();
|
2018-08-09 10:15:43 -04:00
|
|
|
}
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-08-09 10:15:43 -04:00
|
|
|
private Result cancel() {
|
|
|
|
onCanceled();
|
2018-12-07 12:16:37 -08:00
|
|
|
return Result.success(new Data.Builder().putBoolean(KEY_FAILED, true).build());
|
2018-08-09 10:15:43 -04:00
|
|
|
}
|
2018-06-18 12:27:04 -07:00
|
|
|
|
2018-10-03 09:42:35 -07:00
|
|
|
private boolean requirementsMet(@NonNull Data data) {
|
2018-08-09 10:15:43 -04:00
|
|
|
boolean met = true;
|
|
|
|
|
|
|
|
if (data.getBoolean(KEY_REQUIRES_SQLCIPHER, false)) {
|
|
|
|
met &= new SqlCipherMigrationRequirement(getApplicationContext()).isPresent();
|
|
|
|
}
|
|
|
|
|
|
|
|
return met;
|
|
|
|
}
|
|
|
|
|
2018-10-03 09:42:35 -07:00
|
|
|
private boolean withinRetryLimits(@NonNull Data data) {
|
2018-08-09 10:15:43 -04:00
|
|
|
int retryCount = data.getInt(KEY_RETRY_COUNT, 0);
|
|
|
|
long retryUntil = data.getLong(KEY_RETRY_UNTIL, 0);
|
|
|
|
|
|
|
|
if (retryCount > 0) {
|
|
|
|
return getRunAttemptCount() <= retryCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
return System.currentTimeMillis() < retryUntil;
|
|
|
|
}
|
|
|
|
|
|
|
|
private void log(@NonNull String message) {
|
2018-10-01 16:14:45 -07:00
|
|
|
log(message, null);
|
2018-08-09 10:15:43 -04:00
|
|
|
}
|
|
|
|
|
2018-10-01 16:14:45 -07:00
|
|
|
private void log(@NonNull String message, @Nullable Throwable t) {
|
|
|
|
Log.i(TAG, buildLog(getId(), message), t);
|
2018-08-09 10:15:43 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
private void warn(@NonNull String message) {
|
|
|
|
warn(message, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void warn(@NonNull String message, @Nullable Throwable t) {
|
|
|
|
Log.w(TAG, buildLog(getId(), message), t);
|
|
|
|
}
|
|
|
|
|
|
|
|
private String buildLog(@NonNull UUID id, @NonNull String message) {
|
|
|
|
return "[" + id + "] " + getClass().getSimpleName() + " :: " + message;
|
|
|
|
}
|
|
|
|
|
2018-12-06 12:14:20 -08:00
|
|
|
protected String logSuffix() {
|
2018-08-09 10:15:43 -04:00
|
|
|
long timeSinceSubmission = System.currentTimeMillis() - getInputData().getLong(KEY_SUBMIT_TIME, 0);
|
2018-10-28 22:13:51 -07:00
|
|
|
return " (Time since submission: " + timeSinceSubmission + " ms, Run attempt: " + getRunAttemptCount() + ", isStopped: " + isStopped() + ")";
|
2018-08-09 10:15:43 -04:00
|
|
|
}
|
2018-06-18 12:27:04 -07:00
|
|
|
}
|