system-testing 1.0.88 → 1.0.91

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.
@@ -2,7 +2,7 @@
2
2
  export default class SystemTestCommunicator {
3
3
  /**
4
4
  * @param {object} args
5
- * @param {(args: Record<string, any>) => Promise<{result: string} | void>} args.onCommand
5
+ * @param {(args: {data: Record<string, any>}) => Promise<any>} args.onCommand
6
6
  * @param {object} [args.parent]
7
7
  */
8
8
  constructor({ onCommand, parent }) {
@@ -102,4 +102,4 @@ export default class SystemTestCommunicator {
102
102
  this.send({ type: "response", id, data });
103
103
  }
104
104
  }
105
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"system-test-communicator.js","sourceRoot":"","sources":["../src/system-test-communicator.js"],"names":[],"mappings":"AAAA,YAAY;AAEZ,MAAM,CAAC,OAAO,OAAO,sBAAsB;IAYzC;;;;OAIG;IACH,YAAY,EAAC,SAAS,EAAE,MAAM,EAAC;QAhB/B,0FAA0F;QAC1F,eAAU,GAAG,EAAE,CAAA;QAEf,kCAAkC;QAClC,eAAU,GAAG,EAAE,CAAA;QAEf,oBAAe,GAAG,CAAC,CAAA;QAEnB,+BAA+B;QAC/B,OAAE,GAAG,IAAI,CAAA;QAwBT,2BAA2B;QAC3B,YAAO,GAAG,CAAC,KAAK,EAAE,EAAE;YAClB,OAAO,CAAC,KAAK,CAAC,wBAAwB,EAAE,KAAK,CAAC,CAAA;QAChD,CAAC,CAAA;QAED,8BAA8B;QAC9B,cAAS,GAAG,KAAK,EAAE,OAAO,EAAE,EAAE;YAC5B,yEAAyE;YACzE,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAA;YAEhC,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;gBACnB,SAAS;YACX,CAAC;iBAAM,IAAI,IAAI,CAAC,IAAI,IAAI,SAAS,EAAE,CAAC;gBAClC,IAAI,CAAC;oBACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,EAAC,IAAI,EAAE,IAAI,CAAC,IAAI,EAAC,CAAC,CAAA;oBAEtD,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,EAAE,EAAC,MAAM,EAAC,CAAC,CAAA;gBACjC,CAAC;gBAAC,OAAO,KAAK,EAAE,CAAC;oBACf,IAAI,KAAK,YAAY,KAAK,EAAE,CAAC;wBAC3B,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,EAAE,EAAC,KAAK,EAAE,KAAK,CAAC,OAAO,EAAC,CAAC,CAAA;oBAC/C,CAAC;yBAAM,CAAC;wBACN,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,EAAE,EAAC,KAAK,EAAE,KAAK,EAAC,CAAC,CAAA;oBACvC,CAAC;gBACH,CAAC;YACH,CAAC;iBAAM,IAAI,IAAI,CAAC,IAAI,IAAI,UAAU,EAAE,CAAC;gBACnC,MAAM,QAAQ,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;gBAEzC,IAAI,CAAC,QAAQ,EAAE,CAAC;oBACd,MAAM,IAAI,KAAK,CAAC,qBAAqB,IAAI,CAAC,EAAE,EAAE,CAAC,CAAA;gBACjD,CAAC;gBAED,OAAO,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;gBAE/B,IAAI,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC;oBACpB,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;gBAClC,CAAC;qBAAM,CAAC;oBACN,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAA;gBACpC,CAAC;YACH,CAAC;iBAAM,CAAC;gBACN,MAAM,IAAI,KAAK,CAAC,4CAA4C,IAAI,CAAC,IAAI,KAAK,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;YACnG,CAAC;QACH,CAAC,CAAA;QAED,WAAM,GAAG,GAAG,EAAE;YACZ,IAAI,CAAC,cAAc,EAAE,CAAA;QACvB,CAAC,CAAA;QA7DC,IAAI,CAAC,SAAS,GAAG,SAAS,CAAA;QAC1B,IAAI,CAAC,MAAM,GAAG,MAAM,CAAA;IACtB,CAAC;IAED,cAAc;QACZ,OAAO,IAAI,CAAC,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACpC,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAA;YAEpC,IAAI,CAAC,IAAI,CAAC,EAAE,IAAI,IAAI,CAAC,EAAE,CAAC,UAAU,KAAK,CAAC,EAAE,CAAC;gBACzC,MAAM,IAAI,KAAK,CAAC,uBAAuB,CAAC,CAAA;YAC1C,CAAC;YAED,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAA;QACpC,CAAC;IACH,CAAC;IAiDD;;;OAGG;IACH,IAAI,CAAC,IAAI;QACP,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QAE1B,IAAI,IAAI,CAAC,EAAE,EAAE,UAAU,IAAI,CAAC,EAAE,CAAC;YAC7B,IAAI,CAAC,cAAc,EAAE,CAAA;QACvB,CAAC;IACH,CAAC;IAED;;;;OAIG;IACH,WAAW,CAAC,IAAI;QACd,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,MAAM,EAAE,GAAG,IAAI,CAAC,eAAe,CAAA;YAE/B,IAAI,CAAC,eAAe,IAAI,CAAC,CAAA;YACzB,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC,GAAG,EAAC,OAAO,EAAE,MAAM,EAAC,CAAA;YAEvC,IAAI,CAAC,IAAI,CAAC,EAAC,IAAI,EAAE,SAAS,EAAE,EAAE,EAAE,IAAI,EAAC,CAAC,CAAA;QACxC,CAAC,CAAC,CAAA;IACJ,CAAC;IAED;;;;OAIG;IACH,OAAO,CAAC,EAAE,EAAE,IAAI;QACd,IAAI,CAAC,IAAI,CAAC,EAAC,IAAI,EAAE,UAAU,EAAE,EAAE,EAAE,IAAI,EAAC,CAAC,CAAA;IACzC,CAAC;CACF","sourcesContent":["// @ts-check\n\nexport default class SystemTestCommunicator {\n  /** @type {Record<string, {resolve: (data: any) => void, reject: (data: any) => void}>} */\n  _responses = {}\n\n  /** @type {Record<string, any>} */\n  _sendQueue = []\n\n  _sendQueueCount = 0\n\n  /** @type {WebSocket | null} */\n  ws = null\n\n  /**\n   * @param {object} args\n   * @param {(args: Record<string, any>) => Promise<{result: string} | void>} args.onCommand\n   * @param {object} [args.parent]\n   */\n  constructor({onCommand, parent}) {\n    this.onCommand = onCommand\n    this.parent = parent\n  }\n\n  flushSendQueue() {\n    while (this._sendQueue.length !== 0) {\n      const data = this._sendQueue.shift()\n\n      if (!this.ws || this.ws.readyState !== 1) {\n        throw new Error(\"WebSocket is not open\")\n      }\n\n      this.ws.send(JSON.stringify(data))\n    }\n  }\n\n  /** @param {Error} error */\n  onError = (error) => {\n    console.error(\"onWebSocketClientError\", error)\n  }\n\n  /** @param {string} rawData */\n  onMessage = async (rawData) => {\n    /** @type {{data: any, id: number, type: string, isTrusted?: boolean}} */\n    const data = JSON.parse(rawData)\n\n    if (data.isTrusted) {\n      // Ignore\n    } else if (data.type == \"command\") {\n      try {\n        const result = await this.onCommand({data: data.data})\n\n        this.respond(data.id, {result})\n      } catch (error) {\n        if (error instanceof Error) {\n          this.respond(data.id, {error: error.message})\n        } else {\n          this.respond(data.id, {error: error})\n        }\n      }\n    } else if (data.type == \"response\") {\n      const response = this._responses[data.id]\n\n      if (!response) {\n        throw new Error(`No such response: ${data.id}`)\n      }\n\n      delete this._responses[data.id]\n\n      if (data.data.error) {\n        response.reject(data.data.error)\n      } else {\n        response.resolve(data.data.result)\n      }\n    } else {\n      throw new Error(`Unknown type for SystemTestCommunicator: ${data.type}: ${JSON.stringify(data)}`)\n    }\n  }\n\n  onOpen = () => {\n    this.flushSendQueue()\n  }\n\n  /**\n   * @param {Record<string, any>} data\n   * @returns {void}\n   */\n  send(data) {\n    this._sendQueue.push(data)\n\n    if (this.ws?.readyState == 1) {\n      this.flushSendQueue()\n    }\n  }\n\n  /**\n   * Sends a command and returns a promise that resolves with the response.\n   * @param {Record<string, any>} data - The command data to send.\n   * @returns {Promise<void>} A promise that resolves with the response data.\n   */\n  sendCommand(data) {\n    return new Promise((resolve, reject) => {\n      const id = this._sendQueueCount\n\n      this._sendQueueCount += 1\n      this._responses[id] = {resolve, reject}\n\n      this.send({type: \"command\", id, data})\n    })\n  }\n\n  /**\n   * @param {number} id\n   * @param {Record<string, any>} data\n   * @returns {void}\n   */\n  respond(id, data) {\n    this.send({type: \"response\", id, data})\n  }\n}\n"]}
105
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"system-test-communicator.js","sourceRoot":"","sources":["../src/system-test-communicator.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nexport default class SystemTestCommunicator {\n  /** @type {Record<string, {resolve: (data: any) => void, reject: (data: any) => void}>} */\n  _responses = {}\n\n  /** @type {Record<string, any>} */\n  _sendQueue = []\n\n  _sendQueueCount = 0\n\n  /** @type {WebSocket | null} */\n  ws = null\n\n  /**\n   * @param {object} args\n   * @param {(args: {data: Record<string, any>}) => Promise<any>} args.onCommand\n   * @param {object} [args.parent]\n   */\n  constructor({onCommand, parent}) {\n    this.onCommand = onCommand\n    this.parent = parent\n  }\n\n  flushSendQueue() {\n    while (this._sendQueue.length !== 0) {\n      const data = this._sendQueue.shift()\n\n      if (!this.ws || this.ws.readyState !== 1) {\n        throw new Error(\"WebSocket is not open\")\n      }\n\n      this.ws.send(JSON.stringify(data))\n    }\n  }\n\n  /** @param {Error} error */\n  onError = (error) => {\n    console.error(\"onWebSocketClientError\", error)\n  }\n\n  /** @param {string} rawData */\n  onMessage = async (rawData) => {\n    /** @type {{data: any, id: number, type: string, isTrusted?: boolean}} */\n    const data = JSON.parse(rawData)\n\n    if (data.isTrusted) {\n      // Ignore\n    } else if (data.type == \"command\") {\n      try {\n        const result = await this.onCommand({data: data.data})\n\n        this.respond(data.id, {result})\n      } catch (error) {\n        if (error instanceof Error) {\n          this.respond(data.id, {error: error.message})\n        } else {\n          this.respond(data.id, {error: error})\n        }\n      }\n    } else if (data.type == \"response\") {\n      const response = this._responses[data.id]\n\n      if (!response) {\n        throw new Error(`No such response: ${data.id}`)\n      }\n\n      delete this._responses[data.id]\n\n      if (data.data.error) {\n        response.reject(data.data.error)\n      } else {\n        response.resolve(data.data.result)\n      }\n    } else {\n      throw new Error(`Unknown type for SystemTestCommunicator: ${data.type}: ${JSON.stringify(data)}`)\n    }\n  }\n\n  onOpen = () => {\n    this.flushSendQueue()\n  }\n\n  /**\n   * @param {Record<string, any>} data\n   * @returns {void}\n   */\n  send(data) {\n    this._sendQueue.push(data)\n\n    if (this.ws?.readyState == 1) {\n      this.flushSendQueue()\n    }\n  }\n\n  /**\n   * Sends a command and returns a promise that resolves with the response.\n   * @param {Record<string, any>} data - The command data to send.\n   * @returns {Promise<void>} A promise that resolves with the response data.\n   */\n  sendCommand(data) {\n    return new Promise((resolve, reject) => {\n      const id = this._sendQueueCount\n\n      this._sendQueueCount += 1\n      this._responses[id] = {resolve, reject}\n\n      this.send({type: \"command\", id, data})\n    })\n  }\n\n  /**\n   * @param {number} id\n   * @param {Record<string, any>} data\n   * @returns {void}\n   */\n  respond(id, data) {\n    this.send({type: \"response\", id, data})\n  }\n}\n"]}
@@ -12,7 +12,7 @@ export default class SystemTestHttpServer {
12
12
  _host: string;
13
13
  _port: number;
14
14
  _debug: boolean;
15
- _onError: (error: Error) => void;
15
+ _onError: ((error: Error) => void) | undefined;
16
16
  _started: boolean;
17
17
  _connectHost: string;
18
18
  /** @type {Set<import("node:net").Socket>} */
@@ -33,7 +33,7 @@ export default class SystemTestHttpServer {
33
33
  start(): Promise<void>;
34
34
  /** @returns {Promise<void>} */
35
35
  startHttpServer(): Promise<void>;
36
- httpServer: http.Server<typeof http.IncomingMessage, typeof http.ServerResponse>;
36
+ httpServer: http.Server<typeof http.IncomingMessage, typeof http.ServerResponse> | undefined;
37
37
  /**
38
38
  * @param {{timeoutMs?: number}} [args]
39
39
  * @returns {Promise<void>}
@@ -83,14 +83,15 @@ export default class SystemTestHttpServer {
83
83
  }
84
84
  this._connections.clear();
85
85
  }
86
+ const httpServer = this.httpServer;
86
87
  await new Promise((resolve, reject) => {
87
- this.httpServer.close((error) => {
88
+ httpServer.close((error) => {
88
89
  if (error) {
89
90
  reject(error);
90
91
  }
91
92
  else {
92
93
  this._started = false;
93
- resolve();
94
+ resolve(undefined);
94
95
  }
95
96
  });
96
97
  });
@@ -140,6 +141,7 @@ export default class SystemTestHttpServer {
140
141
  try {
141
142
  await new Promise((resolve, reject) => {
142
143
  const request = http.get(url, (response) => {
144
+ /** @type {Buffer[]} */
143
145
  const chunks = [];
144
146
  response.on("data", (chunk) => chunks.push(chunk));
145
147
  response.on("end", () => {
@@ -183,4 +185,4 @@ export default class SystemTestHttpServer {
183
185
  return message.includes("socket hang up");
184
186
  }
185
187
  }
186
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"system-test-http-server.js","sourceRoot":"","sources":["../src/system-test-http-server.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport fs from \"node:fs/promises\"\nimport http from \"node:http\"\nimport mime from \"mime\"\n\nexport default class SystemTestHttpServer {\n  /**\n   * @param {{host?: string, port?: number, debug?: boolean, onError?: (error: Error) => void, connectHost?: string}} [args]\n   */\n  constructor({host = \"localhost\", port = 1984, debug = false, onError, connectHost} = {}) {\n    this._host = host\n    this._port = port\n    this._debug = debug\n    this._onError = onError\n    this._started = false\n    this._connectHost = connectHost ?? (host === \"0.0.0.0\" ? \"127.0.0.1\" : host)\n    /** @type {Set<import(\"node:net\").Socket>} */\n    this._connections = new Set()\n  }\n\n  /** @param {any[]} args */\n  debugError(...args) {\n    console.error(\"[SystemTestHttpServer error]\", ...args)\n  }\n\n  /** @param {any[]} args */\n  debugLog(...args) {\n    if (this._debug) {\n      console.log(\"[SystemTestHttpServer]\", ...args)\n    }\n  }\n\n  /** @returns {Promise<void>} */\n  async close() {\n    if (!this.httpServer) {\n      throw new Error(\"HTTP server is not initialized\")\n    }\n\n    if (this._connections.size > 0) {\n      for (const connection of this._connections) {\n        connection.destroy()\n      }\n      this._connections.clear()\n    }\n\n    await new Promise((resolve, reject) => {\n      this.httpServer.close((error) => {\n        if (error) {\n          reject(error)\n        } else {\n          this._started = false\n          resolve()\n        }\n      })\n    })\n  }\n\n  /**\n   * @param {http.IncomingMessage} request\n   * @param {http.ServerResponse} response\n   * @returns {Promise<void>}\n   */\n  onHttpServerRequest = async (request, response) => {\n    if (!request.url) {\n      response.statusCode = 400\n      response.end(\"Bad Request\")\n      return\n    }\n\n    const baseUrl = `http://${request.headers.host || \"localhost:1984\"}`\n    const {pathname} = new URL(request.url, baseUrl)\n    let filePath = `${process.cwd()}/dist${pathname}`\n\n    if (filePath.endsWith(\"/\")) {\n      filePath += \"index.html\"\n    }\n\n    let fileExists\n\n    try {\n      await fs.stat(filePath)\n      fileExists = true\n    } catch (_error) { // eslint-disable-line no-unused-vars\n      fileExists = false\n    }\n\n    if (!fileExists) {\n      filePath = `${process.cwd()}/dist/index.html`\n    }\n\n    let fileContent\n\n    try {\n      fileContent = await fs.readFile(filePath)\n    } catch (error) {\n      const errorMessage = error instanceof Error ? error.message : String(error)\n\n      this.debugError(`HTTP server read failed: ${errorMessage}`)\n      response.statusCode = 500\n      response.end(\"Internal Server Error\")\n      return\n    }\n\n    const mimeType = mime.getType(filePath)\n\n    response.statusCode = 200\n\n    if (mimeType) {\n      response.setHeader(\"Content-Type\", mimeType)\n    }\n\n    response.end(fileContent)\n  }\n\n  /** @returns {Promise<void>} */\n  async start() {\n    this.debugLog(`Starting HTTP server on ${this._host}:${this._port}`)\n    await this.startHttpServer()\n    this.debugLog(`HTTP server started on ${this._host}:${this._port}`)\n  }\n\n  /** @returns {Promise<void>} */\n  startHttpServer() {\n    return new Promise((resolve, reject) => {\n      this.httpServer = http.createServer(this.onHttpServerRequest)\n      this.httpServer.on(\"connection\", (connection) => {\n        this._connections.add(connection)\n        connection.on(\"close\", () => {\n          this._connections.delete(connection)\n        })\n      })\n      this.httpServer.on(\"error\", (error) => {\n        const errorMessage = error instanceof Error ? error.message : String(error)\n\n        this.debugError(`HTTP server error: ${errorMessage}`)\n\n        if (this._started) {\n          if (this._onError) {\n            this._onError(error instanceof Error ? error : new Error(errorMessage))\n          }\n        } else {\n          reject(error)\n        }\n      })\n      this.httpServer.listen(this._port, this._host, () => {\n        this._started = true\n        resolve()\n      })\n    })\n  }\n\n  /**\n   * @param {{timeoutMs?: number}} [args]\n   * @returns {Promise<void>}\n   */\n  async assertReachable({timeoutMs = 5000} = {}) {\n    const url = `http://${this._connectHost}:${this._port}/`\n    const maxAttempts = 3\n\n    for (let attempt = 1; attempt <= maxAttempts; attempt++) {\n      try {\n        await new Promise((resolve, reject) => {\n          const request = http.get(url, (response) => {\n            const chunks = []\n\n            response.on(\"data\", (chunk) => chunks.push(chunk))\n            response.on(\"end\", () => {\n              const body = Buffer.concat(chunks).toString(\"utf8\")\n\n              if (response.statusCode !== 200) {\n                reject(new Error(`HTTP server health check failed with status ${response.statusCode} for ${url}: ${body.slice(0, 200)}`))\n                return\n              }\n\n              resolve(undefined)\n            })\n          })\n\n          request.on(\"error\", (error) => reject(error))\n\n          request.setTimeout(timeoutMs, () => {\n            request.destroy(new Error(`HTTP server health check timed out after ${timeoutMs}ms for ${url}`))\n          })\n        })\n\n        return\n      } catch (error) {\n        if (attempt === maxAttempts || !this.isRetryableHealthCheckError(error)) {\n          throw error\n        }\n\n        await new Promise((resolve) => setTimeout(resolve, 100))\n      }\n    }\n  }\n\n  /**\n   * @param {unknown} error\n   * @returns {boolean}\n   */\n  isRetryableHealthCheckError(error) {\n    if (error && typeof error === \"object\" && \"code\" in error) {\n      if (error.code === \"ECONNRESET\") return true\n      if (error.code === \"ECONNREFUSED\") return true\n      if (error.code === \"ETIMEDOUT\") return true\n    }\n\n    const message = error instanceof Error ? error.message : String(error)\n\n    return message.includes(\"socket hang up\")\n  }\n}\n"]}
188
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"system-test-http-server.js","sourceRoot":"","sources":["../src/system-test-http-server.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport fs from \"node:fs/promises\"\nimport http from \"node:http\"\nimport mime from \"mime\"\n\nexport default class SystemTestHttpServer {\n  /**\n   * @param {{host?: string, port?: number, debug?: boolean, onError?: (error: Error) => void, connectHost?: string}} [args]\n   */\n  constructor({host = \"localhost\", port = 1984, debug = false, onError, connectHost} = {}) {\n    this._host = host\n    this._port = port\n    this._debug = debug\n    this._onError = onError\n    this._started = false\n    this._connectHost = connectHost ?? (host === \"0.0.0.0\" ? \"127.0.0.1\" : host)\n    /** @type {Set<import(\"node:net\").Socket>} */\n    this._connections = new Set()\n  }\n\n  /** @param {any[]} args */\n  debugError(...args) {\n    console.error(\"[SystemTestHttpServer error]\", ...args)\n  }\n\n  /** @param {any[]} args */\n  debugLog(...args) {\n    if (this._debug) {\n      console.log(\"[SystemTestHttpServer]\", ...args)\n    }\n  }\n\n  /** @returns {Promise<void>} */\n  async close() {\n    if (!this.httpServer) {\n      throw new Error(\"HTTP server is not initialized\")\n    }\n\n    if (this._connections.size > 0) {\n      for (const connection of this._connections) {\n        connection.destroy()\n      }\n      this._connections.clear()\n    }\n\n    const httpServer = this.httpServer\n\n    await new Promise((resolve, reject) => {\n      httpServer.close((error) => {\n        if (error) {\n          reject(error)\n        } else {\n          this._started = false\n          resolve(undefined)\n        }\n      })\n    })\n  }\n\n  /**\n   * @param {http.IncomingMessage} request\n   * @param {http.ServerResponse} response\n   * @returns {Promise<void>}\n   */\n  onHttpServerRequest = async (request, response) => {\n    if (!request.url) {\n      response.statusCode = 400\n      response.end(\"Bad Request\")\n      return\n    }\n\n    const baseUrl = `http://${request.headers.host || \"localhost:1984\"}`\n    const {pathname} = new URL(request.url, baseUrl)\n    let filePath = `${process.cwd()}/dist${pathname}`\n\n    if (filePath.endsWith(\"/\")) {\n      filePath += \"index.html\"\n    }\n\n    let fileExists\n\n    try {\n      await fs.stat(filePath)\n      fileExists = true\n    } catch (_error) { // eslint-disable-line no-unused-vars\n      fileExists = false\n    }\n\n    if (!fileExists) {\n      filePath = `${process.cwd()}/dist/index.html`\n    }\n\n    let fileContent\n\n    try {\n      fileContent = await fs.readFile(filePath)\n    } catch (error) {\n      const errorMessage = error instanceof Error ? error.message : String(error)\n\n      this.debugError(`HTTP server read failed: ${errorMessage}`)\n      response.statusCode = 500\n      response.end(\"Internal Server Error\")\n      return\n    }\n\n    const mimeType = mime.getType(filePath)\n\n    response.statusCode = 200\n\n    if (mimeType) {\n      response.setHeader(\"Content-Type\", mimeType)\n    }\n\n    response.end(fileContent)\n  }\n\n  /** @returns {Promise<void>} */\n  async start() {\n    this.debugLog(`Starting HTTP server on ${this._host}:${this._port}`)\n    await this.startHttpServer()\n    this.debugLog(`HTTP server started on ${this._host}:${this._port}`)\n  }\n\n  /** @returns {Promise<void>} */\n  startHttpServer() {\n    return new Promise((resolve, reject) => {\n      this.httpServer = http.createServer(this.onHttpServerRequest)\n      this.httpServer.on(\"connection\", (connection) => {\n        this._connections.add(connection)\n        connection.on(\"close\", () => {\n          this._connections.delete(connection)\n        })\n      })\n      this.httpServer.on(\"error\", (error) => {\n        const errorMessage = error instanceof Error ? error.message : String(error)\n\n        this.debugError(`HTTP server error: ${errorMessage}`)\n\n        if (this._started) {\n          if (this._onError) {\n            this._onError(error instanceof Error ? error : new Error(errorMessage))\n          }\n        } else {\n          reject(error)\n        }\n      })\n      this.httpServer.listen(this._port, this._host, () => {\n        this._started = true\n        resolve()\n      })\n    })\n  }\n\n  /**\n   * @param {{timeoutMs?: number}} [args]\n   * @returns {Promise<void>}\n   */\n  async assertReachable({timeoutMs = 5000} = {}) {\n    const url = `http://${this._connectHost}:${this._port}/`\n    const maxAttempts = 3\n\n    for (let attempt = 1; attempt <= maxAttempts; attempt++) {\n      try {\n        await new Promise((resolve, reject) => {\n          const request = http.get(url, (response) => {\n            /** @type {Buffer[]} */\n            const chunks = []\n\n            response.on(\"data\", (chunk) => chunks.push(chunk))\n            response.on(\"end\", () => {\n              const body = Buffer.concat(chunks).toString(\"utf8\")\n\n              if (response.statusCode !== 200) {\n                reject(new Error(`HTTP server health check failed with status ${response.statusCode} for ${url}: ${body.slice(0, 200)}`))\n                return\n              }\n\n              resolve(undefined)\n            })\n          })\n\n          request.on(\"error\", (error) => reject(error))\n\n          request.setTimeout(timeoutMs, () => {\n            request.destroy(new Error(`HTTP server health check timed out after ${timeoutMs}ms for ${url}`))\n          })\n        })\n\n        return\n      } catch (error) {\n        if (attempt === maxAttempts || !this.isRetryableHealthCheckError(error)) {\n          throw error\n        }\n\n        await new Promise((resolve) => setTimeout(resolve, 100))\n      }\n    }\n  }\n\n  /**\n   * @param {unknown} error\n   * @returns {boolean}\n   */\n  isRetryableHealthCheckError(error) {\n    if (error && typeof error === \"object\" && \"code\" in error) {\n      if (error.code === \"ECONNRESET\") return true\n      if (error.code === \"ECONNREFUSED\") return true\n      if (error.code === \"ETIMEDOUT\") return true\n    }\n\n    const message = error instanceof Error ? error.message : String(error)\n\n    return message.includes(\"socket hang up\")\n  }\n}\n"]}
@@ -1,40 +1,3 @@
1
- /**
2
- * @typedef {object} SystemTestArgs
3
- * @property {string} [host] Hostname for the app server.
4
- * @property {number} [port] Port for the app server.
5
- * @property {string} [httpHost] Hostname for the static HTTP server.
6
- * @property {number} [httpPort] Port for the static HTTP server.
7
- * @property {string} [httpConnectHost] Hostname used by the driver to reach the HTTP server.
8
- * @property {boolean} [debug] Enable debug logging.
9
- * @property {(error: any) => boolean} [errorFilter] Filter for browser errors (return false to ignore).
10
- * @property {number} [clientWsPort] Port for the browser-command WebSocket server.
11
- * @property {number} [scoundrelPort] Port for the Scoundrel WebSocket server.
12
- * @property {Record<string, any>} [urlArgs] Query params appended to the root path.
13
- * @property {SystemTestDriverConfig} [driver] Driver configuration.
14
- */
15
- /**
16
- * @typedef {object} SystemTestDriverConfig
17
- * @property {"selenium"|"appium"} [type] Driver implementation to use.
18
- * @property {Record<string, any>} [options] Driver-specific options.
19
- */
20
- /**
21
- * @typedef {object} FindArgs
22
- * @property {number} [timeout] Override timeout for lookup.
23
- * @property {boolean | null} [visible] Whether to require elements to be visible (`true`) or hidden (`false`). Use `null` to disable visibility filtering.
24
- * @property {boolean} [scrollTo] Whether to scroll found elements into view before returning them.
25
- * @property {boolean} [useBaseSelector] Whether to scope by the base selector.
26
- */
27
- /**
28
- * @typedef {FindArgs & {withFallback?: boolean}} InteractArgs
29
- */
30
- /**
31
- * @typedef {object} WaitForNoSelectorArgs
32
- * @property {boolean} [useBaseSelector] Whether to scope by the base selector.
33
- */
34
- /**
35
- * @typedef {object} NotificationMessageArgs
36
- * @property {boolean} [dismiss] Whether to dismiss the notification after it appears.
37
- */
38
1
  export default class SystemTest extends Browser {
39
2
  static rootPath: string;
40
3
  /**
@@ -67,8 +30,8 @@ export default class SystemTest extends Browser {
67
30
  _clientWsPort: number;
68
31
  _httpHost: string;
69
32
  _httpPort: number;
70
- /** @type {(error: any) => boolean | undefined} */
71
- _errorFilter: (error: any) => boolean | undefined;
33
+ /** @type {((error: any) => boolean) | undefined} */
34
+ _errorFilter: ((error: any) => boolean) | undefined;
72
35
  _scoundrelPort: number;
73
36
  /** @type {WebSocketServer | undefined} */
74
37
  scoundrelWss: WebSocketServer | undefined;
@@ -96,8 +59,8 @@ export default class SystemTest extends Browser {
96
59
  }): boolean;
97
60
  _host: string;
98
61
  _port: number;
99
- _httpConnectHost: string;
100
- _urlArgs: Record<string, any>;
62
+ _httpConnectHost: string | undefined;
63
+ _urlArgs: Record<string, any> | undefined;
101
64
  _rootPath: string;
102
65
  /** @type {Record<number, object>} */
103
66
  _responses: Record<number, object>;
@@ -107,8 +70,8 @@ export default class SystemTest extends Browser {
107
70
  * @returns {void}
108
71
  */
109
72
  startScoundrel(): void;
110
- serverWebSocket: ServerWebSocket;
111
- server: Server;
73
+ serverWebSocket: ServerWebSocket | undefined;
74
+ server: Server | undefined;
112
75
  /**
113
76
  * @returns {Promise<void>}
114
77
  */
@@ -150,8 +113,8 @@ export default class SystemTest extends Browser {
150
113
  * @returns {Promise<void>}
151
114
  */
152
115
  start(): Promise<void>;
153
- currentUrl: string;
154
- systemTestHttpServer: SystemTestHttpServer;
116
+ currentUrl: string | undefined;
117
+ systemTestHttpServer: SystemTestHttpServer | undefined;
155
118
  /**
156
119
  * @returns {string}
157
120
  */
@@ -165,13 +128,13 @@ export default class SystemTest extends Browser {
165
128
  * @returns {Promise<void>}
166
129
  */
167
130
  waitForClientWebSocket(): Promise<void>;
168
- waitForClientWebSocketPromiseReject: (reason?: any) => void;
169
- waitForClientWebSocketPromiseResolve: (value: any) => void;
131
+ waitForClientWebSocketPromiseReject: ((reason?: any) => void) | undefined;
132
+ waitForClientWebSocketPromiseResolve: ((value: any) => void) | undefined;
170
133
  /**
171
- * Starts the web socket server
172
- * @returns {void}
134
+ * Starts the web socket server and waits for it to be ready.
135
+ * @returns {Promise<void>}
173
136
  */
174
- startWebSocketServer(): void;
137
+ startWebSocketServer(): Promise<void>;
175
138
  /**
176
139
  * Sets the on command callback
177
140
  * @param {function({type: string, data: Record<string, any>}): Promise<void>} callback
@@ -181,22 +144,17 @@ export default class SystemTest extends Browser {
181
144
  type: string;
182
145
  data: Record<string, any>;
183
146
  }) => Promise<void>): void;
184
- _onCommandCallback: (arg0: {
147
+ _onCommandCallback: ((arg0: {
185
148
  type: string;
186
149
  data: Record<string, any>;
187
- }) => Promise<void>;
150
+ }) => Promise<void>) | undefined;
188
151
  /**
189
152
  * Handles a command received from the browser
190
- * @param {{data: {message: string, backtrace?: string, type: string, value: any[]}}} args
153
+ * @param {{data: Record<string, any>}} args
191
154
  * @returns {Promise<any>}
192
155
  */
193
156
  onCommandReceived: ({ data }: {
194
- data: {
195
- message: string;
196
- backtrace?: string;
197
- type: string;
198
- value: any[];
199
- };
157
+ data: Record<string, any>;
200
158
  }) => Promise<any>;
201
159
  /**
202
160
  * Handles a new web socket connection
@@ -204,20 +162,15 @@ export default class SystemTest extends Browser {
204
162
  * @returns {Promise<void>}
205
163
  */
206
164
  onWebSocketConnection: (ws: WebSocket) => Promise<void>;
207
- ws: WebSocket;
165
+ ws: WebSocket | null | undefined;
208
166
  /** @returns {void} */
209
167
  onWebSocketClose: () => void;
210
168
  /**
211
169
  * Handles an error reported from the browser
212
- * @param {object} data
213
- * @param {string} data.message
214
- * @param {string} [data.backtrace]
170
+ * @param {Record<string, any>} data
215
171
  * @returns {void}
216
172
  */
217
- handleError(data: {
218
- message: string;
219
- backtrace?: string;
220
- }): void;
173
+ handleError(data: Record<string, any>): void;
221
174
  /**
222
175
  * Stops the system test
223
176
  * @returns {Promise<void>}
@@ -239,75 +192,75 @@ export type SystemTestArgs = {
239
192
  /**
240
193
  * Hostname for the app server.
241
194
  */
242
- host?: string;
195
+ host?: string | undefined;
243
196
  /**
244
197
  * Port for the app server.
245
198
  */
246
- port?: number;
199
+ port?: number | undefined;
247
200
  /**
248
201
  * Hostname for the static HTTP server.
249
202
  */
250
- httpHost?: string;
203
+ httpHost?: string | undefined;
251
204
  /**
252
205
  * Port for the static HTTP server.
253
206
  */
254
- httpPort?: number;
207
+ httpPort?: number | undefined;
255
208
  /**
256
209
  * Hostname used by the driver to reach the HTTP server.
257
210
  */
258
- httpConnectHost?: string;
211
+ httpConnectHost?: string | undefined;
259
212
  /**
260
213
  * Enable debug logging.
261
214
  */
262
- debug?: boolean;
215
+ debug?: boolean | undefined;
263
216
  /**
264
217
  * Filter for browser errors (return false to ignore).
265
218
  */
266
- errorFilter?: (error: any) => boolean;
219
+ errorFilter?: ((error: any) => boolean) | undefined;
267
220
  /**
268
221
  * Port for the browser-command WebSocket server.
269
222
  */
270
- clientWsPort?: number;
223
+ clientWsPort?: number | undefined;
271
224
  /**
272
225
  * Port for the Scoundrel WebSocket server.
273
226
  */
274
- scoundrelPort?: number;
227
+ scoundrelPort?: number | undefined;
275
228
  /**
276
229
  * Query params appended to the root path.
277
230
  */
278
- urlArgs?: Record<string, any>;
231
+ urlArgs?: Record<string, any> | undefined;
279
232
  /**
280
233
  * Driver configuration.
281
234
  */
282
- driver?: SystemTestDriverConfig;
235
+ driver?: SystemTestDriverConfig | undefined;
283
236
  };
284
237
  export type SystemTestDriverConfig = {
285
238
  /**
286
239
  * Driver implementation to use.
287
240
  */
288
- type?: "selenium" | "appium";
241
+ type?: "appium" | "selenium" | undefined;
289
242
  /**
290
243
  * Driver-specific options.
291
244
  */
292
- options?: Record<string, any>;
245
+ options?: Record<string, any> | undefined;
293
246
  };
294
247
  export type FindArgs = {
295
248
  /**
296
249
  * Override timeout for lookup.
297
250
  */
298
- timeout?: number;
251
+ timeout?: number | undefined;
299
252
  /**
300
253
  * Whether to require elements to be visible (`true`) or hidden (`false`). Use `null` to disable visibility filtering.
301
254
  */
302
- visible?: boolean | null;
255
+ visible?: boolean | null | undefined;
303
256
  /**
304
257
  * Whether to scroll found elements into view before returning them.
305
258
  */
306
- scrollTo?: boolean;
259
+ scrollTo?: boolean | undefined;
307
260
  /**
308
261
  * Whether to scope by the base selector.
309
262
  */
310
- useBaseSelector?: boolean;
263
+ useBaseSelector?: boolean | undefined;
311
264
  };
312
265
  export type InteractArgs = FindArgs & {
313
266
  withFallback?: boolean;
@@ -316,13 +269,13 @@ export type WaitForNoSelectorArgs = {
316
269
  /**
317
270
  * Whether to scope by the base selector.
318
271
  */
319
- useBaseSelector?: boolean;
272
+ useBaseSelector?: boolean | undefined;
320
273
  };
321
274
  export type NotificationMessageArgs = {
322
275
  /**
323
276
  * Whether to dismiss the notification after it appears.
324
277
  */
325
- dismiss?: boolean;
278
+ dismiss?: boolean | undefined;
326
279
  };
327
280
  import Browser from "./browser.js";
328
281
  import { WebSocketServer } from "ws";