Skip to content

Commit f07f2b9

Browse files
authored
Merge pull request #558 from aroben/merge-without-extend
Rewrite merge helpers to work with frozen objects
2 parents 3ca3141 + c7a9d82 commit f07f2b9

7 files changed

+146
-87
lines changed

lib/simplecov.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -172,7 +172,7 @@ def usable?
172172
require "simplecov/filter"
173173
require "simplecov/formatter"
174174
require "simplecov/last_run"
175-
require "simplecov/merge_helpers"
175+
require "simplecov/raw_coverage"
176176
require "simplecov/result_merger"
177177
require "simplecov/command_guesser"
178178
require "simplecov/version"

lib/simplecov/merge_helpers.rb

-37
This file was deleted.

lib/simplecov/raw_coverage.rb

+39
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
module SimpleCov
2+
module RawCoverage
3+
module_function
4+
5+
# Merges multiple Coverage.result hashes
6+
def merge_results(*results)
7+
results.reduce({}) do |result, merged|
8+
merge_resultsets(result, merged)
9+
end
10+
end
11+
12+
# Merges two Coverage.result hashes
13+
def merge_resultsets(result1, result2)
14+
(result1.keys | result2.keys).each_with_object({}) do |filename, merged|
15+
file1 = result1[filename]
16+
file2 = result2[filename]
17+
merged[filename] = merge_file_coverage(file1, file2)
18+
end
19+
end
20+
21+
def merge_file_coverage(file1, file2)
22+
return (file1 || file2).dup unless file1 && file2
23+
24+
file1.map.with_index do |count1, index|
25+
count2 = file2[index]
26+
merge_line_coverage(count1, count2)
27+
end
28+
end
29+
30+
def merge_line_coverage(count1, count2)
31+
sum = count1.to_i + count2.to_i
32+
if sum.zero? && (count1.nil? || count2.nil?)
33+
nil
34+
else
35+
sum
36+
end
37+
end
38+
end
39+
end

lib/simplecov/result.rb

-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,6 @@ class Result
2424
# Initialize a new SimpleCov::Result from given Coverage.result (a Hash of filenames each containing an array of
2525
# coverage data)
2626
def initialize(original_result)
27-
original_result = original_result.dup.extend(SimpleCov::HashMergeHelper) unless original_result.is_a? SimpleCov::HashMergeHelper
2827
@original_result = original_result.freeze
2928
@files = SimpleCov::FileList.new(original_result.map do |filename, coverage|
3029
SimpleCov::SourceFile.new(filename, coverage) if File.file?(filename)

lib/simplecov/result_merger.rb

+12-8
Original file line numberDiff line numberDiff line change
@@ -54,20 +54,24 @@ def results
5454
results
5555
end
5656

57+
# Merge two or more SimpleCov::Results into a new one with merged
58+
# coverage data and the command_name for the result consisting of a join
59+
# on all source result's names
60+
def merge_results(*results)
61+
merged = SimpleCov::RawCoverage.merge_results(*results.map(&:original_result))
62+
result = SimpleCov::Result.new(merged)
63+
# Specify the command name
64+
result.command_name = results.map(&:command_name).sort.join(", ")
65+
result
66+
end
67+
5768
#
5869
# Gets all SimpleCov::Results from cache, merges them and produces a new
5970
# SimpleCov::Result with merged coverage data and the command_name
6071
# for the result consisting of a join on all source result's names
6172
#
6273
def merged_result
63-
merged = {}
64-
results.each do |result|
65-
merged = result.original_result.merge_resultset(merged)
66-
end
67-
result = SimpleCov::Result.new(merged)
68-
# Specify the command name
69-
result.command_name = results.map(&:command_name).sort.join(", ")
70-
result
74+
merge_results(*results)
7175
end
7276

7377
# Saves the given SimpleCov::Result in the resultset cache

spec/raw_coverage_spec.rb

+92
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
require "helper"
2+
3+
if SimpleCov.usable?
4+
describe SimpleCov::RawCoverage do
5+
describe "with two faked coverage resultsets" do
6+
before do
7+
@resultset1 = {
8+
source_fixture("sample.rb") => [nil, 1, 1, 1, nil, nil, 1, 1, nil, nil],
9+
source_fixture("app/models/user.rb") => [nil, 1, 1, 1, nil, nil, 1, 0, nil, nil],
10+
source_fixture("app/controllers/sample_controller.rb") => [nil, 1, 1, 1, nil, nil, 1, 0, nil, nil],
11+
source_fixture("resultset1.rb") => [1, 1, 1, 1],
12+
source_fixture("parallel_tests.rb") => [nil, 0, nil, 0],
13+
source_fixture("conditionally_loaded_1.rb") => [nil, 0, 1], # loaded only in the first resultset
14+
source_fixture("three.rb") => [nil, 1, 1],
15+
}
16+
17+
@resultset2 = {
18+
source_fixture("sample.rb") => [1, nil, 1, 1, nil, nil, 1, 1, nil, nil],
19+
source_fixture("app/models/user.rb") => [nil, 1, 5, 1, nil, nil, 1, 0, nil, nil],
20+
source_fixture("app/controllers/sample_controller.rb") => [nil, 3, 1, nil, nil, nil, 1, 0, nil, nil],
21+
source_fixture("resultset2.rb") => [nil, 1, 1, nil],
22+
source_fixture("parallel_tests.rb") => [nil, nil, 0, 0],
23+
source_fixture("conditionally_loaded_2.rb") => [nil, 0, 1], # loaded only in the second resultset
24+
source_fixture("three.rb") => [nil, 1, 4],
25+
}
26+
27+
@resultset3 = {
28+
source_fixture("three.rb") => [nil, 1, 2],
29+
}
30+
end
31+
32+
context "a merge" do
33+
subject do
34+
SimpleCov::RawCoverage.merge_results(@resultset1, @resultset2, @resultset3)
35+
end
36+
37+
it "has proper results for sample.rb" do
38+
expect(subject[source_fixture("sample.rb")]).to eq([1, 1, 2, 2, nil, nil, 2, 2, nil, nil])
39+
end
40+
41+
it "has proper results for user.rb" do
42+
expect(subject[source_fixture("app/models/user.rb")]).to eq([nil, 2, 6, 2, nil, nil, 2, 0, nil, nil])
43+
end
44+
45+
it "has proper results for sample_controller.rb" do
46+
expect(subject[source_fixture("app/controllers/sample_controller.rb")]).to eq([nil, 4, 2, 1, nil, nil, 2, 0, nil, nil])
47+
end
48+
49+
it "has proper results for resultset1.rb" do
50+
expect(subject[source_fixture("resultset1.rb")]).to eq([1, 1, 1, 1])
51+
end
52+
53+
it "has proper results for resultset2.rb" do
54+
expect(subject[source_fixture("resultset2.rb")]).to eq([nil, 1, 1, nil])
55+
end
56+
57+
it "has proper results for parallel_tests.rb" do
58+
expect(subject[source_fixture("parallel_tests.rb")]).to eq([nil, nil, nil, 0])
59+
end
60+
61+
it "has proper results for conditionally_loaded_1.rb" do
62+
expect(subject[source_fixture("conditionally_loaded_1.rb")]).to eq([nil, 0, 1])
63+
end
64+
65+
it "has proper results for conditionally_loaded_2.rb" do
66+
expect(subject[source_fixture("conditionally_loaded_2.rb")]).to eq([nil, 0, 1])
67+
end
68+
69+
it "has proper results for three.rb" do
70+
expect(subject[source_fixture("three.rb")]).to eq([nil, 3, 7])
71+
end
72+
end
73+
end
74+
75+
it "merges frozen resultsets" do
76+
resultset1 = {
77+
source_fixture("sample.rb").freeze => [nil, 1, 1, 1, nil, nil, 1, 1, nil, nil].freeze,
78+
source_fixture("app/models/user.rb").freeze => [nil, 1, 1, 1, nil, nil, 1, 0, nil, nil].freeze,
79+
}.freeze
80+
81+
resultset2 = {
82+
source_fixture("sample.rb").freeze => [1, nil, 1, 1, nil, nil, 1, 1, nil, nil].freeze,
83+
}.freeze
84+
85+
merged_result = SimpleCov::RawCoverage.merge_results(resultset1, resultset2)
86+
expect(merged_result.keys).to eq(resultset1.keys)
87+
expect(merged_result.values.map(&:frozen?)).to eq([false, false])
88+
expect(merged_result[source_fixture("sample.rb")]).to eq([1, 1, 2, 2, nil, nil, 2, 2, nil, nil])
89+
expect(merged_result[source_fixture("app/models/user.rb")]).to eq([nil, 1, 1, 1, nil, nil, 1, 0, nil, nil])
90+
end
91+
end
92+
end

spec/merge_helpers_spec.rb spec/result_merger_spec.rb

+2-40
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
require "helper"
22

33
if SimpleCov.usable?
4-
describe "merge helpers" do
4+
describe SimpleCov::ResultMerger do
55
describe "with two faked coverage resultsets" do
66
before do
77
SimpleCov.use_merging true
@@ -12,7 +12,7 @@
1212
source_fixture("resultset1.rb") => [1, 1, 1, 1],
1313
source_fixture("parallel_tests.rb") => [nil, 0, nil, 0],
1414
source_fixture("conditionally_loaded_1.rb") => [nil, 0, 1], # loaded only in the first resultset
15-
}.extend(SimpleCov::HashMergeHelper)
15+
}
1616

1717
@resultset2 = {
1818
source_fixture("sample.rb") => [1, nil, 1, 1, nil, nil, 1, 1, nil, nil],
@@ -24,44 +24,6 @@
2424
}
2525
end
2626

27-
context "a merge" do
28-
subject do
29-
@resultset1.merge_resultset(@resultset2)
30-
end
31-
32-
it "has proper results for sample.rb" do
33-
expect(subject[source_fixture("sample.rb")]).to eq([1, 1, 2, 2, nil, nil, 2, 2, nil, nil])
34-
end
35-
36-
it "has proper results for user.rb" do
37-
expect(subject[source_fixture("app/models/user.rb")]).to eq([nil, 2, 6, 2, nil, nil, 2, 0, nil, nil])
38-
end
39-
40-
it "has proper results for sample_controller.rb" do
41-
expect(subject[source_fixture("app/controllers/sample_controller.rb")]).to eq([nil, 4, 2, 1, nil, nil, 2, 0, nil, nil])
42-
end
43-
44-
it "has proper results for resultset1.rb" do
45-
expect(subject[source_fixture("resultset1.rb")]).to eq([1, 1, 1, 1])
46-
end
47-
48-
it "has proper results for resultset2.rb" do
49-
expect(subject[source_fixture("resultset2.rb")]).to eq([nil, 1, 1, nil])
50-
end
51-
52-
it "has proper results for parallel_tests.rb" do
53-
expect(subject[source_fixture("parallel_tests.rb")]).to eq([nil, nil, nil, 0])
54-
end
55-
56-
it "has proper results for conditionally_loaded_1.rb" do
57-
expect(subject[source_fixture("conditionally_loaded_1.rb")]).to eq([nil, 0, 1])
58-
end
59-
60-
it "has proper results for conditionally_loaded_2.rb" do
61-
expect(subject[source_fixture("conditionally_loaded_2.rb")]).to eq([nil, 0, 1])
62-
end
63-
end
64-
6527
# See Github issue #6
6628
it "returns an empty hash when the resultset cache file is empty" do
6729
File.open(SimpleCov::ResultMerger.resultset_path, "w+") { |f| f.puts "" }

0 commit comments

Comments
 (0)