More complex queries
Norm allows you to sort and limit rows and use subqueries for complex WHERE
clauses. To understand how, it helps to keep in mind that Norm essentially generates SQL SELECT
queries after the following pattern: SELECT <fields of model> FROM <table-name specified by model> WHERE <condition>
This means that whatever pieces of SQL come after the WHERE
keyword are thing you can freely specify if need be.
Limiting the number of queried models
To limit the number of queried models, simply use SQL's LIMIT
keyword.
Let's query our Customer
table from earlier and query multiple entries, but only take the first entry:
var
userFoo = newUser("foo@foo.foo")
alice = newCustomer(some "Alice", userFoo)
bob = newCustomer(some "Bob", userFoo)
with dbConn:
insert userFoo
insert alice
insert bob
var customersFoo = @[newCustomer()]
dbConn.select(customersFoo, "User.email = ? LIMIT 1", "foo@foo.foo")
assert customersFoo.len() == 1
echo()
INSERT INTO "User" (email) VALUES(?) <- @['foo@foo.foo'] Object ID is not 0, skipping insertion. Type: User, ID: 1 INSERT INTO "Customer" (name, user) VALUES(?, ?) <- @['Alice', 1] Object ID is not 0, skipping insertion. Type: User, ID: 1 INSERT INTO "Customer" (name, user) VALUES(?, ?) <- @['Bob', 1] SELECT "Customer".name, "Customer".user, "user".email, "user".id, "Customer".id FROM "Customer" LEFT JOIN "User" AS "user" ON "Customer".user = "user".id WHERE User.email = ? LIMIT 1 <- ['foo@foo.foo']
customersFoo
has only 1 entry, despite alice
and bob
both having the email address "foo@foo.foo"
, thanks to the LIMIT
SQL keyword.
Sorting model output
We can of course use ORDER BY
just as we did LIMIT
before:
var sortedCustomersFoo = @[newCustomer()]
dbConn.select(sortedCustomersFoo, "User.email = ? ORDER BY name DESC", "foo@foo.foo")
assert sortedCustomersFoo[0].name.get() == "Bob"
assert sortedCustomersFoo[1].name.get() == "Alice"
echo()
SELECT "Customer".name, "Customer".user, "user".email, "user".id, "Customer".id FROM "Customer" LEFT JOIN "User" AS "user" ON "Customer".user = "user".id WHERE User.email = ? ORDER BY name DESC <- ['foo@foo.foo']
Using Subqueries
Similarly as to ORDER BY
, you can also use subqueries within the WHERE
block:
var subqueryCustomersFoo = @[newCustomer()]
const condition = """
Customer.id IN (SELECT Cust.id FROM Customer AS Cust WHERE Cust.id % 2 == 0)
"""
dbConn.select(subqueryCustomersFoo, condition)
assert subqueryCustomersFoo.len() == 1
assert subqueryCustomersFoo[0].id == 2
echo()
SELECT "Customer".name, "Customer".user, "user".email, "user".id, "Customer".id FROM "Customer" LEFT JOIN "User" AS "user" ON "Customer".user = "user".id WHERE Customer.id IN (SELECT Cust.id FROM Customer AS Cust WHERE Cust.id % 2 == 0) <- []