swc/bundler/tests/.cache/deno/bc2a46e068dbf4e6aab4080efb335923e28821cc.ts
강동윤 bbaf619f63
fix(bundler): Fix bugs (#1437)
swc_bundler:
 - [x] Fix wrapped esms. (denoland/deno#9307)
 - [x] Make test secure.
2021-03-02 17:33:03 +09:00

97 lines
2.6 KiB
TypeScript

// Loaded from https://deno.land/x/mongo@v0.20.0/src/protocol/protocol.ts
import { assert, BufReader, Deferred, deferred } from "../../deps.ts";
import { MongoError, MongoErrorInfo } from "../error.ts";
import { Document } from "../types.ts";
import { handshake } from "./handshake.ts";
import { parseHeader } from "./header.ts";
import { deserializeMessage, Message, serializeMessage } from "./message.ts";
type Socket = Deno.Reader & Deno.Writer;
let nextRequestId = 0;
export class WireProtocol {
#socket: Socket;
#pending = false;
#pendingOps: Map<number, Deferred<Message>> = new Map();
#reader: BufReader;
#connectionId: number = 0;
constructor(socket: Socket) {
this.#socket = socket;
this.#reader = new BufReader(this.#socket);
}
async connect() {
const { connectionId } = await handshake(this);
this.#connectionId = connectionId;
}
async commandSingle<T = Document>(db: string, body: Document): Promise<T> {
const [doc] = await this.command<MongoErrorInfo | T>(db, body);
const maybeError = doc as MongoErrorInfo;
if (maybeError.ok === 0) {
throw new MongoError(maybeError);
}
return doc as T;
}
async command<T = Document>(db: string, body: Document): Promise<T[]> {
const requestId = nextRequestId++;
const chunks = serializeMessage({
requestId,
responseTo: 0,
sections: [
{
document: {
...body,
$db: db,
},
},
],
});
for (const chunk of chunks) {
await Deno.writeAll(this.#socket, chunk);
}
this.#pendingOps.set(requestId, deferred());
this.receive();
const message = await this.#pendingOps.get(requestId);
let documents: T[] = [];
message?.sections.forEach((section) => {
if ("document" in section) {
documents.push(section.document as T);
} else {
documents = documents.concat(section.documents as T[]);
}
});
return documents;
}
private async receive() {
if (this.#pending) return;
this.#pending = true;
while (this.#pendingOps.size > 0) {
const headerBuffer = await this.#reader.readFull(new Uint8Array(16));
assert(headerBuffer);
const header = parseHeader(headerBuffer!);
const bodyBuffer = await this.#reader.readFull(
new Uint8Array(header.messageLength - 16),
);
assert(bodyBuffer);
const reply = deserializeMessage(header, bodyBuffer!);
const pendingMessage = this.#pendingOps.get(header.responseTo);
this.#pendingOps.delete(header.responseTo);
pendingMessage?.resolve(reply);
}
this.#pending = false;
}
}