2012-12-06 15:30:28 -05:00
|
|
|
module Mutant
|
|
|
|
# Generator for mutations
|
|
|
|
class Mutator
|
|
|
|
# Abstract base class for node mutators
|
|
|
|
class Node < self
|
|
|
|
include AbstractType
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
alias_method :node, :input
|
|
|
|
alias_method :dup_node, :dup_input
|
|
|
|
|
|
|
|
# Return source of input node
|
|
|
|
#
|
|
|
|
# @return [String]
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
#
|
|
|
|
def source
|
|
|
|
ToSource.to_source(node)
|
|
|
|
end
|
|
|
|
memoize :source
|
|
|
|
|
|
|
|
# Test if generated node is new
|
|
|
|
#
|
|
|
|
# @return [true]
|
|
|
|
# if generated node is different from input
|
|
|
|
#
|
|
|
|
# @return [false]
|
|
|
|
# otherwise
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
#
|
|
|
|
def new?(node)
|
|
|
|
source != ToSource.to_source(node)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Emit a new AST node
|
|
|
|
#
|
|
|
|
# @param [Rubinis::AST::Node:Class] node_class
|
|
|
|
#
|
|
|
|
# @return [Rubinius::AST::Node]
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
#
|
|
|
|
def emit_node(node_class, *arguments)
|
|
|
|
emit(new(node_class, *arguments))
|
|
|
|
end
|
|
|
|
|
|
|
|
# Create a new AST node with same class as wrapped node
|
|
|
|
#
|
|
|
|
# @return [Rubinius::AST::Node]
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
#
|
|
|
|
def new_self(*arguments)
|
|
|
|
new(node.class, *arguments)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Create a new AST node with Rubnius::AST::NilLiteral class
|
|
|
|
#
|
|
|
|
# @return [Rubinius::AST::Node]
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
#
|
|
|
|
def new_nil
|
|
|
|
new(Rubinius::AST::NilLiteral)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Create a new AST node with the same line as wrapped node
|
|
|
|
#
|
|
|
|
# @param [Class:Rubinius::AST::Node] node_class
|
|
|
|
#
|
|
|
|
# @return [Rubinius::AST::Node]
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
#
|
|
|
|
def new(node_class, *arguments)
|
|
|
|
node_class.new(node.line, *arguments)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Emit a new AST node with same class as wrapped node
|
|
|
|
#
|
|
|
|
# @return [undefined]
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
#
|
|
|
|
def emit_self(*arguments)
|
|
|
|
emit(new_self(*arguments))
|
|
|
|
end
|
|
|
|
|
|
|
|
# Emit a new node with wrapping class for each entry in values
|
|
|
|
#
|
|
|
|
# @param [Array] values
|
|
|
|
#
|
|
|
|
# @return [undefined]
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
#
|
|
|
|
def emit_values(values)
|
|
|
|
values.each do |value|
|
|
|
|
emit_self(value)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Emit body mutations
|
|
|
|
#
|
2012-12-06 16:30:57 -05:00
|
|
|
# @param [Symbol] name
|
|
|
|
#
|
2012-12-06 15:30:28 -05:00
|
|
|
# @return [undefined]
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
#
|
2012-12-06 16:30:57 -05:00
|
|
|
def emit_attribute_mutations(name)
|
|
|
|
body = node.public_send(name)
|
2012-12-06 15:54:18 -05:00
|
|
|
|
|
|
|
Mutator.each(body) do |mutation|
|
|
|
|
dup = dup_node
|
2012-12-06 16:30:57 -05:00
|
|
|
dup.public_send(:"#{name}=", mutation)
|
2012-12-06 15:54:18 -05:00
|
|
|
emit(dup)
|
2012-12-06 15:30:28 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Emit a new AST node with NilLiteral class
|
|
|
|
#
|
|
|
|
# @return [Rubinius::AST::NilLiteral]
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
#
|
|
|
|
def emit_nil
|
|
|
|
emit(new_nil)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Return AST representing send
|
|
|
|
#
|
|
|
|
# @param [Rubinius::AST::Node] receiver
|
|
|
|
# @param [Symbol] name
|
|
|
|
# @param [Rubinius::AST::Node] arguments
|
|
|
|
#
|
|
|
|
# @return [Rubnius::AST::SendWithArguments]
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
#
|
|
|
|
def new_send(receiver, name, arguments=nil)
|
|
|
|
if arguments
|
|
|
|
new(Rubinius::AST::SendWithArguments, receiver, name, arguments)
|
|
|
|
else
|
|
|
|
new(Rubinius::AST::Send, receiver, name)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Return duplicated (unfrozen) node each call
|
|
|
|
#
|
|
|
|
# @return [Rubinius::AST::Node]
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
#
|
|
|
|
def dup_node
|
|
|
|
node.dup
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|