free_mutant/lib/mutant/mutation.rb
Markus Schirp 41d9700473 Refactor runer infrastructure
* Nuke around 800 lock
* Honor LSP with not anymore squeezing something non LSP compatible in
  the same inheritance tree.
* Separate running from result tree.
* Clean up kill logic and early exits on already dead mutations.
* Fix #runnin? smell for reporters.
* Decouple config object from VM state. Makes it serializable to enable
  config loading.
* Fix sequence of global VM events to match PRIOR rspec infects VM with
  gazillions of classes / modules. Thix fixes a startup speed
  degeneration.
* Various fixes to enhance determinism.
* Replace some unneded manual double dispatch with single manual dispatch for
  reporter / runners.
2014-07-05 23:04:38 +00:00

130 lines
2.4 KiB
Ruby

module Mutant
# Represent a mutated node with its subject
class Mutation
include AbstractType, Adamantium::Flat
include Concord::Public.new(:subject, :node)
CODE_DELIMITER = "\0".freeze
CODE_RANGE = (0..4).freeze
# Return mutated root node
#
# @return [Parser::AST::Node]
#
# @api private
#
def root
subject.root(node)
end
memoize :root
# Insert mutated node
#
# FIXME: Cache subject visibility in a better way! Ideally dont mutate it
# implicitly. Also subject.public? should NOT be a public interface it
# is a detail of method mutations.
#
# @return [self]
#
# @api private
#
def insert
subject.public?
Loader::Eval.call(root, subject)
self
end
# Return identification
#
# @return [String]
#
# @api private
#
def identification
"#{self.class::SYMBOL}:#{subject.identification}:#{code}"
end
memoize :identification
# Return mutation code
#
# @return [String]
#
# @api private
#
def code
sha1[CODE_RANGE]
end
memoize :code
# Return source
#
# @return [String]
#
# @api private
#
def source
Unparser.unparse(node)
end
memoize :source
# Return original source
#
# @return [String]
#
# @api private
#
def original_source
subject.source
end
# Test if mutation is killed by test report
#
# @param [Report::Test] test_report
#
# @return [Boolean]
#
# @api private
#
def killed_by?(test_report)
self.class::SHOULD_PASS.equal?(test_report.passed)
end
private
# Return sha1 sum of source and subject identification
#
# @return [String]
#
# @api private
#
def sha1
Digest::SHA1.hexdigest(subject.identification + CODE_DELIMITER + source)
end
memoize :sha1
# Evil mutation that should case mutations to fail tests
class Evil < self
SHOULD_PASS = false
SYMBOL = 'evil'.freeze
end # Evil
# Neutral mutation that should not cause mutations to fail tests
class Neutral < self
SYMBOL = 'neutral'.freeze
SHOULD_PASS = true
end # Neutral
# Noop mutation, special case of neutral
class Noop < self
SYMBOL = 'noop'.freeze
SHOULD_PASS = true
end # Noop
end # Mutation
end # Mutant