-
Notifications
You must be signed in to change notification settings - Fork 1
/
Features.ts
83 lines (63 loc) · 2.04 KB
/
Features.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
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
import domLoaded from "dom-loaded";
import { pageLoaded } from "./utils/pageLoaded";
// eslint-disable-next-line import/no-cycle
import { Config, getConfig } from "./config";
// eslint-disable-next-line import/no-cycle
import { needsToWaitForApi } from "./utils/needsToWaitForApi";
// eslint-disable-next-line import/no-cycle
import { apiLoaded } from "./utils/apiLoaded";
export type FeatureProps = Config & {};
export type Feature = (props: FeatureProps) => any;
export interface FeatureDescription {
id: string;
feature: Feature;
waitForDomLoaded?: boolean;
waitForPageLoaded?: boolean;
needsApi?: boolean;
}
class Features {
private __addedFeatures: FeatureDescription[] = [];
add(ctx: FeatureDescription): void {
this.__addedFeatures.push(ctx);
}
getAll(): FeatureDescription[] {
return [...this.__addedFeatures];
}
async loadAll(): Promise<void> {
const config = getConfig();
const promises: Promise<any>[] = this.getAll().map(
async ({ id, feature, waitForDomLoaded, waitForPageLoaded, needsApi }) => {
if (config.features[id] === false) {
console.log(`⏭ skipping feature because it's disabled in config, id: \`${id}\``);
return;
}
try {
const featureProps: FeatureProps = { ...config };
const requirements: Promise<any>[] = [];
if (waitForDomLoaded) {
requirements.push(domLoaded);
}
if (waitForPageLoaded) {
requirements.push(pageLoaded);
}
if (needsApi && needsToWaitForApi()) {
requirements.push(apiLoaded);
}
if (requirements.length > 0) {
await Promise.allSettled(requirements);
await feature(featureProps);
console.log(`✅ (⏱) feature loaded (after fulfilling requirements), id: \`${id}\``);
} else {
await feature(featureProps);
console.log(`✅ feature loaded (instantly), id: \`${id}\``);
}
} catch (e) {
console.error(`❌ failed to load feature "${id}", error:`, { e });
}
}
);
await Promise.allSettled(promises);
}
}
const features = new Features();
export { features };