An alternative to Core Data for people who like having direct SQL access.
By Marco Arment. See the LICENSE file for license info (it's the MIT license).
FCModel is a generic model layer on top of SQLite. It's intended for people who want some of Core Data's convenience, but with more control over implementation, performance, database schemas, queries, indexes, and migrations, and the ability to use raw SQL queries and SQLite features directly.
FCModel accomplishes a lot of what Brent Simmons wrote about. This is my version of that. (Are you reading objc.io yet? You should be. It's excellent.)
This is a beta. I'm building an app around it, a few others are using it and making contributions, and it's stable for us so far. But changes are still relatively frequent, and the API may still change in minor but backward-incompatible ways.
- Xcode 5 or later
- Deployment on iOS 6 or above, or Mac OS X 10.8 or above (it requires
NSMapTable
) - ARC only
- FMDB, Gus Mueller's excellent Objective-C SQLite wrapper (automatic if you use CocoaPods)
- Linking your project with
sqlite3
(automatic if you use CocoaPods)
There isn't much right now. Check out the FCModel.h
header and the example project. I'll add more here as I get the chance.
SQLite tables are associated with FCModel subclasses of the same name, and database columns map to @property
declarations with the same name. So you could have a table like this:
CREATE TABLE Person (
id INTEGER PRIMARY KEY,
name TEXT NOT NULL DEFAULT '',
createdTime REAL NOT NULL
);
CREATE INDEX IF NOT EXISTS name ON Person (name);
A single-column primary key is required. It can be an integer or a string. If you don't specify a key value upon object creation, FCModel will generate a random 64-bit signed integer key that's unique within the table. You're responsible for creating your own table indexes.
This table's model would look like this:
#import "FCModel.h"
@interface Person : FCModel
@property (nonatomic) int64_t id;
@property (nonatomic, copy) NSString *name;
@property (nonatomic) NSDate *createdTime;
@end
Database-mapped object properties can be:
- Primitives (
int
,double
,BOOL
,int64_t
, etc.) orNSNumber
, limited to SQLite's precision (64-bit signed for integers). NSString
, which is always stored and loaded as UTF-8NSData
forBLOB
columnsNSDate
, which is converted to/fromNSTimeInterval
since 1970 (signeddouble
versions of Unix timestamps) for storage. DeclareNSDate
columns asREAL
in the table.NSURL
, which is converted to/from itsabsoluteString
representation for storage.NSDictionary
orNSArray
, which are converted to/from binary plists for storage (so each contained object must be anNSData
,NSString
,NSArray
,NSDictionary
,NSDate
, orNSNumber
).
To override this behavior or customize it for other types, models may override the methods below. Database values may be NSString
or NSNumber
for INTEGER
/FLOAT
/TEXT
columns, or NSData
for BLOB
columns. For columns that permit NULL
, these methods may receive or return nil
. Overrides must call the super
implementation to convert values that they're not handling.
- (id)serializedDatabaseRepresentationOfValue:(id)instanceValue forPropertyNamed:(NSString *)propertyName;
- (id)unserializedRepresentationOfDatabaseValue:(id)databaseValue forPropertyNamed:(NSString *)propertyName;
You can name your column-property ivars whatever you like. FCModel associates columns with property names, not ivar names.
Models may have properties that have no corresponding database columns. But if any columns in a model's table don't have corresponding properties, FCModel logs a notice to the console at launch.
In your application:didFinishLaunchingWithOptions:
method, before any models are accessed, call FCModel's openDatabaseAtPath:withSchemaBuilder:
. This looks a bit crazy, but bear with me — it's conceptually very simple.
Your schema-builder block is passed int *schemaVersion
, which is an in-out argument:
- FCModel tells you the current schema on the way in (on an empty database, this starts at 0).
- You execute any schema-creation or migration statements to get to the next schema version.
- You update
*schemaVersion
to reflect the new version.
Here's an example from that Person class described above:
NSString *documentsPath = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) objectAtIndex:0];
NSString *dbPath = [documentsPath stringByAppendingPathComponent:@"testDB.sqlite3"];
[FCModel openDatabaseAtPath:dbPath withSchemaBuilder:^(FMDatabase *db, int *schemaVersion) {
[db beginTransaction];
// My custom failure handling. Yours may vary.
void (^failedAt)(int statement) = ^(int statement){
int lastErrorCode = db.lastErrorCode;
NSString *lastErrorMessage = db.lastErrorMessage;
[db rollback];
NSAssert3(0, @"Migration statement %d failed, code %d: %@", statement, lastErrorCode, lastErrorMessage);
};
if (*schemaVersion < 1) {
if (! [db executeUpdate:
@"CREATE TABLE Person ("
@" id INTEGER PRIMARY KEY,"
@" name TEXT NOT NULL DEFAULT '',"
@" createdTime REAL NOT NULL"
@");"
]) failedAt(1);
if (! [db executeUpdate:@"CREATE INDEX IF NOT EXISTS name ON Person (name);"]) failedAt(2);
*schemaVersion = 1;
}
// If you wanted to change the schema in a later app version, you'd add something like this here:
/*
if (*schemaVersion < 2) {
if (! [db executeUpdate:@"ALTER TABLE Person ADD COLUMN title TEXT NOT NULL DEFAULT ''"]) failedAt(3);
*schemaVersion = 2;
}
// And so on...
if (*schemaVersion < 3) {
if (! [db executeUpdate:@"CREATE TABLE..."]) failedAt(4);
*schemaVersion = 3;
}
*/
[db commit];
}];
Once you've shipped a version to customers, never change its construction in your code. That way, on an initial launch of a new version, your schema-builder will see that the customer's existing database is at e.g. schema version 2, and you can execute only what's required to bring it up to version 3.
Creating new instances (INSERTs):
// If you want a random 64-bit signed integer primary key value for .id:
Person *bob = [Person new];
// If you want to specify your own .id value:
Person *bob = [Person instanceWithPrimaryKey:@123];
bob.name = @"Bob";
bob.createdTime = [NSDate date];
[bob save];
SELECT and UPDATE queries should look familiar to FMDB fans: everything's parameterized with ?
placeholders and varargs query functions, and it's passed right through to FMDB. Just as with FMDB, you need to box primitives when passing them as query params, e.g. @1
instead of 1
.
// Find that specific Bob by ID
Person *bob = [Person instanceWithPrimaryKey:@123];
bob.name = @"Robert";
[bob save];
// Or find the first person named Bob
Person *firstBob = [Person firstInstanceWhere:@"name = ? ORDER BY id LIMIT 1", @"Bob"];
// Find all Bobs
NSArray *allBobs = [Person instancesWhere:@"name = ?", @"Bob"];
You can use two shortcuts in queries:
$T
: The model's table name. (e.g. "Person")$PK
: The model's primary-key column name. (e.g. "id")
Now here's where it gets crazy. Suppose you wanted to rename all Bobs to Robert, or delete all people named Sue, without loading them all and doing a million queries. (Hi, Core Data.)
// Suppose these are hanging out here, being retained somewhere (in the UI, maybe)
Person *bob = [Person instanceWithPrimaryKey:@123];
Person *sue = [Person firstInstanceWhere:@"name = 'Sue'"]; // you don't HAVE to parameterize everything
// ...
[Person executeUpdateQuery:@"UPDATE $T SET name = ? WHERE name = ?", @"Robert", @"Bob"];
NSLog(@"This Bob's name is now %@.", bob.name);
// prints: This Bob's name is now Robert.
[Person executeUpdateQuery:@"DELETE FROM $T WHERE name = 'Sue'"];
NSLog(@"Sue is %@.", sue.deleted ? @"deleted" : @"around");
// prints: Sue is deleted.
It works. (Or at least, it should. Please let me know if it doesn't.)
FCModel is not designed to handle this automatically. You're meant to write this from each model's implementation as appropriate. This gives you complete control over schema, index usage, automatic fetching queries (or not), and caching.
If you want automatic relationship mappings, consider using Core Data. It's very good at that.
Each FCModel instance is exclusive in memory by its table and primary-key value. If you load Person ID 1, then some other query loads Person ID 1, they'll be the same instance (unless the first one got deallocated in the meantime).
FCModels are safe to retain for a while, even by the UI. You can use KVO to observe changes. Just check instances' deleted
property where relevant, and watch for change notifications.
FCModels are inherently cached by primary key:
NSArray *allBobs = [Person instancesWhere:@"name = ?", @"Bob"];
// executes query: SELECT * FROM Person WHERE name = 'Bob'
Person *bob = [Person instanceWithPrimaryKey:@123];
// cache hit, no query executed
...but only among what's retained in your app. If you want to cache an entire table, for instance, you'll want to do something like retain its allInstances
array somewhere long-lived (such as the app delegate).
FCModels can be accessed and modified from any thread (again, as far as I know), but all database operations are run synchronously on a serial queue, so you're not likely to see any performance gains by concurrent access.
FCModel's public notifications (FCModelInsertNotification
, etc.) are always posted on the main thread.
For now, it's just right here on GitHub.
...are welcome, with the following guideline:
More than anything else, I'd like to keep FCModel small, simple, and easy to fit in your mental L2 cache.