import { gettext } from "django"; import { LitElement, html, customElement, TemplateResult, property } from "lit-element"; import { DefaultClient } from "../../api/Client"; import "./Message"; import { APIMessage } from "./Message"; export function showMessage(message: APIMessage): void { const container = document.querySelector("ak-message-container"); if (!container) { throw new Error("failed to find message container"); } container.messages.push(message); container.requestUpdate(); } @customElement("ak-message-container") export class MessageContainer extends LitElement { url = DefaultClient.makeUrl(["root", "messages"]); @property({attribute: false}) messages: APIMessage[] = []; messageSocket?: WebSocket; retryDelay = 200; createRenderRoot(): ShadowRoot | Element { return this; } constructor() { super(); try { this.connect(); } catch (error) { console.warn(`authentik/messages: failed to connect to ws ${error}`); } } firstUpdated(): void { this.fetchMessages(); } connect(): void { const wsUrl = `${window.location.protocol.replace("http", "ws")}//${ window.location.host }/ws/client/`; this.messageSocket = new WebSocket(wsUrl); this.messageSocket.addEventListener("open", () => { console.debug(`authentik/messages: connected to ${wsUrl}`); }); this.messageSocket.addEventListener("close", (e) => { console.debug(`authentik/messages: closed ws connection: ${e}`); if (this.retryDelay > 3000) { showMessage({ level_tag: "error", message: gettext("Connection error, reconnecting...") }); } setTimeout(() => { console.debug(`authentik/messages: reconnecting ws in ${this.retryDelay}ms`); this.connect(); }, this.retryDelay); this.retryDelay = this.retryDelay * 2; }); this.messageSocket.addEventListener("message", (e) => { const data = JSON.parse(e.data); this.messages.push(data); this.requestUpdate(); }); this.messageSocket.addEventListener("error", (e) => { console.warn(`authentik/messages: error ${e}`); this.retryDelay = this.retryDelay * 2; }); } /* Fetch messages which were stored in the session. * This mostly gets messages which were created when the user arrives/leaves the site * and especially the login flow */ fetchMessages(): Promise { console.debug("authentik/messages: fetching messages over direct api"); return fetch(this.url) .then((r) => r.json()) .then((r: APIMessage[]) => { r.forEach((m: APIMessage) => { this.messages.push(m); this.requestUpdate(); }); }); } render(): TemplateResult { return html``; } }