SQL databases are amazing. They give you tools to create extremely powerful guarantees about the integrity of your data, evolve that data over time, and ask some pretty complex questions about it with relatively little effort.
As an example, let's say we have an application that handles questionnaires:
-
Each
Questionnaire
has manySections
-
Each
Section
has manyQuestions
-
Each
Question
can have oneAnswer
We could find all the sections with answers where the person loves bread like this!:
SELECT
sections.title AS section_title,
questions.label AS question_label,
questions.category,
questions.answer
FROM questions
INNER JOIN sections ON questions.section_id = sections.id
WHERE
questions.category = 'BREAD' AND
questions.answer ILIKE '%I LOVE%';
At the same time, using well-structured objects or domain types is also a powerful way of modelling pieces of an application.
class Section(
val title: String,
val questions: List<Question>
) {
fun bestAnswers() = questions.filter { it.hasGreatAnswer() }
}
class Question(
val label: String,
val category: Category,
val answer: Answer?
) {
fun hasGreatAnswer() = category == Category.BREAD && answer?.hasMuchLove() ?: false
}
Combining the two approaches, however, is often more awkward than you'd expect. The awkwardness here is that SQL queries produce flat results
:
But our domain model contains multiple nested structures
(e.g. questionnaires contain sections, which contain questions).
This mismatch in shape is sometimes called object-relational impedance mismatch.
How has this been solved?
Generally, there are two categories of common solutions to this:
- Use raw SQL queries and stitch the relevant data together manually (e.g. query the questions and sections table separately, and then for each matching ID build up a list of questions for each section)
- Use an ORM (Object Relational Mapper)
An ORM effectively abstracts the first option away.
Problems with these approaches
Both approaches have their pros and cons.
Manually stitching
gives you the power and flexibility of raw SQL, but the stitching code can be error-prone and painful to maintain.
Using an ORM
gives you stronger guarantees that the data will be constructed correctly, but you lose a lot of the flexibility of raw SQL (or worse, the complications of mixing the two!). And unless you understand how to use the ORM properly, you can easily run into serious performance problems.
In Java / Kotlin, JPA with Hibernate
is the de-facto framework for dealing with relational databases. It's an impressive and mature framework / standard, but it's also extremely complex and a very high level abstraction (what's going on under the hood can often be extremely opaque).
In my particular case, I ran into a few grievances with it:
- The mapping code could be quite awkward and boilerplate heavy—often having to fish out related objects to make a change.
- Queries with deeply nested objects became very slow, and were very difficult to debug.
-
Magical annotations
@Lazy
@Eager
for fetching strategies and remembering where to put@JoinColumn
and@OneToMany
/@OneToOne
left me scratching my head more often than I'd like to admit.
None of these points were show stoppers, and I'm sure with more time and experience they'd become less painful, but I couldn't help but think—is there a less abstract way to do this?
JSON and Jackson
Of the "magical" things you can do in Java / Kotlin, I was surprised by how painless it was using Jackson
, a serialization/deserialization library for Java. Given a JSON object like:
{
"title": "Section 1",
"questions": [
{
"label": "Do you love bread?",
"category": "BREAD",
"answer": "Absolutely"
},
{
"label": "Do you love baguettes?",
"category": "BREAD",
"answer": null
}
]
}
And Kotlin classes:
class Section(
val title: String,
val questions: List<Question>
)
class Question(
val label: String,
val category: Category,
val answer: Answer?
)
You can magically convert that JSON into a Section
:
jacksonObjectMapper().readValue<Section>(jsonString)
What's effectively happening is the Jackson object mapper is taking the name of each object key and matching it with an argument to the constructor of the relevant class.
Hasura GraphQL Engine
At the time I'd been working on a side project that uses Hasura—a GraphQL server that can auto generate an API based from a Postgres schema. I stumbled across a blog post explaining the architecture behind its high performance—taking data from Postgres and returning deeply nested JSON structures for GraphQL responses (it's a fantastic blog post and well worth a read).
At first, they queried the flat results and used a transformation function in Haskell to turn that data into JSON. However, they found that actually using Postgres' JSON aggregation functions produced a 3-6x improvement in speed!
This got me thinking.
We were using Postgres on our project, and Jackson painlessly converts JSON to Kotlin objects. Can we combine the two somehow?
JSON aggregate functions
It's definitely not perfect, but It was a lot less painful than I'd expected. Here's a query for getting sections:
SELECT
json_agg(
json_build_object(
'title', sections.title,
'questions', questions.json
)
)
FROM sections
INNER JOIN (
SELECT
questions.section_id,
json_agg(
json_build_object(
'label', questions.label,
'category', questions.category,
'answer', questions.answer
)
) AS json
FROM questions
GROUP BY questions.section_id
) questions ON questions.section_id = sections.id;
Whilst it might not be to everyone's taste, once the nesting gets deeper you could split the strings up into smaller queries:
@Language("SQL")
const val getQuestions = """
SELECT
questions.section_id,
json_agg(
json_build_object(
'label', questions.label,
'category', questions.category,
'answer', questions.answer
)
) AS json
FROM questions
GROUP BY questions.section_id
"""
@Language("SQL")
const val getSections = """
SELECT
json_agg(
json_build_object(
'title', sections.title,
'questions', questions.json
)
) AS output
FROM sections
INNER JOIN ($getQuestions) questions ON questions.section_id = sections.id
"""
This gives back a single row from postgres:
output: [ { "title": "Section 1", "questions": [ ... ] } ]
Jackson can then take the output string and parse it into a List<Section>
.
Sealed Classes
Sealed classes, in my opinion, are one of the best features of Kotlin—they let you model a restricted subset of a class where all members are known at compile time (it's a similar concept to algebraic data types seen in some typed functional languages).
Say an answer to a question got more complicated and needed the concept of being approved by a reviewer. You could do something like this:
sealed class Answer {
object NotAnswered() : Answer()
class Answered(val answer: String) : Answer()
class Approved(val answer: String, val reviewer: Username) : Answer()
}
Now whenever we use an Answer
in the application, we have to handle all the possible cases explicitly:
when (answer) {
is NotAnswered -> doSomething()
is Answered -> doSomethingWithAnswer(answer.answer)
is Approved -> doSomethingWithReviewer(answer.reviewer)
}
Jackson won't read sealed classes by default, but we can write a custom Deserializer
to create an Answer
from JSON:
class AnswerDeserializer : JsonDeserializer<Answer>() {
override fun deserialize(p: JsonParser, ctxt: DeserializationContext): Answer =
p.readValueAs(Intermediate::class.java).toAnswer()
class Intermediate(
val answer: String?,
val reviewedBy: String?
) {
fun toAnswer(): Answer = when {
answer != null && reviewedBy != null -> Approved(answer, reviewedBy)
answer != null && reviewedBy == null -> Answered(answer)
else -> NotAnswered
}
}
}
And modify our query to have this intermediate structure:
@Language("SQL")
const val getQuestions = """
SELECT
questions.section_id,
json_agg(
json_build_object(
'label', questions.label,
'category', questions.category,
'answer', json_build_object(
'answer', questions.answer,
'reviewedBy', questions.reviewedBy
)
)
) json
FROM questions
GROUP BY questions.section_id
"""
Registering the deserializer on the Jackson object mapper means it can recognise Answers
now.
There are some potential downsides to this approach to watch out for, however:
- You're tied to Postgres (or a database that supports JSON aggregates).
- It's worth taking care to not get overzealous in splitting up queries!
- IntelliJ has some amazing inline autocompletion and schema detection, but only in the paid for ultimate edition.
This combination has been surprisingly effective on our current project though—it's made database code more transparent and sped up nested queries significantly. One query that was taking ~5s went down to ~100ms! (I'm sure we were misusing hibernate horribly!)
You can find a full working example here.