Refactor execution loop to use Interval instead

This commit is contained in:
Nilay Majorwar 2021-12-16 23:05:22 +05:30
parent c9346da331
commit a69d8a42b5
3 changed files with 41 additions and 37 deletions

View File

@ -2,12 +2,12 @@ import { LanguageEngine, StepExecutionResult } from "./types";
type ExecuteAllArgs<RS> = { type ExecuteAllArgs<RS> = {
/** Interval between two execution steps, in milliseconds */ /** Interval between two execution steps, in milliseconds */
interval?: number; interval: number;
/** /**
* Pass to run in streaming-response mode. * Pass to run in streaming-response mode.
* Callback is called with exeuction result on every execution step. * Callback is called with exeuction result on every execution step.
*/ */
onResult?: (result: StepExecutionResult<RS>) => void; onResult: (result: StepExecutionResult<RS>) => void;
}; };
class ExecutionController<RS> { class ExecutionController<RS> {
@ -15,6 +15,7 @@ class ExecutionController<RS> {
private _breakpoints: number[] = []; private _breakpoints: number[] = [];
private _result: StepExecutionResult<RS> | null; private _result: StepExecutionResult<RS> | null;
private _resolvePause: (() => void) | null = null; private _resolvePause: (() => void) | null = null;
private _execInterval: NodeJS.Timeout | null = null;
private _isPaused: boolean = false; private _isPaused: boolean = false;
/** /**
@ -94,44 +95,52 @@ class ExecutionController<RS> {
* Execute the loaded program until stopped. * Execute the loaded program until stopped.
* @param param0.interval Interval between two execution steps * @param param0.interval Interval between two execution steps
* @param param0.onResult Callback called with result on each execution step * @param param0.onResult Callback called with result on each execution step
* @returns Returns last (already used) execution result
*/ */
async executeAll({ interval, onResult }: ExecuteAllArgs<RS>) { async executeAll({ interval, onResult }: ExecuteAllArgs<RS>) {
// Clear paused state // Clear paused state
this._isPaused = false; this._isPaused = false;
console.log(interval);
while (true) { // Run execution loop using an Interval
// Run an execution step in the engine this._execInterval = setInterval(() => {
this._result = this._engine.executeStep(); const doBreak = this.runExecLoopIteration();
onResult(this._result!);
// Check end of program if (doBreak) {
if (!this._result.nextStepLocation) { clearInterval(this._execInterval!);
onResult && onResult(this._result); this._execInterval = null;
this._resolvePause && this._resolvePause(); // In case pause happens on same cycle
break;
} }
}, interval);
}
// Check if execution has been paused /**
if (this._resolvePause) { * Runs a single iteration of execution loop, and sets
this._result.signal = "paused"; * `this._result` to the execution result.
onResult && onResult(this._result); * @returns Boolean - if true, break execution loop.
this._resolvePause && this._resolvePause(); */
break; private runExecLoopIteration(): boolean {
} // Run an execution step in the engine
this._result = this._engine.executeStep();
// Check if next line has breakpoint // Check end of program
if (this._breakpoints.includes(this._result.nextStepLocation.line)) { if (!this._result.nextStepLocation) {
this._result.signal = "paused"; this._resolvePause && this._resolvePause(); // In case pause happens on same cycle
onResult && onResult(this._result); return true;
break;
}
// Continue as usual
onResult && onResult(this._result);
await this.sleep(interval || 0);
} }
return this._result; // Check if execution has been paused
if (this._resolvePause) {
this._result.signal = "paused";
this._resolvePause && this._resolvePause();
return true;
}
// Check if next line has breakpoint
if (this._breakpoints.includes(this._result.nextStepLocation!.line)) {
this._result.signal = "paused";
return true;
}
return false;
} }
/** /**
@ -143,11 +152,6 @@ class ExecutionController<RS> {
this._result.signal = "paused"; this._result.signal = "paused";
return this._result; return this._result;
} }
/** Asynchronously sleep for a period of time */
private async sleep(millis: number) {
return new Promise<void>((resolve) => setTimeout(resolve, millis));
}
} }
export default ExecutionController; export default ExecutionController;

View File

@ -20,7 +20,7 @@ export type WorkerRequestData =
} }
| { | {
type: "Execute"; type: "Execute";
params: { interval?: number }; params: { interval: number };
} }
| { | {
type: "ExecuteStep"; type: "ExecuteStep";

View File

@ -63,7 +63,7 @@ const updateBreakpoints = (points: number[]) => {
* Execute the entire program loaded on engine, * Execute the entire program loaded on engine,
* and return result of execution. * and return result of execution.
*/ */
const execute = (interval?: number) => { const execute = (interval: number) => {
_controller!.executeAll({ _controller!.executeAll({
interval, interval,
onResult: (res) => postMessage(resultMessage(res)), onResult: (res) => postMessage(resultMessage(res)),