1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-10 16:11:13 -05:00

fix: make WebSocket.send() exclusive (#3885)

This commit is contained in:
Yusuke Sakurai 2020-02-06 22:42:32 +09:00 committed by GitHub
parent ed680552a2
commit 699d10bd9e
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 138 additions and 52 deletions

View file

@ -10,6 +10,7 @@ import { readLong, readShort, sliceLongToBytes } from "../io/ioutil.ts";
import { Sha1 } from "./sha1.ts"; import { Sha1 } from "./sha1.ts";
import { writeResponse } from "../http/server.ts"; import { writeResponse } from "../http/server.ts";
import { TextProtoReader } from "../textproto/mod.ts"; import { TextProtoReader } from "../textproto/mod.ts";
import { Deferred, deferred } from "../util/async.ts";
export enum OpCode { export enum OpCode {
Continue = 0x0, Continue = 0x0,
@ -193,21 +194,30 @@ function createMask(): Uint8Array {
} }
class WebSocketImpl implements WebSocket { class WebSocketImpl implements WebSocket {
readonly conn: Conn;
private readonly mask?: Uint8Array; private readonly mask?: Uint8Array;
private readonly bufReader: BufReader; private readonly bufReader: BufReader;
private readonly bufWriter: BufWriter; private readonly bufWriter: BufWriter;
private sendQueue: Array<{
frame: WebSocketFrame;
d: Deferred<void>;
}> = [];
constructor( constructor({
readonly conn: Conn, conn,
opts: { bufReader,
bufReader?: BufReader; bufWriter,
bufWriter?: BufWriter; mask
mask?: Uint8Array; }: {
} conn: Conn;
) { bufReader?: BufReader;
this.mask = opts.mask; bufWriter?: BufWriter;
this.bufReader = opts.bufReader || new BufReader(conn); mask?: Uint8Array;
this.bufWriter = opts.bufWriter || new BufWriter(conn); }) {
this.conn = conn;
this.mask = mask;
this.bufReader = bufReader || new BufReader(conn);
this.bufWriter = bufWriter || new BufWriter(conn);
} }
async *receive(): AsyncIterableIterator<WebSocketEvent> { async *receive(): AsyncIterableIterator<WebSocketEvent> {
@ -250,14 +260,11 @@ class WebSocketImpl implements WebSocket {
yield { code, reason }; yield { code, reason };
return; return;
case OpCode.Ping: case OpCode.Ping:
await writeFrame( await this.enqueue({
{ opcode: OpCode.Pong,
opcode: OpCode.Pong, payload: frame.payload,
payload: frame.payload, isLastFrame: true
isLastFrame: true });
},
this.bufWriter
);
yield ["ping", frame.payload] as WebSocketPingEvent; yield ["ping", frame.payload] as WebSocketPingEvent;
break; break;
case OpCode.Pong: case OpCode.Pong:
@ -268,6 +275,27 @@ class WebSocketImpl implements WebSocket {
} }
} }
private dequeue(): void {
const [e] = this.sendQueue;
if (!e) return;
writeFrame(e.frame, this.bufWriter)
.then(() => e.d.resolve())
.catch(e => e.d.reject(e))
.finally(() => {
this.sendQueue.shift();
this.dequeue();
});
}
private enqueue(frame: WebSocketFrame): Promise<void> {
const d = deferred<void>();
this.sendQueue.push({ d, frame });
if (this.sendQueue.length === 1) {
this.dequeue();
}
return d;
}
async send(data: WebSocketMessage): Promise<void> { async send(data: WebSocketMessage): Promise<void> {
if (this.isClosed) { if (this.isClosed) {
throw new SocketClosedError("socket has been closed"); throw new SocketClosedError("socket has been closed");
@ -276,28 +304,24 @@ class WebSocketImpl implements WebSocket {
typeof data === "string" ? OpCode.TextFrame : OpCode.BinaryFrame; typeof data === "string" ? OpCode.TextFrame : OpCode.BinaryFrame;
const payload = typeof data === "string" ? encode(data) : data; const payload = typeof data === "string" ? encode(data) : data;
const isLastFrame = true; const isLastFrame = true;
await writeFrame( const frame = {
{ isLastFrame,
isLastFrame, opcode,
opcode, payload,
payload, mask: this.mask
mask: this.mask };
}, return this.enqueue(frame);
this.bufWriter
);
} }
async ping(data: WebSocketMessage = ""): Promise<void> { async ping(data: WebSocketMessage = ""): Promise<void> {
const payload = typeof data === "string" ? encode(data) : data; const payload = typeof data === "string" ? encode(data) : data;
await writeFrame( const frame = {
{ isLastFrame: true,
isLastFrame: true, opcode: OpCode.Ping,
opcode: OpCode.Ping, mask: this.mask,
mask: this.mask, payload
payload };
}, return this.enqueue(frame);
this.bufWriter
);
} }
private _isClosed = false; private _isClosed = false;
@ -317,15 +341,12 @@ class WebSocketImpl implements WebSocket {
} else { } else {
payload = new Uint8Array(header); payload = new Uint8Array(header);
} }
await writeFrame( await this.enqueue({
{ isLastFrame: true,
isLastFrame: true, opcode: OpCode.Close,
opcode: OpCode.Close, mask: this.mask,
mask: this.mask, payload
payload });
},
this.bufWriter
);
} catch (e) { } catch (e) {
throw e; throw e;
} finally { } finally {
@ -380,7 +401,7 @@ export async function acceptWebSocket(req: {
}): Promise<WebSocket> { }): Promise<WebSocket> {
const { conn, headers, bufReader, bufWriter } = req; const { conn, headers, bufReader, bufWriter } = req;
if (acceptable(req)) { if (acceptable(req)) {
const sock = new WebSocketImpl(conn, { bufReader, bufWriter }); const sock = new WebSocketImpl({ conn, bufReader, bufWriter });
const secKey = headers.get("sec-websocket-key"); const secKey = headers.get("sec-websocket-key");
if (typeof secKey !== "string") { if (typeof secKey !== "string") {
throw new Error("sec-websocket-key is not provided"); throw new Error("sec-websocket-key is not provided");
@ -499,9 +520,19 @@ export async function connectWebSocket(
conn.close(); conn.close();
throw err; throw err;
} }
return new WebSocketImpl(conn, { return new WebSocketImpl({
conn,
bufWriter, bufWriter,
bufReader, bufReader,
mask: createMask() mask: createMask()
}); });
} }
export function createWebSocket(params: {
conn: Conn;
bufWriter?: BufWriter;
bufReader?: BufReader;
mask?: Uint8Array;
}): WebSocket {
return new WebSocketImpl(params);
}

View file

@ -3,6 +3,7 @@ import { BufReader, BufWriter } from "../io/bufio.ts";
import { assert, assertEquals, assertThrowsAsync } from "../testing/asserts.ts"; import { assert, assertEquals, assertThrowsAsync } from "../testing/asserts.ts";
import { runIfMain, test } from "../testing/mod.ts"; import { runIfMain, test } from "../testing/mod.ts";
import { TextProtoReader } from "../textproto/mod.ts"; import { TextProtoReader } from "../textproto/mod.ts";
import * as bytes from "../bytes/mod.ts";
import { import {
acceptable, acceptable,
connectWebSocket, connectWebSocket,
@ -11,10 +12,13 @@ import {
OpCode, OpCode,
readFrame, readFrame,
unmask, unmask,
writeFrame writeFrame,
createWebSocket
} from "./mod.ts"; } from "./mod.ts";
import { encode } from "../strings/mod.ts"; import { encode, decode } from "../strings/mod.ts";
type Writer = Deno.Writer;
type Reader = Deno.Reader;
type Conn = Deno.Conn;
const { Buffer } = Deno; const { Buffer } = Deno;
test(async function wsReadUnmaskedTextFrame(): Promise<void> { test(async function wsReadUnmaskedTextFrame(): Promise<void> {
@ -30,7 +34,7 @@ test(async function wsReadUnmaskedTextFrame(): Promise<void> {
}); });
test(async function wsReadMaskedTextFrame(): Promise<void> { test(async function wsReadMaskedTextFrame(): Promise<void> {
//a masked single text frame with payload "Hello" // a masked single text frame with payload "Hello"
const buf = new BufReader( const buf = new BufReader(
new Buffer( new Buffer(
new Uint8Array([ new Uint8Array([
@ -272,4 +276,55 @@ test("handshake should send search correctly", async function wsHandshakeWithSea
assertEquals(statusLine, "GET /?a=1 HTTP/1.1"); assertEquals(statusLine, "GET /?a=1 HTTP/1.1");
}); });
function dummyConn(r: Reader, w: Writer): Conn {
return {
rid: -1,
closeRead: (): void => {},
closeWrite: (): void => {},
read: (x): Promise<number | Deno.EOF> => r.read(x),
write: (x): Promise<number> => w.write(x),
close: (): void => {},
localAddr: { transport: "tcp", hostname: "0.0.0.0", port: 0 },
remoteAddr: { transport: "tcp", hostname: "0.0.0.0", port: 0 }
};
}
function delayedWriter(ms: number, dest: Writer): Writer {
return {
write(p: Uint8Array): Promise<number> {
return new Promise<number>(resolve => {
setTimeout(async (): Promise<void> => {
resolve(await dest.write(p));
}, ms);
});
}
};
}
test("WebSocket.send(), WebSocket.ping() should be exclusive", async (): Promise<
void
> => {
const buf = new Buffer();
const conn = dummyConn(new Buffer(), delayedWriter(1, buf));
const sock = createWebSocket({ conn });
// Ensure send call
await Promise.all([
sock.send("first"),
sock.send("second"),
sock.ping(),
sock.send(new Uint8Array([3]))
]);
const bufr = new BufReader(buf);
const first = await readFrame(bufr);
const second = await readFrame(bufr);
const ping = await readFrame(bufr);
const third = await readFrame(bufr);
assertEquals(first.opcode, OpCode.TextFrame);
assertEquals(decode(first.payload), "first");
assertEquals(first.opcode, OpCode.TextFrame);
assertEquals(decode(second.payload), "second");
assertEquals(ping.opcode, OpCode.Ping);
assertEquals(third.opcode, OpCode.BinaryFrame);
assertEquals(bytes.equal(third.payload, new Uint8Array([3])), true);
});
runIfMain(import.meta); runIfMain(import.meta);