factor/extra/peg/ebnf/ebnf-tests.factor

400 lines
9.3 KiB
Factor
Raw Normal View History

2007-11-27 00:13:36 -05:00
! Copyright (C) 2007 Chris Double.
! See http://factorcode.org/license.txt for BSD license.
!
2008-04-28 22:19:14 -04:00
USING: kernel tools.test peg peg.ebnf words math math.parser
sequences accessors ;
2008-03-01 17:00:45 -05:00
IN: peg.ebnf.tests
2007-11-27 00:13:36 -05:00
{ T{ ebnf-non-terminal f "abc" } } [
2008-04-28 22:19:14 -04:00
"abc" 'non-terminal' parse ast>>
2007-11-27 00:13:36 -05:00
] unit-test
{ T{ ebnf-terminal f "55" } } [
2008-04-28 22:19:14 -04:00
"'55'" 'terminal' parse ast>>
2007-11-27 00:13:36 -05:00
] unit-test
2007-11-27 16:28:28 -05:00
{
T{ ebnf-rule f
2007-11-27 19:05:53 -05:00
"digit"
T{ ebnf-choice f
V{ T{ ebnf-terminal f "1" } T{ ebnf-terminal f "2" } }
2007-11-27 16:28:28 -05:00
}
}
} [
2008-04-28 22:19:14 -04:00
"digit = '1' | '2'" 'rule' parse ast>>
2007-11-27 16:28:28 -05:00
] unit-test
{
T{ ebnf-rule f
"digit"
T{ ebnf-sequence f
V{ T{ ebnf-terminal f "1" } T{ ebnf-terminal f "2" } }
2007-11-27 16:28:28 -05:00
}
2007-11-27 19:05:53 -05:00
}
2007-11-27 16:28:28 -05:00
} [
2008-04-28 22:19:14 -04:00
"digit = '1' '2'" 'rule' parse ast>>
] unit-test
{
T{ ebnf-choice f
V{
T{ ebnf-sequence f
V{ T{ ebnf-non-terminal f "one" } T{ ebnf-non-terminal f "two" } }
}
T{ ebnf-non-terminal f "three" }
}
}
} [
2008-04-28 22:19:14 -04:00
"one two | three" 'choice' parse ast>>
2007-11-27 21:26:25 -05:00
] unit-test
{
T{ ebnf-sequence f
V{
T{ ebnf-non-terminal f "one" }
2008-04-28 22:15:05 -04:00
T{ ebnf-whitespace f
T{ ebnf-choice f
V{ T{ ebnf-non-terminal f "two" } T{ ebnf-non-terminal f "three" } }
}
2007-11-27 21:26:25 -05:00
}
}
}
} [
2008-04-28 22:19:14 -04:00
"one {two | three}" 'choice' parse ast>>
2007-11-27 21:32:04 -05:00
] unit-test
{
T{ ebnf-sequence f
V{
T{ ebnf-non-terminal f "one" }
T{ ebnf-repeat0 f
T{ ebnf-sequence f
V{
T{ ebnf-choice f
V{ T{ ebnf-non-terminal f "two" } T{ ebnf-non-terminal f "three" } }
}
T{ ebnf-non-terminal f "four" }
}
}
}
}
}
} [
2008-04-28 22:19:14 -04:00
"one ((two | three) four)*" 'choice' parse ast>>
2007-11-27 21:49:14 -05:00
] unit-test
{
T{ ebnf-sequence f
V{
T{ ebnf-non-terminal f "one" }
T{ ebnf-optional f T{ ebnf-non-terminal f "two" } }
T{ ebnf-non-terminal f "three" }
}
}
} [
2008-04-28 22:19:14 -04:00
"one ( two )? three" 'choice' parse ast>>
2007-11-27 21:49:14 -05:00
] unit-test
2008-03-19 00:34:28 -04:00
{ "foo" } [
2008-04-28 22:19:14 -04:00
"\"foo\"" 'identifier' parse ast>>
2008-03-19 00:34:28 -04:00
] unit-test
{ "foo" } [
2008-04-28 22:19:14 -04:00
"'foo'" 'identifier' parse ast>>
2008-03-19 00:34:28 -04:00
] unit-test
{ "foo" } [
2008-04-28 22:19:14 -04:00
"foo" 'non-terminal' parse ast>> ebnf-non-terminal-symbol
2008-03-19 00:34:28 -04:00
] unit-test
{ "foo" } [
2008-04-28 22:19:14 -04:00
"foo]" 'non-terminal' parse ast>> ebnf-non-terminal-symbol
2008-03-19 00:34:28 -04:00
] unit-test
{ V{ "a" "b" } } [
2008-04-28 22:19:14 -04:00
"ab" [EBNF foo='a' 'b' EBNF] call ast>>
] unit-test
{ V{ 1 "b" } } [
2008-04-28 22:19:14 -04:00
"ab" [EBNF foo=('a')[[ drop 1 ]] 'b' EBNF] call ast>>
] unit-test
{ V{ 1 2 } } [
2008-04-28 22:19:14 -04:00
"ab" [EBNF foo=('a') [[ drop 1 ]] ('b') [[ drop 2 ]] EBNF] call ast>>
] unit-test
{ CHAR: A } [
2008-04-28 22:19:14 -04:00
"A" [EBNF foo=[A-Z] EBNF] call ast>>
] unit-test
{ CHAR: Z } [
2008-04-28 22:19:14 -04:00
"Z" [EBNF foo=[A-Z] EBNF] call ast>>
] unit-test
{ f } [
"0" [EBNF foo=[A-Z] EBNF] call
] unit-test
{ CHAR: 0 } [
2008-04-28 22:19:14 -04:00
"0" [EBNF foo=[^A-Z] EBNF] call ast>>
] unit-test
{ f } [
"A" [EBNF foo=[^A-Z] EBNF] call
] unit-test
{ f } [
"Z" [EBNF foo=[^A-Z] EBNF] call
2008-03-27 06:54:34 -04:00
] unit-test
{ V{ "1" "+" "foo" } } [
2008-04-28 22:19:14 -04:00
"1+1" [EBNF foo='1' '+' '1' [[ drop "foo" ]] EBNF] call ast>>
] unit-test
{ "foo" } [
2008-04-28 22:19:14 -04:00
"1+1" [EBNF foo='1' '+' '1' => [[ drop "foo" ]] EBNF] call ast>>
] unit-test
{ "foo" } [
2008-04-28 22:19:14 -04:00
"1+1" [EBNF foo='1' '+' '1' => [[ drop "foo" ]] | '1' '-' '1' => [[ drop "bar" ]] EBNF] call ast>>
] unit-test
{ "bar" } [
2008-04-28 22:19:14 -04:00
"1-1" [EBNF foo='1' '+' '1' => [[ drop "foo" ]] | '1' '-' '1' => [[ drop "bar" ]] EBNF] call ast>>
] unit-test
{ 6 } [
2008-04-28 22:19:14 -04:00
"4+2" [EBNF num=[0-9] => [[ digit> ]] foo=num:x '+' num:y => [[ drop x y + ]] EBNF] call ast>>
] unit-test
{ 6 } [
2008-04-28 22:19:14 -04:00
"4+2" [EBNF foo=[0-9]:x '+' [0-9]:y => [[ drop x digit> y digit> + ]] EBNF] call ast>>
] unit-test
{ 10 } [
2008-04-28 22:19:14 -04:00
{ 1 2 3 4 } [EBNF num=. ?[ number? ]? list=list:x num:y => [[ drop x y + ]] | num EBNF] call ast>>
] unit-test
{ f } [
{ "a" 2 3 4 } [EBNF num=. ?[ number? ]? list=list:x num:y => [[ drop x y + ]] | num EBNF] call
] unit-test
{ 3 } [
2008-04-28 22:19:14 -04:00
{ 1 2 "a" 4 } [EBNF num=. ?[ number? ]? list=list:x num:y => [[ drop x y + ]] | num EBNF] call ast>>
] unit-test
{ f } [
"ab" [EBNF -=" " | "\t" | "\n" foo="a" - "b" EBNF] call
] unit-test
{ V{ "a" " " "b" } } [
2008-04-28 22:19:14 -04:00
"a b" [EBNF -=" " | "\t" | "\n" foo="a" - "b" EBNF] call ast>>
] unit-test
{ V{ "a" "\t" "b" } } [
2008-04-28 22:19:14 -04:00
"a\tb" [EBNF -=" " | "\t" | "\n" foo="a" - "b" EBNF] call ast>>
] unit-test
{ V{ "a" "\n" "b" } } [
2008-04-28 22:19:14 -04:00
"a\nb" [EBNF -=" " | "\t" | "\n" foo="a" - "b" EBNF] call ast>>
] unit-test
{ V{ "a" f "b" } } [
2008-04-28 22:19:14 -04:00
"ab" [EBNF -=" " | "\t" | "\n" foo="a" (-)? "b" EBNF] call ast>>
] unit-test
{ V{ "a" " " "b" } } [
2008-04-28 22:19:14 -04:00
"a b" [EBNF -=" " | "\t" | "\n" foo="a" (-)? "b" EBNF] call ast>>
] unit-test
{ V{ "a" "\t" "b" } } [
2008-04-28 22:19:14 -04:00
"a\tb" [EBNF -=" " | "\t" | "\n" foo="a" (-)? "b" EBNF] call ast>>
] unit-test
{ V{ "a" "\n" "b" } } [
2008-04-28 22:19:14 -04:00
"a\nb" [EBNF -=" " | "\t" | "\n" foo="a" (-)? "b" EBNF] call ast>>
] unit-test
{ V{ "a" "b" } } [
2008-04-28 22:19:14 -04:00
"ab" [EBNF -=(" " | "\t" | "\n")? => [[ drop ignore ]] foo="a" - "b" EBNF] call ast>>
] unit-test
{ V{ "a" "b" } } [
2008-04-28 22:19:14 -04:00
"a\tb" [EBNF -=(" " | "\t" | "\n")? => [[ drop ignore ]] foo="a" - "b" EBNF] call ast>>
] unit-test
{ V{ "a" "b" } } [
2008-04-28 22:19:14 -04:00
"a\nb" [EBNF -=(" " | "\t" | "\n")? => [[ drop ignore ]] foo="a" - "b" EBNF] call ast>>
] unit-test
{ f } [
"axb" [EBNF -=(" " | "\t" | "\n")? => [[ drop ignore ]] foo="a" - "b" EBNF] call
] unit-test
2008-03-28 09:51:49 -04:00
{ V{ V{ 49 } "+" V{ 49 } } } [
#! Test direct left recursion.
2008-03-28 07:49:39 -04:00
#! Using packrat, so first part of expr fails, causing 2nd choice to be used
2008-04-28 22:19:14 -04:00
"1+1" [EBNF num=([0-9])+ expr=expr "+" num | num EBNF] call ast>>
2008-03-28 07:49:39 -04:00
] unit-test
2008-03-27 06:54:34 -04:00
2008-03-28 09:51:49 -04:00
{ V{ V{ V{ 49 } "+" V{ 49 } } "+" V{ 49 } } } [
#! Test direct left recursion.
2008-03-28 07:41:41 -04:00
#! Using packrat, so first part of expr fails, causing 2nd choice to be used
2008-04-28 22:19:14 -04:00
"1+1+1" [EBNF num=([0-9])+ expr=expr "+" num | num EBNF] call ast>>
2008-03-28 09:51:49 -04:00
] unit-test
{ V{ V{ V{ 49 } "+" V{ 49 } } "+" V{ 49 } } } [
#! Test indirect left recursion.
#! Using packrat, so first part of expr fails, causing 2nd choice to be used
2008-04-28 22:19:14 -04:00
"1+1+1" [EBNF num=([0-9])+ x=expr expr=x "+" num | num EBNF] call ast>>
2008-03-28 07:49:39 -04:00
] unit-test
2008-03-27 06:54:34 -04:00
2008-04-02 23:09:03 -04:00
{ t } [
"abcd='9' | ('8'):x => [[ drop x ]]" 'ebnf' parse parse-result-remaining empty?
] unit-test
EBNF: primary
Primary = PrimaryNoNewArray
PrimaryNoNewArray = ClassInstanceCreationExpression
| MethodInvocation
| FieldAccess
| ArrayAccess
| "this"
ClassInstanceCreationExpression = "new" ClassOrInterfaceType "(" ")"
| Primary "." "new" Identifier "(" ")"
MethodInvocation = Primary "." MethodName "(" ")"
| MethodName "(" ")"
FieldAccess = Primary "." Identifier
| "super" "." Identifier
ArrayAccess = Primary "[" Expression "]"
| ExpressionName "[" Expression "]"
ClassOrInterfaceType = ClassName | InterfaceTypeName
ClassName = "C" | "D"
InterfaceTypeName = "I" | "J"
Identifier = "x" | "y" | ClassOrInterfaceType
MethodName = "m" | "n"
ExpressionName = Identifier
Expression = "i" | "j"
main = Primary
;EBNF
{ "this" } [
2008-04-28 22:19:14 -04:00
"this" primary ast>>
] unit-test
{ V{ "this" "." "x" } } [
2008-04-28 22:19:14 -04:00
"this.x" primary ast>>
] unit-test
{ V{ V{ "this" "." "x" } "." "y" } } [
2008-04-28 22:19:14 -04:00
"this.x.y" primary ast>>
] unit-test
{ V{ V{ "this" "." "x" } "." "m" "(" ")" } } [
2008-04-28 22:19:14 -04:00
"this.x.m()" primary ast>>
] unit-test
{ V{ V{ V{ "x" "[" "i" "]" } "[" "j" "]" } "." "y" } } [
2008-04-28 22:19:14 -04:00
"x[i][j].y" primary ast>>
] unit-test
2008-04-14 06:42:45 -04:00
'ebnf' compile must-infer
{ V{ V{ "a" "b" } "c" } } [
2008-04-28 22:19:14 -04:00
"abc" [EBNF a="a" "b" foo=(a "c") EBNF] call ast>>
] unit-test
2008-04-28 22:15:05 -04:00
{ V{ V{ "a" "b" } "c" } } [
2008-04-28 22:19:14 -04:00
"abc" [EBNF a="a" "b" foo={a "c"} EBNF] call ast>>
2008-04-28 22:15:05 -04:00
] unit-test
{ V{ V{ "a" "b" } "c" } } [
2008-04-28 22:19:14 -04:00
"abc" [EBNF a="a" "b" foo=a "c" EBNF] call ast>>
2008-04-28 22:15:05 -04:00
] unit-test
{ f } [
"a bc" [EBNF a="a" "b" foo=(a "c") EBNF] call
] unit-test
{ f } [
2008-04-28 22:15:05 -04:00
"a bc" [EBNF a="a" "b" foo=a "c" EBNF] call
] unit-test
{ f } [
2008-04-28 22:15:05 -04:00
"a bc" [EBNF a="a" "b" foo={a "c"} EBNF] call
] unit-test
2008-04-28 22:15:05 -04:00
{ f } [
"ab c" [EBNF a="a" "b" foo=a "c" EBNF] call
] unit-test
{ V{ V{ "a" "b" } "c" } } [
2008-04-28 22:19:14 -04:00
"ab c" [EBNF a="a" "b" foo={a "c"} EBNF] call ast>>
2008-04-28 22:15:05 -04:00
] unit-test
{ f } [
"ab c" [EBNF a="a" "b" foo=(a "c") EBNF] call
] unit-test
{ f } [
"a b c" [EBNF a="a" "b" foo=a "c" EBNF] call
] unit-test
{ f } [
"a b c" [EBNF a="a" "b" foo=(a "c") EBNF] call
] unit-test
{ f } [
2008-04-28 22:15:05 -04:00
"a b c" [EBNF a="a" "b" foo={a "c"} EBNF] call
] unit-test
{ V{ V{ V{ "a" "b" } "c" } V{ V{ "a" "b" } "c" } } } [
2008-04-28 22:19:14 -04:00
"ab cab c" [EBNF a="a" "b" foo={a "c"}* EBNF] call ast>>
] unit-test
2008-04-28 22:15:05 -04:00
{ V{ } } [
2008-04-28 22:19:14 -04:00
"ab cab c" [EBNF a="a" "b" foo=(a "c")* EBNF] call ast>>
2008-04-28 22:15:05 -04:00
] unit-test
{ V{ V{ V{ "a" "b" } "c" } V{ V{ "a" "b" } "c" } } } [
2008-04-28 22:19:14 -04:00
"ab c ab c" [EBNF a="a" "b" foo={a "c"}* EBNF] call ast>>
2008-04-28 22:15:05 -04:00
] unit-test
{ V{ } } [
2008-04-28 22:19:14 -04:00
"ab c ab c" [EBNF a="a" "b" foo=(a "c")* EBNF] call ast>>
2008-04-28 22:15:05 -04:00
] unit-test
2008-06-16 01:39:14 -04:00
{ V{ "a" "a" "a" } } [
"aaa" [EBNF a=('a')* b=!('b') a:x => [[ drop x ]] EBNF] call ast>>
] unit-test
{ t } [
"aaa" [EBNF a=('a')* b=!('b') a:x => [[ drop x ]] EBNF] call ast>>
"aaa" [EBNF a=('a')* b=!('b') (a):x => [[ drop x ]] EBNF] call ast>> =
] unit-test
{ V{ "a" "a" "a" } } [
"aaa" [EBNF a=('a')* b=a:x => [[ drop x ]] EBNF] call ast>>
] unit-test
{ t } [
"aaa" [EBNF a=('a')* b=a:x => [[ drop x ]] EBNF] call ast>>
"aaa" [EBNF a=('a')* b=(a):x => [[ drop x ]] EBNF] call ast>> =
] unit-test
{ t } [
"number=(digit)+:n 'a'" 'ebnf' parse remaining>> length zero?
] unit-test
{ t } [
"number=(digit)+ 'a'" 'ebnf' parse remaining>> length zero?
] unit-test
{ t } [
"number=digit+ 'a'" 'ebnf' parse remaining>> length zero?
] unit-test
{ t } [
"number=digit+:n 'a'" 'ebnf' parse remaining>> length zero?
] unit-test