mirror of
https://github.com/samuelclay/NewsBlur.git
synced 2025-08-05 16:58:59 +00:00
244 lines
5.7 KiB
Objective-C
Executable file
244 lines
5.7 KiB
Objective-C
Executable file
//
|
|
// FMDatabasePool.m
|
|
// fmdb
|
|
//
|
|
// Created by August Mueller on 6/22/11.
|
|
// Copyright 2011 Flying Meat Inc. All rights reserved.
|
|
//
|
|
|
|
#import "FMDatabasePool.h"
|
|
#import "FMDatabase.h"
|
|
|
|
@interface FMDatabasePool()
|
|
|
|
- (void)pushDatabaseBackInPool:(FMDatabase*)db;
|
|
- (FMDatabase*)db;
|
|
|
|
@end
|
|
|
|
|
|
@implementation FMDatabasePool
|
|
@synthesize path=_path;
|
|
@synthesize delegate=_delegate;
|
|
@synthesize maximumNumberOfDatabasesToCreate=_maximumNumberOfDatabasesToCreate;
|
|
|
|
|
|
+ (id)databasePoolWithPath:(NSString*)aPath {
|
|
return FMDBReturnAutoreleased([[self alloc] initWithPath:aPath]);
|
|
}
|
|
|
|
- (id)initWithPath:(NSString*)aPath {
|
|
|
|
self = [super init];
|
|
|
|
if (self != nil) {
|
|
_path = [aPath copy];
|
|
_lockQueue = dispatch_queue_create([[NSString stringWithFormat:@"fmdb.%@", self] UTF8String], NULL);
|
|
_databaseInPool = FMDBReturnRetained([NSMutableArray array]);
|
|
_databaseOutPool = FMDBReturnRetained([NSMutableArray array]);
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void)dealloc {
|
|
|
|
_delegate = 0x00;
|
|
FMDBRelease(_path);
|
|
FMDBRelease(_databaseInPool);
|
|
FMDBRelease(_databaseOutPool);
|
|
|
|
if (_lockQueue) {
|
|
FMDBDispatchQueueRelease(_lockQueue);
|
|
_lockQueue = 0x00;
|
|
}
|
|
#if ! __has_feature(objc_arc)
|
|
[super dealloc];
|
|
#endif
|
|
}
|
|
|
|
|
|
- (void)executeLocked:(void (^)(void))aBlock {
|
|
dispatch_sync(_lockQueue, aBlock);
|
|
}
|
|
|
|
- (void)pushDatabaseBackInPool:(FMDatabase*)db {
|
|
|
|
if (!db) { // db can be null if we set an upper bound on the # of databases to create.
|
|
return;
|
|
}
|
|
|
|
[self executeLocked:^() {
|
|
|
|
if ([_databaseInPool containsObject:db]) {
|
|
[[NSException exceptionWithName:@"Database already in pool" reason:@"The FMDatabase being put back into the pool is already present in the pool" userInfo:nil] raise];
|
|
}
|
|
|
|
[_databaseInPool addObject:db];
|
|
[_databaseOutPool removeObject:db];
|
|
|
|
}];
|
|
}
|
|
|
|
- (FMDatabase*)db {
|
|
|
|
__block FMDatabase *db;
|
|
|
|
[self executeLocked:^() {
|
|
db = [_databaseInPool lastObject];
|
|
|
|
if (db) {
|
|
[_databaseOutPool addObject:db];
|
|
[_databaseInPool removeLastObject];
|
|
}
|
|
else {
|
|
|
|
if (_maximumNumberOfDatabasesToCreate) {
|
|
NSUInteger currentCount = [_databaseOutPool count] + [_databaseInPool count];
|
|
|
|
if (currentCount >= _maximumNumberOfDatabasesToCreate) {
|
|
NSLog(@"Maximum number of databases (%ld) has already been reached!", (long)currentCount);
|
|
return;
|
|
}
|
|
}
|
|
|
|
db = [FMDatabase databaseWithPath:_path];
|
|
}
|
|
|
|
//This ensures that the db is opened before returning
|
|
if ([db open]) {
|
|
if ([_delegate respondsToSelector:@selector(databasePool:shouldAddDatabaseToPool:)] && ![_delegate databasePool:self shouldAddDatabaseToPool:db]) {
|
|
[db close];
|
|
db = 0x00;
|
|
}
|
|
else {
|
|
//It should not get added in the pool twice if lastObject was found
|
|
if (![_databaseOutPool containsObject:db]) {
|
|
[_databaseOutPool addObject:db];
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
NSLog(@"Could not open up the database at path %@", _path);
|
|
db = 0x00;
|
|
}
|
|
}];
|
|
|
|
return db;
|
|
}
|
|
|
|
- (NSUInteger)countOfCheckedInDatabases {
|
|
|
|
__block NSUInteger count;
|
|
|
|
[self executeLocked:^() {
|
|
count = [_databaseInPool count];
|
|
}];
|
|
|
|
return count;
|
|
}
|
|
|
|
- (NSUInteger)countOfCheckedOutDatabases {
|
|
|
|
__block NSUInteger count;
|
|
|
|
[self executeLocked:^() {
|
|
count = [_databaseOutPool count];
|
|
}];
|
|
|
|
return count;
|
|
}
|
|
|
|
- (NSUInteger)countOfOpenDatabases {
|
|
__block NSUInteger count;
|
|
|
|
[self executeLocked:^() {
|
|
count = [_databaseOutPool count] + [_databaseInPool count];
|
|
}];
|
|
|
|
return count;
|
|
}
|
|
|
|
- (void)releaseAllDatabases {
|
|
[self executeLocked:^() {
|
|
[_databaseOutPool removeAllObjects];
|
|
[_databaseInPool removeAllObjects];
|
|
}];
|
|
}
|
|
|
|
- (void)inDatabase:(void (^)(FMDatabase *db))block {
|
|
|
|
FMDatabase *db = [self db];
|
|
|
|
block(db);
|
|
|
|
[self pushDatabaseBackInPool:db];
|
|
}
|
|
|
|
- (void)beginTransaction:(BOOL)useDeferred withBlock:(void (^)(FMDatabase *db, BOOL *rollback))block {
|
|
|
|
BOOL shouldRollback = NO;
|
|
|
|
FMDatabase *db = [self db];
|
|
|
|
if (useDeferred) {
|
|
[db beginDeferredTransaction];
|
|
}
|
|
else {
|
|
[db beginTransaction];
|
|
}
|
|
|
|
|
|
block(db, &shouldRollback);
|
|
|
|
if (shouldRollback) {
|
|
[db rollback];
|
|
}
|
|
else {
|
|
[db commit];
|
|
}
|
|
|
|
[self pushDatabaseBackInPool:db];
|
|
}
|
|
|
|
- (void)inDeferredTransaction:(void (^)(FMDatabase *db, BOOL *rollback))block {
|
|
[self beginTransaction:YES withBlock:block];
|
|
}
|
|
|
|
- (void)inTransaction:(void (^)(FMDatabase *db, BOOL *rollback))block {
|
|
[self beginTransaction:NO withBlock:block];
|
|
}
|
|
#if SQLITE_VERSION_NUMBER >= 3007000
|
|
- (NSError*)inSavePoint:(void (^)(FMDatabase *db, BOOL *rollback))block {
|
|
|
|
static unsigned long savePointIdx = 0;
|
|
|
|
NSString *name = [NSString stringWithFormat:@"savePoint%ld", savePointIdx++];
|
|
|
|
BOOL shouldRollback = NO;
|
|
|
|
FMDatabase *db = [self db];
|
|
|
|
NSError *err = 0x00;
|
|
|
|
if (![db startSavePointWithName:name error:&err]) {
|
|
[self pushDatabaseBackInPool:db];
|
|
return err;
|
|
}
|
|
|
|
block(db, &shouldRollback);
|
|
|
|
if (shouldRollback) {
|
|
[db rollbackToSavePointWithName:name error:&err];
|
|
}
|
|
else {
|
|
[db releaseSavePointWithName:name error:&err];
|
|
}
|
|
|
|
[self pushDatabaseBackInPool:db];
|
|
|
|
return err;
|
|
}
|
|
#endif
|
|
|
|
@end
|