toptal/database_validations

View on GitHub
benchmarks/composed_benchmarks.rb

Summary

Maintainability
A
2 hrs
Test Coverage
require 'benchmark/ips'
require 'database_validations'
require_relative 'configurations'
require_relative 'gc_suite'
require_relative 'database_cleaner'

[postgresql_configuration, mysql_configuration].each do |database_configuration|
  ActiveRecord::Base.establish_connection(database_configuration)
  clear_database!(database_configuration)

  ActiveRecord::Schema.define(version: 1) do
    create_table :companies
    create_table :countries

    create_table :users_1 do |t|
      t.string :email
      t.string :full_name
      t.belongs_to :country
      t.belongs_to :company
      t.index :email
      t.index :full_name
    end

    create_table :users_2 do |t|
      t.string :email
      t.string :full_name
      t.belongs_to :country, foreign_key: true
      t.belongs_to :company, foreign_key: true
      t.index :email, unique: true
      t.index :full_name, unique: true
    end
  end
  ActiveRecord::Schema.verbose = false
  ActiveRecord::Base.logger = nil

  class Company < ActiveRecord::Base
  end

  class Country < ActiveRecord::Base
  end

  class Users1 < ActiveRecord::Base
    self.table_name = :users_1

    validates_uniqueness_of :email
    validates_uniqueness_of :full_name

    belongs_to :company, optional: false
    belongs_to :country, optional: false
  end

  class Users2 < ActiveRecord::Base
    self.table_name = :users_2

    validates_db_uniqueness_of :email
    validates_db_uniqueness_of :full_name

    db_belongs_to :company
    db_belongs_to :country
  end

  # ===Benchmarks===
  suite = GCSuite.new
  company = Company.create!
  country = Country.create!
  field = 0

  # ===Save only valid===
  Benchmark.ips do |x|
    x.config(suite: suite)

    x.report("#{database_configuration[:adapter]} optimistic: without gem") { field += 1; Users1.create(company_id: company.id, country_id: country.id, full_name: field.to_s, email: field.to_s) }
    x.report("#{database_configuration[:adapter]} optimistic: with gem") { field += 1; Users2.create(company_id: company.id, country_id: country.id, full_name: field.to_s, email: field.to_s) }

    x.report("#{database_configuration[:adapter]} realistic: without gem") { field += 1; field % 100 == 0 ? Users1.create(company_id: -1, country_id: -1, full_name: 'invalid', email: 'invalid') : Users1.create(company_id: company.id, country_id: country.id, full_name: field.to_s, email: field.to_s) }
    x.report("#{database_configuration[:adapter]} realistic: with gem") { field += 1; field % 100 == 0 ? Users2.create(company_id: -1, country_id: -1, full_name: 'invalid', email: 'invalid') : Users2.create(company_id: company.id, country_id: country.id, full_name: field.to_s, email: field.to_s) }

    x.report("#{database_configuration[:adapter]} pessimistic: without gem") { Users1.create(company_id: -1, country_id: -1, full_name: 'invalid', email: 'invalid') }
    x.report("#{database_configuration[:adapter]} pessimistic: with gem") { Users2.create(company_id: -1, country_id: -1, full_name: 'invalid', email: 'invalid') }
  end

  # Clear the DB
  ActiveRecord::Schema.define(version: 2) do
    drop_table :users_1, if_exists: true, force: :cascade
    drop_table :users_2, if_exists: true, force: :cascade
    drop_table :companies, if_exists: true, force: :cascade
    drop_table :countries, if_exists: true, force: :cascade
  end
end