making grammar look more like ruby

This commit is contained in:
Torsten Ruger 2014-04-28 15:46:57 +03:00
parent 0a3253c5c3
commit e04973fc3c
4 changed files with 65 additions and 27 deletions

View File

@ -6,7 +6,6 @@ module Parser
include Parslet
rule(:space) { match('\s').repeat(1) }
rule(:space?) { space.maybe }
rule(:name) { match('[a-z]').repeat(1).as(:name) >> space? }
rule(:double_quote){ str('"') }
rule(:minus) { str('-') }
rule(:plus) { str('+') }
@ -16,17 +15,22 @@ module Parser
rule(:exponent) { (str('e')| str('E')) }
rule(:escaped_character) { str('\\') >> (match('["\\\\/bfnrt]') | (str('u') >> hexdigit.repeat(4,4))) }
rule(:true) { str('true').as(:true) }
rule(:false) { str('false').as(:false) }
rule(:nil) { str('null').as(:nil) }
rule(:true) { str('true').as(:true) >> space?}
rule(:false) { str('false').as(:false) >> space?}
rule(:nil) { str('null').as(:nil) >> space?}
# identifier must start with lower case
rule(:name) { (match['a-z'] >> match['a-zA-Z0-9'].repeat).as(:name) >> space? }
#anything in double quotes
rule(:string){
double_quote >> (escaped_character | double_quote.absent? >> any ).repeat.as(:string) >> double_quote
double_quote >>
(escaped_character | double_quote.absent? >> any ).repeat.as(:string) >>
double_quote >> space?
}
rule(:integer) { sign.maybe >> digit.repeat(1).as(:integer) >> space? }
rule(:float) { integer >> dot >> integer >>
(exponent >> sign.maybe >> digit.repeat).maybe}
(exponent >> sign.maybe >> digit.repeat).maybe >> space?}
end
end

View File

@ -27,17 +27,23 @@ module Parser
rule(:expression) { conditional | function_call | integer | name }
def delimited_expressions( delimit )
((delimit.absent? >> expression).repeat(1)).as(:expressions) >> delimit
end
rule(:conditional) {
keyword_if >> left_parenthesis >> expression.as(:conditional) >> right_parenthesis >>
block.as(:if_true) >>
keyword_else >>
block.as(:if_false)
delimited_expressions(keyword_else).as(:if_true) >>
delimited_expressions(keyword_end).as(:if_false)
}
rule(:block) { left_brace >> expression.as(:block) >> right_brace }
rule(:expressions_else) { delimited_expressions(keyword_else) }
rule(:expressions_end) { delimited_expressions(keyword_end) }
# rule(:expressions_end) { ((keyword_end.absent? >> expression).repeat(1)).as(:expressions) >> keyword_end }
rule(:function_definition) {
keyword_def >> name.as(:function_definition) >> parmeter_list >> block
keyword_def >> name.as(:function_definition) >> parmeter_list >> expressions_end
}
rule(:parmeter_list) {
@ -45,6 +51,6 @@ module Parser
((name.as(:parmeter) >> (comma >> name.as(:parmeter)).repeat(0)).maybe).as(:parmeter_list) >>
right_parenthesis
}
rule(:root){ function_definition.repeat(0) >> expression | expression | argument_list }
rule(:root){ function_definition | expression }
end
end

View File

@ -4,6 +4,6 @@ module Parser
rule(:keyword_if) { str('if') >> space? }
rule(:keyword_else) { str('else') >> space? }
rule(:keyword_def) { str('def') >> space? }
rule(:keyword_end) { str('end') >> space? }
end
end

View File

@ -50,30 +50,58 @@ class ParserTest < MiniTest::Test
check
end
def test_expression_else
@input = <<HERE
4
5
else
HERE
@expected = {:expressions=>[{:integer=>"4"}, {:integer=>"5"}]}
@parser = @parser.expressions_else
check
end
def test_expression_end
@input = <<HERE
5
name
call(4,6)
end
HERE
@expected = {:expressions => [ { :integer => "5" },
{ :name => "name" },
{ :function_call => { :name => "call" } ,
:argument_list => [ {:argument => { :integer => "4" } } ,
{:argument => { :integer => "6" } } ] } ]}
@parser = @parser.expressions_end
check
end
def test_conditional
@input = <<HERE
if (0) {
if (0)
42
} else {
else
667
}
end
HERE
@expected = {:conditional => {:integer => '0'},
:if_true => {:block => {:integer => '42'}},
:if_false => {:block => {:integer => '667'}}}
@expected = { :conditional => { :integer => "0"},
:if_true => { :expressions => [ { :integer => "42" } ] } ,
:if_false => { :expressions => [ { :integer => "667" } ] } }
@parser = @parser.conditional
check
end
def test_function_definition
@input = <<HERE
def foo(x) {
def foo(x)
5
}
end
HERE
@expected = {:function_definition => {:name => 'foo'},
:parmeter_list => {:parmeter => {:name => 'x'}},
:block => {:integer => '5'}}
:expressions => [{:integer => '5'}]}
@parser = @parser.function_definition
check
end