activerecord-analyze 0.4.0 → 0.7.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/.circleci/config.yml +30 -0
- data/README.md +98 -1
- data/activerecord-analyze.gemspec +3 -0
- data/docker-compose.yml.sample +11 -0
- data/lib/activerecord-analyze/main.rb +24 -2
- data/lib/activerecord-analyze/version.rb +1 -1
- data/query-plan.png +0 -0
- data/spec/main_spec.rb +57 -0
- data/spec/migrations/create_users_migration.rb +10 -0
- data/spec/spec_helper.rb +8 -0
- metadata +54 -3
    
        checksums.yaml
    CHANGED
    
    | @@ -1,7 +1,7 @@ | |
| 1 1 | 
             
            ---
         | 
| 2 2 | 
             
            SHA256:
         | 
| 3 | 
            -
              metadata.gz:  | 
| 4 | 
            -
              data.tar.gz:  | 
| 3 | 
            +
              metadata.gz: b2021239ed9352a040106ce8c1de50503e785fccd9a3093928eeaf5dfa79a0dc
         | 
| 4 | 
            +
              data.tar.gz: 8ba00244e759dc1f31091ec4196aded3887c9007902faf867431ea2ef9ae93dc
         | 
| 5 5 | 
             
            SHA512:
         | 
| 6 | 
            -
              metadata.gz:  | 
| 7 | 
            -
              data.tar.gz:  | 
| 6 | 
            +
              metadata.gz: ad2874f8422dc7299b8b7b3036fbec8e90f29780862ea3dd192629894905aafede52731314cc569afb6c9ce31f61a379b31ac308af0c304fc9e72bea583e58a7
         | 
| 7 | 
            +
              data.tar.gz: 114236b8af394e3f7fad06549e3bc99fa1814e815dd93282c7d3abfc0a0535ee568d90c0360abab021c83a7f1b38d2bfd0bfec31f043720893ab321776fe83fe
         | 
| @@ -0,0 +1,30 @@ | |
| 1 | 
            +
            version: 2
         | 
| 2 | 
            +
            jobs:
         | 
| 3 | 
            +
              test:
         | 
| 4 | 
            +
                docker:
         | 
| 5 | 
            +
                  - image: circleci/ruby:2.6.5
         | 
| 6 | 
            +
                    environment:
         | 
| 7 | 
            +
                      DATABASE_URL: postgresql://postgres:secret@localhost:5432/activerecord-analyze-test
         | 
| 8 | 
            +
                  - image: circleci/postgres:11.5
         | 
| 9 | 
            +
                    environment:
         | 
| 10 | 
            +
                      POSTGRES_USER: postgres
         | 
| 11 | 
            +
                      POSTGRES_DB: activerecord-analyze-test
         | 
| 12 | 
            +
                      POSTGRES_PASSWORD: secret
         | 
| 13 | 
            +
                parallelism: 1
         | 
| 14 | 
            +
                steps:
         | 
| 15 | 
            +
                  - checkout
         | 
| 16 | 
            +
                  - run: gem update --system
         | 
| 17 | 
            +
                  - run: gem install bundler
         | 
| 18 | 
            +
                  - run: bundle install --path vendor/bundle
         | 
| 19 | 
            +
                  - run: sudo apt-get update
         | 
| 20 | 
            +
                  - run: sudo apt install postgresql-client-11
         | 
| 21 | 
            +
                  - run: dockerize -wait tcp://localhost:5432 -timeout 1m
         | 
| 22 | 
            +
                  - run:
         | 
| 23 | 
            +
                      name: Run specs
         | 
| 24 | 
            +
                      command: |
         | 
| 25 | 
            +
                        bundle exec rspec spec/
         | 
| 26 | 
            +
            workflows:
         | 
| 27 | 
            +
              version: 2
         | 
| 28 | 
            +
              test:
         | 
| 29 | 
            +
                jobs:
         | 
| 30 | 
            +
                  - test
         | 
    
        data/README.md
    CHANGED
    
    | @@ -1,7 +1,13 @@ | |
| 1 | 
            -
            # ActiveRecord Analyze
         | 
| 1 | 
            +
            # ActiveRecord Analyze [](https://badge.fury.io/rb/activerecord-analyze) [](https://circleci.com/gh/pawurb/activerecord-analyze)
         | 
| 2 2 |  | 
| 3 3 | 
             
            This gem adds an `analyze` method to Active Record query objects. It executes `EXPLAIN ANALYZE` on a query SQL.
         | 
| 4 4 |  | 
| 5 | 
            +
            You can check out this blog post for more info on how to [debug and fix slow queries in Rails apps](https://pawelurbanek.com/slow-rails-queries).
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            The following `format` options are supported `:json, :hash, :yaml, :text, :xml`. Especially the `:json` format is useful because it let's you visualize a query plan using [a visualizer tool](https://tatiyants.com/pev/#/plans/new).
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            
         | 
| 10 | 
            +
             | 
| 5 11 | 
             
            ## Installation
         | 
| 6 12 |  | 
| 7 13 | 
             
            In your Gemfile:
         | 
| @@ -12,6 +18,97 @@ gem 'activerecord-analyze' | |
| 12 18 |  | 
| 13 19 | 
             
            ```
         | 
| 14 20 |  | 
| 21 | 
            +
            ## Options
         | 
| 22 | 
            +
             | 
| 23 | 
            +
            The `analyze` method supports the following EXPLAIN query options ([PostgreSQL docs reference](https://www.postgresql.org/docs/12/sql-explain.html)):
         | 
| 24 | 
            +
             | 
| 25 | 
            +
            ```
         | 
| 26 | 
            +
            BUFFERS [ boolean ]
         | 
| 27 | 
            +
            VERBOSE [ boolean ]
         | 
| 28 | 
            +
            COSTS [ boolean ]
         | 
| 29 | 
            +
            SETTINGS [ boolean ]
         | 
| 30 | 
            +
            TIMING [ boolean ]
         | 
| 31 | 
            +
            SUMMARY [ boolean ]
         | 
| 32 | 
            +
            FORMAT { TEXT | XML | JSON | YAML }
         | 
| 33 | 
            +
            ```
         | 
| 34 | 
            +
             | 
| 35 | 
            +
            You can execute it like that:
         | 
| 36 | 
            +
             | 
| 37 | 
            +
            ```ruby
         | 
| 38 | 
            +
             | 
| 39 | 
            +
            User.all.analyze(
         | 
| 40 | 
            +
              format: :json,
         | 
| 41 | 
            +
              verbose: true,
         | 
| 42 | 
            +
              costs: true,
         | 
| 43 | 
            +
              settings: true,
         | 
| 44 | 
            +
              buffers: true,
         | 
| 45 | 
            +
              timing: true,
         | 
| 46 | 
            +
              summary: true
         | 
| 47 | 
            +
            )
         | 
| 48 | 
            +
             | 
| 49 | 
            +
            # EXPLAIN (FORMAT JSON, ANALYZE, VERBOSE, COSTS, SETTINGS, BUFFERS, TIMING, SUMMARY)
         | 
| 50 | 
            +
            # SELECT "users".* FROM "users"
         | 
| 51 | 
            +
            # [
         | 
| 52 | 
            +
            #   {
         | 
| 53 | 
            +
            #     "Plan": {
         | 
| 54 | 
            +
            #       "Node Type": "Seq Scan",
         | 
| 55 | 
            +
            #       "Parallel Aware": false,
         | 
| 56 | 
            +
            #       "Relation Name": "users",
         | 
| 57 | 
            +
            #       "Schema": "public",
         | 
| 58 | 
            +
            #       "Alias": "users",
         | 
| 59 | 
            +
            #       "Startup Cost": 0.00,
         | 
| 60 | 
            +
            #       "Total Cost": 11.56,
         | 
| 61 | 
            +
            #       "Plan Rows": 520,
         | 
| 62 | 
            +
            #       "Plan Width": 127,
         | 
| 63 | 
            +
            #       "Actual Startup Time": 0.006,
         | 
| 64 | 
            +
            #       "Actual Total Time": 0.007,
         | 
| 65 | 
            +
            #       "Actual Rows": 2,
         | 
| 66 | 
            +
            #       "Actual Loops": 1,
         | 
| 67 | 
            +
            #       "Output": ["id", "team_id", "email"],
         | 
| 68 | 
            +
            #       "Shared Hit Blocks": 1,
         | 
| 69 | 
            +
            #       "Shared Read Blocks": 0,
         | 
| 70 | 
            +
            #       "Shared Dirtied Blocks": 0,
         | 
| 71 | 
            +
            #       "Shared Written Blocks": 0,
         | 
| 72 | 
            +
            #       "Local Hit Blocks": 0,
         | 
| 73 | 
            +
            #       "Local Read Blocks": 0,
         | 
| 74 | 
            +
            #       "Local Dirtied Blocks": 0,
         | 
| 75 | 
            +
            #       "Local Written Blocks": 0,
         | 
| 76 | 
            +
            #       "Temp Read Blocks": 0,
         | 
| 77 | 
            +
            #       "Temp Written Blocks": 0,
         | 
| 78 | 
            +
            #       "I/O Read Time": 0.000,
         | 
| 79 | 
            +
            #       "I/O Write Time": 0.000
         | 
| 80 | 
            +
            #     },
         | 
| 81 | 
            +
            #     "Settings": {
         | 
| 82 | 
            +
            #       "cpu_index_tuple_cost": "0.001",
         | 
| 83 | 
            +
            #       "cpu_operator_cost": "0.0005",
         | 
| 84 | 
            +
            #       "cpu_tuple_cost": "0.003",
         | 
| 85 | 
            +
            #       "effective_cache_size": "10800000kB",
         | 
| 86 | 
            +
            #       "max_parallel_workers_per_gather": "1",
         | 
| 87 | 
            +
            #       "random_page_cost": "2",
         | 
| 88 | 
            +
            #       "work_mem": "100MB"
         | 
| 89 | 
            +
            #     },
         | 
| 90 | 
            +
            #     "Planning Time": 0.033,
         | 
| 91 | 
            +
            #     "Triggers": [
         | 
| 92 | 
            +
            #     ],
         | 
| 93 | 
            +
            #     "Execution Time": 0.018
         | 
| 94 | 
            +
            #   }
         | 
| 95 | 
            +
            # ]
         | 
| 96 | 
            +
             | 
| 97 | 
            +
            ```
         | 
| 98 | 
            +
             | 
| 99 | 
            +
            Optionally you can disable running the `ANALYZE` query and only generate the plan:
         | 
| 100 | 
            +
             | 
| 101 | 
            +
            ```ruby
         | 
| 102 | 
            +
             | 
| 103 | 
            +
            User.all.analyze(analyze: false)
         | 
| 104 | 
            +
             | 
| 105 | 
            +
            # EXPLAIN ANALYZE for: SELECT "users".* FROM "users"
         | 
| 106 | 
            +
            #                         QUERY PLAN
         | 
| 107 | 
            +
            # ----------------------------------------------------------
         | 
| 108 | 
            +
            #  Seq Scan on users  (cost=0.00..15.20 rows=520 width=127)
         | 
| 109 | 
            +
             | 
| 110 | 
            +
            ```
         | 
| 111 | 
            +
             | 
| 15 112 | 
             
            ### Disclaimer
         | 
| 16 113 |  | 
| 17 114 | 
             
            It is a bit experimental and can break with new Rails release.
         | 
| @@ -7,6 +7,8 @@ module ActiveRecord | |
| 7 7 | 
             
                        case fmt
         | 
| 8 8 | 
             
                        when :json
         | 
| 9 9 | 
             
                          "FORMAT JSON,"
         | 
| 10 | 
            +
                        when :hash
         | 
| 11 | 
            +
                          "FORMAT JSON,"
         | 
| 10 12 | 
             
                        when :yaml
         | 
| 11 13 | 
             
                          "FORMAT YAML,"
         | 
| 12 14 | 
             
                        when :text
         | 
| @@ -50,6 +52,7 @@ module ActiveRecord | |
| 50 52 | 
             
                      .strip.gsub(/\s+/, " ")
         | 
| 51 53 | 
             
                      .gsub(/\(\s?\s?\s?,/, "(")
         | 
| 52 54 | 
             
                      .gsub(/\s,\s/, " ")
         | 
| 55 | 
            +
                      .gsub(/\(\s?\)/, "")
         | 
| 53 56 |  | 
| 54 57 | 
             
                      sql = "EXPLAIN #{opts_sql} #{to_sql(arel, binds)}"
         | 
| 55 58 | 
             
                      PostgreSQL::ExplainPrettyPrinter.new.pp(exec_query(sql, "EXPLAIN #{opts_sql}".strip, binds))
         | 
| @@ -62,7 +65,20 @@ end | |
| 62 65 | 
             
            module ActiveRecord
         | 
| 63 66 | 
             
              class Relation
         | 
| 64 67 | 
             
                def analyze(opts = {})
         | 
| 65 | 
            -
                  exec_analyze(collecting_queries_for_explain { exec_queries }, opts)
         | 
| 68 | 
            +
                  res = exec_analyze(collecting_queries_for_explain { exec_queries }, opts)
         | 
| 69 | 
            +
                  if [:json, :hash].include?(opts[:format])
         | 
| 70 | 
            +
                    start = res.index("[\n")
         | 
| 71 | 
            +
                    finish = res.rindex("]")
         | 
| 72 | 
            +
                    raw_json = res.slice(start, finish - start + 1)
         | 
| 73 | 
            +
             | 
| 74 | 
            +
                    if opts[:format] == :json
         | 
| 75 | 
            +
                      JSON.parse(raw_json).to_json
         | 
| 76 | 
            +
                    elsif opts[:format] == :hash
         | 
| 77 | 
            +
                      JSON.parse(raw_json)
         | 
| 78 | 
            +
                    end
         | 
| 79 | 
            +
                  else
         | 
| 80 | 
            +
                    res
         | 
| 81 | 
            +
                  end
         | 
| 66 82 | 
             
                end
         | 
| 67 83 | 
             
              end
         | 
| 68 84 | 
             
            end
         | 
| @@ -71,7 +87,13 @@ module ActiveRecord | |
| 71 87 | 
             
              module Explain
         | 
| 72 88 | 
             
                def exec_analyze(queries, opts = {}) # :nodoc:
         | 
| 73 89 | 
             
                  str = queries.map do |sql, binds|
         | 
| 74 | 
            -
                     | 
| 90 | 
            +
                    analyze_msg = if opts[:analyze] == false
         | 
| 91 | 
            +
                      ""
         | 
| 92 | 
            +
                    else
         | 
| 93 | 
            +
                      " ANALYZE"
         | 
| 94 | 
            +
                    end
         | 
| 95 | 
            +
             | 
| 96 | 
            +
                    msg = "EXPLAIN#{analyze_msg} for: #{sql}".dup
         | 
| 75 97 | 
             
                    unless binds.empty?
         | 
| 76 98 | 
             
                      msg << " "
         | 
| 77 99 | 
             
                      msg << binds.map { |attr| render_bind(attr) }.inspect
         | 
    
        data/query-plan.png
    ADDED
    
    | Binary file | 
    
        data/spec/main_spec.rb
    ADDED
    
    | @@ -0,0 +1,57 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            require 'spec_helper'
         | 
| 4 | 
            +
            require "migrations/create_users_migration.rb"
         | 
| 5 | 
            +
             | 
| 6 | 
            +
            class User < ActiveRecord::Base; end
         | 
| 7 | 
            +
             | 
| 8 | 
            +
            describe "ActiveRecord analyze" do
         | 
| 9 | 
            +
              before(:all) do
         | 
| 10 | 
            +
                ActiveRecord::Base.establish_connection(
         | 
| 11 | 
            +
                  ENV.fetch("DATABASE_URL")
         | 
| 12 | 
            +
                )
         | 
| 13 | 
            +
             | 
| 14 | 
            +
                @schema_migration = ActiveRecord::Base.connection.schema_migration
         | 
| 15 | 
            +
                ActiveRecord::Migrator.new(:up, [CreateUsers.new], @schema_migration).migrate
         | 
| 16 | 
            +
              end
         | 
| 17 | 
            +
             | 
| 18 | 
            +
              describe "no opts" do
         | 
| 19 | 
            +
                it "works" do
         | 
| 20 | 
            +
                  expect do
         | 
| 21 | 
            +
                    User.all.analyze
         | 
| 22 | 
            +
                  end.not_to raise_error
         | 
| 23 | 
            +
                end
         | 
| 24 | 
            +
              end
         | 
| 25 | 
            +
             | 
| 26 | 
            +
              describe "format json" do
         | 
| 27 | 
            +
                it "works" do
         | 
| 28 | 
            +
                  result = User.all.analyze(format: :json)
         | 
| 29 | 
            +
                  expect(JSON.parse(result)[0].keys.sort).to eq [
         | 
| 30 | 
            +
                    "Execution Time", "Plan", "Planning Time", "Triggers"
         | 
| 31 | 
            +
                  ]
         | 
| 32 | 
            +
                end
         | 
| 33 | 
            +
              end
         | 
| 34 | 
            +
             | 
| 35 | 
            +
              describe "format hash" do
         | 
| 36 | 
            +
                it "works" do
         | 
| 37 | 
            +
                  result = User.all.analyze(format: :hash)
         | 
| 38 | 
            +
                  expect(result[0].keys.sort).to eq [
         | 
| 39 | 
            +
                    "Execution Time", "Plan", "Planning Time", "Triggers"
         | 
| 40 | 
            +
                  ]
         | 
| 41 | 
            +
                end
         | 
| 42 | 
            +
              end
         | 
| 43 | 
            +
             | 
| 44 | 
            +
              describe "supports options" do
         | 
| 45 | 
            +
                it "works" do
         | 
| 46 | 
            +
                  result = User.all.limit(10).where.not(email: nil).analyze(
         | 
| 47 | 
            +
                    format: :hash,
         | 
| 48 | 
            +
                    costs: true,
         | 
| 49 | 
            +
                    timing: true,
         | 
| 50 | 
            +
                    summary: true
         | 
| 51 | 
            +
                  )
         | 
| 52 | 
            +
                  expect(result[0].keys.sort).to eq [
         | 
| 53 | 
            +
                    "Execution Time", "Plan", "Planning Time", "Triggers"
         | 
| 54 | 
            +
                  ]
         | 
| 55 | 
            +
                end
         | 
| 56 | 
            +
              end
         | 
| 57 | 
            +
            end
         | 
    
        data/spec/spec_helper.rb
    ADDED
    
    
    
        metadata
    CHANGED
    
    | @@ -1,14 +1,14 @@ | |
| 1 1 | 
             
            --- !ruby/object:Gem::Specification
         | 
| 2 2 | 
             
            name: activerecord-analyze
         | 
| 3 3 | 
             
            version: !ruby/object:Gem::Version
         | 
| 4 | 
            -
              version: 0. | 
| 4 | 
            +
              version: 0.7.1
         | 
| 5 5 | 
             
            platform: ruby
         | 
| 6 6 | 
             
            authors:
         | 
| 7 7 | 
             
            - pawurb
         | 
| 8 8 | 
             
            autorequire:
         | 
| 9 9 | 
             
            bindir: bin
         | 
| 10 10 | 
             
            cert_chain: []
         | 
| 11 | 
            -
            date: 2021- | 
| 11 | 
            +
            date: 2021-04-09 00:00:00.000000000 Z
         | 
| 12 12 | 
             
            dependencies:
         | 
| 13 13 | 
             
            - !ruby/object:Gem::Dependency
         | 
| 14 14 | 
             
              name: rails
         | 
| @@ -24,6 +24,48 @@ dependencies: | |
| 24 24 | 
             
                - - ">="
         | 
| 25 25 | 
             
                  - !ruby/object:Gem::Version
         | 
| 26 26 | 
             
                    version: 5.1.0
         | 
| 27 | 
            +
            - !ruby/object:Gem::Dependency
         | 
| 28 | 
            +
              name: rake
         | 
| 29 | 
            +
              requirement: !ruby/object:Gem::Requirement
         | 
| 30 | 
            +
                requirements:
         | 
| 31 | 
            +
                - - ">="
         | 
| 32 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 33 | 
            +
                    version: '0'
         | 
| 34 | 
            +
              type: :development
         | 
| 35 | 
            +
              prerelease: false
         | 
| 36 | 
            +
              version_requirements: !ruby/object:Gem::Requirement
         | 
| 37 | 
            +
                requirements:
         | 
| 38 | 
            +
                - - ">="
         | 
| 39 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 40 | 
            +
                    version: '0'
         | 
| 41 | 
            +
            - !ruby/object:Gem::Dependency
         | 
| 42 | 
            +
              name: pg
         | 
| 43 | 
            +
              requirement: !ruby/object:Gem::Requirement
         | 
| 44 | 
            +
                requirements:
         | 
| 45 | 
            +
                - - ">="
         | 
| 46 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 47 | 
            +
                    version: '0'
         | 
| 48 | 
            +
              type: :development
         | 
| 49 | 
            +
              prerelease: false
         | 
| 50 | 
            +
              version_requirements: !ruby/object:Gem::Requirement
         | 
| 51 | 
            +
                requirements:
         | 
| 52 | 
            +
                - - ">="
         | 
| 53 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 54 | 
            +
                    version: '0'
         | 
| 55 | 
            +
            - !ruby/object:Gem::Dependency
         | 
| 56 | 
            +
              name: rspec
         | 
| 57 | 
            +
              requirement: !ruby/object:Gem::Requirement
         | 
| 58 | 
            +
                requirements:
         | 
| 59 | 
            +
                - - ">="
         | 
| 60 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 61 | 
            +
                    version: '0'
         | 
| 62 | 
            +
              type: :development
         | 
| 63 | 
            +
              prerelease: false
         | 
| 64 | 
            +
              version_requirements: !ruby/object:Gem::Requirement
         | 
| 65 | 
            +
                requirements:
         | 
| 66 | 
            +
                - - ">="
         | 
| 67 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 68 | 
            +
                    version: '0'
         | 
| 27 69 | 
             
            description: ' Gem adds an "analyze" method to all Active Record query objects. Compatible
         | 
| 28 70 | 
             
              with PostgreSQL database. '
         | 
| 29 71 | 
             
            email:
         | 
| @@ -32,15 +74,21 @@ executables: [] | |
| 32 74 | 
             
            extensions: []
         | 
| 33 75 | 
             
            extra_rdoc_files: []
         | 
| 34 76 | 
             
            files:
         | 
| 77 | 
            +
            - ".circleci/config.yml"
         | 
| 35 78 | 
             
            - ".gitignore"
         | 
| 36 79 | 
             
            - Gemfile
         | 
| 37 80 | 
             
            - LICENSE.txt
         | 
| 38 81 | 
             
            - README.md
         | 
| 39 82 | 
             
            - Rakefile
         | 
| 40 83 | 
             
            - activerecord-analyze.gemspec
         | 
| 84 | 
            +
            - docker-compose.yml.sample
         | 
| 41 85 | 
             
            - lib/activerecord-analyze.rb
         | 
| 42 86 | 
             
            - lib/activerecord-analyze/main.rb
         | 
| 43 87 | 
             
            - lib/activerecord-analyze/version.rb
         | 
| 88 | 
            +
            - query-plan.png
         | 
| 89 | 
            +
            - spec/main_spec.rb
         | 
| 90 | 
            +
            - spec/migrations/create_users_migration.rb
         | 
| 91 | 
            +
            - spec/spec_helper.rb
         | 
| 44 92 | 
             
            homepage: http://github.com/pawurb/activerecord-analyze
         | 
| 45 93 | 
             
            licenses:
         | 
| 46 94 | 
             
            - MIT
         | 
| @@ -64,4 +112,7 @@ rubygems_version: 3.1.4 | |
| 64 112 | 
             
            signing_key:
         | 
| 65 113 | 
             
            specification_version: 4
         | 
| 66 114 | 
             
            summary: Add EXPLAIN ANALYZE to Active Record query objects
         | 
| 67 | 
            -
            test_files: | 
| 115 | 
            +
            test_files:
         | 
| 116 | 
            +
            - spec/main_spec.rb
         | 
| 117 | 
            +
            - spec/migrations/create_users_migration.rb
         | 
| 118 | 
            +
            - spec/spec_helper.rb
         |