/* Copyright 2016 Aviral Dasgupta Copyright 2016 OpenMarket Ltd Copyright 2018 New Vector Ltd Copyright 2020 The Matrix.org Foundation C.I.C. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ import { MatrixClient } from "matrix-js-sdk/src/client"; import { encodeUnpaddedBase64 } from "matrix-js-sdk/src/crypto/olmlib"; import { logger } from "matrix-js-sdk/src/logger"; import { MatrixEvent } from "matrix-js-sdk/src/models/event"; import { Room } from "matrix-js-sdk/src/models/room"; import { SSOAction } from "matrix-js-sdk/src/@types/auth"; import dis from "./dispatcher/dispatcher"; import BaseEventIndexManager from "./indexing/BaseEventIndexManager"; import { ActionPayload } from "./dispatcher/payloads"; import { CheckUpdatesPayload } from "./dispatcher/payloads/CheckUpdatesPayload"; import { Action } from "./dispatcher/actions"; import { hideToast as hideUpdateToast } from "./toasts/UpdateToast"; import { MatrixClientPeg } from "./MatrixClientPeg"; import { idbLoad, idbSave, idbDelete } from "./utils/StorageManager"; import { ViewRoomPayload } from "./dispatcher/payloads/ViewRoomPayload"; import { IConfigOptions } from "./IConfigOptions"; export const SSO_HOMESERVER_URL_KEY = "mx_sso_hs_url"; export const SSO_ID_SERVER_URL_KEY = "mx_sso_is_url"; export const SSO_IDP_ID_KEY = "mx_sso_idp_id"; export enum UpdateCheckStatus { Checking = "CHECKING", Error = "ERROR", NotAvailable = "NOTAVAILABLE", Downloading = "DOWNLOADING", Ready = "READY", } export interface UpdateStatus { /** * The current phase of the manual update check. */ status: UpdateCheckStatus; /** * Detail string relating to the current status, typically for error details. */ detail?: string; } const UPDATE_DEFER_KEY = "mx_defer_update"; /** * Base class for classes that provide platform-specific functionality * eg. Setting an application badge or displaying notifications * * Instances of this class are provided by the application. */ export default abstract class BasePlatform { protected notificationCount = 0; protected errorDidOccur = false; public constructor() { dis.register(this.onAction); this.startUpdateCheck = this.startUpdateCheck.bind(this); } public abstract getConfig(): Promise; public abstract getDefaultDeviceDisplayName(): string; protected onAction = (payload: ActionPayload): void => { switch (payload.action) { case "on_client_not_viable": case Action.OnLoggedOut: this.setNotificationCount(0); break; } }; // Used primarily for Analytics public abstract getHumanReadableName(): string; public setNotificationCount(count: number): void { this.notificationCount = count; } public setErrorStatus(errorDidOccur: boolean): void { this.errorDidOccur = errorDidOccur; } /** * Whether we can call checkForUpdate on this platform build */ public async canSelfUpdate(): Promise { return false; } public startUpdateCheck(): void { hideUpdateToast(); localStorage.removeItem(UPDATE_DEFER_KEY); dis.dispatch({ action: Action.CheckUpdates, status: UpdateCheckStatus.Checking, }); } /** * Update the currently running app to the latest available version * and replace this instance of the app with the new version. */ public installUpdate(): void {} /** * Check if the version update has been deferred and that deferment is still in effect * @param newVersion the version string to check */ protected shouldShowUpdate(newVersion: string): boolean { // If the user registered on this client in the last 24 hours then do not show them the update toast if (MatrixClientPeg.userRegisteredWithinLastHours(24)) return false; try { const [version, deferUntil] = JSON.parse(localStorage.getItem(UPDATE_DEFER_KEY)!); return newVersion !== version || Date.now() > deferUntil; } catch (e) { return true; } } /** * Ignore the pending update and don't prompt about this version * until the next morning (8am). */ public deferUpdate(newVersion: string): void { const date = new Date(Date.now() + 24 * 60 * 60 * 1000); date.setHours(8, 0, 0, 0); // set to next 8am localStorage.setItem(UPDATE_DEFER_KEY, JSON.stringify([newVersion, date.getTime()])); hideUpdateToast(); } /** * Return true if platform supports multi-language * spell-checking, otherwise false. */ public supportsSpellCheckSettings(): boolean { return false; } /** * Returns true if platform allows overriding native context menus */ public allowOverridingNativeContextMenus(): boolean { return false; } /** * Returns true if the platform supports displaying * notifications, otherwise false. * @returns {boolean} whether the platform supports displaying notifications */ public supportsNotifications(): boolean { return false; } /** * Returns true if the application currently has permission * to display notifications. Otherwise false. * @returns {boolean} whether the application has permission to display notifications */ public maySendNotifications(): boolean { return false; } /** * Requests permission to send notifications. Returns * a promise that is resolved when the user has responded * to the request. The promise has a single string argument * that is 'granted' if the user allowed the request or * 'denied' otherwise. */ public abstract requestNotificationPermission(): Promise; public displayNotification( title: string, msg: string, avatarUrl: string | null, room: Room, ev?: MatrixEvent, ): Notification { const notifBody: NotificationOptions = { body: msg, silent: true, // we play our own sounds }; if (avatarUrl) notifBody["icon"] = avatarUrl; const notification = new window.Notification(title, notifBody); notification.onclick = () => { const payload: ViewRoomPayload = { action: Action.ViewRoom, room_id: room.roomId, metricsTrigger: "Notification", }; if (ev?.getThread()) { payload.event_id = ev.getId(); } dis.dispatch(payload); window.focus(); }; return notification; } public loudNotification(ev: MatrixEvent, room: Room): void {} public clearNotification(notif: Notification): void { // Some browsers don't support this, e.g Safari on iOS // https://developer.mozilla.org/en-US/docs/Web/API/Notification/close if (notif.close) { notif.close(); } } /** * Returns true if the platform requires URL previews in tooltips, otherwise false. * @returns {boolean} whether the platform requires URL previews in tooltips */ public needsUrlTooltips(): boolean { return false; } /** * Returns a promise that resolves to a string representing the current version of the application. */ public abstract getAppVersion(): Promise; /** * Restarts the application, without necessarily reloading * any application code */ public abstract reload(): void; public supportsSetting(settingName?: string): boolean { return false; } public async getSettingValue(settingName: string): Promise { return undefined; } public setSettingValue(settingName: string, value: any): Promise { throw new Error("Unimplemented"); } /** * Get our platform specific EventIndexManager. * * @return {BaseEventIndexManager} The EventIndex manager for our platform, * can be null if the platform doesn't support event indexing. */ public getEventIndexingManager(): BaseEventIndexManager | null { return null; } public setLanguage(preferredLangs: string[]): void {} public setSpellCheckEnabled(enabled: boolean): void {} public async getSpellCheckEnabled(): Promise { return false; } public setSpellCheckLanguages(preferredLangs: string[]): void {} public getSpellCheckLanguages(): Promise | null { return null; } public async getDesktopCapturerSources(options: GetSourcesOptions): Promise> { return []; } public supportsDesktopCapturer(): boolean { return false; } public supportsJitsiScreensharing(): boolean { return true; } public overrideBrowserShortcuts(): boolean { return false; } public navigateForwardBack(back: boolean): void {} public getAvailableSpellCheckLanguages(): Promise | null { return null; } protected getSSOCallbackUrl(fragmentAfterLogin = ""): URL { const url = new URL(window.location.href); url.hash = fragmentAfterLogin; return url; } /** * Begin Single Sign On flows. * @param {MatrixClient} mxClient the matrix client using which we should start the flow * @param {"sso"|"cas"} loginType the type of SSO it is, CAS/SSO. * @param {string} fragmentAfterLogin the hash to pass to the app during sso callback. * @param {SSOAction} action the SSO flow to indicate to the IdP, optional. * @param {string} idpId The ID of the Identity Provider being targeted, optional. */ public startSingleSignOn( mxClient: MatrixClient, loginType: "sso" | "cas", fragmentAfterLogin?: string, idpId?: string, action?: SSOAction, ): void { // persist hs url and is url for when the user is returned to the app with the login token localStorage.setItem(SSO_HOMESERVER_URL_KEY, mxClient.getHomeserverUrl()); if (mxClient.getIdentityServerUrl()) { localStorage.setItem(SSO_ID_SERVER_URL_KEY, mxClient.getIdentityServerUrl()!); } if (idpId) { localStorage.setItem(SSO_IDP_ID_KEY, idpId); } const callbackUrl = this.getSSOCallbackUrl(fragmentAfterLogin); window.location.href = mxClient.getSsoLoginUrl(callbackUrl.toString(), loginType, idpId, action); // redirect to SSO } /** * Get a previously stored pickle key. The pickle key is used for * encrypting libolm objects. * @param {string} userId the user ID for the user that the pickle key is for. * @param {string} userId the device ID that the pickle key is for. * @returns {string|null} the previously stored pickle key, or null if no * pickle key has been stored. */ public async getPickleKey(userId: string, deviceId: string): Promise { if (!window.crypto || !window.crypto.subtle) { return null; } let data; try { data = await idbLoad("pickleKey", [userId, deviceId]); } catch (e) { logger.error("idbLoad for pickleKey failed", e); } if (!data) { return null; } if (!data.encrypted || !data.iv || !data.cryptoKey) { logger.error("Badly formatted pickle key"); return null; } const additionalData = new Uint8Array(userId.length + deviceId.length + 1); for (let i = 0; i < userId.length; i++) { additionalData[i] = userId.charCodeAt(i); } additionalData[userId.length] = 124; // "|" for (let i = 0; i < deviceId.length; i++) { additionalData[userId.length + 1 + i] = deviceId.charCodeAt(i); } try { const key = await crypto.subtle.decrypt( { name: "AES-GCM", iv: data.iv, additionalData }, data.cryptoKey, data.encrypted, ); return encodeUnpaddedBase64(key); } catch (e) { logger.error("Error decrypting pickle key"); return null; } } /** * Create and store a pickle key for encrypting libolm objects. * @param {string} userId the user ID for the user that the pickle key is for. * @param {string} deviceId the device ID that the pickle key is for. * @returns {string|null} the pickle key, or null if the platform does not * support storing pickle keys. */ public async createPickleKey(userId: string, deviceId: string): Promise { if (!window.crypto || !window.crypto.subtle) { return null; } const crypto = window.crypto; const randomArray = new Uint8Array(32); crypto.getRandomValues(randomArray); const cryptoKey = await crypto.subtle.generateKey({ name: "AES-GCM", length: 256 }, false, [ "encrypt", "decrypt", ]); const iv = new Uint8Array(32); crypto.getRandomValues(iv); const additionalData = new Uint8Array(userId.length + deviceId.length + 1); for (let i = 0; i < userId.length; i++) { additionalData[i] = userId.charCodeAt(i); } additionalData[userId.length] = 124; // "|" for (let i = 0; i < deviceId.length; i++) { additionalData[userId.length + 1 + i] = deviceId.charCodeAt(i); } const encrypted = await crypto.subtle.encrypt({ name: "AES-GCM", iv, additionalData }, cryptoKey, randomArray); try { await idbSave("pickleKey", [userId, deviceId], { encrypted, iv, cryptoKey }); } catch (e) { return null; } return encodeUnpaddedBase64(randomArray); } /** * Delete a previously stored pickle key from storage. * @param {string} userId the user ID for the user that the pickle key is for. * @param {string} userId the device ID that the pickle key is for. */ public async destroyPickleKey(userId: string, deviceId: string): Promise { try { await idbDelete("pickleKey", [userId, deviceId]); } catch (e) { logger.error("idbDelete failed in destroyPickleKey", e); } } /** * Clear app storage, called when logging out to perform data clean up. */ public async clearStorage(): Promise { window.sessionStorage.clear(); window.localStorage.clear(); } }