123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380 |
- /**
- * The MIT License (MIT)
- * Copyright (c) 2017-present Dmitry Soshnikov <dmitry.soshnikov@gmail.com>
- */
- 'use strict';
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- var DFAMinimizer = require('./dfa-minimizer');
- var _require = require('../special-symbols'),
- EPSILON_CLOSURE = _require.EPSILON_CLOSURE;
- /**
- * DFA is build by converting from NFA (subset construction).
- */
- var DFA = function () {
- function DFA(nfa) {
- _classCallCheck(this, DFA);
- this._nfa = nfa;
- }
- /**
- * Minimizes DFA.
- */
- _createClass(DFA, [{
- key: 'minimize',
- value: function minimize() {
- this.getTransitionTable();
- this._originalAcceptingStateNumbers = this._acceptingStateNumbers;
- this._originalTransitionTable = this._transitionTable;
- DFAMinimizer.minimize(this);
- }
- /**
- * Returns alphabet for this DFA.
- */
- }, {
- key: 'getAlphabet',
- value: function getAlphabet() {
- return this._nfa.getAlphabet();
- }
- /**
- * Returns accepting states.
- */
- }, {
- key: 'getAcceptingStateNumbers',
- value: function getAcceptingStateNumbers() {
- if (!this._acceptingStateNumbers) {
- // Accepting states are determined during table construction.
- this.getTransitionTable();
- }
- return this._acceptingStateNumbers;
- }
- /**
- * Returns original accepting states.
- */
- }, {
- key: 'getOriginaAcceptingStateNumbers',
- value: function getOriginaAcceptingStateNumbers() {
- if (!this._originalAcceptingStateNumbers) {
- // Accepting states are determined during table construction.
- this.getTransitionTable();
- }
- return this._originalAcceptingStateNumbers;
- }
- /**
- * Sets transition table.
- */
- }, {
- key: 'setTransitionTable',
- value: function setTransitionTable(table) {
- this._transitionTable = table;
- }
- /**
- * Sets accepting states.
- */
- }, {
- key: 'setAcceptingStateNumbers',
- value: function setAcceptingStateNumbers(stateNumbers) {
- this._acceptingStateNumbers = stateNumbers;
- }
- /**
- * DFA transition table is built from NFA table.
- */
- }, {
- key: 'getTransitionTable',
- value: function getTransitionTable() {
- var _this = this;
- if (this._transitionTable) {
- return this._transitionTable;
- }
- // Calculate from NFA transition table.
- var nfaTable = this._nfa.getTransitionTable();
- var nfaStates = Object.keys(nfaTable);
- this._acceptingStateNumbers = new Set();
- // Start state of DFA is E(S[nfa])
- var startState = nfaTable[nfaStates[0]][EPSILON_CLOSURE];
- // Init the worklist (states which should be in the DFA).
- var worklist = [startState];
- var alphabet = this.getAlphabet();
- var nfaAcceptingStates = this._nfa.getAcceptingStateNumbers();
- var dfaTable = {};
- // Determine whether the combined DFA state is accepting.
- var updateAcceptingStates = function updateAcceptingStates(states) {
- var _iteratorNormalCompletion = true;
- var _didIteratorError = false;
- var _iteratorError = undefined;
- try {
- for (var _iterator = nfaAcceptingStates[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
- var nfaAcceptingState = _step.value;
- // If any of the states from NFA is accepting, DFA's
- // state is accepting as well.
- if (states.indexOf(nfaAcceptingState) !== -1) {
- _this._acceptingStateNumbers.add(states.join(','));
- break;
- }
- }
- } catch (err) {
- _didIteratorError = true;
- _iteratorError = err;
- } finally {
- try {
- if (!_iteratorNormalCompletion && _iterator.return) {
- _iterator.return();
- }
- } finally {
- if (_didIteratorError) {
- throw _iteratorError;
- }
- }
- }
- };
- while (worklist.length > 0) {
- var states = worklist.shift();
- var dfaStateLabel = states.join(',');
- dfaTable[dfaStateLabel] = {};
- var _iteratorNormalCompletion2 = true;
- var _didIteratorError2 = false;
- var _iteratorError2 = undefined;
- try {
- for (var _iterator2 = alphabet[Symbol.iterator](), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {
- var symbol = _step2.value;
- var onSymbol = [];
- // Determine whether the combined state is accepting.
- updateAcceptingStates(states);
- var _iteratorNormalCompletion3 = true;
- var _didIteratorError3 = false;
- var _iteratorError3 = undefined;
- try {
- for (var _iterator3 = states[Symbol.iterator](), _step3; !(_iteratorNormalCompletion3 = (_step3 = _iterator3.next()).done); _iteratorNormalCompletion3 = true) {
- var state = _step3.value;
- var nfaStatesOnSymbol = nfaTable[state][symbol];
- if (!nfaStatesOnSymbol) {
- continue;
- }
- var _iteratorNormalCompletion4 = true;
- var _didIteratorError4 = false;
- var _iteratorError4 = undefined;
- try {
- for (var _iterator4 = nfaStatesOnSymbol[Symbol.iterator](), _step4; !(_iteratorNormalCompletion4 = (_step4 = _iterator4.next()).done); _iteratorNormalCompletion4 = true) {
- var nfaStateOnSymbol = _step4.value;
- if (!nfaTable[nfaStateOnSymbol]) {
- continue;
- }
- onSymbol.push.apply(onSymbol, _toConsumableArray(nfaTable[nfaStateOnSymbol][EPSILON_CLOSURE]));
- }
- } catch (err) {
- _didIteratorError4 = true;
- _iteratorError4 = err;
- } finally {
- try {
- if (!_iteratorNormalCompletion4 && _iterator4.return) {
- _iterator4.return();
- }
- } finally {
- if (_didIteratorError4) {
- throw _iteratorError4;
- }
- }
- }
- }
- } catch (err) {
- _didIteratorError3 = true;
- _iteratorError3 = err;
- } finally {
- try {
- if (!_iteratorNormalCompletion3 && _iterator3.return) {
- _iterator3.return();
- }
- } finally {
- if (_didIteratorError3) {
- throw _iteratorError3;
- }
- }
- }
- var dfaStatesOnSymbolSet = new Set(onSymbol);
- var dfaStatesOnSymbol = [].concat(_toConsumableArray(dfaStatesOnSymbolSet));
- if (dfaStatesOnSymbol.length > 0) {
- var dfaOnSymbolStr = dfaStatesOnSymbol.join(',');
- dfaTable[dfaStateLabel][symbol] = dfaOnSymbolStr;
- if (!dfaTable.hasOwnProperty(dfaOnSymbolStr)) {
- worklist.unshift(dfaStatesOnSymbol);
- }
- }
- }
- } catch (err) {
- _didIteratorError2 = true;
- _iteratorError2 = err;
- } finally {
- try {
- if (!_iteratorNormalCompletion2 && _iterator2.return) {
- _iterator2.return();
- }
- } finally {
- if (_didIteratorError2) {
- throw _iteratorError2;
- }
- }
- }
- }
- return this._transitionTable = this._remapStateNumbers(dfaTable);
- }
- /**
- * Remaps state numbers in the resulting table:
- * combined states '1,2,3' -> 1, '3,4' -> 2, etc.
- */
- }, {
- key: '_remapStateNumbers',
- value: function _remapStateNumbers(calculatedDFATable) {
- var newStatesMap = {};
- this._originalTransitionTable = calculatedDFATable;
- var transitionTable = {};
- Object.keys(calculatedDFATable).forEach(function (originalNumber, newNumber) {
- newStatesMap[originalNumber] = newNumber + 1;
- });
- for (var originalNumber in calculatedDFATable) {
- var originalRow = calculatedDFATable[originalNumber];
- var row = {};
- for (var symbol in originalRow) {
- row[symbol] = newStatesMap[originalRow[symbol]];
- }
- transitionTable[newStatesMap[originalNumber]] = row;
- }
- // Remap accepting states.
- this._originalAcceptingStateNumbers = this._acceptingStateNumbers;
- this._acceptingStateNumbers = new Set();
- var _iteratorNormalCompletion5 = true;
- var _didIteratorError5 = false;
- var _iteratorError5 = undefined;
- try {
- for (var _iterator5 = this._originalAcceptingStateNumbers[Symbol.iterator](), _step5; !(_iteratorNormalCompletion5 = (_step5 = _iterator5.next()).done); _iteratorNormalCompletion5 = true) {
- var _originalNumber = _step5.value;
- this._acceptingStateNumbers.add(newStatesMap[_originalNumber]);
- }
- } catch (err) {
- _didIteratorError5 = true;
- _iteratorError5 = err;
- } finally {
- try {
- if (!_iteratorNormalCompletion5 && _iterator5.return) {
- _iterator5.return();
- }
- } finally {
- if (_didIteratorError5) {
- throw _iteratorError5;
- }
- }
- }
- return transitionTable;
- }
- /**
- * Returns original DFA table, where state numbers
- * are combined numbers from NFA.
- */
- }, {
- key: 'getOriginalTransitionTable',
- value: function getOriginalTransitionTable() {
- if (!this._originalTransitionTable) {
- // Original table is determined during table construction.
- this.getTransitionTable();
- }
- return this._originalTransitionTable;
- }
- /**
- * Checks whether this DFA accepts a string.
- */
- }, {
- key: 'matches',
- value: function matches(string) {
- var state = 1;
- var i = 0;
- var table = this.getTransitionTable();
- while (string[i]) {
- state = table[state][string[i++]];
- if (!state) {
- return false;
- }
- }
- if (!this.getAcceptingStateNumbers().has(state)) {
- return false;
- }
- return true;
- }
- }]);
- return DFA;
- }();
- module.exports = DFA;
|