Type-safe SQL in Swift
SwiftQL
Swift Query Language allows you to write type-safe relational database queries using familiar Swift syntax.
The primary goal of SwiftQL is to allow SQL queries to be written in a type-safe manner, so that SQL queries can be verified at compile time, to reduce runtime errors and security issues caused by common mistakes such as incorrect syntax, or improperly escaped input.
SwiftQL uses SQLite and as such attempts to adhere to SQL as understood by SQLite within reason.
SwiftQL is currently only a proof of concept and is not ready for use in production systems.
If you are intereted in using SwiftQL in your project please consider contributing or donating.
Tutorial
SwiftQL works with relational databases. Objects in SwiftQL are derived from tables.
To use an object in SwiftQL, you first need to define meta-data which describes how each object is stored in the database.
At present this definition needs to be implemented by hand.
A future goal of SwiftQL is to derive the schema automatically from Swift objects.
First define the types you would like to use in your database.
Value types are recommended, although reference types can also be used.
SwiftQL currently does not support enums or protocols.
struct Sample: Identifiable, Equatable {
let id: String
let value: Int
}
Next we create a custom class conforming to the Database
protocol, which describes the schema or layout of the database.
Implement the Schema
associated type, which extends from the DatabaseSchema
.
class MyDatabase: Database {
final class Schema: DatabaseSchema {
Within the schema class we need to define a class which describes the schema for each table in the database. In this example we define a schema for the Samples
table.
We need to define a property on the class for each field in the table. Our Sample table has an id
primary key field which is a string, and value
field which is an integer.
SwiftQL supports the following types for fields:
Boolean
: Stored as 64-bit integer, where false
is stored as 0
and true
is stored as 1
.
Int
: Stored as 64-bit integer.
Double
: Stored as IEEE double precision floating point.
String
: Stored as UTF-8 encoded null terminated string.
Data
: Stored as raw bytes.
URL
: Stored as text using the .absoluteString
representation.
UUID
: Stored as text using the .uuidString
representation.
Date
: Stored as text using ISO8601
representation
Each field requires the name of the column in the table.
We also need to define the tableName
which is the name used in SQL queries.
tableFields
should return an array containing all of the fields for the table. This is used by SwiftQL to process some queries, such as table create
statements.
The entity(from:)
method should return an instance of the entity stored in the table, deserialized from the SQLRow
provided in the parameter. This method is used by SwiftQL to instantiate the entity when returning a result from a select query.
Finally, the values(entity:)
method should return a dictionary that represents an instance of the Swift entity in a form that is understood by SQL.
final class SampleSchema: BaseTableSchema, TableSchema {
lazy var id = PrimaryKeyField<String>(name: "id", table: self)
lazy var value = Field<Int>(name: "value", table: self)
static let tableName = SQLIdentifier(stringLiteral: "samples")
var tableFields: [AnyField] {
return [id, value]
}
func entity(from row: SQLRow) -> Sample {
Sample(
id: row.field(id),
value: row.field(value)
)
}
func values(entity: Sample) -> [SQLIdentifier : SQLExpression] {
[
id.name: StringLiteral(entity.id),
value.name: IntegerLiteral(entity.value)
]
}
}
Lastly we need to provide a variable to access the schema, and a constructor to instantiate the database.
func samples() -> SampleSchema {
schema(table: SampleSchema.self)
}
}
let connection: SQLite.Connection
init(connection: SQLite.Connection) {
self.connection = connection
}
}
}
Once we have the database definition in place we can instantiate the database and start to use it. The code below opens the database file named mydatabase.sqlite3
in the cache directory.
let directory = FileManager.default.urls(for: .cachesDirectory, in: .userDomainMask).first!
let fileURL = directory.appendingPathComponent("mydatabase").appendingPathExtension("sqlite3")
let resource = SQLite.Resource(fileURL: fileURL)
let connection = try resource.connect()
let database = MyDatabase(connection: connection)
Create each table:
try database.execute { db in
// CREATE TABLE IF NOT EXISTS `samples`
// `id` TEXT PRIMARY KEY NOT NULL,
// `value` INT NOT NULL
Create(db.samples())
}
Insert data into the database:
try database.execute { db in
// INSERT INTO `samples`
// ( `id`, `value` )
// VALUES ( ?1, ?2 )"
let sample = db.samples()
Insert(sample, Sample(id: "foo", value: 7))
}
Fetch data from the database
try database.execute { db in
// SELECT `t0`.`id`, `t0`.`value`
// FROM `samples` AS `t0`
let sample = db.samples()
Select(sample)
From(sample)
}
Note in the above examples that we used a variable sample
to refer to the schema. This is used when we need to refer to the same table multiple times in a query. In queries where a table is used only once, such as in the insert
example, the table instance can be used directly without assigning it to a variable.
TODO:
Below are some of the goals for SwiftQL, roughly in order of highest priority first.
- Observable live updating select query results.
- Define table schemas using the Create statement instead of as metadata classes.
- SQL GROUP BY syntax.
- Handle SQLITE_BUSY errors.
- Allow only read (select) statements to be observed.
- Optimize observable statements: Only re-query when the tables and fields mentioned in the query are changed.
- Bulk import from CSV and data frames.
- Automatic schema migration when opening a populated database which has an outdated schema.
- Migrate documentation to Swift-DocC.
- Interface to other databases including Postgres and MySQL.
License:
SwiftQL is free for commercial and non-commercial use.
Attribution to the original source is required.