1 | ;
|
2 | Object.defineProperty(exports, "__esModule", { value: true });
|
3 | const tslib_1 = require("tslib");
|
4 | const fs = tslib_1.__importStar(require("fs-extra"));
|
5 | const path = tslib_1.__importStar(require("path"));
|
6 | const rxjs_1 = require("rxjs");
|
7 | const operators_1 = require("rxjs/operators");
|
8 | const seamless_immutable_1 = tslib_1.__importDefault(require("seamless-immutable"));
|
9 | const watchConfig$ = (file) => rxjs_1.Observable.create((observer) => {
|
10 | // import('chokidar').FSWatcher
|
11 | // tslint:disable-next-line no-any
|
12 | let watcher;
|
13 | let closed = false;
|
14 | Promise.resolve().then(() => tslib_1.__importStar(require('chokidar'))).then((chokidar) => {
|
15 | if (!closed) {
|
16 | watcher = chokidar.watch(file, { ignoreInitial: false });
|
17 | watcher.on('add', () => {
|
18 | observer.next('change');
|
19 | });
|
20 | watcher.on('change', () => {
|
21 | observer.next('change');
|
22 | });
|
23 | watcher.on('error', (error) => {
|
24 | observer.error(error);
|
25 | });
|
26 | watcher.on('unlink', () => {
|
27 | observer.error(new Error('Configuration file deleted.'));
|
28 | });
|
29 | }
|
30 | })
|
31 | .catch((error) => observer.error(error));
|
32 | return () => {
|
33 | closed = true;
|
34 | if (watcher !== undefined) {
|
35 | watcher.close();
|
36 | }
|
37 | };
|
38 | });
|
39 | class Config {
|
40 | constructor({ name: configName, defaultConfig, schema, configPath, }) {
|
41 | this.configPath = path.resolve(configPath, `${configName}.json`);
|
42 | this.defaultConfig = defaultConfig;
|
43 | this.schema = schema;
|
44 | this.config$ = rxjs_1.defer(async () => this.getConfig()).pipe(operators_1.switchMap((config) => watchConfig$(this.configPath).pipe(operators_1.mergeScan((prevConfig) => rxjs_1.defer(async () => this.getConfig({ config: prevConfig })), config, 1))), operators_1.distinctUntilChanged());
|
45 | }
|
46 | async update({ config }) {
|
47 | await this.validate(config);
|
48 | await fs.ensureDir(path.dirname(this.configPath));
|
49 | await fs.writeFile(this.configPath, JSON.stringify(config));
|
50 | return config;
|
51 | }
|
52 | async getConfig({ config = this.defaultConfig } = {}) {
|
53 | let contents;
|
54 | try {
|
55 | contents = await fs.readFile(this.configPath, 'utf8');
|
56 | }
|
57 | catch (error) {
|
58 | if (error.code === 'ENOENT') {
|
59 | return this.update({ config });
|
60 | }
|
61 | throw error;
|
62 | }
|
63 | const currentConfig = JSON.parse(contents);
|
64 | await this.validate(currentConfig);
|
65 | if (config !== undefined) {
|
66 | // tslint:disable-next-line no-any
|
67 | return seamless_immutable_1.default.merge(config, currentConfig, {
|
68 | deep: true,
|
69 | });
|
70 | }
|
71 | return currentConfig;
|
72 | }
|
73 | async validate(config) {
|
74 | const validateConfig = await this.getValidateConfig();
|
75 | const isValid = validateConfig(config);
|
76 | if (!isValid) {
|
77 | const error = new Error('Invalid config');
|
78 | // tslint:disable-next-line no-object-mutation no-any
|
79 | error.errors = validateConfig.errors;
|
80 | throw error;
|
81 | }
|
82 | }
|
83 | // Promise<import('ajv').ValidateFunction>
|
84 | // tslint:disable-next-line no-any
|
85 | async getValidateConfig() {
|
86 | if (this.mutableValidateConfig !== undefined) {
|
87 | return this.mutableValidateConfig;
|
88 | }
|
89 | const ajv = await Promise.resolve().then(() => tslib_1.__importStar(require('ajv')));
|
90 | // tslint:disable-next-line no-any
|
91 | const validateConfig = new ajv.default().compile(this.schema);
|
92 | this.mutableValidateConfig = validateConfig;
|
93 | return validateConfig;
|
94 | }
|
95 | }
|
96 | exports.Config = Config;
|
97 |
|
98 | //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["Config.ts"],"names":[],"mappings":"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","file":"neo-one-server-plugin/src/Config.js","sourcesContent":["import * as fs from 'fs-extra';\nimport * as path from 'path';\nimport { defer, Observable, Observer } from 'rxjs';\nimport { distinctUntilChanged, mergeScan, switchMap } from 'rxjs/operators';\nimport SeamlessImmutable from 'seamless-immutable';\n\ntype Event = 'change';\n\nconst watchConfig$ = (file: string): Observable<Event> =>\n  Observable.create((observer: Observer<string>) => {\n    // import('chokidar').FSWatcher\n    // tslint:disable-next-line no-any\n    let watcher: any | undefined;\n    let closed = false;\n    import('chokidar')\n      .then((chokidar) => {\n        if (!closed) {\n          watcher = chokidar.watch(file, { ignoreInitial: false });\n          watcher.on('add', () => {\n            observer.next('change');\n          });\n          watcher.on('change', () => {\n            observer.next('change');\n          });\n          watcher.on('error', (error: Error) => {\n            observer.error(error);\n          });\n          watcher.on('unlink', () => {\n            observer.error(new Error('Configuration file deleted.'));\n          });\n        }\n      })\n      .catch((error) => observer.error(error));\n\n    return () => {\n      closed = true;\n      if (watcher !== undefined) {\n        watcher.close();\n      }\n    };\n  });\n\nexport class Config<TConfig extends object> {\n  public readonly config$: Observable<TConfig>;\n  public readonly configPath: string;\n  private readonly defaultConfig: TConfig;\n  // tslint:disable-next-line no-any\n  private readonly schema: any;\n  // tslint:disable-next-line no-any\n  private mutableValidateConfig: any;\n\n  public constructor({\n    name: configName,\n    defaultConfig,\n    schema,\n    configPath,\n  }: {\n    readonly name: string;\n    readonly defaultConfig: TConfig;\n    // tslint:disable-next-line no-any\n    readonly schema: any;\n    readonly configPath: string;\n  }) {\n    this.configPath = path.resolve(configPath, `${configName}.json`);\n    this.defaultConfig = defaultConfig;\n    this.schema = schema;\n    this.config$ = defer(async () => this.getConfig()).pipe(\n      switchMap((config) =>\n        watchConfig$(this.configPath).pipe(\n          mergeScan((prevConfig) => defer(async () => this.getConfig({ config: prevConfig })), config, 1),\n        ),\n      ),\n\n      distinctUntilChanged(),\n    );\n  }\n\n  public async update({ config }: { readonly config: TConfig }): Promise<TConfig> {\n    await this.validate(config);\n    await fs.ensureDir(path.dirname(this.configPath));\n    await fs.writeFile(this.configPath, JSON.stringify(config));\n\n    return config;\n  }\n\n  private async getConfig({ config = this.defaultConfig }: { readonly config?: TConfig } = {}): Promise<TConfig> {\n    let contents;\n    try {\n      contents = await fs.readFile(this.configPath, 'utf8');\n    } catch (error) {\n      if (error.code === 'ENOENT') {\n        return this.update({ config });\n      }\n\n      throw error;\n    }\n\n    const currentConfig = JSON.parse(contents);\n    await this.validate(currentConfig);\n\n    if (config !== undefined) {\n      // tslint:disable-next-line no-any\n      return (SeamlessImmutable as any).merge(config, currentConfig, {\n        deep: true,\n      });\n    }\n\n    return currentConfig;\n  }\n\n  private async validate(config: TConfig): Promise<void> {\n    const validateConfig = await this.getValidateConfig();\n    const isValid = validateConfig(config);\n    if (!isValid) {\n      const error = new Error('Invalid config');\n      // tslint:disable-next-line no-object-mutation no-any\n      (error as any).errors = validateConfig.errors;\n      throw error;\n    }\n  }\n\n  // Promise<import('ajv').ValidateFunction>\n  // tslint:disable-next-line no-any\n  private async getValidateConfig(): Promise<any> {\n    if (this.mutableValidateConfig !== undefined) {\n      return this.mutableValidateConfig;\n    }\n\n    const ajv = await import('ajv');\n    // tslint:disable-next-line no-any\n    const validateConfig = new ajv.default().compile(this.schema);\n    this.mutableValidateConfig = validateConfig;\n\n    return validateConfig;\n  }\n}\n"]}
|