summaryrefslogtreecommitdiff
path: root/ui/lib/session.svelte.js
diff options
context:
space:
mode:
authorOwen Jacobson <owen@grimoire.ca>2025-04-10 20:50:13 -0400
committerOwen Jacobson <owen@grimoire.ca>2025-04-10 20:50:13 -0400
commit1ef57107b1c355ef896327f0714344277df7ae18 (patch)
tree9874d3d61f0bdb13913c6c4d079fbb82b336f656 /ui/lib/session.svelte.js
parent0fc3057b05dddb4eba142deeb6373ed37e312c60 (diff)
parent1ee129176eb71f5e246462b66fd9c9862ed1ee7a (diff)
Use a heartbeat to allow the client to reconnect after network failures.
Diffstat (limited to 'ui/lib/session.svelte.js')
-rw-r--r--ui/lib/session.svelte.js60
1 files changed, 54 insertions, 6 deletions
diff --git a/ui/lib/session.svelte.js b/ui/lib/session.svelte.js
index 67155ab..b953d9c 100644
--- a/ui/lib/session.svelte.js
+++ b/ui/lib/session.svelte.js
@@ -1,8 +1,11 @@
import { redirect } from '@sveltejs/kit';
+import { goto } from '$app/navigation';
+
import * as api from './apiServer.js';
import * as r from './state/remote/state.svelte.js';
import * as l from './state/local/channels.svelte.js';
+import { Watchdog } from './watchdog.js';
class Session {
remote = $state();
@@ -16,19 +19,32 @@ class Session {
)
);
- static boot({ user, users, channels, messages, resume_point }) {
+ static boot({ user, users, channels, messages, resume_point, heartbeat }) {
const remote = r.State.boot({
currentUser: user,
users,
channels,
messages,
- resumePoint: resume_point
+ resumePoint: resume_point,
+ heartbeat
});
const local = l.Channels.fromLocalStorage();
return new Session(remote, local);
}
+ reboot({ user, users, channels, messages, resume_point, heartbeat }) {
+ this.remote = r.State.boot({
+ currentUser: user,
+ users,
+ channels,
+ messages,
+ resumePoint: resume_point,
+ heartbeat
+ });
+ }
+
constructor(remote, local) {
+ this.watchdog = new Watchdog(this.watchdogExpired.bind(this));
this.remote = remote;
this.local = local;
}
@@ -36,30 +52,62 @@ class Session {
begin() {
this.events = api.subscribeToEvents(this.remote.resumePoint);
this.events.onmessage = this.onMessage.bind(this);
+ this.watchdog.reset(this.heartbeatMillis());
}
end() {
+ this.watchdog.stop();
this.events.close();
this.events = null;
}
+ active() {
+ return this.events !== null;
+ }
+
onMessage(message) {
const event = JSON.parse(message.data);
this.remote.onEvent(event);
this.local.retainChannels(this.remote.channels.all.keys());
+ this.watchdog.reset(this.heartbeatMillis());
+ }
+
+ heartbeatMillis() {
+ return this.remote.heartbeat /* in seconds */ * 1000 /* millis */;
+ }
+
+ async watchdogExpired() {
+ // We leave `this.events` set here as a marker that the interruption is temporary. That's then
+ // used below, after a potential delay, to decide whether to start the stream back up again or
+ // not.
+ this.events.close();
+ this.watchdog.stop();
+
+ const response = await bootOrNavigate(goto);
+ // Session abandoned; give up here. We need to do this after each await, because that's time in
+ // which the session may have been abandoned.
+ if (!this.active()) return;
+
+ this.reboot(response);
+ this.begin();
}
}
-export async function boot() {
+async function bootOrNavigate(navigateTo) {
const response = await api.boot();
switch (response.status) {
case 401:
- redirect(307, '/login');
+ await navigateTo('/login');
break;
case 503:
- redirect(307, '/setup');
+ await navigateTo('/setup');
break;
case 200:
- return Session.boot(response.data);
+ return response.data;
}
}
+
+export async function boot() {
+ const response = await bootOrNavigate(async (url) => redirect(307, url));
+ return Session.boot(response);
+}