Heisenbug: A bug that happened in background only

3 minute read

The bug

Few months back I discovered encountered a Heisenbug (software bug that seems to disappear or alter its behaviour when one attempts to study it.)

How it all started ?

We were about to release a new settlement process for a set of users, one feature of the settlement process was keeping old weeks settled. So, even if any ledger was added to a settled week, system would automatically counter it to maintain the original amount for that week.

We tested it thoroughly and were about to release it next day post final desk check.

I decided to test one more condition to be really sure and that’s when I encountered the heisenbug.

I tried to balance the amount of a user for a settlement week.

I did so by adding two ledgers for a settled week but for the same day.

I noticed that the processed week’s amount is not correct and found counters ledgers are not correct.

In place of one counter ledger of X amount, system made two counter ledgers of X amount !

Thinking that I must have made some changes, I reset the development branch to last good commit, test the normal scenario on rails console and it worked, system balanced the amount.

I ran it again via the original flow, system failed to balance it.

Trying to find the issue, I added debug logs, fired up the console, triggered the function that did all of the work, it worked just fine !

I made the script, ran it through rails runner mode, it worked again.

But when I used the main flow it didn’t work. Main flow was triggered inside a Sidekiq Worker

Struggling to find the bug, I opened a tmux window with split planes. Enabled activerecord query logs on both.

One with same flow in console and another containing sidekiq logs and stared at them.

Few minutes ago, Sajal from android team asked if rails does funny thing with threads and caching and I said don’t be absurd, rails can’t do this.

But then looking at the logs, it hit me.

Rails is indeed doing funny thing with sidekiq processes. Subsequent queries inside sidekiq was being cached.

Why/How it happened ?

Console flow had following query

# System computes difference 
(2.1ms) SELECT SUM from table where date = 'DATE'
# System balances first entry
(2.1ms) SELECT SUM from table where date = 'DATE'
# System balances second entry

while sidekiq flow only had one real query and subsequent queries were cached

# System computes difference 
(2.1ms) SELECT SUM from table where date = 'DATE'
# System balances first entry
# System computes difference
CACHE (0ms) SELECT SUM from table where date = 'DATE'
# System balances second entry

After balancing first entry, system computed the difference after adding second query but since query was same, rails didn’t even fire the query, even though amount was affected and returned stale data and system tried to balance amount based on the stale data and reached in-consistent state.

We were trying to do aggregation for the same week and rails provided the stale query, since we performing the same query in rapid succession. Ideally rails clears cache on any write but it didn’t do it in this case, for some reason.

Now I knew what was happening I fixed the problem but let’s find out how it happened.

Quick Googling pointed to the Sidekiq Wiki where it said

Even when performing batched reads correctly, as above, the ActiveRecord query cache can cause memory bloat by storing query result

sets unnecessarily. Since Rails 5.0, the query cache is enabled by default for background jobs, including Sidekiq workers. If your job performs a large number of batched reads and is still using lots of memory, try disabling the query cache:

ActiveRecord functionality was added to Sidekiq by this PR https://github.com/mperham/sidekiq/pull/3718/files

So if we are using Rails 5.x, queries in Sidekiq are cached by default

Fixing the bug

Fixing the bug was simple, we don’t want rails to cache the queries

So we can either.

  1. Clear the cache before every query ( Not good)
  2. Disable caching for the method (Good)

We can either wrap the entry block of sidekiq inside an uncached scope

like this

 class Worker 
 include Sidekiq::Worker 
 ...
 def perform 
     ActiveRecord::Base.uncached do
      # do stuff
     end
 end
 
 end

or just wrap the function where calculation is performed. This ensures that it is never cached, no matter if it’s executed from console or from Sidekiq

   def calculate
      Model.uncached do
        Model.where(date: Range).sum(:amount)
      end 
   end 

Updated:

Comments