langium
Version:
A language engineering tool for the Language Server Protocol
115 lines (101 loc) • 4.83 kB
text/typescript
/******************************************************************************
* Copyright 2023 TypeFox GmbH
* This program and the accompanying materials are made available under the
* terms of the MIT License, which is available in the project root.
******************************************************************************/
import { type AbstractCancellationTokenSource, CancellationToken, CancellationTokenSource } from '../utils/cancellation.js';
import { Deferred, isOperationCancelled, startCancelableOperation, type MaybePromise } from '../utils/promise-utils.js';
/**
* Utility service to execute mutually exclusive actions.
*/
export interface WorkspaceLock {
/**
* Performs a single async action, like initializing the workspace or processing document changes.
* Only one action will be executed at a time.
*
* When another action is queued up, the token provided for the action will be cancelled.
* Assuming the action makes use of this token, the next action only has to wait for the current action to finish cancellation.
*/
write(action: (token: CancellationToken) => MaybePromise<void>): Promise<void>;
/**
* Performs a single action, like computing completion results or providing workspace symbols.
* Read actions will only be executed after all write actions have finished. They will be executed in parallel if possible.
*
* If a write action is currently running, the read action will be queued up and executed afterwards.
* If a new write action is queued up while a read action is waiting, the write action will receive priority and will be handled before the read action.
*
* Note that read actions are not allowed to modify anything in the workspace. Please use {@link write} instead.
*/
read<T>(action: () => MaybePromise<T>): Promise<T>;
/**
* Cancels the last queued write action. All previous write actions already have been cancelled.
*/
cancelWrite(): void;
}
type LockAction<T = void> = (token: CancellationToken) => MaybePromise<T>;
interface LockEntry {
action: LockAction<unknown>;
deferred: Deferred<unknown>;
cancellationToken: CancellationToken;
}
export class DefaultWorkspaceLock implements WorkspaceLock {
private previousTokenSource: AbstractCancellationTokenSource = new CancellationTokenSource();
private writeQueue: LockEntry[] = [];
private readQueue: LockEntry[] = [];
private done = true;
write(action: (token: CancellationToken) => MaybePromise<void>): Promise<void> {
this.cancelWrite();
const tokenSource = startCancelableOperation();
this.previousTokenSource = tokenSource;
return this.enqueue(this.writeQueue, action, tokenSource.token);
}
read<T>(action: () => MaybePromise<T>): Promise<T> {
return this.enqueue(this.readQueue, action);
}
private enqueue<T = void>(queue: LockEntry[], action: LockAction<T>, cancellationToken = CancellationToken.None): Promise<T> {
const deferred = new Deferred<unknown>();
const entry: LockEntry = {
action,
deferred,
cancellationToken
};
queue.push(entry);
this.performNextOperation();
return deferred.promise as Promise<T>;
}
private async performNextOperation(): Promise<void> {
if (!this.done) {
return;
}
const entries: LockEntry[] = [];
if (this.writeQueue.length > 0) {
// Just perform the next write action
entries.push(this.writeQueue.shift()!);
} else if (this.readQueue.length > 0) {
// Empty the read queue and perform all actions in parallel
entries.push(...this.readQueue.splice(0, this.readQueue.length));
} else {
return;
}
this.done = false;
await Promise.all(entries.map(async ({ action, deferred, cancellationToken }) => {
try {
// Move the execution of the action to the next event loop tick via `Promise.resolve()`
const result = await Promise.resolve().then(() => action(cancellationToken));
deferred.resolve(result);
} catch (err) {
if (isOperationCancelled(err)) {
// If the operation was cancelled, we don't want to reject the promise
deferred.resolve(undefined);
} else {
deferred.reject(err);
}
}
}));
this.done = true;
this.performNextOperation();
}
cancelWrite(): void {
this.previousTokenSource.cancel();
}
}