2022-05-27 18:30:13 +08:00
|
|
|
/*
|
|
|
|
Copyright 2022 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.
|
|
|
|
*/
|
|
|
|
|
2022-10-13 01:59:10 +08:00
|
|
|
import fetchMock from "fetch-mock-jest";
|
2022-05-27 18:30:13 +08:00
|
|
|
import { UpdateCheckStatus } from "matrix-react-sdk/src/BasePlatform";
|
|
|
|
import { MatrixClientPeg } from "matrix-react-sdk/src/MatrixClientPeg";
|
|
|
|
|
|
|
|
import WebPlatform from "../../../../src/vector/platform/WebPlatform";
|
2023-09-26 19:28:31 +08:00
|
|
|
import { setupLanguageMock } from "../../../setup/setupLanguage";
|
2022-05-27 18:30:13 +08:00
|
|
|
|
2022-10-13 01:59:10 +08:00
|
|
|
fetchMock.config.overwriteRoutes = true;
|
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
describe("WebPlatform", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
jest.clearAllMocks();
|
2023-09-26 19:28:31 +08:00
|
|
|
setupLanguageMock();
|
2022-05-27 18:30:13 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
it("returns human readable name", () => {
|
|
|
|
const platform = new WebPlatform();
|
|
|
|
expect(platform.getHumanReadableName()).toEqual("Web Platform");
|
|
|
|
});
|
|
|
|
|
2022-07-11 20:22:37 +08:00
|
|
|
it("registers service worker", () => {
|
|
|
|
// @ts-ignore - mocking readonly object
|
|
|
|
navigator.serviceWorker = { register: jest.fn() };
|
|
|
|
new WebPlatform();
|
|
|
|
expect(navigator.serviceWorker.register).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
2022-10-01 01:27:59 +08:00
|
|
|
it("should call reload on window location object", () => {
|
2023-04-25 16:36:17 +08:00
|
|
|
Object.defineProperty(window, "location", { value: { reload: jest.fn() }, writable: true });
|
2022-10-01 01:27:59 +08:00
|
|
|
|
|
|
|
const platform = new WebPlatform();
|
|
|
|
expect(window.location.reload).not.toHaveBeenCalled();
|
|
|
|
platform.reload();
|
|
|
|
expect(window.location.reload).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should call reload to install update", () => {
|
2023-04-25 16:36:17 +08:00
|
|
|
Object.defineProperty(window, "location", { value: { reload: jest.fn() }, writable: true });
|
2022-10-01 01:27:59 +08:00
|
|
|
|
|
|
|
const platform = new WebPlatform();
|
|
|
|
expect(window.location.reload).not.toHaveBeenCalled();
|
|
|
|
platform.installUpdate();
|
|
|
|
expect(window.location.reload).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("getDefaultDeviceDisplayName", () => {
|
|
|
|
it.each([
|
2022-12-09 20:28:29 +08:00
|
|
|
[
|
2022-10-01 01:27:59 +08:00
|
|
|
"https://develop.element.io/#/room/!foo:bar",
|
2022-10-13 16:22:34 +08:00
|
|
|
"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) " +
|
|
|
|
"Chrome/105.0.0.0 Safari/537.36",
|
2022-10-12 21:35:52 +08:00
|
|
|
"develop.element.io: Chrome on macOS",
|
2022-12-09 20:28:29 +08:00
|
|
|
],
|
2022-10-01 01:27:59 +08:00
|
|
|
])("%s & %s = %s", (url, userAgent, result) => {
|
2023-04-25 16:36:17 +08:00
|
|
|
Object.defineProperty(window, "navigator", { value: { userAgent }, writable: true });
|
|
|
|
Object.defineProperty(window, "location", { value: { href: url }, writable: true });
|
2022-10-01 01:27:59 +08:00
|
|
|
const platform = new WebPlatform();
|
|
|
|
expect(platform.getDefaultDeviceDisplayName()).toEqual(result);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
describe("notification support", () => {
|
|
|
|
const mockNotification = {
|
|
|
|
requestPermission: jest.fn(),
|
|
|
|
permission: "notGranted",
|
2022-10-13 16:22:34 +08:00
|
|
|
};
|
2022-05-27 18:30:13 +08:00
|
|
|
beforeEach(() => {
|
|
|
|
// @ts-ignore
|
|
|
|
window.Notification = mockNotification;
|
|
|
|
mockNotification.permission = "notGranted";
|
|
|
|
});
|
|
|
|
|
|
|
|
it("supportsNotifications returns false when platform does not support notifications", () => {
|
|
|
|
// @ts-ignore
|
|
|
|
window.Notification = undefined;
|
|
|
|
expect(new WebPlatform().supportsNotifications()).toBe(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("supportsNotifications returns true when platform supports notifications", () => {
|
|
|
|
expect(new WebPlatform().supportsNotifications()).toBe(true);
|
|
|
|
});
|
2022-07-11 20:22:37 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
it("maySendNotifications returns true when notification permissions are not granted", () => {
|
|
|
|
expect(new WebPlatform().maySendNotifications()).toBe(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("maySendNotifications returns true when notification permissions are granted", () => {
|
2022-10-13 16:22:34 +08:00
|
|
|
mockNotification.permission = "granted";
|
2022-05-27 18:30:13 +08:00
|
|
|
expect(new WebPlatform().maySendNotifications()).toBe(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("requests notification permissions and returns result ", async () => {
|
|
|
|
mockNotification.requestPermission.mockImplementation((callback) => callback("test"));
|
|
|
|
|
|
|
|
const platform = new WebPlatform();
|
|
|
|
const result = await platform.requestNotificationPermission();
|
|
|
|
expect(result).toEqual("test");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("app version", () => {
|
|
|
|
const envVersion = process.env.VERSION;
|
|
|
|
const prodVersion = "1.10.13";
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
jest.spyOn(MatrixClientPeg, "userRegisteredWithinLastHours").mockReturnValue(false);
|
2022-10-13 16:22:34 +08:00
|
|
|
});
|
2022-05-27 18:30:13 +08:00
|
|
|
|
|
|
|
afterAll(() => {
|
2023-04-25 16:36:17 +08:00
|
|
|
// @ts-ignore
|
|
|
|
WebPlatform.VERSION = envVersion;
|
2022-05-27 18:30:13 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should return true from canSelfUpdate()", async () => {
|
|
|
|
const platform = new WebPlatform();
|
|
|
|
const result = await platform.canSelfUpdate();
|
|
|
|
expect(result).toBe(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("getAppVersion returns normalized app version", async () => {
|
2023-04-25 16:36:17 +08:00
|
|
|
// @ts-ignore
|
|
|
|
WebPlatform.VERSION = prodVersion;
|
2022-05-27 18:30:13 +08:00
|
|
|
const platform = new WebPlatform();
|
|
|
|
|
|
|
|
const version = await platform.getAppVersion();
|
|
|
|
expect(version).toEqual(prodVersion);
|
|
|
|
|
2023-04-25 16:36:17 +08:00
|
|
|
// @ts-ignore
|
|
|
|
WebPlatform.VERSION = `v${prodVersion}`;
|
2022-05-27 18:30:13 +08:00
|
|
|
const version2 = await platform.getAppVersion();
|
|
|
|
// v prefix removed
|
|
|
|
expect(version2).toEqual(prodVersion);
|
|
|
|
|
2023-04-25 16:36:17 +08:00
|
|
|
// @ts-ignore
|
|
|
|
WebPlatform.VERSION = `version not like semver`;
|
2022-05-27 18:30:13 +08:00
|
|
|
const notSemverVersion = await platform.getAppVersion();
|
|
|
|
expect(notSemverVersion).toEqual(`version not like semver`);
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("pollForUpdate()", () => {
|
2022-10-13 16:22:34 +08:00
|
|
|
it(
|
|
|
|
"should return not available and call showNoUpdate when current version " +
|
|
|
|
"matches most recent version",
|
|
|
|
async () => {
|
2023-04-25 16:36:17 +08:00
|
|
|
// @ts-ignore
|
|
|
|
WebPlatform.VERSION = prodVersion;
|
2022-10-13 01:59:10 +08:00
|
|
|
fetchMock.getOnce("/version", prodVersion);
|
2022-05-27 18:30:13 +08:00
|
|
|
const platform = new WebPlatform();
|
2022-12-09 20:28:29 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
const showUpdate = jest.fn();
|
|
|
|
const showNoUpdate = jest.fn();
|
|
|
|
const result = await platform.pollForUpdate(showUpdate, showNoUpdate);
|
2022-12-09 20:28:29 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
expect(result).toEqual({ status: UpdateCheckStatus.NotAvailable });
|
|
|
|
expect(showUpdate).not.toHaveBeenCalled();
|
|
|
|
expect(showNoUpdate).toHaveBeenCalled();
|
|
|
|
},
|
|
|
|
);
|
2022-12-09 20:28:29 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
it("should strip v prefix from versions before comparing", async () => {
|
2023-04-25 16:36:17 +08:00
|
|
|
// @ts-ignore
|
|
|
|
WebPlatform.VERSION = prodVersion;
|
2022-10-13 01:59:10 +08:00
|
|
|
fetchMock.getOnce("/version", `v${prodVersion}`);
|
2022-05-27 18:30:13 +08:00
|
|
|
const platform = new WebPlatform();
|
2022-07-11 20:22:37 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
const showUpdate = jest.fn();
|
|
|
|
const showNoUpdate = jest.fn();
|
|
|
|
const result = await platform.pollForUpdate(showUpdate, showNoUpdate);
|
2022-07-11 20:22:37 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
// versions only differ by v prefix, no update
|
|
|
|
expect(result).toEqual({ status: UpdateCheckStatus.NotAvailable });
|
|
|
|
expect(showUpdate).not.toHaveBeenCalled();
|
|
|
|
expect(showNoUpdate).toHaveBeenCalled();
|
|
|
|
});
|
2022-07-11 20:22:37 +08:00
|
|
|
|
2022-10-13 16:22:34 +08:00
|
|
|
it(
|
|
|
|
"should return ready and call showUpdate when current version " + "differs from most recent version",
|
|
|
|
async () => {
|
2023-04-25 16:36:17 +08:00
|
|
|
// @ts-ignore
|
|
|
|
WebPlatform.VERSION = "0.0.0"; // old version
|
2022-10-13 01:59:10 +08:00
|
|
|
fetchMock.getOnce("/version", prodVersion);
|
2022-05-27 18:30:13 +08:00
|
|
|
const platform = new WebPlatform();
|
2022-12-09 20:28:29 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
const showUpdate = jest.fn();
|
|
|
|
const showNoUpdate = jest.fn();
|
|
|
|
const result = await platform.pollForUpdate(showUpdate, showNoUpdate);
|
2022-12-09 20:28:29 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
expect(result).toEqual({ status: UpdateCheckStatus.Ready });
|
|
|
|
expect(showUpdate).toHaveBeenCalledWith("0.0.0", prodVersion);
|
|
|
|
expect(showNoUpdate).not.toHaveBeenCalled();
|
|
|
|
},
|
|
|
|
);
|
2022-12-09 20:28:29 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
it("should return ready without showing update when user registered in last 24", async () => {
|
2023-04-25 16:36:17 +08:00
|
|
|
// @ts-ignore
|
|
|
|
WebPlatform.VERSION = "0.0.0"; // old version
|
2022-05-27 18:30:13 +08:00
|
|
|
jest.spyOn(MatrixClientPeg, "userRegisteredWithinLastHours").mockReturnValue(true);
|
2022-10-13 01:59:10 +08:00
|
|
|
fetchMock.getOnce("/version", prodVersion);
|
2022-05-27 18:30:13 +08:00
|
|
|
const platform = new WebPlatform();
|
2022-07-11 20:22:37 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
const showUpdate = jest.fn();
|
|
|
|
const showNoUpdate = jest.fn();
|
|
|
|
const result = await platform.pollForUpdate(showUpdate, showNoUpdate);
|
2022-07-11 20:22:37 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
expect(result).toEqual({ status: UpdateCheckStatus.Ready });
|
|
|
|
expect(showUpdate).not.toHaveBeenCalled();
|
|
|
|
expect(showNoUpdate).not.toHaveBeenCalled();
|
|
|
|
});
|
2022-07-11 20:22:37 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
it("should return error when version check fails", async () => {
|
2022-10-13 01:59:10 +08:00
|
|
|
fetchMock.getOnce("/version", { throws: "oups" });
|
2022-05-27 18:30:13 +08:00
|
|
|
const platform = new WebPlatform();
|
2022-07-11 20:22:37 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
const showUpdate = jest.fn();
|
|
|
|
const showNoUpdate = jest.fn();
|
|
|
|
const result = await platform.pollForUpdate(showUpdate, showNoUpdate);
|
2022-07-11 20:22:37 +08:00
|
|
|
|
2022-05-27 18:30:13 +08:00
|
|
|
expect(result).toEqual({ status: UpdateCheckStatus.Error, detail: "Unknown Error" });
|
|
|
|
expect(showUpdate).not.toHaveBeenCalled();
|
|
|
|
expect(showNoUpdate).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|