2018-02-26 09:58:18 -08:00
|
|
|
package org.thoughtcrime.securesms.jobs;
|
|
|
|
|
|
|
|
|
|
|
|
import android.Manifest;
|
2020-08-19 10:06:26 +10:00
|
|
|
import androidx.annotation.NonNull;
|
2018-08-09 10:15:43 -04:00
|
|
|
|
2019-02-07 09:47:06 -08:00
|
|
|
import org.thoughtcrime.securesms.backup.BackupPassphrase;
|
2018-02-26 09:58:18 -08:00
|
|
|
import org.thoughtcrime.securesms.backup.FullBackupExporter;
|
|
|
|
import org.thoughtcrime.securesms.crypto.AttachmentSecretProvider;
|
|
|
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
|
|
|
import org.thoughtcrime.securesms.database.NoExternalStorageException;
|
2020-05-29 14:08:13 +10:00
|
|
|
import org.thoughtcrime.securesms.jobmanager.Data;
|
|
|
|
import org.thoughtcrime.securesms.jobmanager.Job;
|
|
|
|
import org.thoughtcrime.securesms.logging.Log;
|
2018-08-06 12:20:24 -04:00
|
|
|
import org.thoughtcrime.securesms.notifications.NotificationChannels;
|
2018-02-26 09:58:18 -08:00
|
|
|
import org.thoughtcrime.securesms.permissions.Permissions;
|
|
|
|
import org.thoughtcrime.securesms.service.GenericForegroundService;
|
2020-09-11 16:29:08 +10:00
|
|
|
import org.thoughtcrime.securesms.util.BackupUtilOld;
|
2020-09-04 17:27:57 +10:00
|
|
|
import org.thoughtcrime.securesms.util.ExternalStorageUtil;
|
2018-02-26 09:58:18 -08:00
|
|
|
|
|
|
|
import java.io.File;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.text.SimpleDateFormat;
|
|
|
|
import java.util.Date;
|
|
|
|
import java.util.Locale;
|
2018-08-09 10:15:43 -04:00
|
|
|
|
2020-05-29 14:08:13 +10:00
|
|
|
import network.loki.messenger.R;
|
|
|
|
|
2020-09-11 12:11:46 +10:00
|
|
|
//TODO AC: Needs to be refactored to use Storage Access Framework or Media Store API.
|
2019-03-28 08:56:35 -07:00
|
|
|
public class LocalBackupJob extends BaseJob {
|
2018-02-26 09:58:18 -08:00
|
|
|
|
2019-03-28 08:56:35 -07:00
|
|
|
public static final String KEY = "LocalBackupJob";
|
2018-02-26 09:58:18 -08:00
|
|
|
|
|
|
|
private static final String TAG = LocalBackupJob.class.getSimpleName();
|
|
|
|
|
2019-03-28 08:56:35 -07:00
|
|
|
public LocalBackupJob() {
|
|
|
|
this(new Job.Parameters.Builder()
|
|
|
|
.setQueue("__LOCAL_BACKUP__")
|
|
|
|
.setMaxInstances(1)
|
|
|
|
.setMaxAttempts(3)
|
|
|
|
.build());
|
2018-08-09 10:15:43 -04:00
|
|
|
}
|
|
|
|
|
2019-03-28 08:56:35 -07:00
|
|
|
private LocalBackupJob(@NonNull Job.Parameters parameters) {
|
|
|
|
super(parameters);
|
2018-02-26 09:58:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2019-03-28 08:56:35 -07:00
|
|
|
public @NonNull Data serialize() {
|
|
|
|
return Data.EMPTY;
|
2018-08-09 10:15:43 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2019-03-28 08:56:35 -07:00
|
|
|
public @NonNull String getFactoryKey() {
|
|
|
|
return KEY;
|
2018-08-09 10:15:43 -04:00
|
|
|
}
|
2018-02-26 09:58:18 -08:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onRun() throws NoExternalStorageException, IOException {
|
2018-08-02 09:25:33 -04:00
|
|
|
Log.i(TAG, "Executing backup job...");
|
2018-02-26 09:58:18 -08:00
|
|
|
|
|
|
|
if (!Permissions.hasAll(context, Manifest.permission.WRITE_EXTERNAL_STORAGE)) {
|
|
|
|
throw new IOException("No external storage permission!");
|
|
|
|
}
|
|
|
|
|
2018-04-06 13:13:53 -04:00
|
|
|
GenericForegroundService.startForegroundTask(context,
|
2018-08-06 12:20:24 -04:00
|
|
|
context.getString(R.string.LocalBackupJob_creating_backup),
|
2018-10-27 23:34:18 -07:00
|
|
|
NotificationChannels.BACKUPS,
|
2020-05-28 14:55:17 +10:00
|
|
|
R.drawable.ic_launcher_foreground);
|
2020-05-29 14:08:13 +10:00
|
|
|
|
|
|
|
// TODO: Maybe create a new backup icon like ic_signal_backup?
|
|
|
|
|
2018-02-26 09:58:18 -08:00
|
|
|
try {
|
2019-02-07 09:47:06 -08:00
|
|
|
String backupPassword = BackupPassphrase.get(context);
|
2020-09-04 17:27:57 +10:00
|
|
|
File backupDirectory = ExternalStorageUtil.getBackupDir(context);
|
2018-02-26 09:58:18 -08:00
|
|
|
String timestamp = new SimpleDateFormat("yyyy-MM-dd-HH-mm-ss", Locale.US).format(new Date());
|
2020-05-28 14:55:17 +10:00
|
|
|
String fileName = String.format("session-%s.backup", timestamp);
|
2018-02-26 09:58:18 -08:00
|
|
|
File backupFile = new File(backupDirectory, fileName);
|
|
|
|
|
|
|
|
if (backupFile.exists()) {
|
|
|
|
throw new IOException("Backup file already exists?");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (backupPassword == null) {
|
|
|
|
throw new IOException("Backup password is null");
|
|
|
|
}
|
|
|
|
|
2020-09-04 17:27:57 +10:00
|
|
|
File tempFile = File.createTempFile("backup", "tmp", ExternalStorageUtil.getCacheDir(context));
|
2018-02-26 09:58:18 -08:00
|
|
|
|
|
|
|
FullBackupExporter.export(context,
|
|
|
|
AttachmentSecretProvider.getInstance(context).getOrCreateAttachmentSecret(),
|
|
|
|
DatabaseFactory.getBackupDatabase(context),
|
|
|
|
tempFile,
|
|
|
|
backupPassword);
|
|
|
|
|
|
|
|
if (!tempFile.renameTo(backupFile)) {
|
|
|
|
tempFile.delete();
|
|
|
|
throw new IOException("Renaming temporary backup file failed!");
|
|
|
|
}
|
|
|
|
|
2020-09-11 16:29:08 +10:00
|
|
|
BackupUtilOld.deleteOldBackups(context);
|
2018-02-26 09:58:18 -08:00
|
|
|
} finally {
|
|
|
|
GenericForegroundService.stopForegroundTask(context);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2019-05-22 13:51:56 -03:00
|
|
|
public boolean onShouldRetry(@NonNull Exception e) {
|
2018-02-26 09:58:18 -08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onCanceled() {
|
2019-03-28 08:56:35 -07:00
|
|
|
}
|
2018-02-26 09:58:18 -08:00
|
|
|
|
2019-03-28 08:56:35 -07:00
|
|
|
public static class Factory implements Job.Factory<LocalBackupJob> {
|
|
|
|
@Override
|
|
|
|
public @NonNull LocalBackupJob create(@NonNull Parameters parameters, @NonNull Data data) {
|
|
|
|
return new LocalBackupJob(parameters);
|
|
|
|
}
|
2018-02-26 09:58:18 -08:00
|
|
|
}
|
|
|
|
}
|