Query API
Loaders
Loaders are methods for each table that perform super fast batched and cached loads on indexed columns. This is designed to minimise performance issues from the n+1 problem.
Loaders are the primary way of retrieving data using Gybson.
loadOne
loadOne(...)
returns a single record or null.
loadOne
filters by a unique key (often primary key).
A compound unique key example:
loadMany
loadMany(...)
returns an array of records.
loadMany
filters on non-unique key columns (often foreign keys).
An order can be specified:
findMany
Similar to loadMany
, findMany
loads many rows from a table. The difference is that findMany
provides a very flexible query API including relation filtering.
Due to this flexibility, findMany
does not perform batching or caching.
tip
When building GraphQL APIs, we recommend only using findMany
near the top of your query tree.
When used deeper in the tree you should be wary of performance issues due to the n+1 problem.
where
The where
field is used to specify filters on the rows returned.
You can filter by any combination of columns and relations. For all filtering options see Filtering
i.e. Find all users where age is not 10 and they have a pet with the type "dog".
orderBy
The orderBy
field is used to specify the order rows should be returned in.
You can specify one or more ordering columns as well as the order direction (ascending, descending)
i.e. Find all users ordered by first_name and last_name in descending order, age in ascending order.
paginate
The paginate
field is used to specify a subset of matching rows to return.
Both offset-limit
and cursor
pagination is supported.
When using pagination, you should always specify and orderBy
clause as well.
Offset-limit pagination
Returns rows before or after a specific number of rows.
Offset-limit pagination is simpler than cursor pagination but does not work as well for tables that change often.
i.e. Find the first 4 users in ascending last_name order after row 300.
Cursor pagination
Returns rows before or after a specific row. You can use any column (or combination) as your cursor.
Cursor pagination is generally better than offset-limit as it still works the number of rows in the table can change.
note
When using cursor pagination, you should specify the same columns in orderBy as your cursor. If you don't you may get unexpected result sets.
i.e. Find the first 4 users in ascending order with a last_name after 'Jones'.
insert
Inserts one or more rows into the database. This will automatically apply DEFAULT values for any
columns that are undefined
.
Insert returns the id of the first row inserted.
insert a single row
insert multiple rows
You can also choose to skip duplicate rows during a multi-row insert:
upsert
Inserts multiple row into the database. If a row already exists with the same primary key, the row will be updated.
tip
Upsert is implemented using native SQL functionality: ON CONFLICT DO in Postgres ON DUPLICATE KEY UPDATE in MySQL
Because of this, the databases may behave differently with the same query. For instance, MySQL will upsert on any unique constraint conflict. PostgreSQL will only conflict on the primary key.
In the case of conflicts, you can specify how you want to update the existing row(s):
mergeColumns
will overwrite the values on the existing row with the values of the new row for the selected columns:
Will update only the first_name
and age
values if the row already exists.
update
will update existing rows with the specifed values:
Will set the users first_name to John 2
if they already exist.
Upsert multiple rows
Upsert with soft-deletes
You can also specify whether to reinstate (remove soft delete) on a row that has previously been soft-deleted:
update
Updates all rows that match a filter.
update
supports all where filter options.
softDelete
A soft-delete allows you to treat a row as deleted while maintaining it for record-keeping or recovery purposes.
By default any tables with a column deleted
or deleted_at
support soft deletes.
softDelete
will set the deleted
column to true or the current timestamp and cause the row to be filtered from future queries.
softDelete
supports all where filter options.
delete
delete
will delete a row permanently.
delete
supports all where filter options.
transactions
Use _transactions
to run a set of queries as a single atomic query. This means if any
of the queries fail then none of the changes will be committed.
You can include a query in the transaction by passing in the connection
argument.
Manual connection handling
Most query functions allow an optional connection
argument. You can pass a MySQL or PostgreSQL connection
and the query will use it instead of the internal Knex connection.
This can be useful for apps with existing connection handling or more complex transaction handling requirements.
Example with MySQL: