456 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			456 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| var parse = require('../definition-syntax/parse');
 | |
| 
 | |
| var MATCH = { type: 'Match' };
 | |
| var MISMATCH = { type: 'Mismatch' };
 | |
| var DISALLOW_EMPTY = { type: 'DisallowEmpty' };
 | |
| var LEFTPARENTHESIS = 40;  // (
 | |
| var RIGHTPARENTHESIS = 41; // )
 | |
| 
 | |
| function createCondition(match, thenBranch, elseBranch) {
 | |
|     // reduce node count
 | |
|     if (thenBranch === MATCH && elseBranch === MISMATCH) {
 | |
|         return match;
 | |
|     }
 | |
| 
 | |
|     if (match === MATCH && thenBranch === MATCH && elseBranch === MATCH) {
 | |
|         return match;
 | |
|     }
 | |
| 
 | |
|     if (match.type === 'If' && match.else === MISMATCH && thenBranch === MATCH) {
 | |
|         thenBranch = match.then;
 | |
|         match = match.match;
 | |
|     }
 | |
| 
 | |
|     return {
 | |
|         type: 'If',
 | |
|         match: match,
 | |
|         then: thenBranch,
 | |
|         else: elseBranch
 | |
|     };
 | |
| }
 | |
| 
 | |
| function isFunctionType(name) {
 | |
|     return (
 | |
|         name.length > 2 &&
 | |
|         name.charCodeAt(name.length - 2) === LEFTPARENTHESIS &&
 | |
|         name.charCodeAt(name.length - 1) === RIGHTPARENTHESIS
 | |
|     );
 | |
| }
 | |
| 
 | |
| function isEnumCapatible(term) {
 | |
|     return (
 | |
|         term.type === 'Keyword' ||
 | |
|         term.type === 'AtKeyword' ||
 | |
|         term.type === 'Function' ||
 | |
|         term.type === 'Type' && isFunctionType(term.name)
 | |
|     );
 | |
| }
 | |
| 
 | |
| function buildGroupMatchGraph(combinator, terms, atLeastOneTermMatched) {
 | |
|     switch (combinator) {
 | |
|         case ' ':
 | |
|             // Juxtaposing components means that all of them must occur, in the given order.
 | |
|             //
 | |
|             // a b c
 | |
|             // =
 | |
|             // match a
 | |
|             //   then match b
 | |
|             //     then match c
 | |
|             //       then MATCH
 | |
|             //       else MISMATCH
 | |
|             //     else MISMATCH
 | |
|             //   else MISMATCH
 | |
|             var result = MATCH;
 | |
| 
 | |
|             for (var i = terms.length - 1; i >= 0; i--) {
 | |
|                 var term = terms[i];
 | |
| 
 | |
|                 result = createCondition(
 | |
|                     term,
 | |
|                     result,
 | |
|                     MISMATCH
 | |
|                 );
 | |
|             };
 | |
| 
 | |
|             return result;
 | |
| 
 | |
|         case '|':
 | |
|             // A bar (|) separates two or more alternatives: exactly one of them must occur.
 | |
|             //
 | |
|             // a | b | c
 | |
|             // =
 | |
|             // match a
 | |
|             //   then MATCH
 | |
|             //   else match b
 | |
|             //     then MATCH
 | |
|             //     else match c
 | |
|             //       then MATCH
 | |
|             //       else MISMATCH
 | |
| 
 | |
|             var result = MISMATCH;
 | |
|             var map = null;
 | |
| 
 | |
|             for (var i = terms.length - 1; i >= 0; i--) {
 | |
|                 var term = terms[i];
 | |
| 
 | |
|                 // reduce sequence of keywords into a Enum
 | |
|                 if (isEnumCapatible(term)) {
 | |
|                     if (map === null && i > 0 && isEnumCapatible(terms[i - 1])) {
 | |
|                         map = Object.create(null);
 | |
|                         result = createCondition(
 | |
|                             {
 | |
|                                 type: 'Enum',
 | |
|                                 map: map
 | |
|                             },
 | |
|                             MATCH,
 | |
|                             result
 | |
|                         );
 | |
|                     }
 | |
| 
 | |
|                     if (map !== null) {
 | |
|                         var key = (isFunctionType(term.name) ? term.name.slice(0, -1) : term.name).toLowerCase();
 | |
|                         if (key in map === false) {
 | |
|                             map[key] = term;
 | |
|                             continue;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 map = null;
 | |
| 
 | |
|                 // create a new conditonal node
 | |
|                 result = createCondition(
 | |
|                     term,
 | |
|                     MATCH,
 | |
|                     result
 | |
|                 );
 | |
|             };
 | |
| 
 | |
|             return result;
 | |
| 
 | |
|         case '&&':
 | |
|             // A double ampersand (&&) separates two or more components,
 | |
|             // all of which must occur, in any order.
 | |
| 
 | |
|             // Use MatchOnce for groups with a large number of terms,
 | |
|             // since &&-groups produces at least N!-node trees
 | |
|             if (terms.length > 5) {
 | |
|                 return {
 | |
|                     type: 'MatchOnce',
 | |
|                     terms: terms,
 | |
|                     all: true
 | |
|                 };
 | |
|             }
 | |
| 
 | |
|             // Use a combination tree for groups with small number of terms
 | |
|             //
 | |
|             // a && b && c
 | |
|             // =
 | |
|             // match a
 | |
|             //   then [b && c]
 | |
|             //   else match b
 | |
|             //     then [a && c]
 | |
|             //     else match c
 | |
|             //       then [a && b]
 | |
|             //       else MISMATCH
 | |
|             //
 | |
|             // a && b
 | |
|             // =
 | |
|             // match a
 | |
|             //   then match b
 | |
|             //     then MATCH
 | |
|             //     else MISMATCH
 | |
|             //   else match b
 | |
|             //     then match a
 | |
|             //       then MATCH
 | |
|             //       else MISMATCH
 | |
|             //     else MISMATCH
 | |
|             var result = MISMATCH;
 | |
| 
 | |
|             for (var i = terms.length - 1; i >= 0; i--) {
 | |
|                 var term = terms[i];
 | |
|                 var thenClause;
 | |
| 
 | |
|                 if (terms.length > 1) {
 | |
|                     thenClause = buildGroupMatchGraph(
 | |
|                         combinator,
 | |
|                         terms.filter(function(newGroupTerm) {
 | |
|                             return newGroupTerm !== term;
 | |
|                         }),
 | |
|                         false
 | |
|                     );
 | |
|                 } else {
 | |
|                     thenClause = MATCH;
 | |
|                 }
 | |
| 
 | |
|                 result = createCondition(
 | |
|                     term,
 | |
|                     thenClause,
 | |
|                     result
 | |
|                 );
 | |
|             };
 | |
| 
 | |
|             return result;
 | |
| 
 | |
|         case '||':
 | |
|             // A double bar (||) separates two or more options:
 | |
|             // one or more of them must occur, in any order.
 | |
| 
 | |
|             // Use MatchOnce for groups with a large number of terms,
 | |
|             // since ||-groups produces at least N!-node trees
 | |
|             if (terms.length > 5) {
 | |
|                 return {
 | |
|                     type: 'MatchOnce',
 | |
|                     terms: terms,
 | |
|                     all: false
 | |
|                 };
 | |
|             }
 | |
| 
 | |
|             // Use a combination tree for groups with small number of terms
 | |
|             //
 | |
|             // a || b || c
 | |
|             // =
 | |
|             // match a
 | |
|             //   then [b || c]
 | |
|             //   else match b
 | |
|             //     then [a || c]
 | |
|             //     else match c
 | |
|             //       then [a || b]
 | |
|             //       else MISMATCH
 | |
|             //
 | |
|             // a || b
 | |
|             // =
 | |
|             // match a
 | |
|             //   then match b
 | |
|             //     then MATCH
 | |
|             //     else MATCH
 | |
|             //   else match b
 | |
|             //     then match a
 | |
|             //       then MATCH
 | |
|             //       else MATCH
 | |
|             //     else MISMATCH
 | |
|             var result = atLeastOneTermMatched ? MATCH : MISMATCH;
 | |
| 
 | |
|             for (var i = terms.length - 1; i >= 0; i--) {
 | |
|                 var term = terms[i];
 | |
|                 var thenClause;
 | |
| 
 | |
|                 if (terms.length > 1) {
 | |
|                     thenClause = buildGroupMatchGraph(
 | |
|                         combinator,
 | |
|                         terms.filter(function(newGroupTerm) {
 | |
|                             return newGroupTerm !== term;
 | |
|                         }),
 | |
|                         true
 | |
|                     );
 | |
|                 } else {
 | |
|                     thenClause = MATCH;
 | |
|                 }
 | |
| 
 | |
|                 result = createCondition(
 | |
|                     term,
 | |
|                     thenClause,
 | |
|                     result
 | |
|                 );
 | |
|             };
 | |
| 
 | |
|             return result;
 | |
|     }
 | |
| }
 | |
| 
 | |
| function buildMultiplierMatchGraph(node) {
 | |
|     var result = MATCH;
 | |
|     var matchTerm = buildMatchGraph(node.term);
 | |
| 
 | |
|     if (node.max === 0) {
 | |
|         // disable repeating of empty match to prevent infinite loop
 | |
|         matchTerm = createCondition(
 | |
|             matchTerm,
 | |
|             DISALLOW_EMPTY,
 | |
|             MISMATCH
 | |
|         );
 | |
| 
 | |
|         // an occurrence count is not limited, make a cycle;
 | |
|         // to collect more terms on each following matching mismatch
 | |
|         result = createCondition(
 | |
|             matchTerm,
 | |
|             null, // will be a loop
 | |
|             MISMATCH
 | |
|         );
 | |
| 
 | |
|         result.then = createCondition(
 | |
|             MATCH,
 | |
|             MATCH,
 | |
|             result // make a loop
 | |
|         );
 | |
| 
 | |
|         if (node.comma) {
 | |
|             result.then.else = createCondition(
 | |
|                 { type: 'Comma', syntax: node },
 | |
|                 result,
 | |
|                 MISMATCH
 | |
|             );
 | |
|         }
 | |
|     } else {
 | |
|         // create a match node chain for [min .. max] interval with optional matches
 | |
|         for (var i = node.min || 1; i <= node.max; i++) {
 | |
|             if (node.comma && result !== MATCH) {
 | |
|                 result = createCondition(
 | |
|                     { type: 'Comma', syntax: node },
 | |
|                     result,
 | |
|                     MISMATCH
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             result = createCondition(
 | |
|                 matchTerm,
 | |
|                 createCondition(
 | |
|                     MATCH,
 | |
|                     MATCH,
 | |
|                     result
 | |
|                 ),
 | |
|                 MISMATCH
 | |
|             );
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (node.min === 0) {
 | |
|         // allow zero match
 | |
|         result = createCondition(
 | |
|             MATCH,
 | |
|             MATCH,
 | |
|             result
 | |
|         );
 | |
|     } else {
 | |
|         // create a match node chain to collect [0 ... min - 1] required matches
 | |
|         for (var i = 0; i < node.min - 1; i++) {
 | |
|             if (node.comma && result !== MATCH) {
 | |
|                 result = createCondition(
 | |
|                     { type: 'Comma', syntax: node },
 | |
|                     result,
 | |
|                     MISMATCH
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             result = createCondition(
 | |
|                 matchTerm,
 | |
|                 result,
 | |
|                 MISMATCH
 | |
|             );
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| function buildMatchGraph(node) {
 | |
|     if (typeof node === 'function') {
 | |
|         return {
 | |
|             type: 'Generic',
 | |
|             fn: node
 | |
|         };
 | |
|     }
 | |
| 
 | |
|     switch (node.type) {
 | |
|         case 'Group':
 | |
|             var result = buildGroupMatchGraph(
 | |
|                 node.combinator,
 | |
|                 node.terms.map(buildMatchGraph),
 | |
|                 false
 | |
|             );
 | |
| 
 | |
|             if (node.disallowEmpty) {
 | |
|                 result = createCondition(
 | |
|                     result,
 | |
|                     DISALLOW_EMPTY,
 | |
|                     MISMATCH
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             return result;
 | |
| 
 | |
|         case 'Multiplier':
 | |
|             return buildMultiplierMatchGraph(node);
 | |
| 
 | |
|         case 'Type':
 | |
|         case 'Property':
 | |
|             return {
 | |
|                 type: node.type,
 | |
|                 name: node.name,
 | |
|                 syntax: node
 | |
|             };
 | |
| 
 | |
|         case 'Keyword':
 | |
|             return {
 | |
|                 type: node.type,
 | |
|                 name: node.name.toLowerCase(),
 | |
|                 syntax: node
 | |
|             };
 | |
| 
 | |
|         case 'AtKeyword':
 | |
|             return {
 | |
|                 type: node.type,
 | |
|                 name: '@' + node.name.toLowerCase(),
 | |
|                 syntax: node
 | |
|             };
 | |
| 
 | |
|         case 'Function':
 | |
|             return {
 | |
|                 type: node.type,
 | |
|                 name: node.name.toLowerCase() + '(',
 | |
|                 syntax: node
 | |
|             };
 | |
| 
 | |
|         case 'String':
 | |
|             // convert a one char length String to a Token
 | |
|             if (node.value.length === 3) {
 | |
|                 return {
 | |
|                     type: 'Token',
 | |
|                     value: node.value.charAt(1),
 | |
|                     syntax: node
 | |
|                 };
 | |
|             }
 | |
| 
 | |
|             // otherwise use it as is
 | |
|             return {
 | |
|                 type: node.type,
 | |
|                 value: node.value.substr(1, node.value.length - 2).replace(/\\'/g, '\''),
 | |
|                 syntax: node
 | |
|             };
 | |
| 
 | |
|         case 'Token':
 | |
|             return {
 | |
|                 type: node.type,
 | |
|                 value: node.value,
 | |
|                 syntax: node
 | |
|             };
 | |
| 
 | |
|         case 'Comma':
 | |
|             return {
 | |
|                 type: node.type,
 | |
|                 syntax: node
 | |
|             };
 | |
| 
 | |
|         default:
 | |
|             throw new Error('Unknown node type:', node.type);
 | |
|     }
 | |
| }
 | |
| 
 | |
| module.exports = {
 | |
|     MATCH: MATCH,
 | |
|     MISMATCH: MISMATCH,
 | |
|     DISALLOW_EMPTY: DISALLOW_EMPTY,
 | |
|     buildMatchGraph: function(syntaxTree, ref) {
 | |
|         if (typeof syntaxTree === 'string') {
 | |
|             syntaxTree = parse(syntaxTree);
 | |
|         }
 | |
| 
 | |
|         return {
 | |
|             type: 'MatchGraph',
 | |
|             match: buildMatchGraph(syntaxTree),
 | |
|             syntax: ref || null,
 | |
|             source: syntaxTree
 | |
|         };
 | |
|     }
 | |
| };
 |