1# frozen_string_literal: false
2require "test/unit"
3require "coverage"
4require "tmpdir"
5require "envutil"
6
7class TestCoverage < Test::Unit::TestCase
8  def test_result_without_start
9    assert_in_out_err(%w[-rcoverage], <<-"end;", [], /coverage measurement is not enabled/)
10      Coverage.result
11      p :NG
12    end;
13  end
14
15  def test_peek_result_without_start
16    assert_in_out_err(%w[-rcoverage], <<-"end;", [], /coverage measurement is not enabled/)
17      Coverage.peek_result
18      p :NG
19    end;
20  end
21
22  def test_result_with_nothing
23    assert_in_out_err(%w[-rcoverage], <<-"end;", ["{}"], [])
24      Coverage.start
25      p Coverage.result
26    end;
27  end
28
29  def test_coverage_running?
30    assert_in_out_err(%w[-rcoverage], <<-"end;", ["false", "true", "true", "false"], [])
31      p Coverage.running?
32      Coverage.start
33      p Coverage.running?
34      Coverage.peek_result
35      p Coverage.running?
36      Coverage.result
37      p Coverage.running?
38    end;
39  end
40
41  def test_coverage_snapshot
42    Dir.mktmpdir {|tmp|
43      Dir.chdir(tmp) {
44        File.open("test.rb", "w") do |f|
45          f.puts <<-EOS
46            def coverage_test_snapshot
47              :ok
48            end
49          EOS
50        end
51
52        assert_in_out_err(%w[-rcoverage], <<-"end;", ["[1, 0, nil]", "[1, 1, nil]", "[1, 1, nil]"], [])
53          Coverage.start
54          tmp = Dir.pwd
55          require tmp + "/test.rb"
56          cov = Coverage.peek_result[tmp + "/test.rb"]
57          coverage_test_snapshot
58          cov2 = Coverage.peek_result[tmp + "/test.rb"]
59          p cov
60          p cov2
61          p Coverage.result[tmp + "/test.rb"]
62        end;
63      }
64    }
65  end
66
67  def test_restarting_coverage
68    Dir.mktmpdir {|tmp|
69      Dir.chdir(tmp) {
70        tmp = Dir.pwd
71        File.open("test.rb", "w") do |f|
72          f.puts <<-EOS
73            def coverage_test_restarting
74              :ok
75            end
76          EOS
77        end
78
79        File.open("test2.rb", "w") do |f|
80          f.puts <<-EOS
81            itself
82          EOS
83        end
84
85        exp1 = { "#{tmp}/test.rb" => [1, 0, nil] }.inspect
86        exp2 = {}.inspect
87        exp3 = { "#{tmp}/test2.rb" => [1] }.inspect
88        assert_in_out_err(%w[-rcoverage], <<-"end;", [exp1, exp2, exp3], [])
89          Coverage.start
90          tmp = Dir.pwd
91          require tmp + "/test.rb"
92          p Coverage.result
93
94          # Restart coverage but '/test.rb' is required before restart,
95          # so coverage is not recorded.
96          Coverage.start
97          coverage_test_restarting
98          p Coverage.result
99
100          # Restart coverage and '/test2.rb' is required after restart,
101          # so coverage is recorded.
102          Coverage.start
103          require tmp + "/test2.rb"
104          p Coverage.result
105        end;
106      }
107    }
108  end
109
110  def test_big_code
111    Dir.mktmpdir {|tmp|
112      Dir.chdir(tmp) {
113        File.open("test.rb", "w") do |f|
114          f.puts "__id__\n" * 10000
115          f.puts "def ignore(x); end"
116          f.puts "ignore([1"
117          f.puts "])"
118        end
119
120        assert_in_out_err(%w[-rcoverage], <<-"end;", ["10003"], [])
121          Coverage.start
122          tmp = Dir.pwd
123          require tmp + '/test.rb'
124          p Coverage.result[tmp + '/test.rb'].size
125        end;
126      }
127    }
128  end
129
130  def test_eval
131    bug13305 = '[ruby-core:80079] [Bug #13305]'
132
133    Dir.mktmpdir {|tmp|
134      Dir.chdir(tmp) {
135        File.open("test.rb", "w") do |f|
136          f.puts 'REPEATS = 400'
137          f.puts 'def add_method(target)'
138          f.puts '  REPEATS.times do'
139          f.puts '    target.class_eval(<<~RUBY, __FILE__, __LINE__ + 1)'
140          f.puts '      def foo'
141          f.puts '        #{"\n" * rand(REPEATS)}'
142          f.puts '      end'
143          f.puts '      1'
144          f.puts '    RUBY'
145          f.puts '  end'
146          f.puts 'end'
147        end
148
149        assert_in_out_err(%w[-W0 -rcoverage], <<-"end;", ["[1, 1, 1, 400, nil, nil, nil, nil, nil, nil, nil]"], [], bug13305)
150          Coverage.start
151          tmp = Dir.pwd
152          require tmp + '/test.rb'
153          add_method(Class.new)
154          p Coverage.result[tmp + "/test.rb"]
155        end;
156      }
157    }
158  end
159
160  def test_nocoverage_optimized_line
161    assert_ruby_status(%w[], "#{<<-"begin;"}\n#{<<-'end;'}")
162    begin;
163      def foo(x)
164        x # optimized away
165        nil
166      end
167    end;
168  end
169
170  def test_coverage_optimized_branch
171    result = {
172      :branches => {
173        [:"&.", 0, 1, 0, 1, 8] => {
174          [:then, 1, 1, 0, 1, 8] => 0,
175          [:else, 2, 1, 0, 1, 8] => 1,
176        },
177      },
178    }
179    assert_coverage(<<~"end;", { branches: true }, result) # Bug #15476
180      nil&.foo
181    end;
182  end
183
184  def assert_coverage(code, opt, stdout)
185    stdout = [stdout] unless stdout.is_a?(Array)
186    stdout = stdout.map {|s| s.to_s }
187    Dir.mktmpdir {|tmp|
188      Dir.chdir(tmp) {
189        File.write("test.rb", code)
190
191        assert_in_out_err(%w[-W0 -rcoverage], <<-"end;", stdout, [])
192          Coverage.start(#{ opt })
193          tmp = Dir.pwd
194          require tmp + '/test.rb'
195          r = Coverage.result[tmp + "/test.rb"]
196          if r[:methods]
197            h = {}
198            r[:methods].keys.sort_by {|key| key.drop(1) }.each do |key|
199              h[key] = r[:methods][key]
200            end
201            r[:methods].replace h
202          end
203          p r
204        end;
205      }
206    }
207  end
208
209  def test_line_coverage_for_multiple_lines
210    result = {
211      :lines => [nil, 1, nil, nil, nil, 1, nil, nil, nil, 1, nil, 1, nil, nil, nil, nil, 1, 1, nil, 1, nil, nil, nil, nil, 1]
212    }
213    assert_coverage(<<~"end;", { lines: true }, result) # Bug #14191
214      FOO = [
215        { foo: 'bar' },
216        { bar: 'baz' }
217      ]
218
219      'some string'.split
220                   .map(&:length)
221
222      some =
223        'value'
224
225      Struct.new(
226        :foo,
227        :bar
228      ).new
229
230      class Test
231        def foo(bar)
232          {
233            foo: bar
234          }
235        end
236      end
237
238      Test.new.foo(Object.new)
239    end;
240  end
241
242  def test_branch_coverage_for_if_statement
243    result = {
244      :branches => {
245        [:if    ,  0,  2, 2,  6,  5] => {[:then,  1,  3,  4,  3,  5]=>2, [:else,  2,  5,  4,  5,  5]=>1},
246        [:unless,  3,  8, 2, 12,  5] => {[:else,  4, 11,  4, 11,  5]=>2, [:then,  5,  9,  4,  9,  5]=>1},
247        [:if    ,  6, 14, 2, 16,  5] => {[:then,  7, 15,  4, 15,  5]=>2, [:else,  8, 14,  2, 16,  5]=>1},
248        [:unless,  9, 18, 2, 20,  5] => {[:else, 10, 18,  2, 20,  5]=>2, [:then, 11, 19,  4, 19,  5]=>1},
249        [:if    , 12, 22, 2, 22, 13] => {[:then, 13, 22,  2, 22,  3]=>2, [:else, 14, 22,  2, 22, 13]=>1},
250        [:unless, 15, 23, 2, 23, 17] => {[:else, 16, 23,  2, 23, 17]=>2, [:then, 17, 23,  2, 23,  3]=>1},
251        [:if    , 18, 25, 2, 25, 16] => {[:then, 19, 25, 11, 25, 12]=>2, [:else, 20, 25, 15, 25, 16]=>1},
252      }
253    }
254    assert_coverage(<<~"end;", { branches: true }, result)
255      def foo(x)
256        if x == 0
257          0
258        else
259          1
260        end
261
262        unless x == 0
263          0
264        else
265          1
266        end
267
268        if x == 0
269          0
270        end
271
272        unless x == 0
273          0
274        end
275
276        0 if x == 0
277        0 unless x == 0
278
279        x == 0 ? 0 : 1
280      end
281
282      foo(0)
283      foo(0)
284      foo(1)
285    end;
286  end
287
288  def test_branch_coverage_for_while_statement
289    result = {
290      :branches => {
291        [:while, 0,  2, 0,  4,  3] => {[:body, 1,  3, 2,  3, 8]=> 3},
292        [:until, 2,  5, 0,  7,  3] => {[:body, 3,  6, 2,  6, 8]=>10},
293        [:while, 4, 10, 0, 10, 18] => {[:body, 5, 10, 0, 10, 6]=> 3},
294        [:until, 6, 11, 0, 11, 20] => {[:body, 7, 11, 0, 11, 6]=>10},
295      }
296    }
297    assert_coverage(<<~"end;", { branches: true }, result)
298      x = 3
299      while x > 0
300        x -= 1
301      end
302      until x == 10
303        x += 1
304      end
305
306      y = 3
307      y -= 1 while y > 0
308      y += 1 until y == 10
309    end;
310  end
311
312  def test_branch_coverage_for_case_statement
313    result = {
314      :branches => {
315        [:case,  0,  2, 2,  7, 5] => {[:when,  1,  4, 4,  4, 5]=>2, [:when,  2,  6, 4,  6, 5]=>0, [:else,  3,  2, 2,  7,  5]=>1},
316        [:case,  4,  9, 2, 14, 5] => {[:when,  5, 11, 4, 11, 5]=>2, [:when,  6, 13, 4, 13, 5]=>0, [:else,  7,  9, 2, 14,  5]=>1},
317        [:case,  8, 16, 2, 23, 5] => {[:when,  9, 18, 4, 18, 5]=>2, [:when, 10, 20, 4, 20, 5]=>0, [:else, 11, 22, 4, 22, 10]=>1},
318        [:case, 12, 25, 2, 32, 5] => {[:when, 13, 27, 4, 27, 5]=>2, [:when, 14, 29, 4, 29, 5]=>0, [:else, 15, 31, 4, 31, 10]=>1},
319      }
320    }
321    assert_coverage(<<~"end;", { branches: true }, result)
322      def foo(x)
323        case x
324        when 0
325          0
326        when 1
327          1
328        end
329
330        case
331        when x == 0
332          0
333        when x == 1
334          1
335        end
336
337        case x
338        when 0
339          0
340        when 1
341          1
342        else
343          :other
344        end
345
346        case
347        when x == 0
348          0
349        when x == 1
350          1
351        else
352          :other
353        end
354      end
355
356      foo(0)
357      foo(0)
358      foo(2)
359    end;
360  end
361
362  def test_branch_coverage_for_safe_method_invocation
363    result = {
364      :branches=>{
365        [:"&.", 0, 6, 0, 6,  6] => {[:then,  1, 6, 0, 6,  6]=>1, [:else,  2, 6, 0, 6,  6]=>0},
366        [:"&.", 3, 7, 0, 7,  6] => {[:then,  4, 7, 0, 7,  6]=>0, [:else,  5, 7, 0, 7,  6]=>1},
367        [:"&.", 6, 8, 0, 8, 10] => {[:then,  7, 8, 0, 8, 10]=>1, [:else,  8, 8, 0, 8, 10]=>0},
368        [:"&.", 9, 9, 0, 9, 10] => {[:then, 10, 9, 0, 9, 10]=>0, [:else, 11, 9, 0, 9, 10]=>1},
369      }
370    }
371    assert_coverage(<<~"end;", { branches: true }, result)
372      class Dummy; def foo; end; def foo=(x); end; end
373      a = Dummy.new
374      b = nil
375      c = Dummy.new
376      d = nil
377      a&.foo
378      b&.foo
379      c&.foo = 1
380      d&.foo = 1
381    end;
382  end
383
384  def test_method_coverage
385    result = {
386      :methods => {
387        [Object, :bar, 2, 0, 3, 3] => 1,
388        [Object, :baz, 4, 1, 4, 13] => 0,
389        [Object, :foo, 1, 0, 1, 12] => 2,
390      }
391    }
392    assert_coverage(<<~"end;", { methods: true }, result)
393      def foo; end
394      def bar
395      end
396       def baz; end
397
398      foo
399      foo
400      bar
401    end;
402  end
403
404  def test_method_coverage_for_define_method
405    result = {
406      :methods => {
407        [Object, :a, 6, 18, 6, 25] => 2,
408        [Object, :b, 7, 18, 8, 3] => 0,
409        [Object, :bar, 2, 20, 3, 1] => 1,
410        [Object, :baz, 4, 9, 4, 11] => 0,
411        [Object, :foo, 1, 20, 1, 22] => 2,
412      }
413    }
414    assert_coverage(<<~"end;", { methods: true }, result)
415      define_method(:foo) {}
416      define_method(:bar) {
417      }
418      f = proc {}
419      define_method(:baz, &f)
420      define_method(:a) do; end
421      define_method(:b) do
422      end
423
424      foo
425      foo
426      bar
427      a
428      a
429    end;
430  end
431
432  class DummyConstant < String
433    def inspect
434      self
435    end
436  end
437
438  def test_method_coverage_for_alias
439    _C = DummyConstant.new("C")
440    _M = DummyConstant.new("M")
441    code = <<~"end;"
442      module M
443        def foo
444        end
445        alias bar foo
446      end
447      class C
448        include M
449        def baz
450        end
451        alias qux baz
452      end
453    end;
454
455    result = {
456      :methods => {
457        [_C, :baz, 8, 2, 9, 5] => 0,
458        [_M, :foo, 2, 2, 3, 5] => 0,
459      }
460    }
461    assert_coverage(code, { methods: true }, result)
462
463    result = {
464      :methods => {
465        [_C, :baz, 8, 2, 9, 5] => 12,
466        [_M, :foo, 2, 2, 3, 5] =>  3,
467      }
468    }
469    assert_coverage(code + <<~"end;", { methods: true }, result)
470      obj = C.new
471      1.times { obj.foo }
472      2.times { obj.bar }
473      4.times { obj.baz }
474      8.times { obj.qux }
475    end;
476  end
477
478  def test_method_coverage_for_singleton_class
479    _singleton_Foo = DummyConstant.new("#<Class:Foo>")
480    _Foo = DummyConstant.new("Foo")
481    code = <<~"end;"
482      class Foo
483        def foo
484        end
485        alias bar foo
486        def self.baz
487        end
488        class << self
489          alias qux baz
490        end
491      end
492
493      1.times { Foo.new.foo }
494      2.times { Foo.new.bar }
495      4.times { Foo.baz }
496      8.times { Foo.qux }
497    end;
498
499    result = {
500      :methods => {
501        [_singleton_Foo, :baz, 5, 2, 6, 5] => 12,
502        [_Foo, :foo, 2, 2, 3, 5] => 3,
503      }
504    }
505    assert_coverage(code, { methods: true }, result)
506  end
507
508  def test_oneshot_line_coverage
509    result = {
510      :oneshot_lines => [2, 6, 10, 12, 17, 18, 25, 20]
511    }
512    assert_coverage(<<~"end;", { oneshot_lines: true }, result)
513      FOO = [
514        { foo: 'bar' }, # 2
515        { bar: 'baz' }
516      ]
517
518      'some string'.split # 6
519                   .map(&:length)
520
521      some =
522        'value' # 10
523
524      Struct.new( # 12
525        :foo,
526        :bar
527      ).new
528
529      class Test # 17
530        def foo(bar) # 18
531          {
532            foo: bar # 20
533          }
534        end
535      end
536
537      Test.new.foo(Object.new) # 25
538    end;
539  end
540
541  def test_clear_with_lines
542    Dir.mktmpdir {|tmp|
543      Dir.chdir(tmp) {
544        File.open("test.rb", "w") do |f|
545          f.puts "def foo(x)"
546          f.puts "  if x > 0"
547          f.puts "    :pos"
548          f.puts "  else"
549          f.puts "    :non_pos"
550          f.puts "  end"
551          f.puts "end"
552        end
553
554        exp = [
555          "{:lines=>[1, 0, 0, nil, 0, nil, nil]}",
556          "{:lines=>[0, 1, 1, nil, 0, nil, nil]}",
557          "{:lines=>[0, 1, 0, nil, 1, nil, nil]}",
558        ]
559        assert_in_out_err(%w[-rcoverage], <<-"end;", exp, [])
560          Coverage.start(lines: true)
561          tmp = Dir.pwd
562          f = tmp + "/test.rb"
563          require f
564          p Coverage.result(stop: false, clear: true)[f]
565          foo(1)
566          p Coverage.result(stop: false, clear: true)[f]
567          foo(-1)
568          p Coverage.result[f]
569        end;
570      }
571    }
572  end
573
574  def test_clear_with_branches
575    Dir.mktmpdir {|tmp|
576      Dir.chdir(tmp) {
577        File.open("test.rb", "w") do |f|
578          f.puts "def foo(x)"
579          f.puts "  if x > 0"
580          f.puts "    :pos"
581          f.puts "  else"
582          f.puts "    :non_pos"
583          f.puts "  end"
584          f.puts "end"
585        end
586
587        exp = [
588          "{:branches=>{[:if, 0, 2, 2, 6, 5]=>{[:then, 1, 3, 4, 3, 8]=>0, [:else, 2, 5, 4, 5, 12]=>0}}}",
589          "{:branches=>{[:if, 0, 2, 2, 6, 5]=>{[:then, 1, 3, 4, 3, 8]=>1, [:else, 2, 5, 4, 5, 12]=>0}}}",
590          "{:branches=>{[:if, 0, 2, 2, 6, 5]=>{[:then, 1, 3, 4, 3, 8]=>0, [:else, 2, 5, 4, 5, 12]=>1}}}",
591          "{:branches=>{[:if, 0, 2, 2, 6, 5]=>{[:then, 1, 3, 4, 3, 8]=>0, [:else, 2, 5, 4, 5, 12]=>1}}}",
592        ]
593        assert_in_out_err(%w[-rcoverage], <<-"end;", exp, [])
594          Coverage.start(branches: true)
595          tmp = Dir.pwd
596          f = tmp + "/test.rb"
597          require f
598          p Coverage.result(stop: false, clear: true)[f]
599          foo(1)
600          p Coverage.result(stop: false, clear: true)[f]
601          foo(-1)
602          p Coverage.result(stop: false, clear: true)[f]
603          foo(-1)
604          p Coverage.result(stop: false, clear: true)[f]
605        end;
606      }
607    }
608  end
609
610  def test_clear_with_methods
611    Dir.mktmpdir {|tmp|
612      Dir.chdir(tmp) {
613        File.open("test.rb", "w") do |f|
614          f.puts "def foo(x)"
615          f.puts "  if x > 0"
616          f.puts "    :pos"
617          f.puts "  else"
618          f.puts "    :non_pos"
619          f.puts "  end"
620          f.puts "end"
621        end
622
623        exp = [
624          "{:methods=>{[Object, :foo, 1, 0, 7, 3]=>0}}",
625          "{:methods=>{[Object, :foo, 1, 0, 7, 3]=>1}}",
626          "{:methods=>{[Object, :foo, 1, 0, 7, 3]=>1}}",
627          "{:methods=>{[Object, :foo, 1, 0, 7, 3]=>1}}"
628        ]
629        assert_in_out_err(%w[-rcoverage], <<-"end;", exp, [])
630          Coverage.start(methods: true)
631          tmp = Dir.pwd
632          f = tmp + "/test.rb"
633          require f
634          p Coverage.result(stop: false, clear: true)[f]
635          foo(1)
636          p Coverage.result(stop: false, clear: true)[f]
637          foo(-1)
638          p Coverage.result(stop: false, clear: true)[f]
639          foo(-1)
640          p Coverage.result(stop: false, clear: true)[f]
641        end;
642      }
643    }
644  end
645
646  def test_clear_with_oneshot_lines
647    Dir.mktmpdir {|tmp|
648      Dir.chdir(tmp) {
649        File.open("test.rb", "w") do |f|
650          f.puts "def foo(x)"
651          f.puts "  if x > 0"
652          f.puts "    :pos"
653          f.puts "  else"
654          f.puts "    :non_pos"
655          f.puts "  end"
656          f.puts "end"
657        end
658
659        exp = [
660          "{:oneshot_lines=>[1]}",
661          "{:oneshot_lines=>[2, 3]}",
662          "{:oneshot_lines=>[5]}",
663          "{:oneshot_lines=>[]}",
664        ]
665        assert_in_out_err(%w[-rcoverage], <<-"end;", exp, [])
666          Coverage.start(oneshot_lines: true)
667          tmp = Dir.pwd
668          f = tmp + "/test.rb"
669          require f
670          p Coverage.result(stop: false, clear: true)[f]
671          foo(1)
672          p Coverage.result(stop: false, clear: true)[f]
673          foo(-1)
674          p Coverage.result(stop: false, clear: true)[f]
675          foo(-1)
676          p Coverage.result(stop: false, clear: true)[f]
677        end;
678      }
679    }
680  end
681
682  def test_line_stub
683    Dir.mktmpdir {|tmp|
684      Dir.chdir(tmp) {
685        File.open("test.rb", "w") do |f|
686          f.puts "def foo(x)"
687          f.puts "  if x > 0"
688          f.puts "    :pos"
689          f.puts "  else"
690          f.puts "    :non_pos"
691          f.puts "  end"
692          f.puts "end"
693        end
694
695        assert_equal([0, 0, 0, nil, 0, nil, nil], Coverage.line_stub("test.rb"))
696      }
697    }
698  end
699
700  def test_stop_wrong_peephole_optimization
701    result = {
702      :lines => [1, 1, 1, nil]
703    }
704    assert_coverage(<<~"end;", { lines: true }, result)
705      raise if 1 == 2
706      while true
707        break
708      end
709    end;
710  end
711end
712