You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

203 lines
5.6 KiB

  1. //
  2. // PTDatabaseManager.m
  3. // PTDatabaseReader
  4. //
  5. // Created by Peng Tao on 15/11/23.
  6. // Copyright © 2015Peng Tao. All rights reserved.
  7. //
  8. #import "FLEXSQLiteDatabaseManager.h"
  9. #import "FLEXManager.h"
  10. #import <sqlite3.h>
  11. static NSString *const QUERY_TABLENAMES_SQL = @"SELECT name FROM sqlite_master WHERE type='table' ORDER BY name";
  12. @implementation FLEXSQLiteDatabaseManager
  13. {
  14. sqlite3* _db;
  15. NSString* _databasePath;
  16. }
  17. - (instancetype)initWithPath:(NSString*)aPath
  18. {
  19. self = [super init];
  20. if (self) {
  21. _databasePath = [aPath copy];
  22. }
  23. return self;
  24. }
  25. - (BOOL)open {
  26. if (_db) {
  27. return YES;
  28. }
  29. int err = sqlite3_open([_databasePath UTF8String], &_db);
  30. #if SQLITE_HAS_CODEC
  31. NSString *defaultSqliteDatabasePassword = [FLEXManager sharedManager].defaultSqliteDatabasePassword;
  32. if (defaultSqliteDatabasePassword) {
  33. const char *key = defaultSqliteDatabasePassword.UTF8String;
  34. sqlite3_key(_db, key, (int)strlen(key));
  35. }
  36. #endif
  37. if(err != SQLITE_OK) {
  38. NSLog(@"error opening!: %d", err);
  39. return NO;
  40. }
  41. return YES;
  42. }
  43. - (BOOL)close {
  44. if (!_db) {
  45. return YES;
  46. }
  47. int rc;
  48. BOOL retry;
  49. BOOL triedFinalizingOpenStatements = NO;
  50. do {
  51. retry = NO;
  52. rc = sqlite3_close(_db);
  53. if (SQLITE_BUSY == rc || SQLITE_LOCKED == rc) {
  54. if (!triedFinalizingOpenStatements) {
  55. triedFinalizingOpenStatements = YES;
  56. sqlite3_stmt *pStmt;
  57. while ((pStmt = sqlite3_next_stmt(_db, nil)) !=0) {
  58. NSLog(@"Closing leaked statement");
  59. sqlite3_finalize(pStmt);
  60. retry = YES;
  61. }
  62. }
  63. }
  64. else if (SQLITE_OK != rc) {
  65. NSLog(@"error closing!: %d", rc);
  66. }
  67. }
  68. while (retry);
  69. _db = nil;
  70. return YES;
  71. }
  72. - (NSArray<NSDictionary<NSString *, id> *> *)queryAllTables
  73. {
  74. return [self executeQuery:QUERY_TABLENAMES_SQL];
  75. }
  76. - (NSArray<NSString *> *)queryAllColumnsWithTableName:(NSString *)tableName
  77. {
  78. NSString *sql = [NSString stringWithFormat:@"PRAGMA table_info('%@')",tableName];
  79. NSArray<NSDictionary<NSString *, id> *> *resultArray = [self executeQuery:sql];
  80. NSMutableArray<NSString *> *array = [NSMutableArray array];
  81. for (NSDictionary<NSString *, id> *dict in resultArray) {
  82. NSString *columnName = (NSString *)dict[@"name"] ?: @"";
  83. [array addObject:columnName];
  84. }
  85. return array;
  86. }
  87. - (NSArray<NSDictionary<NSString *, id> *> *)queryAllDataWithTableName:(NSString *)tableName
  88. {
  89. NSString *sql = [NSString stringWithFormat:@"SELECT * FROM %@",tableName];
  90. return [self executeQuery:sql];
  91. }
  92. #pragma mark -
  93. #pragma mark - Private
  94. - (NSArray<NSDictionary<NSString *, id> *> *)executeQuery:(NSString *)sql
  95. {
  96. [self open];
  97. NSMutableArray<NSDictionary<NSString *, id> *> *resultArray = [NSMutableArray array];
  98. sqlite3_stmt *pstmt;
  99. if (sqlite3_prepare_v2(_db, [sql UTF8String], -1, &pstmt, 0) == SQLITE_OK) {
  100. while (sqlite3_step(pstmt) == SQLITE_ROW) {
  101. NSUInteger num_cols = (NSUInteger)sqlite3_data_count(pstmt);
  102. if (num_cols > 0) {
  103. NSMutableDictionary<NSString *, id> *dict = [NSMutableDictionary dictionaryWithCapacity:num_cols];
  104. int columnCount = sqlite3_column_count(pstmt);
  105. int columnIdx = 0;
  106. for (columnIdx = 0; columnIdx < columnCount; columnIdx++) {
  107. NSString *columnName = [NSString stringWithUTF8String:sqlite3_column_name(pstmt, columnIdx)];
  108. id objectValue = [self objectForColumnIndex:columnIdx stmt:pstmt];
  109. [dict setObject:objectValue forKey:columnName];
  110. }
  111. [resultArray addObject:dict];
  112. }
  113. }
  114. }
  115. [self close];
  116. return resultArray;
  117. }
  118. - (id)objectForColumnIndex:(int)columnIdx stmt:(sqlite3_stmt*)stmt {
  119. int columnType = sqlite3_column_type(stmt, columnIdx);
  120. id returnValue = nil;
  121. if (columnType == SQLITE_INTEGER) {
  122. returnValue = [NSNumber numberWithLongLong:sqlite3_column_int64(stmt, columnIdx)];
  123. }
  124. else if (columnType == SQLITE_FLOAT) {
  125. returnValue = [NSNumber numberWithDouble:sqlite3_column_double(stmt, columnIdx)];
  126. }
  127. else if (columnType == SQLITE_BLOB) {
  128. returnValue = [self dataForColumnIndex:columnIdx stmt:stmt];
  129. }
  130. else {
  131. //default to a string for everything else
  132. returnValue = [self stringForColumnIndex:columnIdx stmt:stmt];
  133. }
  134. if (returnValue == nil) {
  135. returnValue = [NSNull null];
  136. }
  137. return returnValue;
  138. }
  139. - (NSString *)stringForColumnIndex:(int)columnIdx stmt:(sqlite3_stmt *)stmt {
  140. if (sqlite3_column_type(stmt, columnIdx) == SQLITE_NULL || (columnIdx < 0)) {
  141. return nil;
  142. }
  143. const char *c = (const char *)sqlite3_column_text(stmt, columnIdx);
  144. if (!c) {
  145. // null row.
  146. return nil;
  147. }
  148. return [NSString stringWithUTF8String:c];
  149. }
  150. - (NSData *)dataForColumnIndex:(int)columnIdx stmt:(sqlite3_stmt *)stmt{
  151. if (sqlite3_column_type(stmt, columnIdx) == SQLITE_NULL || (columnIdx < 0)) {
  152. return nil;
  153. }
  154. const char *dataBuffer = sqlite3_column_blob(stmt, columnIdx);
  155. int dataSize = sqlite3_column_bytes(stmt, columnIdx);
  156. if (dataBuffer == NULL) {
  157. return nil;
  158. }
  159. return [NSData dataWithBytes:(const void *)dataBuffer length:(NSUInteger)dataSize];
  160. }
  161. @end