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