require 'lucid/step_match'
require 'lucid/core_ext/string'
require 'lucid/core_ext/proc'
require 'lucid/interface_rb/regexp_argument_matcher'
module Lucid
module InterfaceRb
# A Ruby test definition holds a Regexp and a Proc, and is created
# by calling Given, When or Then.
#
# Example:
#
# Given /there are (\d+) lucid tests to run/ do
# # some code here
# end
#
class RbStepDefinition
class MissingProc < StandardError
def message
'Test definitions must always have a proc or symbol'
end
end
class << self
def new(rb_language, pattern, proc_or_sym, options)
raise MissingProc if proc_or_sym.nil?
super rb_language, parse_pattern(pattern), create_proc(proc_or_sym, options)
end
private
def parse_pattern(pattern)
return pattern if pattern.is_a?(Regexp)
raise ArgumentError unless pattern.is_a?(String)
p = Regexp.escape(pattern)
p = p.gsub(/\\\$\w+/, '(.*)') # Replace $var with (.*)
Regexp.new("^#{p}$")
end
def create_proc(proc_or_sym, options)
return proc_or_sym if proc_or_sym.is_a?(Proc)
raise ArgumentError unless proc_or_sym.is_a?(Symbol)
message = proc_or_sym
target_proc = parse_target_proc_from(options)
lambda do |*args|
target = instance_exec(&target_proc)
target.send(message, *args)
end
end
def parse_target_proc_from(options)
return lambda { self } unless options.key?(:on)
target = options[:on]
case target
when Proc
target
when Symbol
lambda { self.send(target) }
else
lambda { raise ArgumentError, 'Target must be a symbol or a proc' }
end
end
end
def initialize(rb_language, regexp, proc)
@rb_language, @regexp, @proc = rb_language, regexp, proc
@rb_language.available_step_definition(regexp_source, file_colon_line)
end
def regexp_source
@regexp.inspect
end
def to_hash
flags = ''
flags += 'm' if (@regexp.options & Regexp::MULTILINE) != 0
flags += 'i' if (@regexp.options & Regexp::IGNORECASE) != 0
flags += 'x' if (@regexp.options & Regexp::EXTENDED) != 0
{'source' => @regexp.source, 'flags' => flags}
end
def ==(step_definition)
regexp_source == step_definition.regexp_source
end
def arguments_from(step_name)
args = RegexpArgumentMatcher.arguments_from(@regexp, step_name)
@rb_language.invoked_step_definition(regexp_source, file_colon_line) if args
args
end
def invoke(args)
begin
args = @rb_language.execute_transforms(args)
@rb_language.current_domain.lucid_instance_exec(true, regexp_source, *args, &@proc)
rescue Lucid::ArityMismatchError => e
e.backtrace.unshift(self.backtrace_line)
raise e
end
end
def backtrace_line
@proc.backtrace_line(regexp_source)
end
def file_colon_line
case @proc
when Proc
@proc.file_colon_line
when Symbol
":#{@proc}"
end
end
def file
@file ||= file_colon_line.split(':')[0]
end
end
end
end