import { createBrowserHistory, History } from "history"; import { computed, configure, when } from "mobx"; import { RouterStore, syncHistoryWithStore } from "mobx-react-router"; import { WebSocketRpcClient } from "@client/sprinklersRpc/WebSocketRpcClient"; import HttpApi from "@client/state/HttpApi"; import { UiStore } from "@client/state/UiStore"; import { UserStore } from "@client/state/UserStore"; import ApiError from "@common/ApiError"; import { ErrorCode } from "@common/ErrorCode"; import log from "@common/logger"; import { DefaultEvents, TypedEventEmitter } from "@common/TypedEventEmitter"; interface AppEvents extends DefaultEvents { checkToken(): void; hasToken(): void; } export default class AppState extends TypedEventEmitter { history: History = createBrowserHistory(); routerStore = new RouterStore(); uiStore = new UiStore(); userStore = new UserStore(); httpApi = new HttpApi(); tokenStore = this.httpApi.tokenStore; sprinklersRpc = new WebSocketRpcClient(this.tokenStore); constructor() { super(); this.sprinklersRpc.on("newUserData", this.userStore.receiveUserData); this.sprinklersRpc.on("tokenError", this.checkToken); this.httpApi.on("tokenError", this.checkToken); this.on("checkToken", this.doCheckToken); this.on("hasToken", () => { when(() => !this.tokenStore.accessToken.isValid, this.checkToken); this.sprinklersRpc.start(); }); } @computed get isLoggedIn() { return this.tokenStore.accessToken.isValid; } async start() { configure({ enforceActions: true, }); syncHistoryWithStore(this.history, this.routerStore); await this.tokenStore.loadLocalStorage(); await this.checkToken(); } checkToken = () => { this.emit("checkToken"); } private doCheckToken = async () => { const { accessToken, refreshToken } = this.tokenStore; accessToken.updateCurrentTime(); if (accessToken.isValid) { // if the access token is valid, we are good this.emit("hasToken"); return; } if (!refreshToken.isValid) { // if the refresh token is not valid, need to login again this.history.push("/login"); return; } try { await this.httpApi.grantRefresh(); this.emit("hasToken"); } catch (err) { if (err instanceof ApiError && err.code === ErrorCode.BadToken) { log.warn({ err }, "refresh is bad for some reason, erasing"); this.tokenStore.clear(); this.history.push("/login"); } else { log.error({ err }, "could not refresh access token"); // TODO: some kind of error page? } } } }