OSX和iOS之间的iCloud Core数据共享

时间:2015-09-01 11:50:25

标签: ios macos core-data icloud

我有一个OSX和iOS的应用程序,都使用iCloud在它们之间共享核心数据。当我在iPhone上进行更改时,我可以在OSX应用程序中看到它们,并在两个应用程序中调用NSPersistentStoreDidImportUbiquitousContentChangesNotification,因此iOS - > Mac效果很好。但是当我在Mac应用程序中进行一些更改时,我无法在iPhone中看到它们。 iPhone永远不会调用NSPersistentStoreDidImportUbiquitousContentChangesNotification。我能改变Mac的唯一方法是在iPhone上做一些改变。然后我可以看到所有的变化。

两个项目中的iCloud集成是一样的。核心数据数据模型是相同的。权利也是一样的。

我的所有代码都基于此库:http://ossh.com.au/design-and-technology/software-development/sample-library-style-ios-core-data-app-with-icloud-integration/

为什么在我在iPhone上进行一些更改之前iPhone没有改变Mac?有什么想法吗?

另一个问题

在我的应用程序中,当用户启用iCloud时,我会检查iCloud上是否已有文件。如果文件存在,则用户可以选择从iCloud文件还原核心数据或从本地存储启动新副本。 要开始新的副本,我使用以下代码:

- (void)startNewCopy
{
    [self removeICloudStore];
    [self moveStoreToIcloud];
}

- (void)removeICloudStore
{
    BOOL result;
    NSError *error;
    // Now delete the iCloud content and file
    result = [NSPersistentStoreCoordinator removeUbiquitousContentAndPersistentStoreAtURL:[self icloudStoreURL]
                                                                             options:[self icloudStoreOptions]
                                                                             error:&error];
    if (!result)
    {
        NSLog(@"Error removing store");
    }
    else
    {
        NSLog(@"Core Data store removed.");
        // Now delete the local file
        [self deleteLocalCopyOfiCloudStore];
    }

}


- (void)deleteLocalCopyOfiCloudStore {
   // We need to get the URL to the store
   NSError *error = nil;
   [[NSFileManager defaultManager] removeItemAtURL:[self localUbiquitySupportURL] error:&error];
}

- (void)moveStoreToIcloud
{
     // Open the store
     NSPersistentStore *sourceStore = [[_persistentStoreCoordinator persistentStores] firstObject];

     if (!sourceStore)
     {
         NSLog(@" failed to add old store");
     }
     else
     {
         NSLog(@" Successfully added store to migrate");
         NSError *error;
         id migrationSuccess = [_persistentStoreCoordinator migratePersistentStore:sourceStore toURL:[self icloudStoreURL] options:[self icloudStoreOptions] withType:NSSQLiteStoreType error:&error];

         if (migrationSuccess)
         {
             NSLog(@"Store migrated to iCloud");

             _persistentStoreCoordinator = nil;
             _managedObjectContext = nil;

             // Now delete the local file
             [self deleteLocalStore];

         }
         else
         {
             NSLog(@"Failed to migrate store: %@, %@", error, error.userInfo);

         }
     }

}

- (void)deleteLocalStore
{
     NSError *error = nil;
     [[NSFileManager defaultManager] removeItemAtURL:[self localStoreURL] error:&error];
}

从单个设备执行此操作似乎都运行良好,但是当我尝试使用多个设备时,我遇到了一些错误。

示例:

我有两个设备。第一个没有连接到iCloud,第二个连接到iCloud。 在启用iCloud的第一个设备中,我选择startNewCopy,然后在第二个设备中出现此错误:

  

CoreData:Ubiquity:Librarian在开始下载时返回了一个严重错误错误Domain = BRCloudDocsErrorDomain Code = 5“操作无法完成。(BRCloudDocsErrorDomain错误5 - URL上没有文档)”

错误之前永远不会调用NSPersistentStoreCoordinatorStoresDidChangeNotification

这是我的通知代码:

- (void)processStoresDidChange:(NSNotification *)notification
{
      NSLog(@"processStoresDidChange");
      // Post notification to trigger UI updates

      // Check type of transition
      NSNumber *type = [notification.userInfo objectForKey:NSPersistentStoreUbiquitousTransitionTypeKey];

      //NSLog(@" userInfo is %@", notification.userInfo);
      //NSLog(@" transition type is %@", type);

      if (type.intValue == NSPersistentStoreUbiquitousTransitionTypeInitialImportCompleted) {

          NSLog(@" transition type is NSPersistentStoreUbiquitousTransitionTypeInitialImportCompleted");

      } else if (type.intValue == NSPersistentStoreUbiquitousTransitionTypeAccountAdded) {
          NSLog(@" transition type is NSPersistentStoreUbiquitousTransitionTypeAccountAdded");
      } else if (type.intValue == NSPersistentStoreUbiquitousTransitionTypeAccountRemoved) {
          NSLog(@" transition type is NSPersistentStoreUbiquitousTransitionTypeAccountRemoved");
      } else if (type.intValue == NSPersistentStoreUbiquitousTransitionTypeContentRemoved) {
          NSLog(@" transition type is NSPersistentStoreUbiquitousTransitionTypeContentRemoved");
      }

      [[NSOperationQueue mainQueue] addOperationWithBlock:^ {

          if (type.intValue == NSPersistentStoreUbiquitousTransitionTypeContentRemoved) {

              [self deleteLocalStore];
              _persistentStoreCoordinator = nil;
              _managedObjectContext = nil;

              NSLog(@" iCloud store was removed! Wait for empty store");
          }

          // Refresh user Interface
          [[NSNotificationCenter defaultCenter] postNotificationName:@"notiIcloud" object:@"storeChanged"];
      }];

}

如何检测iCloud内容已被删除并避免上述错误?

2 个答案:

答案 0 :(得分:0)

在我看来,你可能没有使用适当的接口。

您可以在不使用NSPersistentStore方法的情况下将文件移入和移出iCloud。 通常,相应的操作也必须包含在NSFileCoordinator电话中。

这是我在iOS上做的事情。 toLocal表示移入或移出本地(设备)存储:

//
/// Move file between stores (assumed no name clash).
/// Return new URL.
/// Note: cloud file moves are asynchronous.
///
- (NSURL *)moveStoreFile:(NSURL *)url toRootURL:(NSURL *)rootURL toLocal:(BOOL)toLocal
{
    NSURL *newURL = rootURL;
    if (!toLocal)
        newURL = [newURL URLByAppendingPathComponent:@"Documents"];
    newURL = [newURL URLByAppendingPathComponent:url.lastPathComponent];

    [self backupFile:url isLocal:!toLocal completionHandler:
     ^(BOOL success) {

        MRLOG(@"MOVE %s %@ to %s %@", toLocal? "icloud" : "local", [url lastPathComponent],
              toLocal? "local" : "icloud", [newURL lastPathComponent]);

        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0),
        ^{
           NSError *error;
           // setUbiquitous does file coordination
           BOOL msuccess = [[NSFileManager defaultManager]
                           setUbiquitous:!toLocal itemAtURL:url destinationURL:newURL error:&error];

           dispatch_async(dispatch_get_main_queue(),
           ^{
              if (!msuccess)
                  MRLOG(@"move failed: %@", error);
              [self.delegate moveStoreFileCompletedWithStatus:success error:error];
           });
        });
    }];

    return newURL;
}

删除需要明确的文件协调:

///
/// Purge backup file (really delete it).
/// Note: cloud deletes are asynchronous.
///
- (void)purgeFile:(NSURL *)url isLocal:(BOOL)isLocal
{
    MRLOG(@"PURGE %s %@", isLocal? "local" : "icloud", [url lastPathComponent]);

    if (isLocal)
        [self removeFile:url];
    else
        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0),
        ^{
            NSError *coordinationError;
            NSFileCoordinator *coordinator = [[NSFileCoordinator alloc] initWithFilePresenter:nil];

            [coordinator coordinateWritingItemAtURL:url options:NSFileCoordinatorWritingForDeleting
                                                    error:&coordinationError
                                     byAccessor:
             ^(NSURL* writingURL)
             {
                 [self removeFile:writingURL];
             }];
            if (coordinationError) {
                MRLOG(@"coordination error: %@", coordinationError);
                [(SSApplication *)[SSApplication sharedApplication] fileErrorAlert:coordinationError];
            }
        });
}

要检测您需要使用NSMetadataQuery的文件的更改,并为NSMetadataQueryDidUpdateNotification添加观察者。

答案 1 :(得分:0)

我有一个类似的问题,将iOS与我的Mac同步。

此错误5实际上意味着您的商店存在问题。

您可以尝试以下步骤:

  • 在iOS上,您需要重置模拟器的内容并清除iCloud内容(仅适用于您的应用)。

您可以通过致电重置您的iCloud内容(抱歉,我在Swift中拥有它):

try! NSPersistentStoreCoordinator.removeUbiquitousContentAndPersistentStoreAtURL(storeURL, options: self.options)
  • 在Mac OS上,您需要删除此文件夹的内容" CoreDataUbiquitySupport / YourAppId "。这基本上是您的Core Data Persistent Store(例如:SQLite文件)。

当Model(xcdatamodeld文件)发生更改且现有数据尚未清除时,会发生此问题。您最终得到一个与旧模型下生成的数据相关联的新模型,并且您尝试跨设备同步该混合...

将来,要避免与模型相关的问题,请查看核心数据模型版本控制和数据迁移。 https://developer.apple.com/library/mac/documentation/Cocoa/Conceptual/CoreDataVersioning/Articles/Introduction.html

希望它能解决你的问题。