@tai-kun/surrealdb
Version:
The SurrealDB SDK for JavaScript
461 lines (459 loc) • 46 kB
JavaScript
var __defProp = Object.defineProperty;
var __name = (target, value) => __defProp(target, "name", { value, configurable: true });
// src/engines/websocket/engine.ts
import {
EngineAbc,
processQueryRequest
} from "../../engine/index.mjs";
import {
ConnectionUnavailableError,
MissingNamespaceError,
RpcResponseError,
ServerResponseError,
SurrealTypeError,
unreachable,
WebSocketEngineError
} from "../../errors/index.mjs";
import {
getTimeoutSignal,
isLiveResult,
isRpcResponse,
Serial,
StatefulPromise,
throwIfAborted
} from "../../utils/index.mjs";
var _WebSocketEngine = class _WebSocketEngine extends EngineAbc {
constructor(config) {
super(config);
this.name = "websocket";
this.ws = null;
this.id = new Serial();
const {
pingInterval,
createWebSocket
} = config;
this.newWs = createWebSocket;
this.pingInterval = typeof pingInterval === "function" ? pingInterval : /* @__PURE__ */ ((ms) => () => ms)(Math.max(3e3, pingInterval ?? 3e4));
}
async connect({ endpoint, signal }) {
throwIfAborted(signal);
const conn = this.getConnectionInfo();
if (conn.state === "open") {
return;
}
if (conn.state !== "closed") {
unreachable(conn);
}
await this.transition(
{
state: "connecting",
endpoint
},
() => "closed"
);
const ws = await this.newWs(new URL(endpoint), this.fmt.wsProtocols);
ws.addEventListener("error", (evt) => {
if (
/* __TAI_KUN__SURREALDB__DIAGNOSTICS_CHANNEL__ */
false
) {
channel.publish("websocket:ws:error", evt);
}
this.ee.emit(
"error",
new WebSocketEngineError(
3150,
// イベントに message プロパティが含まれているかどうかは分かりません。
//
// - Mozilla: https://developer.mozilla.org/en-US/docs/Web/API/WebSocket/error_event#event_type
// - Node.js:
// - undici: https://github.com/nodejs/undici/blob/v6.18.1/types/websocket.d.ts#L127-L133
// - ws: https://github.com/DefinitelyTyped/DefinitelyTyped/blob/master/types/ws/index.d.ts#L289-L294
// - Deno: https://github.com/denoland/deno/blob/v1.43.6/ext/web/02_event.js#L1064-L1133
"message" in evt ? String(evt.message) : 'The "error" event was caught.',
{
..."error" in evt ? { cause: evt.error } : {},
fatal: true
}
)
);
});
ws.addEventListener("close", async (evt) => {
if (
/* __TAI_KUN__SURREALDB__DIAGNOSTICS_CHANNEL__ */
false
) {
channel.publish("websocket:ws:close", evt);
}
this.ws = null;
this.id.reset();
switch (evt.code) {
// 正常系
case 1e3:
// Normal Closure
// 予約済み
case 1004:
// Reserved
case 1005:
// No Status
case 1015:
// TLS Handshake
// 早期の切断に由来するエラーコードをエラーとして扱いません。
case 1001:
// Going Away
case 1006:
break;
case 1002:
// Protocol Error
case 1003:
// Unsupported Data
case 1007:
// Invalid Frame Payload Data
case 1008:
// Policy Violation
case 1009:
// Message Too Big
case 1010:
// Mandatory Extension
case 1011:
// Internal Server Error
case 1012:
// Service Restart
case 1013:
// Try Again Later
case 1014:
this.ee.emit(
"error",
new WebSocketEngineError(evt.code, evt.reason)
);
}
try {
await this.transition("closed", () => "closed");
} catch (e) {
this.ee.emit(
"error",
new WebSocketEngineError(
3154,
'An error occurred within the handler for the "close" event.',
{
cause: e,
fatal: false
}
)
);
}
});
ws.addEventListener("open", async (evt) => {
if (
/* __TAI_KUN__SURREALDB__DIAGNOSTICS_CHANNEL__ */
false
) {
channel.publish("websocket:ws:open", evt);
}
try {
this.ws = ws;
await this.transition(
{
state: "open",
endpoint
},
() => {
this.ws = null;
return "closed";
}
);
} catch (e) {
this.ee.emit(
"error",
new WebSocketEngineError(
// open イベントハンドラー内で発生したエラーを、
// カスタムエラーコード 3151 として報告します。
3151,
'An error occurred within the handler for the "open" event.',
{
cause: e,
fatal: true
}
)
);
}
});
ws.addEventListener("message", async (evt) => {
if (
/* __TAI_KUN__SURREALDB__DIAGNOSTICS_CHANNEL__ */
false
) {
channel.publish("websocket:ws:message", evt);
}
try {
const data = evt.data instanceof Blob ? await evt.data.arrayBuffer() : evt.data;
const rpcResp = this.fmt.decodeSync(data);
if (!isRpcResponse(rpcResp)) {
throw new ServerResponseError("Missing rpc response.", {
// cause: (evt.data as Buffer).map(v => v).join(),
cause: {
endpoint: endpoint.href,
response: rpcResp
// TODO(tai-kun): 情報を追加
// ...cause,
}
});
}
if ("id" in rpcResp) {
this.ee.emit(`rpc_${rpcResp.id}`, rpcResp);
} else if ("result" in rpcResp) {
if (!isLiveResult(rpcResp.result)) {
throw new ServerResponseError("Missing live result", {
cause: {
endpoint: endpoint.href,
liveResult: rpcResp.result
// TODO(tai-kun): 情報を追加
// ...cause,
}
});
}
const { id, ...payload } = rpcResp.result;
this.ee.emit(`live_${String(id)}`, payload);
} else {
throw new RpcResponseError(rpcResp, {
cause: {
endpoint: endpoint.href
}
});
}
} catch (e) {
this.ee.emit(
"error",
new WebSocketEngineError(
// message イベントハンドラー内で発生したエラーを、
// カスタムエラーコード 3152 として報告します。
3152,
'An error occurred within the handler for the "message" event.',
{
cause: e,
fatal: false
}
)
);
}
});
const openPromise = this.ee.once("open", { signal });
const errorPromise = this.ee.once("error", { signal });
try {
const [result] = await Promise.race([
openPromise,
errorPromise
]);
if (result instanceof Error) {
throw result;
}
if ("error" in result) {
throw result.error;
}
} finally {
await Promise.all([
openPromise.cancel(),
errorPromise.cancel()
]);
}
const createPromise = /* @__PURE__ */ __name(() => {
let i, int, res, pro = new StatefulPromise((r) => {
i = setTimeout(res = r, int = this.pingInterval(), false);
});
return Object.assign(pro, {
interval: int,
// 現在未使用
cancel() {
if (pro.state === "pending") {
clearTimeout(i);
res(true);
}
}
});
}, "createPromise");
let promise = createPromise();
const pinger = (async () => {
while (true) {
const canceled = await promise;
if (canceled) {
break;
}
try {
if (
/* __TAI_KUN__SURREALDB__DIAGNOSTICS_CHANNEL__ */
false
) {
channel.publish("websocket:ping", { type: "ping" });
}
const timeout = 5e3;
const rpcResp = await this.rpc({
signal: getTimeoutSignal(timeout),
request: {
method: "ping"
}
});
if (rpcResp.error) {
throw new RpcResponseError(rpcResp, {
cause: {
endpoint: endpoint.href
}
});
}
if (
/* __TAI_KUN__SURREALDB__DIAGNOSTICS_CHANNEL__ */
false
) {
channel.publish("websocket:pong", { type: "pong" });
}
} catch (e) {
this.ee.emit(
"error",
new WebSocketEngineError(
// ping メッセージの送信に失敗したエラーを、
// カスタムエラーコード 3153 として報告します。
3153,
"Failed to send a ping message.",
{
cause: e,
fatal: false
}
)
);
} finally {
promise = createPromise();
}
}
})();
this.ee.on(
"closing",
/* @__PURE__ */ __name(async function stop() {
this.off("closing", stop);
promise.cancel();
await pinger;
}, "stop")
);
}
async close({ signal }) {
throwIfAborted(signal);
const conn = this.getConnectionInfo();
if (conn.state === "closed") {
return;
}
if (conn.state !== "open") {
unreachable(conn);
}
await this.transition(
{
state: "closing",
endpoint: conn.endpoint
},
() => ({
state: "closing",
endpoint: conn.endpoint
})
);
const promiseClosed = this.ee.once("closed", { signal });
if (this.ws && this.ws.readyState !== 3 && this.ws.readyState !== 2) {
this.ws.close();
} else {
this.ee.emit("closed", {
state: "closed"
});
}
const [result] = await promiseClosed;
if ("error" in result) {
throw result.error;
}
}
async rpc({ request, signal }) {
const conn = this.getConnectionInfo();
if (!this.ws || conn.state !== "open") {
throw new ConnectionUnavailableError({
cause: "The connection is not established via the .connect() method."
});
}
switch (request.method) {
case "use": {
let { namespace, database } = conn;
const [ns, db] = request.params;
if (ns !== void 0) {
namespace = ns;
}
if (db !== void 0) {
database = db;
}
if (namespace === null && database !== null) {
throw new MissingNamespaceError(database);
}
{
if (ns === void 0 && conn.namespace !== null) {
request = {
method: request.method,
params: [conn.namespace, request.params[1]]
};
}
if (db === void 0 && conn.database !== null) {
request = {
method: request.method,
params: [request.params[0], conn.database]
};
}
}
break;
}
case "query": {
request = processQueryRequest(request);
break;
}
}
const id = `${request.method}_${this.id.next()}`;
const body = this.fmt.encodeSync({
id,
method: request.method,
params: request.params || []
});
if (typeof body !== "string" && !(body instanceof Uint8Array)) {
throw new SurrealTypeError(["String", "Uint8Array"], body);
}
const resp = this.ee.once(`rpc_${id}`, { signal });
this.ws.send(body);
const [rawResp] = await resp;
const rpcResp = rawResp;
if ("result" in rpcResp) {
const rpc = {
method: request.method,
params: request.params,
result: rpcResp.result
};
switch (rpc.method) {
case "use": {
const [ns, db] = rpc.params;
if (ns !== void 0) {
this.namespace = ns;
}
if (db !== void 0) {
this.database = db;
}
break;
}
case "signin":
case "signup":
this.token = rpc.result;
break;
case "authenticate":
[this.token] = rpc.params;
break;
case "invalidate":
this.token = null;
break;
}
}
return rpcResp;
}
};
__name(_WebSocketEngine, "WebSocketEngine");
var WebSocketEngine = _WebSocketEngine;
export {
WebSocketEngine as default
};
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/engines/websocket/engine.ts"],
  "sourcesContent": ["import {\n  type CloseArgs,\n  type ConnectArgs,\n  EngineAbc,\n  type EngineAbcConfig,\n  type EngineEventMap,\n  processQueryRequest,\n  type RpcArgs,\n} from \"@tai-kun/surrealdb/engine\";\nimport {\n  ConnectionUnavailableError,\n  MissingNamespaceError,\n  RpcResponseError,\n  ServerResponseError,\n  SurrealTypeError,\n  unreachable,\n  WebSocketEngineError,\n} from \"@tai-kun/surrealdb/errors\";\nimport type {\n  BidirectionalRpcResponse,\n  RpcParams,\n  RpcQueryRequest,\n  RpcResult,\n} from \"@tai-kun/surrealdb/types\";\nimport {\n  channel,\n  getTimeoutSignal,\n  isLiveResult,\n  isRpcResponse,\n  Serial,\n  StatefulPromise,\n  type TaskEmitter,\n  throwIfAborted,\n} from \"@tai-kun/surrealdb/utils\";\nimport type { Promisable } from \"type-fest\";\nimport type { WebSocket as WsWebSocket } from \"ws\";\n\ntype InferWebSocket<T> = T extends\n  { readonly WebSocket: new(...args: any) => infer W } ? W : never;\n\ntype NativeWebSocket = InferWebSocket<\n  | typeof global\n  | typeof window\n  | typeof self\n>;\n\ntype WebSocket = WsWebSocket | NativeWebSocket; // isows \u306E\u578B\u5B9A\u7FA9\u306F\u3053\u308C\u3088\u308A\u72ED\u3044\u3002\n\nexport type CreateWebSocket = (\n  address: URL,\n  protocols: string[],\n) => Promisable<WebSocket>;\n\nexport interface WebSocketEngineConfig extends EngineAbcConfig {\n  readonly createWebSocket: CreateWebSocket;\n  readonly pingInterval?: number | (() => number) | undefined;\n}\n\nexport default class WebSocketEngine extends EngineAbc {\n  readonly name = \"websocket\";\n\n  protected ws: WebSocket | null = null;\n  protected readonly id = new Serial();\n  protected readonly newWs: CreateWebSocket;\n  protected readonly pingInterval: () => number;\n\n  constructor(config: WebSocketEngineConfig) {\n    super(config);\n    const {\n      pingInterval,\n      createWebSocket,\n    } = config;\n    this.newWs = createWebSocket;\n    this.pingInterval = typeof pingInterval === \"function\"\n      ? pingInterval\n      : (ms => () => ms)(Math.max(3_000, pingInterval ?? 30_000));\n  }\n\n  async connect({ endpoint, signal }: ConnectArgs): Promise<void> {\n    throwIfAborted(signal);\n    const conn = this.getConnectionInfo();\n\n    if (conn.state === \"open\") {\n      return;\n    }\n\n    if (conn.state !== \"closed\") {\n      unreachable(conn as never);\n    }\n\n    await this.transition(\n      {\n        state: \"connecting\",\n        endpoint,\n      },\n      () => \"closed\",\n    );\n    const ws = await this.newWs(new URL(endpoint), this.fmt.wsProtocols);\n    ws.addEventListener(\"error\", evt => {\n      if (/* __TAI_KUN__SURREALDB__DIAGNOSTICS_CHANNEL__ */ false) {\n        channel.publish(\"websocket:ws:error\", evt);\n      }\n\n      this.ee.emit(\n        \"error\",\n        new WebSocketEngineError(\n          3150,\n          // \u30A4\u30D9\u30F3\u30C8\u306B message \u30D7\u30ED\u30D1\u30C6\u30A3\u304C\u542B\u307E\u308C\u3066\u3044\u308B\u304B\u3069\u3046\u304B\u306F\u5206\u304B\u308A\u307E\u305B\u3093\u3002\n          //\n          // - Mozilla: https://developer.mozilla.org/en-US/docs/Web/API/WebSocket/error_event#event_type\n          // - Node.js:\n          //     - undici: https://github.com/nodejs/undici/blob/v6.18.1/types/websocket.d.ts#L127-L133\n          //     - ws: https://github.com/DefinitelyTyped/DefinitelyTyped/blob/master/types/ws/index.d.ts#L289-L294\n          // - Deno: https://github.com/denoland/deno/blob/v1.43.6/ext/web/02_event.js#L1064-L1133\n          \"message\" in evt\n            ? String(evt.message)\n            : \"The \\\"error\\\" event was caught.\",\n          {\n            ...(\"error\" in evt ? { cause: evt.error } : {}),\n            fatal: true,\n          },\n        ),\n      );\n    });\n    ws.addEventListener(\"close\", async evt => {\n      if (/* __TAI_KUN__SURREALDB__DIAGNOSTICS_CHANNEL__ */ false) {\n        channel.publish(\"websocket:ws:close\", evt);\n      }\n\n      this.ws = null;\n      this.id.reset();\n\n      switch (evt.code) {\n        // \u6B63\u5E38\u7CFB\n        case 1000: // Normal Closure\n        // \u4E88\u7D04\u6E08\u307F\n        case 1004: // Reserved\n        case 1005: // No Status\n        case 1015: // TLS Handshake\n        // \u65E9\u671F\u306E\u5207\u65AD\u306B\u7531\u6765\u3059\u308B\u30A8\u30E9\u30FC\u30B3\u30FC\u30C9\u3092\u30A8\u30E9\u30FC\u3068\u3057\u3066\u6271\u3044\u307E\u305B\u3093\u3002\n        case 1001: // Going Away\n        case 1006: // Abnormal Closure (\u304B\u3064\u3001\u4E88\u7D04\u6E08\u307F)\n          break;\n\n        case 1002: // Protocol Error\n        case 1003: // Unsupported Data\n        case 1007: // Invalid Frame Payload Data\n        case 1008: // Policy Violation\n        case 1009: // Message Too Big\n        case 1010: // Mandatory Extension\n        case 1011: // Internal Server Error\n        case 1012: // Service Restart\n        case 1013: // Try Again Later\n        case 1014: // Bad Gateway\n          this.ee.emit(\n            \"error\",\n            new WebSocketEngineError(evt.code, evt.reason),\n          );\n      }\n\n      try {\n        await this.transition(\"closed\", () => \"closed\");\n      } catch (e) {\n        this.ee.emit(\n          \"error\",\n          new WebSocketEngineError(\n            3154,\n            \"An error occurred within the handler for the \\\"close\\\" event.\",\n            {\n              cause: e,\n              fatal: false,\n            },\n          ),\n        );\n      }\n    });\n    ws.addEventListener(\"open\", async evt => {\n      if (/* __TAI_KUN__SURREALDB__DIAGNOSTICS_CHANNEL__ */ false) {\n        channel.publish(\"websocket:ws:open\", evt);\n      }\n\n      try {\n        this.ws = ws;\n        await this.transition(\n          {\n            state: \"open\",\n            endpoint,\n          },\n          () => {\n            this.ws = null;\n\n            return \"closed\";\n          },\n        );\n      } catch (e) {\n        this.ee.emit(\n          \"error\",\n          new WebSocketEngineError(\n            // open \u30A4\u30D9\u30F3\u30C8\u30CF\u30F3\u30C9\u30E9\u30FC\u5185\u3067\u767A\u751F\u3057\u305F\u30A8\u30E9\u30FC\u3092\u3001\n            // \u30AB\u30B9\u30BF\u30E0\u30A8\u30E9\u30FC\u30B3\u30FC\u30C9 3151 \u3068\u3057\u3066\u5831\u544A\u3057\u307E\u3059\u3002\n            3151,\n            \"An error occurred within the handler for the \\\"open\\\" event.\",\n            {\n              cause: e,\n              fatal: true,\n            },\n          ),\n        );\n      }\n    });\n    ws.addEventListener(\"message\", async evt => {\n      if (/* __TAI_KUN__SURREALDB__DIAGNOSTICS_CHANNEL__ */ false) {\n        channel.publish(\"websocket:ws:message\", evt);\n      }\n\n      try {\n        // TODO(tai-kun): Blob \u306E .stream \u3092\u6D3B\u7528\u3067\u304D\u308B\u3088\u3046\u306B\u3002CBOR \u306E\u975E\u540C\u671F\u30C7\u30B3\u30FC\u30C9\u3092\u5B9F\u88C5\n        // \u3057\u305F\u3051\u3069\u3001\u30DC\u30C7\u30A3\u30B5\u30A4\u30BA\u306B\u3088\u3063\u3066\u306F\u540C\u671F\u30C7\u30B3\u30FC\u30C9\u306E\u65B9\u304C\u9AD8\u901F\u3060\u3068\u601D\u3046\u306E\u3067\u3001\u305D\u306E\u3078\u3093\u3069\u3046\u3059\u308B\u304B\u691C\u8A0E\n\n        // Node.js v22 \u3068 ws v8.18.0 \u4EE5\u964D\u306F Blob \u3082\u6765\u308B\u3002\n        const data = evt.data instanceof Blob\n          ? await (evt.data as Blob).arrayBuffer()\n          : evt.data;\n        const rpcResp = this.fmt.decodeSync(data);\n\n        if (!isRpcResponse(rpcResp)) {\n          throw new ServerResponseError(\"Missing rpc response.\", {\n            // cause: (evt.data as Buffer).map(v => v).join(),\n            cause: {\n              endpoint: endpoint.href,\n              response: rpcResp,\n              // TODO(tai-kun): \u60C5\u5831\u3092\u8FFD\u52A0\n              // ...cause,\n            },\n          });\n        }\n\n        if (\"id\" in rpcResp) {\n          this.ee.emit(`rpc_${rpcResp.id}`, rpcResp);\n        } else if (\"result\" in rpcResp) {\n          if (!isLiveResult(rpcResp.result)) {\n            throw new ServerResponseError(\"Missing live result\", {\n              cause: {\n                endpoint: endpoint.href,\n                liveResult: rpcResp.result,\n                // TODO(tai-kun): \u60C5\u5831\u3092\u8FFD\u52A0\n                // ...cause,\n              },\n            });\n          }\n\n          const { id, ...payload } = rpcResp.result;\n\n          this.ee.emit(`live_${String(id)}`, payload);\n        } else {\n          throw new RpcResponseError(rpcResp, {\n            cause: {\n              endpoint: endpoint.href,\n            },\n          });\n        }\n      } catch (e) {\n        this.ee.emit(\n          \"error\",\n          new WebSocketEngineError(\n            // message \u30A4\u30D9\u30F3\u30C8\u30CF\u30F3\u30C9\u30E9\u30FC\u5185\u3067\u767A\u751F\u3057\u305F\u30A8\u30E9\u30FC\u3092\u3001\n            // \u30AB\u30B9\u30BF\u30E0\u30A8\u30E9\u30FC\u30B3\u30FC\u30C9 3152 \u3068\u3057\u3066\u5831\u544A\u3057\u307E\u3059\u3002\n            3152,\n            \"An error occurred within the handler for the \\\"message\\\" event.\",\n            {\n              cause: e,\n              fatal: false,\n            },\n          ),\n        );\n      }\n    });\n\n    const openPromise = this.ee.once(\"open\", { signal });\n    const errorPromise = this.ee.once(\"error\", { signal });\n\n    try {\n      const [result] = await Promise.race([\n        openPromise,\n        errorPromise,\n      ]);\n\n      if (result instanceof Error) {\n        throw result;\n      }\n\n      if (\"error\" in result) {\n        throw result.error;\n      }\n    } finally {\n      await Promise.all([\n        openPromise.cancel(),\n        errorPromise.cancel(),\n      ]);\n    }\n\n    const createPromise = () => {\n      let i: any,\n        int: number,\n        res: (v: boolean) => void,\n        pro = new StatefulPromise<boolean>(r => {\n          i = setTimeout(res = r, int = this.pingInterval(), false);\n        });\n\n      return Object.assign(pro, {\n        interval: int!, // \u73FE\u5728\u672A\u4F7F\u7528\n        cancel() {\n          if (pro.state === \"pending\") {\n            clearTimeout(i);\n            res(true);\n          }\n        },\n      });\n    };\n    let promise = createPromise();\n    const pinger = (async () => {\n      while (true) {\n        const canceled = await promise;\n\n        if (canceled) {\n          break;\n        }\n\n        try {\n          if (/* __TAI_KUN__SURREALDB__DIAGNOSTICS_CHANNEL__ */ false) {\n            channel.publish(\"websocket:ping\", { type: \"ping\" });\n          }\n\n          const timeout = 5_000;\n          const rpcResp = await this.rpc({\n            signal: getTimeoutSignal(timeout),\n            request: {\n              method: \"ping\",\n            },\n          });\n\n          if (rpcResp.error) {\n            throw new RpcResponseError(rpcResp, {\n              cause: {\n                endpoint: endpoint.href,\n              },\n            });\n          }\n\n          if (/* __TAI_KUN__SURREALDB__DIAGNOSTICS_CHANNEL__ */ false) {\n            channel.publish(\"websocket:pong\", { type: \"pong\" });\n          }\n        } catch (e) {\n          this.ee.emit(\n            \"error\",\n            new WebSocketEngineError(\n              // ping \u30E1\u30C3\u30BB\u30FC\u30B8\u306E\u9001\u4FE1\u306B\u5931\u6557\u3057\u305F\u30A8\u30E9\u30FC\u3092\u3001\n              // \u30AB\u30B9\u30BF\u30E0\u30A8\u30E9\u30FC\u30B3\u30FC\u30C9 3153 \u3068\u3057\u3066\u5831\u544A\u3057\u307E\u3059\u3002\n              3153,\n              \"Failed to send a ping message.\",\n              {\n                cause: e,\n                fatal: false,\n              },\n            ),\n          );\n        } finally {\n          promise = createPromise();\n        }\n      }\n    })();\n\n    this.ee.on(\n      \"closing\",\n      async function stop(this: TaskEmitter<EngineEventMap>) {\n        this.off(\"closing\", stop);\n        promise.cancel();\n        await pinger;\n      },\n    );\n  }\n\n  async close({ signal }: CloseArgs): Promise<void> {\n    throwIfAborted(signal);\n    const conn = this.getConnectionInfo();\n\n    if (conn.state === \"closed\") {\n      return;\n    }\n\n    if (conn.state !== \"open\") {\n      unreachable(conn as never);\n    }\n\n    await this.transition(\n      {\n        state: \"closing\",\n        endpoint: conn.endpoint,\n      },\n      () => ({\n        state: \"closing\",\n        endpoint: conn.endpoint,\n      }),\n    );\n    const promiseClosed = this.ee.once(\"closed\", { signal });\n\n    if (\n      this.ws\n      && this.ws.readyState !== 3 // WebSocket.CLOSED\n      && this.ws.readyState !== 2 // WebSocket.CLOSING\n    ) {\n      this.ws.close();\n    } else {\n      this.ee.emit(\"closed\", {\n        state: \"closed\",\n      });\n    }\n\n    const [result] = await promiseClosed;\n\n    if (\"error\" in result) {\n      throw result.error;\n    }\n  }\n\n  async rpc({ request, signal }: RpcArgs): Promise<BidirectionalRpcResponse> {\n    // \u63A5\u7D9A\u60C5\u5831\u306E\u30B9\u30CA\u30C3\u30D7\u30B7\u30E7\u30C3\u30C8\u3092\u53D6\u5F97\u3057\u307E\u3059\u3002\n    // \u4EE5\u964D\u3001\u63A5\u7D9A\u60C5\u5831\u3092\u53C2\u7167\u3059\u308B\u969B\u306F\u3053\u308C\u3092\u4F7F\u7528\u3057\u307E\u3059\u3002\n    const conn = this.getConnectionInfo();\n\n    if (!this.ws || conn.state !== \"open\") {\n      throw new ConnectionUnavailableError({\n        cause: \"The connection is not established via the .connect() method.\",\n      });\n    }\n\n    switch (request.method) {\n      case \"use\": {\n        let { namespace, database } = conn;\n        const [ns, db] = request.params;\n\n        if (ns !== undefined) {\n          namespace = ns;\n        }\n\n        if (db !== undefined) {\n          database = db;\n        }\n\n        if (namespace === null && database !== null) {\n          throw new MissingNamespaceError(database);\n        }\n\n        // - JSON \u3067\u306F null \u3068 undefined \u3092\u533A\u5225\u3057\u306A\u3044\u3051\u3069\u3001SurealDB \u306F\u533A\u5225\u3059\u308B\u3002\n        // - null \u306F\u9078\u629E\u4E2D\u306E\u540D\u524D\u7A7A\u9593\u307E\u305F\u306F\u30C7\u30FC\u30BF\u30D9\u30FC\u30B9\u3092\u672A\u9078\u629E\u306B\u3059\u308B\u306E\u3067\u3001undefined \u3092\u4F7F\u3048\u308B\n        //   CBOR \u30D5\u30A9\u30FC\u30DE\u30C3\u30BF\u30FC\u3068\u540C\u3058\u3088\u3046\u306A\u4F7F\u3044\u65B9\u3092 JSON \u30D5\u30A9\u30FC\u30DE\u30C3\u30BF\u30FC\u3067\u3059\u308B\u3068\u4E8B\u6545\u308B\u53EF\u80FD\u6027\u304C\u9AD8\u3044\u3002\n        // \u3068\u3044\u3046\u52D5\u6A5F\u304C\u3042\u3063\u3066\u3001undefined \u3092\u6587\u5B57\u5217\u306B\u7F6E\u304D\u63DB\u3048\u308B\u3088\u3046\u306B\u52AA\u529B\u3059\u308B\u3002\n        {\n          if (ns === undefined && conn.namespace !== null) {\n            request = {\n              method: request.method,\n              params: [conn.namespace, request.params[1]],\n            };\n          }\n\n          if (db === undefined && conn.database !== null) {\n            request = {\n              method: request.method,\n              params: [request.params[0], conn.database],\n            };\n          }\n        }\n\n        break;\n      }\n\n      case \"query\": {\n        request = processQueryRequest(request) as RpcQueryRequest;\n        break;\n      }\n    }\n\n    const id: BidirectionalRpcResponse[\"id\"] =\n      `${request.method}_${this.id.next()}`;\n    const body: unknown = this.fmt.encodeSync({\n      id,\n      method: request.method,\n      params: request.params || [],\n    });\n\n    if (typeof body !== \"string\" && !(body instanceof Uint8Array)) {\n      throw new SurrealTypeError([\"String\", \"Uint8Array\"], body);\n    }\n\n    const resp = this.ee.once(`rpc_${id}`, { signal });\n    this.ws.send(body);\n    const [rawResp] = await resp;\n    // `rawResp` \u306F message \u30A4\u30D9\u30F3\u30C8\u30CF\u30F3\u30C9\u30E9\u30FC\u5185\u3067\u691C\u8A3C\u6E08\u307F\u306A\u306E\u3067\u3001\u3053\u3053\u3067\u306F\u30AD\u30E3\u30B9\u30C8\u3059\u308B\u3060\u3051\u3002\n    const rpcResp = rawResp as BidirectionalRpcResponse;\n\n    if (\"result\" in rpcResp) {\n      const rpc = {\n        method: request.method,\n        params: request.params,\n        result: rpcResp.result,\n      } as {\n        [M in (typeof request)[\"method\"]]: {\n          method: M;\n          params: RpcParams<M>;\n          result: RpcResult<M>;\n        };\n      }[(typeof request)[\"method\"]];\n\n      switch (rpc.method) {\n        case \"use\": {\n          const [ns, db] = rpc.params;\n\n          if (ns !== undefined) {\n            this.namespace = ns;\n          }\n\n          if (db !== undefined) {\n            this.database = db;\n          }\n\n          break;\n        }\n\n        case \"signin\":\n        case \"signup\":\n          this.token = rpc.result;\n          break;\n\n        case \"authenticate\":\n          [this.token] = rpc.params;\n          break;\n\n        case \"invalidate\":\n          this.token = null;\n          break;\n      }\n    }\n\n    return rpcResp;\n  }\n}\n"],
  "mappings": "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",
  "names": []
}
