Allow labeled expressions in the metagrammar (without any meaning yet)

This commit is contained in:
David Majda 2010-05-23 19:29:53 +02:00
parent 409ddf2ae8
commit 52704593cd
5 changed files with 1217 additions and 1065 deletions

View file

@ -275,6 +275,7 @@ PEG.Compiler = {
rule: checkExpression, rule: checkExpression,
choice: checkSubnodes("alternatives"), choice: checkSubnodes("alternatives"),
sequence: checkSubnodes("elements"), sequence: checkSubnodes("elements"),
labeled: checkExpression,
and_predicate: checkExpression, and_predicate: checkExpression,
not_predicate: checkExpression, not_predicate: checkExpression,
optional: checkExpression, optional: checkExpression,
@ -340,6 +341,7 @@ PEG.Compiler = {
} }
}, },
labeled: checkExpression,
and_predicate: checkExpression, and_predicate: checkExpression,
not_predicate: checkExpression, not_predicate: checkExpression,
optional: checkExpression, optional: checkExpression,
@ -406,6 +408,7 @@ PEG.Compiler = {
rule: replaceInExpression, rule: replaceInExpression,
choice: replaceInSubnodes("alternatives"), choice: replaceInSubnodes("alternatives"),
sequence: replaceInSubnodes("elements"), sequence: replaceInSubnodes("elements"),
labeled: replaceInExpression,
and_predicate: replaceInExpression, and_predicate: replaceInExpression,
not_predicate: replaceInExpression, not_predicate: replaceInExpression,
optional: replaceInExpression, optional: replaceInExpression,
@ -596,6 +599,10 @@ PEG.Compiler = {
); );
}, },
labeled: function(node, resultVar) {
return PEG.Compiler.compileNode(node.expression, resultVar);
},
and_predicate: function(node, resultVar) { and_predicate: function(node, resultVar) {
var savedPosVar = PEG.Compiler.generateUniqueIdentifier("savedPos"); var savedPosVar = PEG.Compiler.generateUniqueIdentifier("savedPos");
var savedReportMatchFailuresVar = PEG.Compiler.generateUniqueIdentifier("savedReportMatchFailuresVar"); var savedReportMatchFailuresVar = PEG.Compiler.generateUniqueIdentifier("savedReportMatchFailuresVar");

File diff suppressed because it is too large Load diff

View file

@ -35,7 +35,7 @@ choice
} }
sequence sequence
= prefixed* action { = labeled* action {
var expression = $1.length != 1 var expression = $1.length != 1
? { ? {
type: "sequence", type: "sequence",
@ -48,7 +48,7 @@ sequence
action: $2 action: $2
}; };
} }
/ prefixed* { / labeled* {
return $1.length != 1 return $1.length != 1
? { ? {
type: "sequence", type: "sequence",
@ -57,6 +57,16 @@ sequence
: $1[0]; : $1[0];
} }
labeled
= identifier colon prefixed {
return {
type: "labeled",
label: $1,
expression: $3
};
}
/ prefixed
prefixed prefixed
= and suffixed { return { type: "and_predicate", expression: $2 }; } = and suffixed { return { type: "and_predicate", expression: $2 }; }
/ not suffixed { return { type: "not_predicate", expression: $2 }; } / not suffixed { return { type: "not_predicate", expression: $2 }; }
@ -90,6 +100,7 @@ nonBraceCharacter
= [^{}] = [^{}]
equals = "=" __ { return $1; } equals = "=" __ { return $1; }
colon = ":" __ { return $1; }
slash = "/" __ { return $1; } slash = "/" __ { return $1; }
and = "&" __ { return $1; } and = "&" __ { return $1; }
not = "!" __ { return $1; } not = "!" __ { return $1; }

View file

@ -206,6 +206,7 @@ test("buildParser reports missing referenced rules", function() {
'start = "a" / "b" / missing', 'start = "a" / "b" / missing',
'start = missing "a" "b"', 'start = missing "a" "b"',
'start = "a" "b" missing', 'start = "a" "b" missing',
'start = label:missing',
'start = &missing', 'start = &missing',
'start = !missing', 'start = !missing',
'start = missing?', 'start = missing?',
@ -230,6 +231,7 @@ test("buildParser reports left recursion", function() {
'start = start / "a" / "b"', 'start = start / "a" / "b"',
'start = "a" / "b" / start', 'start = "a" / "b" / start',
'start = start "a" "b"', 'start = start "a" "b"',
'start = label:start',
'start = &start', 'start = &start',
'start = !start', 'start = !start',
'start = start?', 'start = start?',
@ -289,6 +291,12 @@ test("sequences", function() {
parses(posTestParser, "a", "a"); parses(posTestParser, "a", "a");
}); });
test("labels", function() {
var parser = PEG.buildParser('start = label:"a"');
parses(parser, "a", "a");
doesNotParse(parser, "b");
});
test("and predicate", function() { test("and predicate", function() {
var parser = PEG.buildParser('start = "a" &"b" "b"'); var parser = PEG.buildParser('start = "a" &"b" "b"');
parses(parser, "ab", ["a", "", "b"]); parses(parser, "ab", ["a", "", "b"]);

View file

@ -43,6 +43,14 @@ function sequence(elements) {
}; };
} }
function labeled(label, expression) {
return {
type: "labeled",
label: label,
expression: expression
};
}
function nodeWithExpressionConstructor(type) { function nodeWithExpressionConstructor(type) {
return function(expression) { return function(expression) {
return { return {
@ -100,12 +108,12 @@ var literalIjkl = literal("ijkl");
var optionalLiteral = optional(literalAbcd); var optionalLiteral = optional(literalAbcd);
var notAbcd = notPredicate(literalAbcd); var labeledAbcd = labeled("a", literalAbcd);
var notEfgh = notPredicate(literalEfgh); var labeledEfgh = labeled("e", literalEfgh);
var notIjkl = notPredicate(literalIjkl); var labeledIjkl = labeled("i", literalIjkl);
var sequenceEmpty = sequence([]); var sequenceEmpty = sequence([]);
var sequenceNots = sequence([notAbcd, notEfgh, notIjkl]); var sequenceLabeleds = sequence([labeledAbcd, labeledEfgh, labeledIjkl]);
var sequenceLiterals = sequence([literalAbcd, literalEfgh, literalIjkl]); var sequenceLiterals = sequence([literalAbcd, literalEfgh, literalIjkl]);
var choiceLiterals = choice([literalAbcd, literalEfgh, literalIjkl]); var choiceLiterals = choice([literalAbcd, literalEfgh, literalIjkl]);
@ -194,19 +202,31 @@ test("parses sequence", function() {
oneRuleGrammar(action(sequenceEmpty, " code ")) oneRuleGrammar(action(sequenceEmpty, " code "))
); );
grammarParserParses( grammarParserParses(
'start = !"abcd" { code }', 'start = a:"abcd" { code }',
oneRuleGrammar(action(notAbcd, " code ")) oneRuleGrammar(action(labeledAbcd, " code "))
); );
grammarParserParses( grammarParserParses(
'start = !"abcd" !"efgh" !"ijkl" { code }', 'start = a:"abcd" e:"efgh" i:"ijkl" { code }',
oneRuleGrammar(action(sequenceNots, " code ")) oneRuleGrammar(action(sequenceLabeleds, " code "))
); );
grammarParserParses('start = ', oneRuleGrammar(sequenceEmpty)); grammarParserParses('start = ', oneRuleGrammar(sequenceEmpty));
grammarParserParses('start = !"abcd"', oneRuleGrammar(notAbcd)); grammarParserParses('start = a:"abcd"', oneRuleGrammar(labeledAbcd));
grammarParserParses( grammarParserParses(
'start = !"abcd" !"efgh" !"ijkl"', 'start = a:"abcd" e:"efgh" i:"ijkl"',
oneRuleGrammar(sequenceNots) oneRuleGrammar(sequenceLabeleds)
);
});
/* Canonical labeled is "label:\"abcd\"". */
test("parses labeled", function() {
grammarParserParses(
'start = label:!"abcd"',
oneRuleGrammar(labeled("label", notPredicate(literalAbcd)))
);
grammarParserParses(
'start = !"abcd"',
oneRuleGrammar(notPredicate(literalAbcd))
); );
}); });