2021-12-06 18:31:19 +00:00
|
|
|
import 'dart:async';
|
|
|
|
|
2022-02-17 23:37:15 +00:00
|
|
|
import 'package:easy_localization/easy_localization.dart';
|
2021-12-06 18:31:19 +00:00
|
|
|
import 'package:selfprivacy/config/get_it_config.dart';
|
2022-11-16 00:24:40 +00:00
|
|
|
import 'package:selfprivacy/logic/api_maps/graphql_maps/server_api/server_api.dart';
|
2024-01-31 05:14:23 +00:00
|
|
|
import 'package:selfprivacy/logic/api_maps/rest_maps/backblaze.dart';
|
2021-12-06 18:31:19 +00:00
|
|
|
import 'package:selfprivacy/logic/cubit/app_config_dependent/authentication_dependend_cubit.dart';
|
2023-06-29 09:52:09 +00:00
|
|
|
import 'package:selfprivacy/logic/models/backup.dart';
|
2024-01-31 05:14:23 +00:00
|
|
|
import 'package:selfprivacy/logic/models/hive/backblaze_bucket.dart';
|
2023-06-29 09:52:09 +00:00
|
|
|
import 'package:selfprivacy/logic/models/hive/backups_credential.dart';
|
|
|
|
import 'package:selfprivacy/logic/models/initialize_repository_input.dart';
|
|
|
|
import 'package:selfprivacy/logic/models/service.dart';
|
2021-12-06 18:31:19 +00:00
|
|
|
|
|
|
|
part 'backups_state.dart';
|
|
|
|
|
2022-05-17 13:31:34 +00:00
|
|
|
class BackupsCubit extends ServerInstallationDependendCubit<BackupsState> {
|
2022-06-05 19:36:32 +00:00
|
|
|
BackupsCubit(final ServerInstallationCubit serverInstallationCubit)
|
2022-05-24 18:55:39 +00:00
|
|
|
: super(
|
2022-06-05 22:40:34 +00:00
|
|
|
serverInstallationCubit,
|
|
|
|
const BackupsState(preventActions: true),
|
|
|
|
);
|
2021-12-06 18:31:19 +00:00
|
|
|
|
2022-06-05 19:36:32 +00:00
|
|
|
final ServerApi api = ServerApi();
|
|
|
|
final BackblazeApi backblaze = BackblazeApi();
|
2021-12-06 18:31:19 +00:00
|
|
|
|
2022-05-24 18:55:39 +00:00
|
|
|
@override
|
2021-12-06 18:31:19 +00:00
|
|
|
Future<void> load() async {
|
2022-05-17 20:08:28 +00:00
|
|
|
if (serverInstallationCubit.state is ServerInstallationFinished) {
|
2022-06-05 19:36:32 +00:00
|
|
|
final BackblazeBucket? bucket = getIt<ApiConfigModel>().backblazeBucket;
|
2023-06-29 09:52:09 +00:00
|
|
|
final BackupConfiguration? backupConfig =
|
|
|
|
await api.getBackupsConfiguration();
|
|
|
|
final List<Backup> backups = await api.getBackups();
|
2023-07-02 11:41:31 +00:00
|
|
|
backups.sort((final a, final b) => b.time.compareTo(a.time));
|
2023-06-29 09:52:09 +00:00
|
|
|
emit(
|
|
|
|
state.copyWith(
|
|
|
|
backblazeBucket: bucket,
|
|
|
|
isInitialized: backupConfig?.isInitialized,
|
2023-07-02 15:23:12 +00:00
|
|
|
autobackupPeriod: backupConfig?.autobackupPeriod ?? Duration.zero,
|
2023-09-09 18:13:27 +00:00
|
|
|
autobackupQuotas: backupConfig?.autobackupQuotas,
|
2023-06-29 09:52:09 +00:00
|
|
|
backups: backups,
|
|
|
|
preventActions: false,
|
|
|
|
refreshing: false,
|
|
|
|
),
|
|
|
|
);
|
2021-12-06 18:31:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-29 09:52:09 +00:00
|
|
|
Future<void> initializeBackups() async {
|
2021-12-06 18:31:19 +00:00
|
|
|
emit(state.copyWith(preventActions: true));
|
2023-06-29 09:52:09 +00:00
|
|
|
final String? encryptionKey =
|
|
|
|
(await api.getBackupsConfiguration())?.encryptionKey;
|
|
|
|
if (encryptionKey == null) {
|
|
|
|
getIt<NavigationService>()
|
|
|
|
.showSnackBar("Couldn't get encryption key from your server.");
|
|
|
|
emit(state.copyWith(preventActions: false));
|
|
|
|
return;
|
2021-12-06 18:31:19 +00:00
|
|
|
}
|
|
|
|
|
2023-06-29 09:52:09 +00:00
|
|
|
final BackblazeBucket bucket;
|
|
|
|
|
|
|
|
if (state.backblazeBucket == null) {
|
|
|
|
final String domain = serverInstallationCubit
|
|
|
|
.state.serverDomain!.domainName
|
|
|
|
.replaceAll(RegExp(r'[^a-zA-Z0-9]'), '-');
|
|
|
|
final int serverId = serverInstallationCubit.state.serverDetails!.id;
|
2023-12-02 20:44:32 +00:00
|
|
|
String bucketName =
|
2023-12-03 17:58:56 +00:00
|
|
|
'${DateTime.now().millisecondsSinceEpoch}-$serverId-$domain';
|
2023-06-29 09:52:09 +00:00
|
|
|
if (bucketName.length > 49) {
|
|
|
|
bucketName = bucketName.substring(0, 49);
|
|
|
|
}
|
|
|
|
final String bucketId = await backblaze.createBucket(bucketName);
|
|
|
|
|
|
|
|
final BackblazeApplicationKey key = await backblaze.createKey(bucketId);
|
|
|
|
bucket = BackblazeBucket(
|
|
|
|
bucketId: bucketId,
|
|
|
|
bucketName: bucketName,
|
|
|
|
applicationKey: key.applicationKey,
|
|
|
|
applicationKeyId: key.applicationKeyId,
|
|
|
|
encryptionKey: encryptionKey,
|
|
|
|
);
|
|
|
|
|
|
|
|
await getIt<ApiConfigModel>().storeBackblazeBucket(bucket);
|
|
|
|
emit(state.copyWith(backblazeBucket: bucket));
|
|
|
|
} else {
|
|
|
|
bucket = state.backblazeBucket!;
|
|
|
|
}
|
|
|
|
|
|
|
|
final GenericResult result = await api.initializeRepository(
|
|
|
|
InitializeRepositoryInput(
|
|
|
|
provider: BackupsProviderType.backblaze,
|
|
|
|
locationId: bucket.bucketId,
|
|
|
|
locationName: bucket.bucketName,
|
|
|
|
login: bucket.applicationKeyId,
|
|
|
|
password: bucket.applicationKey,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
if (result.success == false) {
|
|
|
|
getIt<NavigationService>()
|
|
|
|
.showSnackBar(result.message ?? 'Unknown error');
|
|
|
|
emit(state.copyWith(preventActions: false));
|
|
|
|
return;
|
|
|
|
}
|
2021-12-06 18:31:19 +00:00
|
|
|
await updateBackups();
|
2023-06-29 09:52:09 +00:00
|
|
|
getIt<NavigationService>().showSnackBar(
|
2023-06-29 10:51:38 +00:00
|
|
|
'Backups repository is now initializing. It may take a while.',
|
|
|
|
);
|
2021-12-06 18:31:19 +00:00
|
|
|
|
2023-06-29 09:52:09 +00:00
|
|
|
emit(state.copyWith(preventActions: false));
|
2021-12-06 18:31:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> reuploadKey() async {
|
|
|
|
emit(state.copyWith(preventActions: true));
|
2023-09-05 21:36:49 +00:00
|
|
|
BackblazeBucket? bucket = getIt<ApiConfigModel>().backblazeBucket;
|
2021-12-06 18:31:19 +00:00
|
|
|
if (bucket == null) {
|
|
|
|
emit(state.copyWith(isInitialized: false));
|
|
|
|
} else {
|
2023-09-05 21:36:49 +00:00
|
|
|
String login = bucket.applicationKeyId;
|
|
|
|
String password = bucket.applicationKey;
|
|
|
|
if (login.isEmpty || password.isEmpty) {
|
|
|
|
final BackblazeApplicationKey key =
|
|
|
|
await backblaze.createKey(bucket.bucketId);
|
|
|
|
login = key.applicationKeyId;
|
|
|
|
password = key.applicationKey;
|
|
|
|
bucket = BackblazeBucket(
|
|
|
|
bucketId: bucket.bucketId,
|
|
|
|
bucketName: bucket.bucketName,
|
|
|
|
encryptionKey: bucket.encryptionKey,
|
|
|
|
applicationKey: password,
|
|
|
|
applicationKeyId: login,
|
|
|
|
);
|
|
|
|
await getIt<ApiConfigModel>().storeBackblazeBucket(bucket);
|
|
|
|
emit(state.copyWith(backblazeBucket: bucket));
|
|
|
|
}
|
2023-06-29 09:52:09 +00:00
|
|
|
final GenericResult result = await api.initializeRepository(
|
|
|
|
InitializeRepositoryInput(
|
|
|
|
provider: BackupsProviderType.backblaze,
|
|
|
|
locationId: bucket.bucketId,
|
|
|
|
locationName: bucket.bucketName,
|
2023-09-05 21:36:49 +00:00
|
|
|
login: login,
|
|
|
|
password: password,
|
2023-06-29 09:52:09 +00:00
|
|
|
),
|
|
|
|
);
|
|
|
|
if (result.success == false) {
|
|
|
|
getIt<NavigationService>()
|
|
|
|
.showSnackBar(result.message ?? 'Unknown error');
|
|
|
|
emit(state.copyWith(preventActions: false));
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
emit(state.copyWith(preventActions: false));
|
2023-09-05 21:36:49 +00:00
|
|
|
getIt<NavigationService>().showSnackBar('backup.reuploaded_key'.tr());
|
2023-06-29 09:52:09 +00:00
|
|
|
await updateBackups();
|
|
|
|
}
|
2021-12-06 18:31:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-29 09:52:09 +00:00
|
|
|
@Deprecated("we don't have states")
|
|
|
|
Duration refreshTimeFromState() => const Duration(seconds: 60);
|
2021-12-06 18:31:19 +00:00
|
|
|
|
2022-06-05 19:36:32 +00:00
|
|
|
Future<void> updateBackups({final bool useTimer = false}) async {
|
2021-12-06 18:31:19 +00:00
|
|
|
emit(state.copyWith(refreshing: true));
|
2023-06-29 09:52:09 +00:00
|
|
|
final backups = await api.getBackups();
|
2023-07-02 11:41:31 +00:00
|
|
|
backups.sort((final a, final b) => b.time.compareTo(a.time));
|
2023-06-29 09:52:09 +00:00
|
|
|
final backupConfig = await api.getBackupsConfiguration();
|
2023-06-22 14:14:00 +00:00
|
|
|
|
2022-06-05 22:40:34 +00:00
|
|
|
emit(
|
|
|
|
state.copyWith(
|
|
|
|
backups: backups,
|
2023-06-29 09:52:09 +00:00
|
|
|
refreshTimer: refreshTimeFromState(),
|
2022-06-05 22:40:34 +00:00
|
|
|
refreshing: false,
|
2023-06-29 09:52:09 +00:00
|
|
|
isInitialized: backupConfig?.isInitialized ?? false,
|
|
|
|
autobackupPeriod: backupConfig?.autobackupPeriod,
|
2023-09-09 18:13:27 +00:00
|
|
|
autobackupQuotas: backupConfig?.autobackupQuotas,
|
2022-06-05 22:40:34 +00:00
|
|
|
),
|
|
|
|
);
|
2022-05-24 18:55:39 +00:00
|
|
|
if (useTimer) {
|
2021-12-06 18:31:19 +00:00
|
|
|
Timer(state.refreshTimer, () => updateBackups(useTimer: true));
|
2022-05-24 18:55:39 +00:00
|
|
|
}
|
2021-12-06 18:31:19 +00:00
|
|
|
}
|
|
|
|
|
2021-12-09 03:35:15 +00:00
|
|
|
Future<void> forceUpdateBackups() async {
|
|
|
|
emit(state.copyWith(preventActions: true));
|
2022-10-03 23:32:35 +00:00
|
|
|
getIt<NavigationService>().showSnackBar('backup.refetching_list'.tr());
|
2023-07-02 15:23:12 +00:00
|
|
|
await api.forceBackupListReload();
|
2021-12-09 03:35:15 +00:00
|
|
|
emit(state.copyWith(preventActions: false));
|
|
|
|
}
|
|
|
|
|
2023-06-29 09:52:09 +00:00
|
|
|
Future<void> createMultipleBackups(final List<Service> services) async {
|
|
|
|
emit(state.copyWith(preventActions: true));
|
|
|
|
for (final service in services) {
|
|
|
|
await api.startBackup(service.id);
|
|
|
|
}
|
|
|
|
await updateBackups();
|
|
|
|
emit(state.copyWith(preventActions: false));
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> createBackup(final String serviceId) async {
|
2021-12-06 18:31:19 +00:00
|
|
|
emit(state.copyWith(preventActions: true));
|
2023-06-29 09:52:09 +00:00
|
|
|
await api.startBackup(serviceId);
|
2021-12-06 18:31:19 +00:00
|
|
|
await updateBackups();
|
|
|
|
emit(state.copyWith(preventActions: false));
|
|
|
|
}
|
|
|
|
|
2023-08-07 12:23:48 +00:00
|
|
|
Future<void> restoreBackup(
|
|
|
|
final String backupId,
|
|
|
|
final BackupRestoreStrategy strategy,
|
|
|
|
) async {
|
2021-12-06 18:31:19 +00:00
|
|
|
emit(state.copyWith(preventActions: true));
|
2023-08-07 12:23:48 +00:00
|
|
|
await api.restoreBackup(backupId, strategy);
|
2021-12-06 18:31:19 +00:00
|
|
|
emit(state.copyWith(preventActions: false));
|
|
|
|
}
|
|
|
|
|
2023-07-02 15:23:12 +00:00
|
|
|
Future<void> setAutobackupPeriod(final Duration? period) async {
|
|
|
|
emit(state.copyWith(preventActions: true));
|
|
|
|
final result = await api.setAutobackupPeriod(period: period?.inMinutes);
|
|
|
|
if (result.success == false) {
|
|
|
|
getIt<NavigationService>()
|
|
|
|
.showSnackBar(result.message ?? 'Unknown error');
|
|
|
|
emit(state.copyWith(preventActions: false));
|
|
|
|
} else {
|
|
|
|
getIt<NavigationService>()
|
|
|
|
.showSnackBar('backup.autobackup_period_set'.tr());
|
|
|
|
emit(
|
|
|
|
state.copyWith(
|
|
|
|
preventActions: false,
|
|
|
|
autobackupPeriod: period ?? Duration.zero,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
await updateBackups();
|
|
|
|
}
|
|
|
|
|
2023-09-09 18:13:27 +00:00
|
|
|
Future<void> setAutobackupQuotas(final AutobackupQuotas quotas) async {
|
|
|
|
emit(state.copyWith(preventActions: true));
|
|
|
|
final result = await api.setAutobackupQuotas(quotas);
|
|
|
|
if (result.success == false) {
|
|
|
|
getIt<NavigationService>()
|
|
|
|
.showSnackBar(result.message ?? 'Unknown error');
|
|
|
|
emit(state.copyWith(preventActions: false));
|
|
|
|
} else {
|
|
|
|
getIt<NavigationService>().showSnackBar('backup.quotas_set'.tr());
|
|
|
|
emit(
|
|
|
|
state.copyWith(
|
|
|
|
preventActions: false,
|
|
|
|
autobackupQuotas: quotas,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
await updateBackups();
|
|
|
|
}
|
|
|
|
|
2023-08-07 12:44:26 +00:00
|
|
|
Future<void> forgetSnapshot(final String snapshotId) async {
|
|
|
|
final result = await api.forgetSnapshot(snapshotId);
|
|
|
|
if (!result.success) {
|
|
|
|
getIt<NavigationService>().showSnackBar('jobs.generic_error'.tr());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (result.data == false) {
|
|
|
|
getIt<NavigationService>()
|
|
|
|
.showSnackBar('backup.forget_snapshot_error'.tr());
|
|
|
|
}
|
2023-08-14 04:10:15 +00:00
|
|
|
|
|
|
|
// Optimistic update
|
|
|
|
final backups = state.backups;
|
|
|
|
final index =
|
|
|
|
backups.indexWhere((final snapshot) => snapshot.id == snapshotId);
|
|
|
|
if (index != -1) {
|
|
|
|
backups.removeAt(index);
|
|
|
|
emit(state.copyWith(backups: backups));
|
|
|
|
}
|
|
|
|
|
|
|
|
await updateBackups();
|
2023-08-07 12:44:26 +00:00
|
|
|
}
|
|
|
|
|
2021-12-06 18:31:19 +00:00
|
|
|
@override
|
|
|
|
void clear() async {
|
2022-05-24 18:55:39 +00:00
|
|
|
emit(const BackupsState());
|
2021-12-06 18:31:19 +00:00
|
|
|
}
|
|
|
|
}
|