add ping/pong between client and server
the purpose of this is to hopefully prevent nginx from reaping the connection, since there's data flowing through it every so often
This commit is contained in:
parent
b3911b7af1
commit
bf4d538651
3 changed files with 36 additions and 0 deletions
|
@ -300,6 +300,10 @@ impl SessionLoop {
|
|||
|
||||
async fn process_request(&mut self, ws: &mut WebSocket, request: Request) -> eyre::Result<()> {
|
||||
match request {
|
||||
Request::Ping => {
|
||||
ws.send(to_message(&Notify::Pong)).await?;
|
||||
}
|
||||
|
||||
Request::Wall { wall_event } => {
|
||||
match &wall_event {
|
||||
// This match only concerns itself with drawing-related events to offload
|
||||
|
|
|
@ -69,6 +69,8 @@ pub enum LoginResponse {
|
|||
rename_all_fields = "camelCase"
|
||||
)]
|
||||
pub enum Request {
|
||||
Ping,
|
||||
|
||||
Wall {
|
||||
wall_event: wall::EventKind,
|
||||
},
|
||||
|
@ -95,6 +97,8 @@ pub struct ChunkInfo {
|
|||
rename_all_fields = "camelCase"
|
||||
)]
|
||||
pub enum Notify {
|
||||
Pong,
|
||||
|
||||
Wall {
|
||||
wall_event: wall::Event,
|
||||
},
|
||||
|
|
|
@ -75,6 +75,8 @@ export async function registerUser(nickname) {
|
|||
}
|
||||
|
||||
class Session extends EventTarget {
|
||||
#sentPing = false;
|
||||
|
||||
constructor(userId, secret) {
|
||||
super();
|
||||
this.userId = userId;
|
||||
|
@ -193,6 +195,8 @@ class Session extends EventTarget {
|
|||
}
|
||||
|
||||
async eventLoop() {
|
||||
this.#pingLoop();
|
||||
|
||||
try {
|
||||
while (true) {
|
||||
let event = await listen([this.ws, "message"]);
|
||||
|
@ -207,7 +211,21 @@ class Session extends EventTarget {
|
|||
}
|
||||
}
|
||||
|
||||
#pingLoop() {
|
||||
// Send small ping packets every 30 seconds to prevent reverse proxies from reaping the
|
||||
// connection if the tab is in the background.
|
||||
// (Browsers don't seem to send standard WebSocket pings if the tab is unfocused.)
|
||||
// We don't actually use this packet for anything else, like establishing whether
|
||||
// we still have a connection to the server, because the browser can handle that for us.
|
||||
setInterval(() => this.sendPing(), 30000);
|
||||
}
|
||||
|
||||
async #processNotify(notify) {
|
||||
if (notify.notify == "pong") {
|
||||
console.debug("pong received");
|
||||
this.#sentPing = false;
|
||||
}
|
||||
|
||||
if (notify.notify == "wall") {
|
||||
this.dispatchEvent(
|
||||
Object.assign(new Event("wallEvent"), {
|
||||
|
@ -229,6 +247,16 @@ class Session extends EventTarget {
|
|||
}
|
||||
}
|
||||
|
||||
sendPing() {
|
||||
if (!this.#sentPing) {
|
||||
console.debug("ping sent; waiting for pong");
|
||||
this.#sendJson({
|
||||
request: "ping",
|
||||
});
|
||||
this.#sentPing = true;
|
||||
}
|
||||
}
|
||||
|
||||
sendCursor(x, y) {
|
||||
this.#sendJson({
|
||||
request: "wall",
|
||||
|
|
Loading…
Reference in a new issue