ruby-prof 0.18.0-x64-mingw32

Sign up to get free protection for your applications and to get access to all the features.
Files changed (108) hide show
  1. checksums.yaml +7 -0
  2. data/CHANGES +500 -0
  3. data/LICENSE +25 -0
  4. data/README.rdoc +487 -0
  5. data/Rakefile +113 -0
  6. data/bin/ruby-prof +345 -0
  7. data/bin/ruby-prof-check-trace +45 -0
  8. data/examples/flat.txt +50 -0
  9. data/examples/graph.dot +84 -0
  10. data/examples/graph.html +823 -0
  11. data/examples/graph.txt +139 -0
  12. data/examples/multi.flat.txt +23 -0
  13. data/examples/multi.graph.html +760 -0
  14. data/examples/multi.grind.dat +114 -0
  15. data/examples/multi.stack.html +547 -0
  16. data/examples/stack.html +547 -0
  17. data/ext/ruby_prof/extconf.rb +68 -0
  18. data/ext/ruby_prof/rp_call_info.c +425 -0
  19. data/ext/ruby_prof/rp_call_info.h +53 -0
  20. data/ext/ruby_prof/rp_measure.c +40 -0
  21. data/ext/ruby_prof/rp_measure.h +45 -0
  22. data/ext/ruby_prof/rp_measure_allocations.c +76 -0
  23. data/ext/ruby_prof/rp_measure_cpu_time.c +136 -0
  24. data/ext/ruby_prof/rp_measure_gc_runs.c +73 -0
  25. data/ext/ruby_prof/rp_measure_gc_time.c +60 -0
  26. data/ext/ruby_prof/rp_measure_memory.c +77 -0
  27. data/ext/ruby_prof/rp_measure_process_time.c +71 -0
  28. data/ext/ruby_prof/rp_measure_wall_time.c +45 -0
  29. data/ext/ruby_prof/rp_method.c +630 -0
  30. data/ext/ruby_prof/rp_method.h +75 -0
  31. data/ext/ruby_prof/rp_stack.c +173 -0
  32. data/ext/ruby_prof/rp_stack.h +63 -0
  33. data/ext/ruby_prof/rp_thread.c +277 -0
  34. data/ext/ruby_prof/rp_thread.h +27 -0
  35. data/ext/ruby_prof/ruby_prof.c +794 -0
  36. data/ext/ruby_prof/ruby_prof.h +60 -0
  37. data/ext/ruby_prof/vc/ruby_prof.sln +31 -0
  38. data/ext/ruby_prof/vc/ruby_prof.vcxproj +141 -0
  39. data/lib/2.6.3/ruby_prof.so +0 -0
  40. data/lib/ruby-prof.rb +68 -0
  41. data/lib/ruby-prof/aggregate_call_info.rb +76 -0
  42. data/lib/ruby-prof/assets/call_stack_printer.css.html +117 -0
  43. data/lib/ruby-prof/assets/call_stack_printer.js.html +385 -0
  44. data/lib/ruby-prof/assets/call_stack_printer.png +0 -0
  45. data/lib/ruby-prof/call_info.rb +115 -0
  46. data/lib/ruby-prof/call_info_visitor.rb +40 -0
  47. data/lib/ruby-prof/compatibility.rb +179 -0
  48. data/lib/ruby-prof/method_info.rb +121 -0
  49. data/lib/ruby-prof/printers/abstract_printer.rb +104 -0
  50. data/lib/ruby-prof/printers/call_info_printer.rb +41 -0
  51. data/lib/ruby-prof/printers/call_stack_printer.rb +265 -0
  52. data/lib/ruby-prof/printers/call_tree_printer.rb +143 -0
  53. data/lib/ruby-prof/printers/dot_printer.rb +132 -0
  54. data/lib/ruby-prof/printers/flat_printer.rb +70 -0
  55. data/lib/ruby-prof/printers/flat_printer_with_line_numbers.rb +83 -0
  56. data/lib/ruby-prof/printers/graph_html_printer.rb +249 -0
  57. data/lib/ruby-prof/printers/graph_printer.rb +116 -0
  58. data/lib/ruby-prof/printers/multi_printer.rb +84 -0
  59. data/lib/ruby-prof/profile.rb +26 -0
  60. data/lib/ruby-prof/profile/exclude_common_methods.rb +207 -0
  61. data/lib/ruby-prof/profile/legacy_method_elimination.rb +50 -0
  62. data/lib/ruby-prof/rack.rb +174 -0
  63. data/lib/ruby-prof/task.rb +147 -0
  64. data/lib/ruby-prof/thread.rb +35 -0
  65. data/lib/ruby-prof/version.rb +3 -0
  66. data/lib/unprof.rb +10 -0
  67. data/ruby-prof.gemspec +58 -0
  68. data/test/abstract_printer_test.rb +53 -0
  69. data/test/aggregate_test.rb +136 -0
  70. data/test/basic_test.rb +128 -0
  71. data/test/block_test.rb +74 -0
  72. data/test/call_info_test.rb +78 -0
  73. data/test/call_info_visitor_test.rb +31 -0
  74. data/test/duplicate_names_test.rb +32 -0
  75. data/test/dynamic_method_test.rb +55 -0
  76. data/test/enumerable_test.rb +21 -0
  77. data/test/exceptions_test.rb +24 -0
  78. data/test/exclude_methods_test.rb +146 -0
  79. data/test/exclude_threads_test.rb +53 -0
  80. data/test/fiber_test.rb +79 -0
  81. data/test/issue137_test.rb +63 -0
  82. data/test/line_number_test.rb +80 -0
  83. data/test/measure_allocations_test.rb +26 -0
  84. data/test/measure_cpu_time_test.rb +212 -0
  85. data/test/measure_gc_runs_test.rb +32 -0
  86. data/test/measure_gc_time_test.rb +36 -0
  87. data/test/measure_memory_test.rb +33 -0
  88. data/test/measure_process_time_test.rb +61 -0
  89. data/test/measure_wall_time_test.rb +255 -0
  90. data/test/method_elimination_test.rb +84 -0
  91. data/test/module_test.rb +45 -0
  92. data/test/multi_printer_test.rb +104 -0
  93. data/test/no_method_class_test.rb +15 -0
  94. data/test/pause_resume_test.rb +166 -0
  95. data/test/prime.rb +54 -0
  96. data/test/printers_test.rb +275 -0
  97. data/test/printing_recursive_graph_test.rb +127 -0
  98. data/test/rack_test.rb +157 -0
  99. data/test/recursive_test.rb +215 -0
  100. data/test/singleton_test.rb +38 -0
  101. data/test/stack_printer_test.rb +77 -0
  102. data/test/stack_test.rb +138 -0
  103. data/test/start_stop_test.rb +112 -0
  104. data/test/test_helper.rb +267 -0
  105. data/test/thread_test.rb +187 -0
  106. data/test/unique_call_path_test.rb +202 -0
  107. data/test/yarv_test.rb +55 -0
  108. metadata +199 -0
@@ -0,0 +1,128 @@
1
+ #!/usr/bin/env ruby
2
+ # encoding: UTF-8
3
+
4
+ require File.expand_path('../test_helper', __FILE__)
5
+
6
+ class BasicTest < TestCase
7
+ def setup
8
+ # Need to use wall time for this test due to the sleep calls
9
+ RubyProf::measure_mode = RubyProf::WALL_TIME
10
+ end
11
+
12
+ def start
13
+ RubyProf.start
14
+ RubyProf::C1.hello
15
+ end
16
+
17
+ def test_running
18
+ assert(!RubyProf.running?)
19
+ RubyProf.start
20
+ assert(RubyProf.running?)
21
+ RubyProf.stop
22
+ assert(!RubyProf.running?)
23
+ end
24
+
25
+ def test_double_profile
26
+ RubyProf.start
27
+ assert_raises(RuntimeError) do
28
+ RubyProf.start
29
+ end
30
+ RubyProf.stop
31
+ end
32
+
33
+ def test_no_block
34
+ assert_raises(ArgumentError) do
35
+ RubyProf.profile
36
+ end
37
+ end
38
+
39
+ def test_traceback
40
+ RubyProf.start
41
+ assert_raises(NoMethodError) do
42
+ RubyProf.xxx
43
+ end
44
+
45
+ RubyProf.stop
46
+ end
47
+
48
+ def test_leave_method
49
+ start
50
+ sleep 0.15
51
+ profile = RubyProf.stop
52
+
53
+ assert_equal(1, profile.threads.count)
54
+
55
+ thread = profile.threads.first
56
+ assert_in_delta(0.25, thread.total_time, 0.015)
57
+
58
+ top_methods = thread.top_methods.sort
59
+ assert_equal(2, top_methods.count)
60
+ assert_equal("BasicTest#start", top_methods[0].full_name)
61
+ assert_equal("BasicTest#test_leave_method", top_methods[1].full_name)
62
+
63
+ assert_equal(4, thread.methods.length)
64
+ methods = profile.threads.first.methods.sort
65
+
66
+ # Check times
67
+ assert_equal("<Class::RubyProf::C1>#hello", methods[0].full_name)
68
+ assert_in_delta(0.1, methods[0].total_time, 0.015)
69
+ assert_in_delta(0.0, methods[0].wait_time, 0.015)
70
+ assert_in_delta(0.0, methods[0].self_time, 0.015)
71
+
72
+ assert_equal("BasicTest#start", methods[1].full_name)
73
+ assert_in_delta(0.1, methods[1].total_time, 0.015)
74
+ assert_in_delta(0.0, methods[1].wait_time, 0.015)
75
+ assert_in_delta(0.0, methods[1].self_time, 0.015)
76
+
77
+ assert_equal("BasicTest#test_leave_method", methods[2].full_name)
78
+ assert_in_delta(0.15, methods[2].total_time, 0.015)
79
+ assert_in_delta(0.0, methods[2].wait_time, 0.015)
80
+ assert_in_delta(0.0, methods[2].self_time, 0.015)
81
+
82
+ assert_equal("Kernel#sleep", methods[3].full_name)
83
+ assert_in_delta(0.25, methods[3].total_time, 0.015)
84
+ assert_in_delta(0.0, methods[3].wait_time, 0.015)
85
+ assert_in_delta(0.25, methods[3].self_time, 0.015)
86
+ end
87
+
88
+ def test_leave_method_2
89
+ start
90
+ RubyProf::C1.hello
91
+ RubyProf::C1.hello
92
+ profile = RubyProf.stop
93
+
94
+ assert_equal(1, profile.threads.count)
95
+
96
+ thread = profile.threads.first
97
+ assert_in_delta(0.3, thread.total_time, 0.015)
98
+
99
+ top_methods = thread.top_methods.sort
100
+ assert_equal(2, top_methods.count)
101
+ assert_equal("BasicTest#start", top_methods[0].full_name)
102
+ assert_equal("BasicTest#test_leave_method_2", top_methods[1].full_name)
103
+
104
+ assert_equal(4, thread.methods.length)
105
+ methods = profile.threads.first.methods.sort
106
+
107
+ # Check times
108
+ assert_equal("BasicTest#start", methods[0].full_name)
109
+ assert_in_delta(0.1, methods[0].total_time, 0.015)
110
+ assert_in_delta(0.0, methods[0].wait_time, 0.015)
111
+ assert_in_delta(0.0, methods[0].self_time, 0.015)
112
+
113
+ assert_equal("BasicTest#test_leave_method_2", methods[1].full_name)
114
+ assert_in_delta(0.2, methods[1].total_time, 0.015)
115
+ assert_in_delta(0.0, methods[1].wait_time, 0.015)
116
+ assert_in_delta(0.0, methods[1].self_time, 0.015)
117
+
118
+ assert_equal("Kernel#sleep", methods[2].full_name)
119
+ assert_in_delta(0.3, methods[2].total_time, 0.015)
120
+ assert_in_delta(0.0, methods[2].wait_time, 0.015)
121
+ assert_in_delta(0.3, methods[2].self_time, 0.015)
122
+
123
+ assert_equal("<Class::RubyProf::C1>#hello", methods[3].full_name)
124
+ assert_in_delta(0.3, methods[3].total_time, 0.015)
125
+ assert_in_delta(0.0, methods[3].wait_time, 0.015)
126
+ assert_in_delta(0.0, methods[3].self_time, 0.015)
127
+ end
128
+ end
@@ -0,0 +1,74 @@
1
+ #!/usr/bin/env ruby
2
+ # encoding: UTF-8
3
+
4
+ require File.expand_path("../test_helper", __FILE__)
5
+
6
+ class BlockMethodTest < TestCase
7
+ def setup
8
+ # Need to use wall time for this test due to the sleep calls
9
+ RubyProf::measure_mode = RubyProf::WALL_TIME
10
+ end
11
+
12
+ def test_block
13
+ result = RubyProf.profile do
14
+ 1.times { RubyProf::C1.new.hello }
15
+ end
16
+
17
+ # Methods called
18
+ # Kernel#sleep
19
+ # <Class::BasicObject>#allocate
20
+ # #{RubyProf.parent_object}#inizialize
21
+ # RubyProf::C1#hello
22
+ # Class#new
23
+ # Integer#times
24
+ # BlockMethodTest#test_block
25
+
26
+ methods = result.threads.first.methods.sort.reverse
27
+ assert_equal(RubyProf.ruby_2? ? 6 : 7, methods.length)
28
+
29
+ # Check times
30
+ assert_equal("BlockMethodTest#test_block", methods[0].full_name)
31
+ assert_in_delta(0.2, methods[0].total_time, 0.02)
32
+ assert_in_delta(0.0, methods[0].wait_time, 0.02)
33
+ assert_in_delta(0.0, methods[0].self_time, 0.02)
34
+
35
+ assert_equal("Integer#times", methods[1].full_name)
36
+ assert_in_delta(0.2, methods[1].total_time, 0.02)
37
+ assert_in_delta(0.0, methods[1].wait_time, 0.02)
38
+ assert_in_delta(0.0, methods[1].self_time, 0.02)
39
+
40
+ assert_equal("RubyProf::C1#hello", methods[2].full_name)
41
+ assert_in_delta(0.2, methods[2].total_time, 0.02)
42
+ assert_in_delta(0.0, methods[2].wait_time, 0.02)
43
+ assert_in_delta(0.0, methods[2].self_time, 0.02)
44
+
45
+ assert_equal("Kernel#sleep", methods[3].full_name)
46
+ assert_in_delta(0.2, methods[3].total_time, 0.01)
47
+ assert_in_delta(0.0, methods[3].wait_time, 0.01)
48
+ assert_in_delta(0.2, methods[3].self_time, 0.01)
49
+
50
+ assert_equal("Class#new", methods[4].full_name)
51
+ assert_in_delta(0.0, methods[4].total_time, 0.01)
52
+ assert_in_delta(0.0, methods[4].wait_time, 0.01)
53
+ assert_in_delta(0.0, methods[4].self_time, 0.01)
54
+
55
+ # the timing difference between #initialize and #allocate is so small
56
+ # that we cannot rely on #initialize appearing first.
57
+ # so switch them, if necessary
58
+ if methods[5].full_name =~ /#allocate/
59
+ methods[5], methods[6] = methods[6], methods[5]
60
+ end
61
+
62
+ assert_equal("#{RubyProf.parent_object}#initialize", methods[5].full_name)
63
+ assert_in_delta(0.0, methods[5].total_time, 0.01)
64
+ assert_in_delta(0.0, methods[5].wait_time, 0.01)
65
+ assert_in_delta(0.0, methods[5].self_time, 0.01)
66
+
67
+ unless RubyProf.ruby_2?
68
+ assert_equal("<Class::#{RubyProf.parent_object}>#allocate", methods[6].full_name)
69
+ assert_in_delta(0.0, methods[6].total_time, 0.01)
70
+ assert_in_delta(0.0, methods[6].wait_time, 0.01)
71
+ assert_in_delta(0.0, methods[6].self_time, 0.01)
72
+ end
73
+ end
74
+ end
@@ -0,0 +1,78 @@
1
+ #!/usr/bin/env ruby
2
+ # encoding: UTF-8
3
+
4
+ require File.expand_path('../test_helper', __FILE__)
5
+
6
+ class CallInfoTest < TestCase
7
+ def setup
8
+ # Need to use wall time for this test due to the sleep calls
9
+ RubyProf::measure_mode = RubyProf::WALL_TIME
10
+ end
11
+
12
+ # def test_clone
13
+ # result = RubyProf.profile do
14
+ # RubyProf::C1.hello
15
+ # end
16
+ #
17
+ # method = result.threads.first.top_methods.first
18
+ # call_info = method.call_infos.first
19
+ # call_info_clone = call_info.clone
20
+ #
21
+ ## assert_equal(call_info.target, call_info_clone.target)
22
+ # assert_equal(call_info.total_time, call_info_clone.total_time)
23
+ # end
24
+
25
+ def test_merge
26
+ result1 = RubyProf.profile do
27
+ RubyProf::C1.hello
28
+ end
29
+
30
+ methods = result1.threads.first.methods.sort.reverse
31
+ assert_equal(3, methods.length)
32
+
33
+ assert_equal('CallInfoTest#test_merge', methods[0].full_name)
34
+ assert_in_delta(0.1, methods[0].total_time, 0.01)
35
+ assert_in_delta(0, methods[0].wait_time, 0.01)
36
+ assert_in_delta(0, methods[0].self_time, 0.01)
37
+ assert_equal(1, methods[0].called)
38
+
39
+ assert_equal('<Class::RubyProf::C1>#hello', methods[1].full_name)
40
+ assert_in_delta(0.1, methods[1].total_time, 0.01)
41
+ assert_in_delta(0, methods[1].wait_time, 0.01)
42
+ assert_in_delta(0, methods[1].self_time, 0.01)
43
+ assert_equal(1, methods[1].called)
44
+
45
+ assert_equal('Kernel#sleep', methods[2].full_name)
46
+ assert_in_delta(0.1, methods[2].total_time, 0.01)
47
+ assert_in_delta(0, methods[2].wait_time, 0.01)
48
+ assert_in_delta(0.1, methods[2].self_time, 0.01)
49
+ assert_equal(1, methods[2].called)
50
+
51
+ RubyProf.profile do
52
+ RubyProf::C1.hello
53
+ end
54
+
55
+ # Merge the trees
56
+ methods = result1.threads.first.methods.sort.reverse
57
+ assert_equal(3, methods.length)
58
+
59
+ assert_equal('CallInfoTest#test_merge', methods[0].full_name)
60
+ assert_in_delta(0.1, methods[0].total_time, 0.01)
61
+ assert_in_delta(0, methods[0].wait_time, 0.01)
62
+ assert_in_delta(0, methods[0].self_time, 0.01)
63
+ assert_equal(1, methods[0].called)
64
+
65
+ assert_equal('<Class::RubyProf::C1>#hello', methods[1].full_name)
66
+ assert_in_delta(0.1, methods[1].total_time, 0.01)
67
+ assert_in_delta(0, methods[1].wait_time, 0.01)
68
+ assert_in_delta(0, methods[1].self_time, 0.01)
69
+ assert_equal(1, methods[1].called)
70
+
71
+ assert_equal('Kernel#sleep', methods[2].full_name)
72
+ assert_in_delta(0.1, methods[2].total_time, 0.01)
73
+ assert_in_delta(0, methods[2].wait_time, 0.01)
74
+ assert_in_delta(0.1, methods[2].self_time, 0.01)
75
+ assert_equal(1, methods[2].called)
76
+ end
77
+ end
78
+
@@ -0,0 +1,31 @@
1
+ #!/usr/bin/env ruby
2
+ # encoding: UTF-8
3
+
4
+ require File.expand_path('../test_helper', __FILE__)
5
+
6
+ class CallInfoVisitorTest < TestCase
7
+ def setup
8
+ # Need to use wall time for this test due to the sleep calls
9
+ RubyProf::measure_mode = RubyProf::WALL_TIME
10
+ end
11
+
12
+ def test_visit
13
+ result = RubyProf.profile do
14
+ RubyProf::C1.hello
15
+ end
16
+
17
+ visitor = RubyProf::CallInfoVisitor.new(result.threads.first.top_call_infos)
18
+
19
+ method_names = Array.new
20
+
21
+ visitor.visit do |call_info, event|
22
+ method_names << call_info.target.full_name if event == :enter
23
+ end
24
+
25
+ assert_equal(3, method_names.length)
26
+ assert_equal("CallInfoVisitorTest#test_visit", method_names[0])
27
+ assert_equal("<Class::RubyProf::C1>#hello", method_names[1])
28
+ assert_equal("Kernel#sleep", method_names[2])
29
+ end
30
+ end
31
+
@@ -0,0 +1,32 @@
1
+ #!/usr/bin/env ruby
2
+ # encoding: UTF-8
3
+
4
+ require File.expand_path('../test_helper', __FILE__)
5
+
6
+ class DuplicateNames < TestCase
7
+ def test_names
8
+ result = RubyProf::profile do
9
+ str = %{module Foo; class Bar; def foo; end end end}
10
+
11
+ eval str
12
+ Foo::Bar.new.foo
13
+ DuplicateNames.class_eval {remove_const :Foo}
14
+
15
+ eval str
16
+ Foo::Bar.new.foo
17
+ DuplicateNames.class_eval {remove_const :Foo}
18
+
19
+ eval str
20
+ Foo::Bar.new.foo
21
+ end
22
+
23
+ # There should be 3 foo methods
24
+ methods = result.threads.first.methods.sort.reverse
25
+
26
+ methods = methods.select do |method|
27
+ method.full_name == 'DuplicateNames::Foo::Bar#foo'
28
+ end
29
+
30
+ assert_equal(3, methods.length)
31
+ end
32
+ end
@@ -0,0 +1,55 @@
1
+ #!/usr/bin/env ruby
2
+ # encoding: UTF-8
3
+
4
+ require File.expand_path("../test_helper", __FILE__)
5
+
6
+ class DynamicMethodTest < TestCase
7
+
8
+ class FruitMedley
9
+ define_method(:apple) do
10
+ sleep(0.1)
11
+ "I'm a peach"
12
+ end
13
+
14
+ define_method(:orange) do
15
+ sleep(0.2)
16
+ "I'm an orange"
17
+ end
18
+
19
+ [:banana, :peach].each_with_index do |fruit,i|
20
+ define_method(fruit) do
21
+ sleep(i == 0 ? 0.3 : 0.4)
22
+ "I'm a #{fruit}"
23
+ end
24
+ end
25
+ end
26
+
27
+ def setup
28
+ # Need to use wall time for this test due to the sleep calls
29
+ RubyProf::measure_mode = RubyProf::WALL_TIME
30
+ end
31
+
32
+ def test_dynamic_method
33
+ medley = FruitMedley.new
34
+ result = RubyProf.profile do
35
+ medley.apple
36
+ medley.orange
37
+ medley.banana
38
+ medley.peach
39
+ end
40
+
41
+ # RubyProf::FlatPrinter.new(result).print(STDOUT)
42
+
43
+ methods = result.threads.first.methods.sort.reverse
44
+ expected_method_names = %w(
45
+ DynamicMethodTest#test_dynamic_method
46
+ Kernel#sleep
47
+ DynamicMethodTest::FruitMedley#peach
48
+ DynamicMethodTest::FruitMedley#banana
49
+ DynamicMethodTest::FruitMedley#orange
50
+ DynamicMethodTest::FruitMedley#apple
51
+ Symbol#to_s
52
+ )
53
+ assert_equal expected_method_names.join("\n"), methods.map(&:full_name).join("\n")
54
+ end
55
+ end
@@ -0,0 +1,21 @@
1
+ #!/usr/bin/env ruby
2
+ # encoding: UTF-8
3
+
4
+ require File.expand_path('../test_helper', __FILE__)
5
+
6
+ # -- Test for bug
7
+ # http://github.com/rdp/ruby-prof/issues#issue/12
8
+
9
+ class EnumerableTest < TestCase
10
+ def test_enumerable
11
+ result = RubyProf.profile do
12
+ 3.times { [1,2,3].any? {|n| n} }
13
+ end
14
+ methods = if RUBY_VERSION >= "2.2.0"
15
+ %w(EnumerableTest#test_enumerable Integer#times Array#any?)
16
+ else
17
+ %w(EnumerableTest#test_enumerable Integer#times Enumerable#any? Array#each)
18
+ end
19
+ assert_equal(methods, result.threads.first.methods.map(&:full_name))
20
+ end
21
+ end
@@ -0,0 +1,24 @@
1
+ #!/usr/bin/env ruby
2
+ # encoding: UTF-8
3
+
4
+ require File.expand_path('../test_helper', __FILE__)
5
+
6
+ class ExceptionsTest < TestCase
7
+ def test_profile
8
+ result = begin
9
+ RubyProf.profile do
10
+ raise(RuntimeError, 'Test error')
11
+ end
12
+ rescue
13
+ end
14
+ assert_kind_of(RubyProf::Profile, result)
15
+ end
16
+
17
+ def test_profile_allows_exceptions
18
+ assert_raises(RuntimeError) do
19
+ RubyProf.profile(:allow_exceptions => true) do
20
+ raise(RuntimeError, 'Test error')
21
+ end
22
+ end
23
+ end
24
+ end