free_mutant/lib/mutant/matcher/method.rb

177 lines
3.4 KiB
Ruby
Raw Normal View History

module Mutant
class Matcher
# Matcher for subjects that are a specific method
2012-08-09 17:07:22 -04:00
class Method < self
2013-05-21 20:40:33 -04:00
include Adamantium::Flat, Concord::Public.new(:scope, :method)
2013-04-17 23:31:21 -04:00
# Methods within rbx kernel directory are precompiled and their source
# cannot be accessed via reading source location
BLACKLIST = /\Akernel\//.freeze
# Enumerate matches
#
# @return [Enumerable]
# returns enumerable when no block given
#
# @return [self]
# returns self when block given
#
# @api private
2013-04-17 23:31:21 -04:00
#
def each(&block)
return to_enum unless block_given?
2012-12-07 17:27:21 -05:00
return self if skip?
2012-12-07 17:27:21 -05:00
util = subject
yield util if util
self
end
private
# Test if method is skipped
#
# @return [true]
# true and print warning if location must be filtered
#
# @return [false]
# otherwise
#
# @api private
#
def skip?
location = source_location
if location.nil? or BLACKLIST.match(location.first)
$stderr.puts "#{method.inspect} does not have valid source location so mutant is unable to emit matcher"
return true
end
false
end
# Return method name
#
# @return [String]
#
# @api private
#
def method_name
method.name
end
2012-12-07 17:27:21 -05:00
# Return context
#
2012-12-07 17:27:21 -05:00
# @return [Context::Scope]
#
# @api private
#
2012-12-07 17:27:21 -05:00
def context
Context::Scope.new(scope, source_path)
2012-12-07 17:27:21 -05:00
end
# Return full ast
#
# @return [Parser::AST::Node]
#
# @api private
#
def ast
Parser::CurrentRuby.parse(File.read(source_path))
end
2012-08-14 06:26:56 -04:00
# Return path to source
#
# @return [String]
#
# @api private
#
2012-08-14 06:26:56 -04:00
def source_path
source_location.first
end
# Return source file line
#
# @return [Integer]
#
# @api private
#
2012-08-14 16:45:34 -04:00
def source_line
source_location.last
end
# Return source location
#
# @return [Array]
#
# @api private
#
def source_location
method.source_location
end
2012-08-01 12:34:03 -04:00
# Return subject
#
2012-08-01 12:34:03 -04:00
# @return [Subject]
# returns subject if there is a matched node
#
# @return [nil]
# otherwise
#
# @api private
#
2012-08-01 12:34:03 -04:00
def subject
node = matched_node
return unless node
self.class::SUBJECT_CLASS.new(context, node)
end
2012-08-01 12:34:03 -04:00
memoize :subject
class Finder
def self.run(root, &predicate)
new(root, predicate).match
end
private_class_method :new
attr_reader :match
private
def initialize(root, predicate)
@root, @predicate = root, predicate
test(root)
end
def test(node)
if @predicate.call(node)
@match = node
end
node.children.each do |child|
test(child) if child.kind_of?(Parser::AST::Node)
end
end
end
# Return matched node
#
# @return [Parser::AST::Node]
# if node could be found
#
# @return [nil]
# otherwise
#
# @api private
#
def matched_node
Finder.run(ast) do |node|
match?(node)
end
end
end # Method
end # Matcher
end # Mutant