free_mutant/lib/mutant/mutator/node/send.rb

229 lines
5.1 KiB
Ruby
Raw Normal View History

module Mutant
class Mutator
class Node
# Namespace for send mutators
class Send < self
handle(:send)
children :receiver, :selector
2014-03-30 14:56:51 +00:00
SELECTOR_REPLACEMENTS = IceNine.deep_freeze(
reverse_map: [:map, :each],
reverse_each: [:each],
map: [:each],
send: [:public_send],
gsub: [:sub],
eql?: [:equal?],
:== => [:eql?, :equal?]
2014-03-30 14:56:51 +00:00
)
INDEX_REFERENCE = :[]
INDEX_ASSIGN = :[]=
VARIABLE_ASSIGN = :'='
ASSIGNMENT_OPERATORS = [INDEX_ASSIGN, VARIABLE_ASSIGN].to_set.freeze
ATTRIBUTE_ASSIGNMENT = /\A[a-z\d_]+=\z/.freeze
private
# Perform dispatch
#
# @return [undefined]
#
# @api private
#
def dispatch
emit_singletons
case selector
when INDEX_REFERENCE
run(Index::Reference)
when INDEX_ASSIGN
run(Index::Assign)
else
non_index_dispatch
end
end
# Perform non index dispatch
#
# @return [undefined]
#
# @api private
#
def non_index_dispatch
case
when binary_operator?
run(Binary)
when attribute_assignment?
run(AttributeAssignment)
else
normal_dispatch
end
end
2013-06-21 15:58:35 +02:00
# Return arguments
#
# @return [Enumerable<Parser::AST::Node>]
#
# @api private
#
alias_method :arguments, :remaining_children
# Perform normal, non special case dispatch
#
# @return [undefined]
#
# @api private
#
def normal_dispatch
emit_naked_receiver
2013-09-02 21:29:44 +02:00
emit_selector_replacement
mutate_receiver
2013-06-15 17:16:34 +02:00
emit_argument_propagation
mutate_arguments
end
2013-09-02 21:29:44 +02:00
# Emit selector replacement
#
# @return [undefined]
#
# @api private
#
2013-09-02 21:29:44 +02:00
def emit_selector_replacement
2014-03-30 14:56:51 +00:00
SELECTOR_REPLACEMENTS.fetch(selector, EMPTY_ARRAY).each do |replacement|
emit_selector(replacement)
end
end
# Emit naked receiver mutation
#
# @return [undefined]
#
# @api private
#
def emit_naked_receiver
emit(receiver) if receiver && !(OP_ASSIGN.include?(parent_type) && NOT_ASSIGNABLE.include?(receiver.type))
end
# Test for binary operator
#
# @return [true]
# if send is a binary operator
#
# @return [false]
# otherwise
#
# @api private
#
def binary_operator?
arguments.one? && BINARY_METHOD_OPERATORS.include?(selector)
end
# Test for attribute assignment
#
# @return [Boolean]
#
# @api private
#
def attribute_assignment?
arguments.one? && ATTRIBUTE_ASSIGNMENT =~ selector
end
# Mutate arguments
#
# @return [undefined]
#
# @api private
#
def mutate_arguments
2014-06-03 03:51:22 +00:00
emit_type(receiver, selector)
2014-05-27 15:12:36 +00:00
remaining_children_with_index.each do |_node, index|
mutate_child(index)
delete_child(index)
end
end
2013-06-15 17:16:34 +02:00
# Emit argument propagation
#
2013-06-17 11:47:27 +02:00
# @return [undefined]
2013-06-15 17:16:34 +02:00
#
# @api private
#
def emit_argument_propagation
return unless arguments.one?
node = arguments.first
2014-06-15 12:16:55 +00:00
emit(node) unless NOT_STANDALONE.include?(node.type)
2013-06-15 17:16:34 +02:00
end
# Emit receiver mutations
#
# @return [undefined]
#
# @api private
#
def mutate_receiver
return unless receiver
emit_implicit_self
emit_receiver_mutations
end
# Emit implicit self mutation
#
# @return [undefined]
#
2013-06-15 16:37:43 +02:00
# @api private
#
def emit_implicit_self
2014-06-08 13:01:26 +00:00
emit_receiver(nil) if allow_implicit_self?
end
# Test if implicit self is allowed
#
# @return [Boolean]
#
# @api private
#
def allow_implicit_self?
2014-06-15 12:43:08 +00:00
n_self?(receiver) && !(
KEYWORDS.include?(selector) ||
OP_ASSIGN.include?(parent_type) ||
attribute_assignment?
)
end
# Test for assignment
#
2014-06-15 12:20:00 +00:00
# @return [Boolean]
#
# @api private
#
def assignment?
arguments.one? && (ASSIGNMENT_OPERATORS.include?(selector) || attribute_assignment?)
end
# Test if node is part of an mlhs
#
2014-06-15 12:20:00 +00:00
# @return [Boolean]
#
# @api private
#
def mlhs?
assignment? && !arguments?
end
# Test for empty arguments
#
2014-06-15 12:20:00 +00:00
# @return [Boolean]
#
# @api private
#
def arguments?
arguments.any?
end
end # Send
end # Node
end # Mutator
end # Mutant