Project

testoscope

0.0
No commit activity in last 3 years
No release in over 3 years
This is simple and nice tool to inspect how application operates with current DB structure while testing app, meaning redundant indexes, sequential scans, dummy requests and any other unintended behaviour customized by user.
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
 Dependencies

Development

~> 5.0
>= 2.2.10
>= 12.3.3

Runtime

>= 0.1.23
 Project Readme

Testoscope

Inspect in practice how well is your data organized while testing your application!

Features

Finds out of the box: sequential scans, dummy One-Timer calls to DB, unused indexes. Out of the box it works for PG only, if you need to use it with MySQL or other DB, you need to do some customization.

Highly customizable: you can define your own unintended markers, inspect only some of you tables and so.

May work in a error mode raising exception on unintended behaviour, that way you can protect from perfomance break-out in production.

Best suits with high-level testing: controller tests, integration tests, api tests and so.

Output example: alt text

Out of the box inspections

Sequential scans, dummy One-Timer calls to DB, unused indexes

Sequential scans

It can happend when you are:

  • truly missing an index
  • when you are intend to use a partial index but unintentionally miss index condition in a query

One-Timers

Some times ORM can produce dummy query, in Postgres Query Plan they look like this:

              QUERY PLAN
--------------------------------------------
Result  (cost=0.00..0.00 rows=0 width=194)
 One-Time Filter: false
(2 rows)

in SQL query you are looking for some WHERE false:

SELECT "tags".* 
  FROM "tags" 
  WHERE "tags"."parent_id" = $1 AND 1=0

and in ORM it doesn't look alarming:

sub_tags.where( name: names )

But when names is empty, we get a dummy request.

They are not a big deal from a performance perspective, but you are occupying DB connection pool and cluttering your channel with empty noise.

So it's better to change the underlying logic other than be OK with it.

Unused index

Testoscope can find and warn you about unused index. Possible reasons for them are:

  • you forgot to remove index after code refactoring, and now you have redundant unused index
  • you already have another index more suitable which is preferred by the planner
  • your tests doesn't cover all use-cases

In either cases you may have a problem, but also may not.

Rem: It seems to be much less useful than I thought, at least with PG, because lack of statistic or tiny DB size for test environment makes it very vulnerable to false positive index assumptions. Still you can use it within staging server and get the info from less automatic testing.

How it works?

Testoscope hooks to exec_query of a connection adapter, when sql query reach exec_query it runs two times: first with EXPLAIN for tstoscope to analyze, and the second - for original a caller purpose.

After retrieving explain result, testoscope search for specified unintended behaviour markers, like a Seq Scan substring in Postgres QUERY PLAN explained, and also collects indexes used by all queries for a final summary.

Unintended Behaviours

By default markers are preconfigured for PosgtreSQL and full list of tables:

 config.unintened_key_words = ['Seq Scan', 'One-Time Filter']
 config.tables = :all

But you can set any regexp you want to track inside EXPLAIN results, and also you can track partial list of tables:

config.tables = ['cards', 'users']

Installation

Add this line to your application's Gemfile:

group :test do 
    gem 'testoscope'
end

And then execute:

$ bundle

Usage

In test_helper.rb:

require 'testoscope'

# since doubling all requests to DB is not free, 
# you may use ENV variable to run on demand 
if ENV[:RUN_TESTOSCOPE]
    Testoscope.configure { |c| 
         # include only trace from folders
         c.back_trace_paths = [Rails.root.to_s]
         
         # additionaly exclude subfolders 
         c.back_trace_exclude_paths = ["#{Rails.root.to_s}/test", "#{Rails.root.to_s}/spec"]
         
         # unintended behaviour markers
         c.unintened_key_words = ['Seq Scan', 'One-Time Filter']
         
         # set true to raise error on unintended behaviour 
         c.raise_when_unintended = false
         
         # true to analyze everything, false to analize only by demand using suspend_global_analyze
         c.analyze = true
         
         # :all - to complete list, or array of specific table names: [ 'users', 'comments' ... ] 
         c.tables = :all
    }
    
    MiniTest::Unit.after_tests {
       Testoscope.print_results
    }
end 

TODO

  • Add Tests!
  • Add inspection scripts for suspicious behaviours, like when you using scopes in query and additionaly instantinate them later:
@users = User.scope
# collection doesn't instantinate, and used as subquery, it's OK if we don;t need instatination, 
# but we instatinate it later 
@docs = Doc.for_users( @users )
render { users: @users, docs: @docs }

Contributing

Bug reports and pull requests are welcome on GitHub at https://github.com/alekseyl/testoscope.

License

The gem is available as open source under the terms of the MIT License.