Refactor execution loop to use Interval instead
This commit is contained in:
parent
c9346da331
commit
a69d8a42b5
@ -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;
|
||||||
|
@ -20,7 +20,7 @@ export type WorkerRequestData =
|
|||||||
}
|
}
|
||||||
| {
|
| {
|
||||||
type: "Execute";
|
type: "Execute";
|
||||||
params: { interval?: number };
|
params: { interval: number };
|
||||||
}
|
}
|
||||||
| {
|
| {
|
||||||
type: "ExecuteStep";
|
type: "ExecuteStep";
|
||||||
|
@ -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)),
|
||||||
|
Loading…
x
Reference in New Issue
Block a user