# frozen_string_literal: true require "n_plus_one_control" module NPlusOneControl # Minitest assertions module MinitestHelper def assert_perform_constant_number_of_queries( exact = nil, populate: nil, matching: nil, scale_factors: nil, warmup: nil ) raise ArgumentError, "Block is required" unless block_given? warming_up warmup @executor = NPlusOneControl::Executor.new( population: populate || population_method, matching: matching || NPlusOneControl.default_matching, scale_factors: scale_factors || NPlusOneControl.default_scale_factors ) queries = @executor.call { yield } counts = queries.map(&:last).map(&:size) if exact assert counts.all? { _1 == exact }, NPlusOneControl.failure_message(:number_of_queries, queries) else assert counts.max == counts.min, NPlusOneControl.failure_message(:constant_queries, queries) end end def assert_perform_linear_number_of_queries( slope: 1, populate: nil, matching: nil, scale_factors: nil, warmup: nil ) raise ArgumentError, "Block is required" unless block_given? warming_up warmup @executor = NPlusOneControl::Executor.new( population: populate || population_method, matching: matching || NPlusOneControl.default_matching, scale_factors: scale_factors || NPlusOneControl.default_scale_factors ) queries = @executor.call { yield } assert linear?(queries, slope: slope), NPlusOneControl.failure_message(:linear_queries, queries) end def current_scale @executor&.current_scale end private def warming_up(warmup) own_methods = methods(false) + private_methods(false) (warmup || (own_methods.include?(:warmup) ? method(:warmup) : nil))&.call end def population_method own_methods = methods(false) + private_methods(false) own_methods.include?(:populate) ? method(:populate) : nil end def linear?(queries, slope:) queries.each_cons(2).all? do |pair| scales = pair.map(&:first) query_lists = pair.map(&:last) actual_slope = (query_lists[1].size - query_lists[0].size) / (scales[1] - scales[0]) actual_slope <= slope end end end end Minitest::Test.include NPlusOneControl::MinitestHelper