Advanced ActiveRecord

Learning Goals

  • Diagram database relationships.
  • Identify the tables in a database that hold information required to complete complex queries.
  • Generate complex ActiveRecord queries using joins, group, order, select, and merge.
  • Use the rails dbconsole and rails console to generate ActiveRecord queries.

Resources

  • Video from a past class and the core ideas

Objective

Find the 5 most expensive invoices with successful transactions.

Warmup (5 mins)

  • What tables would we need to query?
  • What information would we need from each table?
  • What calculations would we need to perform?
  • What SQL would we be able to use to create a table with this information?

Lecture

WARNING: This lesson is most beneficial if you don’t follow along in class.

Creating a Visual Representation

Have you ever attempted to build a complex ActiveRecord query by throwing spaghetti at the wall? Eventually something will stick. But there’s a better way.

As you start to dive into more difficult problems, problems that have many moving parts, it will become increasingly difficult to hold all of the relevant information in your head at one time. Additionally, the more complex the problem, the easier it is to slowly drift away from the goal and building the wrong thing. Starting with a clear goal and a plan will increase the likelihood of you hitting your target in a timely fashion.

There are official ways of diagramming SQL queries but we’re going to do something a little different. This technique isn’t an official technique so feel free to tweak or find something that works for you. If you’re interested in diving deeper into a more official strategy check out this blog post on Introductory SQL Diagramming.

We’ll start by sketching out all of the columns we need including the table the column is located in.

relevant database tables and columns

Next up, let’s make a note about all of the things we think need to happen with each of these columns.

tables and columns with SQL terms and why we need them

Converting to ActiveRecord

As we work through the next section we should update the diagram if our understanding changes or if we successfully accomplish one of our goals. We can do something simple like adding a checkmark when we successfully account for each item we documented. When complete we could have a board that looks like this:

query plan with checkmarks

To start through this process let’s fire up a Rails console. We’ll start by building our query slowly, piece by piece. Figuring out where to start is often a struggle. When we are writing a query that needs to return specific rows, invoices in this case, we should start by writing our query from the Model representing those rows. Sometimes, there will be a temptation to start with a join table since they are similar to a hub with spokes that branch out to the tables we need. Avoid this temptation. This will usually result in needing to make more queries than is necessary.

The next thing that is usually good to try is to tack on the easiest portions of the query first and read the output in the console to make sure it matches our expectations. One way to make this easier to read is to use the .to_sql method.

irb(main)> puts Invoice.joins(:invoice_items).to_sql

We added the puts to make it easier to read which remove escape characters.

The output should look something like this:

SELECT "invoices".* FROM "invoices" INNER JOIN "invoice_items" ON "invoice_items"."invoice_id" = "invoices"."id"

This looks the way we would expect. Another great tool to use as we’re trying to solve these issues is the rails db command line tool which will connect us to the database specified in our database.yml file. Let’s fire that up in another tab in our terminal. It’s worth noting that this is going to have access to any SQL commands in a similar way to using something like psql but it does not have access to the Rails environment so we won’t be able to write any Ruby. This can be a nice place to run your queries and have a table based representation of the output. This can be especially handy when you start adding aliases and things like that. If we paste that query, add on a semi-colon, and run it we should see something like this.

rales_engine_development=# SELECT "invoices".* FROM "invoices" INNER JOIN "invoice_items" ON "invoice_items"."invoice_id" = "invoices"."id";
  id  | customer_id | merchant_id | status  |     created_at      |     updated_at
------+-------------+-------------+---------+---------------------+---------------------
    1 |           1 |          26 | shipped | 2012-03-25 09:54:09 | 2012-03-25 09:54:09
    1 |           1 |          26 | shipped | 2012-03-25 09:54:09 | 2012-03-25 09:54:09
    1 |           1 |          26 | shipped | 2012-03-25 09:54:09 | 2012-03-25 09:54:09
    1 |           1 |          26 | shipped | 2012-03-25 09:54:09 | 2012-03-25 09:54:09
    1 |           1 |          26 | shipped | 2012-03-25 09:54:09 | 2012-03-25 09:54:09
    1 |           1 |          26 | shipped | 2012-03-25 09:54:09 | 2012-03-25 09:54:09
    1 |           1 |          26 | shipped | 2012-03-25 09:54:09 | 2012-03-25 09:54:09
    1 |           1 |          26 | shipped | 2012-03-25 09:54:09 | 2012-03-25 09:54:09
    2 |           1 |          75 | shipped | 2012-03-12 05:54:09 | 2012-03-12 05:54:09

It’s important to make sure we are seeing what we would expect. Why do we see multiples of the same record?

This looks the way that we would expect. Mark it off on the diagram. Then, let’s get to the next part of the query.

Invoice.joins(:invoice_items, :transactions)
  1. Confirm the SQL looks the way it should.
  2. Mark it off on our diagram.

Now we’ll tack on the where method to filter by transactions that have a result of success. There’s two ways we can do this. Let’s take a look at each option.

Option 1. We can specify the table (transactions) and column (result) as well as the value we want to filter by (success).

Adding where

Invoice.joins(:invoice_items, :transactions).where("transactions.result = ?", "success")

Option 2: Recall that every ActiveRecord method we chain here is executed from the perspective of starting at the Invoice model. That being the case, we can leverage the relationships set up in that model for queries like this. Before we dive in, understand that Invoice.where(status: "shipped") is the same as Invoice.where({status: "shipped"}). We don’t need the curly braces in the first example because we are only referencing one table. When traversing across multiple tables however, we will need to include the curly braces like so…

Invoice.joins(:invoice_items, :transactions).where(transactions: {result: "success"})

Adding select

Our next steps get more complex since we need to run an aggregate function on the quantity and unit_price from the invoice_items table. Let’s first add the select statement to return exactly the same thing as it currently is.

# Broken onto multiple lines for readability
Invoice.select("invoices.*")
  .joins(:invoice_items, :transactions)
  .where(transactions: {result: "success"})

And now for the aggregate function.

# Broken onto multiple lines for readability
Invoice.select("invoices.*, SUM(invoice_items.quantity * invoice_items.unit_price) AS revenue")
  .joins(:invoice_items, :transactions)
  .where(transactions: {result: "success"})

When we run this code we get an error. Let’s take the SQL that’s generated from calling to_sql and paste it into rails db.

Sure enough, we see the same-ish error:

column "invoices.id" must appear in the GROUP BY clause or be used in an aggregate function

What’s happening? Well, since the result of the aggregate function, revenue, isn’t part of the table that invoices.* comes from, Postgres isn’t sure what to do with it. Telling it to group by invoices.id let’s it know how to squish things down into corresponding rows.

Invoice.select("invoices.*, SUM(invoice_items.quantity * invoice_items.unit_price) AS revenue")
  .joins(:invoice_items, :transactions)
  .where(transactions: {result: "success"})
  .group(:id)

Again, the query is starting at Invoice so when we say group(:id) it infers that we mean invoices.id. Unfortunately, we don’t see the revenue appear in our output. But it is there as a virtual attribute. Virtual attributes get created when we have aliases in our select statements. We can access these virtual attributes the same as other columns in the table.

invoices = Invoice.select("invoices.*, SUM(invoice_items.quantity * invoice_items.unit_price) AS revenue")
  .joins(:invoice_items, :transactions)
  .where(transactions: {result: "success"})
  .group(:id)

invoices.first.revenue
# => 2106777

Noice. Sometimes it can be difficult to see whether you are getting back what you want or whether the query is doing what you think it should. This is a great time to call to_sql and paste the result into our rails db console. You should see output that looks like this:

 id  | customer_id | merchant_id | status  |     created_at      |     updated_at      | revenue
------+-------------+-------------+---------+---------------------+---------------------+---------
1489 |         289 |          46 | shipped | 2012-03-26 22:55:20 | 2012-03-26 22:55:20 |  944894
4790 |         988 |          78 | shipped | 2012-03-24 15:58:12 | 2012-03-24 15:58:12 |   48718
273  |          59 |          72 | shipped | 2012-03-08 21:54:23 | 2012-03-08 21:54:23 |  861533
3936 |         802 |          94 | shipped | 2012-03-20 08:57:27 | 2012-03-20 08:57:27 | 1085352
2574 |         521 |          79 | shipped | 2012-03-11 19:56:13 | 2012-03-11 19:56:13 |  975672
951  |         184 |          87 | shipped | 2012-03-19 15:54:55 | 2012-03-19 15:54:55 | 1832399
4326 |         881 |          21 | shipped | 2012-03-14 18:57:48 | 2012-03-14 18:57:48 |  536897
2614 |         526 |           1 | shipped | 2012-03-23 09:56:15 | 2012-03-23 09:56:15 | 1379489

Notice the revenue column. Looks like we need to order these. But before we do that don’t forget to update our diagram with our progress.

ORDER BY and LIMIT

Next, we’ll tack on order.

Invoice.select("invoices.*, SUM(invoice_items.quantity * invoice_items.unit_price) AS revenue")
  .joins(:invoice_items, :transactions)
  .where(transactions: {result: "success"})
  .group(:id)
  .order("revenue DESC")

We can’t pass in :revenue since it’s not a column in the database and therefore we’ll need to send it in as a string and tell it we want to go in descending order. Now we can tack on our limit(5).

Invoice.select("invoices.*, SUM(invoice_items.quantity * invoice_items.unit_price) AS revenue")
  .joins(:invoice_items, :transactions)
  .where(transactions: {result: "success"})
  .group(:id)
  .order("revenue DESC")
  .limit(5)

We should be good. Use to_sql in your Rails console and paste the results in your db console. You should see something like this:

id  | customer_id | merchant_id | status  |     created_at      |     updated_at      | revenue
------+-------------+-------------+---------+---------------------+---------------------+---------
1560 |         300 |          10 | shipped | 2012-03-14 02:55:24 | 2012-03-14 02:55:24 | 4896887
3394 |         697 |          64 | shipped | 2012-03-20 16:56:57 | 2012-03-20 16:56:57 | 4787797
4377 |         893 |          68 | shipped | 2012-03-14 18:57:51 | 2012-03-14 18:57:51 | 4722254
3584 |         729 |          15 | shipped | 2012-03-19 11:57:08 | 2012-03-19 11:57:08 | 4695214
2711 |         543 |          14 | shipped | 2012-03-09 18:56:20 | 2012-03-09 18:56:20 | 4692545

Now we should move this to our Invoice model.

# models/invoice.rb

def self.most_expensive(limit = 5, sorting = "DESC")
  select("invoices.*, SUM(invoice_items.quantity * invoice_items.unit_price) AS revenue")
    .joins(:invoice_items, :transactions)
    .where(transactions: {result: "success"})
    .group(:id)
    .order("revenue #{sorting}")
    .limit(limit)

Notice that we removed Invoice from this query. A wise person once said “If you want to master Ruby, always know what self is.” We’re in a class method in Invoice which means self is Invoice so we can remove that reference. We also set two default parameters for limit and sorting. We do this to make the method more flexible and reusable.

Houston, we have a problem. The idea of the Invoice model tracking what it means for a transaction to be successful is a violation of SRP. We should extract .where(transactions: {result: "success"}) and move it over to the Transaction model. Enter .merge

Merging Scopes

It’s common to want to share model logic and concepts, such as “a successful transaction”, across multiple models. .merge allows us to pull logic from predefined methods called scopes into different portions of our codebase.

Let’s define a scope to capture this idea of a successful transaction in the Transaction model.

# models/transaction.rb

class Transaction < ApplicationRecord
  scope :successful, -> { where(result: "success") }
end

That arrow looking thing is what we in the Ruby community refer to as a stabby lambda. Explaining lambdas is outside the scope of this lesson but here’s more info if you’re interested.

Reload your Rails console to be able to use this scope (reload!). Now run Transaction.successful and take a look at the SQL output. You should see a WHERE clause filtering by what we specified in our lambda.

Now let’s replace where with merge to reuse this bit of logic in Invoice.most_expensive.

# models/invoice.rb

def self.most_expensive(limit = 5, sorting = "DESC")
  select("invoices.*, SUM(invoice_items.quantity * invoice_items.unit_price) AS revenue")
    .joins(:invoice_items, :transactions)
    .merge(Transaction.successful)
    .group(:id)
    .order("revenue #{sorting}")
    .limit(limit)
end

One of the big advantages here is if the concept of what it means to be a successful transaction ever changes we hopefully only need to update that in one place. The original way we wrote this was vulnerable to changes not associated with this particular table or model.

As you can imagine, scopes are widely used on Rails projects and learning to wield their power is important. Be sure to create a successful scope on Transaction and merge it in for your other BI challenges.

But what if you want to have a scope applied to every query called on a model?

default_scope

What order does Postgres return its rows in? Read the first paragraph in this (Sorting Rows)[https://www.postgresql.org/docs/9.1/queries-order.html] documentation.

So then, what order will our results come back in if we call Transaction.all? What about Transaction.limit(10)?

Enter default_scope. This will allow us to apply a scope to every ActiveRecord query. For example:

# models/transaction.rb

class Transaction < ApplicationRecord
  default_scope { order(id: :asc) }

  scope :successful, -> { where(result: "success") }
end

Reload your console and run Transaction.all. You should now see an ORDER BY clause.

A Word of Caution about default_scope

Because most of our business intelligence is running analytics on only the successful transactions you might have had an idea to try something like this:

class Transaction < ApplicationRecord
  default_scope { where(result: "success") }
end

Whoa! It works!? Ship it!

NOT SO FAST. Reload your console if you haven’t already and run Transaction.new. What’s the value of result? Surely you didn’t intend for every new transaction to be set to success. This sort of thing can cause big problems in production apps where it becomes nearly impossible to remember every model with a default_scope set. Some prefer not to use default_scope at all. My personal preference is to limit its use to ordering but even this can have some unintended consequences. Let’s change our code back to what it was.

# models/transaction.rb

class Transaction < ApplicationRecord
  default_scope { order(id: :asc) }

  scope :successful, -> { where(result: "success") }
end

# models/invoice.rb
def self.most_expensive(limit = 5, sorting = "DESC")
  select("invoices.*, SUM(invoice_items.quantity * invoice_items.unit_price) AS revenue")
    .joins(:invoice_items, :transactions)
    .merge(Transaction.successful)
    .group(:id)
    .order("revenue #{sorting}")
    .limit(limit)
end

Reload your console and call Invoice.most_expensive. You should see an error that looks something like this:

ActiveRecord::StatementInvalid: PG::GroupingError: ERROR:  column "transactions.id" must appear in the GROUP BY clause or be used in an aggregate function

When we merge the successful scope in Invoice.most_expensive we now get the default_scope from the Transaction model as well. Sometimes this is good and sometimes this is bad. Right now it’s bad since our query is now attempting to order by the id from the transactions table which we haven’t selected so it blows up. The fix is simple: unscoped.

# models/invoice.rb
def self.most_expensive(limit = 5, sorting = "DESC")
  select("invoices.*, SUM(invoice_items.quantity * invoice_items.unit_price) AS revenue")
    .joins(:invoice_items, :transactions)
    .merge(Transaction.unscoped.successful)
    .group(:id)
    .order("revenue #{sorting}")
    .limit(limit)
end

unscoped removes all scopes that occur before it is called but leaves the scopes that occur after it is intact. This applies to both default_scope and scope. A tradeoff with this approach is if we were merging Transaction.successful in many queries adding default_scope might break them. That said, calling Transaction.order(id: :asc) all over our codebase might be equally bad. YMMV but be aware of the tradeoffs.

When You Need to Run Raw SQL

Sometimes it’s not possible to use ActiveRecord to build the query you need and instead you need to run raw SQL. There are two ways to do this using ActiveRecord, find_by_sql and ActiveRecord.connection.execute.

If you want to get an array of objects instantiated from rows in a table you can use find_by_sql. Let’s see how this works…

sql = Invoice.most_expensive.to_sql
# => "SELECT  invoices.*, SUM(invoice_items.quantity*invoice_items.unit_price) AS revenue FROM \"invoices\" INNER JOIN \"invoice_items\" ON \"invoice_items\".\"invoice_id\" = \"invoices\".\"id\" INNER JOIN \"transactions\" ON \"transactions\".\"invoice_id\" = \"invoices\".\"id\" WHERE \"transactions\".\"result\" = 'success' GROUP BY \"invoices\".\"id\" ORDER BY revenue DESC LIMIT 5"

# sql is simply a string of our query
sql.class
# => String

# Now we run it
Invoice.find_by_sql(sql)

It’s worth noting that this returns as array and not an ActiveRecord_Relation object which means we cannot chain additional ActiveRecord methods on it. As a word of caution, make sure you call find_by_sql on the model that matches the table in your SELECT statement. Try this out and play around: Merchant.find_by_sql(sql). Does it work? Which table’s attributes do you have access to?

One last way we can run this query is to use ActiveRecord.connection.execute. This can be useful when the data you are building doesn’t correspond to a model and might be a temporary table built in memory as a result of other filtering and calculations.

invoices = ActiveRecord::Base.connection.execute(sql)

invoices.class
# => PG::Result
# (if you're using postgres)

invoices.to_a

As you can see, we don’t get model objects. Instead we see an array of hashes. What might you do if you wanted to turn each of these hashes into an object?

Both find_by_sql and ActiveRecord::Base.connection.execute have their uses. If you can get by with find_by_sql that’s probably the better choice but there are times where ActiveRecord::Base.connection.execute is necessary.

Lesson Search Results

Showing top 10 results