|
|
|
@ -740,8 +740,6 @@ NS_ASSUME_NONNULL_BEGIN
|
|
|
|
|
// This method returns YES IFF "work was done and there might be more work to do".
|
|
|
|
|
- (AnyPromise *)saveDatabaseFilesToCloud
|
|
|
|
|
{
|
|
|
|
|
// AnyPromise *promise = [AnyPromise promiseWithValue:@(1)];
|
|
|
|
|
|
|
|
|
|
if (self.isComplete) {
|
|
|
|
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Backup export no longer active.")];
|
|
|
|
|
}
|
|
|
|
@ -759,7 +757,7 @@ NS_ASSUME_NONNULL_BEGIN
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// TODO: Expose progress.
|
|
|
|
|
return [OWSBackupAPI saveRecordsToCloudObjcWithRecords:records].thenInBackground(^(NSString *recordName) {
|
|
|
|
|
return [OWSBackupAPI saveRecordsToCloudObjcWithRecords:records].thenInBackground(^{
|
|
|
|
|
OWSAssertDebug(items.count == records.count);
|
|
|
|
|
NSUInteger count = MIN(items.count, records.count);
|
|
|
|
|
for (NSUInteger i = 0; i < count; i++) {
|
|
|
|
@ -778,121 +776,152 @@ NS_ASSUME_NONNULL_BEGIN
|
|
|
|
|
// This method returns YES IFF "work was done and there might be more work to do".
|
|
|
|
|
- (AnyPromise *)saveAttachmentFilesToCloud
|
|
|
|
|
{
|
|
|
|
|
if (self.isComplete) {
|
|
|
|
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Backup export no longer active.")];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
AnyPromise *promise = [AnyPromise promiseWithValue:@(1)];
|
|
|
|
|
NSMutableArray<OWSAttachmentExport *> *items = [NSMutableArray new];
|
|
|
|
|
NSMutableArray<CKRecord *> *records = [NSMutableArray new];
|
|
|
|
|
|
|
|
|
|
for (OWSAttachmentExport *attachmentExport in self.unsavedAttachmentExports) {
|
|
|
|
|
if ([self tryToSkipAttachmentUpload:attachmentExport]) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
promise = promise.thenInBackground(^{
|
|
|
|
|
if (self.isComplete) {
|
|
|
|
|
return [AnyPromise promiseWithValue:OWSBackupErrorWithDescription(@"Backup export no longer active.")];
|
|
|
|
|
@autoreleasepool {
|
|
|
|
|
// OWSAttachmentExport is used to lazily write an encrypted copy of the
|
|
|
|
|
// attachment to disk.
|
|
|
|
|
if (![attachmentExport prepareForUpload]) {
|
|
|
|
|
// Attachment files are non-critical so any error uploading them is recoverable.
|
|
|
|
|
return @(1);
|
|
|
|
|
}
|
|
|
|
|
OWSAssertDebug(attachmentExport.relativeFilePath.length > 0);
|
|
|
|
|
OWSAssertDebug(attachmentExport.encryptedItem);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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];
|
|
|
|
|
}();
|
|
|
|
|
|
|
|
|
|
if (!fileUrl) {
|
|
|
|
|
// Attachment files are non-critical so any error uploading them is recoverable.
|
|
|
|
|
return @(1);
|
|
|
|
|
}
|
|
|
|
|
return [self saveAttachmentFileToCloud:attachmentExport];
|
|
|
|
|
|
|
|
|
|
NSString *recordName =
|
|
|
|
|
[OWSBackupAPI recordNameForPersistentFileWithRecipientId:self.recipientId
|
|
|
|
|
fileId:attachmentExport.attachmentId];
|
|
|
|
|
CKRecord *record = [OWSBackupAPI recordForFileUrl:fileUrl recordName:recordName];
|
|
|
|
|
[records addObject:record];
|
|
|
|
|
[items addObject:attachmentExport];
|
|
|
|
|
return @(1);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
[self.unsavedAttachmentExports removeAllObjects];
|
|
|
|
|
return promise;
|
|
|
|
|
|
|
|
|
|
// TODO: Expose progress.
|
|
|
|
|
dispatch_queue_t backgroundQueue = dispatch_get_global_queue(0, 0);
|
|
|
|
|
return promise
|
|
|
|
|
.thenInBackground(^{
|
|
|
|
|
return [OWSBackupAPI saveRecordsToCloudObjcWithRecords:records];
|
|
|
|
|
})
|
|
|
|
|
.ensureOn(backgroundQueue,
|
|
|
|
|
^{
|
|
|
|
|
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.
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
.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;
|
|
|
|
|
[self.dbConnection readWriteWithBlock:^(YapDatabaseReadWriteTransaction *transaction) {
|
|
|
|
|
[backupFragment saveWithTransaction:transaction];
|
|
|
|
|
}];
|
|
|
|
|
|
|
|
|
|
OWSLogVerbose(@"saved attachment: %@ as %@",
|
|
|
|
|
attachmentExport.attachmentFilePath,
|
|
|
|
|
attachmentExport.relativeFilePath);
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
.catchInBackground(^{
|
|
|
|
|
// Attachment files are non-critical so any error uploading them is recoverable.
|
|
|
|
|
return [AnyPromise promiseWithValue:@(1)];
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (AnyPromise *)saveAttachmentFileToCloud:(OWSAttachmentExport *)attachmentExport
|
|
|
|
|
- (BOOL)tryToSkipAttachmentUpload:(OWSAttachmentExport *)attachmentExport
|
|
|
|
|
{
|
|
|
|
|
if (self.lastValidRecordNames) {
|
|
|
|
|
// 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 *lastRecordName =
|
|
|
|
|
[OWSBackupAPI recordNameForPersistentFileWithRecipientId:self.recipientId
|
|
|
|
|
fileId:attachmentExport.attachmentId];
|
|
|
|
|
OWSBackupFragment *_Nullable lastBackupFragment = [OWSBackupFragment fetchObjectWithUniqueID:lastRecordName];
|
|
|
|
|
if (lastBackupFragment && [self.lastValidRecordNames containsObject:lastRecordName]) {
|
|
|
|
|
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 = lastRecordName;
|
|
|
|
|
exportItem.attachmentExport = attachmentExport;
|
|
|
|
|
[self.savedAttachmentItems addObject:exportItem];
|
|
|
|
|
|
|
|
|
|
OWSLogVerbose(@"recycled attachment: %@ as %@",
|
|
|
|
|
attachmentExport.attachmentFilePath,
|
|
|
|
|
attachmentExport.relativeFilePath);
|
|
|
|
|
return [AnyPromise promiseWithValue:@(1)];
|
|
|
|
|
}
|
|
|
|
|
if (!self.lastValidRecordNames) {
|
|
|
|
|
return NO;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@autoreleasepool {
|
|
|
|
|
// OWSAttachmentExport is used to lazily write an encrypted copy of the
|
|
|
|
|
// attachment to disk.
|
|
|
|
|
if (![attachmentExport prepareForUpload]) {
|
|
|
|
|
// Attachment files are non-critical so any error uploading them is recoverable.
|
|
|
|
|
return [AnyPromise promiseWithValue:@(1)];
|
|
|
|
|
}
|
|
|
|
|
OWSAssertDebug(attachmentExport.relativeFilePath.length > 0);
|
|
|
|
|
OWSAssertDebug(attachmentExport.encryptedItem);
|
|
|
|
|
// 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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return [OWSBackupAPI
|
|
|
|
|
savePersistentFileOnceToCloudObjcWithRecipientId:self.recipientId
|
|
|
|
|
fileId:attachmentExport.attachmentId
|
|
|
|
|
fileUrlBlock:^{
|
|
|
|
|
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];
|
|
|
|
|
}]
|
|
|
|
|
.thenInBackground(^(NSString *recordName) {
|
|
|
|
|
if (![attachmentExport cleanUp]) {
|
|
|
|
|
OWSLogError(@"couldn't clean up attachment export.");
|
|
|
|
|
// Attachment files are non-critical so any error uploading them is recoverable.
|
|
|
|
|
}
|
|
|
|
|
OWSAssertDebug(lastBackupFragment.encryptionKey.length > 0);
|
|
|
|
|
OWSAssertDebug(lastBackupFragment.relativeFilePath.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;
|
|
|
|
|
[self.dbConnection readWriteWithBlock:^(YapDatabaseReadWriteTransaction *transaction) {
|
|
|
|
|
[backupFragment saveWithTransaction:transaction];
|
|
|
|
|
}];
|
|
|
|
|
// Recycle the metadata from the last backup's manifest.
|
|
|
|
|
OWSBackupEncryptedItem *encryptedItem = [OWSBackupEncryptedItem new];
|
|
|
|
|
encryptedItem.encryptionKey = lastBackupFragment.encryptionKey;
|
|
|
|
|
attachmentExport.encryptedItem = encryptedItem;
|
|
|
|
|
attachmentExport.relativeFilePath = lastBackupFragment.relativeFilePath;
|
|
|
|
|
|
|
|
|
|
OWSLogVerbose(
|
|
|
|
|
@"saved attachment: %@ as %@", attachmentExport.attachmentFilePath, attachmentExport.relativeFilePath);
|
|
|
|
|
})
|
|
|
|
|
.catchInBackground(^{
|
|
|
|
|
if (![attachmentExport cleanUp]) {
|
|
|
|
|
OWSLogError(@"couldn't clean up attachment export.");
|
|
|
|
|
// Attachment files are non-critical so any error uploading them is recoverable.
|
|
|
|
|
}
|
|
|
|
|
OWSBackupExportItem *exportItem = [OWSBackupExportItem new];
|
|
|
|
|
exportItem.encryptedItem = attachmentExport.encryptedItem;
|
|
|
|
|
exportItem.recordName = recordName;
|
|
|
|
|
exportItem.attachmentExport = attachmentExport;
|
|
|
|
|
[self.savedAttachmentItems addObject:exportItem];
|
|
|
|
|
|
|
|
|
|
// Attachment files are non-critical so any error uploading them is recoverable.
|
|
|
|
|
return [AnyPromise promiseWithValue:@(1)];
|
|
|
|
|
});
|
|
|
|
|
OWSLogVerbose(
|
|
|
|
|
@"recycled attachment: %@ as %@", attachmentExport.attachmentFilePath, attachmentExport.relativeFilePath);
|
|
|
|
|
return YES;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (AnyPromise *)saveLocalProfileAvatarToCloud
|
|
|
|
@ -922,8 +951,6 @@ NS_ASSUME_NONNULL_BEGIN
|
|
|
|
|
return [OWSBackupAPI saveRecordsToCloudObjcWithRecords:@[ record ]].thenInBackground(^{
|
|
|
|
|
exportItem.recordName = recordName;
|
|
|
|
|
self.localProfileAvatarItem = exportItem;
|
|
|
|
|
|
|
|
|
|
return [AnyPromise promiseWithValue:@(1)];
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -941,14 +968,14 @@ NS_ASSUME_NONNULL_BEGIN
|
|
|
|
|
OWSBackupExportItem *exportItem = [OWSBackupExportItem new];
|
|
|
|
|
exportItem.encryptedItem = encryptedItem;
|
|
|
|
|
|
|
|
|
|
return [OWSBackupAPI upsertManifestFileToCloudObjcWithRecipientId:self.recipientId
|
|
|
|
|
fileUrl:[NSURL fileURLWithPath:encryptedItem.filePath]]
|
|
|
|
|
.thenInBackground(^(NSString *recordName) {
|
|
|
|
|
exportItem.recordName = recordName;
|
|
|
|
|
self.manifestItem = exportItem;
|
|
|
|
|
|
|
|
|
|
// All files have been saved to the cloud.
|
|
|
|
|
});
|
|
|
|
|
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;
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (nullable OWSBackupEncryptedItem *)writeManifestFile
|
|
|
|
|