puter/packages/strataparse/strata_impls/ContextSwitchingPStratumImpl.js

99 lines
3.3 KiB
JavaScript
Raw Normal View History

2024-04-13 08:53:44 +08:00
/*
* Copyright (C) 2024 Puter Technologies Inc.
*
* This file is part of Phoenix Shell.
*
* Phoenix Shell is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published
* by the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
import { AcceptParserUtil, ParseResult, Parser } from "../parse.js";
export default class ContextSwitchingPStratumImpl {
constructor ({ contexts, entry }) {
this.contexts = { ...contexts };
for ( const key in this.contexts ) {
const new_array = [];
for ( const parser of this.contexts[key] ) {
if ( parser.hasOwnProperty('transition') ) {
new_array.push({
...parser,
parser: AcceptParserUtil.adapt(parser.parser),
})
} else {
new_array.push(AcceptParserUtil.adapt(parser));
}
}
this.contexts[key] = new_array;
}
this.stack = [{
context_name: entry,
}];
this.valid = true;
this.lastvalue = null;
}
get stack_top () {
return this.stack[this.stack.length - 1];
}
get current_context () {
return this.contexts[this.stack_top.context_name];
}
next (api) {
if ( ! this.valid ) return { done: true };
const lexer = api.delegate;
const context = this.current_context;
for ( const spec of context ) {
{
const { done, value } = lexer.look();
this.anti_cycle_i = value === this.lastvalue ? (this.anti_cycle_i || 0) + 1 : 0;
if ( this.anti_cycle_i > 30 ) {
throw new Error('infinite loop');
}
this.lastvalue = value;
if ( done ) return { done };
}
let parser, transition, peek;
if ( spec.hasOwnProperty('parser') ) {
({ parser, transition, peek } = spec);
} else {
parser = spec;
}
const subLexer = lexer.fork();
const result = parser.parse(subLexer);
if ( result.status === ParseResult.UNRECOGNIZED ) {
continue;
}
if ( result.status === ParseResult.INVALID ) {
return { done: true, value: result };
}
if ( ! peek ) lexer.join(subLexer);
if ( transition ) {
if ( transition.pop ) this.stack.pop();
if ( transition.to ) this.stack.push({
context_name: transition.to,
});
}
if ( result.value.$discard || peek ) return this.next(api);
return { done: false, value: result.value };
}
return { done: true, value: 'ran out of parsers' };
}
}