我在保存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,以便共享它们。谢谢。
答案 0 :(得分:2)
即使共享操作被取消(因为您没有明确表示取消的意思),您可能实际创建并保存在私人数据库中的CKShare记录。您只是取消了更改其他用户的参与状态(例如,邀请)。因此,当您重复此过程时,您尝试重新保存相同的CKShare,尽管再次设置,因此可能会修改更改标记。 CloudKit看到这一点并返回“服务器记录已更改”错误。
如果我是对的,你可以采用以下两种方式之一。首先,将CKModifyRecordsOperation
变量savePolicy
的值设置为.allKeys
,例如,设置qualityOfService
var的位置。这将强制要求即使CloudKit在您的服务器上找到记录,它也会自动覆盖它。如果您确定上传的版本应该总是获胜,这是一个很好的方法(并且只会导致单个网络呼叫)。
大多数人使用的第二种更通用的方法是首先使用根记录的CKFetchRecordsOperation
var recordID创建share
。所有CKRecord
个对象的share
var都是CKReference
,但如果尚未共享该对象,则该值为零。如果我是对的,在取消后的第二次尝试中,您将找到根记录的共享参考,以包含您在第一次尝试时设置的CKShare。因此,抓住该份额,然后使用它,像以前一样启动父级和其他引用的其余部分!