alphalang 0.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/bin/alphalang +59 -0
- data/lib/alpha.rb +168 -0
- data/lib/lang_creator.rb +124 -0
- data/lib/locales/de +17 -0
- data/lib/locales/en +17 -0
- data/lib/locales/locale_template +17 -0
- data/lib/locales/sv +17 -0
- data/lib/nodes/#stmtnodes.rb# +196 -0
- data/lib/nodes/basenodes.rb +195 -0
- data/lib/nodes/stmtnodes.rb +200 -0
- data/lib/rdparse.rb +238 -0
- data/lib/rdparse_quiet.rb +237 -0
- data/lib/tester/fibonacci.alpha +19 -0
- data/lib/tester/test_unit.rb +217 -0
- metadata +98 -0
@@ -0,0 +1,195 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
####################################################
|
3
|
+
|
4
|
+
$scopes = [{}]
|
5
|
+
$scope_lvl = 0
|
6
|
+
$test_nodes = false
|
7
|
+
|
8
|
+
####################################################
|
9
|
+
|
10
|
+
class TrueClass
|
11
|
+
def evaluate
|
12
|
+
self
|
13
|
+
end
|
14
|
+
end
|
15
|
+
|
16
|
+
class FalseClass
|
17
|
+
def evaluate
|
18
|
+
self
|
19
|
+
end
|
20
|
+
end
|
21
|
+
|
22
|
+
class String
|
23
|
+
def name
|
24
|
+
self
|
25
|
+
end
|
26
|
+
|
27
|
+
def evaluate
|
28
|
+
self
|
29
|
+
end
|
30
|
+
end
|
31
|
+
|
32
|
+
####################################################
|
33
|
+
|
34
|
+
class Node
|
35
|
+
attr_accessor :value
|
36
|
+
|
37
|
+
def initialize(value)
|
38
|
+
@value = value
|
39
|
+
end
|
40
|
+
|
41
|
+
def evaluate
|
42
|
+
@value.evaluate
|
43
|
+
end
|
44
|
+
end
|
45
|
+
|
46
|
+
class NumberNode < Node
|
47
|
+
def initialize(value)
|
48
|
+
super(value)
|
49
|
+
end
|
50
|
+
|
51
|
+
def evaluate
|
52
|
+
if @value.include?('.')
|
53
|
+
@value.to_f
|
54
|
+
else
|
55
|
+
@value.to_i
|
56
|
+
end
|
57
|
+
end
|
58
|
+
end
|
59
|
+
|
60
|
+
class BoolNode < Node
|
61
|
+
def initialize(value)
|
62
|
+
@value = false
|
63
|
+
@value = true if value == "true"
|
64
|
+
@value = false if value == "false"
|
65
|
+
end
|
66
|
+
|
67
|
+
def evaluate
|
68
|
+
@value
|
69
|
+
end
|
70
|
+
end
|
71
|
+
|
72
|
+
class AndNode < Node
|
73
|
+
def initialize(node1, node2)
|
74
|
+
@node1, @node2 = node1, node2
|
75
|
+
if @node2.class.method_defined? (:lhs)
|
76
|
+
if @node2.lhs == nil
|
77
|
+
@node2.lhs = @node1.lhs
|
78
|
+
end
|
79
|
+
end
|
80
|
+
end
|
81
|
+
|
82
|
+
def evaluate
|
83
|
+
@node1.evaluate && @node2.evaluate
|
84
|
+
end
|
85
|
+
end
|
86
|
+
|
87
|
+
class OrNode < Node
|
88
|
+
def initialize(node1, node2)
|
89
|
+
@node1, @node2 = node1, node2
|
90
|
+
if @node2.class.method_defined? (:lhs)
|
91
|
+
if @node2.lhs == nil
|
92
|
+
@node2.lhs = @node1.lhs
|
93
|
+
end
|
94
|
+
end
|
95
|
+
end
|
96
|
+
|
97
|
+
def evaluate
|
98
|
+
@node1.evaluate || @node2.evaluate
|
99
|
+
end
|
100
|
+
end
|
101
|
+
|
102
|
+
class NotNode < Node
|
103
|
+
def initialize(node)
|
104
|
+
@node = node
|
105
|
+
end
|
106
|
+
|
107
|
+
def evaluate
|
108
|
+
not @node.evaluate
|
109
|
+
end
|
110
|
+
end
|
111
|
+
|
112
|
+
####################################################
|
113
|
+
|
114
|
+
class CompareNode < Node
|
115
|
+
attr_accessor :lhs, :op, :rhs
|
116
|
+
|
117
|
+
def initialize(node1, op, node2)
|
118
|
+
@lhs, @op, @rhs = node1, op, node2
|
119
|
+
end
|
120
|
+
|
121
|
+
def evaluate
|
122
|
+
@value = @lhs.evaluate.send(@op, @rhs.evaluate)
|
123
|
+
end
|
124
|
+
end
|
125
|
+
|
126
|
+
class BinaryOperationNode < Node
|
127
|
+
attr_accessor :lhs, :op, :rhs
|
128
|
+
|
129
|
+
def initialize(lhs, op, rhs)
|
130
|
+
super(op)
|
131
|
+
@lhs, @op, @rhs = lhs, op, rhs
|
132
|
+
end
|
133
|
+
|
134
|
+
def evaluate
|
135
|
+
if @op == '/'
|
136
|
+
@value = @lhs.evaluate.to_f.send(@op, @rhs.evaluate)
|
137
|
+
else
|
138
|
+
@value = @lhs.evaluate.send(@op, @rhs.evaluate)
|
139
|
+
end
|
140
|
+
end
|
141
|
+
end
|
142
|
+
|
143
|
+
####################################################
|
144
|
+
|
145
|
+
class CompStmtNode < Node
|
146
|
+
|
147
|
+
def initialize(stmt_compstmt)
|
148
|
+
super
|
149
|
+
@comp_statements = stmt_compstmt
|
150
|
+
end
|
151
|
+
|
152
|
+
def evaluate
|
153
|
+
@comp_statements[0].evaluate
|
154
|
+
@comp_statements[1].evaluate # kanske bör loopas osv??? Root blir nu sista evalueringen.
|
155
|
+
end
|
156
|
+
end
|
157
|
+
|
158
|
+
|
159
|
+
####################################################
|
160
|
+
|
161
|
+
class ArgListNode < Node
|
162
|
+
attr_accessor :lhs, :rhs
|
163
|
+
|
164
|
+
def initialize(lhs, rhs)
|
165
|
+
@lhs, @rhs = lhs, rhs
|
166
|
+
|
167
|
+
if @rhs == NilClass
|
168
|
+
@value = [@lhs]
|
169
|
+
else
|
170
|
+
result = [@lhs]
|
171
|
+
@rhs.each do |element|
|
172
|
+
result << element
|
173
|
+
end
|
174
|
+
@value = result
|
175
|
+
end
|
176
|
+
end
|
177
|
+
|
178
|
+
def [](index)
|
179
|
+
@value[index]
|
180
|
+
end
|
181
|
+
|
182
|
+
def []=(index, value)
|
183
|
+
@value[index] = value
|
184
|
+
end
|
185
|
+
|
186
|
+
def each
|
187
|
+
@value.each_with_index do |val, index|
|
188
|
+
yield val, index
|
189
|
+
end
|
190
|
+
end
|
191
|
+
|
192
|
+
def evaluate
|
193
|
+
@value
|
194
|
+
end
|
195
|
+
end
|
@@ -0,0 +1,200 @@
|
|
1
|
+
require_relative 'basenodes'
|
2
|
+
|
3
|
+
####################################################
|
4
|
+
class VariableCallNode < Node
|
5
|
+
attr_accessor :name
|
6
|
+
|
7
|
+
def initialize(name)
|
8
|
+
@name = name
|
9
|
+
end
|
10
|
+
|
11
|
+
def lookup_var(name)
|
12
|
+
temp_scope_lvl = $scope_lvl
|
13
|
+
while temp_scope_lvl >= 0
|
14
|
+
if not $scopes[temp_scope_lvl].has_key?(name)
|
15
|
+
temp_scope_lvl -= 1
|
16
|
+
else
|
17
|
+
return $scopes[temp_scope_lvl][name]
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
21
|
+
|
22
|
+
def evaluate
|
23
|
+
return @value = lookup_var(@name)
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
class VariableDecNode < Node
|
28
|
+
attr_accessor :name
|
29
|
+
|
30
|
+
def initialize(name, value)
|
31
|
+
super(value)
|
32
|
+
@name = name
|
33
|
+
end
|
34
|
+
|
35
|
+
def evaluate
|
36
|
+
$scopes[$scope_lvl][@name.name] = @value.evaluate
|
37
|
+
self
|
38
|
+
# return nil
|
39
|
+
end
|
40
|
+
end
|
41
|
+
|
42
|
+
####################################################
|
43
|
+
|
44
|
+
class FunctionDecNode < Node
|
45
|
+
def initialize(node, value)
|
46
|
+
super(value)
|
47
|
+
@name = node
|
48
|
+
@args = node.args
|
49
|
+
end
|
50
|
+
|
51
|
+
def evaluate
|
52
|
+
$scopes[0][@name.name] = [@value, @args]
|
53
|
+
return nil
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
57
|
+
class FuncCallNode < Node
|
58
|
+
attr_accessor :name, :args
|
59
|
+
|
60
|
+
def initialize(name, args)
|
61
|
+
@name = name
|
62
|
+
@args = args
|
63
|
+
end
|
64
|
+
|
65
|
+
def lookup_var(name)
|
66
|
+
return $scopes[0][name]
|
67
|
+
end
|
68
|
+
|
69
|
+
def evaluate
|
70
|
+
$scopes.push({})
|
71
|
+
func = lookup_var(@name)
|
72
|
+
function_body = func[0]
|
73
|
+
function_param = func[1]
|
74
|
+
|
75
|
+
return nil if func.is_a?(NilClass)
|
76
|
+
|
77
|
+
$scope_lvl += 1
|
78
|
+
|
79
|
+
if function_param.is_a?(ArgListNode)
|
80
|
+
function_param.each do |val, index|
|
81
|
+
function_param[index] = VariableDecNode.new(function_param[index].name, @args[index])
|
82
|
+
function_param[index].evaluate
|
83
|
+
end
|
84
|
+
end
|
85
|
+
|
86
|
+
func_result = function_body.evaluate
|
87
|
+
old_scope_lvl = $scope_lvl
|
88
|
+
# Om en assign är returnvärde så declareras variablen i globalt scope
|
89
|
+
if func_result.is_a?(VariableDecNode)
|
90
|
+
$scope_lvl = 0
|
91
|
+
func_result.evaluate
|
92
|
+
$scope_lvl = old_scope_lvl
|
93
|
+
end
|
94
|
+
|
95
|
+
$scope_lvl -= 1
|
96
|
+
$scopes.pop
|
97
|
+
return func_result
|
98
|
+
end
|
99
|
+
end
|
100
|
+
|
101
|
+
####################################################
|
102
|
+
|
103
|
+
class IfNode < Node
|
104
|
+
attr_accessor :argument
|
105
|
+
|
106
|
+
def initialize(argument, node)
|
107
|
+
@argument, @node = argument, node
|
108
|
+
end
|
109
|
+
|
110
|
+
def evaluate
|
111
|
+
if @argument.evaluate
|
112
|
+
@value = @node.evaluate
|
113
|
+
else
|
114
|
+
@value = nil
|
115
|
+
end
|
116
|
+
@value
|
117
|
+
end
|
118
|
+
end
|
119
|
+
|
120
|
+
class ElseifNode < Node
|
121
|
+
attr_accessor :argument
|
122
|
+
def initialize(argument, node)
|
123
|
+
@argument, @node = argument, node
|
124
|
+
end
|
125
|
+
def evaluate
|
126
|
+
if @argument.evaluate
|
127
|
+
@value = @node.evaluate
|
128
|
+
end
|
129
|
+
end
|
130
|
+
end
|
131
|
+
|
132
|
+
class ElseNode < Node
|
133
|
+
attr_accessor :argument, :node
|
134
|
+
def initialize(node)
|
135
|
+
@node = node
|
136
|
+
@argument = true
|
137
|
+
end
|
138
|
+
|
139
|
+
def evaluate
|
140
|
+
@value = @node.evaluate
|
141
|
+
end
|
142
|
+
end
|
143
|
+
|
144
|
+
class IfCompStmtNode < Node
|
145
|
+
def initialize(*nodes)
|
146
|
+
@nodes = nodes.flatten
|
147
|
+
end
|
148
|
+
|
149
|
+
def evaluate
|
150
|
+
@nodes.each do |node|
|
151
|
+
if node.argument.evaluate
|
152
|
+
return node.evaluate
|
153
|
+
end
|
154
|
+
end
|
155
|
+
return nil
|
156
|
+
end
|
157
|
+
end
|
158
|
+
|
159
|
+
####################################################
|
160
|
+
|
161
|
+
class WhileLoopNode < Node
|
162
|
+
attr_accessor :condition
|
163
|
+
|
164
|
+
def initialize(condition, statement)
|
165
|
+
@condition = condition
|
166
|
+
super(statement)
|
167
|
+
end
|
168
|
+
|
169
|
+
def evaluate
|
170
|
+
while @condition.evaluate
|
171
|
+
@value.evaluate
|
172
|
+
end
|
173
|
+
self.class
|
174
|
+
end
|
175
|
+
end
|
176
|
+
|
177
|
+
####################################################
|
178
|
+
|
179
|
+
class PrintNode
|
180
|
+
attr_accessor :value
|
181
|
+
|
182
|
+
def initialize(value)
|
183
|
+
@value = value
|
184
|
+
end
|
185
|
+
|
186
|
+
def evaluate
|
187
|
+
puts @value.evaluate
|
188
|
+
self.class # detta kanske är trevligare än nil, åter igen den diskussionen.
|
189
|
+
end
|
190
|
+
end
|
191
|
+
|
192
|
+
class PauseNode < Node
|
193
|
+
def initialize(value)
|
194
|
+
super(value)
|
195
|
+
end
|
196
|
+
|
197
|
+
def evaluate
|
198
|
+
sleep @value.evaluate
|
199
|
+
end
|
200
|
+
end
|
data/lib/rdparse.rb
ADDED
@@ -0,0 +1,238 @@
|
|
1
|
+
# This file is called rdparse.rb because it implements a Recursive
|
2
|
+
# Descent Parser. Read more about the theory on e.g.
|
3
|
+
# http://en.wikipedia.org/wiki/Recursive_descent_parser
|
4
|
+
|
5
|
+
# 2010-02-11 New version of this file for the 2010 instance of TDP007
|
6
|
+
# which handles false return values during parsing, and has an easy way
|
7
|
+
# of turning on and off debug messages.
|
8
|
+
# 2014-02-16 New version that handles { false } blocks and :empty tokens.
|
9
|
+
|
10
|
+
require 'logger'
|
11
|
+
|
12
|
+
class Rule
|
13
|
+
|
14
|
+
# A rule is created through the rule method of the Parser class, like this:
|
15
|
+
# rule :term do
|
16
|
+
# match(:term, '*', :dice) {|a, _, b| a * b }
|
17
|
+
# match(:term, '/', :dice) {|a, _, b| a / b }
|
18
|
+
# match(:dice)
|
19
|
+
# end
|
20
|
+
|
21
|
+
Match = Struct.new :pattern, :block
|
22
|
+
|
23
|
+
def initialize(name, parser)
|
24
|
+
@logger = parser.logger
|
25
|
+
# The name of the expressions this rule matches
|
26
|
+
@name = name
|
27
|
+
# We need the parser to recursively parse sub-expressions occurring
|
28
|
+
# within the pattern of the match objects associated with this rule
|
29
|
+
@parser = parser
|
30
|
+
@matches = []
|
31
|
+
# Left-recursive matches
|
32
|
+
@lrmatches = []
|
33
|
+
end
|
34
|
+
|
35
|
+
# Add a matching expression to this rule, as in this example:
|
36
|
+
# match(:term, '*', :dice) {|a, _, b| a * b }
|
37
|
+
# The arguments to 'match' describe the constituents of this expression.
|
38
|
+
def match(*pattern, &block)
|
39
|
+
match = Match.new(pattern, block)
|
40
|
+
# If the pattern is left-recursive, then add it to the left-recursive set
|
41
|
+
if pattern[0] == @name
|
42
|
+
pattern.shift
|
43
|
+
@lrmatches << match
|
44
|
+
else
|
45
|
+
@matches << match
|
46
|
+
end
|
47
|
+
end
|
48
|
+
|
49
|
+
def parse
|
50
|
+
# Try non-left-recursive matches first, to avoid infinite recursion
|
51
|
+
match_result = try_matches(@matches)
|
52
|
+
return nil if match_result.nil?
|
53
|
+
loop do
|
54
|
+
result = try_matches(@lrmatches, match_result)
|
55
|
+
return match_result if result.nil?
|
56
|
+
match_result = result
|
57
|
+
end
|
58
|
+
end
|
59
|
+
|
60
|
+
private
|
61
|
+
|
62
|
+
# Try out all matching patterns of this rule
|
63
|
+
def try_matches(matches, pre_result = nil)
|
64
|
+
match_result = nil
|
65
|
+
# Begin at the current position in the input string of the parser
|
66
|
+
start = @parser.pos
|
67
|
+
matches.each do |match|
|
68
|
+
# pre_result is a previously available result from evaluating expressions
|
69
|
+
result = pre_result.nil? ? [] : [pre_result]
|
70
|
+
|
71
|
+
# We iterate through the parts of the pattern, which may be e.g.
|
72
|
+
# [:expr,'*',:term]
|
73
|
+
match.pattern.each_with_index do |token,index|
|
74
|
+
|
75
|
+
# If this "token" is a compound term, add the result of
|
76
|
+
# parsing it to the "result" array
|
77
|
+
if @parser.rules[token]
|
78
|
+
result << @parser.rules[token].parse
|
79
|
+
if result.last.nil?
|
80
|
+
result = nil
|
81
|
+
break
|
82
|
+
end
|
83
|
+
@logger.debug("Matched '#{@name} = #{match.pattern[index..-1].inspect}'")
|
84
|
+
else
|
85
|
+
# Otherwise, we consume the token as part of applying this rule
|
86
|
+
nt = @parser.expect(token)
|
87
|
+
if nt
|
88
|
+
result << nt
|
89
|
+
if @lrmatches.include?(match.pattern) then
|
90
|
+
pattern = [@name]+match.pattern
|
91
|
+
else
|
92
|
+
pattern = match.pattern
|
93
|
+
end
|
94
|
+
@logger.debug("Matched token '#{nt}' as part of rule '#{@name} <= #{pattern.inspect}'")
|
95
|
+
else
|
96
|
+
result = nil
|
97
|
+
break
|
98
|
+
end
|
99
|
+
end # pattern.each
|
100
|
+
end # matches.each
|
101
|
+
if result
|
102
|
+
if match.block
|
103
|
+
match_result = match.block.call(*result)
|
104
|
+
else
|
105
|
+
match_result = result[0]
|
106
|
+
end
|
107
|
+
@logger.debug("'#{@parser.string[start..@parser.pos-1]}' matched '#{@name}' and generated '#{match_result.inspect}'") unless match_result.nil?
|
108
|
+
break
|
109
|
+
else
|
110
|
+
# If this rule did not match the current token list, move
|
111
|
+
# back to the scan position of the last match
|
112
|
+
@parser.pos = start
|
113
|
+
end
|
114
|
+
end
|
115
|
+
|
116
|
+
return match_result
|
117
|
+
end
|
118
|
+
end
|
119
|
+
|
120
|
+
class Parser
|
121
|
+
|
122
|
+
attr_accessor :pos
|
123
|
+
attr_reader :rules, :string, :logger
|
124
|
+
|
125
|
+
class ParseError < RuntimeError
|
126
|
+
end
|
127
|
+
|
128
|
+
def initialize(language_name, locale='en', &block)
|
129
|
+
|
130
|
+
locales_path = File.join(__dir__, "locales", locale)
|
131
|
+
token_pairs = File.readlines(locales_path)
|
132
|
+
@token_list = Hash.new
|
133
|
+
token_pairs.each do |pair|
|
134
|
+
default_value, locale_value = pair.split(' ')
|
135
|
+
@token_list[default_value] = locale_value
|
136
|
+
end
|
137
|
+
|
138
|
+
@logger = Logger.new(STDOUT)
|
139
|
+
@lex_tokens = []
|
140
|
+
@rules = {}
|
141
|
+
@start = nil
|
142
|
+
@language_name = language_name
|
143
|
+
instance_eval(&block)
|
144
|
+
end
|
145
|
+
|
146
|
+
# Tokenize the string into small pieces
|
147
|
+
def tokenize(string)
|
148
|
+
@tokens = []
|
149
|
+
@string = string.clone
|
150
|
+
until string.empty?
|
151
|
+
# Unless any of the valid tokens of our language are the prefix of
|
152
|
+
# 'string', we fail with an exception
|
153
|
+
raise ParseError, "unable to lex '#{string}" unless @lex_tokens.any? do |tok|
|
154
|
+
match = tok.pattern.match(string)
|
155
|
+
# The regular expression of a token has matched the beginning of 'string'
|
156
|
+
if match
|
157
|
+
@logger.debug("Token #{match[0]} consumed as #{match[0]}")
|
158
|
+
# Also, evaluate this expression by using the block
|
159
|
+
# associated with the token
|
160
|
+
@tokens << tok.block.call(match.to_s) if tok.block
|
161
|
+
# consume the match and proceed with the rest of the string
|
162
|
+
string = match.post_match
|
163
|
+
true
|
164
|
+
else
|
165
|
+
# this token pattern did not match, try the next
|
166
|
+
false
|
167
|
+
end # if
|
168
|
+
end # raise
|
169
|
+
end # until
|
170
|
+
end
|
171
|
+
|
172
|
+
def parse(string)
|
173
|
+
# First, split the string according to the "token" instructions given.
|
174
|
+
# Afterwards @tokens contains all tokens that are to be parsed.
|
175
|
+
tokenize(string)
|
176
|
+
|
177
|
+
# These variables are used to match if the total number of tokens
|
178
|
+
# are consumed by the parser
|
179
|
+
@pos = 0
|
180
|
+
@max_pos = 0
|
181
|
+
@expected = []
|
182
|
+
# Parse (and evaluate) the tokens received
|
183
|
+
result = @start.parse
|
184
|
+
# If there are unparsed extra tokens, signal error
|
185
|
+
if @pos != @tokens.size
|
186
|
+
raise ParseError, "Parse error. expected: '#{@expected.join(', ')}', found '#{@tokens[@max_pos]}'"
|
187
|
+
end
|
188
|
+
return result
|
189
|
+
end
|
190
|
+
|
191
|
+
def next_token
|
192
|
+
@pos += 1
|
193
|
+
return @tokens[@pos - 1]
|
194
|
+
end
|
195
|
+
|
196
|
+
# Return the next token in the queue
|
197
|
+
def expect(tok)
|
198
|
+
return tok if tok == :empty
|
199
|
+
t = next_token
|
200
|
+
if @pos - 1 > @max_pos
|
201
|
+
@max_pos = @pos - 1
|
202
|
+
@expected = []
|
203
|
+
end
|
204
|
+
return t if tok === t
|
205
|
+
@expected << tok if @max_pos == @pos - 1 && !@expected.include?(tok)
|
206
|
+
return nil
|
207
|
+
end
|
208
|
+
|
209
|
+
def to_s
|
210
|
+
"Parser for #{@language_name}"
|
211
|
+
end
|
212
|
+
|
213
|
+
private
|
214
|
+
|
215
|
+
LexToken = Struct.new(:pattern, :block)
|
216
|
+
|
217
|
+
def token(pattern, &block)
|
218
|
+
@lex_tokens << LexToken.new(Regexp.new('\\A' + @token_list[pattern.source]), block)
|
219
|
+
end
|
220
|
+
|
221
|
+
def start(name, &block)
|
222
|
+
rule(name, &block)
|
223
|
+
@start = @rules[name]
|
224
|
+
end
|
225
|
+
|
226
|
+
def rule(name,&block)
|
227
|
+
@current_rule = Rule.new(name, self)
|
228
|
+
@rules[name] = @current_rule
|
229
|
+
instance_eval &block # In practise, calls match 1..N times
|
230
|
+
@current_rule = nil
|
231
|
+
end
|
232
|
+
|
233
|
+
def match(*pattern, &block)
|
234
|
+
# Basically calls memberfunction "match(*pattern, &block)
|
235
|
+
@current_rule.send(:match, *pattern, &block)
|
236
|
+
end
|
237
|
+
|
238
|
+
end
|