CKShare“服务器记录已更改”错误

时间:2018-01-22 23:22:36

标签: ios swift cloudkit

我在保存CKShare时遇到了问题。执行共享时,一切正常,但如果共享被取消并尝试再次共享,则会收到错误:

"Server Record Changed" (14/2004); server message = "client oplock error updating record"

以下是我的分享代码:

func share(_ deck: Deck) {

    let zoneID = CKManager.defaultManager.sharedZone?.zoneID


    if let deckName = deck.name {
        selectedDeckForPrivateShare = deckName
    }

    var records = [CKRecord]()


    let deckRecord = deck.deckToCKRecord(zoneID)

    records.append(deckRecord)


    let reference = CKReference(record: deckRecord, action: .none)

    for index in deck.cards {
        let cardRecord = index.cardToCKRecord(zoneID)
        cardRecord["deck"] = reference
        cardRecord.parent = reference

        records.append(cardRecord)

    }


    let share = CKShare(rootRecord: deckRecord)
    share[CKShareTitleKey] = "\(String(describing: deck.name)))" as CKRecordValue?
    records.append(share)
    let sharingController = UICloudSharingController { (controller, preparationCompletion) in


        let modifyOP = CKModifyRecordsOperation(recordsToSave: records, recordIDsToDelete: nil)

        modifyOP.qualityOfService = .userInteractive
        modifyOP.modifyRecordsCompletionBlock = { (savedRecords,deletedRecordIDs,error) in
            preparationCompletion(share, CKContainer.default(), error)

            controller.delegate = self

            if let error = error {
                print(error)

            }
        }


        self.privateDatabase.add(modifyOP)

    }


    sharingController.delegate = self

    self.present(sharingController, animated: true, completion: {

    })   
}

我使用两种方法deckToCKRecord()cardToCKRecord()将核心数据转换为CKRecord,以便共享它们。谢谢。

1 个答案:

答案 0 :(得分:2)

即使共享操作被取消(因为您没有明确表示取消的意思),您可能实际创建并保存在私人数据库中的CKShare记录。您只是取消了更改其他用户的参与状态(例如,邀请)。因此,当您重复此过程时,您尝试重新保存相同的CKShare,尽管再次设置,因此可能会修改更改标记。 CloudKit看到这一点并返回“服务器记录已更改”错误。

如果我是对的,你可以采用以下两种方式之一。首先,将CKModifyRecordsOperation变量savePolicy的值设置为.allKeys,例如,设置qualityOfService var的位置。这将强制要求即使CloudKit在您的服务器上找到记录,它也会自动覆盖它。如果您确定上传的版本应该总是获胜,这是一个很好的方法(并且只会导致单个网络呼叫)。

大多数人使用的第二种更通用的方法是首先使用根记录的CKFetchRecordsOperation var recordID创建share。所有CKRecord个对象的share var都是CKReference,但如果尚未共享该对象,则该值为零。如果我是对的,在取消后的第二次尝试中,您将找到根记录的共享参考,以包含您在第一次尝试时设置的CKShare。因此,抓住该份额,然后使用它,像以前一样启动父级和其他引用的其余部分!