Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
microsoft
GitHub Repository: microsoft/vscode
Path: blob/main/extensions/microsoft-authentication/src/common/async.ts
3320 views
1
/*---------------------------------------------------------------------------------------------
2
* Copyright (c) Microsoft Corporation. All rights reserved.
3
* Licensed under the MIT License. See License.txt in the project root for license information.
4
*--------------------------------------------------------------------------------------------*/
5
6
import { CancellationError, CancellationToken, Disposable, Event } from 'vscode';
7
8
export class SequencerByKey<TKey> {
9
10
private promiseMap = new Map<TKey, Promise<unknown>>();
11
12
queue<T>(key: TKey, promiseTask: () => Promise<T>): Promise<T> {
13
const runningPromise = this.promiseMap.get(key) ?? Promise.resolve();
14
const newPromise = runningPromise
15
.catch(() => { })
16
.then(promiseTask)
17
.finally(() => {
18
if (this.promiseMap.get(key) === newPromise) {
19
this.promiseMap.delete(key);
20
}
21
});
22
this.promiseMap.set(key, newPromise);
23
return newPromise;
24
}
25
}
26
27
export class IntervalTimer extends Disposable {
28
29
private _token: any;
30
31
constructor() {
32
super(() => this.cancel());
33
this._token = -1;
34
}
35
36
cancel(): void {
37
if (this._token !== -1) {
38
clearInterval(this._token);
39
this._token = -1;
40
}
41
}
42
43
cancelAndSet(runner: () => void, interval: number): void {
44
this.cancel();
45
this._token = setInterval(() => {
46
runner();
47
}, interval);
48
}
49
}
50
51
/**
52
* Returns a promise that rejects with an {@CancellationError} as soon as the passed token is cancelled.
53
* @see {@link raceCancellation}
54
*/
55
function raceCancellationError<T>(promise: Promise<T>, token: CancellationToken): Promise<T> {
56
return new Promise((resolve, reject) => {
57
const ref = token.onCancellationRequested(() => {
58
ref.dispose();
59
reject(new CancellationError());
60
});
61
promise.then(resolve, reject).finally(() => ref.dispose());
62
});
63
}
64
65
function raceTimeoutError<T>(promise: Promise<T>, timeout: number): Promise<T> {
66
return new Promise((resolve, reject) => {
67
const ref = setTimeout(() => {
68
reject(new CancellationError());
69
}, timeout);
70
promise.then(resolve, reject).finally(() => clearTimeout(ref));
71
});
72
}
73
74
export function raceCancellationAndTimeoutError<T>(promise: Promise<T>, token: CancellationToken, timeout: number): Promise<T> {
75
return raceCancellationError(raceTimeoutError(promise, timeout), token);
76
}
77
78
/**
79
* Given an event, returns another event which only fires once.
80
*
81
* @param event The event source for the new event.
82
*/
83
function once<T>(event: Event<T>): Event<T> {
84
return (listener, thisArgs = null, disposables?) => {
85
// we need this, in case the event fires during the listener call
86
let didFire = false;
87
let result: Disposable | undefined = undefined;
88
result = event(e => {
89
if (didFire) {
90
return;
91
} else if (result) {
92
result.dispose();
93
} else {
94
didFire = true;
95
}
96
97
return listener.call(thisArgs, e);
98
}, null, disposables);
99
100
if (didFire) {
101
result.dispose();
102
}
103
104
return result;
105
};
106
}
107
108
/**
109
* Creates a promise out of an event, using the {@link Event.once} helper.
110
*/
111
export function toPromise<T>(event: Event<T>): Promise<T> {
112
return new Promise(resolve => once(event)(resolve));
113
}
114
115
//#region DeferredPromise
116
117
export type ValueCallback<T = unknown> = (value: T | Promise<T>) => void;
118
119
const enum DeferredOutcome {
120
Resolved,
121
Rejected
122
}
123
124
/**
125
* Creates a promise whose resolution or rejection can be controlled imperatively.
126
*/
127
export class DeferredPromise<T> {
128
129
private completeCallback!: ValueCallback<T>;
130
private errorCallback!: (err: unknown) => void;
131
private outcome?: { outcome: DeferredOutcome.Rejected; value: any } | { outcome: DeferredOutcome.Resolved; value: T };
132
133
public get isRejected() {
134
return this.outcome?.outcome === DeferredOutcome.Rejected;
135
}
136
137
public get isResolved() {
138
return this.outcome?.outcome === DeferredOutcome.Resolved;
139
}
140
141
public get isSettled() {
142
return !!this.outcome;
143
}
144
145
public get value() {
146
return this.outcome?.outcome === DeferredOutcome.Resolved ? this.outcome?.value : undefined;
147
}
148
149
public readonly p: Promise<T>;
150
151
constructor() {
152
this.p = new Promise<T>((c, e) => {
153
this.completeCallback = c;
154
this.errorCallback = e;
155
});
156
}
157
158
public complete(value: T) {
159
return new Promise<void>(resolve => {
160
this.completeCallback(value);
161
this.outcome = { outcome: DeferredOutcome.Resolved, value };
162
resolve();
163
});
164
}
165
166
public error(err: unknown) {
167
return new Promise<void>(resolve => {
168
this.errorCallback(err);
169
this.outcome = { outcome: DeferredOutcome.Rejected, value: err };
170
resolve();
171
});
172
}
173
174
public cancel() {
175
return this.error(new CancellationError());
176
}
177
}
178
179
//#endregion
180
181