first run at ruby block to_vool

leaving the parser structure in the ruby layer,
but adopting the constant approach in vool
This commit is contained in:
Torsten Ruger 2018-07-20 20:07:15 +03:00
parent 0238874c20
commit 235853ab2d
2 changed files with 52 additions and 2 deletions

View File

@ -1,5 +1,10 @@
module Ruby
# The way the ruby parser presents a call with a block is by wrapping the
# whole thing in a :block scope. It includes the send and the block definition.
#
# A block is in essence quite like a method, so the block definition is like a
# method definition, except it is not bound to the class direcly, but the enclosing
# method. The enclosing method also provides the scope.
class BlockStatement < Statement
attr_reader :send , :args , :body
@ -8,8 +13,25 @@ module Ruby
raise "no bod" unless @body
end
# In Vool we "hoist" the block definition through a local assignment, much
# as we would other complex args (bit like in the Normalizer)
# The block is then passed as a normal variable to the send. In other words, the
# BlockStatement resolves to a list of Statements, the last of which is the send
#
def to_vool
BlockStatement.new( @args , @body.normalize)
block_name = "implicit_block_#{object_id}".to_sym
block = Vool::BlockStatement.new( @args.dup , @body.to_vool)
assign = Vool::LocalAssignment.new( block_name , block)
sendd = @send.to_vool
if(sendd.is_a?(Vool::Statements))
ret = sendd
sendd = sendd.last
else
ret = Vool::Statements.new([sendd])
end
sendd.arguments << LocalVariable.new(block_name).to_vool
ret.prepend(assign)
ret
end
end

View File

@ -25,4 +25,32 @@ module Ruby
assert_equal 1 , @lst.body.arguments.length
end
end
class TestBlockStatementVool < MiniTest::Test
include RubyTests
def setup()
input = "plus_one{|arg1| arg1 + 1 } "
@lst = compile( input ).to_vool
end
def test_block
assert_equal Vool::Statements , @lst.class
end
def test_first
assert_equal Vool::LocalAssignment , @lst.first.class
assert @lst.first.name.to_s.start_with?("implicit_block_")
assert_equal Vool::BlockStatement , @lst.first.value.class
end
def test_last_send
assert_equal 2 , @lst.length
assert_equal Vool::SendStatement , @lst.last.class
assert_equal :plus_one , @lst.last.name
end
def test_send_block_arg
assert_equal 1 , @lst.last.arguments.length
assert @lst.last.arguments.first.name.to_s.start_with?("implicit_block_")
end
def test_block_args
assert_equal [:arg1] , @lst.first.value.args
end
end
end