Skip to content

Commit

Permalink
fix(server): Acknowledge connection before notifying the client to av…
Browse files Browse the repository at this point in the history
…oid race conditions with slow sends (#506)

Closes #501
  • Loading branch information
enisdenjo authored Sep 28, 2023
1 parent 039cab0 commit 8cb82bd
Show file tree
Hide file tree
Showing 2 changed files with 60 additions and 3 deletions.
55 changes: 55 additions & 0 deletions src/__tests__/server.ts
Original file line number Diff line number Diff line change
Expand Up @@ -635,6 +635,61 @@ describe('Connect', () => {
expect(event.wasClean).toBeTruthy();
});
});

it("should have acknowledged connection even if ack message send didn't resolve", (done) => {
let sent: Promise<void> | null = null;
let resolveSend = () => {
// noop
};
makeServer({
schema,
onSubscribe(ctx) {
expect(ctx.acknowledged).toBeTruthy();
resolveSend();
done();
},
}).opened(
{
protocol: GRAPHQL_TRANSPORT_WS_PROTOCOL,
send: async () => {
// if already set, this is a subsequent send happening after the test
if (sent) {
return;
}

// message was sent and delivered to the client...
sent = new Promise((resolve) => {
resolve();
});
await sent;

// ...but something else is slow - leading to a potential race condition on the `acknowledged` flag
await new Promise<void>((resolve) => (resolveSend = resolve));
},
close: (code, reason) => {
fail(`Unexpected close with ${code}: ${reason}`);
},
onMessage: async (cb) => {
cb(stringifyMessage({ type: MessageType.ConnectionInit }));
await sent;
cb(
stringifyMessage({
id: '1',
type: MessageType.Subscribe,
payload: { query: '{ getValue }' },
}),
);
},
onPing: () => {
/**/
},
onPong: () => {
/**/
},
},
{},
);
});
});

describe('Ping/Pong', () => {
Expand Down
8 changes: 5 additions & 3 deletions src/server.ts
Original file line number Diff line number Diff line change
Expand Up @@ -613,6 +613,11 @@ export function makeServer<
if (permittedOrPayload === false)
return socket.close(CloseCode.Forbidden, 'Forbidden');

// we should acknowledge before send to avoid race conditions (like those exampled in https://github.com/enisdenjo/graphql-ws/issues/501)
// even if the send fails/throws, the connection should be closed because its malfunctioning
// @ts-expect-error: I can write
ctx.acknowledged = true;

await socket.send(
stringifyMessage<MessageType.ConnectionAck>(
isObject(permittedOrPayload)
Expand All @@ -627,9 +632,6 @@ export function makeServer<
replacer,
),
);

// @ts-expect-error: I can write
ctx.acknowledged = true;
return;
}
case MessageType.Ping: {
Expand Down

0 comments on commit 8cb82bd

Please sign in to comment.