Everything should be made as simple as possible, but no simpler.
—Albert Einstein
The Skull Swift package offers a bare bones (400 LOC) interface for SQLite. Emphasising simplicity, its synchronous API implements a minimal set of functions for interacting with SQLite.
import Foundation
import Skull
let skull: DispatchQueue = DispatchQueue(label: "ink.codes.skull")
let db = skull.sync {
return try! Skull()
}
skull.async {
let sql = "create table planets (id integer primary key, au double, name text);"
try! db.exec(sql)
}
skull.async {
let sql = "insert into planets values (?, ?, ?);"
try! db.update(sql, 0, 0.4, "Mercury")
try! db.update(sql, 1, 0.7, "Venus")
try! db.update(sql, 2, 1, "Earth")
try! db.update(sql, 3, 1.5, "Mars")
}
skull.sync {
let sql = "select name from planets where au=1;"
try! db.query(sql) { er, row in
assert(er == nil)
let name = row?["name"] as! String
assert(name == "Earth")
print(name)
return 0
}
}Skull is deliberately thin, its tiny API leaves access serialization to users. Leveraging a dedicated serial queue, as shown in the example above, intuitively ensures serialized access.
enum SkullError: ErrorSkullError enumerates explicit errors.
alreadyOpen(String)failedToFinalize(Array<Error>)invalidURLnotOpensqliteError(Int, String)sqliteMessage(String)unsupportedType
typealias SkullRow = Dictionary<String, Any>SkullRow models a row within a SQLite table. Being a Dictionary, it offers subscript access to column values, which can be of three essential types:
StringIntDouble
For example:
row["a"] as String == "500.0"
row["b"] as Int == 500
row["c"] as Double == 500.0class Skull: SQLDatabaseSkull, the main object of this module, represents a SQLite database connection. It adopts the SQLDatabase protocol, which defines its interface:
protocol SQLDatabase {
var url: URL? { get }
func flush() throws
func exec(_ sql: String, cb: @escaping (SkullError?, [String : String]) -> Int) throws
func query(_ sql: String, cb: (SkullError?, SkullRow?) -> Int) throws
func update(_ sql: String, _ params: Any?...) throws
}To open a database connection you initialize a new Skull object.
init(_ url: URL? = nil) throwsurlThe location of the database file to open.
Opens the database located at file url. If the file does not exist, it is created. Skipping url or passing nil opens an in-memory database.
A Skull object, representing a database connection, offers following methods for accessing the database.
func exec(sql: String, cb: ((SkullError?, [String:String]) -> Int)?)) throwssqlZero or more UTF-8 encoded, semicolon-separated SQL statements.cbA callback to handle results or abort by returning non-zero.
Executes SQL statements and applies the callback for each result, limited to strings in this case. The callback is optional, if provided, it can abort execution by returning non-zero. The callback doesn't just handle results, it can also monitor execution and, if need be, abort the operation; it is applied zero or more times.
func query(sql: String, cb: (SkullError?, SkullRow?) -> Int) throwssqlThe SQL statement to query the database with.cbThe callback to handle resuting errors and rows.
Queries the database with the specified selective SQL statement and applies the callback for each resulting row or occuring error.
func update(sql: String, params: Any?...) throwssqlThe SQL statement to apply.paramsThe parameters to bind to the statement.
Updates the database by binding the specified parameters to an SQLite statement, for example:
let sql = "insert into planets values (?, ?, ?);"
try! db.update(sql, 0, 0.4, "Mercury")This method may throw SkullError.sqliteError(Int, String) or SkullError.unsupportedType.
func flush() throwsRemoves and finalizes all cached prepared statements.
var url: URL? { get }The location of the database file.
Close the database by simply dismissing the Skull object.
📦 Add https://github.com/michaelnisi/skull to your package manifest.