mirror of https://github.com/oxen-io/session-ios
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1182 lines
44 KiB
Matlab
1182 lines
44 KiB
Matlab
7 years ago
|
//
|
||
|
// Copyright (c) 2018 Open Whisper Systems. All rights reserved.
|
||
|
//
|
||
|
|
||
7 years ago
|
#import "OWSBackupExportJob.h"
|
||
7 years ago
|
#import "OWSBackupIO.h"
|
||
7 years ago
|
#import "OWSDatabaseMigration.h"
|
||
6 years ago
|
#import "Session-Swift.h"
|
||
6 years ago
|
#import <PromiseKit/AnyPromise.h>
|
||
4 years ago
|
#import <SignalCoreKit/NSData+OWS.h>
|
||
|
#import <SignalCoreKit/NSDate+OWS.h>
|
||
|
#import <SignalCoreKit/Threading.h>
|
||
4 years ago
|
#import <SessionMessagingKit/OWSBackgroundTask.h>
|
||
4 years ago
|
#import <SignalUtilitiesKit/OWSError.h>
|
||
4 years ago
|
#import <SessionUtilitiesKit/OWSFileSystem.h>
|
||
4 years ago
|
#import <SessionMessagingKit/TSAttachment.h>
|
||
|
#import <SessionMessagingKit/TSAttachmentStream.h>
|
||
4 years ago
|
#import <SessionMessagingKit/TSMessage.h>
|
||
4 years ago
|
#import <SessionMessagingKit/TSThread.h>
|
||
7 years ago
|
|
||
6 years ago
|
@import CloudKit;
|
||
|
|
||
7 years ago
|
NS_ASSUME_NONNULL_BEGIN
|
||
|
|
||
7 years ago
|
@class OWSAttachmentExport;
|
||
7 years ago
|
|
||
7 years ago
|
@interface OWSBackupExportItem : NSObject
|
||
7 years ago
|
|
||
7 years ago
|
@property (nonatomic) OWSBackupEncryptedItem *encryptedItem;
|
||
7 years ago
|
|
||
7 years ago
|
@property (nonatomic) NSString *recordName;
|
||
7 years ago
|
|
||
7 years ago
|
// This property is optional and is only used for attachments.
|
||
|
@property (nonatomic, nullable) OWSAttachmentExport *attachmentExport;
|
||
7 years ago
|
|
||
7 years ago
|
// This property is optional.
|
||
7 years ago
|
//
|
||
|
// See comments in `OWSBackupIO`.
|
||
7 years ago
|
@property (nonatomic, nullable) NSNumber *uncompressedDataLength;
|
||
|
|
||
7 years ago
|
- (instancetype)init NS_UNAVAILABLE;
|
||
7 years ago
|
|
||
|
@end
|
||
|
|
||
|
#pragma mark -
|
||
|
|
||
7 years ago
|
@implementation OWSBackupExportItem
|
||
|
|
||
|
- (instancetype)initWithEncryptedItem:(OWSBackupEncryptedItem *)encryptedItem
|
||
|
{
|
||
|
if (!(self = [super init])) {
|
||
|
return self;
|
||
|
}
|
||
|
|
||
7 years ago
|
OWSAssertDebug(encryptedItem);
|
||
7 years ago
|
|
||
|
self.encryptedItem = encryptedItem;
|
||
|
|
||
|
return self;
|
||
|
}
|
||
7 years ago
|
|
||
|
@end
|
||
|
|
||
|
#pragma mark -
|
||
|
|
||
7 years ago
|
// Used to serialize database snapshot contents.
|
||
|
// Writes db entities using protobufs into snapshot fragments.
|
||
|
// Snapshot fragments are compressed (they compress _very well_,
|
||
|
// around 20x smaller) then encrypted. Ordering matters in
|
||
7 years ago
|
// snapshot contents (entities should be restored in the same
|
||
7 years ago
|
// order they are serialized), so we are always careful to preserve
|
||
|
// ordering of entities within a snapshot AND ordering of snapshot
|
||
|
// fragments within a bakckup.
|
||
|
//
|
||
|
// This stream is used to write entities one at a time and takes
|
||
|
// care of sharding them into fragments, compressing and encrypting
|
||
|
// those fragments. Fragment size is fixed to reduce worst case
|
||
|
// memory usage.
|
||
7 years ago
|
@interface OWSDBExportStream : NSObject
|
||
|
|
||
7 years ago
|
@property (nonatomic) OWSBackupIO *backupIO;
|
||
7 years ago
|
|
||
7 years ago
|
@property (nonatomic) NSMutableArray<OWSBackupExportItem *> *exportItems;
|
||
7 years ago
|
|
||
7 years ago
|
@property (nonatomic, nullable) SignalIOSProtoBackupSnapshotBuilder *backupSnapshotBuilder;
|
||
7 years ago
|
|
||
7 years ago
|
@property (nonatomic) NSUInteger cachedItemCount;
|
||
7 years ago
|
|
||
7 years ago
|
@property (nonatomic) NSUInteger totalItemCount;
|
||
|
|
||
|
- (instancetype)init NS_UNAVAILABLE;
|
||
7 years ago
|
|
||
|
@end
|
||
|
|
||
|
#pragma mark -
|
||
|
|
||
7 years ago
|
@implementation OWSDBExportStream
|
||
7 years ago
|
|
||
7 years ago
|
- (instancetype)initWithBackupIO:(OWSBackupIO *)backupIO
|
||
7 years ago
|
{
|
||
7 years ago
|
if (!(self = [super init])) {
|
||
|
return self;
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
OWSAssertDebug(backupIO);
|
||
7 years ago
|
|
||
7 years ago
|
self.exportItems = [NSMutableArray new];
|
||
7 years ago
|
self.backupIO = backupIO;
|
||
7 years ago
|
|
||
7 years ago
|
return self;
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
|
||
|
// It isn't strictly necessary to capture the entity type (the importer doesn't
|
||
|
// use this state), but I think it'll be helpful to have around to future-proof
|
||
|
// this work, help with debugging issue, etc.
|
||
6 years ago
|
- (BOOL)writeObject:(NSObject *)object
|
||
|
collection:(NSString *)collection
|
||
|
key:(NSString *)key
|
||
|
entityType:(SignalIOSProtoBackupSnapshotBackupEntityType)entityType
|
||
7 years ago
|
{
|
||
7 years ago
|
OWSAssertDebug(object);
|
||
6 years ago
|
OWSAssertDebug(collection.length > 0);
|
||
|
OWSAssertDebug(key.length > 0);
|
||
7 years ago
|
|
||
|
NSData *_Nullable data = [NSKeyedArchiver archivedDataWithRootObject:object];
|
||
|
if (!data) {
|
||
7 years ago
|
OWSFailDebug(@"couldn't serialize database object: %@", [object class]);
|
||
7 years ago
|
return NO;
|
||
|
}
|
||
|
|
||
7 years ago
|
if (!self.backupSnapshotBuilder) {
|
||
7 years ago
|
self.backupSnapshotBuilder = [SignalIOSProtoBackupSnapshot builder];
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
SignalIOSProtoBackupSnapshotBackupEntityBuilder *entityBuilder =
|
||
6 years ago
|
[SignalIOSProtoBackupSnapshotBackupEntity builderWithType:entityType
|
||
|
entityData:data
|
||
|
collection:collection
|
||
|
key:key];
|
||
7 years ago
|
|
||
7 years ago
|
NSError *error;
|
||
|
SignalIOSProtoBackupSnapshotBackupEntity *_Nullable entity = [entityBuilder buildAndReturnError:&error];
|
||
|
if (!entity || error) {
|
||
7 years ago
|
OWSFailDebug(@"couldn't build proto: %@", error);
|
||
7 years ago
|
return NO;
|
||
|
}
|
||
|
|
||
|
[self.backupSnapshotBuilder addEntity:entity];
|
||
7 years ago
|
|
||
|
self.cachedItemCount = self.cachedItemCount + 1;
|
||
|
self.totalItemCount = self.totalItemCount + 1;
|
||
|
|
||
|
static const int kMaxDBSnapshotSize = 1000;
|
||
|
if (self.cachedItemCount > kMaxDBSnapshotSize) {
|
||
7 years ago
|
@autoreleasepool {
|
||
|
return [self flush];
|
||
|
}
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
return YES;
|
||
|
}
|
||
|
|
||
7 years ago
|
// Write cached data to disk, if necessary.
|
||
7 years ago
|
//
|
||
|
// Returns YES on success.
|
||
7 years ago
|
- (BOOL)flush
|
||
7 years ago
|
{
|
||
7 years ago
|
if (!self.backupSnapshotBuilder) {
|
||
7 years ago
|
// No data to flush to disk.
|
||
7 years ago
|
return YES;
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
// Try to release allocated buffers ASAP.
|
||
|
@autoreleasepool {
|
||
7 years ago
|
NSError *error;
|
||
|
NSData *_Nullable uncompressedData = [self.backupSnapshotBuilder buildSerializedDataAndReturnError:&error];
|
||
|
if (!uncompressedData || error) {
|
||
7 years ago
|
OWSFailDebug(@"couldn't serialize proto: %@", error);
|
||
7 years ago
|
return NO;
|
||
|
}
|
||
|
|
||
7 years ago
|
NSUInteger uncompressedDataLength = uncompressedData.length;
|
||
|
self.backupSnapshotBuilder = nil;
|
||
|
self.cachedItemCount = 0;
|
||
|
if (!uncompressedData) {
|
||
7 years ago
|
OWSFailDebug(@"couldn't convert database snapshot to data.");
|
||
7 years ago
|
return NO;
|
||
|
}
|
||
7 years ago
|
|
||
7 years ago
|
NSData *compressedData = [self.backupIO compressData:uncompressedData];
|
||
7 years ago
|
|
||
7 years ago
|
OWSBackupEncryptedItem *_Nullable encryptedItem = [self.backupIO encryptDataAsTempFile:compressedData];
|
||
|
if (!encryptedItem) {
|
||
7 years ago
|
OWSFailDebug(@"couldn't encrypt database snapshot.");
|
||
7 years ago
|
return NO;
|
||
|
}
|
||
|
|
||
|
OWSBackupExportItem *exportItem = [[OWSBackupExportItem alloc] initWithEncryptedItem:encryptedItem];
|
||
|
exportItem.uncompressedDataLength = @(uncompressedDataLength);
|
||
|
[self.exportItems addObject:exportItem];
|
||
|
}
|
||
7 years ago
|
|
||
7 years ago
|
return YES;
|
||
|
}
|
||
|
|
||
|
@end
|
||
|
|
||
|
#pragma mark -
|
||
|
|
||
7 years ago
|
// This class is used to:
|
||
|
//
|
||
|
// * Lazy-encrypt and eagerly cleanup attachment uploads.
|
||
|
// To reduce disk footprint of backup export process,
|
||
|
// we only want to have one attachment export on disk
|
||
|
// at a time.
|
||
7 years ago
|
@interface OWSAttachmentExport : NSObject
|
||
|
|
||
7 years ago
|
@property (nonatomic) OWSBackupIO *backupIO;
|
||
7 years ago
|
@property (nonatomic) NSString *attachmentId;
|
||
|
@property (nonatomic) NSString *attachmentFilePath;
|
||
|
@property (nonatomic, nullable) NSString *relativeFilePath;
|
||
7 years ago
|
@property (nonatomic) OWSBackupEncryptedItem *encryptedItem;
|
||
|
|
||
|
- (instancetype)init NS_UNAVAILABLE;
|
||
7 years ago
|
|
||
|
@end
|
||
|
|
||
|
#pragma mark -
|
||
|
|
||
|
@implementation OWSAttachmentExport
|
||
|
|
||
7 years ago
|
- (instancetype)initWithBackupIO:(OWSBackupIO *)backupIO
|
||
|
attachmentId:(NSString *)attachmentId
|
||
|
attachmentFilePath:(NSString *)attachmentFilePath
|
||
7 years ago
|
{
|
||
|
if (!(self = [super init])) {
|
||
|
return self;
|
||
|
}
|
||
|
|
||
7 years ago
|
OWSAssertDebug(backupIO);
|
||
|
OWSAssertDebug(attachmentId.length > 0);
|
||
|
OWSAssertDebug(attachmentFilePath.length > 0);
|
||
7 years ago
|
|
||
7 years ago
|
self.backupIO = backupIO;
|
||
7 years ago
|
self.attachmentId = attachmentId;
|
||
|
self.attachmentFilePath = attachmentFilePath;
|
||
|
|
||
|
return self;
|
||
|
}
|
||
|
|
||
7 years ago
|
- (void)dealloc
|
||
|
{
|
||
|
// Surface memory leaks by logging the deallocation.
|
||
7 years ago
|
OWSLogVerbose(@"Dealloc: %@", self.class);
|
||
7 years ago
|
|
||
|
[self cleanUp];
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
// On success, encryptedItem will be non-nil.
|
||
7 years ago
|
//
|
||
|
// Returns YES on success.
|
||
7 years ago
|
- (BOOL)prepareForUpload
|
||
7 years ago
|
{
|
||
7 years ago
|
OWSAssertDebug(self.attachmentId.length > 0);
|
||
|
OWSAssertDebug(self.attachmentFilePath.length > 0);
|
||
7 years ago
|
|
||
|
NSString *attachmentsDirPath = [TSAttachmentStream attachmentsFolder];
|
||
|
if (![self.attachmentFilePath hasPrefix:attachmentsDirPath]) {
|
||
7 years ago
|
OWSFailDebug(@"attachment has unexpected path: %@", self.attachmentFilePath);
|
||
7 years ago
|
return NO;
|
||
7 years ago
|
}
|
||
|
NSString *relativeFilePath = [self.attachmentFilePath substringFromIndex:attachmentsDirPath.length];
|
||
|
NSString *pathSeparator = @"/";
|
||
|
if ([relativeFilePath hasPrefix:pathSeparator]) {
|
||
|
relativeFilePath = [relativeFilePath substringFromIndex:pathSeparator.length];
|
||
|
}
|
||
|
self.relativeFilePath = relativeFilePath;
|
||
|
|
||
7 years ago
|
OWSBackupEncryptedItem *_Nullable encryptedItem = [self.backupIO encryptFileAsTempFile:self.attachmentFilePath];
|
||
7 years ago
|
if (!encryptedItem) {
|
||
7 years ago
|
OWSFailDebug(@"attachment could not be encrypted: %@", self.attachmentFilePath);
|
||
7 years ago
|
return NO;
|
||
7 years ago
|
}
|
||
7 years ago
|
self.encryptedItem = encryptedItem;
|
||
|
return YES;
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
// Returns YES on success.
|
||
|
- (BOOL)cleanUp
|
||
|
{
|
||
|
return [OWSFileSystem deleteFileIfExists:self.encryptedItem.filePath];
|
||
|
}
|
||
|
|
||
7 years ago
|
@end
|
||
|
|
||
|
#pragma mark -
|
||
|
|
||
7 years ago
|
@interface OWSBackupExportJob ()
|
||
7 years ago
|
|
||
7 years ago
|
@property (nonatomic, nullable) OWSBackgroundTask *backgroundTask;
|
||
7 years ago
|
|
||
7 years ago
|
@property (nonatomic) OWSBackupIO *backupIO;
|
||
7 years ago
|
|
||
|
@property (nonatomic) NSMutableArray<OWSBackupExportItem *> *unsavedDatabaseItems;
|
||
7 years ago
|
|
||
7 years ago
|
@property (nonatomic) NSMutableArray<OWSAttachmentExport *> *unsavedAttachmentExports;
|
||
7 years ago
|
|
||
7 years ago
|
@property (nonatomic) NSMutableArray<OWSBackupExportItem *> *savedDatabaseItems;
|
||
|
|
||
|
@property (nonatomic) NSMutableArray<OWSBackupExportItem *> *savedAttachmentItems;
|
||
|
|
||
6 years ago
|
@property (nonatomic, nullable) OWSBackupExportItem *localProfileAvatarItem;
|
||
|
|
||
7 years ago
|
@property (nonatomic, nullable) OWSBackupExportItem *manifestItem;
|
||
7 years ago
|
|
||
7 years ago
|
// If we are replacing an existing backup, we use some of its contents for continuity.
|
||
7 years ago
|
@property (nonatomic, nullable) NSSet<NSString *> *lastValidRecordNames;
|
||
7 years ago
|
|
||
6 years ago
|
@property (nonatomic, nullable) YapDatabaseConnection *dbConnection;
|
||
|
|
||
7 years ago
|
@end
|
||
|
|
||
|
#pragma mark -
|
||
|
|
||
7 years ago
|
@implementation OWSBackupExportJob
|
||
7 years ago
|
|
||
6 years ago
|
#pragma mark - Dependencies
|
||
|
|
||
|
- (OWSPrimaryStorage *)primaryStorage
|
||
|
{
|
||
|
OWSAssertDebug(SSKEnvironment.shared.primaryStorage);
|
||
|
|
||
|
return SSKEnvironment.shared.primaryStorage;
|
||
|
}
|
||
|
|
||
6 years ago
|
- (OWSBackup *)backup
|
||
|
{
|
||
|
OWSAssertDebug(AppEnvironment.shared.backup);
|
||
|
|
||
|
return AppEnvironment.shared.backup;
|
||
|
}
|
||
|
|
||
6 years ago
|
- (OWSProfileManager *)profileManager
|
||
|
{
|
||
|
return [OWSProfileManager sharedManager];
|
||
|
}
|
||
|
|
||
6 years ago
|
#pragma mark -
|
||
|
|
||
6 years ago
|
- (void)start
|
||
7 years ago
|
{
|
||
7 years ago
|
OWSAssertIsOnMainThread();
|
||
|
|
||
7 years ago
|
OWSLogInfo(@"");
|
||
7 years ago
|
|
||
7 years ago
|
self.backgroundTask = [OWSBackgroundTask backgroundTaskWithLabelStr:__PRETTY_FUNCTION__];
|
||
|
|
||
7 years ago
|
[self updateProgressWithDescription:nil progress:nil];
|
||
|
|
||
6 years ago
|
self.dbConnection = self.primaryStorage.newDatabaseConnection;
|
||
|
|
||
6 years ago
|
[[self.backup ensureCloudKitAccess]
|
||
6 years ago
|
.thenInBackground(^{
|
||
6 years ago
|
[self updateProgressWithDescription:NSLocalizedString(@"BACKUP_EXPORT_PHASE_CONFIGURATION",
|
||
|
@"Indicates that the backup export is being configured.")
|
||
|
progress:nil];
|
||
7 years ago
|
|
||
6 years ago
|
return [self configureExport];
|
||
|
})
|
||
6 years ago
|
.thenInBackground(^{
|
||
|
return [self fetchAllRecords];
|
||
|
})
|
||
|
.thenInBackground(^{
|
||
|
[self updateProgressWithDescription:NSLocalizedString(@"BACKUP_EXPORT_PHASE_EXPORT",
|
||
|
@"Indicates that the backup export data is being exported.")
|
||
|
progress:nil];
|
||
7 years ago
|
|
||
6 years ago
|
return [self exportDatabase];
|
||
|
})
|
||
|
.thenInBackground(^{
|
||
|
return [self saveToCloud];
|
||
|
})
|
||
|
.thenInBackground(^{
|
||
|
return [self cleanUp];
|
||
|
})
|
||
|
.thenInBackground(^{
|
||
|
[self succeed];
|
||
|
})
|
||
|
.catch(^(NSError *error) {
|
||
|
OWSFailDebug(@"Backup export failed with error: %@.", error);
|
||
7 years ago
|
|
||
6 years ago
|
[self failWithErrorDescription:
|
||
|
NSLocalizedString(@"BACKUP_EXPORT_ERROR_COULD_NOT_EXPORT",
|
||
|
@"Error indicating the backup export could not export the user's data.")];
|
||
6 years ago
|
}) retainUntilComplete];
|
||
7 years ago
|
}
|
||
|
|
||
6 years ago
|
- (AnyPromise *)configureExport
|
||
7 years ago
|
{
|
||
7 years ago
|
OWSLogVerbose(@"");
|
||
7 years ago
|
|
||
6 years ago
|
if (self.isComplete) {
|
||
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Backup export no longer active.")];
|
||
|
}
|
||
|
|
||
7 years ago
|
if (![self ensureJobTempDir]) {
|
||
7 years ago
|
OWSFailDebug(@"Could not create jobTempDirPath.");
|
||
6 years ago
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Could not create jobTempDirPath.")];
|
||
7 years ago
|
}
|
||
7 years ago
|
|
||
7 years ago
|
self.backupIO = [[OWSBackupIO alloc] initWithJobTempDirPath:self.jobTempDirPath];
|
||
7 years ago
|
|
||
4 years ago
|
return [AnyPromise promiseWithValue:@(1)];
|
||
7 years ago
|
}
|
||
|
|
||
6 years ago
|
- (AnyPromise *)fetchAllRecords
|
||
7 years ago
|
{
|
||
6 years ago
|
OWSLogVerbose(@"");
|
||
7 years ago
|
|
||
|
if (self.isComplete) {
|
||
6 years ago
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Backup export no longer active.")];
|
||
7 years ago
|
}
|
||
|
|
||
6 years ago
|
return [AnyPromise promiseWithResolverBlock:^(PMKResolver resolve) {
|
||
6 years ago
|
[OWSBackupAPI fetchAllRecordNamesWithRecipientId:self.recipientId
|
||
|
success:^(NSArray<NSString *> *recordNames) {
|
||
6 years ago
|
if (self.isComplete) {
|
||
6 years ago
|
return resolve(OWSBackupErrorWithDescription(@"Backup export no longer active."));
|
||
6 years ago
|
}
|
||
6 years ago
|
self.lastValidRecordNames = [NSSet setWithArray:recordNames];
|
||
6 years ago
|
resolve(@(1));
|
||
|
}
|
||
|
failure:^(NSError *error) {
|
||
|
resolve(error);
|
||
|
}];
|
||
|
}];
|
||
|
}
|
||
|
|
||
|
- (AnyPromise *)exportDatabase
|
||
|
{
|
||
|
OWSAssertDebug(self.backupIO);
|
||
|
|
||
|
OWSLogVerbose(@"");
|
||
|
|
||
|
if (self.isComplete) {
|
||
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Backup export no longer active.")];
|
||
|
}
|
||
|
|
||
|
return [AnyPromise promiseWithResolverBlock:^(PMKResolver resolve) {
|
||
6 years ago
|
if (![self performExportDatabase]) {
|
||
6 years ago
|
NSError *error = OWSBackupErrorWithDescription(@"Backup export failed.");
|
||
|
return resolve(error);
|
||
|
}
|
||
6 years ago
|
|
||
6 years ago
|
resolve(@(1));
|
||
|
}];
|
||
7 years ago
|
}
|
||
|
|
||
6 years ago
|
- (BOOL)performExportDatabase
|
||
7 years ago
|
{
|
||
7 years ago
|
OWSAssertDebug(self.backupIO);
|
||
7 years ago
|
|
||
7 years ago
|
OWSLogVerbose(@"");
|
||
7 years ago
|
|
||
7 years ago
|
[self updateProgressWithDescription:NSLocalizedString(@"BACKUP_EXPORT_PHASE_DATABASE_EXPORT",
|
||
|
@"Indicates that the database data is being exported.")
|
||
|
progress:nil];
|
||
7 years ago
|
|
||
7 years ago
|
OWSDBExportStream *exportStream = [[OWSDBExportStream alloc] initWithBackupIO:self.backupIO];
|
||
7 years ago
|
|
||
7 years ago
|
__block BOOL aborted = NO;
|
||
|
typedef BOOL (^EntityFilter)(id object);
|
||
|
typedef NSUInteger (^ExportBlock)(YapDatabaseReadTransaction *,
|
||
|
NSString *,
|
||
|
Class,
|
||
|
EntityFilter _Nullable,
|
||
7 years ago
|
SignalIOSProtoBackupSnapshotBackupEntityType);
|
||
6 years ago
|
NSMutableSet<NSString *> *exportedCollections = [NSMutableSet new];
|
||
7 years ago
|
ExportBlock exportEntities = ^(YapDatabaseReadTransaction *transaction,
|
||
|
NSString *collection,
|
||
|
Class expectedClass,
|
||
|
EntityFilter _Nullable filter,
|
||
7 years ago
|
SignalIOSProtoBackupSnapshotBackupEntityType entityType) {
|
||
6 years ago
|
[exportedCollections addObject:collection];
|
||
|
|
||
7 years ago
|
__block NSUInteger count = 0;
|
||
6 years ago
|
[transaction enumerateKeysAndObjectsInCollection:collection
|
||
|
usingBlock:^(NSString *key, id object, BOOL *stop) {
|
||
|
if (self.isComplete) {
|
||
|
*stop = YES;
|
||
|
return;
|
||
|
}
|
||
|
if (filter && !filter(object)) {
|
||
|
return;
|
||
|
}
|
||
|
if (![object isKindOfClass:expectedClass]) {
|
||
|
OWSFailDebug(@"unexpected class: %@", [object class]);
|
||
|
return;
|
||
|
}
|
||
6 years ago
|
NSObject *entity = object;
|
||
6 years ago
|
count++;
|
||
|
|
||
6 years ago
|
if ([entity isKindOfClass:[TSAttachmentStream class]]) {
|
||
|
// Convert attachment streams to pointers,
|
||
|
// since we'll need to restore them.
|
||
|
TSAttachmentStream *attachmentStream
|
||
|
= (TSAttachmentStream *)entity;
|
||
|
TSAttachmentPointer *attachmentPointer =
|
||
|
[[TSAttachmentPointer alloc]
|
||
|
initForRestoreWithAttachmentStream:attachmentStream];
|
||
|
entity = attachmentPointer;
|
||
|
}
|
||
|
|
||
6 years ago
|
if (![exportStream writeObject:entity
|
||
|
collection:collection
|
||
|
key:key
|
||
|
entityType:entityType]) {
|
||
6 years ago
|
*stop = YES;
|
||
|
aborted = YES;
|
||
|
return;
|
||
|
}
|
||
|
}];
|
||
7 years ago
|
return count;
|
||
|
};
|
||
7 years ago
|
|
||
7 years ago
|
__block NSUInteger copiedThreads = 0;
|
||
|
__block NSUInteger copiedInteractions = 0;
|
||
|
__block NSUInteger copiedAttachments = 0;
|
||
|
__block NSUInteger copiedMigrations = 0;
|
||
6 years ago
|
__block NSUInteger copiedMisc = 0;
|
||
7 years ago
|
self.unsavedAttachmentExports = [NSMutableArray new];
|
||
6 years ago
|
[self.dbConnection readWithBlock:^(YapDatabaseReadTransaction *transaction) {
|
||
7 years ago
|
copiedThreads = exportEntities(transaction,
|
||
|
[TSThread collection],
|
||
|
[TSThread class],
|
||
|
nil,
|
||
7 years ago
|
SignalIOSProtoBackupSnapshotBackupEntityTypeThread);
|
||
7 years ago
|
if (aborted) {
|
||
|
return;
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
copiedAttachments = exportEntities(transaction,
|
||
|
[TSAttachment collection],
|
||
|
[TSAttachment class],
|
||
|
^(id object) {
|
||
|
if (![object isKindOfClass:[TSAttachmentStream class]]) {
|
||
6 years ago
|
// No need to backup the contents (e.g. the file on disk)
|
||
|
// of attachment pointers.
|
||
|
// After a restore, users will be able "tap to retry".
|
||
|
return YES;
|
||
7 years ago
|
}
|
||
|
TSAttachmentStream *attachmentStream = object;
|
||
7 years ago
|
NSString *_Nullable filePath = attachmentStream.originalFilePath;
|
||
6 years ago
|
if (!filePath || ![NSFileManager.defaultManager fileExistsAtPath:filePath]) {
|
||
|
OWSFailDebug(@"attachment is missing file.");
|
||
7 years ago
|
return NO;
|
||
|
}
|
||
7 years ago
|
|
||
7 years ago
|
// OWSAttachmentExport is used to lazily write an encrypted copy of the
|
||
|
// attachment to disk.
|
||
|
OWSAttachmentExport *attachmentExport =
|
||
7 years ago
|
[[OWSAttachmentExport alloc] initWithBackupIO:self.backupIO
|
||
|
attachmentId:attachmentStream.uniqueId
|
||
|
attachmentFilePath:filePath];
|
||
7 years ago
|
[self.unsavedAttachmentExports addObject:attachmentExport];
|
||
|
|
||
|
return YES;
|
||
|
},
|
||
7 years ago
|
SignalIOSProtoBackupSnapshotBackupEntityTypeAttachment);
|
||
7 years ago
|
if (aborted) {
|
||
|
return;
|
||
|
}
|
||
7 years ago
|
|
||
7 years ago
|
// Interactions refer to threads and attachments, so copy after them.
|
||
|
copiedInteractions = exportEntities(transaction,
|
||
|
[TSInteraction collection],
|
||
|
[TSInteraction class],
|
||
|
^(id object) {
|
||
|
// Ignore disappearing messages.
|
||
|
if ([object isKindOfClass:[TSMessage class]]) {
|
||
|
TSMessage *message = object;
|
||
|
if (message.isExpiringMessage) {
|
||
|
return NO;
|
||
7 years ago
|
}
|
||
7 years ago
|
}
|
||
|
TSInteraction *interaction = object;
|
||
|
// Ignore dynamic interactions.
|
||
|
if (interaction.isDynamicInteraction) {
|
||
|
return NO;
|
||
|
}
|
||
|
return YES;
|
||
|
},
|
||
7 years ago
|
SignalIOSProtoBackupSnapshotBackupEntityTypeInteraction);
|
||
7 years ago
|
if (aborted) {
|
||
|
return;
|
||
|
}
|
||
7 years ago
|
|
||
7 years ago
|
copiedMigrations = exportEntities(transaction,
|
||
|
[OWSDatabaseMigration collection],
|
||
|
[OWSDatabaseMigration class],
|
||
|
nil,
|
||
7 years ago
|
SignalIOSProtoBackupSnapshotBackupEntityTypeMigration);
|
||
6 years ago
|
if (aborted) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
for (NSString *collection in MiscCollectionsToBackup()) {
|
||
|
copiedMisc += exportEntities(
|
||
|
transaction, collection, [NSObject class], nil, SignalIOSProtoBackupSnapshotBackupEntityTypeMisc);
|
||
|
if (aborted) {
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
6 years ago
|
for (NSString *collection in [exportedCollections.allObjects sortedArrayUsingSelector:@selector(compare:)]) {
|
||
|
OWSLogVerbose(@"Exported collection: %@", collection);
|
||
|
}
|
||
|
OWSLogVerbose(@"Exported collections: %lu", (unsigned long)exportedCollections.count);
|
||
|
|
||
6 years ago
|
NSSet<NSString *> *allCollections = [NSSet setWithArray:transaction.allCollections];
|
||
|
NSMutableSet *unexportedCollections = [allCollections mutableCopy];
|
||
|
[unexportedCollections minusSet:exportedCollections];
|
||
|
for (NSString *collection in [unexportedCollections.allObjects sortedArrayUsingSelector:@selector(compare:)]) {
|
||
|
OWSLogVerbose(@"Unexported collection: %@", collection);
|
||
|
}
|
||
|
OWSLogVerbose(@"Unexported collections: %lu", (unsigned long)unexportedCollections.count);
|
||
7 years ago
|
}];
|
||
|
|
||
7 years ago
|
if (aborted || self.isComplete) {
|
||
|
return NO;
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
@autoreleasepool {
|
||
|
if (![exportStream flush]) {
|
||
7 years ago
|
OWSFailDebug(@"Could not flush database snapshots.");
|
||
7 years ago
|
return NO;
|
||
|
}
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
self.unsavedDatabaseItems = [exportStream.exportItems mutableCopy];
|
||
7 years ago
|
|
||
7 years ago
|
// TODO: Should we do a database checkpoint?
|
||
|
|
||
7 years ago
|
OWSLogInfo(@"copiedThreads: %zd", copiedThreads);
|
||
|
OWSLogInfo(@"copiedMessages: %zd", copiedInteractions);
|
||
|
OWSLogInfo(@"copiedAttachments: %zd", copiedAttachments);
|
||
|
OWSLogInfo(@"copiedMigrations: %zd", copiedMigrations);
|
||
6 years ago
|
OWSLogInfo(@"copiedMisc: %zd", copiedMisc);
|
||
7 years ago
|
OWSLogInfo(@"copiedEntities: %zd", exportStream.totalItemCount);
|
||
7 years ago
|
|
||
7 years ago
|
return YES;
|
||
7 years ago
|
}
|
||
7 years ago
|
|
||
6 years ago
|
- (AnyPromise *)saveToCloud
|
||
7 years ago
|
{
|
||
7 years ago
|
OWSLogVerbose(@"");
|
||
7 years ago
|
|
||
6 years ago
|
if (self.isComplete) {
|
||
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Backup export no longer active.")];
|
||
|
}
|
||
|
|
||
7 years ago
|
self.savedDatabaseItems = [NSMutableArray new];
|
||
|
self.savedAttachmentItems = [NSMutableArray new];
|
||
7 years ago
|
|
||
7 years ago
|
unsigned long long totalFileSize = 0;
|
||
|
NSUInteger totalFileCount = 0;
|
||
7 years ago
|
{
|
||
7 years ago
|
unsigned long long databaseFileSize = 0;
|
||
7 years ago
|
for (OWSBackupExportItem *item in self.unsavedDatabaseItems) {
|
||
7 years ago
|
unsigned long long fileSize =
|
||
|
[OWSFileSystem fileSizeOfPath:item.encryptedItem.filePath].unsignedLongLongValue;
|
||
|
ows_add_overflow(databaseFileSize, fileSize, &databaseFileSize);
|
||
7 years ago
|
}
|
||
7 years ago
|
OWSLogInfo(@"exporting %@: count: %zd, bytes: %llu.",
|
||
7 years ago
|
@"database items",
|
||
|
self.unsavedDatabaseItems.count,
|
||
7 years ago
|
databaseFileSize);
|
||
7 years ago
|
ows_add_overflow(totalFileSize, databaseFileSize, &totalFileSize);
|
||
|
ows_add_overflow(totalFileCount, self.unsavedDatabaseItems.count, &totalFileCount);
|
||
7 years ago
|
}
|
||
|
{
|
||
7 years ago
|
unsigned long long attachmentFileSize = 0;
|
||
7 years ago
|
for (OWSAttachmentExport *attachmentExport in self.unsavedAttachmentExports) {
|
||
7 years ago
|
unsigned long long fileSize =
|
||
7 years ago
|
[OWSFileSystem fileSizeOfPath:attachmentExport.attachmentFilePath].unsignedLongLongValue;
|
||
7 years ago
|
ows_add_overflow(attachmentFileSize, fileSize, &attachmentFileSize);
|
||
7 years ago
|
}
|
||
7 years ago
|
OWSLogInfo(@"exporting %@: count: %zd, bytes: %llu.",
|
||
7 years ago
|
@"attachment items",
|
||
|
self.unsavedAttachmentExports.count,
|
||
7 years ago
|
attachmentFileSize);
|
||
7 years ago
|
ows_add_overflow(totalFileSize, attachmentFileSize, &totalFileSize);
|
||
|
ows_add_overflow(totalFileCount, self.unsavedAttachmentExports.count, &totalFileSize);
|
||
7 years ago
|
}
|
||
7 years ago
|
OWSLogInfo(@"exporting %@: count: %zd, bytes: %llu.", @"all items", totalFileCount, totalFileSize);
|
||
7 years ago
|
|
||
7 years ago
|
// Add one for the manifest
|
||
|
NSUInteger unsavedCount = (self.unsavedDatabaseItems.count + self.unsavedAttachmentExports.count + 1);
|
||
|
NSUInteger savedCount = (self.savedDatabaseItems.count + self.savedAttachmentItems.count);
|
||
6 years ago
|
// Ignore localProfileAvatarItem for now.
|
||
7 years ago
|
|
||
|
CGFloat progress = (savedCount / (CGFloat)(unsavedCount + savedCount));
|
||
7 years ago
|
[self updateProgressWithDescription:NSLocalizedString(@"BACKUP_EXPORT_PHASE_UPLOAD",
|
||
|
@"Indicates that the backup export data is being uploaded.")
|
||
|
progress:@(progress)];
|
||
|
|
||
6 years ago
|
// Save attachment files _before_ anything else, since they
|
||
|
// are the only reusable backup records.
|
||
|
return [self saveAttachmentFilesToCloud]
|
||
|
.thenInBackground(^{
|
||
|
return [self saveDatabaseFilesToCloud];
|
||
|
})
|
||
6 years ago
|
.thenInBackground(^{
|
||
|
return [self saveLocalProfileAvatarToCloud];
|
||
|
})
|
||
6 years ago
|
.thenInBackground(^{
|
||
|
return [self saveManifestFileToCloud];
|
||
|
});
|
||
7 years ago
|
}
|
||
7 years ago
|
|
||
7 years ago
|
// This method returns YES IFF "work was done and there might be more work to do".
|
||
6 years ago
|
- (AnyPromise *)saveDatabaseFilesToCloud
|
||
7 years ago
|
{
|
||
6 years ago
|
if (self.isComplete) {
|
||
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Backup export no longer active.")];
|
||
|
}
|
||
7 years ago
|
|
||
6 years ago
|
NSArray<OWSBackupExportItem *> *items = [self.unsavedDatabaseItems copy];
|
||
|
NSMutableArray<CKRecord *> *records = [NSMutableArray new];
|
||
|
for (OWSBackupExportItem *item in items) {
|
||
6 years ago
|
OWSAssertDebug(item.encryptedItem.filePath.length > 0);
|
||
7 years ago
|
|
||
6 years ago
|
NSString *recordName =
|
||
|
[OWSBackupAPI recordNameForEphemeralFileWithRecipientId:self.recipientId label:@"database"];
|
||
|
CKRecord *record =
|
||
|
[OWSBackupAPI recordForFileUrl:[NSURL fileURLWithPath:item.encryptedItem.filePath] recordName:recordName];
|
||
|
[records addObject:record];
|
||
6 years ago
|
}
|
||
6 years ago
|
|
||
|
// TODO: Expose progress.
|
||
6 years ago
|
return [OWSBackupAPI saveRecordsToCloudObjcWithRecords:records].thenInBackground(^{
|
||
6 years ago
|
OWSAssertDebug(items.count == records.count);
|
||
|
NSUInteger count = MIN(items.count, records.count);
|
||
|
for (NSUInteger i = 0; i < count; i++) {
|
||
|
OWSBackupExportItem *item = items[i];
|
||
|
CKRecord *record = records[i];
|
||
|
|
||
|
OWSAssertDebug(record.recordID.recordName.length > 0);
|
||
|
item.recordName = record.recordID.recordName;
|
||
|
}
|
||
|
|
||
|
[self.savedDatabaseItems addObjectsFromArray:items];
|
||
|
[self.unsavedDatabaseItems removeObjectsInArray:items];
|
||
|
});
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
// This method returns YES IFF "work was done and there might be more work to do".
|
||
6 years ago
|
- (AnyPromise *)saveAttachmentFilesToCloud
|
||
7 years ago
|
{
|
||
6 years ago
|
if (self.isComplete) {
|
||
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Backup export no longer active.")];
|
||
|
}
|
||
|
|
||
6 years ago
|
AnyPromise *promise = [AnyPromise promiseWithValue:@(1)];
|
||
6 years ago
|
NSMutableArray<OWSAttachmentExport *> *items = [NSMutableArray new];
|
||
|
NSMutableArray<CKRecord *> *records = [NSMutableArray new];
|
||
7 years ago
|
|
||
6 years ago
|
for (OWSAttachmentExport *attachmentExport in self.unsavedAttachmentExports) {
|
||
6 years ago
|
if ([self tryToSkipAttachmentUpload:attachmentExport]) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
6 years ago
|
promise = promise.thenInBackground(^{
|
||
6 years ago
|
@autoreleasepool {
|
||
|
// OWSAttachmentExport is used to lazily write an encrypted copy of the
|
||
|
// attachment to disk.
|
||
|
if (![attachmentExport prepareForUpload]) {
|
||
6 years ago
|
// Attachment files are non-critical so any error preparing them is recoverable.
|
||
6 years ago
|
return @(1);
|
||
|
}
|
||
|
OWSAssertDebug(attachmentExport.relativeFilePath.length > 0);
|
||
|
OWSAssertDebug(attachmentExport.encryptedItem);
|
||
6 years ago
|
}
|
||
7 years ago
|
|
||
6 years ago
|
NSURL *_Nullable fileUrl = ^{
|
||
|
if (attachmentExport.encryptedItem.filePath.length < 1) {
|
||
|
OWSLogError(@"attachment export missing temp file path");
|
||
|
return (NSURL *)nil;
|
||
|
}
|
||
|
if (attachmentExport.relativeFilePath.length < 1) {
|
||
|
OWSLogError(@"attachment export missing relative file path");
|
||
|
return (NSURL *)nil;
|
||
|
}
|
||
|
return [NSURL fileURLWithPath:attachmentExport.encryptedItem.filePath];
|
||
|
}();
|
||
7 years ago
|
|
||
6 years ago
|
if (!fileUrl) {
|
||
6 years ago
|
// Attachment files are non-critical so any error preparing them is recoverable.
|
||
6 years ago
|
return @(1);
|
||
6 years ago
|
}
|
||
7 years ago
|
|
||
6 years ago
|
NSString *recordName =
|
||
|
[OWSBackupAPI recordNameForPersistentFileWithRecipientId:self.recipientId
|
||
|
fileId:attachmentExport.attachmentId];
|
||
|
CKRecord *record = [OWSBackupAPI recordForFileUrl:fileUrl recordName:recordName];
|
||
|
[records addObject:record];
|
||
|
[items addObject:attachmentExport];
|
||
|
return @(1);
|
||
|
});
|
||
|
}
|
||
6 years ago
|
|
||
6 years ago
|
void (^cleanup)(void) = ^{
|
||
|
for (OWSAttachmentExport *attachmentExport in items) {
|
||
|
if (![attachmentExport cleanUp]) {
|
||
|
OWSLogError(@"couldn't clean up attachment export.");
|
||
|
// Attachment files are non-critical so any error uploading them is recoverable.
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
6 years ago
|
// TODO: Expose progress.
|
||
|
return promise
|
||
|
.thenInBackground(^{
|
||
|
return [OWSBackupAPI saveRecordsToCloudObjcWithRecords:records];
|
||
6 years ago
|
})
|
||
6 years ago
|
.thenInBackground(^{
|
||
|
OWSAssertDebug(items.count == records.count);
|
||
|
NSUInteger count = MIN(items.count, records.count);
|
||
|
for (NSUInteger i = 0; i < count; i++) {
|
||
|
OWSAttachmentExport *attachmentExport = items[i];
|
||
|
CKRecord *record = records[i];
|
||
|
NSString *recordName = record.recordID.recordName;
|
||
|
OWSAssertDebug(recordName.length > 0);
|
||
|
|
||
|
OWSBackupExportItem *exportItem = [OWSBackupExportItem new];
|
||
|
exportItem.encryptedItem = attachmentExport.encryptedItem;
|
||
|
exportItem.recordName = recordName;
|
||
|
exportItem.attachmentExport = attachmentExport;
|
||
|
[self.savedAttachmentItems addObject:exportItem];
|
||
|
|
||
|
// Immediately save the record metadata to facilitate export resume.
|
||
|
OWSBackupFragment *backupFragment = [[OWSBackupFragment alloc] initWithUniqueId:recordName];
|
||
|
backupFragment.recordName = recordName;
|
||
|
backupFragment.encryptionKey = exportItem.encryptedItem.encryptionKey;
|
||
|
backupFragment.relativeFilePath = attachmentExport.relativeFilePath;
|
||
|
backupFragment.attachmentId = attachmentExport.attachmentId;
|
||
|
backupFragment.uncompressedDataLength = exportItem.uncompressedDataLength;
|
||
5 years ago
|
[LKStorage writeSyncWithBlock:^(YapDatabaseReadWriteTransaction *transaction) {
|
||
6 years ago
|
[backupFragment saveWithTransaction:transaction];
|
||
4 years ago
|
}];
|
||
6 years ago
|
|
||
|
OWSLogVerbose(@"saved attachment: %@ as %@",
|
||
|
attachmentExport.attachmentFilePath,
|
||
|
attachmentExport.relativeFilePath);
|
||
6 years ago
|
}
|
||
6 years ago
|
})
|
||
6 years ago
|
.thenInBackground(^{
|
||
|
cleanup();
|
||
|
})
|
||
|
.catchInBackground(^(NSError *error) {
|
||
|
cleanup();
|
||
|
|
||
|
return error;
|
||
6 years ago
|
});
|
||
7 years ago
|
}
|
||
|
|
||
6 years ago
|
- (BOOL)tryToSkipAttachmentUpload:(OWSAttachmentExport *)attachmentExport
|
||
|
{
|
||
|
if (!self.lastValidRecordNames) {
|
||
|
return NO;
|
||
|
}
|
||
|
|
||
|
// Wherever possible, we do incremental backups and re-use fragments of the last
|
||
|
// backup and/or restore.
|
||
|
// Recycling fragments doesn't just reduce redundant network activity,
|
||
|
// it allows us to skip the local export work, i.e. encryption.
|
||
|
// To do so, we must preserve the metadata for these fragments.
|
||
|
//
|
||
|
// We check two things:
|
||
|
//
|
||
|
// * That we already know the metadata for this fragment (from a previous backup
|
||
|
// or restore).
|
||
|
// * That this record does in fact exist in our CloudKit database.
|
||
|
NSString *recordName =
|
||
|
[OWSBackupAPI recordNameForPersistentFileWithRecipientId:self.recipientId fileId:attachmentExport.attachmentId];
|
||
|
OWSBackupFragment *_Nullable lastBackupFragment = [OWSBackupFragment fetchObjectWithUniqueID:recordName];
|
||
|
if (!lastBackupFragment || ![self.lastValidRecordNames containsObject:recordName]) {
|
||
|
return NO;
|
||
|
}
|
||
|
|
||
|
OWSAssertDebug(lastBackupFragment.encryptionKey.length > 0);
|
||
|
OWSAssertDebug(lastBackupFragment.relativeFilePath.length > 0);
|
||
|
|
||
|
// Recycle the metadata from the last backup's manifest.
|
||
|
OWSBackupEncryptedItem *encryptedItem = [OWSBackupEncryptedItem new];
|
||
|
encryptedItem.encryptionKey = lastBackupFragment.encryptionKey;
|
||
|
attachmentExport.encryptedItem = encryptedItem;
|
||
|
attachmentExport.relativeFilePath = lastBackupFragment.relativeFilePath;
|
||
|
|
||
|
OWSBackupExportItem *exportItem = [OWSBackupExportItem new];
|
||
|
exportItem.encryptedItem = attachmentExport.encryptedItem;
|
||
|
exportItem.recordName = recordName;
|
||
|
exportItem.attachmentExport = attachmentExport;
|
||
|
[self.savedAttachmentItems addObject:exportItem];
|
||
|
|
||
|
OWSLogVerbose(
|
||
|
@"recycled attachment: %@ as %@", attachmentExport.attachmentFilePath, attachmentExport.relativeFilePath);
|
||
|
return YES;
|
||
|
}
|
||
|
|
||
6 years ago
|
- (AnyPromise *)saveLocalProfileAvatarToCloud
|
||
|
{
|
||
|
if (self.isComplete) {
|
||
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Backup export no longer active.")];
|
||
|
}
|
||
|
|
||
|
NSData *_Nullable localProfileAvatarData = self.profileManager.localProfileAvatarData;
|
||
|
if (localProfileAvatarData.length < 1) {
|
||
|
// No profile avatar to backup.
|
||
|
return [AnyPromise promiseWithValue:@(1)];
|
||
|
}
|
||
|
OWSBackupEncryptedItem *_Nullable encryptedItem =
|
||
|
[self.backupIO encryptDataAsTempFile:localProfileAvatarData encryptionKey:self.delegate.backupEncryptionKey];
|
||
|
if (!encryptedItem) {
|
||
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Could not encrypt local profile avatar.")];
|
||
|
}
|
||
|
|
||
|
OWSBackupExportItem *exportItem = [OWSBackupExportItem new];
|
||
|
exportItem.encryptedItem = encryptedItem;
|
||
|
|
||
6 years ago
|
NSString *recordName =
|
||
|
[OWSBackupAPI recordNameForEphemeralFileWithRecipientId:self.recipientId label:@"local-profile-avatar"];
|
||
|
CKRecord *record =
|
||
|
[OWSBackupAPI recordForFileUrl:[NSURL fileURLWithPath:encryptedItem.filePath] recordName:recordName];
|
||
|
return [OWSBackupAPI saveRecordsToCloudObjcWithRecords:@[ record ]].thenInBackground(^{
|
||
|
exportItem.recordName = recordName;
|
||
|
self.localProfileAvatarItem = exportItem;
|
||
|
});
|
||
6 years ago
|
}
|
||
|
|
||
6 years ago
|
- (AnyPromise *)saveManifestFileToCloud
|
||
7 years ago
|
{
|
||
6 years ago
|
if (self.isComplete) {
|
||
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Backup export no longer active.")];
|
||
|
}
|
||
7 years ago
|
|
||
7 years ago
|
OWSBackupEncryptedItem *_Nullable encryptedItem = [self writeManifestFile];
|
||
|
if (!encryptedItem) {
|
||
6 years ago
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Could not generate manifest.")];
|
||
7 years ago
|
}
|
||
7 years ago
|
|
||
|
OWSBackupExportItem *exportItem = [OWSBackupExportItem new];
|
||
|
exportItem.encryptedItem = encryptedItem;
|
||
7 years ago
|
|
||
7 years ago
|
|
||
6 years ago
|
NSString *recordName = [OWSBackupAPI recordNameForManifestWithRecipientId:self.recipientId];
|
||
|
CKRecord *record =
|
||
|
[OWSBackupAPI recordForFileUrl:[NSURL fileURLWithPath:encryptedItem.filePath] recordName:recordName];
|
||
|
return [OWSBackupAPI saveRecordsToCloudObjcWithRecords:@[ record ]].thenInBackground(^{
|
||
|
exportItem.recordName = recordName;
|
||
|
self.manifestItem = exportItem;
|
||
|
});
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
- (nullable OWSBackupEncryptedItem *)writeManifestFile
|
||
7 years ago
|
{
|
||
7 years ago
|
OWSAssertDebug(self.savedDatabaseItems.count > 0);
|
||
|
OWSAssertDebug(self.savedAttachmentItems);
|
||
|
OWSAssertDebug(self.jobTempDirPath.length > 0);
|
||
|
OWSAssertDebug(self.backupIO);
|
||
7 years ago
|
|
||
6 years ago
|
NSMutableDictionary *json = [@{
|
||
7 years ago
|
kOWSBackup_ManifestKey_DatabaseFiles : [self jsonForItems:self.savedDatabaseItems],
|
||
|
kOWSBackup_ManifestKey_AttachmentFiles : [self jsonForItems:self.savedAttachmentItems],
|
||
6 years ago
|
} mutableCopy];
|
||
|
|
||
|
NSString *_Nullable localProfileName = self.profileManager.localProfileName;
|
||
|
if (localProfileName.length > 0) {
|
||
|
json[kOWSBackup_ManifestKey_LocalProfileName] = localProfileName;
|
||
|
}
|
||
|
|
||
|
if (self.localProfileAvatarItem) {
|
||
|
json[kOWSBackup_ManifestKey_LocalProfileAvatar] = [self jsonForItems:@[ self.localProfileAvatarItem ]];
|
||
|
}
|
||
7 years ago
|
|
||
7 years ago
|
OWSLogVerbose(@"json: %@", json);
|
||
7 years ago
|
|
||
7 years ago
|
NSError *error;
|
||
|
NSData *_Nullable jsonData =
|
||
|
[NSJSONSerialization dataWithJSONObject:json options:NSJSONWritingPrettyPrinted error:&error];
|
||
|
if (!jsonData || error) {
|
||
7 years ago
|
OWSFailDebug(@"error encoding manifest file: %@", error);
|
||
7 years ago
|
return nil;
|
||
|
}
|
||
7 years ago
|
return [self.backupIO encryptDataAsTempFile:jsonData encryptionKey:self.delegate.backupEncryptionKey];
|
||
7 years ago
|
}
|
||
|
|
||
|
- (NSArray<NSDictionary<NSString *, id> *> *)jsonForItems:(NSArray<OWSBackupExportItem *> *)items
|
||
|
{
|
||
|
NSMutableArray *result = [NSMutableArray new];
|
||
|
for (OWSBackupExportItem *item in items) {
|
||
|
NSMutableDictionary<NSString *, id> *itemJson = [NSMutableDictionary new];
|
||
7 years ago
|
OWSAssertDebug(item.recordName.length > 0);
|
||
7 years ago
|
|
||
|
itemJson[kOWSBackup_ManifestKey_RecordName] = item.recordName;
|
||
7 years ago
|
OWSAssertDebug(item.encryptedItem.encryptionKey.length > 0);
|
||
7 years ago
|
itemJson[kOWSBackup_ManifestKey_EncryptionKey] = item.encryptedItem.encryptionKey.base64EncodedString;
|
||
|
if (item.attachmentExport) {
|
||
7 years ago
|
OWSAssertDebug(item.attachmentExport.relativeFilePath.length > 0);
|
||
7 years ago
|
itemJson[kOWSBackup_ManifestKey_RelativeFilePath] = item.attachmentExport.relativeFilePath;
|
||
|
}
|
||
7 years ago
|
if (item.attachmentExport.attachmentId) {
|
||
7 years ago
|
OWSAssertDebug(item.attachmentExport.attachmentId.length > 0);
|
||
7 years ago
|
itemJson[kOWSBackup_ManifestKey_AttachmentId] = item.attachmentExport.attachmentId;
|
||
|
}
|
||
7 years ago
|
if (item.uncompressedDataLength) {
|
||
|
itemJson[kOWSBackup_ManifestKey_DataSize] = item.uncompressedDataLength;
|
||
|
}
|
||
7 years ago
|
[result addObject:itemJson];
|
||
7 years ago
|
}
|
||
7 years ago
|
|
||
|
return result;
|
||
7 years ago
|
}
|
||
|
|
||
6 years ago
|
- (AnyPromise *)cleanUp
|
||
7 years ago
|
{
|
||
7 years ago
|
if (self.isComplete) {
|
||
|
// Job was aborted.
|
||
6 years ago
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Backup export no longer active.")];
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
OWSLogVerbose(@"");
|
||
7 years ago
|
|
||
7 years ago
|
[self updateProgressWithDescription:NSLocalizedString(@"BACKUP_EXPORT_PHASE_CLEAN_UP",
|
||
|
@"Indicates that the cloud is being cleaned up.")
|
||
|
progress:nil];
|
||
|
|
||
7 years ago
|
// Now that our backup export has successfully completed,
|
||
|
// we try to clean up the cloud. We can safely delete any
|
||
|
// records not involved in this backup export.
|
||
|
NSMutableSet<NSString *> *activeRecordNames = [NSMutableSet new];
|
||
|
|
||
7 years ago
|
OWSAssertDebug(self.savedDatabaseItems.count > 0);
|
||
7 years ago
|
for (OWSBackupExportItem *item in self.savedDatabaseItems) {
|
||
7 years ago
|
OWSAssertDebug(item.recordName.length > 0);
|
||
|
OWSAssertDebug(![activeRecordNames containsObject:item.recordName]);
|
||
7 years ago
|
[activeRecordNames addObject:item.recordName];
|
||
|
}
|
||
|
for (OWSBackupExportItem *item in self.savedAttachmentItems) {
|
||
7 years ago
|
OWSAssertDebug(item.recordName.length > 0);
|
||
|
OWSAssertDebug(![activeRecordNames containsObject:item.recordName]);
|
||
7 years ago
|
[activeRecordNames addObject:item.recordName];
|
||
6 years ago
|
}
|
||
|
if (self.localProfileAvatarItem) {
|
||
|
OWSBackupExportItem *item = self.localProfileAvatarItem;
|
||
|
OWSAssertDebug(item.recordName.length > 0);
|
||
|
OWSAssertDebug(![activeRecordNames containsObject:item.recordName]);
|
||
|
[activeRecordNames addObject:item.recordName];
|
||
7 years ago
|
}
|
||
7 years ago
|
OWSAssertDebug(self.manifestItem.recordName.length > 0);
|
||
|
OWSAssertDebug(![activeRecordNames containsObject:self.manifestItem.recordName]);
|
||
7 years ago
|
[activeRecordNames addObject:self.manifestItem.recordName];
|
||
7 years ago
|
|
||
7 years ago
|
// Because we do "lazy attachment restores", we need to include the record names for all
|
||
7 years ago
|
// records that haven't been restored yet.
|
||
7 years ago
|
NSArray<NSString *> *restoringRecordNames = [OWSBackup.sharedManager attachmentRecordNamesForLazyRestore];
|
||
|
[activeRecordNames addObjectsFromArray:restoringRecordNames];
|
||
7 years ago
|
|
||
7 years ago
|
[self cleanUpMetadataCacheWithActiveRecordNames:activeRecordNames];
|
||
|
|
||
6 years ago
|
return [self cleanUpCloudWithActiveRecordNames:activeRecordNames];
|
||
7 years ago
|
}
|
||
|
|
||
|
- (void)cleanUpMetadataCacheWithActiveRecordNames:(NSSet<NSString *> *)activeRecordNames
|
||
|
{
|
||
7 years ago
|
OWSAssertDebug(activeRecordNames.count > 0);
|
||
7 years ago
|
|
||
|
if (self.isComplete) {
|
||
|
// Job was aborted.
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// After every successful backup export, we can (and should) cull metadata
|
||
|
// for any backup fragment (i.e. CloudKit record) that wasn't involved in
|
||
|
// the latest backup export.
|
||
5 years ago
|
[LKStorage writeSyncWithBlock:^(YapDatabaseReadWriteTransaction *transaction) {
|
||
6 years ago
|
NSArray<NSString *> *allRecordNames = [transaction allKeysInCollection:[OWSBackupFragment collection]];
|
||
|
|
||
7 years ago
|
NSMutableSet<NSString *> *obsoleteRecordNames = [NSMutableSet new];
|
||
6 years ago
|
[obsoleteRecordNames addObjectsFromArray:allRecordNames];
|
||
7 years ago
|
[obsoleteRecordNames minusSet:activeRecordNames];
|
||
6 years ago
|
|
||
7 years ago
|
[transaction removeObjectsForKeys:obsoleteRecordNames.allObjects inCollection:[OWSBackupFragment collection]];
|
||
4 years ago
|
}];
|
||
7 years ago
|
}
|
||
|
|
||
6 years ago
|
- (AnyPromise *)cleanUpCloudWithActiveRecordNames:(NSSet<NSString *> *)activeRecordNames
|
||
7 years ago
|
{
|
||
7 years ago
|
OWSAssertDebug(activeRecordNames.count > 0);
|
||
7 years ago
|
|
||
|
if (self.isComplete) {
|
||
|
// Job was aborted.
|
||
6 years ago
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Backup export no longer active.")];
|
||
7 years ago
|
}
|
||
|
|
||
6 years ago
|
return [AnyPromise promiseWithResolverBlock:^(PMKResolver resolve) {
|
||
|
[OWSBackupAPI fetchAllRecordNamesWithRecipientId:self.recipientId
|
||
|
success:^(NSArray<NSString *> *recordNames) {
|
||
6 years ago
|
NSMutableSet<NSString *> *obsoleteRecordNames = [NSMutableSet new];
|
||
|
[obsoleteRecordNames addObjectsFromArray:recordNames];
|
||
|
[obsoleteRecordNames minusSet:activeRecordNames];
|
||
|
|
||
|
OWSLogVerbose(@"recordNames: %zd - activeRecordNames: %zd = obsoleteRecordNames: %zd",
|
||
|
recordNames.count,
|
||
|
activeRecordNames.count,
|
||
|
obsoleteRecordNames.count);
|
||
|
|
||
6 years ago
|
[self deleteRecordsFromCloud:[obsoleteRecordNames.allObjects mutableCopy]
|
||
|
deletedCount:0
|
||
|
completion:^(NSError *_Nullable error) {
|
||
|
// Cloud cleanup is non-critical so any error is recoverable.
|
||
|
resolve(@(1));
|
||
|
}];
|
||
6 years ago
|
}
|
||
|
failure:^(NSError *error) {
|
||
7 years ago
|
// Cloud cleanup is non-critical so any error is recoverable.
|
||
6 years ago
|
resolve(@(1));
|
||
|
}];
|
||
|
}];
|
||
7 years ago
|
}
|
||
|
|
||
|
- (void)deleteRecordsFromCloud:(NSMutableArray<NSString *> *)obsoleteRecordNames
|
||
7 years ago
|
deletedCount:(NSUInteger)deletedCount
|
||
7 years ago
|
completion:(OWSBackupJobCompletion)completion
|
||
7 years ago
|
{
|
||
7 years ago
|
OWSAssertDebug(obsoleteRecordNames);
|
||
|
OWSAssertDebug(completion);
|
||
7 years ago
|
|
||
7 years ago
|
OWSLogVerbose(@"");
|
||
7 years ago
|
|
||
|
if (obsoleteRecordNames.count < 1) {
|
||
|
// No more records to delete; cleanup is complete.
|
||
7 years ago
|
return completion(nil);
|
||
|
}
|
||
|
|
||
|
if (self.isComplete) {
|
||
|
// Job was aborted.
|
||
|
return completion(nil);
|
||
7 years ago
|
}
|
||
|
|
||
7 years ago
|
CGFloat progress = (obsoleteRecordNames.count / (CGFloat)(obsoleteRecordNames.count + deletedCount));
|
||
|
[self updateProgressWithDescription:NSLocalizedString(@"BACKUP_EXPORT_PHASE_CLEAN_UP",
|
||
|
@"Indicates that the cloud is being cleaned up.")
|
||
|
progress:@(progress)];
|
||
|
|
||
7 years ago
|
static const NSUInteger kMaxBatchSize = 100;
|
||
|
NSMutableArray<NSString *> *batchRecordNames = [NSMutableArray new];
|
||
|
while (obsoleteRecordNames.count > 0 && batchRecordNames.count < kMaxBatchSize) {
|
||
|
NSString *recordName = obsoleteRecordNames.lastObject;
|
||
|
[obsoleteRecordNames removeLastObject];
|
||
|
[batchRecordNames addObject:recordName];
|
||
|
}
|
||
7 years ago
|
|
||
7 years ago
|
[OWSBackupAPI deleteRecordsFromCloudWithRecordNames:batchRecordNames
|
||
7 years ago
|
success:^{
|
||
6 years ago
|
[self deleteRecordsFromCloud:obsoleteRecordNames
|
||
|
deletedCount:deletedCount + batchRecordNames.count
|
||
|
completion:completion];
|
||
7 years ago
|
}
|
||
|
failure:^(NSError *error) {
|
||
6 years ago
|
// Cloud cleanup is non-critical so any error is recoverable.
|
||
6 years ago
|
[self deleteRecordsFromCloud:obsoleteRecordNames
|
||
|
deletedCount:deletedCount + batchRecordNames.count
|
||
|
completion:completion];
|
||
7 years ago
|
}];
|
||
|
}
|
||
|
|
||
7 years ago
|
@end
|
||
|
|
||
|
NS_ASSUME_NONNULL_END
|