|
|
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; } };
|