SQLite.swift 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  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 databaseConnection: OpaquePointer = nil
  17. let openResult = path.withCString { sqlite3_open($0, &databaseConnection) }
  18. guard openResult == SQLITE_OK else {
  19. throw SQLiteError.OpenFailed(String(cString: sqlite3_errmsg(databaseConnection)))
  20. }
  21. return SQLite(databaseConnection)
  22. }
  23. private init(_ databaseConnection: OpaquePointer) {
  24. self.databaseConnection = databaseConnection
  25. }
  26. private struct ExecCContext { var callback: ([String: String] -> Void)? }
  27. public func exec(sql: String) throws {
  28. try exec(sql, nil)
  29. }
  30. public func exec(sql: String, _ params: [String?]? = nil, _ step: ([String: String?] -> Void)? = nil) throws {
  31. var statement: OpaquePointer = nil
  32. let prepareResult = sql.withCString { sqlite3_prepare_v2(databaseConnection, $0, Int32(sql.utf8.count), &statement, nil) }
  33. guard prepareResult == SQLITE_OK else {
  34. throw SQLiteError.ExecFailed(String(cString: sqlite3_errmsg(databaseConnection)))
  35. }
  36. for (index, value) in (params ?? [String?]()).enumerated() {
  37. let bindResult = value?.withCString({ sqlite3_bind_text(statement, index + 1, $0, -1 /* take zero terminator. */) { _ in } })
  38. ?? sqlite3_bind_null(statement, index + 1)
  39. guard bindResult == SQLITE_OK else {
  40. throw SQLiteError.BindFailed(String(cString: sqlite3_errmsg(databaseConnection)))
  41. }
  42. }
  43. while true {
  44. let stepResult = sqlite3_step(statement)
  45. switch stepResult {
  46. case SQLITE_ROW:
  47. var content = [String: String?]()
  48. for i in 0..<sqlite3_column_count(statement) {
  49. let name = String(cString: UnsafePointer<CChar>(sqlite3_column_name(statement, i)))
  50. let pointer = sqlite3_column_text(statement, i)
  51. if pointer == nil {
  52. content[name] = nil
  53. } else {
  54. content[name] = String(cString: UnsafePointer<CChar>(pointer))
  55. }
  56. }
  57. step?(content)
  58. case SQLITE_DONE:
  59. return
  60. case SQLITE_ERROR:
  61. throw SQLiteError.ExecFailed("sqlite3_step() returned SQLITE_ERROR.")
  62. default:
  63. throw SQLiteError.ExecFailed("Unknown result for sqlite3_step(): \(stepResult)")
  64. }
  65. }
  66. }
  67. public func enumerate(sql: String) throws -> StatmentSequence {
  68. var statement: OpaquePointer = nil
  69. let prepareResult = sql.withCString { sqlite3_prepare_v2(databaseConnection, $0, Int32(sql.utf8.count), &statement, nil) }
  70. guard prepareResult == SQLITE_OK else {
  71. throw SQLiteError.ExecFailed(String(cString: sqlite3_errmsg(databaseConnection)))
  72. }
  73. return StatmentSequence(statement: statement)
  74. }
  75. public struct StatmentGenerator: IteratorProtocol {
  76. public let statement: OpaquePointer
  77. public func next() -> [String: String]? {
  78. switch sqlite3_step(statement) {
  79. case SQLITE_ROW:
  80. var content = [String: String]()
  81. for i in 0..<sqlite3_column_count(statement) {
  82. let name = String(cString: UnsafePointer<CChar>(sqlite3_column_name(statement, i)))
  83. let pointer = sqlite3_column_text(statement, i)
  84. if pointer == nil {
  85. content[name] = nil
  86. } else {
  87. content[name] = String(cString: UnsafePointer<CChar>(pointer))
  88. }
  89. }
  90. return content
  91. default:
  92. sqlite3_finalize(statement)
  93. return nil
  94. }
  95. }
  96. }
  97. public struct StatmentSequence: Sequence {
  98. public let statement: OpaquePointer
  99. public func makeIterator() -> StatmentGenerator {
  100. return StatmentGenerator(statement: statement)
  101. }
  102. }
  103. public func close() {
  104. sqlite3_close(databaseConnection)
  105. }
  106. }