UNPKG

13.9 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3const tslib_1 = require("tslib");
4const fs = tslib_1.__importStar(require("fs-extra"));
5const path = tslib_1.__importStar(require("path"));
6const rxjs_1 = require("rxjs");
7const operators_1 = require("rxjs/operators");
8const seamless_immutable_1 = tslib_1.__importDefault(require("seamless-immutable"));
9const 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});
39class 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}
96exports.Config = Config;
97
98//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["Config.ts"],"names":[],"mappings":";;;AAAA,qDAA+B;AAC/B,mDAA6B;AAC7B,+BAAmD;AACnD,8CAA4E;AAC5E,oFAAmD;AAInD,MAAM,YAAY,GAAG,CAAC,IAAY,EAAqB,EAAE,CACvD,iBAAU,CAAC,MAAM,CAAC,CAAC,QAA0B,EAAE,EAAE;IAC/C,+BAA+B;IAC/B,kCAAkC;IAClC,IAAI,OAAwB,CAAC;IAC7B,IAAI,MAAM,GAAG,KAAK,CAAC;IACnB,0DAAO,UAAU,IACd,IAAI,CAAC,CAAC,QAAQ,EAAE,EAAE;QACjB,IAAI,CAAC,MAAM,EAAE;YACX,OAAO,GAAG,QAAQ,CAAC,KAAK,CAAC,IAAI,EAAE,EAAE,aAAa,EAAE,KAAK,EAAE,CAAC,CAAC;YACzD,OAAO,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE;gBACrB,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC1B,CAAC,CAAC,CAAC;YACH,OAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,GAAG,EAAE;gBACxB,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC1B,CAAC,CAAC,CAAC;YACH,OAAO,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,KAAY,EAAE,EAAE;gBACnC,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YACxB,CAAC,CAAC,CAAC;YACH,OAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,GAAG,EAAE;gBACxB,QAAQ,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,6BAA6B,CAAC,CAAC,CAAC;YAC3D,CAAC,CAAC,CAAC;SACJ;IACH,CAAC,CAAC;SACD,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;IAE3C,OAAO,GAAG,EAAE;QACV,MAAM,GAAG,IAAI,CAAC;QACd,IAAI,OAAO,KAAK,SAAS,EAAE;YACzB,OAAO,CAAC,KAAK,EAAE,CAAC;SACjB;IACH,CAAC,CAAC;AACJ,CAAC,CAAC,CAAC;AAEL,MAAa,MAAM;IASjB,YAAmB,EACjB,IAAI,EAAE,UAAU,EAChB,aAAa,EACb,MAAM,EACN,UAAU,GAOX;QACC,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,GAAG,UAAU,OAAO,CAAC,CAAC;QACjE,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC;QACnC,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACrB,IAAI,CAAC,OAAO,GAAG,YAAK,CAAC,KAAK,IAAI,EAAE,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC,IAAI,CACrD,qBAAS,CAAC,CAAC,MAAM,EAAE,EAAE,CACnB,YAAY,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,IAAI,CAChC,qBAAS,CAAC,CAAC,UAAU,EAAE,EAAE,CAAC,YAAK,CAAC,KAAK,IAAI,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,MAAM,EAAE,UAAU,EAAE,CAAC,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC,CAChG,CACF,EAED,gCAAoB,EAAE,CACvB,CAAC;IACJ,CAAC;IAEM,KAAK,CAAC,MAAM,CAAC,EAAE,MAAM,EAAgC;QAC1D,MAAM,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;QAC5B,MAAM,EAAE,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC;QAClD,MAAM,EAAE,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC;QAE5D,OAAO,MAAM,CAAC;IAChB,CAAC;IAEO,KAAK,CAAC,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC,aAAa,KAAoC,EAAE;QACzF,IAAI,QAAQ,CAAC;QACb,IAAI;YACF,QAAQ,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,UAAU,EAAE,MAAM,CAAC,CAAC;SACvD;QAAC,OAAO,KAAK,EAAE;YACd,IAAI,KAAK,CAAC,IAAI,KAAK,QAAQ,EAAE;gBAC3B,OAAO,IAAI,CAAC,MAAM,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC;aAChC;YAED,MAAM,KAAK,CAAC;SACb;QAED,MAAM,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;QAC3C,MAAM,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC;QAEnC,IAAI,MAAM,KAAK,SAAS,EAAE;YACxB,kCAAkC;YAClC,OAAQ,4BAAyB,CAAC,KAAK,CAAC,MAAM,EAAE,aAAa,EAAE;gBAC7D,IAAI,EAAE,IAAI;aACX,CAAC,CAAC;SACJ;QAED,OAAO,aAAa,CAAC;IACvB,CAAC;IAEO,KAAK,CAAC,QAAQ,CAAC,MAAe;QACpC,MAAM,cAAc,GAAG,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACtD,MAAM,OAAO,GAAG,cAAc,CAAC,MAAM,CAAC,CAAC;QACvC,IAAI,CAAC,OAAO,EAAE;YACZ,MAAM,KAAK,GAAG,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAC;YAC1C,qDAAqD;YACpD,KAAa,CAAC,MAAM,GAAG,cAAc,CAAC,MAAM,CAAC;YAC9C,MAAM,KAAK,CAAC;SACb;IACH,CAAC;IAED,0CAA0C;IAC1C,kCAAkC;IAC1B,KAAK,CAAC,iBAAiB;QAC7B,IAAI,IAAI,CAAC,qBAAqB,KAAK,SAAS,EAAE;YAC5C,OAAO,IAAI,CAAC,qBAAqB,CAAC;SACnC;QAED,MAAM,GAAG,GAAG,gEAAa,KAAK,GAAC,CAAC;QAChC,kCAAkC;QAClC,MAAM,cAAc,GAAG,IAAI,GAAG,CAAC,OAAO,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC9D,IAAI,CAAC,qBAAqB,GAAG,cAAc,CAAC;QAE5C,OAAO,cAAc,CAAC;IACxB,CAAC;CACF;AA7FD,wBA6FC","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"]}