mirror of
https://github.com/HeyPuter/puter
synced 2024-11-15 06:15:47 +00:00
99 lines
3.3 KiB
JavaScript
99 lines
3.3 KiB
JavaScript
/*
|
|
* 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' };
|
|
}
|
|
}
|