All files / src/util/async SequencedExecutor.ts

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62                                  1x 1x 1x             1x   1x   1x   3x   1x 4x   4x     4x       2x   2x     4x       4x     4x   4x     1x
/**
 * Copyright 2017-2020 Plexus Interop Deutsche Bank AG
 * SPDX-License-Identifier: Apache-2.0
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
import { Logger } from '../../logger/Logger';
import { default as Queue } from 'typescript-collections/dist/lib/Queue';
import { AsyncHelper } from './AsyncHelper';
import { LoggerFactory } from '../../logger/LoggerFactory';
 
type PendingTask = {
    id: number,
    task: () => Promise<void>;
};
 
export class SequencedExecutor {
 
    private outQueue: Queue<PendingTask> = new Queue<PendingTask>();
 
    private currentId: number = 0;
 
    constructor(private readonly Elog: Logger = LoggerFactory.getLogger('SequencedExecutor')) { }
 
    public async submit(task: () => Promise<void>): Promise<void> {
        const id = ++this.currentId;
        /* istanbul ignore if */
        if (this.log.isTraceEnabled()) {
            this.log.trace(`Scheduling [${id}] task`);
        }
        this.outQueue.enqueue({
            id: this.currentId,
            task
        });
        if (this.outQueue.size() > 1) {
            /* istanbul ignore if */
            if (this.log.isTraceEnabled()) {
                this.log.trace(`Waiting for [${id}] task to execute`);
            }
            await AsyncHelper.waitFor(() => this.outQueue.peek().id === id);
        }
        try {
            /* istanbul ignore if */
            if (this.log.isTraceEnabled()) {
                this.log.trace(`Executing [${id}] task`);
            }
            await this.outQueue.peek().task();
        } finally {
            this.outQueue.dequeue();
        }
    }
}