Class: MiniTest::Unit::TestCase
- Inherits:
-
Object
- Object
- MiniTest::Unit::TestCase
- Includes:
- Assertions
- Defined in:
- lib/minitest/unit.rb,
lib/minitest/benchmark.rb
Overview
Subclass TestCase to create your own tests. Typically you'll want a TestCase subclass per implementation class.
See MiniTest::Assertions
Direct Known Subclasses
Constant Summary
- PASSTHROUGH_EXCEPTIONS =
[NoMemoryError, SignalException, Interrupt, SystemExit]
- SUPPORTS_INFO_SIGNAL =
:nodoc:
Signal.list['INFO']
Constants included from Assertions
Assertions::UNDEFINED, Assertions::WINDOZE
Instance Attribute Summary (collapse)
-
- (Object) __name__
readonly
:nodoc:.
Class Method Summary (collapse)
-
+ (Object) add_setup_hook(arg = nil, &block)
Adds a block of code that will be executed before every TestCase is run.
-
+ (Object) add_teardown_hook(arg = nil, &block)
Adds a block of code that will be executed after every TestCase is run.
-
+ (Object) bench_exp(min, max, base = 10)
Returns a set of ranges stepped exponentially from min to max by powers of base.
-
+ (Object) bench_linear(min, max, step = 10)
Returns a set of ranges stepped linearly from min to max by step.
-
+ (Object) bench_range
Specifies the ranges used for benchmarking for that class.
-
+ (Object) benchmark_methods
Returns the benchmark methods (methods that start with bench_) for that class.
-
+ (Object) benchmark_suites
Returns all test suites that have benchmark methods.
-
+ (Object) i_suck_and_my_tests_are_order_dependent!
Call this at the top of your tests when you absolutely positively need to have ordered tests.
-
+ (Object) inherited(klass)
:nodoc:.
-
+ (Object) reset
:nodoc:.
-
+ (Object) reset_setup_teardown_hooks
:nodoc:.
-
+ (Object) setup_hooks
:nodoc:.
-
+ (Object) teardown_hooks
:nodoc:.
-
+ (Object) test_methods
:nodoc:.
-
+ (Object) test_order
:nodoc:.
-
+ (Object) test_suites
:nodoc:.
Instance Method Summary (collapse)
-
- (Object) assert_performance(validation, &work)
Runs the given work, gathering the times of each run.
-
- (Object) assert_performance_constant(threshold = 0.99, &work)
Runs the given work and asserts that the times gathered fit to match a constant rate (eg, linear slope == 0) within a given threshold.
-
- (Object) assert_performance_exponential(threshold = 0.99, &work)
Runs the given work and asserts that the times gathered fit to match a exponential curve within a given error threshold.
-
- (Object) assert_performance_linear(threshold = 0.99, &work)
Runs the given work and asserts that the times gathered fit to match a straight line within a given error threshold.
-
- (Object) assert_performance_power(threshold = 0.99, &work)
Runs the given work and asserts that the times gathered curve fit to match a power curve within a given error threshold.
-
- (Object) fit_error(xys)
Takes an array of x/y pairs and calculates the general R^2 value.
-
- (Object) fit_exponential(xs, ys)
To fit a functional form: y = ae^(bx).
-
- (Object) fit_linear(xs, ys)
Fits the functional form: a + bx.
-
- (Object) fit_power(xs, ys)
To fit a functional form: y = ax^b.
-
- (TestCase) initialize(name)
constructor
:nodoc:.
- - (Object) io
- - (Boolean) io?
-
- (Boolean) passed?
Returns true if the test passed.
-
- (Object) run(runner)
Runs the tests reporting the status to runner.
-
- (Object) run_setup_hooks
:nodoc:.
-
- (Object) run_teardown_hooks
:nodoc:.
-
- (Object) setup
Runs before every test.
-
- (Object) sigma(enum, &block)
Enumerates over enum mapping block if given, returning the sum of the result.
-
- (Object) teardown
Runs after every test.
-
- (Object) validation_for_fit(msg, threshold)
Returns a proc that calls the specified fit method and asserts that the error is within a tolerable threshold.
Methods included from Assertions
#_assertions, #_assertions=, #assert, #assert_block, #assert_empty, #assert_equal, #assert_in_delta, #assert_in_epsilon, #assert_includes, #assert_instance_of, #assert_kind_of, #assert_match, #assert_nil, #assert_operator, #assert_output, #assert_predicate, #assert_raises, #assert_respond_to, #assert_same, #assert_send, #assert_silent, #assert_throws, #capture_io, diff, #diff, diff=, #exception_details, #flunk, #message, #mu_pp, #mu_pp_for_diff, #pass, #refute, #refute_empty, #refute_equal, #refute_in_delta, #refute_in_epsilon, #refute_includes, #refute_instance_of, #refute_kind_of, #refute_match, #refute_nil, #refute_operator, #refute_predicate, #refute_respond_to, #refute_same, #skip
Constructor Details
- (TestCase) initialize(name)
:nodoc:
996 997 998 999 1000 |
# File 'lib/minitest/unit.rb', line 996 def initialize name # :nodoc: @__name__ = name @__io__ = nil @passed = nil end |
Instance Attribute Details
- (Object) __name__ (readonly)
:nodoc:
952 953 954 |
# File 'lib/minitest/unit.rb', line 952 def __name__ @__name__ end |
Class Method Details
+ (Object) add_setup_hook(arg = nil, &block)
Adds a block of code that will be executed before every TestCase is run. Equivalent to setup, but usable multiple times and without re-opening any classes.
All of the setup hooks will run in order after the setup method, if one is defined.
The argument can be any object that responds to #call or a block. That means that this call,
MiniTest::TestCase.add_setup_hook { puts "foo" }
... is equivalent to:
module MyTestSetup
def call
puts "foo"
end
end
MiniTest::TestCase.add_setup_hook MyTestSetup
The blocks passed to add_setup_hook take an optional parameter that will be the TestCase instance that is executing the block.
1106 1107 1108 1109 |
# File 'lib/minitest/unit.rb', line 1106 def self.add_setup_hook arg=nil, &block hook = arg || block @setup_hooks << hook end |
+ (Object) add_teardown_hook(arg = nil, &block)
Adds a block of code that will be executed after every TestCase is run. Equivalent to teardown, but usable multiple times and without re-opening any classes.
All of the teardown hooks will run in reverse order after the teardown method, if one is defined.
The argument can be any object that responds to #call or a block. That means that this call,
MiniTest::TestCase.add_teardown_hook { puts "foo" }
... is equivalent to:
module MyTestTeardown
def call
puts "foo"
end
end
MiniTest::TestCase.add_teardown_hook MyTestTeardown
The blocks passed to add_teardown_hook take an optional parameter that will be the TestCase instance that is executing the block.
1155 1156 1157 1158 |
# File 'lib/minitest/unit.rb', line 1155 def self.add_teardown_hook arg=nil, &block hook = arg || block @teardown_hooks << hook end |
+ (Object) bench_exp(min, max, base = 10)
Returns a set of ranges stepped exponentially from min to max by powers of base. Eg:
bench_exp(2, 16, 2) # => [2, 4, 8, 16]
22 23 24 25 26 27 |
# File 'lib/minitest/benchmark.rb', line 22 def self.bench_exp min, max, base = 10 min = (Math.log10(min) / Math.log10(base)).to_i max = (Math.log10(max) / Math.log10(base)).to_i (min..max).map { |m| base ** m }.to_a end |
+ (Object) bench_linear(min, max, step = 10)
Returns a set of ranges stepped linearly from min to max by step. Eg:
bench_linear(20, 40, 10) # => [20, 30, 40]
35 36 37 38 39 |
# File 'lib/minitest/benchmark.rb', line 35 def self.bench_linear min, max, step = 10 (min..max).step(step).to_a rescue LocalJumpError # 1.8.6 r = []; (min..max).step(step) { |n| r << n }; r end |
+ (Object) bench_range
Specifies the ranges used for benchmarking for that class. Defaults to exponential growth from 1 to 10k by powers of 10. Override if you need different ranges for your benchmarks.
See also: ::bench_exp and ::bench_linear.
63 64 65 |
# File 'lib/minitest/benchmark.rb', line 63 def self.bench_range bench_exp 1, 10_000 end |
+ (Object) benchmark_methods
Returns the benchmark methods (methods that start with bench_) for that class.
45 46 47 |
# File 'lib/minitest/benchmark.rb', line 45 def self.benchmark_methods # :nodoc: public_instance_methods(true).grep(/^bench_/).map { |m| m.to_s }.sort end |
+ (Object) benchmark_suites
Returns all test suites that have benchmark methods.
52 53 54 |
# File 'lib/minitest/benchmark.rb', line 52 def self.benchmark_suites TestCase.test_suites.reject { |s| s.benchmark_methods.empty? } end |
+ (Object) i_suck_and_my_tests_are_order_dependent!
Call this at the top of your tests when you absolutely positively need to have ordered tests. In doing so, you're admitting that you suck and your tests are weak.
1022 1023 1024 1025 1026 |
# File 'lib/minitest/unit.rb', line 1022 def self.i_suck_and_my_tests_are_order_dependent! class << self define_method :test_order do :alpha end end end |
+ (Object) inherited(klass)
:nodoc:
1028 1029 1030 1031 1032 |
# File 'lib/minitest/unit.rb', line 1028 def self.inherited klass # :nodoc: @@test_suites[klass] = true klass.reset_setup_teardown_hooks super end |
+ (Object) reset
:nodoc:
1011 1012 1013 |
# File 'lib/minitest/unit.rb', line 1011 def self.reset # :nodoc: @@test_suites = {} end |
+ (Object) reset_setup_teardown_hooks
:nodoc:
1073 1074 1075 1076 |
# File 'lib/minitest/unit.rb', line 1073 def self.reset_setup_teardown_hooks # :nodoc: @setup_hooks = [] @teardown_hooks = [] end |
+ (Object) setup_hooks
:nodoc:
1111 1112 1113 1114 1115 1116 1117 |
# File 'lib/minitest/unit.rb', line 1111 def self.setup_hooks # :nodoc: if superclass.respond_to? :setup_hooks then superclass.setup_hooks else [] end + @setup_hooks end |
+ (Object) teardown_hooks
:nodoc:
1160 1161 1162 1163 1164 1165 1166 |
# File 'lib/minitest/unit.rb', line 1160 def self.teardown_hooks # :nodoc: if superclass.respond_to? :teardown_hooks then superclass.teardown_hooks else [] end + @teardown_hooks end |
+ (Object) test_methods
:nodoc:
1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 |
# File 'lib/minitest/unit.rb', line 1042 def self.test_methods # :nodoc: methods = public_instance_methods(true).grep(/^test/).map { |m| m.to_s } case self.test_order when :random then max = methods.size methods.sort.sort_by { rand max } when :alpha, :sorted then methods.sort else raise "Unknown test_order: #{self.test_order.inspect}" end end |
+ (Object) test_order
:nodoc:
1034 1035 1036 |
# File 'lib/minitest/unit.rb', line 1034 def self.test_order # :nodoc: :random end |
+ (Object) test_suites
:nodoc:
1038 1039 1040 |
# File 'lib/minitest/unit.rb', line 1038 def self.test_suites # :nodoc: @@test_suites.keys.sort_by { |ts| ts.name.to_s } end |
Instance Method Details
- (Object) assert_performance(validation, &work)
Runs the given work, gathering the times of each run. Range and times are then passed to a given validation proc. Outputs the benchmark name and times in tab-separated format, making it easy to paste into a spreadsheet for graphing or further analysis.
Ranges are specified by ::bench_range.
Eg:
def bench_algorithm
validation = proc { |x, y| ... }
assert_performance validation do |x|
@obj.algorithm
end
end
85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 |
# File 'lib/minitest/benchmark.rb', line 85 def assert_performance validation, &work range = self.class.bench_range io.print "#{__name__}" times = [] range.each do |x| GC.start t0 = Time.now instance_exec(x, &work) t = Time.now - t0 io.print "\t%9.6f" % t times << t end io.puts validation[range, times] end |
- (Object) assert_performance_constant(threshold = 0.99, &work)
Runs the given work and asserts that the times gathered fit to match a constant rate (eg, linear slope == 0) within a given threshold. Note: because we're testing for a slope of 0, R^2 is not a good determining factor for the fit, so the threshold is applied against the slope itself. As such, you probably want to tighten it from the default.
See www.graphpad.com/curvefit/goodness_of_fit.htm for more details.
Fit is calculated by #fit_linear.
Ranges are specified by ::bench_range.
Eg:
def bench_algorithm
assert_performance_constant 0.9999 do |x|
@obj.algorithm
end
end
129 130 131 132 133 134 135 136 137 |
# File 'lib/minitest/benchmark.rb', line 129 def assert_performance_constant threshold = 0.99, &work validation = proc do |range, times| a, b, rr = fit_linear range, times assert_in_delta 0, b, 1 - threshold [a, b, rr] end assert_performance validation, &work end |
- (Object) assert_performance_exponential(threshold = 0.99, &work)
Runs the given work and asserts that the times gathered fit to match a exponential curve within a given error threshold.
Fit is calculated by #fit_exponential.
Ranges are specified by ::bench_range.
Eg:
def bench_algorithm
assert_performance_exponential 0.9999 do |x|
@obj.algorithm
end
end
155 156 157 |
# File 'lib/minitest/benchmark.rb', line 155 def assert_performance_exponential threshold = 0.99, &work assert_performance validation_for_fit(:exponential, threshold), &work end |
- (Object) assert_performance_linear(threshold = 0.99, &work)
Runs the given work and asserts that the times gathered fit to match a straight line within a given error threshold.
Fit is calculated by #fit_linear.
Ranges are specified by ::bench_range.
Eg:
def bench_algorithm
assert_performance_linear 0.9999 do |x|
@obj.algorithm
end
end
175 176 177 |
# File 'lib/minitest/benchmark.rb', line 175 def assert_performance_linear threshold = 0.99, &work assert_performance validation_for_fit(:linear, threshold), &work end |
- (Object) assert_performance_power(threshold = 0.99, &work)
Runs the given work and asserts that the times gathered curve fit to match a power curve within a given error threshold.
Fit is calculated by #fit_power.
Ranges are specified by ::bench_range.
Eg:
def bench_algorithm
assert_performance_power 0.9999 do |x|
@obj.algorithm
end
end
195 196 197 |
# File 'lib/minitest/benchmark.rb', line 195 def assert_performance_power threshold = 0.99, &work assert_performance validation_for_fit(:power, threshold), &work end |
- (Object) fit_error(xys)
Takes an array of x/y pairs and calculates the general R^2 value.
204 205 206 207 208 209 210 |
# File 'lib/minitest/benchmark.rb', line 204 def fit_error xys = sigma(xys) { |x, y| y } / xys.size.to_f ss_tot = sigma(xys) { |x, y| (y - ) ** 2 } ss_err = sigma(xys) { |x, y| (yield(x) - y) ** 2 } 1 - (ss_err / ss_tot) end |
- (Object) fit_exponential(xs, ys)
To fit a functional form: y = ae^(bx).
Takes x and y values and returns [a, b, r^2].
See: mathworld.wolfram.com/LeastSquaresFittingExponential.html
219 220 221 222 223 224 225 226 227 228 229 230 231 232 |
# File 'lib/minitest/benchmark.rb', line 219 def fit_exponential xs, ys n = xs.size xys = xs.zip(ys) sxlny = sigma(xys) { |x,y| x * Math.log(y) } slny = sigma(xys) { |x,y| Math.log(y) } sx2 = sigma(xys) { |x,y| x * x } sx = sigma xs c = n * sx2 - sx ** 2 a = (slny * sx2 - sx * sxlny) / c b = ( n * sxlny - sx * slny ) / c return Math.exp(a), b, fit_error(xys) { |x| Math.exp(a + b * x) } end |
- (Object) fit_linear(xs, ys)
Fits the functional form: a + bx.
Takes x and y values and returns [a, b, r^2].
241 242 243 244 245 246 247 248 249 250 251 252 253 254 |
# File 'lib/minitest/benchmark.rb', line 241 def fit_linear xs, ys n = xs.size xys = xs.zip(ys) sx = sigma xs sy = sigma ys sx2 = sigma(xs) { |x| x ** 2 } sxy = sigma(xys) { |x,y| x * y } c = n * sx2 - sx**2 a = (sy * sx2 - sx * sxy) / c b = ( n * sxy - sx * sy ) / c return a, b, fit_error(xys) { |x| a + b * x } end |
- (Object) fit_power(xs, ys)
To fit a functional form: y = ax^b.
Takes x and y values and returns [a, b, r^2].
263 264 265 266 267 268 269 270 271 272 273 274 275 |
# File 'lib/minitest/benchmark.rb', line 263 def fit_power xs, ys n = xs.size xys = xs.zip(ys) slnxlny = sigma(xys) { |x, y| Math.log(x) * Math.log(y) } slnx = sigma(xs) { |x | Math.log(x) } slny = sigma(ys) { | y| Math.log(y) } slnx2 = sigma(xs) { |x | Math.log(x) ** 2 } b = (n * slnxlny - slnx * slny) / (n * slnx2 - slnx ** 2); a = (slny - b * slnx) / n return Math.exp(a), b, fit_error(xys) { |x| (Math.exp(a) * (x ** b)) } end |
- (Object) io
1002 1003 1004 1005 |
# File 'lib/minitest/unit.rb', line 1002 def io @__io__ = true MiniTest::Unit.output end |
- (Boolean) io?
1007 1008 1009 |
# File 'lib/minitest/unit.rb', line 1007 def io? @__io__ end |
- (Boolean) passed?
Returns true if the test passed.
1059 1060 1061 |
# File 'lib/minitest/unit.rb', line 1059 def passed? @passed end |
- (Object) run(runner)
Runs the tests reporting the status to runner
962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 |
# File 'lib/minitest/unit.rb', line 962 def run runner trap "INFO" do time = runner.start_time ? Time.now - runner.start_time : 0 warn "%s#%s %.2fs" % [self.class, self.__name__, time] runner.status $stderr end if SUPPORTS_INFO_SIGNAL result = "" begin @passed = nil self.setup self.run_setup_hooks self.__send__ self.__name__ result = "." unless io? @passed = true rescue *PASSTHROUGH_EXCEPTIONS raise rescue Exception => e @passed = false result = runner.puke self.class, self.__name__, e ensure begin self.run_teardown_hooks self.teardown rescue *PASSTHROUGH_EXCEPTIONS raise rescue Exception => e result = runner.puke self.class, self.__name__, e end trap 'INFO', 'DEFAULT' if SUPPORTS_INFO_SIGNAL end result end |
- (Object) run_setup_hooks
:nodoc:
1119 1120 1121 1122 1123 1124 1125 1126 1127 |
# File 'lib/minitest/unit.rb', line 1119 def run_setup_hooks # :nodoc: self.class.setup_hooks.each do |hook| if hook.respond_to?(:arity) && hook.arity == 1 hook.call(self) else hook.call end end end |
- (Object) run_teardown_hooks
:nodoc:
1168 1169 1170 1171 1172 1173 1174 1175 1176 |
# File 'lib/minitest/unit.rb', line 1168 def run_teardown_hooks # :nodoc: self.class.teardown_hooks.reverse.each do |hook| if hook.respond_to?(:arity) && hook.arity == 1 hook.call(self) else hook.call end end end |
- (Object) setup
Runs before every test. Use this to refactor test initialization.
1066 |
# File 'lib/minitest/unit.rb', line 1066 def setup; end |
- (Object) sigma(enum, &block)
Enumerates over enum mapping block if given, returning the sum of the result. Eg:
sigma([1, 2, 3]) # => 1 + 2 + 3 => 7
sigma([1, 2, 3]) { |n| n ** 2 } # => 1 + 4 + 9 => 14
284 285 286 287 |
# File 'lib/minitest/benchmark.rb', line 284 def sigma enum, &block enum = enum.map(&block) if block enum.inject { |sum, n| sum + n } end |
- (Object) teardown
Runs after every test. Use this to refactor test cleanup.
1071 |
# File 'lib/minitest/unit.rb', line 1071 def teardown; end |
- (Object) validation_for_fit(msg, threshold)
Returns a proc that calls the specified fit method and asserts that the error is within a tolerable threshold.
293 294 295 296 297 298 299 |
# File 'lib/minitest/benchmark.rb', line 293 def validation_for_fit msg, threshold proc do |range, times| a, b, rr = send "fit_#{msg}", range, times assert_operator rr, :>=, threshold [a, b, rr] end end |