- Associations Benefits
- Required Protocols
- The Types of Associations
- Associations and the Database Schema
- Building Requests from Associations
- Fetching Values from Associations
- Association Aggregates
- DerivableRequest Protocol
- How do I filter records and only keep those that are associated to another record?
- How do I filter records and only keep those that are NOT associated to another record?
- How do I select only one column of an associated record?
An association is a connection between two Record types.
Associations streamline common operations in your code, make them safer, and more efficient. For example, consider a library application that has two record types, author and book:
struct Author {
var id: Int64
var name: String
}
struct Book {
var id: Int64
var authorId: Int64?
var title: String
}
Now, suppose we wanted to load all books from an existing author. We'd need to do something like this:
let author: Author = ...
let books = try Book
.filter(Column("authorId") == author.id)
.fetchAll(db)
Or, loading all pairs of books along with their authors:
struct BookInfo {
var book: Book
var author: Author?
}
let books = try Book.fetchAll(db)
let bookInfos = books.map { book -> BookInfo in
let author = try Author.fetchOne(db, id: book.authorId)
return BookInfo(book: book, author: author)
}
With GRDB associations, we can streamline these operations (and others), by declaring the connections between books and authors. Here is how we define associations, and properties that access them:
extension Author {
static let books = hasMany(Book.self)
var books: QueryInterfaceRequest<Book> {
request(for: Author.books)
}
}
extension Book {
static let author = belongsTo(Author.self)
var author: QueryInterfaceRequest<Author> {
request(for: Book.author)
}
}
Loading all books from an existing author is now easier:
let books = try author.books.fetchAll(db)
As for loading all pairs of books and authors, it is not only easier, but also much more efficient:
struct BookInfo: FetchableRecord, Decodable {
let book: Book
let author: Author?
}
let request = Book.including(optional: Book.author)
let bookInfos = BookInfo.fetchAll(db, request)
Before we dive in, please remember that associations can not generate all possible SQL queries that involve several tables. You may also prefer writing SQL, and this is just OK, because your SQL skills are welcome. The splittingRowAdapters(columnCounts:)
method can help you consume the rows fetched from joined queries, as in SELECT book.*, author.* FROM ...
.
Associations are available on types that adopt the necessary supporting protocols.
Associations are based on the TableRecord, FetchableRecord, and EncodableRecord protocols:
-
TableRecord is the protocol that lets you declare associations between record types:
extension Author: TableRecord { static let books = hasMany(Book.self) } extension Book: TableRecord { static let author = belongsTo(Author.self) }
-
FetchableRecord makes it possible to fetch records from the database:
extension Author: FetchableRecord { } // Who's prolific? let authors = try dbQueue.read { db in try Author .having(Author.books.count >= 20) .fetchAll(db) // [Author] }
FetchableRecord conformance can be derived from the standard Decodable protocol. See Codable Records for more information.
-
EncodableRecord makes it possible to fetch associated records with the
request(for:)
method:extension Book: EncodableRecord { // The request for the author of a book. var author: QueryInterfaceRequest<Author> { request(for: Book.author) } } // Who wrote this book? let book: Book = ... let author = try dbQueue.read { db in try book.author.fetchOne(db) // Author? }
A record type can conform to EncodableRecord via the PersistableRecord protocol. However, PersistableRecord also grants persistence methods, the ones that are able to insert, update, and delete rows in the database. When you'd rather keep a record type read-only, and yet profit from associations, all you need is EncodableRecord.
EncodableRecord conformance can be derived from the standard Encodable protocol. See Codable Records for more information.
GRDB handles several types of associations:
- BelongsTo
- HasMany
- HasOne
- HasManyThrough
- HasOneThrough
- Associations to common table expressions
An association generally declares a link from a record type to another, as in "one book belongs to its author". It instructs GRDB to use the foreign keys declared in the database as support for Swift methods.
Each one of these associations is appropriate for a particular database situation.
Associations to common table expressions are specific enough and are documented in Associations to Common Table Expressions.
- BelongsTo
- HasMany
- HasOne
- HasManyThrough
- HasOneThrough
- Choosing Between BelongsTo and HasOne
- Self Joins
The BelongsTo association sets up a one-to-one connection from a record type to another record type, such as each instance of the declaring record "belongs to" an instance of the other record.
For example, if your application includes authors and books, and each book is assigned its author, you'd declare the Book.author
association as below:
struct Book: TableRecord {
static let author = belongsTo(Author.self)
...
}
struct Author: TableRecord {
...
}
The BelongsTo association between a book and its author needs that the database table for books has a column that points to the table for authors:
See Convention for the BelongsTo Association for some sample code that defines the database schema for such an association, and Building Requests from Associations in order to learn how to use it.
The HasMany association indicates a one-to-many connection between two record types, such as each instance of the declaring record "has many" instances of the other record. You'll often find this association on the other side of a BelongsTo association.
For example, if your application includes authors and books, and each author is assigned zero or more books, you'd declare the Author.books
association as below:
struct Author: TableRecord {
static let books = hasMany(Book.self)
}
struct Book: TableRecord {
...
}
The HasMany association between an author and its books needs that the database table for books has a column that points to the table for authors:
See Convention for the HasMany Association for some sample code that defines the database schema for such an association, and Building Requests from Associations in order to learn how to use it.
The HasOne association, like BelongsTo, sets up a one-to-one connection from a record type to another record type, but with different semantics, and underlying database schema. It is usually used when an entity has been denormalized into two database tables.
For example, if your application has one database table for countries, and another for their demographic profiles, you'd declare the Country.demographics
association as below:
struct Country: TableRecord {
static let demographics = hasOne(Demographics.self, key: "demographics")
...
}
struct Demographics: TableRecord {
...
}
The HasOne association between a country and its demographics needs that the database table for demographics has a column that points to the table for countries:
Note that this demographics example of HasOne association uses an explicit "demographics"
key, unlike the BelongsTo and HasMany associations above. This key is necessary when you use a plural name for a one-to-one association. See Convention for Database Table Names for more information.
See Convention for the HasOne Association for some sample code that defines the database schema for such an association, and Building Requests from Associations in order to learn how to use it.
The HasManyThrough association is often used to set up a many-to-many connection with another record. This association indicates that the declaring record can be matched with zero or more instances of another record by proceeding through a third record. For example, consider the practice of passport delivery. The relevant association declarations could look like this:
struct Country: TableRecord {
static let passports = hasMany(Passport.self)
static let citizens = hasMany(Citizen.self, through: passports, using: Passport.citizen)
...
}
struct Passport: TableRecord {
static let country = belongsTo(Country.self)
static let citizen = belongsTo(Citizen.self)
}
struct Citizen: TableRecord {
static let passports = hasMany(Passport.self)
static let countries = hasMany(Country.self, through: passports, using: Passport.country)
...
}
The HasManyThrough association is also useful for setting up "shortcuts" through nested associations. For example, if a document has many sections, and a section has many paragraphs, you may sometimes want to get a simple collection of all paragraphs in the document. You could set that up this way:
struct Document: TableRecord {
static let sections = hasMany(Section.self)
static let paragraphs = hasMany(Paragraph.self, through: sections, using: Section.paragraphs)
...
}
struct Section: TableRecord {
static let paragraphs = hasMany(Paragraph.self)
...
}
struct Paragraph: TableRecord {
...
}
As in the examples above, HasManyThrough association is always built from two other associations: the through:
and using:
arguments. Those associations can be any other association (BelongsTo, HasMany, HasManyThrough, etc). The above Document.paragraphs
association can also be defined, in a much more explicit way, as below:
struct Document: TableRecord {
static let paragraphs = hasMany(
Paragraph.self,
through: Document.hasMany(Section.self),
using: Section.hasMany(Paragraph.self))
...
}
See Building Requests from Associations in order to learn how to use the HasManyThrough association.
A HasOneThrough association sets up a one-to-one connection with another record. This association indicates that the declaring record can be matched with one instance of another record by proceeding through a third record. For example, if each book belongs to a library, and each library has one address, then one knows where the book should be returned to:
struct Book: TableRecord {
static let library = belongsTo(Library.self)
static let returnAddress = hasOne(Address.self, through: library, using: Library.address)
...
}
struct Library: TableRecord {
static let address = hasOne(Address.self)
...
}
struct Address: TableRecord {
...
}
As in the example above, HasOneThrough association is always built from two other associations: the through:
and using:
arguments. Those associations can be any other association to one (BelongsTo, HasOne, HasOneThrough). The above Book.returnAddress
association can also be defined, in a much more explicit way, as below:
struct Book: TableRecord {
static let returnAddress = hasOne(
Address.self,
through: Book.belongsTo(Library.self),
using: Library.hasOne(Address.self))
...
}
See Building Requests from Associations in order to learn how to use the HasOneThrough association.
When you want to set up a one-to-one relationship between two record types, you'll need to add a BelongsTo association to one, and a HasOne association to the other. How do you know which is which?
The distinction is in where you place the database foreign key. The record that points to the other one has the BelongsTo association. The other record has the HasOne association:
A country has one demographic profile, a demographic profile belongs to a country:
struct Country: TableRecord {
static let demographics = hasOne(Demographics.self)
...
}
struct Demographics: TableRecord {
static let country = belongsTo(Country.self)
...
}
When designing your data model, you will sometimes find a record that should have a relation to itself. For example, you may want to store all employees in a single database table, but be able to trace relationships such as between manager and subordinates. This situation can be modeled with self-joining associations:
struct Employee {
static let subordinates = hasMany(Employee.self, key: "subordinates")
static let manager = belongsTo(Employee.self, key: "manager")
...
}
The matching migration would look like:
migrator.registerMigration("Employees") { db in
try db.create(table: "employee") { t in
t.autoIncrementedPrimaryKey("id")
t.belongsTo("manager", inTable: "employee", onDelete: .setNull)
t.column("name", .text)
}
}
Note that the associations on both sides of the self-join use a customized association key. This helps consuming this association. For example:
struct EmployeeInfo: FetchableRecord, Decodable {
var employee: Employee
var manager: Employee?
var subordinates: Set<Employee>
}
let request = Employee
.including(optional: Employee.manager)
.including(all: Employee.subordinates)
let employeeInfos: [EmployeeInfo] = try EmployeeInfo.fetchAll(db, request)
See Fetching Values from Associations for more information.
Associations are grounded in the database schema, the way database tables are defined.
For example, a BelongsTo association between a book and its author needs that the database table for books has a column that points to the table for authors.
GRDB also comes with several conventions for defining your database schema.
Those conventions help associations be convenient and, generally, "just work". When you can't, or don't want to follow conventions, you will have to override the expected defaults in your Swift code.
- Convention for Database Table Names
- Convention for the BelongsTo Association
- Convention for the HasMany Association
- Convention for the HasOne Association
- Foreign Keys
Database table names should be written in English, singular, and camelCased.
Make them look like Swift identifiers: book
, author
, postalAddress
.
If the database schema does not follow this convention, and has, for example, database tables which are named with underscores (postal_address
), you can still use associations. But you need to help row consumption by naming your associations with a customized key:
// Setup for table names that does not follow the expected convention
struct PostalAddress: TableRecord {
// Customized table name
static let databaseTableName = "postal_address"
}
extension Author {
// Customized association key
static let postalAddress = belongsTo(PostalAddress.self, key: "postalAddress")
}
GRDB will automatically pluralize or singularize names in order to help you easily associate records.
For example, the Book and Author records will automatically feed properties named books
, author
, or bookCount
in your decoded records, without any explicit configuration, as long as the names of the backing database tables are "book" and "author".
The GRDB pluralization mechanisms are very powerful, being capable of pluralizing and singularizing both regular and irregular words (it's directly inspired from the battle-tested Ruby on Rails inflections).
When using class names composed of two or more words, the table name should use the camelCase singular form:
RecordType | Table Name | Derived identifiers |
---|---|---|
Book | book | book , books , bookCount |
LineItem | lineItem | lineItem , lineItems , lineItemPriceSum |
Mouse | mouse | mouse , mice , maxMouseSize |
Person | person | person , people , personCount |
If your application relies on non-English names, GRDB may generate unexpected identifiers. If this happens, please open an issue.
See The Structure of a Joined Request for more information.
extension Book: TableRecord {
static let author = belongsTo(Author.self)
}
Here is the recommended migration for the BelongsTo association:
migrator.registerMigration("Books and Authors") { db in
try db.create(table: "author") { t in
t.autoIncrementedPrimaryKey("id") // (1)
t.column("name", .text)
}
try db.create(table: "book") { t in
t.autoIncrementedPrimaryKey("id")
t.belongsTo("author", onDelete: .cascade) // (2)
.notNull() // (3)
t.column("title", .text)
}
}
- The
author
table has a primary key. - The
book.authorId
column is used to link a book to the author it belongs to. This column is indexed in order to ease the selection of an author's books. A foreign key is defined frombook.authorId
column toauthors.id
, so that SQLite guarantees that no book refers to a missing author. TheonDelete: .cascade
option has SQLite automatically delete all of an author's books when that author is deleted. See Foreign Key Actions for more information. - Make the
book.authorId
column not null if you want SQLite to guarantee that all books have an author.
The example above uses auto-incremented primary keys. But generally speaking, all primary keys are supported, including composite primary keys that span several columns.
Following this convention lets you write, for example:
struct Book: TableRecord {
static let author = belongsTo(Author.self)
}
struct Author: TableRecord {
}
If the database schema does not follow this convention, and does not define foreign keys between tables, you can still use BelongsTo associations. But your help is needed to define the missing foreign key:
struct Book: TableRecord {
static let author = belongsTo(Author.self, using: ForeignKey(...))
}
See Foreign Keys for more information.
extension Author: TableRecord {
static let books = hasMany(Book.self)
}
Here is the recommended migration for the HasMany association:
migrator.registerMigration("Books and Authors") { db in
try db.create(table: "author") { t in
t.autoIncrementedPrimaryKey("id") // (1)
t.column("name", .text)
}
try db.create(table: "book") { t in
t.autoIncrementedPrimaryKey("id")
t.belongsTo("author", onDelete: .cascade) // (2)
.notNull() // (3)
t.column("title", .text)
}
}
- The
author
table has a primary key. - The
book.authorId
column is used to link a book to the author it belongs to. This column is indexed in order to ease the selection of an author's books. A foreign key is defined frombook.authorId
column toauthors.id
, so that SQLite guarantees that no book refers to a missing author. TheonDelete: .cascade
option has SQLite automatically delete all of an author's books when that author is deleted. See Foreign Key Actions for more information. - Make the
book.authorId
column not null if you want SQLite to guarantee that all books have an author.
The example above uses auto-incremented primary keys. But generally speaking, all primary keys are supported, including composite primary keys that span several columns.
Following this convention lets you write, for example:
struct Book: TableRecord {
}
struct Author: TableRecord {
static let books = hasMany(Book.self)
}
If the database schema does not follow this convention, and does not define foreign keys between tables, you can still use HasMany associations. But your help is needed to define the missing foreign key:
struct Author: TableRecord {
static let books = hasMany(Book.self, using: ForeignKey(...))
}
See Foreign Keys for more information.
extension Country: TableRecord {
static let demographics = hasOne(Demographics.self)
}
Here is the recommended migration for the HasOne association:
migrator.registerMigration("Countries") { db in
try db.create(table: "country") { t in
t.primaryKey("code", .text) // (1)
t.column("name", .text)
}
try db.create(table: "demographics") { t in
t.autoIncrementedPrimaryKey("id")
t.belongsTo("country", onDelete: .cascade) // (2)
.notNull() // (3)
.unique() // (4)
t.column("population", .integer)
t.column("density", .double)
}
}
- The
country
table has a primary key. - The
demographics.countryCode
column is used to link a demographic profile to the country it belongs to. This column is indexed in order to ease the selection of the demographics of a country. A foreign key is defined fromdemographics.countryCode
column tocountry.code
, so that SQLite guarantees that no profile refers to a missing country. TheonDelete: .cascade
option has SQLite automatically delete a profile when its country is deleted. See Foreign Key Actions for more information. - Make the
demographics.countryCode
column not null if you want SQLite to guarantee that all profiles are linked to a country. - Create a unique index on the
demographics.countryCode
column in order to guarantee the unicity of any country's profile.
The example above uses a string primary key for the "country" table. But generally speaking, all primary keys are supported, including composite primary keys that span several columns.
Following this convention lets you write, for example:
struct Country: TableRecord {
static let demographics = hasOne(Demographics.self)
}
struct Demographics: TableRecord {
}
If the database schema does not follow this convention, and does not define foreign keys between tables, you can still use HasOne associations. But your help is needed to define the missing foreign key:
struct Country: TableRecord {
static let demographics = hasOne(Demographics.self, using: ForeignKey(...))
}
See Foreign Keys for more information.
Associations can automatically infer the foreign keys that define how two database tables are linked together.
In the example below, the book.authorId
column is automatically used to link a book to its author, because the database schema defines a foreign key between the book and author database tables (see Convention for the BelongsTo Association).
struct Book: TableRecord {
static let author = belongsTo(Author.self)
}
struct Author: TableRecord {
static let books = hasMany(Book.self)
}
Note: Generally speaking, all foreign keys are supported, including composite keys that span several columns.
Warning: SQLite voids foreign key constraints when one or more of a foreign key column is NULL (see SQLite Foreign Key Support). GRDB does not match foreign keys that involve a NULL value either.
Sometimes the database schema does not define any foreign key. And sometimes, there are several foreign keys from a table to another.
// The migration that has created the above schema
migrator.registerMigration("Library") { db in
try db.create(table: "person") { t in
t.autoIncrementedPrimaryKey("id")
t.column("name", .text)
}
try db.create(table: "book") { t in
t.autoIncrementedPrimaryKey("id")
t.belongsTo("author", inTable: "person")
t.belongsTo("translator", inTable: "person")
t.column("title", .text)
}
}
When this happens, associations can't be automatically inferred from the database schema. GRDB will complain with a fatal error such as "Ambiguous foreign key from book to person", or "Could not infer foreign key from book to person".
Your help is needed. You have to instruct GRDB which foreign key to use:
struct Book: TableRecord {
// Define foreign keys
static let authorForeignKey = ForeignKey(["authorId"])
static let translatorForeignKey = ForeignKey(["translatorId"])
// Use foreign keys to define associations:
static let author = belongsTo(Person.self, using: authorForeignKey)
static let translator = belongsTo(Person.self, using: translatorForeignKey)
}
Foreign keys are always defined from the table that contains the columns at the origin of the foreign key. Person's symmetric HasMany associations reuse Book's foreign keys:
struct Person: TableRecord {
static let writtenBooks = hasMany(Book.self, using: Book.authorForeignKey)
static let translatedBooks = hasMany(Book.self, using: Book.translatorForeignKey)
}
Foreign keys can also be defined from query interface columns:
struct Book: TableRecord {
enum Columns: String, ColumnExpression {
case id, title, authorId, translatorId
}
static let authorForeignKey = ForeignKey([Columns.authorId])
static let translatorForeignKey = ForeignKey([Columns.translatorId])
}
When the destination table of a foreign key does not define any primary key, you need to provide the full definition of a foreign key:
struct Book: TableRecord {
static let authorForeignKey = ForeignKey(["authorId"], to: ["id"])
static let author = belongsTo(Person.self, using: authorForeignKey)
}
Once you have defined associations, you can define fetch request that involve several record types.
Fetch requests do not visit the database until you fetch values from them. This will be covered in Fetching Values from Associations. But before you can fetch anything, you have to describe what you want to fetch. This is the topic of this chapter.
- Requesting Associated Records
- Joining And Prefetching Associated Records
- Combining Associations
- Filtering Associations
- Sorting Associations
- Ordered Associations
- Columns Selected by an Association
- Further Refinements to Associations
- Table Aliases
- Refining Association Requests
You can use associations to build requests for associated records.
For example, given a Book.author
BelongsTo association, you can build a request for the author of a book with the request(for:)
method. In the example below, we return this request from the Book.author
property:
struct Book: TableRecord, EncodableRecord {
/// The association from a book to is author
static let author = belongsTo(Author.self)
/// The request for the author of a book
var author: QueryInterfaceRequest<Author> {
request(for: Book.author)
}
}
You can now fetch the author of a book:
let book: Book = ...
let author = try book.author.fetchOne(db) // Author?
All other associations, HasOne, HasMany, HasOneThrough, and HasManyThrough, can also build requests for associated records. For example:
struct Author: TableRecord, EncodableRecord {
/// The association from an author to its books
static let books = hasMany(Book.self)
/// The request for the books of an author
var books: QueryInterfaceRequest<Book> {
request(for: Author.books)
}
}
let author: Author = ...
let books = try author.books.fetchAll(db) // [Book]
Requests for associated records can be filtered and ordered like all query interface requests:
let novels = try author
.books
.filter(Column("kind") == BookKind.novel)
.order(Column("publishDate").desc)
.fetchAll(db) // [Book]
You build requests that involve associations with one of the following "joining methods":
- Prefetch associated records:
- Prefetch only a few columns of an associated record:
- Join associated records without prefetching:
- Choosing a Joining Method Given the Shape of the Decoded Type
For example, fetch books along with their author:
// SELECT book.*, author.*
// FROM book
// JOIN author ON author.id = book.authorId
let request = Book.including(required: Book.author)
This method accepts any association. It has the base record fetched along with one associated record, which is "included" in the fetched results. When the associated record does not exist, records are not present in the fetched results: the associated record is "required".
To fetch results from such a request, you define a dedicated record type:
// Fetch all books along with their author
struct BookInfo: Decodable, FetchableRecord {
var book: Book // The base record
var author: Author // The required associated record
}
let bookInfos = try Book
.including(required: Book.author)
.asRequest(of: BookInfo.self)
.fetchAll(db)
The CodingKey for the BookInfo.author
property must match the association key of the Book.author
association. The association key is, by default, the name of the associated database table. This can be configured with the association forKey(_:)
method:
struct Author: TableRecord {
static let databaseTableName = "writer"
}
struct Book: TableRecord {
// Replace the default "writer" association key with "author"
static let author = belongsTo(Author.self).forKey("author")
}
// Fetch all books along with their author
struct BookInfo: Decodable, FetchableRecord {
var book: Book
var author: Author // Matches the "author" association key
}
let bookInfos = try Book
.including(required: Book.author)
.asRequest(of: BookInfo.self)
.fetchAll(db)
When you only need a few columns of the associated record, you can define a partial type with one property per selected column of the associated record:
// Fetch all books along with the name and country of their author
struct BookInfo: Decodable, FetchableRecord {
struct PartialAuthor: Decodable {
var name: String
var country: String
}
var book: Book
var author: PartialAuthor
}
let bookInfos = try Book
.including(required: Book.author.select(Column("name"), Column("country")))
.asRequest(of: BookInfo.self)
.fetchAll(db)
But if you'd rather avoid this extra partial type, prefer annotated(withRequired:)
:
// Fetch all books along with the country of their author
struct BookInfo: Decodable, FetchableRecord {
var book: Book
var country: String
}
let bookInfos = try Book
.annotated(withRequired: Book.author.select(Column("country")))
.asRequest(of: BookInfo.self)
.fetchAll(db)
For example, fetch books along with their eventual author:
// SELECT book.*, author.*
// FROM book
// LEFT JOIN author ON author.id = book.authorId
let request = Book.including(optional: Book.author)
This method accepts any association. It has the base record fetched along with one associated record, which is "included" in the fetched results. The associated record can be missing: it is "optional".
To fetch results from such a request, you define a dedicated record type:
// Fetch all books along with their eventual author
struct BookInfo: Decodable, FetchableRecord {
var book: Book // The base record
var author: Author? // The optional associated record
}
let bookInfos = try Book
.including(optional: Book.author)
.asRequest(of: BookInfo.self)
.fetchAll(db)
The CodingKey for the BookInfo.author
property must match the association key of the Book.author
association. See including(required:)
for more information.
When you only need a few columns of the associated record, you can define a partial type with one property per selected column of the associated record:
// Fetch all books along with the name and country of their eventual author
struct BookInfo: Decodable, FetchableRecord {
struct PartialAuthor: Decodable {
var name: String
var country: String
}
var book: Book
var author: PartialAuthor?
}
let bookInfos = try Book
.including(optional: Book.author.select(Column("name"), Column("country")))
.asRequest(of: BookInfo.self)
.fetchAll(db)
But if you'd rather avoid this extra partial type, prefer annotated(withOptional:)
:
// Fetch all books along with the country of their eventual author
struct BookInfo: Decodable, FetchableRecord {
var book: Book
var country: String?
}
let bookInfos = try Book
.annotated(withOptional: Book.author.select(Column("country")))
.asRequest(of: BookInfo.self)
.fetchAll(db)
For example, fetch authors along with their books:
// SELECT author.* FROM author
// SELECT book.* FROM book WHERE authorId IN (...)
let request = Author.including(all: Author.books)
This method accepts any to-many association (HasMany or HasManyThrough). It has the base record fetched along with all its associated records, which are "included" in the fetched results.
To fetch results from such a request, you define a dedicated record type:
// Fetch all authors along with their books
struct AuthorInfo: Decodable, FetchableRecord {
var author: Author // The base record
var books: [Book] // A collection of associated records
}
let authorInfos = try Author
.including(all: Author.books)
.asRequest(of: AuthorInfo.self)
.fetchAll(db)
The associated records can be stored into an Array as in the above example, but also a Set, and generally speaking any decodable Swift collection.
The CodingKey for the AuthorInfo.books
property must match the association key of the Author.books
association. The association key is, by default, the pluralized name of the associated database table. This can be configured with the association forKey(_:)
method:
struct Book: TableRecord {
static let databaseTableName = "publication"
}
struct Author: TableRecord {
// Replace the default "publications" association key with "books"
static let books = hasMany(Book.self).forKey("books")
}
// Fetch all authors along with their books
struct AuthorInfo: Decodable, FetchableRecord {
var author: Author
var books: [Book] // Matches the "books" association key
}
let authorInfos = try Author
.including(all: Author.books)
.asRequest(of: AuthorInfo.self)
.fetchAll(db)
When you only need a few columns of the associated records, you can define a partial type with one property per selected column of the associated record:
// Fetch all authors along with the titles and years of their books
struct AuthorInfo: Decodable, FetchableRecord {
struct PartialBook: Decodable {
var title: String
var year: Int
}
var author: Author
var books: [PartialBook]
}
let authorInfos = try Author
.including(all: Author.books.select(Column("title"), Column("year")))
.asRequest(of: AuthorInfo.self)
.fetchAll(db)
When you need a single column of the associated records, avoid the extra partial type, and instead use the association forKey(_:)
method in order to match the name of the decoded property:
// Fetch all authors along with the titles of their books
struct AuthorInfo: Decodable, FetchableRecord {
var author: Author
var bookTitles: [String]
}
let authorInfos = try Author
.including(all: Author.books
.select(Column("title"))
.forKey("bookTitles"))
.asRequest(of: AuthorInfo.self)
.fetchAll(db)
For example, fetch books along with the name and country of their author:
// SELECT book.*, author.name, author.country
// FROM book
// JOIN author ON author.id = book.authorId
let request = Book.annotated(withRequired: Book.author.select(Column("name"), Column("country")))
This method accepts any association. The base record is annotated with the selected columns of one associated record. When the associated record does not exist, records are not present in the fetched results: the associated record is "required".
To fetch results from such a request, you define a dedicated record type:
// Fetch all books along with the name and country of their author
struct BookInfo: Decodable, FetchableRecord {
var book: Book // The base record
var name: String // A column of the required associated record
var country: String // A column of the required associated record
}
let bookInfos = try Book
.annotated(withRequired: Book.author.select(Column("name"), Column("country")))
.asRequest(of: BookInfo.self)
.fetchAll(db)
If the name of a column of the associated record is also the name of a column of the base record, rename the associated column with the column forKey(_:)
method, and accordingly rename the property of the decoded record type:
// Fetch all books along with the name of their author
struct BookInfo: Decodable, FetchableRecord {
var book: Book
var authorName: String
}
let bookInfos = try Book
.annotated(withRequired: Book.author.select(Column("name").forKey("authorName")))
.asRequest(of: BookInfo.self)
.fetchAll(db)
For example, fetch books along with the name and country of their eventual author:
// SELECT book.*, author.name, author.country
// FROM book
// LEFT JOIN author ON author.id = book.authorId
let request = Book.annotated(withOptional: Book.author.select(Column("name"), Column("country")))
This method accepts any association. The base record is annotated with the selected columns of one associated record. When the associated record does not exist, the columns of the associated record are NULL: the associated record is "optional".
To fetch results from such a request, you define a dedicated record type:
// Fetch all books along with the name of their eventual author
struct BookInfo: Decodable, FetchableRecord {
var book: Book // The base record
var name: String? // A column of the optional associated record
var country: String? // A column of the optional associated record
}
let bookInfos = try Book
.annotated(withOptional: Book.author.select(Column("name"), Column("country")))
.asRequest(of: BookInfo.self)
.fetchAll(db)
If the name of a column of the associated record is also the name of a column of the base record, rename the associated column with the column forKey(_:)
method, and accordingly rename the property of the decoded record type:
// Fetch all books along with the name of their eventual author
struct BookInfo: Decodable, FetchableRecord {
var book: Book
var authorName: String?
}
let bookInfos = try Book
.annotated(withOptional: Book.author.select(Column("name").forKey("authorName")))
.asRequest(of: BookInfo.self)
.fetchAll(db)
For example, fetch books by French authors:
// SELECT book.*
// FROM book
// JOIN author ON author.id = book.authorId
// AND author.country = 'France'
let request = Book.joining(required: Book.author.filter(Column("country") == "France"))
This method accepts any association. It has the base record "joined" with one associated record, which is not included in the fetched results. When the associated record does not exist, the base record is not present in the fetched results: the associated record is "required".
To fetch results from such a request, you do not need to define a dedicated record type:
// Fetch all books by French authors
let books = try Book
.joining(required: Book.author.filter(Column("country") == "France"))
.fetchAll(db)
// SELECT book.*
// FROM book
// LEFT JOIN author ON author.id = book.authorId
let request = Book.joining(optional: Book.author)
This method accepts any association. It has the base record "joined" with one associated record, which is not included in the fetched results. The associated record can be missing: it is "optional".
This method has no observable effect unless the associated record is used in a way or another. We'll see examples later in this documentation.
In the description of the joining methods above, we have seen that you need to define dedicated record types in order to prefetch associated records. Each joining method needs a dedicated record type that has a specific shape.
In this chapter, we take the reversed perspective. We list various shapes of decoded record types. When you find the type you want, you'll know the joining method you need.
Note: If you don't find the type you want, chances are that you are fighting the framework, and should reconsider your position. Your escape hatch is the low-level apis described in Decoding a Joined Request with FetchableRecord.
-
struct BookInfo: Decodable, FetchableRecord { var book: Book // The base record var author: Author // The associated record } let bookInfos = try Book .including(required: Book.author) .asRequest(of: BookInfo.self) .fetchAll(db)
struct BookInfo: Decodable, FetchableRecord { struct PartialAuthor: Decodable { var name: String var country: String } var book: Book // The base record var author: PartialAuthor // The partial associated record } let bookInfos = try Book .including(required: Book.author.select(Column("name"), Column("country"))) .asRequest(of: BookInfo.self) .fetchAll(db)
-
struct BookInfo: Decodable, FetchableRecord { var book: Book // The base record var authorName: String // A column of the associated record var country: String // A column of the associated record } let bookInfos = try Book .annotated(withRequired: Book.author.select( Column("name").forKey("authorName"), Column("country"))) .asRequest(of: BookInfo.self) .fetchAll(db)
-
struct BookInfo: Decodable, FetchableRecord { var book: Book // The base record var author: Author? // The eventual associated record } let bookInfos = try Book .including(optional: Book.author) .asRequest(of: BookInfo.self) .fetchAll(db)
struct BookInfo: Decodable, FetchableRecord { struct PartialAuthor: Decodable { var name: String var country: String } var book: Book // The base record var author: PartialAuthor? // The eventual partial associated record } let bookInfos = try Book .including(optional: Book.author.select(Column("name"), Column("country"))) .asRequest(of: BookInfo.self) .fetchAll(db)
-
struct BookInfo: Decodable, FetchableRecord { var book: Book // The base record var authorName: String? // A column of the eventual associated record var country: String? // A column of the eventual associated record } let bookInfos = try Book .annotated(withOptional: Book.author.select( Column("name").forKey("authorName"), Column("country"))) .asRequest(of: BookInfo.self) .fetchAll(db)
-
struct AuthorInfo: Decodable, FetchableRecord { var author: Author // The base record var books: [Book] // A collection of associated records } let authorInfos = try Author .including(all: Author.books) .asRequest(of: AuthorInfo.self) .fetchAll(db)
struct AuthorInfo: Decodable, FetchableRecord { struct PartialBook: Decodable { var title: String var year: Int } var author: Author // The base record var books: [PartialBook] // A collection of partial associated records } let authorInfos = try Author .including(all: Author.books.select(Column("title"), Column("year"))) .asRequest(of: AuthorInfo.self) .fetchAll(db)
struct AuthorInfo: Decodable, FetchableRecord { var author: Author // The base record var bookTitles: [String] // A collection of one column of the associated records } let authorInfos = try Author .including(all: Author.books .select(Column("title")) .forKey("bookTitles")) .asRequest(of: AuthorInfo.self) .fetchAll(db)
Associations can be combined in order to build more complex requests.
You can join several associations in parallel:
// SELECT book.*, person1.*, person2.*
// FROM book
// JOIN person person1 ON person1.id = book.authorId
// LEFT JOIN person person2 ON person2.id = book.translatorId
let request = Book
.including(required: Book.author)
.including(optional: Book.translator)
// This request can feed the following record:
struct BookInfo: FetchableRecord, Decodable {
var book: Book
var author: Person
var translator: Person?
}
let bookInfos: [BookInfo] = try BookInfo.fetchAll(db, request)
The request above fetches all books, along with their author and eventual translator.
You can chain associations in order to jump from a record to another:
// SELECT book.*, person.*, country.*
// FROM book
// JOIN person ON person.id = book.authorId
// LEFT JOIN country ON country.code = person.countryCode
let request = Book
.including(required: Book.author
.including(optional: Person.country))
// This request can feed the following record:
struct BookInfo: FetchableRecord, Decodable {
var book: Book
var author: Author
var country: Country?
}
let bookInfos: [BookInfo] = try BookInfo.fetchAll(db, request)
The request above fetches all books, along with their author, and their author's country.
When you chain associations, you can avoid fetching intermediate tables by replacing the including
method with joining
. The request below fetches all books, along with their author's country, but does not include the intermediate authors in the fetched results:
// SELECT book.*, country.*
// FROM book
// LEFT JOIN person ON person.id = book.authorId
// LEFT JOIN country ON country.code = person.countryCode
let request = Book
.joining(optional: Book.author
.including(optional: Person.country))
// This request can feed the following record:
struct BookInfo: FetchableRecord, Decodable {
var book: Book
var country: Country?
}
let bookInfos: [BookInfo] = try BookInfo.fetchAll(db, request)
HasOneThrough and HasManyThrough associations provide a shortcut for those requests that skip intermediate tables:
// SELECT book.*, country.*
// FROM book
// LEFT JOIN person ON person.id = book.authorId
// LEFT JOIN country ON country.code = person.countryCode
let request = Book.including(optional: Book.country)
// This request can feed the following record:
struct BookInfo: FetchableRecord, Decodable {
var book: Book
var country: Country?
}
let bookInfos: [BookInfo] = try BookInfo.fetchAll(db, request)
Warning: you can not currently chain a required association behind an optional association:
// Not implemented let request = Book .joining(optional: Book.author .including(required: Person.country))This code compiles, but you'll get a runtime fatal error "Not implemented: chaining a required association behind an optional association". Future versions of GRDB may allow such requests.
You can filter associated records.
The filter(_:)
, filter(id:)
, filter(ids:)
, filter(key:)
and filter(keys:)
methods, that you already know for filtering simple requests, can filter associated records as well:
// SELECT book.*
// FROM book
// JOIN person ON person.id = book.authorId
// AND person.countryCode = 'FR'
let frenchAuthor = Book.author.filter(Column("countryCode") == "FR")
let request = Book.joining(required: frenchAuthor)
// This request feeds the Book record:
let books: [Book] = try request.fetchAll(db)
The request above fetches all books written by a French author.
The one below fetches all authors along with their novels and poems:
let request = Author
.including(all: Author.book
.filter(Column("kind") == "novel")
.forKey("novels"))
.including(all: Author.book
.filter(Column("kind") == "poems")
.forKey("poems"))
// This request can feed the following record:
struct AuthorInfo: FetchableRecord, Decodable {
var author: Author
var novels: [Book]
var poems: [Book]
}
let authorInfos: [AuthorInfo] = try AuthorInfo.fetchAll(db, request)
There are more filtering options:
- Filtering on conditions that involve several tables.
- Filtering in the WHERE clause instead of the ON clause (can be useful when you are skilled enough in SQL to make the difference).
Those extra filtering options require Table Aliases, introduced below.
You can sort fetched results according to associated records.
The order()
method, that you already know for sorting simple requests, can sort associated records as well:
// SELECT book.*, person.*
// FROM book
// JOIN person ON person.id = book.authorId
// ORDER BY person.name
let sortedAuthor = Book.author.order(Column("name"))
let request = Book.including(required: sortedAuthor)
When you sort both the base record and the associated record, the request is sorted on the base record first, and on the associated record next:
// SELECT book.*, person.*
// FROM book
// JOIN person ON person.id = book.authorId
// ORDER BY book.publishDate DESC, person.name
let sortedAuthor = Book.author.order(Column("name"))
let request = Book
.including(required: sortedAuthor)
.order(Column("publishDate").desc)
There are more sorting options:
- Sorting on expressions that involve several tables.
- Changing the order of the sorting terms (such as sorting on author name first, and then publish date).
Those extra sorting options require Table Aliases, introduced below.
By default, HasMany or HasManyThrough associations are unordered: the order of associated records is undefined unless explicitly specified on each request.
But you can build an ordering right into the definition of an association, so that it becomes the default ordering for this association. For example, let's model soccer teams and players, ordered by the number printed on their shirt.
Let's start with a HasMany association. Each player knows its position in its team:
struct Team: FetchableRecord, TableRecord {
var id: Int64
var name: String
}
struct Player: FetchableRecord, TableRecord {
var id: Int64
var teamId: Int64
var name: String
var position: Int
}
The Team.players
association is ordered by position, so that all team players are loaded well-sorted by default:
extension Team {
static let players = hasMany(Player.self).order(Column("position"))
var players: QueryInterfaceRequest<Player> {
request(for: Team.players)
}
}
Things are very similar for HasManyThrough associations. Now each player knows its position in the teams it belongs to:
struct Team: FetchableRecord, TableRecord {
var id: Int64
var name: String
}
struct PlayerRole: FetchableRecord, TableRecord {
var teamId: Int64
var playerId: Int64
var position: Int
}
struct Player: FetchableRecord, TableRecord {
var id: Int64
var name: String
}
Again, the Team.players
association is ordered by position, so that all team players are loaded well-sorted by default:
extension Team {
static let playerRoles = hasMany(PlayerRole.self).order(Column("position"))
static let players = hasMany(Player.self, through: playerRoles, using: PlayerRole.player)
var players: QueryInterfaceRequest<Player> {
request(for: Team.players)
}
}
extension PlayerRole {
static let player = belongsTo(Player.self)
}
In both cases, you can escape the default ordering when you need it:
struct TeamInfo: Decodable, FetchableRecord {
var team: Team
var players: [Player]
}
// Default ordering by position
let team: Team = ...
let players = try team.players.fetchAll(db)
let teamInfos = try Team
.including(all: Team.players)
.asRequest(of: TeamInfo.self)
.fetchAll(db)
// Custom ordering
let team: Team = ...
let players = try team.players.order(Column("name")).fetchAll(db)
let teamInfos = try Team
.including(all: Team.players.order(Column("name")))
.asRequest(of: TeamInfo.self)
.fetchAll(db)
By default, associated records, like all records, include all their columns:
// SELECT book.*, author.*
// FROM book
// JOIN author ON author.id = book.authorId
let request = Book.including(required: Book.author)
The selection can be changed for each individual request, or for all requests including a given type.
To specify the default selection for a given record type, see Columns Selected by a Request.
To specify the selection in a specific request, use the select
method:
// SELECT book.*, author.id, author.name
// FROM book
// JOIN author ON author.id = book.authorId
let restrictedAuthor = Book.author.select(Column("id"), Column("name"))
let request = Book.including(required: restrictedAuthor)
In order to fetch from such requests of partial records, see the documentation of the joining methods.
Associations support more refinements:
-
distinct
Fetch all authors with the kinds of books they write (novels, poems, plays, etc):
struct AuthorInfo: Decodable, FetchableRecord { var author: Author var bookKinds: Set<Book.Kind> } let distinctBookKinds = Author.books .select(Column("kind")) .distinct() .forKey("bookKinds") let authorInfos: [AuthorInfo] = try Author .including(all: distinctBookKinds) .asRequest(of: AuthorInfo.self) .fetchAll(db)
-
group
,having
Fetch all authors with the year of their latest book for each kind (novels, poems, plays, etc):
struct BookKindInfo: Decodable { var kind: Book.Kind var maxYear: Int } struct AuthorInfo: Decodable, FetchableRecord { var author: Author var bookKindInfos: [BookKindInfo] } let bookKindInfos = Author.books .select( Column("kind"), max(Column("year")).forKey("maxYear")) .group(Column("kind")) .forKey("bookKindInfos") let authorInfos: [AuthorInfo] = try Author .including(all: bookKindInfos) .asRequest(of: AuthorInfo.self) .fetchAll(db)
-
Fetch all authors with their awarded books:
struct AuthorInfo: FetchableRecord, Decodable { var author: Author var awardedBooks: [Book] } let awardedBooks = Author.books .having(Book.awards.isEmpty == false) .forKey("awardedBooks") let authorInfos: [AuthorInfo] = try Author .including(all: awardedBooks) .asRequest(of: AuthorInfo.self) .fetchAll(db)
-
Association can use their own CTEs:
struct AuthorInfo: FetchableRecord, Decodable { var author: Author var specialBooks: [Book] } let specialCTE = CommonTableExpression(...) let specialBooks = Author.books .with(specialCTE) ... // use the CTE in the book association .forKey("specialBooks") let authorInfos = try Author .including(all: specialBooks) .asRequest(of: AuthorInfo.self) .fetchAll(db)
Warning: associations refined with
limit
,distinct
,group
,having
, or association aggregates can only be used withincluding(all:)
. You will get a fatal error if you use them with other joining methods:including(required:)
, etc.
In all examples we have seen so far, all associated records are joined, included, filtered, and sorted independently. We could not filter them on conditions that involve several records, for example.
Let's say we look for posthumous books, published after their author has died. We need to compare a book publication date with an author eventual death date.
Let's first see a wrong way to do it:
// A wrong request:
// SELECT book.*
// FROM book
// JOIN person ON person.id = book.authorId
// WHERE book.publishDate >= book.deathDate
let request = Book
.joining(required: Book.author)
.filter(Column("publishDate") >= Column("deathDate"))
When executed, we'll get a DatabaseError of code 1, "no such column: book.deathDate".
That is because the "deathDate" column has been used for filtering books, when it is defined on the person database table.
To fix this error, we need a table alias:
let authorAlias = TableAlias()
We modify the Book.author
association so that it uses this table alias, and we use the table alias to qualify author columns where needed:
// SELECT book.*
// FROM book
// JOIN person ON person.id = book.authorId
// WHERE book.publishDate >= person.deathDate
let request = Book
.joining(required: Book.author.aliased(authorAlias))
.filter(Column("publishDate") >= authorAlias[Column("deathDate")])
Table aliases can also improve control over the ordering of request results. In the example below, we override the default ordering of associated records by sorting on author names first:
// SELECT book.*
// FROM book
// JOIN person ON person.id = book.authorId
// ORDER BY person.name, book.publishDate
let request = Book
.joining(required: Book.author.aliased(authorAlias))
.order(authorAlias[Column("name")], Column("publishDate"))
Table aliases can be given a name. This name is guaranteed to be used as the table alias in the SQL query. This guarantee lets you write SQL snippets when you need it:
// SELECT b.*
// FROM book b
// JOIN person a ON a.id = b.authorId
// AND a.countryCode = 'FR'
// WHERE b.publishDate >= a.deathDate
let bookAlias = TableAlias(name: "b")
let authorAlias = TableAlias(name: "a")
let request = Book.aliased(bookAlias)
.joining(required: Book.author.aliased(authorAlias)
.filter(sql: "a.countryCode = ?", arguments: ["FR"]))
.filter(sql: "b.publishDate >= a.deathDate")
Note: avoid reusing table aliases between several tables or requests, because you will get a fatal error:
// Fatal error: A TableAlias most not be used to refer to multiple tables let alias = TableAlias() let books = Book.aliased(alias)... let people = Person.aliased(alias)...Note: you can't use the
including(all:)
method and use table aliases to filter the associated records on other records:// NOT IMPLEMENTED: loading all authors along with their posthumous books let authorAlias = TableAlias() let request = Author .aliased(authorAlias) .including(all: Author.books .filter(Column("publishDate") >= authorAlias[Column("deathDate")]))
When you join or include an association several times, with the same association key, GRDB will apply the following rules:
-
including
wins overjoining
:// Equivalent to Record.including(optional: association) Record .including(optional: association) .joining(optional: association)
-
required
wins overoptional
:// Equivalent to Record.including(required: association) Record .including(required: association) .including(optional: association)
-
All filters are applied:
// Equivalent to Record.including(required: association.filter(condition1 && condition2)) Record .including(required: association.filter(condition1)) .including(optional: association.filter(condition1))
-
The last ordering wins:
// Equivalent to Record.including(required: association.order(ordering2)) Record .including(required: association.order(ordering1)) .including(optional: association.order(ordering2))
-
The last selection wins:
// Equivalent to Record.including(required: association.select(selection2)) Record .including(required: association.select(selection1)) .including(optional: association.select(selection2))
Those rules exist so that you can design fluent interfaces that build complex requests out of simple building blocks.
For example, we can start by defining base requests as extensions to the DerivableRequest Protocol:
// Author requests
extension DerivableRequest<Author> {
/// Filters authors by country
func filter(country: String) -> Self {
filter(Column("country") == country)
}
}
// Book requests
extension DerivableRequest<Book> {
/// Filters books by author country
func filter(authorCountry: String) -> Self {
joining(required: Book.author.filter(country: country))
}
/// Order books by author name and then book title
func orderedByAuthorNameAndYear() -> Self {
let authorAlias = TableAlias()
return self
.joining(optional: Book.author.aliased(authorAlias))
.order(
authorAlias[Column("name")].collating(.localizedCaseInsensitiveCompare),
Column("year"))
}
}
And then compose those in a fluent style:
struct BookInfo: FetchableRecord, Decodable {
var book: Book
var author: Author
}
// SELECT book.*, author.*
// FROM book
// JOIN author ON author.id = book.authorId AND author.country = 'FR'
// ORDER BY author.name COLLATE ..., book.year
let bookInfos = try Book.all()
.filter(authorCountry: "FR")
.orderedByAuthorNameAndYear()
.including(required: Book.author)
.asRequest(of: BookInfo.self)
.fetchAll(db)
Remember that those refinement rules only apply when an association is joined or included several times, with the same association key. Changing this key stops merging associations together. See Isolation of Multiple Aggregates for a longer discussion.
We have seen in Joining And Prefetching Associated Records how to define requests that involve several records.
To consume those requests, you will generally define a record type that matches the structure of the request. You'll make it adopt the FetchableRecord protocol, so that it can decode database rows.
Often, you'll also make it adopt the standard Decodable protocol, because the compiler will generate the decoding code for you.
Each association included in the request can feed a property of the decoded record:
-
including(optional:)
feeds an optional property:let request = Employee.including(optional: Employee.manager) struct EmployeeInfo: FetchableRecord, Decodable { var employee: Employee var manager: Employee? // the optional associated manager } let employeeInfos: [EmployeeInfo] = try EmployeeInfo.fetchAll(db, request)
-
including(required:)
feeds an non-optional property:let request = Book.including(required: Book.author) struct BookInfo: FetchableRecord, Decodable { var book: Book var author: Author // the required associated author } let bookInfos: [BookInfo] = try BookInfo.fetchAll(db, request)
-
including(all:)
feeds an Array or Set property:let request = Author.including(all: Author.books) struct AuthorInfo: FetchableRecord, Decodable { var author: Author var books: [Book] // all associated books } let authorInfos: [AuthorInfo] = try AuthorInfo.fetchAll(db, request)
-
Recommended Practices for Designing Record Types - in this general guide about records, check out the "Associations" chapter.
Joined request defines a tree of associated records identified by "association keys".
Below, author and cover image are both associated to book, and country is associated to author:
let request = Book
.including(required: Book.author
.including(optional: Author.country))
.including(optional: Book.coverImage)
This request builds the following tree of association keys:
Requests can feed record types whose property names match those association keys:
struct BookInfo: FetchableRecord, Decodable {
var book: Book
var author: Author
var country: Country?
var coverImage: CoverImage?
}
let bookInfos: [BookInfo] = try BookInfo.fetchAll(db, request)
By default, association keys are the names of the database tables of associated records. Keys are automatically singularized or pluralized, depending of the cardinality of the included association:
extension Author {
static let books = hasMany(Book.self)
}
Author.including(all: Author.books) // association key "books"
extension Book {
static let author = belongsTo(Author.self)
}
Book.including(required: Book.author) // association key "author"
Keys can be customized when the association is defined:
extension Employee {
static let manager = belongsTo(Employee.self, key: "manager")
}
Employee.including(optional: Employee.manager) // association key "manager"
Keys can also be customized with the forKey
method:
extension Author {
static let novels = books
.filter(Column("kind") == "novel")
.forKey("novels")
}
Author.including(all: Author.novels) // association key "novels"
When association keys match the property names of a Decodable record, you get free decoding of joined requests into this record:
let request = Book
.including(required: Book.author
.including(optional: Author.country))
.including(optional: Book.coverImage)
struct BookInfo: FetchableRecord, Decodable {
var book: Book
var author: Author
var country: Country?
var coverImage: CoverImage?
}
let bookInfos: [BookInfo] = try BookInfo.fetchAll(db, request)
We see that a hierarchical tree has been flattened in the BookInfo
record.
But sometimes your decoded records will have better reflect the hierarchical structure of the request:
Some requests are better decoded with a Decodable record that reflects the hierarchical structure of the request.
let request = Book
.including(optional: Book.coverImage)
.including(required: Book.author
.including(optional: Person.country))
.including(optional: Book.translator
.including(optional: Person.country))
This requests for all books, with their cover images, and their authors and translators. Those people are themselves decorated with their respective nationalities.
We plan to decode this request into is the following nested record:
struct BookInfo: FetchableRecord, Decodable {
struct PersonInfo: Decodable {
var person: Person
var country: Country?
}
var book: Book
var authorInfo: PersonInfo
var translatorInfo: PersonInfo?
var coverImage: CoverImage?
}
This request needs a little preparation: we need association keys that match the coding keys for the authorInfo and translatorInfo properties.
And who is the most able to know those coding keys? BookInfo itself, thanks to its CodingKeys
enum that was automatically generated by the Swift compiler. We thus define the BookInfo.all()
method that builds our request:
extension BookInfo {
static func all() -> QueryInterfaceRequest<BookInfo> {
Book.including(optional: Book.coverImage)
.including(required: Book.author
.forKey(CodingKeys.authorInfo) // (1)
.including(optional: Person.country))
.including(optional: Book.translator
.forKey(CodingKeys.translatorInfo) // (1)
.including(optional: Person.country))
.asRequest(of: BookInfo.self) // (2)
}
}
let bookInfos = try BookInfo.all().fetchAll(db, request) // [BookInfo]
- The
forKey(_:)
method changes the association key, so that the associated records can feed their target properties. - The
asRequest(of:)
method turns the request into a request of BookInfo. See Custom Requests for more information.
When Decodable records provides convenient decoding of joined rows, you may want a little more control over row decoding.
The init(row:)
initializer of the FetchableRecord protocol is what you look after:
let request = Book
.including(required: Book.author
.including(optional: Author.country))
.including(optional: Book.coverImage)
struct BookInfo: FetchableRecord {
var book: Book
var author: Author
var country: Country?
var coverImage: CoverImage?
init(row: Row) throws {
book = try Book(row: row)
author = row["author"]
country = row["country"]
coverImage = row["coverImage"]
}
}
let bookInfos: [BookInfo] = try BookInfo.fetchAll(db, request)
You are already familiar with row subscripts to decode database values:
let name: String = row["name"]
When you extract a record instead of a value from a row, GRDB looks up the tree of association keys. If the key is not found, or only associated with columns that all contain NULL values, an optional record is decoded as nil:
let author: Author = row["author"]
let country: Country? = row["country"]
You can also perform custom navigation in the tree by using row scopes. See Row Adapters for more information.
When you use the include(all:)
method, you can decode an Array or a Set of records:
let request = Author.including(all: Author.books)
struct AuthorInfo: FetchableRecord {
var author: Author
var books: [Book]
init(row: Row) throws {
author = try Author(row: row)
books = row["books"]
}
}
let authorInfos: [AuthorInfo] = try AuthorInfo.fetchAll(db, request)
When you have difficulties building a Decodable record that successfully decodes a joined request, we advise to temporarily decode raw database rows, and inspect them.
let request = Book
.including(required: Book.author
.including(optional: Author.country))
.including(optional: Book.coverImage)
.including(all: Book.prizes)
let rows = try Row.fetchAll(db, request)
print(rows[0].debugDescription)
// Prints:
// ▿ [id:1, authorId:2, title:"Moby-Dick"]
// unadapted: [id:1, authorId:2, title:"Moby-Dick", id:2, name:"Herman Melville", countryCode:"US", code:"US", name:"United States of America", id:NULL, imageId:NULL, path:NULL]
// - author: [id:2, name:"Herman Melville", countryCode:"US"]
// - country: [code:"US", name:"United States of America"]
// - coverImage: [id:NULL, imageId:NULL, path:NULL]
// + prizes: 3 rows
Watch in the row debugging description:
- the association keys: "person", "country", "coverImage" and "prizes" in our example
- associated rows that contain only null values ("coverImage", above).
The associated rows that contain only null values are easy to deal with: null rows loaded from optional associated records should be decoded into Swift optionals:
struct BookInfo: FetchableRecord, Decodable {
var book: Book
var author: Author // .including(required: Book.author)
var country: Country? // .including(optional: Author.country)
var coverImage: CoverImage? // .including(optional: Book.coverImage)
var prizes: [Prize] // .including(all: Book.prizes)
}
When the association keys don't match your expectations, change them (see The Structure of a Joined Request):
let request = Book
.including(optional: Book.author.forKey("writer")) // customized association key
let rows = try Row.fetchAll(db, request)
print(rows[0].debugDescription)
// Prints:
// ▿ [id:1, authorId:2, title:"Moby-Dick"]
// unadapted: [id:1, authorId:2, title:"Moby-Dick", id:2, name:"Herman Melville"]
// - writer: [id:2, name:"Herman Melville", countryCode:"US"]
It is possible to fetch aggregated values from HasMany and HasManyThrough associations:
Counting associated records, fetching the minimum, maximum, average value of an associated record column, computing the sum of an associated record column, these are all aggregation operations.
When you need to compute aggregates from a single record, you use regular aggregating methods on requests for associated records. For example:
struct Author: TableRecord, EncodableRecord {
static let books = hasMany(Book.self)
var books: QueryInterfaceRequest<Book> {
request(for: Author.books)
}
}
let author: Author = ...
// The number of books by this author
let bookCount = try author.books.fetchCount(db) // Int
// The year of the most recent book by this author
let request = author.books.select(max(yearColumn))
let maxBookYear = try Int.fetchOne(db, request) // Int?
When you need to compute aggregates from several record, in a single shot, you'll use an association aggregate. Those are the topic of this chapter.
For example, you'll use the isEmpty
aggregate when you want, say, to fetch all authors who wrote no book at all, or some books:
let lazyAuthors = try Author
.having(Author.books.isEmpty)
.fetchAll(db) // [Author]
let productiveAuthors: [Author] = try Author
.having(Author.books.isEmpty == false)
.fetchAll(db) // [Author]
And you'll use the count
aggregate in order to fetch all authors along with the number of books they wrote:
struct AuthorInfo: Decodable, FetchableRecord {
var author: Author
var bookCount: Int
}
let request = Author.annotated(with: Author.books.count)
let authorInfos: [AuthorInfo] = try AuthorInfo.fetchAll(db, request)
for info in authorInfos {
print("\(info.author.name) wrote \(info.bookCount) book(s).")
}
HasMany and HasManyThrough associations let you build the following association aggregates:
books.count
books.isEmpty
books.min(column)
books.max(column)
books.average(column)
books.sum(column)
books.total(column)
The annotated(with:)
method appends aggregated values to the selected columns of a request. You can append as many aggregates values as needed, from one or several associations.
In order to access those values, you fetch records that have matching properties.
For example:
struct AuthorInfo: Decodable, FetchableRecord {
var author: Author
var bookCount: Int
var maxBookYear: Int?
}
// SELECT author.*,
// COUNT(DISTINCT book.id) AS bookCount,
// MAX(book.year) AS maxBookYear,
// FROM author
// LEFT JOIN book ON book.authorId = author.id
// GROUP BY author.id
let request = Author.annotated(with:
Author.books.count,
Author.books.max(Column("year")))
let authorInfos: [AuthorInfo] = try AuthorInfo.fetchAll(db, request)
for info in authorInfos {
print(info.author.name)
print("- number of books: \(info.bookCount)")
print("- last book published on: \(info.maxBookYear)")
}
As seen in the above example, aggregated values are given a default name, such as "bookCount" or "maxBookYear", which directly feeds the decoded records.
The default name is built from the aggregating method, the association key, and the aggregated column name:
Method | Association Key | Aggregated Column | Aggregate name |
---|---|---|---|
Author.books.isEmpty . |
books |
- | hasNoBook |
Author.books.count . |
books |
- | bookCount |
Author.books.min(Column("year")) |
books |
year |
minBookYear |
Author.books.max(Column("year")) |
books |
year |
maxBookYear |
Author.books.average(Column("price")) |
books |
price |
averageBookPrice |
Author.books.sum(Column("awards")) |
books |
awards |
bookAwardsSum |
Author.books.total(Column("awards")) |
books |
awards |
bookAwardsSum ¹ |
¹ The default name of the total
aggregate has a Sum
suffix, just like the sum
aggregate. Both compute sums, one with the SUM
SQL function, the other with TOTAL
. See SQLite documentation for the difference between these aggregate functions.
Those default names are lost whenever an aggregate is modified (negated, added, multiplied, whatever).
You can name or rename aggregates with the forKey
method:
struct AuthorInfo: Decodable, FetchableRecord {
var author: Author
var numberOfBooks: Int
}
let numberOfBooks = Author.books.count.forKey("numberOfBooks") // <--
let request = Author.annotated(with: numberOfBooks)
let authorInfos: [AuthorInfo] = try AuthorInfo.fetchAll(db, request)
struct AuthorInfo: Decodable, FetchableRecord {
var author: Author
var hasBooks: Bool
}
let hasBooks = (Author.books.isEmpty == false).forKey("hasBooks") // <--
let request = Author.annotated(with: hasBooks)
let authorInfos: [AuthorInfo] = try AuthorInfo.fetchAll(db, request)
struct AuthorInfo: Decodable, FetchableRecord {
var author: Author
var workCount: Int
}
let workCount = (Author.books.count + Author.paintings.count).forKey("workCount") // <--
let request = Author.annotated(with: workCount)
let authorInfos: [AuthorInfo] = try AuthorInfo.fetchAll(db, request)
Coding keys are also accepted:
struct AuthorInfo: Decodable, FetchableRecord {
var author: Author
var numberOfBooks: Int
static func all() -> QueryInterfaceRequest<AuthorInfo> {
let numberOfBooks = Author.books.count.forKey(CodingKey.numberOfBooks) // <--
return Author
.annotated(with: numberOfBooks)
.asRequest(of: AuthorInfo.self)
}
}
let authorInfos: [AuthorInfo] = try AuthorInfo.all().fetchAll(db)
The having(_:)
method filters a request according to an aggregated value. You can append as many aggregate conditions as needed, from one or several associations.
-
Authors who did not write any book:
SQL
SELECT author.* FROM author LEFT JOIN book ON book.authorId = author.id GROUP BY author.id HAVING COUNT(DISTINCT book.id) = 0
let request = Author.having(Author.books.isEmpty)
-
Authors who wrote at least one book:
SQL
SELECT author.* FROM author LEFT JOIN book ON book.authorId = author.id GROUP BY author.id HAVING COUNT(DISTINCT book.id) > 0
let request = Author.having(Author.books.isEmpty == false)
-
Authors who wrote at least two books:
SQL
SELECT author.* FROM author LEFT JOIN book ON book.authorId = author.id GROUP BY author.id HAVING COUNT(DISTINCT book.id) >= 2
let request = Author.having(Author.books.count >= 2)
-
Authors who wrote at least one book after 2010:
SQL
SELECT author.* FROM author LEFT JOIN book ON book.authorId = author.id GROUP BY author.id HAVING MAX(book.year) >= 2010
let request = Author.having(Author.books.max(Column("year")) >= 2010)
-
Authors who wrote at least one book of kind "novel":
SQL
SELECT author.* FROM author LEFT JOIN book ON book.authorId = author.id AND book.kind = 'novel' GROUP BY author.id HAVING COUNT(DISTINCT book.id) > 0
let novels = Author.books.filter(Column("kind") == "novel") let request = Author.having(novels.isEmpty == false)
-
Authors who wrote more books than they made paintings:
SQL
SELECT author.* FROM author LEFT JOIN book ON book.authorId = author.id LEFT JOIN painting ON painting.authorId = author.id GROUP BY author.id HAVING COUNT(DISTINCT book.id) > COUNT(DISTINCT painting.id)
let request = Author.having(Author.books.count > Author.paintings.count)
-
Authors who wrote no book, but made at least one painting:
SQL
SELECT author.* FROM author LEFT JOIN book ON book.authorId = author.id LEFT JOIN painting ON painting.authorId = author.id GROUP BY author.id HAVING ((COUNT(DISTINCT book.id) = 0) AND (COUNT(DISTINCT painting.id) > 0))
let request = Author.having(Author.books.isEmpty && !Author.paintings.isEmpty)
Aggregates can be modified and combined with Swift operators:
-
Logical operators
&&
,||
and!
SQL
SELECT author.* FROM author LEFT JOIN book ON book.authorId = author.id LEFT JOIN painting ON painting.authorId = author.id GROUP BY author.id HAVING ((COUNT(DISTINCT book.id) = 0) AND (COUNT(DISTINCT painting.id) = 0))
let condition = Author.books.isEmpty && Author.paintings.isEmpty let request = Author.having(condition)
-
Comparison operators
<
,<=
,=
,!=
,>=
,>
SQL
SELECT author.* FROM author LEFT JOIN book ON book.authorId = author.id GROUP BY author.id HAVING MAX(book.year) >= 2010
let request = Author.having(Author.books.max(Column("year")) >= 2010)
-
Arithmetic operators
+
,-
,*
,/
SQL
SELECT author.*, (COUNT(DISTINCT book.id) + COUNT(DISTINCT painting.id)) AS workCount FROM author LEFT JOIN book ON book.authorId = author.id LEFT JOIN painting ON painting.authorId = author.id GROUP BY author.id
let workCount = Author.books.count + Author.paintings.count) let request = Author.annotated(with: workCount.forKey("workCount"))
-
IFNULL operator
??
SQL
SELECT "team".*, IFNULL(MIN("player"."score"), 0) AS "minPlayerScore" FROM "team" LEFT JOIN "player" ON ("player"."teamId" = "team"."id") GROUP BY "team"."id"
let request = Team.annotated(with: Team.players.min(Column("score")) ?? 0)
-
SQL functions
ABS
,CAST
, andLENGTH
are available as theabs
,cast
, andlength
Swift functions:SQL
SELECT "team".*, ABS(MAX("player"."score")) FROM "team" LEFT JOIN "player" ON ("player"."teamId" = "team"."id") GROUP BY "team"."id"
let request = Team.annotated(with: abs(Team.players.max(Column("score"))))
When you compute multiple aggregates, make sure they use as many distinct association keys as there are distinct populations of associated records.
In the example below, we use compute two aggregates from the same association Author.books
. Both aggregates are computed on the same population of associated records, and so we want them to share the same association key:
-
Authors with the publishing year of their first and last book:
SQL
SELECT author.*, MIN(book.year) AS minBookYear, MAX(book.year) AS maxBookYea FROM author LEFT JOIN book ON book.authorId = author.id GROUP BY author.id
struct Author: TableRecord { static let books = hasMany(Book.self) // association key "books" } struct AuthorInfo: Decodable, FetchableRecord { var author: Author var minBookYear: Int? var maxBookYear: Int? } let request = Author.annotated(with: Author.books.min(Column("year")), // association key "books" Author.books.max(Column("year"))) // association key "books" let authorInfos: [AuthorInfo] = try AuthorInfo.fetchAll(db, request)
In this other example, the Author.books
and Author.paintings
have the distinct book
and painting
keys. They don't interfere, and provide the expected results:
-
Authors with their number of books and paintings:
SQL
SELECT author.*, (COUNT(DISTINCT book.id) + COUNT(DISTINCT painting.id)) AS workCount FROM author LEFT JOIN book ON book.authorId = author.id LEFT JOIN painting ON painting.authorId = author.id GROUP BY author.id
struct Author: TableRecord { static let books = hasMany(Book.self) // association key "books" static let paintings = hasMany(Painting.self) // association key "paintings" } struct AuthorInfo: Decodable, FetchableRecord { var author: Author var workCount: Int } let aggregate = Author.books.count + // association key "books" Author.paintings.count // association key "paintings" let request = Author.annotated(with: aggregate.forKey("workCount")) let authorInfos: [AuthorInfo] = try AuthorInfo.fetchAll(db, request)
But in the following example, we use the same association Author.books
twice, in order to compute aggregates on two distinct populations of associated books. We must provide explicit keys in order to make sure both aggregates are computed independently:
-
Authors with their number of novels and theatre plays:
SQL
SELECT author.*, COUNT(DISTINCT book1.id) AS novelCount, COUNT(DISTINCT book2.id) AS theatrePlayCount FROM author LEFT JOIN book book1 ON book1.authorId = author.id AND book1.kind = 'novel' LEFT JOIN book book2 ON book2.authorId = author.id AND book2.kind = 'theatrePlay' GROUP BY author.id
struct Author: TableRecord { static let books = hasMany(Book.self) // association key "books" } struct AuthorInfo: Decodable, FetchableRecord { var author: Author var novelCount: Int var theatrePlayCount: Int } let novelCount = Author.books .filter(Column("kind") == "novel") .forKey("novels") // association key "novels" .count let theatrePlayCount = Author.books .filter(Column("kind") == "theatrePlay") .forKey("theatrePlays") // association key "theatrePlays" .count let request = Author.annotated(with: novelCount, theatrePlayCount) let authorInfos: [AuthorInfo] = try AuthorInfo.fetchAll(db, request)
When one doesn't use distinct association keys for novels and theatre plays, GRDB will not count two distinct sets of associated books, and will not fetch the expected results:
SQL
SELECT author.*, COUNT(DISTINCT book.id) AS novelCount, COUNT(DISTINCT book.id) AS theatrePlayCount FROM author LEFT JOIN book ON book.authorId = author.id AND (book.kind = 'novel' AND book.kind = 'theatrePlay') GROUP BY author.id
// WRONG: not counting distinct sets of associated books let novelCount = Author.books // association key "books" .filter(Column("kind") == "novel") .count .forKey("novelCount") let theatrePlayCount = Author.books // association key "books" .filter(Column("kind") == "theatrePlay") .count .forKey("theatrePlayCount") let request = Author.annotated(with: novelCount, theatrePlayCount) let authorInfos: [AuthorInfo] = try AuthorInfo.fetchAll(db, request)
The DerivableRequest
protocol is adopted by both query interface requests such as Author.all()
and associations such as Book.author
. It is intended for you to use as a customization point when you want to extend the built-in GRDB apis.
For example, we may want to define orderedByName()
and filter(country:)
request methods that make our requests easier to read:
// Authors sorted by name
let request = Author.all().orderedByName()
// French authors ordered by name
let request = Author.all().filter(country: "FR").orderedByName()
// Spanish books
let request = Book.all().filter(country: "ES")
Those methods are defined on extensions to the DerivableRequest
protocol:
extension DerivableRequest<Author> {
func filter(country: String) -> Self {
filter(Column("country") == country)
}
func orderedByName() -> Self {
order(Column("name").collating(.localizedCaseInsensitiveCompare))
}
}
extension DerivableRequest<Book> {
func filter(country: String) -> Self {
joining(required: Book.author.filter(country: country))
}
}
See Recommended Practices for Designing Record Types for more information.
-
You can't chain a required association on an optional association:
// NOT IMPLEMENTED let request = Book .joining(optional: Book.author .including(required: Person.country))
This code compiles, but you'll get a runtime fatal error "Not implemented: chaining a required association behind an optional association". Future versions of GRDB may allow such requests.
-
You can't use the
including(all:)
method and use table aliases to filter the associated records on other records:// NOT IMPLEMENTED: loading all authors along with their posthumous books let authorAlias = TableAlias() let request = Author .aliased(authorAlias) .including(all: Author.books .filter(Column("publishDate") >= authorAlias[Column("deathDate")]))
-
You can't use the
including(all:)
method with a HasMany and a HasManyThrough associations that share the same base association in the same request:// NOT IMPLEMENTED let request = Country .including(all: Country.passports) .including(all: Country.citizens)
This code compiles, but you'll get a runtime fatal error "Not implemented: merging a direct association and an indirect one with including(all:)". Future versions of GRDB may allow such requests.
The workaround is to nest the most remote association:
// Workaround let request = Country .including(all: Country.passports .including(required: Passport.citizen))
This documentation owns a lot to the Active Record Associations guide, which is an immensely well-written introduction to database relations. Many thanks to the Rails team and contributors.
GRDB
Copyright (C) 2015-2023 Gwendal Roué
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
Ruby on Rails documentation
Copyright (c) 2005-2018 David Heinemeier Hansson
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.