SFSQLiteStatement.m [plain text]
/*
* Copyright (c) 2017 Apple Inc. All Rights Reserved.
*
* @APPLE_LICENSE_HEADER_START@
*
* This file contains Original Code and/or Modifications of Original Code
* as defined in and that are subject to the Apple Public Source License
* Version 2.0 (the 'License'). You may not use this file except in
* compliance with the License. Please obtain a copy of the License at
* http://www.opensource.apple.com/apsl/ and read it before using this
* file.
*
* The Original Code and all software distributed under the License are
* distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
* EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
* INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
* Please see the License for the specific language governing rights and
* limitations under the License.
*
* @APPLE_LICENSE_HEADER_END@
*/
#if __OBJC2__
#import <Foundation/NSKeyedArchiver_Private.h>
#import "SFSQLite.h"
#import "SFSQLiteStatement.h"
#import "SFObjCType.h"
#import "debugging.h"
@interface SFSQLiteStatement ()
@property (nonatomic, strong) NSMutableArray *temporaryBoundObjects;
@end
@implementation SFSQLiteStatement
@synthesize SQLite = _SQLite;
@synthesize SQL = _SQL;
@synthesize handle = _handle;
@synthesize reset = _reset;
@synthesize temporaryBoundObjects = _temporaryBoundObjects;
- (id)initWithSQLite:(SFSQLite *)SQLite SQL:(NSString *)SQL handle:(sqlite3_stmt *)handle {
if ((self = [super init])) {
_SQLite = SQLite;
_SQL = SQL;
_handle = handle;
_reset = YES;
}
return self;
}
- (void)finalizeStatement {
if (!_reset) {
secerror("sfsqlite: Statement not reset after last use: \" return;
}
if (sqlite3_finalize(_handle)) {
secerror("sfsqlite: Error finalizing prepared statement: \" return;
}
}
- (void)resetAfterStepError
{
if (!_reset) {
(void)sqlite3_reset(_handle); // we expect this to return an error
(void)sqlite3_clear_bindings(_handle);
[_temporaryBoundObjects removeAllObjects];
_reset = YES;
}
}
- (BOOL)step {
if (_reset) {
_reset = NO;
}
int rc = sqlite3_step(_handle);
if ((rc & 0x00FF) == SQLITE_ROW) {
return YES;
} else if ((rc & 0x00FF) == SQLITE_DONE) {
return NO;
} else {
[self resetAfterStepError];
secerror("sfsqlite: Failed to step ( return NO;
}
}
- (void)reset {
if (!_reset) {
if (sqlite3_reset(_handle)) {
secerror("sfsqlite: Error resetting prepared statement: \" return;
}
if (sqlite3_clear_bindings(_handle)) {
secerror("sfsqlite: Error clearing prepared statement bindings: \" return;
}
[_temporaryBoundObjects removeAllObjects];
_reset = YES;
}
}
- (void)bindInt:(SInt32)value atIndex:(NSUInteger)index {
if (!_reset) {
secerror("sfsqlite: Statement is not reset: \" return;
}
if (sqlite3_bind_int(_handle, (int)index+1, value)) {
secerror("sfsqlite: Error binding int at return;
}
}
- (void)bindInt64:(SInt64)value atIndex:(NSUInteger)index {
if (!_reset) {
secerror("sfsqlite: Statement is not reset: \" return;
}
if (sqlite3_bind_int64(_handle, (int)index+1, value)) {
secerror("sfsqlite: Error binding int64 at return;
}
}
- (void)bindDouble:(double)value atIndex:(NSUInteger)index {
if (!_reset) {
secerror("sfsqlite: Statement is not reset: \" return;
}
if (sqlite3_bind_double(_handle, (int)index+1, value)) {
secerror("sfsqlite: Error binding double at return;
}
}
- (void)bindBlob:(NSData *)value atIndex:(NSUInteger)index {
if (!_reset) {
secerror("sfsqlite: Statement is not reset: \" return;
}
if (value) {
NS_VALID_UNTIL_END_OF_SCOPE NSData *arcSafeValue = value;
if (sqlite3_bind_blob(_handle, (int)index+1, [arcSafeValue bytes], (int)[arcSafeValue length], NULL)) {
secerror("sfsqlite: Error binding blob at return;
}
} else {
[self bindNullAtIndex:index];
}
}
- (void)bindText:(NSString *)value atIndex:(NSUInteger)index {
if (!_reset) {
secerror("sfsqlite: Statement is not reset: \" return;
}
if (value) {
NS_VALID_UNTIL_END_OF_SCOPE NSString *arcSafeValue = value;
if (sqlite3_bind_text(_handle, (int)index+1, [arcSafeValue UTF8String], -1, NULL)) {
secerror("sfsqlite: Error binding text at return;
}
} else {
[self bindNullAtIndex:index];
}
}
- (void)bindNullAtIndex:(NSUInteger)index {
int rc = sqlite3_bind_null(_handle, (int)index+1);
if ((rc & 0x00FF) != SQLITE_OK) {
secerror("sfsqlite: sqlite3_bind_null error");
return;
}
}
- (id)retainedTemporaryBoundObject:(id)object
{
if (!_temporaryBoundObjects) {
_temporaryBoundObjects = [NSMutableArray new];
}
[_temporaryBoundObjects addObject:object];
return object;
}
- (void)bindValue:(id)value atIndex:(NSUInteger)index {
if ([value isKindOfClass:[NSNumber class]]) {
SFObjCType *type = [SFObjCType typeForValue:value];
if (type.isIntegerNumber) {
if (type.size <= 4) {
[self bindInt:[value intValue] atIndex:index];
} else {
[self bindInt64:[value longLongValue] atIndex:index];
}
} else {
NSAssert(type.isFloatingPointNumber, @"Expected number type to be either integer or floating point");
NSAssert(type.code == SFObjCTypeDouble || type.code == SFObjCTypeFloat, @"Unexpected floating point number type: [self bindDouble:[value doubleValue] atIndex:index];
}
} else if ([value isKindOfClass:[NSData class]]) {
[self bindBlob:value atIndex:index];
} else if ([value isKindOfClass:[NSUUID class]]) {
uuid_t uuid;
[(NSUUID *)value getUUIDBytes:uuid];
[self bindBlob:[self retainedTemporaryBoundObject:[NSData dataWithBytes:uuid length:sizeof(uuid_t)]] atIndex:index];
} else if ([value isKindOfClass:[NSString class]]) {
[self bindText:value atIndex:index];
} else if ([value isKindOfClass:[NSNull class]]) {
[self bindNullAtIndex:index];
} else if ([value isKindOfClass:[NSDate class]]) {
[self bindDouble:[(NSDate *)value timeIntervalSinceReferenceDate] atIndex:index];
} else if ([value isKindOfClass:[NSError class]]) {
[self bindBlob:[self retainedTemporaryBoundObject:[NSKeyedArchiver archivedDataWithRootObject:value requiringSecureCoding:YES error:nil]] atIndex:index];
} else if ([value isKindOfClass:[NSURL class]]) {
[self bindText:[self retainedTemporaryBoundObject:[value absoluteString]] atIndex:index];
} else {
secerror("sfsqlite: Can't bind object of type return;
}
}
- (void)bindValues:(NSArray *)values {
for (NSUInteger i = 0; i < values.count; i++) {
[self bindValue:values[i] atIndex:i];
}
}
- (NSUInteger)columnCount {
NSAssert(!_reset, @"Statement is reset: \"
return sqlite3_column_count(_handle);
}
- (int)columnTypeAtIndex:(NSUInteger)index {
NSAssert(!_reset, @"Statement is reset: \"
return sqlite3_column_type(_handle, (int)index);
}
- (NSString *)columnNameAtIndex:(NSUInteger)index {
NSAssert(!_reset, @"Statement is reset: \"
return @(sqlite3_column_name(_handle, (int)index));
}
- (SInt32)intAtIndex:(NSUInteger)index {
NSAssert(!_reset, @"Statement is reset: \"
return sqlite3_column_int(_handle, (int)index);
}
- (SInt64)int64AtIndex:(NSUInteger)index {
NSAssert(!_reset, @"Statement is reset: \"
return sqlite3_column_int64(_handle, (int)index);
}
- (double)doubleAtIndex:(NSUInteger)index {
NSAssert(!_reset, @"Statement is reset: \"
return sqlite3_column_double(_handle, (int)index);
}
- (NSData *)blobAtIndex:(NSUInteger)index {
NSAssert(!_reset, @"Statement is reset: \"
const void *bytes = sqlite3_column_blob(_handle, (int)index);
if (bytes) {
int length = sqlite3_column_bytes(_handle, (int)index);
return [NSData dataWithBytes:bytes length:length];
} else {
return nil;
}
}
- (NSString *)textAtIndex:(NSUInteger)index {
NSAssert(!_reset, @"Statement is reset: \"
const char *text = (const char *)sqlite3_column_text(_handle, (int)index);
if (text) {
return @(text);
} else {
return nil;
}
}
- (id)objectAtIndex:(NSUInteger)index {
int type = [self columnTypeAtIndex:index];
switch (type) {
case SQLITE_INTEGER:
return @([self int64AtIndex:index]);
case SQLITE_FLOAT:
return @([self doubleAtIndex:index]);
case SQLITE_TEXT:
return [self textAtIndex:index];
case SQLITE_BLOB:
return [self blobAtIndex:index];
case SQLITE_NULL:
return nil;
default:
secerror("sfsqlite: Unexpected column type: return nil;
}
}
- (NSArray *)allObjects {
NSUInteger columnCount = [self columnCount];
NSMutableArray *objects = [NSMutableArray arrayWithCapacity:columnCount];
for (NSUInteger i = 0; i < columnCount; i++) {
objects[i] = [self objectAtIndex:i] ?: [NSNull null];
}
return objects;
}
- (NSDictionary *)allObjectsByColumnName {
NSUInteger columnCount = [self columnCount];
NSMutableDictionary *objectsByColumnName = [NSMutableDictionary dictionaryWithCapacity:columnCount];
for (NSUInteger i = 0; i < columnCount; i++) {
NSString *columnName = [self columnNameAtIndex:i];
id object = [self objectAtIndex:i];
if (object) {
objectsByColumnName[columnName] = object;
}
}
return objectsByColumnName;
}
@end
#endif