SQLite.swift 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. //
  2. // SQLite.swift
  3. // Swifter
  4. //
  5. // Copyright (c) 2014-2016 Damian Kołakowski. All rights reserved.
  6. //
  7. import Foundation
  8. public enum SQLiteError: ErrorProtocol {
  9. case OpenFailed(String?)
  10. case ExecFailed(String?)
  11. case BindFailed(String?)
  12. }
  13. public class SQLite {
  14. private let databaseConnection: OpaquePointer
  15. public static func open(_ path: String) throws -> SQLite {
  16. var databaseConnectionPointer: OpaquePointer? = nil
  17. let openResult = path.withCString { sqlite3_open($0, &databaseConnectionPointer) }
  18. guard let databaseConnection = databaseConnectionPointer else {
  19. throw SQLiteError.ExecFailed("Invalid pointer.")
  20. }
  21. guard openResult == SQLITE_OK else {
  22. throw SQLiteError.OpenFailed(String(cString: sqlite3_errmsg(databaseConnection)))
  23. }
  24. return SQLite(databaseConnection)
  25. }
  26. private init(_ databaseConnection: OpaquePointer) {
  27. self.databaseConnection = databaseConnection
  28. }
  29. private struct ExecCContext { var callback: ([String: String] -> Void)? }
  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. let pointer = sqlite3_column_text(statement, i)
  57. if pointer == nil {
  58. content[name] = nil
  59. } else {
  60. content[name] = String(cString: UnsafePointer<CChar>(pointer))
  61. }
  62. }
  63. step?(content)
  64. case SQLITE_DONE:
  65. return
  66. case SQLITE_ERROR:
  67. throw SQLiteError.ExecFailed("sqlite3_step() returned SQLITE_ERROR.")
  68. default:
  69. throw SQLiteError.ExecFailed("Unknown result for sqlite3_step(): \(stepResult)")
  70. }
  71. }
  72. }
  73. public func enumerate(_ sql: String) throws -> StatmentSequence {
  74. var statement: OpaquePointer? = nil
  75. let prepareResult = sql.withCString { sqlite3_prepare_v2(databaseConnection, $0, Int32(sql.utf8.count), &statement, nil) }
  76. guard prepareResult == SQLITE_OK else {
  77. throw SQLiteError.ExecFailed(String(cString: sqlite3_errmsg(databaseConnection)))
  78. }
  79. guard let readyStatement = statement else {
  80. throw SQLiteError.ExecFailed("Invalid pointer.")
  81. }
  82. return StatmentSequence(statement: readyStatement)
  83. }
  84. public struct StatmentGenerator: IteratorProtocol {
  85. public let statement: OpaquePointer
  86. public func next() -> [String: String]? {
  87. switch sqlite3_step(statement) {
  88. case SQLITE_ROW:
  89. var content = [String: String]()
  90. for i in 0..<sqlite3_column_count(statement) {
  91. let name = String(cString: UnsafePointer<CChar>(sqlite3_column_name(statement, i)))
  92. let pointer = sqlite3_column_text(statement, i)
  93. if pointer == nil {
  94. content[name] = nil
  95. } else {
  96. content[name] = String(cString: UnsafePointer<CChar>(pointer))
  97. }
  98. }
  99. return content
  100. default:
  101. sqlite3_finalize(statement)
  102. return nil
  103. }
  104. }
  105. }
  106. public struct StatmentSequence: Sequence {
  107. public let statement: OpaquePointer
  108. public func makeIterator() -> StatmentGenerator {
  109. return StatmentGenerator(statement: statement)
  110. }
  111. }
  112. public func close() {
  113. sqlite3_close(databaseConnection)
  114. }
  115. }