CloudKitDictionarySyncer is a utility that you can use in your app to save an NSDictionary to both a local plist file and remotely in the user's CloudKit private database. If CloudKit is not available, because the user is offline, not logged in or whatever , the NSDictionary will be saved to the local plist only. CloudKitDictionarySyncer is written in Swift.
At app startup, when you load the dictionary, CloudKitDictionarySyncer will return a single dictionary if the plist and iCloud version are identical. If not, a Conflict tuple containing both dictionaries will be returned. Handle the conflict in a way appropriate for your app, for example by merging the data in both dictionaries or by simply choosing the last saved dictionary (last method not recommended). After the conflict is solved, you should save the dictionary immediately. This will resync the local plist dictionary and the iCloud dictionary.
##Installation
Add the file CloudKitDictionarySyncer.swift to your project
##Usage
-
Turn on iCloud in the Capabilities tab for your build target. Make sure to enable CloudKit using the checkbox that shows up.
-
In the iCloud dashboard, add a new Record type called Plists. In the Plists type, add two string attributes: dictname and plistxml. The resulting Record Type should look like this:
If you have trouble setting up Cloudkit, read this excellent blog post
Create a syncer object for each dictionary that you want to persist/sync. If you set the debug flag to true you will get some debug log messages in your console log.
let exampledictSyncer = CloudKitDictionarySyncer(dictname: "exampledict", debug: true)
This should be done only once per app session. Pass a function that can receive a LoadResult enum, that will contain either a NSMutableDictionary or a Conflict tuple.
self.exampledictSyncer.loadDictionary(onComplete: {
loadResult in
switch loadResult {
case .Dict(let loadeddict):
// No conflict
self.dict = loadeddict
case .Conflict(let plistdict, let clouddict, let latest):
// Handle conflict, for example by merging.
self.dict = myMergeFunction(plistdict, clouddict)
// Save the merged dict immediately
self.syncer.saveDictionary(self.dict, onComplete: {
status in
println("Resaved merged dict. Save status = \(status)")
})
}
})
To be safe, you should do this whenever you have updated your dictionary. For the battery savers and the more adventurous, save when the app goes into background. Pass a function that takes a String, that will contain an informational status message from the save operation.
self.exampledictSyncer.saveDictionary(self.dict, onComplete: {
status in
println("Save status = \(status)")
})
##Example Project
For more info, have a look at the example project in this repo.
##Why should I use this?
If your app wants to save any data, you will need to persist it somewhere. If you only persist the data locally, the user cannot see the same data on other devices or new devices. If you only persist the data remotely, the user cannot use your app when being offline.
One solution to persist the data both locally and remotely is to use NSUbiquitousKeyValueStore in iCloud, but in my experience, the key-value store syncing at app start can be delayed up to 30 seconds, which is way too long. You would need to wait for the syncing to complete before letting the user update any of the persisted data.
With CloudKit, you can now use iCloud like a regular remote database, giving you much more control over reading and writing data from iCloud. Since CloudKit seems to have very short response times, CloudKitDictionarySyncer can quickly load your data from iCloud, or from a local plist file if CloudKit is not available. Because the loading is so quick, you can let your user wait while the initial loading of your data is executing.
CloudKitDictionarySyncer will also give you full control over the conflict handling. You decide what to do if the remote and local dictionaries are not identical.
Still, this is a very simple utility, that doesn't take advantage of all aspects of CloudKit. The plist xml is simply saved in a single field in the users private CloudKit database. If you are looking for more sophisticated syncing, I would recommend using Couchbase Lite IOS in combination with a Couchbase or CouchDB server. That is a better solution for syncing the user data. However, if you want to avoid explicit user logins in your app, with all the hassle of creating login and registration views, forgot-password functionality etc, you would still need to save a generated username in iCloud. For that, you could use CloudKitDictionarySyncer :)
##Feedback and Contribution
All feedback and contribution is very appreciated. Please send pull requests, create issues or just send an email to mats.melke@gmail.com.
##Copyrights
- Copyright (c) 2015- Mats Melke. Please see LICENSE.txt for details.