diff --git a/engines/execution-controller.ts b/engines/execution-controller.ts index e00de83..22cabba 100644 --- a/engines/execution-controller.ts +++ b/engines/execution-controller.ts @@ -2,12 +2,12 @@ import { LanguageEngine, StepExecutionResult } from "./types"; type ExecuteAllArgs = { /** Interval between two execution steps, in milliseconds */ - interval?: number; + interval: number; /** * Pass to run in streaming-response mode. * Callback is called with exeuction result on every execution step. */ - onResult?: (result: StepExecutionResult) => void; + onResult: (result: StepExecutionResult) => void; }; class ExecutionController { @@ -15,6 +15,7 @@ class ExecutionController { private _breakpoints: number[] = []; private _result: StepExecutionResult | null; private _resolvePause: (() => void) | null = null; + private _execInterval: NodeJS.Timeout | null = null; private _isPaused: boolean = false; /** @@ -94,44 +95,52 @@ class ExecutionController { * Execute the loaded program until stopped. * @param param0.interval Interval between two execution steps * @param param0.onResult Callback called with result on each execution step - * @returns Returns last (already used) execution result */ async executeAll({ interval, onResult }: ExecuteAllArgs) { // Clear paused state this._isPaused = false; + console.log(interval); - while (true) { - // Run an execution step in the engine - this._result = this._engine.executeStep(); - - // Check end of program - if (!this._result.nextStepLocation) { - onResult && onResult(this._result); - this._resolvePause && this._resolvePause(); // In case pause happens on same cycle - break; + // Run execution loop using an Interval + this._execInterval = setInterval(() => { + const doBreak = this.runExecLoopIteration(); + onResult(this._result!); + if (doBreak) { + clearInterval(this._execInterval!); + this._execInterval = null; } + }, interval); + } - // Check if execution has been paused - if (this._resolvePause) { - this._result.signal = "paused"; - onResult && onResult(this._result); - this._resolvePause && this._resolvePause(); - break; - } + /** + * Runs a single iteration of execution loop, and sets + * `this._result` to the execution result. + * @returns Boolean - if true, break execution loop. + */ + private runExecLoopIteration(): boolean { + // Run an execution step in the engine + this._result = this._engine.executeStep(); - // Check if next line has breakpoint - if (this._breakpoints.includes(this._result.nextStepLocation.line)) { - this._result.signal = "paused"; - onResult && onResult(this._result); - break; - } - - // Continue as usual - onResult && onResult(this._result); - await this.sleep(interval || 0); + // Check end of program + if (!this._result.nextStepLocation) { + this._resolvePause && this._resolvePause(); // In case pause happens on same cycle + return true; } - 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 { this._result.signal = "paused"; return this._result; } - - /** Asynchronously sleep for a period of time */ - private async sleep(millis: number) { - return new Promise((resolve) => setTimeout(resolve, millis)); - } } export default ExecutionController; diff --git a/engines/worker-constants.ts b/engines/worker-constants.ts index 0091899..8fe325c 100644 --- a/engines/worker-constants.ts +++ b/engines/worker-constants.ts @@ -20,7 +20,7 @@ export type WorkerRequestData = } | { type: "Execute"; - params: { interval?: number }; + params: { interval: number }; } | { type: "ExecuteStep"; diff --git a/engines/worker.ts b/engines/worker.ts index 3840fb4..23062a9 100644 --- a/engines/worker.ts +++ b/engines/worker.ts @@ -63,7 +63,7 @@ const updateBreakpoints = (points: number[]) => { * Execute the entire program loaded on engine, * and return result of execution. */ -const execute = (interval?: number) => { +const execute = (interval: number) => { _controller!.executeAll({ interval, onResult: (res) => postMessage(resultMessage(res)),