velocious 1.0.291 → 1.0.293

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,94 @@
1
+ /**
2
+ * Development-only file watcher that asks the HTTP server to recycle workers
3
+ * when application source files change.
4
+ */
5
+ export default class VelociousHttpServerDevelopmentReloader {
6
+ /**
7
+ * @param {object} args - Options object.
8
+ * @param {import("../configuration.js").default} args.configuration - Configuration instance.
9
+ * @param {function({changedPath: string}) : Promise<void>} args.onReload - Reload callback.
10
+ * @param {number} [args.debounceMs] - Debounce window for grouped changes.
11
+ * @param {typeof fsWatch} [args.watchFactory] - File watch factory.
12
+ * @param {typeof fs.readdir} [args.readdir] - Directory reader.
13
+ * @param {typeof fs.stat} [args.stat] - Stat reader.
14
+ */
15
+ constructor({ configuration, onReload, debounceMs, watchFactory, readdir, stat }: {
16
+ configuration: import("../configuration.js").default;
17
+ onReload: (arg0: {
18
+ changedPath: string;
19
+ }) => Promise<void>;
20
+ debounceMs?: number | undefined;
21
+ watchFactory?: typeof fsWatch | undefined;
22
+ readdir?: typeof fs.readdir | undefined;
23
+ stat?: typeof fs.stat | undefined;
24
+ });
25
+ configuration: import("../configuration.js").default;
26
+ debounceMs: number;
27
+ logger: Logger;
28
+ onReload: (arg0: {
29
+ changedPath: string;
30
+ }) => Promise<void>;
31
+ readdir: typeof fs.readdir;
32
+ stat: typeof fs.stat;
33
+ watchFactory: typeof fsWatch;
34
+ /** @type {ReturnType<typeof setTimeout> | undefined} */
35
+ reloadTimer: ReturnType<typeof setTimeout> | undefined;
36
+ /** @type {string | undefined} */
37
+ pendingChangedPath: string | undefined;
38
+ /** @type {Map<string, import("fs").FSWatcher>} */
39
+ watchers: Map<string, import("fs").FSWatcher>;
40
+ /** @returns {Promise<void>} - Resolves when watching has started. */
41
+ start(): Promise<void>;
42
+ /** @returns {string[]} - Source directories to watch. */
43
+ watchRootPaths(): string[];
44
+ /**
45
+ * @param {string} directoryPath - Directory path.
46
+ * @returns {Promise<void>} - Resolves when child directories are watched.
47
+ */
48
+ watchDirectoryRecursive(directoryPath: string): Promise<void>;
49
+ /**
50
+ * @param {object} args - Options object.
51
+ * @param {string} args.directoryPath - Watched directory path.
52
+ * @param {string} args.eventType - Watch event type.
53
+ * @param {string | Buffer | null} args.fileName - Relative changed filename.
54
+ * @returns {Promise<void>} - Resolves when complete.
55
+ */
56
+ onWatcherEvent({ directoryPath, eventType, fileName }: {
57
+ directoryPath: string;
58
+ eventType: string;
59
+ fileName: string | Buffer | null;
60
+ }): Promise<void>;
61
+ /**
62
+ * @param {object} args - Options object.
63
+ * @param {string} args.changedPath - Changed path.
64
+ * @param {string | Buffer | null} args.fileName - Raw filename from fs.watch.
65
+ * @returns {boolean} - Whether the path should trigger reload.
66
+ */
67
+ shouldReloadPath({ changedPath, fileName }: {
68
+ changedPath: string;
69
+ fileName: string | Buffer | null;
70
+ }): boolean;
71
+ /**
72
+ * @param {string} changedPath - Candidate directory path.
73
+ * @returns {Promise<void>} - Resolves when any new directory watchers are added.
74
+ */
75
+ watchPotentialDirectory(changedPath: string): Promise<void>;
76
+ /**
77
+ * @param {string} changedPath - Changed path.
78
+ * @returns {void} - No return value.
79
+ */
80
+ scheduleReload(changedPath: string): void;
81
+ /** @returns {Promise<void>} - Resolves when the queued reload is handled. */
82
+ flushReload(): Promise<void>;
83
+ /**
84
+ * @param {Error} error - Watcher error.
85
+ * @returns {void} - No return value.
86
+ */
87
+ onWatcherError: (error: Error) => void;
88
+ /** @returns {Promise<void>} - Resolves when watchers are closed. */
89
+ stop(): Promise<void>;
90
+ }
91
+ import Logger from "../logger.js";
92
+ import fs from "fs/promises";
93
+ import { watch as fsWatch } from "fs";
94
+ //# sourceMappingURL=development-reloader.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"development-reloader.d.ts","sourceRoot":"","sources":["../../../src/http-server/development-reloader.js"],"names":[],"mappings":"AAeA;;;GAGG;AACH;IACE;;;;;;;;OAQG;IACH,kFAPG;QAAoD,aAAa,EAAzD,OAAO,qBAAqB,EAAE,OAAO;QACiB,QAAQ,EAA9D,CAAS,IAAqB,EAArB;YAAC,WAAW,EAAE,MAAM,CAAA;SAAC,KAAI,OAAO,CAAC,IAAI,CAAC;QACjC,UAAU;QACF,YAAY;QACT,OAAO;QACV,IAAI;KACpC,EAkBA;IAhBC,qDAAkC;IAClC,mBAA4B;IAC5B,eAAgE;IAChE,iBAVkB;QAAC,WAAW,EAAE,MAAM,CAAA;KAAC,KAAI,OAAO,CAAC,IAAI,CAAC,CAUhC;IACxB,2BAAsB;IACtB,qBAAgB;IAChB,6BAAgC;IAEhC,wDAAwD;IACxD,aADW,UAAU,CAAC,OAAO,UAAU,CAAC,GAAG,SAAS,CACxB;IAE5B,iCAAiC;IACjC,oBADW,MAAM,GAAG,SAAS,CACM;IAEnC,kDAAkD;IAClD,UADW,GAAG,CAAC,MAAM,EAAE,OAAO,IAAI,EAAE,SAAS,CAAC,CACrB;IAG3B,qEAAqE;IACrE,SADc,OAAO,CAAC,IAAI,CAAC,CAK1B;IAED,yDAAyD;IACzD,kBADc,MAAM,EAAE,CAarB;IAED;;;OAGG;IACH,uCAHW,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAgCzB;IAED;;;;;;OAMG;IACH,uDALG;QAAqB,aAAa,EAA1B,MAAM;QACO,SAAS,EAAtB,MAAM;QACuB,QAAQ,EAArC,MAAM,GAAG,MAAM,GAAG,IAAI;KAC9B,GAAU,OAAO,CAAC,IAAI,CAAC,CAczB;IAED;;;;;OAKG;IACH,4CAJG;QAAqB,WAAW,EAAxB,MAAM;QACuB,QAAQ,EAArC,MAAM,GAAG,MAAM,GAAG,IAAI;KAC9B,GAAU,OAAO,CAQnB;IAED;;;OAGG;IACH,qCAHW,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAczB;IAED;;;OAGG;IACH,4BAHW,MAAM,GACJ,IAAI,CAYhB;IAED,6EAA6E;IAC7E,eADc,OAAO,CAAC,IAAI,CAAC,CAU1B;IAED;;;OAGG;IACH,iBAAkB,OAHP,KAGY,KAFV,IAAI,CAIhB;IAED,oEAAoE;IACpE,QADc,OAAO,CAAC,IAAI,CAAC,CAc1B;CACF;mBA9MkB,cAAc;eADlB,aAAa;iCADG,IAAI"}
@@ -0,0 +1,180 @@
1
+ // @ts-check
2
+ import { watch as fsWatch } from "fs";
3
+ import fs from "fs/promises";
4
+ import Logger from "../logger.js";
5
+ import path from "path";
6
+ const RELOADABLE_EXTENSIONS = new Set([
7
+ ".cjs",
8
+ ".ejs",
9
+ ".js",
10
+ ".json",
11
+ ".mjs"
12
+ ]);
13
+ /**
14
+ * Development-only file watcher that asks the HTTP server to recycle workers
15
+ * when application source files change.
16
+ */
17
+ export default class VelociousHttpServerDevelopmentReloader {
18
+ /**
19
+ * @param {object} args - Options object.
20
+ * @param {import("../configuration.js").default} args.configuration - Configuration instance.
21
+ * @param {function({changedPath: string}) : Promise<void>} args.onReload - Reload callback.
22
+ * @param {number} [args.debounceMs] - Debounce window for grouped changes.
23
+ * @param {typeof fsWatch} [args.watchFactory] - File watch factory.
24
+ * @param {typeof fs.readdir} [args.readdir] - Directory reader.
25
+ * @param {typeof fs.stat} [args.stat] - Stat reader.
26
+ */
27
+ constructor({ configuration, onReload, debounceMs = 75, watchFactory = fsWatch, readdir = fs.readdir, stat = fs.stat }) {
28
+ this.configuration = configuration;
29
+ this.debounceMs = debounceMs;
30
+ this.logger = new Logger("DevelopmentReloader", { configuration });
31
+ this.onReload = onReload;
32
+ this.readdir = readdir;
33
+ this.stat = stat;
34
+ this.watchFactory = watchFactory;
35
+ /** @type {ReturnType<typeof setTimeout> | undefined} */
36
+ this.reloadTimer = undefined;
37
+ /** @type {string | undefined} */
38
+ this.pendingChangedPath = undefined;
39
+ /** @type {Map<string, import("fs").FSWatcher>} */
40
+ this.watchers = new Map();
41
+ }
42
+ /** @returns {Promise<void>} - Resolves when watching has started. */
43
+ async start() {
44
+ for (const rootPath of this.watchRootPaths()) {
45
+ await this.watchDirectoryRecursive(rootPath);
46
+ }
47
+ }
48
+ /** @returns {string[]} - Source directories to watch. */
49
+ watchRootPaths() {
50
+ const rootPaths = new Set();
51
+ const configurationDirectory = this.configuration.getDirectory();
52
+ rootPaths.add(path.join(configurationDirectory, "src"));
53
+ for (const backendProject of this.configuration.getBackendProjects()) {
54
+ if (!backendProject?.path)
55
+ continue;
56
+ rootPaths.add(path.join(backendProject.path, "src"));
57
+ }
58
+ return Array.from(rootPaths);
59
+ }
60
+ /**
61
+ * @param {string} directoryPath - Directory path.
62
+ * @returns {Promise<void>} - Resolves when child directories are watched.
63
+ */
64
+ async watchDirectoryRecursive(directoryPath) {
65
+ const resolvedDirectoryPath = path.resolve(directoryPath);
66
+ if (this.watchers.has(resolvedDirectoryPath))
67
+ return;
68
+ let entries;
69
+ try {
70
+ entries = await this.readdir(resolvedDirectoryPath, { withFileTypes: true });
71
+ }
72
+ catch (error) {
73
+ if ( /** @type {{code?: string}} */(error)?.code === "ENOENT")
74
+ return;
75
+ throw error;
76
+ }
77
+ const watcher = this.watchFactory(resolvedDirectoryPath, (eventType, fileName) => {
78
+ void this.onWatcherEvent({
79
+ directoryPath: resolvedDirectoryPath,
80
+ eventType,
81
+ fileName
82
+ });
83
+ });
84
+ watcher.on("error", this.onWatcherError);
85
+ this.watchers.set(resolvedDirectoryPath, watcher);
86
+ for (const entry of entries) {
87
+ if (!entry.isDirectory())
88
+ continue;
89
+ await this.watchDirectoryRecursive(path.join(resolvedDirectoryPath, entry.name));
90
+ }
91
+ }
92
+ /**
93
+ * @param {object} args - Options object.
94
+ * @param {string} args.directoryPath - Watched directory path.
95
+ * @param {string} args.eventType - Watch event type.
96
+ * @param {string | Buffer | null} args.fileName - Relative changed filename.
97
+ * @returns {Promise<void>} - Resolves when complete.
98
+ */
99
+ async onWatcherEvent({ directoryPath, eventType, fileName }) {
100
+ const changedPath = fileName
101
+ ? path.join(directoryPath, fileName.toString())
102
+ : directoryPath;
103
+ await this.watchPotentialDirectory(changedPath);
104
+ if (!this.shouldReloadPath({ changedPath, fileName }))
105
+ return;
106
+ this.scheduleReload(changedPath);
107
+ await this.logger.debug(() => ["Queued development hot reload", { changedPath, eventType }]);
108
+ }
109
+ /**
110
+ * @param {object} args - Options object.
111
+ * @param {string} args.changedPath - Changed path.
112
+ * @param {string | Buffer | null} args.fileName - Raw filename from fs.watch.
113
+ * @returns {boolean} - Whether the path should trigger reload.
114
+ */
115
+ shouldReloadPath({ changedPath, fileName }) {
116
+ if (!fileName)
117
+ return true;
118
+ const extension = path.extname(changedPath).toLowerCase();
119
+ return RELOADABLE_EXTENSIONS.has(extension);
120
+ }
121
+ /**
122
+ * @param {string} changedPath - Candidate directory path.
123
+ * @returns {Promise<void>} - Resolves when any new directory watchers are added.
124
+ */
125
+ async watchPotentialDirectory(changedPath) {
126
+ try {
127
+ const stat = await this.stat(changedPath);
128
+ if (stat.isDirectory()) {
129
+ await this.watchDirectoryRecursive(changedPath);
130
+ }
131
+ }
132
+ catch (error) {
133
+ if ( /** @type {{code?: string}} */(error)?.code !== "ENOENT") {
134
+ throw error;
135
+ }
136
+ }
137
+ }
138
+ /**
139
+ * @param {string} changedPath - Changed path.
140
+ * @returns {void} - No return value.
141
+ */
142
+ scheduleReload(changedPath) {
143
+ this.pendingChangedPath = changedPath;
144
+ if (this.reloadTimer) {
145
+ clearTimeout(this.reloadTimer);
146
+ }
147
+ this.reloadTimer = setTimeout(() => {
148
+ void this.flushReload();
149
+ }, this.debounceMs);
150
+ }
151
+ /** @returns {Promise<void>} - Resolves when the queued reload is handled. */
152
+ async flushReload() {
153
+ this.reloadTimer = undefined;
154
+ const changedPath = this.pendingChangedPath;
155
+ if (!changedPath)
156
+ return;
157
+ this.pendingChangedPath = undefined;
158
+ await this.onReload({ changedPath });
159
+ }
160
+ /**
161
+ * @param {Error} error - Watcher error.
162
+ * @returns {void} - No return value.
163
+ */
164
+ onWatcherError = (error) => {
165
+ void this.logger.warn("Development hot reload watcher error", error);
166
+ };
167
+ /** @returns {Promise<void>} - Resolves when watchers are closed. */
168
+ async stop() {
169
+ if (this.reloadTimer) {
170
+ clearTimeout(this.reloadTimer);
171
+ this.reloadTimer = undefined;
172
+ }
173
+ this.pendingChangedPath = undefined;
174
+ for (const watcher of this.watchers.values()) {
175
+ watcher.close();
176
+ }
177
+ this.watchers.clear();
178
+ }
179
+ }
180
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"development-reloader.js","sourceRoot":"","sources":["../../../src/http-server/development-reloader.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport {watch as fsWatch} from \"fs\"\nimport fs from \"fs/promises\"\nimport Logger from \"../logger.js\"\nimport path from \"path\"\n\nconst RELOADABLE_EXTENSIONS = new Set([\n  \".cjs\",\n  \".ejs\",\n  \".js\",\n  \".json\",\n  \".mjs\"\n])\n\n/**\n * Development-only file watcher that asks the HTTP server to recycle workers\n * when application source files change.\n */\nexport default class VelociousHttpServerDevelopmentReloader {\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {function({changedPath: string}) : Promise<void>} args.onReload - Reload callback.\n   * @param {number} [args.debounceMs] - Debounce window for grouped changes.\n   * @param {typeof fsWatch} [args.watchFactory] - File watch factory.\n   * @param {typeof fs.readdir} [args.readdir] - Directory reader.\n   * @param {typeof fs.stat} [args.stat] - Stat reader.\n   */\n  constructor({configuration, onReload, debounceMs = 75, watchFactory = fsWatch, readdir = fs.readdir, stat = fs.stat}) {\n    this.configuration = configuration\n    this.debounceMs = debounceMs\n    this.logger = new Logger(\"DevelopmentReloader\", {configuration})\n    this.onReload = onReload\n    this.readdir = readdir\n    this.stat = stat\n    this.watchFactory = watchFactory\n\n    /** @type {ReturnType<typeof setTimeout> | undefined} */\n    this.reloadTimer = undefined\n\n    /** @type {string | undefined} */\n    this.pendingChangedPath = undefined\n\n    /** @type {Map<string, import(\"fs\").FSWatcher>} */\n    this.watchers = new Map()\n  }\n\n  /** @returns {Promise<void>} - Resolves when watching has started. */\n  async start() {\n    for (const rootPath of this.watchRootPaths()) {\n      await this.watchDirectoryRecursive(rootPath)\n    }\n  }\n\n  /** @returns {string[]} - Source directories to watch. */\n  watchRootPaths() {\n    const rootPaths = new Set()\n    const configurationDirectory = this.configuration.getDirectory()\n\n    rootPaths.add(path.join(configurationDirectory, \"src\"))\n\n    for (const backendProject of this.configuration.getBackendProjects()) {\n      if (!backendProject?.path) continue\n      rootPaths.add(path.join(backendProject.path, \"src\"))\n    }\n\n    return Array.from(rootPaths)\n  }\n\n  /**\n   * @param {string} directoryPath - Directory path.\n   * @returns {Promise<void>} - Resolves when child directories are watched.\n   */\n  async watchDirectoryRecursive(directoryPath) {\n    const resolvedDirectoryPath = path.resolve(directoryPath)\n\n    if (this.watchers.has(resolvedDirectoryPath)) return\n\n    let entries\n\n    try {\n      entries = await this.readdir(resolvedDirectoryPath, {withFileTypes: true})\n    } catch (error) {\n      if (/** @type {{code?: string}} */ (error)?.code === \"ENOENT\") return\n      throw error\n    }\n\n    const watcher = this.watchFactory(resolvedDirectoryPath, (eventType, fileName) => {\n      void this.onWatcherEvent({\n        directoryPath: resolvedDirectoryPath,\n        eventType,\n        fileName\n      })\n    })\n\n    watcher.on(\"error\", this.onWatcherError)\n    this.watchers.set(resolvedDirectoryPath, watcher)\n\n    for (const entry of entries) {\n      if (!entry.isDirectory()) continue\n\n      await this.watchDirectoryRecursive(path.join(resolvedDirectoryPath, entry.name))\n    }\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string} args.directoryPath - Watched directory path.\n   * @param {string} args.eventType - Watch event type.\n   * @param {string | Buffer | null} args.fileName - Relative changed filename.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async onWatcherEvent({directoryPath, eventType, fileName}) {\n    const changedPath = fileName\n      ? path.join(directoryPath, fileName.toString())\n      : directoryPath\n\n    await this.watchPotentialDirectory(changedPath)\n\n    if (!this.shouldReloadPath({changedPath, fileName})) return\n\n    this.scheduleReload(changedPath)\n\n    await this.logger.debug(() => [\"Queued development hot reload\", {changedPath, eventType}])\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string} args.changedPath - Changed path.\n   * @param {string | Buffer | null} args.fileName - Raw filename from fs.watch.\n   * @returns {boolean} - Whether the path should trigger reload.\n   */\n  shouldReloadPath({changedPath, fileName}) {\n    if (!fileName) return true\n\n    const extension = path.extname(changedPath).toLowerCase()\n\n    return RELOADABLE_EXTENSIONS.has(extension)\n  }\n\n  /**\n   * @param {string} changedPath - Candidate directory path.\n   * @returns {Promise<void>} - Resolves when any new directory watchers are added.\n   */\n  async watchPotentialDirectory(changedPath) {\n    try {\n      const stat = await this.stat(changedPath)\n\n      if (stat.isDirectory()) {\n        await this.watchDirectoryRecursive(changedPath)\n      }\n    } catch (error) {\n      if (/** @type {{code?: string}} */ (error)?.code !== \"ENOENT\") {\n        throw error\n      }\n    }\n  }\n\n  /**\n   * @param {string} changedPath - Changed path.\n   * @returns {void} - No return value.\n   */\n  scheduleReload(changedPath) {\n    this.pendingChangedPath = changedPath\n\n    if (this.reloadTimer) {\n      clearTimeout(this.reloadTimer)\n    }\n\n    this.reloadTimer = setTimeout(() => {\n      void this.flushReload()\n    }, this.debounceMs)\n  }\n\n  /** @returns {Promise<void>} - Resolves when the queued reload is handled. */\n  async flushReload() {\n    this.reloadTimer = undefined\n\n    const changedPath = this.pendingChangedPath\n\n    if (!changedPath) return\n\n    this.pendingChangedPath = undefined\n    await this.onReload({changedPath})\n  }\n\n  /**\n   * @param {Error} error - Watcher error.\n   * @returns {void} - No return value.\n   */\n  onWatcherError = (error) => {\n    void this.logger.warn(\"Development hot reload watcher error\", error)\n  }\n\n  /** @returns {Promise<void>} - Resolves when watchers are closed. */\n  async stop() {\n    if (this.reloadTimer) {\n      clearTimeout(this.reloadTimer)\n      this.reloadTimer = undefined\n    }\n\n    this.pendingChangedPath = undefined\n\n    for (const watcher of this.watchers.values()) {\n      watcher.close()\n    }\n\n    this.watchers.clear()\n  }\n}\n"]}
@@ -6,13 +6,23 @@ export default class VelociousHttpServer {
6
6
  * @param {boolean} [args.inProcess] - Run HTTP handlers in the main thread instead of worker threads.
7
7
  * @param {number} [args.port] - Port.
8
8
  * @param {number} [args.maxWorkers] - Max workers.
9
+ * @param {function({configuration: import("../configuration.js").default, onReload: function({changedPath: string}) : Promise<void>}) : {start: () => Promise<void>, stop: () => Promise<void>}} [args.developmentReloaderFactory] - Development reloader factory.
9
10
  */
10
- constructor({ configuration, host, inProcess, maxWorkers, port }: {
11
+ constructor({ configuration, developmentReloaderFactory, host, inProcess, maxWorkers, port }: {
11
12
  configuration: import("../configuration.js").default;
12
13
  host?: string | undefined;
13
14
  inProcess?: boolean | undefined;
14
15
  port?: number | undefined;
15
16
  maxWorkers?: number | undefined;
17
+ developmentReloaderFactory?: ((arg0: {
18
+ configuration: import("../configuration.js").default;
19
+ onReload: (arg0: {
20
+ changedPath: string;
21
+ }) => Promise<void>;
22
+ }) => {
23
+ start: () => Promise<void>;
24
+ stop: () => Promise<void>;
25
+ }) | undefined;
16
26
  });
17
27
  clientCount: number;
18
28
  /** @type {Record<string, ServerClient>} */
@@ -22,6 +32,15 @@ export default class VelociousHttpServer {
22
32
  /** @type {Array<WorkerHandler | InProcessHandler>} */
23
33
  workerHandlers: Array<WorkerHandler | InProcessHandler>;
24
34
  configuration: import("../configuration.js").default;
35
+ developmentReloaderFactory: ((arg0: {
36
+ configuration: import("../configuration.js").default;
37
+ onReload: (arg0: {
38
+ changedPath: string;
39
+ }) => Promise<void>;
40
+ }) => {
41
+ start: () => Promise<void>;
42
+ stop: () => Promise<void>;
43
+ }) | undefined;
25
44
  inProcess: boolean;
26
45
  logger: Logger;
27
46
  host: string;
@@ -42,6 +61,11 @@ export default class VelociousHttpServer {
42
61
  stopServer(): Promise<void>;
43
62
  /** @returns {Promise<void>} - Resolves when complete. */
44
63
  stop(): Promise<void>;
64
+ _stopping: boolean | undefined;
65
+ developmentReloader: {
66
+ start: () => Promise<void>;
67
+ stop: () => Promise<void>;
68
+ } | undefined;
45
69
  /** @returns {void} - No return value. */
46
70
  onClose: () => void;
47
71
  /**
@@ -61,8 +85,18 @@ export default class VelociousHttpServer {
61
85
  onClientClose: (client: ServerClient) => void;
62
86
  /** @returns {Promise<void>} - Resolves when complete. */
63
87
  spawnWorker(): Promise<void>;
88
+ /** @returns {Promise<WorkerHandler | InProcessHandler>} - Started worker handler. */
89
+ _buildWorkerHandler(): Promise<WorkerHandler | InProcessHandler>;
64
90
  /** @returns {WorkerHandler | InProcessHandler} - The worker handler to use. */
65
91
  workerHandlerToUse(): WorkerHandler | InProcessHandler;
92
+ /** @returns {boolean} - Whether development worker hot reload should run. */
93
+ shouldUseDevelopmentHotReload(): boolean;
94
+ /** @returns {Promise<void>} - Resolves when watcher setup finishes. */
95
+ _startDevelopmentReloader(): Promise<void>;
96
+ /** @returns {Promise<void>} - Resolves when workers have been refreshed. */
97
+ reloadWorkersForDevelopment(): Promise<void>;
98
+ _reloadWorkersForDevelopmentQueued: boolean | undefined;
99
+ _reloadingWorkersForDevelopment: boolean | undefined;
66
100
  }
67
101
  import ServerClient from "./server-client.js";
68
102
  import WorkerHandler from "./worker-handler/index.js";
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/http-server/index.js"],"names":[],"mappings":"AAUA;IAYE;;;;;;;OAOG;IACH,kEANG;QAAoD,aAAa,EAAzD,OAAO,qBAAqB,EAAE,OAAO;QACvB,IAAI;QACH,SAAS;QACV,IAAI;QACJ,UAAU;KAClC,EAQA;IA1BD,oBAAe;IAEf,4CAA4C;IAC5C,SADW,MAAM,CAAC,MAAM,EAAE,YAAY,CAAC,CAC3B;IAEZ,mEAA2B;IAC3B,oBAAe;IAEf,sDAAsD;IACtD,gBADW,KAAK,CAAC,aAAa,GAAG,gBAAgB,CAAC,CAC/B;IAWjB,qDAAkC;IAClC,mBAAmC;IACnC,eAA8B;IAC9B,aAA6B;IAC7B,aAAwB;IACxB,mBAAkC;IAGpC,0DAA0D;IAC1D,SADc,OAAO,CAAC,IAAI,CAAC,CAQ1B;IALC,kCAAiC;IAOnC,0DAA0D;IAC1D,oBADc,OAAO,CAAC,IAAI,CAAC,CAc1B;IAED,0DAA0D;IAC1D,2BADc,OAAO,CAAC,IAAI,CAAC,CAK1B;IAED,4CAA4C;IAC5C,YADc,OAAO,CAOpB;IAED,0DAA0D;IAC1D,eADc,OAAO,CAAC,IAAI,CAAC,CAW1B;IAED,0DAA0D;IAC1D,cADc,OAAO,CAAC,IAAI,CAAC,CAgB1B;IAED,0DAA0D;IAC1D,QADc,OAAO,CAAC,IAAI,CAAC,CAQ1B;IAED,0CAA0C;IAC1C,eADc,IAAI,CAGjB;IAED;;;OAGG;IACH,gBAAiB,OAHN,KAGW,KAFT,IAAI,CAIhB;IAED;;;OAGG;IACH,eAAgB,QAHL,OAAO,KAAK,EAAE,MAGH,KAFT,IAAI,CA8BhB;IAED;;;OAGG;IACH,gBAAiB,QAHN,YAGY,KAFV,IAAI,CAahB;IAED,0DAA0D;IAC1D,eADc,OAAO,CAAC,IAAI,CAAC,CAc1B;IAED,+EAA+E;IAC/E,sBADc,aAAa,GAAG,gBAAgB,CAY7C;CACF;yBAhNwB,oBAAoB;0BACnB,2BAA2B;6BAJxB,gCAAgC;mBAC1C,cAAc;gBACjB,KAAK"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/http-server/index.js"],"names":[],"mappings":"AAWA;IAYE;;;;;;;;OAQG;IACH,8FAPG;QAAoD,aAAa,EAAzD,OAAO,qBAAqB,EAAE,OAAO;QACvB,IAAI;QACH,SAAS;QACV,IAAI;QACJ,UAAU;QACqK,0BAA0B,WAA9M;YAAC,aAAa,EAAE,OAAO,qBAAqB,EAAE,OAAO,CAAC;YAAC,QAAQ,EAAE,CAAS,IAAqB,EAArB;gBAAC,WAAW,EAAE,MAAM,CAAA;aAAC,KAAI,OAAO,CAAC,IAAI,CAAC,CAAA;SAAC,KAAI;YAAC,KAAK,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;YAAC,IAAI,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAA;SAAC;KAC/L,EASA;IA5BD,oBAAe;IAEf,4CAA4C;IAC5C,SADW,MAAM,CAAC,MAAM,EAAE,YAAY,CAAC,CAC3B;IAEZ,mEAA2B;IAC3B,oBAAe;IAEf,sDAAsD;IACtD,gBADW,KAAK,CAAC,aAAa,GAAG,gBAAgB,CAAC,CAC/B;IAYjB,qDAAkC;IAClC,oCAJkB;QAAC,aAAa,EAAE,OAAO,qBAAqB,EAAE,OAAO,CAAC;QAAC,QAAQ,EAAE,CAAS,IAAqB,EAArB;YAAC,WAAW,EAAE,MAAM,CAAA;SAAC,KAAI,OAAO,CAAC,IAAI,CAAC,CAAA;KAAC,KAAI;QAAC,KAAK,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;QAAC,IAAI,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAA;KAAC,cAIlI;IAC5D,mBAAmC;IACnC,eAA8B;IAC9B,aAA6B;IAC7B,aAAwB;IACxB,mBAAkC;IAGpC,0DAA0D;IAC1D,SADc,OAAO,CAAC,IAAI,CAAC,CAS1B;IALC,kCAAiC;IAOnC,0DAA0D;IAC1D,oBADc,OAAO,CAAC,IAAI,CAAC,CAc1B;IAED,0DAA0D;IAC1D,2BADc,OAAO,CAAC,IAAI,CAAC,CAK1B;IAED,4CAA4C;IAC5C,YADc,OAAO,CAOpB;IAED,0DAA0D;IAC1D,eADc,OAAO,CAAC,IAAI,CAAC,CAW1B;IAED,0DAA0D;IAC1D,cADc,OAAO,CAAC,IAAI,CAAC,CAgB1B;IAED,0DAA0D;IAC1D,QADc,OAAO,CAAC,IAAI,CAAC,CAW1B;IATC,+BAAqB;IAErB;eA1F+I,MAAM,OAAO,CAAC,IAAI,CAAC;cAAQ,MAAM,OAAO,CAAC,IAAI,CAAC;kBA0FzJ;IAStC,0CAA0C;IAC1C,eADc,IAAI,CAGjB;IAED;;;OAGG;IACH,gBAAiB,OAHN,KAGW,KAFT,IAAI,CAIhB;IAED;;;OAGG;IACH,eAAgB,QAHL,OAAO,KAAK,EAAE,MAGH,KAFT,IAAI,CA8BhB;IAED;;;OAGG;IACH,gBAAiB,QAHN,YAGY,KAFV,IAAI,CAahB;IAED,0DAA0D;IAC1D,eADc,OAAO,CAAC,IAAI,CAAC,CAK1B;IAED,qFAAqF;IACrF,uBADc,OAAO,CAAC,aAAa,GAAG,gBAAgB,CAAC,CAetD;IAED,+EAA+E;IAC/E,sBADc,aAAa,GAAG,gBAAgB,CAY7C;IAED,6EAA6E;IAC7E,iCADc,OAAO,CAGpB;IAED,uEAAuE;IACvE,6BADc,OAAO,CAAC,IAAI,CAAC,CAiB1B;IAED,4EAA4E;IAC5E,+BADc,OAAO,CAAC,IAAI,CAAC,CAyB1B;IApBG,wDAA8C;IAIhD,qDAA2C;CAiB9C;yBAjRwB,oBAAoB;0BACnB,2BAA2B;6BAJxB,gCAAgC;mBAC1C,cAAc;gBACjB,KAAK"}
@@ -1,5 +1,6 @@
1
1
  // @ts-check
2
2
  import { digg } from "diggerize";
3
+ import DevelopmentReloader from "./development-reloader.js";
3
4
  import EventEmitter from "../utils/event-emitter.js";
4
5
  import InProcessHandler from "./worker-handler/in-process.js";
5
6
  import Logger from "../logger.js";
@@ -21,9 +22,11 @@ export default class VelociousHttpServer {
21
22
  * @param {boolean} [args.inProcess] - Run HTTP handlers in the main thread instead of worker threads.
22
23
  * @param {number} [args.port] - Port.
23
24
  * @param {number} [args.maxWorkers] - Max workers.
25
+ * @param {function({configuration: import("../configuration.js").default, onReload: function({changedPath: string}) : Promise<void>}) : {start: () => Promise<void>, stop: () => Promise<void>}} [args.developmentReloaderFactory] - Development reloader factory.
24
26
  */
25
- constructor({ configuration, host, inProcess, maxWorkers, port }) {
27
+ constructor({ configuration, developmentReloaderFactory, host, inProcess, maxWorkers, port }) {
26
28
  this.configuration = configuration;
29
+ this.developmentReloaderFactory = developmentReloaderFactory;
27
30
  this.inProcess = inProcess || false;
28
31
  this.logger = new Logger(this);
29
32
  this.host = host || "0.0.0.0";
@@ -33,6 +36,7 @@ export default class VelociousHttpServer {
33
36
  /** @returns {Promise<void>} - Resolves when complete. */
34
37
  async start() {
35
38
  await this._ensureAtLeastOneWorker();
39
+ await this._startDevelopmentReloader();
36
40
  this.netServer = new Net.Server();
37
41
  this.netServer.on("close", this.onClose);
38
42
  this.netServer.on("connection", this.onConnection);
@@ -96,6 +100,9 @@ export default class VelociousHttpServer {
96
100
  }
97
101
  /** @returns {Promise<void>} - Resolves when complete. */
98
102
  async stop() {
103
+ this._stopping = true;
104
+ await this.developmentReloader?.stop();
105
+ this.developmentReloader = undefined;
99
106
  await this.stopClients();
100
107
  await this.stopServer();
101
108
  const stopTasks = this.workerHandlers.map((handler) => handler.stop());
@@ -158,6 +165,11 @@ export default class VelociousHttpServer {
158
165
  };
159
166
  /** @returns {Promise<void>} - Resolves when complete. */
160
167
  async spawnWorker() {
168
+ const workerHandler = await this._buildWorkerHandler();
169
+ this.workerHandlers.push(workerHandler);
170
+ }
171
+ /** @returns {Promise<WorkerHandler | InProcessHandler>} - Started worker handler. */
172
+ async _buildWorkerHandler() {
161
173
  const workerCount = this.workerCount;
162
174
  this.workerCount++;
163
175
  const Handler = this.inProcess ? InProcessHandler : WorkerHandler;
@@ -166,7 +178,7 @@ export default class VelociousHttpServer {
166
178
  workerCount
167
179
  });
168
180
  await workerHandler.start();
169
- this.workerHandlers.push(workerHandler);
181
+ return workerHandler;
170
182
  }
171
183
  /** @returns {WorkerHandler | InProcessHandler} - The worker handler to use. */
172
184
  workerHandlerToUse() {
@@ -178,5 +190,48 @@ export default class VelociousHttpServer {
178
190
  }
179
191
  return workerHandler;
180
192
  }
193
+ /** @returns {boolean} - Whether development worker hot reload should run. */
194
+ shouldUseDevelopmentHotReload() {
195
+ return !this.inProcess && this.configuration.getEnvironment() === "development";
196
+ }
197
+ /** @returns {Promise<void>} - Resolves when watcher setup finishes. */
198
+ async _startDevelopmentReloader() {
199
+ if (!this.shouldUseDevelopmentHotReload())
200
+ return;
201
+ if (this.developmentReloader)
202
+ return;
203
+ const createDevelopmentReloader = this.developmentReloaderFactory
204
+ || ((args) => new DevelopmentReloader(args));
205
+ this.developmentReloader = createDevelopmentReloader({
206
+ configuration: this.configuration,
207
+ onReload: async ({ changedPath }) => {
208
+ await this.logger.info(`Development hot reload detected change in ${changedPath}`);
209
+ await this.reloadWorkersForDevelopment();
210
+ }
211
+ });
212
+ await this.developmentReloader.start();
213
+ }
214
+ /** @returns {Promise<void>} - Resolves when workers have been refreshed. */
215
+ async reloadWorkersForDevelopment() {
216
+ if (this._stopping)
217
+ return;
218
+ if (this._reloadingWorkersForDevelopment) {
219
+ this._reloadWorkersForDevelopmentQueued = true;
220
+ return;
221
+ }
222
+ this._reloadingWorkersForDevelopment = true;
223
+ try {
224
+ do {
225
+ this._reloadWorkersForDevelopmentQueued = false;
226
+ const oldWorkerHandlers = [...this.workerHandlers];
227
+ const newWorkerHandler = await this._buildWorkerHandler();
228
+ this.workerHandlers = [newWorkerHandler];
229
+ await Promise.all(oldWorkerHandlers.map((workerHandler) => workerHandler.stop()));
230
+ } while (this._reloadWorkersForDevelopmentQueued && !this._stopping);
231
+ }
232
+ finally {
233
+ this._reloadingWorkersForDevelopment = false;
234
+ }
235
+ }
181
236
  }
182
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/http-server/index.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport {digg} from \"diggerize\"\nimport EventEmitter from \"../utils/event-emitter.js\"\nimport InProcessHandler from \"./worker-handler/in-process.js\"\nimport Logger from \"../logger.js\"\nimport Net from \"net\"\nimport ServerClient from \"./server-client.js\"\nimport WorkerHandler from \"./worker-handler/index.js\"\n\nexport default class VelociousHttpServer {\n  clientCount = 0\n\n  /** @type {Record<string, ServerClient>}  */\n  clients = {}\n\n  events = new EventEmitter()\n  workerCount = 0\n\n  /** @type {Array<WorkerHandler | InProcessHandler>} */\n  workerHandlers = []\n\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {string} [args.host] - Host.\n   * @param {boolean} [args.inProcess] - Run HTTP handlers in the main thread instead of worker threads.\n   * @param {number} [args.port] - Port.\n   * @param {number} [args.maxWorkers] - Max workers.\n   */\n  constructor({configuration, host, inProcess, maxWorkers, port}) {\n    this.configuration = configuration\n    this.inProcess = inProcess || false\n    this.logger = new Logger(this)\n    this.host = host || \"0.0.0.0\"\n    this.port = port || 3006\n    this.maxWorkers = maxWorkers || 16\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async start() {\n    await this._ensureAtLeastOneWorker()\n    this.netServer = new Net.Server()\n    this.netServer.on(\"close\", this.onClose)\n    this.netServer.on(\"connection\", this.onConnection)\n    this.netServer.on(\"error\", this.onServerError)\n    await this._netServerListen()\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  _netServerListen() {\n    return new Promise((resolve, reject) => {\n      if (!this.netServer) throw new Error(\"No netServer\")\n\n      try {\n        this.netServer.listen(this.port, this.host, () => {\n          this.logger.debug(`Velocious listening on ${this.host}:${this.port}`)\n          resolve(undefined)\n        })\n      } catch (error) {\n        reject(error)\n      }\n    })\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async _ensureAtLeastOneWorker() {\n    if (this.workerHandlers.length == 0) {\n      await this.spawnWorker()\n    }\n  }\n\n  /** @returns {boolean} - Whether active.  */\n  isActive() {\n    if (this.netServer) {\n      return this.netServer.listening\n    }\n\n    return false\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async stopClients() {\n    const promises = []\n\n    for (const clientCount in this.clients) {\n      const client = this.clients[clientCount]\n\n      promises.push(client.end())\n    }\n\n    await Promise.all(promises)\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  stopServer() {\n    return new Promise((resolve, reject) => {\n      if (!this.netServer || !this.netServer.listening) {\n        resolve(undefined)\n        return\n      }\n\n      this.netServer.close((error) => {\n        if (error) {\n          reject(error)\n        } else {\n          resolve(undefined)\n        }\n      })\n    })\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async stop() {\n    await this.stopClients()\n    await this.stopServer()\n\n    const stopTasks = this.workerHandlers.map((handler) => handler.stop())\n    await Promise.all(stopTasks)\n    this.workerHandlers = []\n  }\n\n  /** @returns {void} - No return value.  */\n  onClose = () => {\n    this.events.emit(\"close\")\n  }\n\n  /**\n   * @param {Error} error - Server socket error.\n   * @returns {void} - No return value.\n   */\n  onServerError = (error) => {\n    this.logger.error(`Velocious HTTP server socket error on ${this.host}:${this.port}`, error)\n  }\n\n  /**\n   * @param {import(\"net\").Socket} socket - Socket instance.\n   * @returns {void} - No return value.\n   */\n  onConnection = (socket) => {\n    const clientCount = this.clientCount\n\n    this.logger.debug(() => [\"New client\", {\n      clientCount,\n      remoteAddress: socket.remoteAddress,\n      remoteFamily: socket.remoteFamily,\n      remotePort: socket.remotePort\n    }])\n    this.clientCount++\n\n    try {\n      const workerHandler = this.workerHandlerToUse()\n      const client = new ServerClient({\n        clientCount,\n        configuration: this.configuration,\n        socket\n      })\n\n      client.events.on(\"close\", this.onClientClose)\n\n      this.logger.debug(`Gave client ${clientCount} to worker ${workerHandler.workerCount}`)\n      workerHandler.addSocketConnection(client)\n      this.clients[clientCount] = client\n    } catch (error) {\n      this.logger.error(`Failed to initialize client ${clientCount} on new connection`, error)\n      socket.destroy()\n    }\n  }\n\n  /**\n   * @param {ServerClient} client - Client instance.\n   * @returns {void} - No return value.\n   */\n  onClientClose = (client) => {\n    const clientCount = digg(client, \"clientCount\")\n    const oldClientsLength = Object.keys(this.clients).length\n\n    delete this.clients[clientCount]\n\n    const newClientsLength = Object.keys(this.clients).length\n\n    if (newClientsLength != (oldClientsLength - 1)) {\n      this.logger.error(`Expected client to have been removed but length didn't change from ${oldClientsLength} to ${oldClientsLength - 1}`)\n    }\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async spawnWorker() {\n    const workerCount = this.workerCount\n\n    this.workerCount++\n\n    const Handler = this.inProcess ? InProcessHandler : WorkerHandler\n    const workerHandler = new Handler({\n      configuration: this.configuration,\n      workerCount\n    })\n\n    await workerHandler.start()\n    this.workerHandlers.push(workerHandler)\n  }\n\n  /** @returns {WorkerHandler | InProcessHandler} - The worker handler to use. */\n  workerHandlerToUse() {\n    this.logger.debug(`Worker handlers length: ${this.workerHandlers.length}`)\n\n    const randomWorkerNumber = Math.floor(Math.random() * this.workerHandlers.length)\n    const workerHandler = this.workerHandlers[randomWorkerNumber]\n\n    if (!workerHandler) {\n      throw new Error(`No workerHandler by that number: ${randomWorkerNumber}`)\n    }\n\n    return workerHandler\n  }\n}\n"]}
237
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/http-server/index.js"],"names":[],"mappings":"AAAA,YAAY;AAEZ,OAAO,EAAC,IAAI,EAAC,MAAM,WAAW,CAAA;AAC9B,OAAO,mBAAmB,MAAM,2BAA2B,CAAA;AAC3D,OAAO,YAAY,MAAM,2BAA2B,CAAA;AACpD,OAAO,gBAAgB,MAAM,gCAAgC,CAAA;AAC7D,OAAO,MAAM,MAAM,cAAc,CAAA;AACjC,OAAO,GAAG,MAAM,KAAK,CAAA;AACrB,OAAO,YAAY,MAAM,oBAAoB,CAAA;AAC7C,OAAO,aAAa,MAAM,2BAA2B,CAAA;AAErD,MAAM,CAAC,OAAO,OAAO,mBAAmB;IACtC,WAAW,GAAG,CAAC,CAAA;IAEf,4CAA4C;IAC5C,OAAO,GAAG,EAAE,CAAA;IAEZ,MAAM,GAAG,IAAI,YAAY,EAAE,CAAA;IAC3B,WAAW,GAAG,CAAC,CAAA;IAEf,sDAAsD;IACtD,cAAc,GAAG,EAAE,CAAA;IAEnB;;;;;;;;OAQG;IACH,YAAY,EAAC,aAAa,EAAE,0BAA0B,EAAE,IAAI,EAAE,SAAS,EAAE,UAAU,EAAE,IAAI,EAAC;QACxF,IAAI,CAAC,aAAa,GAAG,aAAa,CAAA;QAClC,IAAI,CAAC,0BAA0B,GAAG,0BAA0B,CAAA;QAC5D,IAAI,CAAC,SAAS,GAAG,SAAS,IAAI,KAAK,CAAA;QACnC,IAAI,CAAC,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,CAAA;QAC9B,IAAI,CAAC,IAAI,GAAG,IAAI,IAAI,SAAS,CAAA;QAC7B,IAAI,CAAC,IAAI,GAAG,IAAI,IAAI,IAAI,CAAA;QACxB,IAAI,CAAC,UAAU,GAAG,UAAU,IAAI,EAAE,CAAA;IACpC,CAAC;IAED,0DAA0D;IAC1D,KAAK,CAAC,KAAK;QACT,MAAM,IAAI,CAAC,uBAAuB,EAAE,CAAA;QACpC,MAAM,IAAI,CAAC,yBAAyB,EAAE,CAAA;QACtC,IAAI,CAAC,SAAS,GAAG,IAAI,GAAG,CAAC,MAAM,EAAE,CAAA;QACjC,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,OAAO,EAAE,IAAI,CAAC,OAAO,CAAC,CAAA;QACxC,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,YAAY,EAAE,IAAI,CAAC,YAAY,CAAC,CAAA;QAClD,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,OAAO,EAAE,IAAI,CAAC,aAAa,CAAC,CAAA;QAC9C,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAA;IAC/B,CAAC;IAED,0DAA0D;IAC1D,gBAAgB;QACd,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,IAAI,CAAC,IAAI,CAAC,SAAS;gBAAE,MAAM,IAAI,KAAK,CAAC,cAAc,CAAC,CAAA;YAEpD,IAAI,CAAC;gBACH,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,GAAG,EAAE;oBAC/C,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,0BAA0B,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC,CAAA;oBACrE,OAAO,CAAC,SAAS,CAAC,CAAA;gBACpB,CAAC,CAAC,CAAA;YACJ,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,MAAM,CAAC,KAAK,CAAC,CAAA;YACf,CAAC;QACH,CAAC,CAAC,CAAA;IACJ,CAAC;IAED,0DAA0D;IAC1D,KAAK,CAAC,uBAAuB;QAC3B,IAAI,IAAI,CAAC,cAAc,CAAC,MAAM,IAAI,CAAC,EAAE,CAAC;YACpC,MAAM,IAAI,CAAC,WAAW,EAAE,CAAA;QAC1B,CAAC;IACH,CAAC;IAED,4CAA4C;IAC5C,QAAQ;QACN,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,OAAO,IAAI,CAAC,SAAS,CAAC,SAAS,CAAA;QACjC,CAAC;QAED,OAAO,KAAK,CAAA;IACd,CAAC;IAED,0DAA0D;IAC1D,KAAK,CAAC,WAAW;QACf,MAAM,QAAQ,GAAG,EAAE,CAAA;QAEnB,KAAK,MAAM,WAAW,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YACvC,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,CAAA;YAExC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,CAAA;QAC7B,CAAC;QAED,MAAM,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAA;IAC7B,CAAC;IAED,0DAA0D;IAC1D,UAAU;QACR,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,EAAE,CAAC;gBACjD,OAAO,CAAC,SAAS,CAAC,CAAA;gBAClB,OAAM;YACR,CAAC;YAED,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE;gBAC7B,IAAI,KAAK,EAAE,CAAC;oBACV,MAAM,CAAC,KAAK,CAAC,CAAA;gBACf,CAAC;qBAAM,CAAC;oBACN,OAAO,CAAC,SAAS,CAAC,CAAA;gBACpB,CAAC;YACH,CAAC,CAAC,CAAA;QACJ,CAAC,CAAC,CAAA;IACJ,CAAC;IAED,0DAA0D;IAC1D,KAAK,CAAC,IAAI;QACR,IAAI,CAAC,SAAS,GAAG,IAAI,CAAA;QACrB,MAAM,IAAI,CAAC,mBAAmB,EAAE,IAAI,EAAE,CAAA;QACtC,IAAI,CAAC,mBAAmB,GAAG,SAAS,CAAA;QACpC,MAAM,IAAI,CAAC,WAAW,EAAE,CAAA;QACxB,MAAM,IAAI,CAAC,UAAU,EAAE,CAAA;QAEvB,MAAM,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAA;QACtE,MAAM,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC,CAAA;QAC5B,IAAI,CAAC,cAAc,GAAG,EAAE,CAAA;IAC1B,CAAC;IAED,0CAA0C;IAC1C,OAAO,GAAG,GAAG,EAAE;QACb,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAA;IAC3B,CAAC,CAAA;IAED;;;OAGG;IACH,aAAa,GAAG,CAAC,KAAK,EAAE,EAAE;QACxB,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,yCAAyC,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,EAAE,EAAE,KAAK,CAAC,CAAA;IAC7F,CAAC,CAAA;IAED;;;OAGG;IACH,YAAY,GAAG,CAAC,MAAM,EAAE,EAAE;QACxB,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAA;QAEpC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,CAAC,YAAY,EAAE;gBACrC,WAAW;gBACX,aAAa,EAAE,MAAM,CAAC,aAAa;gBACnC,YAAY,EAAE,MAAM,CAAC,YAAY;gBACjC,UAAU,EAAE,MAAM,CAAC,UAAU;aAC9B,CAAC,CAAC,CAAA;QACH,IAAI,CAAC,WAAW,EAAE,CAAA;QAElB,IAAI,CAAC;YACH,MAAM,aAAa,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAA;YAC/C,MAAM,MAAM,GAAG,IAAI,YAAY,CAAC;gBAC9B,WAAW;gBACX,aAAa,EAAE,IAAI,CAAC,aAAa;gBACjC,MAAM;aACP,CAAC,CAAA;YAEF,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,IAAI,CAAC,aAAa,CAAC,CAAA;YAE7C,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,eAAe,WAAW,cAAc,aAAa,CAAC,WAAW,EAAE,CAAC,CAAA;YACtF,aAAa,CAAC,mBAAmB,CAAC,MAAM,CAAC,CAAA;YACzC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,GAAG,MAAM,CAAA;QACpC,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,+BAA+B,WAAW,oBAAoB,EAAE,KAAK,CAAC,CAAA;YACxF,MAAM,CAAC,OAAO,EAAE,CAAA;QAClB,CAAC;IACH,CAAC,CAAA;IAED;;;OAGG;IACH,aAAa,GAAG,CAAC,MAAM,EAAE,EAAE;QACzB,MAAM,WAAW,GAAG,IAAI,CAAC,MAAM,EAAE,aAAa,CAAC,CAAA;QAC/C,MAAM,gBAAgB,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,MAAM,CAAA;QAEzD,OAAO,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,CAAA;QAEhC,MAAM,gBAAgB,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,MAAM,CAAA;QAEzD,IAAI,gBAAgB,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAC,EAAE,CAAC;YAC/C,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,sEAAsE,gBAAgB,OAAO,gBAAgB,GAAG,CAAC,EAAE,CAAC,CAAA;QACxI,CAAC;IACH,CAAC,CAAA;IAED,0DAA0D;IAC1D,KAAK,CAAC,WAAW;QACf,MAAM,aAAa,GAAG,MAAM,IAAI,CAAC,mBAAmB,EAAE,CAAA;QAEtD,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,aAAa,CAAC,CAAA;IACzC,CAAC;IAED,qFAAqF;IACrF,KAAK,CAAC,mBAAmB;QACvB,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAA;QAEpC,IAAI,CAAC,WAAW,EAAE,CAAA;QAElB,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC,aAAa,CAAA;QACjE,MAAM,aAAa,GAAG,IAAI,OAAO,CAAC;YAChC,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,WAAW;SACZ,CAAC,CAAA;QAEF,MAAM,aAAa,CAAC,KAAK,EAAE,CAAA;QAE3B,OAAO,aAAa,CAAA;IACtB,CAAC;IAED,+EAA+E;IAC/E,kBAAkB;QAChB,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,2BAA2B,IAAI,CAAC,cAAc,CAAC,MAAM,EAAE,CAAC,CAAA;QAE1E,MAAM,kBAAkB,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,CAAA;QACjF,MAAM,aAAa,GAAG,IAAI,CAAC,cAAc,CAAC,kBAAkB,CAAC,CAAA;QAE7D,IAAI,CAAC,aAAa,EAAE,CAAC;YACnB,MAAM,IAAI,KAAK,CAAC,oCAAoC,kBAAkB,EAAE,CAAC,CAAA;QAC3E,CAAC;QAED,OAAO,aAAa,CAAA;IACtB,CAAC;IAED,6EAA6E;IAC7E,6BAA6B;QAC3B,OAAO,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,aAAa,CAAC,cAAc,EAAE,KAAK,aAAa,CAAA;IACjF,CAAC;IAED,uEAAuE;IACvE,KAAK,CAAC,yBAAyB;QAC7B,IAAI,CAAC,IAAI,CAAC,6BAA6B,EAAE;YAAE,OAAM;QACjD,IAAI,IAAI,CAAC,mBAAmB;YAAE,OAAM;QAEpC,MAAM,yBAAyB,GAAG,IAAI,CAAC,0BAA0B;eAC5D,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,mBAAmB,CAAC,IAAI,CAAC,CAAC,CAAA;QAE9C,IAAI,CAAC,mBAAmB,GAAG,yBAAyB,CAAC;YACnD,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,QAAQ,EAAE,KAAK,EAAE,EAAC,WAAW,EAAC,EAAE,EAAE;gBAChC,MAAM,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,6CAA6C,WAAW,EAAE,CAAC,CAAA;gBAClF,MAAM,IAAI,CAAC,2BAA2B,EAAE,CAAA;YAC1C,CAAC;SACF,CAAC,CAAA;QAEF,MAAM,IAAI,CAAC,mBAAmB,CAAC,KAAK,EAAE,CAAA;IACxC,CAAC;IAED,4EAA4E;IAC5E,KAAK,CAAC,2BAA2B;QAC/B,IAAI,IAAI,CAAC,SAAS;YAAE,OAAM;QAE1B,IAAI,IAAI,CAAC,+BAA+B,EAAE,CAAC;YACzC,IAAI,CAAC,kCAAkC,GAAG,IAAI,CAAA;YAC9C,OAAM;QACR,CAAC;QAED,IAAI,CAAC,+BAA+B,GAAG,IAAI,CAAA;QAE3C,IAAI,CAAC;YACH,GAAG,CAAC;gBACF,IAAI,CAAC,kCAAkC,GAAG,KAAK,CAAA;gBAE/C,MAAM,iBAAiB,GAAG,CAAC,GAAG,IAAI,CAAC,cAAc,CAAC,CAAA;gBAClD,MAAM,gBAAgB,GAAG,MAAM,IAAI,CAAC,mBAAmB,EAAE,CAAA;gBAEzD,IAAI,CAAC,cAAc,GAAG,CAAC,gBAAgB,CAAC,CAAA;gBAExC,MAAM,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC,GAAG,CAAC,CAAC,aAAa,EAAE,EAAE,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC,CAAC,CAAA;YACnF,CAAC,QAAQ,IAAI,CAAC,kCAAkC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAC;QACtE,CAAC;gBAAS,CAAC;YACT,IAAI,CAAC,+BAA+B,GAAG,KAAK,CAAA;QAC9C,CAAC;IACH,CAAC;CACF","sourcesContent":["// @ts-check\n\nimport {digg} from \"diggerize\"\nimport DevelopmentReloader from \"./development-reloader.js\"\nimport EventEmitter from \"../utils/event-emitter.js\"\nimport InProcessHandler from \"./worker-handler/in-process.js\"\nimport Logger from \"../logger.js\"\nimport Net from \"net\"\nimport ServerClient from \"./server-client.js\"\nimport WorkerHandler from \"./worker-handler/index.js\"\n\nexport default class VelociousHttpServer {\n  clientCount = 0\n\n  /** @type {Record<string, ServerClient>}  */\n  clients = {}\n\n  events = new EventEmitter()\n  workerCount = 0\n\n  /** @type {Array<WorkerHandler | InProcessHandler>} */\n  workerHandlers = []\n\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {string} [args.host] - Host.\n   * @param {boolean} [args.inProcess] - Run HTTP handlers in the main thread instead of worker threads.\n   * @param {number} [args.port] - Port.\n   * @param {number} [args.maxWorkers] - Max workers.\n   * @param {function({configuration: import(\"../configuration.js\").default, onReload: function({changedPath: string}) : Promise<void>}) : {start: () => Promise<void>, stop: () => Promise<void>}} [args.developmentReloaderFactory] - Development reloader factory.\n   */\n  constructor({configuration, developmentReloaderFactory, host, inProcess, maxWorkers, port}) {\n    this.configuration = configuration\n    this.developmentReloaderFactory = developmentReloaderFactory\n    this.inProcess = inProcess || false\n    this.logger = new Logger(this)\n    this.host = host || \"0.0.0.0\"\n    this.port = port || 3006\n    this.maxWorkers = maxWorkers || 16\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async start() {\n    await this._ensureAtLeastOneWorker()\n    await this._startDevelopmentReloader()\n    this.netServer = new Net.Server()\n    this.netServer.on(\"close\", this.onClose)\n    this.netServer.on(\"connection\", this.onConnection)\n    this.netServer.on(\"error\", this.onServerError)\n    await this._netServerListen()\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  _netServerListen() {\n    return new Promise((resolve, reject) => {\n      if (!this.netServer) throw new Error(\"No netServer\")\n\n      try {\n        this.netServer.listen(this.port, this.host, () => {\n          this.logger.debug(`Velocious listening on ${this.host}:${this.port}`)\n          resolve(undefined)\n        })\n      } catch (error) {\n        reject(error)\n      }\n    })\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async _ensureAtLeastOneWorker() {\n    if (this.workerHandlers.length == 0) {\n      await this.spawnWorker()\n    }\n  }\n\n  /** @returns {boolean} - Whether active.  */\n  isActive() {\n    if (this.netServer) {\n      return this.netServer.listening\n    }\n\n    return false\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async stopClients() {\n    const promises = []\n\n    for (const clientCount in this.clients) {\n      const client = this.clients[clientCount]\n\n      promises.push(client.end())\n    }\n\n    await Promise.all(promises)\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  stopServer() {\n    return new Promise((resolve, reject) => {\n      if (!this.netServer || !this.netServer.listening) {\n        resolve(undefined)\n        return\n      }\n\n      this.netServer.close((error) => {\n        if (error) {\n          reject(error)\n        } else {\n          resolve(undefined)\n        }\n      })\n    })\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async stop() {\n    this._stopping = true\n    await this.developmentReloader?.stop()\n    this.developmentReloader = undefined\n    await this.stopClients()\n    await this.stopServer()\n\n    const stopTasks = this.workerHandlers.map((handler) => handler.stop())\n    await Promise.all(stopTasks)\n    this.workerHandlers = []\n  }\n\n  /** @returns {void} - No return value.  */\n  onClose = () => {\n    this.events.emit(\"close\")\n  }\n\n  /**\n   * @param {Error} error - Server socket error.\n   * @returns {void} - No return value.\n   */\n  onServerError = (error) => {\n    this.logger.error(`Velocious HTTP server socket error on ${this.host}:${this.port}`, error)\n  }\n\n  /**\n   * @param {import(\"net\").Socket} socket - Socket instance.\n   * @returns {void} - No return value.\n   */\n  onConnection = (socket) => {\n    const clientCount = this.clientCount\n\n    this.logger.debug(() => [\"New client\", {\n      clientCount,\n      remoteAddress: socket.remoteAddress,\n      remoteFamily: socket.remoteFamily,\n      remotePort: socket.remotePort\n    }])\n    this.clientCount++\n\n    try {\n      const workerHandler = this.workerHandlerToUse()\n      const client = new ServerClient({\n        clientCount,\n        configuration: this.configuration,\n        socket\n      })\n\n      client.events.on(\"close\", this.onClientClose)\n\n      this.logger.debug(`Gave client ${clientCount} to worker ${workerHandler.workerCount}`)\n      workerHandler.addSocketConnection(client)\n      this.clients[clientCount] = client\n    } catch (error) {\n      this.logger.error(`Failed to initialize client ${clientCount} on new connection`, error)\n      socket.destroy()\n    }\n  }\n\n  /**\n   * @param {ServerClient} client - Client instance.\n   * @returns {void} - No return value.\n   */\n  onClientClose = (client) => {\n    const clientCount = digg(client, \"clientCount\")\n    const oldClientsLength = Object.keys(this.clients).length\n\n    delete this.clients[clientCount]\n\n    const newClientsLength = Object.keys(this.clients).length\n\n    if (newClientsLength != (oldClientsLength - 1)) {\n      this.logger.error(`Expected client to have been removed but length didn't change from ${oldClientsLength} to ${oldClientsLength - 1}`)\n    }\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async spawnWorker() {\n    const workerHandler = await this._buildWorkerHandler()\n\n    this.workerHandlers.push(workerHandler)\n  }\n\n  /** @returns {Promise<WorkerHandler | InProcessHandler>} - Started worker handler. */\n  async _buildWorkerHandler() {\n    const workerCount = this.workerCount\n\n    this.workerCount++\n\n    const Handler = this.inProcess ? InProcessHandler : WorkerHandler\n    const workerHandler = new Handler({\n      configuration: this.configuration,\n      workerCount\n    })\n\n    await workerHandler.start()\n\n    return workerHandler\n  }\n\n  /** @returns {WorkerHandler | InProcessHandler} - The worker handler to use. */\n  workerHandlerToUse() {\n    this.logger.debug(`Worker handlers length: ${this.workerHandlers.length}`)\n\n    const randomWorkerNumber = Math.floor(Math.random() * this.workerHandlers.length)\n    const workerHandler = this.workerHandlers[randomWorkerNumber]\n\n    if (!workerHandler) {\n      throw new Error(`No workerHandler by that number: ${randomWorkerNumber}`)\n    }\n\n    return workerHandler\n  }\n\n  /** @returns {boolean} - Whether development worker hot reload should run. */\n  shouldUseDevelopmentHotReload() {\n    return !this.inProcess && this.configuration.getEnvironment() === \"development\"\n  }\n\n  /** @returns {Promise<void>} - Resolves when watcher setup finishes. */\n  async _startDevelopmentReloader() {\n    if (!this.shouldUseDevelopmentHotReload()) return\n    if (this.developmentReloader) return\n\n    const createDevelopmentReloader = this.developmentReloaderFactory\n      || ((args) => new DevelopmentReloader(args))\n\n    this.developmentReloader = createDevelopmentReloader({\n      configuration: this.configuration,\n      onReload: async ({changedPath}) => {\n        await this.logger.info(`Development hot reload detected change in ${changedPath}`)\n        await this.reloadWorkersForDevelopment()\n      }\n    })\n\n    await this.developmentReloader.start()\n  }\n\n  /** @returns {Promise<void>} - Resolves when workers have been refreshed. */\n  async reloadWorkersForDevelopment() {\n    if (this._stopping) return\n\n    if (this._reloadingWorkersForDevelopment) {\n      this._reloadWorkersForDevelopmentQueued = true\n      return\n    }\n\n    this._reloadingWorkersForDevelopment = true\n\n    try {\n      do {\n        this._reloadWorkersForDevelopmentQueued = false\n\n        const oldWorkerHandlers = [...this.workerHandlers]\n        const newWorkerHandler = await this._buildWorkerHandler()\n\n        this.workerHandlers = [newWorkerHandler]\n\n        await Promise.all(oldWorkerHandlers.map((workerHandler) => workerHandler.stop()))\n      } while (this._reloadWorkersForDevelopmentQueued && !this._stopping)\n    } finally {\n      this._reloadingWorkersForDevelopment = false\n    }\n  }\n}\n"]}