242 lines
5 KiB
Ruby
242 lines
5 KiB
Ruby
module Mutant
|
|
# Namespace and mixon module for results
|
|
module Result
|
|
|
|
# Coverage mixin
|
|
module Coverage
|
|
FULL_COVERAGE = Rational(1).freeze
|
|
private_constant(*constants(false))
|
|
|
|
# Observed coverage
|
|
#
|
|
# @return [Rational]
|
|
#
|
|
# @api private
|
|
def coverage
|
|
if amount_mutation_results.zero?
|
|
FULL_COVERAGE
|
|
else
|
|
Rational(amount_mutations_killed, amount_mutation_results)
|
|
end
|
|
end
|
|
end # Coverage
|
|
|
|
# Class level mixin
|
|
module ClassMethods
|
|
|
|
# Generate a sum method from name and collection
|
|
#
|
|
# @param [Symbol] name
|
|
# the attribute name on collection item and method name to use
|
|
#
|
|
# @param [Symbol] collection
|
|
# the attribute name used to receive collection
|
|
#
|
|
# @return [undefined]
|
|
#
|
|
# @api private
|
|
def sum(name, collection)
|
|
define_method(name) do
|
|
public_send(collection).map(&name).reduce(0, :+)
|
|
end
|
|
memoize(name)
|
|
end
|
|
end # ClassMethods
|
|
|
|
private_constant(*constants(false))
|
|
|
|
# Mutant overhead running mutatet tests
|
|
#
|
|
# This is NOT the overhead of mutation testing, just an engine specific
|
|
# measurement for the efficiency of the parellelization engine, kill
|
|
# isolation etc.
|
|
#
|
|
# @return [Float]
|
|
#
|
|
# @api private
|
|
def overhead
|
|
runtime - killtime
|
|
end
|
|
|
|
# Hook called when module gets included
|
|
#
|
|
# @param [Class, Module] host
|
|
#
|
|
# @return [undefined]
|
|
#
|
|
# @api private
|
|
def self.included(host)
|
|
host.class_eval do
|
|
include Adamantium
|
|
extend ClassMethods
|
|
end
|
|
end
|
|
|
|
# Env result object
|
|
class Env
|
|
include Coverage, Result, Anima.new(:runtime, :env, :subject_results)
|
|
|
|
# Test if run is successful
|
|
#
|
|
# @return [Boolean]
|
|
#
|
|
# @api private
|
|
def success?
|
|
coverage.eql?(env.config.expected_coverage)
|
|
end
|
|
memoize :success?
|
|
|
|
# Failed subject results
|
|
#
|
|
# @return [Array<Result::Subject>]
|
|
#
|
|
# @api private
|
|
def failed_subject_results
|
|
subject_results.reject(&:success?)
|
|
end
|
|
|
|
sum :amount_mutation_results, :subject_results
|
|
sum :amount_mutations_alive, :subject_results
|
|
sum :amount_mutations_killed, :subject_results
|
|
sum :killtime, :subject_results
|
|
|
|
# Amount of mutations
|
|
#
|
|
# @return [Fixnum]
|
|
#
|
|
# @api private
|
|
def amount_mutations
|
|
env.mutations.length
|
|
end
|
|
|
|
# Amount of subjects
|
|
#
|
|
# @return [Fixnum]
|
|
#
|
|
# @api private
|
|
def amount_subjects
|
|
env.subjects.length
|
|
end
|
|
|
|
end # Env
|
|
|
|
# Test result
|
|
class Test
|
|
include Result, Anima.new(
|
|
:tests,
|
|
:output,
|
|
:passed,
|
|
:runtime
|
|
)
|
|
end # Test
|
|
|
|
# Subject result
|
|
class Subject
|
|
include Coverage, Result, Anima.new(:subject, :tests, :mutation_results)
|
|
|
|
sum :killtime, :mutation_results
|
|
sum :runtime, :mutation_results
|
|
|
|
# Test if subject was processed successful
|
|
#
|
|
# @return [Boolean]
|
|
#
|
|
# @api private
|
|
def success?
|
|
alive_mutation_results.empty?
|
|
end
|
|
|
|
# Test if runner should continue on subject
|
|
#
|
|
# @return [Boolean]
|
|
#
|
|
# @api private
|
|
def continue?
|
|
mutation_results.all?(&:success?)
|
|
end
|
|
|
|
# Killed mutations
|
|
#
|
|
# @return [Array<Result::Mutation>]
|
|
#
|
|
# @api private
|
|
def alive_mutation_results
|
|
mutation_results.reject(&:success?)
|
|
end
|
|
memoize :alive_mutation_results
|
|
|
|
# Amount of mutations
|
|
#
|
|
# @return [Fixnum]
|
|
#
|
|
# @api private
|
|
def amount_mutation_results
|
|
mutation_results.length
|
|
end
|
|
|
|
# Amount of mutations
|
|
#
|
|
# @return [Fixnum]
|
|
#
|
|
# @api private
|
|
def amount_mutations
|
|
subject.mutations.length
|
|
end
|
|
|
|
# Number of killed mutations
|
|
#
|
|
# @return [Fixnum]
|
|
#
|
|
# @api private
|
|
def amount_mutations_killed
|
|
killed_mutation_results.length
|
|
end
|
|
|
|
# Number of alive mutations
|
|
#
|
|
# @return [Fixnum]
|
|
#
|
|
# @api private
|
|
def amount_mutations_alive
|
|
alive_mutation_results.length
|
|
end
|
|
|
|
# Alive mutations
|
|
#
|
|
# @return [Array<Result::Mutation>]
|
|
#
|
|
# @api private
|
|
def killed_mutation_results
|
|
mutation_results.select(&:success?)
|
|
end
|
|
memoize :killed_mutation_results
|
|
|
|
end # Subject
|
|
|
|
# Mutation result
|
|
class Mutation
|
|
include Result, Anima.new(:mutation, :test_result)
|
|
|
|
# The runtime
|
|
#
|
|
# @return [Float]
|
|
#
|
|
# @api private
|
|
def runtime
|
|
test_result.runtime
|
|
end
|
|
|
|
alias_method :killtime, :runtime
|
|
|
|
# Test if mutation was handled successfully
|
|
#
|
|
# @return [Boolean]
|
|
#
|
|
# @api private
|
|
def success?
|
|
mutation.class.success?(test_result)
|
|
end
|
|
|
|
end # Mutation
|
|
end # Result
|
|
end # Mutant
|