For various the CDTIncrementalStore
project has been spun off
into its own cocoapod'able project. The current work can be found at
the following places:
- github: https://github.com/jimix/CDTIncrementalStore
- cocoapods: https://cocoapods.org/pods/CDTIncrementalStore
Applications use Cloudant Sync to store, index and query local JSON data on a device and to synchronise data between many devices. Synchronisation is under the control of the application, rather than being controlled by the underlying system. Conflicts are also easy to manage and resolve, either on the local device or in the remote database.
Cloudant Sync is an Apache CouchDB™ replication-protocol-compatible datastore for devices that don't want or need to run a full CouchDB instance. It's built by Cloudant, building on the work of many others, and is available under the Apache 2.0 licence.
The API is quite different from CouchDB's; we retain the MVCC data model but not the HTTP-centric API.
This library is for iOS, an Android version is also available.
If you have questions, please join our mailing list and drop us a line.
CDTDatastore is available through CocoaPods, to install it add the following line to your Podfile:
pod "CDTDatastore"
CDTDatastore is useable from Swift out of the box with a few small quirks. Install as per the instructions above, and import CloudantSync.h into your bridging header. If you need to iterate over the CDTQueryResult class, you need to create a small extension before you can do so in Swift:
extension CDTQueryResult: SequenceType {
public func generate() -> NSFastGenerator {
return NSFastGenerator(self)
}
}
The Overview section below has examples in both Objective-C and Swift.
There is an example project in the Project
folder, for iOS 7. To get
this up and running independently of the main codebase, a Podfile is
included:
$ cd Project
$ pod install
$ open Project.xcworkspace
See CONTRIBUTING.
CDTDatastore gets regularly tested on the following platforms:
- OS X 10.10.2 (Yosemite)
- iPhone 4S (Simulator), iOS 8.1 and 7.1
- iPhone 5 (Simulator), iOS 8.1 and 7.1
- iPhone 5S (Simulator), iOS 8.1 and 7.1
- iPhone 6 Plus (Simulator), iOS 8.1
- iPhone 6 (Simulator), iOS 8.1
- iPad 2 (Simulator), iOS 8.1 and 7.1
- iPad Air (Simulator), iOS 8.1 and 7.1
- iPad Retina (Simulator), iOS 8.1 and 7.1
Once the libraries are added to a project, the basics of adding and reading a document are:
#import <CloudantSync.h>
// Create a CDTDatastoreManager using application internal storage path
NSError *outError = nil;
NSFileManager *fileManager= [NSFileManager defaultManager];
NSURL *documentsDir = [[fileManager URLsForDirectory:NSDocumentDirectory
inDomains:NSUserDomainMask] lastObject];
NSURL *storeURL = [documentsDir URLByAppendingPathComponent:@"cloudant-sync-datastore"];
NSString *path = [storeURL path];
CDTDatastoreManager *manager =
[[CDTDatastoreManager alloc] initWithDirectory:path
error:&outError];
CDTDatastore *datastore = [manager datastoreNamed:@"my_datastore"
error:&outError];
// Create a document
CDTMutableDocumentRevision *rev = [CDTMutableDocumentRevision revision];
rev.docId = @"doc1"; // Or don't and get an ID generated for you
rev.body = @{
@"description": @"Buy milk",
@"completed": @NO,
@"type": @"com.cloudant.sync.example.task"
};
// Add an attachment -- binary data like a JPEG
CDTUnsavedFileAttachment *att1 = [[CDTUnsavedFileAttachment alloc]
initWithPath:@"/path/to/image.jpg"
name:@"cute_cat.jpg"
type:@"image/jpeg"];
rev.attachments = @{ att1.name:att1 };
// Save the document to the database
CDTDocumentRevision *revision = [datastore createDocumentFromRevision:rev
error:&error];
// Read a document
NSString *docId = revision.docId;
CDTDocumentRevision *retrieved = [datastore getDocumentWithId:docId
error:&error];
If you are using Swift, install the libraries as per the instructions above, and configure a bridging header for your project. Pull in the CloudantSync.h header into the bridging header, and you should be good to go:
#import <CloudantSync.h>
To add, and read documents in Swift, the basics are:
var error: NSError?
let fileManager = NSFileManager.defaultManager()
let documentsDir = fileManager.URLsForDirectory(.DocumentDirectory, inDomains: .UserDomainMask)[0]
as NSURL
let storeURL = documentsDir.URLByAppendingPathComponent("cloudant-sync-datastore")
let path = storeURL.path
// Create Datastore Manager
let manager = CDTDatastoreManager(directory: path, error: &error)
if let err = error {
println("Error creating datastore manager: \(err.localizedDescription)")
}
// Note: data store name must not contain capital letters
let datastore = manager.datastoreNamed("my_datastore", error: &error)
if let err = error {
println("Error creating datastore: \(err.localizedDescription)")
}
// Create a document
var rev = CDTMutableDocumentRevision()
rev.docId = "doc1" // Or don't and get an ID generated for you
var body = [
"description": "Buy milk",
"completed": false,
"type": "com.cloudant.sync.example.task"
] as NSMutableDictionary
rev.setBody(body)
// Save the document to the database
let revision = datastore.createDocumentFromRevision(rev, error: &error)
if let err = error {
println("Error storing document: \(err.localizedDescription)")
}
// Read a document
let docId = revision.docId
let retrieved = datastore.getDocumentWithId(docId, error: &error)
if let err = error {
println("Error retrieving document: \(err.localizedDescription)")
}
Read more in the CRUD document.
You can subscribe for notifications of changes in the database, which is described in the events documentation. It's still a bit raw right now:
- You receive a notification for all new revisions in replication (which can be more than one per updated document).
Replication is used to synchronise data between the local datastore and a remote database, either a CouchDB instance or a Cloudant database. Many datastores can replicate with the same remote database, meaning that cross-device syncronisation is acheived by setting up replications from each device the the remote database.
Replication is simple to get started in the common cases:
#import <CloudantSync.h>
// Create and start the replicator -- -start is essential!
CDTReplicatorFactory *replicatorFactory =
[[CDTReplicatorFactory alloc] initWithDatastoreManager:manager];
NSString *s = @"https://apikey:apipassword@username.cloudant.com/my_database";
NSURL *remoteDatabaseURL = [NSURL URLWithString:s];
CDTDatastore *datastore = [manager datastoreNamed:@"my_datastore"];
// Replicate from the local to remote database
CDTPushReplication *pushReplication = [CDTPushReplication replicationWithSource:datastore
target:remoteDatabaseURL];
NSError *error;
CDTReplicator *replicator = [replicatorFactory oneWay:pushReplication error:&error];
//check error
// Fire-and-forget (there are easy ways to monitor the state too)
[replicator start];
Read more in the replication docs.
Once you have thousands of documents in a database, it's important to have efficient ways of finding them. We've added an easy-to-use querying API. Once the appropriate indexes are set up, querying is as follows:
NSDictionary *query = @{
@"name": @"John", // name equals John
@"age": @{ @"$gt" : @25} // age greater than 25
};
CDTQResultSet *result = [datastore find:query];
[result enumerateObjectsUsingBlock:^(CDTDocumentRevision *rev, NSUInteger idx, BOOL *stop) {
// do something
}];
As of version 0.16.0 the indexing and querying code has been re-written and has more features than the previous implementation. For details about migrating to a 0.16.0+ indexing and query version from a previous version see Index and Querying Migration.
An obvious repercussion of being able to replicate documents about the place is that sometimes you might edit them in more than one place at the same time. When the databases containing these concurrent edits replicate, there needs to be some way to bring these divergent documents back together. Cloudant's MVCC data-model is used to do this.
A document is really a tree of the document and its history. This is neat because it allows us to store multiple versions of a document. In the main, there's a single, linear tree -- just a single branch -- running from the creation of the document to the current revision. It's possible, however, to create further branches in the tree. At this point your document is conflicted and needs some surgury to resolve the conflicts and bring it back to full health.
All requirements are included in the source code or pulled in as dependecies
via pod install
.
See CONTRIBUTORS.
See CONTRIBUTING.
See LICENSE
Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
- MYUtilities is licensed under the BSD licence (portions copied into vendor directory).
- FMDB, by Gus Mueller, is under the MIT License.
- Google Toolbox For Mac is under the Apache License 2.0.