1
0

SQLite.swift 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. //
  2. // SQLite.swift
  3. // Swifter
  4. //
  5. // Copyright (c) 2014-2016 Damian Kołakowski. All rights reserved.
  6. //
  7. import Foundation
  8. import CSQLite
  9. public enum SQLiteError: Error {
  10. case openFailed(String?)
  11. case execFailed(String?)
  12. case bindFailed(String?)
  13. }
  14. public class SQLite {
  15. private let databaseConnection: OpaquePointer
  16. public static func open(_ path: String) throws -> SQLite {
  17. var databaseConnectionPointer: OpaquePointer? = nil
  18. let openResult = path.withCString { sqlite3_open($0, &databaseConnectionPointer) }
  19. guard let databaseConnection = databaseConnectionPointer else {
  20. throw SQLiteError.execFailed("Invalid pointer.")
  21. }
  22. guard openResult == SQLITE_OK else {
  23. throw SQLiteError.openFailed(String(cString: sqlite3_errmsg(databaseConnection)))
  24. }
  25. return SQLite(databaseConnection)
  26. }
  27. private init(_ databaseConnection: OpaquePointer) {
  28. self.databaseConnection = databaseConnection
  29. }
  30. public func exec(_ sql: String) throws {
  31. try exec(sql, nil)
  32. }
  33. public func exec(_ sql: String, _ params: [String?]? = nil, _ step: (([String: String?]) -> Void)? = nil) throws {
  34. var statementPointer: OpaquePointer? = nil
  35. let prepareResult = sql.withCString { sqlite3_prepare_v2(databaseConnection, $0, Int32(sql.utf8.count), &statementPointer, nil) }
  36. guard prepareResult == SQLITE_OK else {
  37. throw SQLiteError.execFailed(String(cString: sqlite3_errmsg(databaseConnection)))
  38. }
  39. guard let statement = statementPointer else {
  40. throw SQLiteError.execFailed("Invalid pointer.")
  41. }
  42. for (index, value) in (params ?? [String?]()).enumerated() {
  43. let bindResult = value?.withCString({ sqlite3_bind_text(statement, index + 1, $0, -1 /* take zero terminator. */) { _ in } })
  44. ?? sqlite3_bind_null(statement, index + 1)
  45. guard bindResult == SQLITE_OK else {
  46. throw SQLiteError.bindFailed(String(cString: sqlite3_errmsg(databaseConnection)))
  47. }
  48. }
  49. while true {
  50. let stepResult = sqlite3_step(statement)
  51. switch stepResult {
  52. case SQLITE_ROW:
  53. var content = [String: String?]()
  54. for i in 0..<sqlite3_column_count(statement) {
  55. let name = String(cString: UnsafePointer<CChar>(sqlite3_column_name(statement, i)))
  56. if let pointer = sqlite3_column_text(statement, i) {
  57. content[name] = String(cString: UnsafePointer<CChar>(pointer))
  58. } else {
  59. content[name] = nil
  60. }
  61. }
  62. step?(content)
  63. case SQLITE_DONE:
  64. return
  65. case SQLITE_ERROR:
  66. throw SQLiteError.execFailed("sqlite3_step() returned SQLITE_ERROR.")
  67. default:
  68. throw SQLiteError.execFailed("Unknown result for sqlite3_step(): \(stepResult)")
  69. }
  70. }
  71. }
  72. public func enumerate(_ sql: String) throws -> StatmentSequence {
  73. var statement: OpaquePointer? = nil
  74. let prepareResult = sql.withCString { sqlite3_prepare_v2(databaseConnection, $0, Int32(sql.utf8.count), &statement, nil) }
  75. guard prepareResult == SQLITE_OK else {
  76. throw SQLiteError.execFailed(String(cString: sqlite3_errmsg(databaseConnection)))
  77. }
  78. guard let readyStatement = statement else {
  79. throw SQLiteError.execFailed("Invalid pointer.")
  80. }
  81. return StatmentSequence(statement: readyStatement)
  82. }
  83. public struct StatmentGenerator: IteratorProtocol {
  84. public let statement: OpaquePointer
  85. public func next() -> [String: String]? {
  86. switch sqlite3_step(statement) {
  87. case SQLITE_ROW:
  88. var content = [String: String]()
  89. for i in 0..<sqlite3_column_count(statement) {
  90. let name = String(cString: UnsafePointer<CChar>(sqlite3_column_name(statement, i)))
  91. if let pointer = sqlite3_column_text(statement, i) {
  92. content[name] = String(cString: UnsafePointer<CChar>(pointer))
  93. } else {
  94. content[name] = nil
  95. }
  96. }
  97. return content
  98. default:
  99. sqlite3_finalize(statement)
  100. return nil
  101. }
  102. }
  103. }
  104. public struct StatmentSequence: Sequence {
  105. public let statement: OpaquePointer
  106. public func makeIterator() -> StatmentGenerator {
  107. return StatmentGenerator(statement: statement)
  108. }
  109. }
  110. public func close() {
  111. sqlite3_close(databaseConnection)
  112. }
  113. }