| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629 | 
							- var hasOwnProperty = Object.prototype.hasOwnProperty;
 
- var matchGraph = require('./match-graph');
 
- var MATCH = matchGraph.MATCH;
 
- var MISMATCH = matchGraph.MISMATCH;
 
- var DISALLOW_EMPTY = matchGraph.DISALLOW_EMPTY;
 
- var TYPE = require('../tokenizer/const').TYPE;
 
- var STUB = 0;
 
- var TOKEN = 1;
 
- var OPEN_SYNTAX = 2;
 
- var CLOSE_SYNTAX = 3;
 
- var EXIT_REASON_MATCH = 'Match';
 
- var EXIT_REASON_MISMATCH = 'Mismatch';
 
- var EXIT_REASON_ITERATION_LIMIT = 'Maximum iteration number exceeded (please fill an issue on https://github.com/csstree/csstree/issues)';
 
- var ITERATION_LIMIT = 15000;
 
- var totalIterationCount = 0;
 
- function reverseList(list) {
 
-     var prev = null;
 
-     var next = null;
 
-     var item = list;
 
-     while (item !== null) {
 
-         next = item.prev;
 
-         item.prev = prev;
 
-         prev = item;
 
-         item = next;
 
-     }
 
-     return prev;
 
- }
 
- function areStringsEqualCaseInsensitive(testStr, referenceStr) {
 
-     if (testStr.length !== referenceStr.length) {
 
-         return false;
 
-     }
 
-     for (var i = 0; i < testStr.length; i++) {
 
-         var testCode = testStr.charCodeAt(i);
 
-         var referenceCode = referenceStr.charCodeAt(i);
 
-         // testCode.toLowerCase() for U+0041 LATIN CAPITAL LETTER A (A) .. U+005A LATIN CAPITAL LETTER Z (Z).
 
-         if (testCode >= 0x0041 && testCode <= 0x005A) {
 
-             testCode = testCode | 32;
 
-         }
 
-         if (testCode !== referenceCode) {
 
-             return false;
 
-         }
 
-     }
 
-     return true;
 
- }
 
- function isCommaContextStart(token) {
 
-     if (token === null) {
 
-         return true;
 
-     }
 
-     return (
 
-         token.type === TYPE.Comma ||
 
-         token.type === TYPE.Function ||
 
-         token.type === TYPE.LeftParenthesis ||
 
-         token.type === TYPE.LeftSquareBracket ||
 
-         token.type === TYPE.LeftCurlyBracket ||
 
-         token.type === TYPE.Delim
 
-     );
 
- }
 
- function isCommaContextEnd(token) {
 
-     if (token === null) {
 
-         return true;
 
-     }
 
-     return (
 
-         token.type === TYPE.RightParenthesis ||
 
-         token.type === TYPE.RightSquareBracket ||
 
-         token.type === TYPE.RightCurlyBracket ||
 
-         token.type === TYPE.Delim
 
-     );
 
- }
 
- function internalMatch(tokens, state, syntaxes) {
 
-     function moveToNextToken() {
 
-         do {
 
-             tokenIndex++;
 
-             token = tokenIndex < tokens.length ? tokens[tokenIndex] : null;
 
-         } while (token !== null && (token.type === TYPE.WhiteSpace || token.type === TYPE.Comment));
 
-     }
 
-     function getNextToken(offset) {
 
-         var nextIndex = tokenIndex + offset;
 
-         return nextIndex < tokens.length ? tokens[nextIndex] : null;
 
-     }
 
-     function stateSnapshotFromSyntax(nextState, prev) {
 
-         return {
 
-             nextState: nextState,
 
-             matchStack: matchStack,
 
-             syntaxStack: syntaxStack,
 
-             thenStack: thenStack,
 
-             tokenIndex: tokenIndex,
 
-             prev: prev
 
-         };
 
-     }
 
-     function pushThenStack(nextState) {
 
-         thenStack = {
 
-             nextState: nextState,
 
-             matchStack: matchStack,
 
-             syntaxStack: syntaxStack,
 
-             prev: thenStack
 
-         };
 
-     }
 
-     function pushElseStack(nextState) {
 
-         elseStack = stateSnapshotFromSyntax(nextState, elseStack);
 
-     }
 
-     function addTokenToMatch() {
 
-         matchStack = {
 
-             type: TOKEN,
 
-             syntax: state.syntax,
 
-             token: token,
 
-             prev: matchStack
 
-         };
 
-         moveToNextToken();
 
-         syntaxStash = null;
 
-         if (tokenIndex > longestMatch) {
 
-             longestMatch = tokenIndex;
 
-         }
 
-     }
 
-     function openSyntax() {
 
-         syntaxStack = {
 
-             syntax: state.syntax,
 
-             opts: state.syntax.opts || (syntaxStack !== null && syntaxStack.opts) || null,
 
-             prev: syntaxStack
 
-         };
 
-         matchStack = {
 
-             type: OPEN_SYNTAX,
 
-             syntax: state.syntax,
 
-             token: matchStack.token,
 
-             prev: matchStack
 
-         };
 
-     }
 
-     function closeSyntax() {
 
-         if (matchStack.type === OPEN_SYNTAX) {
 
-             matchStack = matchStack.prev;
 
-         } else {
 
-             matchStack = {
 
-                 type: CLOSE_SYNTAX,
 
-                 syntax: syntaxStack.syntax,
 
-                 token: matchStack.token,
 
-                 prev: matchStack
 
-             };
 
-         }
 
-         syntaxStack = syntaxStack.prev;
 
-     }
 
-     var syntaxStack = null;
 
-     var thenStack = null;
 
-     var elseStack = null;
 
-     // null – stashing allowed, nothing stashed
 
-     // false – stashing disabled, nothing stashed
 
-     // anithing else – fail stashable syntaxes, some syntax stashed
 
-     var syntaxStash = null;
 
-     var iterationCount = 0; // count iterations and prevent infinite loop
 
-     var exitReason = null;
 
-     var token = null;
 
-     var tokenIndex = -1;
 
-     var longestMatch = 0;
 
-     var matchStack = {
 
-         type: STUB,
 
-         syntax: null,
 
-         token: null,
 
-         prev: null
 
-     };
 
-     moveToNextToken();
 
-     while (exitReason === null && ++iterationCount < ITERATION_LIMIT) {
 
-         // function mapList(list, fn) {
 
-         //     var result = [];
 
-         //     while (list) {
 
-         //         result.unshift(fn(list));
 
-         //         list = list.prev;
 
-         //     }
 
-         //     return result;
 
-         // }
 
-         // console.log('--\n',
 
-         //     '#' + iterationCount,
 
-         //     require('util').inspect({
 
-         //         match: mapList(matchStack, x => x.type === TOKEN ? x.token && x.token.value : x.syntax ? ({ [OPEN_SYNTAX]: '<', [CLOSE_SYNTAX]: '</' }[x.type] || x.type) + '!' + x.syntax.name : null),
 
-         //         token: token && token.value,
 
-         //         tokenIndex,
 
-         //         syntax: syntax.type + (syntax.id ? ' #' + syntax.id : '')
 
-         //     }, { depth: null })
 
-         // );
 
-         switch (state.type) {
 
-             case 'Match':
 
-                 if (thenStack === null) {
 
-                     // turn to MISMATCH when some tokens left unmatched
 
-                     if (token !== null) {
 
-                         // doesn't mismatch if just one token left and it's an IE hack
 
-                         if (tokenIndex !== tokens.length - 1 || (token.value !== '\\0' && token.value !== '\\9')) {
 
-                             state = MISMATCH;
 
-                             break;
 
-                         }
 
-                     }
 
-                     // break the main loop, return a result - MATCH
 
-                     exitReason = EXIT_REASON_MATCH;
 
-                     break;
 
-                 }
 
-                 // go to next syntax (`then` branch)
 
-                 state = thenStack.nextState;
 
-                 // check match is not empty
 
-                 if (state === DISALLOW_EMPTY) {
 
-                     if (thenStack.matchStack === matchStack) {
 
-                         state = MISMATCH;
 
-                         break;
 
-                     } else {
 
-                         state = MATCH;
 
-                     }
 
-                 }
 
-                 // close syntax if needed
 
-                 while (thenStack.syntaxStack !== syntaxStack) {
 
-                     closeSyntax();
 
-                 }
 
-                 // pop stack
 
-                 thenStack = thenStack.prev;
 
-                 break;
 
-             case 'Mismatch':
 
-                 // when some syntax is stashed
 
-                 if (syntaxStash !== null && syntaxStash !== false) {
 
-                     // there is no else branches or a branch reduce match stack
 
-                     if (elseStack === null || tokenIndex > elseStack.tokenIndex) {
 
-                         // restore state from the stash
 
-                         elseStack = syntaxStash;
 
-                         syntaxStash = false; // disable stashing
 
-                     }
 
-                 } else if (elseStack === null) {
 
-                     // no else branches -> break the main loop
 
-                     // return a result - MISMATCH
 
-                     exitReason = EXIT_REASON_MISMATCH;
 
-                     break;
 
-                 }
 
-                 // go to next syntax (`else` branch)
 
-                 state = elseStack.nextState;
 
-                 // restore all the rest stack states
 
-                 thenStack = elseStack.thenStack;
 
-                 syntaxStack = elseStack.syntaxStack;
 
-                 matchStack = elseStack.matchStack;
 
-                 tokenIndex = elseStack.tokenIndex;
 
-                 token = tokenIndex < tokens.length ? tokens[tokenIndex] : null;
 
-                 // pop stack
 
-                 elseStack = elseStack.prev;
 
-                 break;
 
-             case 'MatchGraph':
 
-                 state = state.match;
 
-                 break;
 
-             case 'If':
 
-                 // IMPORTANT: else stack push must go first,
 
-                 // since it stores the state of thenStack before changes
 
-                 if (state.else !== MISMATCH) {
 
-                     pushElseStack(state.else);
 
-                 }
 
-                 if (state.then !== MATCH) {
 
-                     pushThenStack(state.then);
 
-                 }
 
-                 state = state.match;
 
-                 break;
 
-             case 'MatchOnce':
 
-                 state = {
 
-                     type: 'MatchOnceBuffer',
 
-                     syntax: state,
 
-                     index: 0,
 
-                     mask: 0
 
-                 };
 
-                 break;
 
-             case 'MatchOnceBuffer':
 
-                 var terms = state.syntax.terms;
 
-                 if (state.index === terms.length) {
 
-                     // no matches at all or it's required all terms to be matched
 
-                     if (state.mask === 0 || state.syntax.all) {
 
-                         state = MISMATCH;
 
-                         break;
 
-                     }
 
-                     // a partial match is ok
 
-                     state = MATCH;
 
-                     break;
 
-                 }
 
-                 // all terms are matched
 
-                 if (state.mask === (1 << terms.length) - 1) {
 
-                     state = MATCH;
 
-                     break;
 
-                 }
 
-                 for (; state.index < terms.length; state.index++) {
 
-                     var matchFlag = 1 << state.index;
 
-                     if ((state.mask & matchFlag) === 0) {
 
-                         // IMPORTANT: else stack push must go first,
 
-                         // since it stores the state of thenStack before changes
 
-                         pushElseStack(state);
 
-                         pushThenStack({
 
-                             type: 'AddMatchOnce',
 
-                             syntax: state.syntax,
 
-                             mask: state.mask | matchFlag
 
-                         });
 
-                         // match
 
-                         state = terms[state.index++];
 
-                         break;
 
-                     }
 
-                 }
 
-                 break;
 
-             case 'AddMatchOnce':
 
-                 state = {
 
-                     type: 'MatchOnceBuffer',
 
-                     syntax: state.syntax,
 
-                     index: 0,
 
-                     mask: state.mask
 
-                 };
 
-                 break;
 
-             case 'Enum':
 
-                 if (token !== null) {
 
-                     var name = token.value.toLowerCase();
 
-                     // drop \0 and \9 hack from keyword name
 
-                     if (name.indexOf('\\') !== -1) {
 
-                         name = name.replace(/\\[09].*$/, '');
 
-                     }
 
-                     if (hasOwnProperty.call(state.map, name)) {
 
-                         state = state.map[name];
 
-                         break;
 
-                     }
 
-                 }
 
-                 state = MISMATCH;
 
-                 break;
 
-             case 'Generic':
 
-                 var opts = syntaxStack !== null ? syntaxStack.opts : null;
 
-                 var lastTokenIndex = tokenIndex + Math.floor(state.fn(token, getNextToken, opts));
 
-                 if (!isNaN(lastTokenIndex) && lastTokenIndex > tokenIndex) {
 
-                     while (tokenIndex < lastTokenIndex) {
 
-                         addTokenToMatch();
 
-                     }
 
-                     state = MATCH;
 
-                 } else {
 
-                     state = MISMATCH;
 
-                 }
 
-                 break;
 
-             case 'Type':
 
-             case 'Property':
 
-                 var syntaxDict = state.type === 'Type' ? 'types' : 'properties';
 
-                 var dictSyntax = hasOwnProperty.call(syntaxes, syntaxDict) ? syntaxes[syntaxDict][state.name] : null;
 
-                 if (!dictSyntax || !dictSyntax.match) {
 
-                     throw new Error(
 
-                         'Bad syntax reference: ' +
 
-                         (state.type === 'Type'
 
-                             ? '<' + state.name + '>'
 
-                             : '<\'' + state.name + '\'>')
 
-                     );
 
-                 }
 
-                 // stash a syntax for types with low priority
 
-                 if (syntaxStash !== false && token !== null && state.type === 'Type') {
 
-                     var lowPriorityMatching =
 
-                         // https://drafts.csswg.org/css-values-4/#custom-idents
 
-                         // When parsing positionally-ambiguous keywords in a property value, a <custom-ident> production
 
-                         // can only claim the keyword if no other unfulfilled production can claim it.
 
-                         (state.name === 'custom-ident' && token.type === TYPE.Ident) ||
 
-                         // https://drafts.csswg.org/css-values-4/#lengths
 
-                         // ... if a `0` could be parsed as either a <number> or a <length> in a property (such as line-height),
 
-                         // it must parse as a <number>
 
-                         (state.name === 'length' && token.value === '0');
 
-                     if (lowPriorityMatching) {
 
-                         if (syntaxStash === null) {
 
-                             syntaxStash = stateSnapshotFromSyntax(state, elseStack);
 
-                         }
 
-                         state = MISMATCH;
 
-                         break;
 
-                     }
 
-                 }
 
-                 openSyntax();
 
-                 state = dictSyntax.match;
 
-                 break;
 
-             case 'Keyword':
 
-                 var name = state.name;
 
-                 if (token !== null) {
 
-                     var keywordName = token.value;
 
-                     // drop \0 and \9 hack from keyword name
 
-                     if (keywordName.indexOf('\\') !== -1) {
 
-                         keywordName = keywordName.replace(/\\[09].*$/, '');
 
-                     }
 
-                     if (areStringsEqualCaseInsensitive(keywordName, name)) {
 
-                         addTokenToMatch();
 
-                         state = MATCH;
 
-                         break;
 
-                     }
 
-                 }
 
-                 state = MISMATCH;
 
-                 break;
 
-             case 'AtKeyword':
 
-             case 'Function':
 
-                 if (token !== null && areStringsEqualCaseInsensitive(token.value, state.name)) {
 
-                     addTokenToMatch();
 
-                     state = MATCH;
 
-                     break;
 
-                 }
 
-                 state = MISMATCH;
 
-                 break;
 
-             case 'Token':
 
-                 if (token !== null && token.value === state.value) {
 
-                     addTokenToMatch();
 
-                     state = MATCH;
 
-                     break;
 
-                 }
 
-                 state = MISMATCH;
 
-                 break;
 
-             case 'Comma':
 
-                 if (token !== null && token.type === TYPE.Comma) {
 
-                     if (isCommaContextStart(matchStack.token)) {
 
-                         state = MISMATCH;
 
-                     } else {
 
-                         addTokenToMatch();
 
-                         state = isCommaContextEnd(token) ? MISMATCH : MATCH;
 
-                     }
 
-                 } else {
 
-                     state = isCommaContextStart(matchStack.token) || isCommaContextEnd(token) ? MATCH : MISMATCH;
 
-                 }
 
-                 break;
 
-             case 'String':
 
-                 var string = '';
 
-                 for (var lastTokenIndex = tokenIndex; lastTokenIndex < tokens.length && string.length < state.value.length; lastTokenIndex++) {
 
-                     string += tokens[lastTokenIndex].value;
 
-                 }
 
-                 if (areStringsEqualCaseInsensitive(string, state.value)) {
 
-                     while (tokenIndex < lastTokenIndex) {
 
-                         addTokenToMatch();
 
-                     }
 
-                     state = MATCH;
 
-                 } else {
 
-                     state = MISMATCH;
 
-                 }
 
-                 break;
 
-             default:
 
-                 throw new Error('Unknown node type: ' + state.type);
 
-         }
 
-     }
 
-     totalIterationCount += iterationCount;
 
-     switch (exitReason) {
 
-         case null:
 
-             console.warn('[csstree-match] BREAK after ' + ITERATION_LIMIT + ' iterations');
 
-             exitReason = EXIT_REASON_ITERATION_LIMIT;
 
-             matchStack = null;
 
-             break;
 
-         case EXIT_REASON_MATCH:
 
-             while (syntaxStack !== null) {
 
-                 closeSyntax();
 
-             }
 
-             break;
 
-         default:
 
-             matchStack = null;
 
-     }
 
-     return {
 
-         tokens: tokens,
 
-         reason: exitReason,
 
-         iterations: iterationCount,
 
-         match: matchStack,
 
-         longestMatch: longestMatch
 
-     };
 
- }
 
- function matchAsList(tokens, matchGraph, syntaxes) {
 
-     var matchResult = internalMatch(tokens, matchGraph, syntaxes || {});
 
-     if (matchResult.match !== null) {
 
-         var item = reverseList(matchResult.match).prev;
 
-         matchResult.match = [];
 
-         while (item !== null) {
 
-             switch (item.type) {
 
-                 case STUB:
 
-                     break;
 
-                 case OPEN_SYNTAX:
 
-                 case CLOSE_SYNTAX:
 
-                     matchResult.match.push({
 
-                         type: item.type,
 
-                         syntax: item.syntax
 
-                     });
 
-                     break;
 
-                 default:
 
-                     matchResult.match.push({
 
-                         token: item.token.value,
 
-                         node: item.token.node
 
-                     });
 
-                     break;
 
-             }
 
-             item = item.prev;
 
-         }
 
-     }
 
-     return matchResult;
 
- }
 
- function matchAsTree(tokens, matchGraph, syntaxes) {
 
-     var matchResult = internalMatch(tokens, matchGraph, syntaxes || {});
 
-     if (matchResult.match === null) {
 
-         return matchResult;
 
-     }
 
-     var item = matchResult.match;
 
-     var host = matchResult.match = {
 
-         syntax: matchGraph.syntax || null,
 
-         match: []
 
-     };
 
-     var hostStack = [host];
 
-     // revert a list and start with 2nd item since 1st is a stub item
 
-     item = reverseList(item).prev;
 
-     // build a tree
 
-     while (item !== null) {
 
-         switch (item.type) {
 
-             case OPEN_SYNTAX:
 
-                 host.match.push(host = {
 
-                     syntax: item.syntax,
 
-                     match: []
 
-                 });
 
-                 hostStack.push(host);
 
-                 break;
 
-             case CLOSE_SYNTAX:
 
-                 hostStack.pop();
 
-                 host = hostStack[hostStack.length - 1];
 
-                 break;
 
-             default:
 
-                 host.match.push({
 
-                     syntax: item.syntax || null,
 
-                     token: item.token.value,
 
-                     node: item.token.node
 
-                 });
 
-         }
 
-         item = item.prev;
 
-     }
 
-     return matchResult;
 
- }
 
- module.exports = {
 
-     matchAsList: matchAsList,
 
-     matchAsTree: matchAsTree,
 
-     getTotalIterationCount: function() {
 
-         return totalIterationCount;
 
-     }
 
- };
 
 
  |