Skip to content
This repository has been archived by the owner on Sep 11, 2024. It is now read-only.

Commit

Permalink
Enable the rust-crypto labs button
Browse files Browse the repository at this point in the history
  • Loading branch information
richvdh committed Jan 10, 2024
1 parent a74f932 commit 773f5f6
Show file tree
Hide file tree
Showing 5 changed files with 127 additions and 11 deletions.
3 changes: 2 additions & 1 deletion src/MatrixClientPeg.ts
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,7 @@ import ErrorDialog from "./components/views/dialogs/ErrorDialog";
import PlatformPeg from "./PlatformPeg";
import { formatList } from "./utils/FormattingUtils";
import SdkConfig from "./SdkConfig";
import { Features } from "./settings/Settings";

export interface IMatrixClientCreds {
homeserverUrl: string;
Expand Down Expand Up @@ -301,7 +302,7 @@ class MatrixClientPegClass implements IMatrixClientPeg {
throw new Error("createClient must be called first");
}

const useRustCrypto = SettingsStore.getValue("feature_rust_crypto");
const useRustCrypto = SettingsStore.getValue(Features.RustCrypto);

// we want to make sure that the same crypto implementation is used throughout the lifetime of a device,
// so persist the setting at the device layer
Expand Down
5 changes: 4 additions & 1 deletion src/i18n/strings/en_EN.json
Original file line number Diff line number Diff line change
Expand Up @@ -1446,7 +1446,10 @@
"report_to_moderators": "Report to moderators",
"report_to_moderators_description": "In rooms that support moderation, the “Report” button will let you report abuse to room moderators.",
"rust_crypto": "Rust cryptography implementation",
"rust_crypto_disabled_notice": "Can currently only be enabled via config.json",
"rust_crypto_in_config": "Rust cryptography cannot be disabled on this deployment of %(brand)s",
"rust_crypto_in_config_description": "Switching to the Rust cryptography requires a migration process that may take several minutes. It cannot be disabled; use with caution!",
"rust_crypto_optin_warning": "Switching to the Rust cryptography requires a migration process that may take several minutes. To disable you will need to log out and back in; use with caution!",
"rust_crypto_requires_logout": "Once enabled, Rust cryptography can only be disabled by logging out and in again",
"sliding_sync": "Sliding Sync mode",
"sliding_sync_checking": "Checking…",
"sliding_sync_configuration": "Sliding Sync configuration",
Expand Down
19 changes: 14 additions & 5 deletions src/settings/Settings.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ import RustCryptoSdkController from "./controllers/RustCryptoSdkController";
import ServerSupportUnstableFeatureController from "./controllers/ServerSupportUnstableFeatureController";
import { WatchManager } from "./WatchManager";
import { CustomTheme } from "../theme";
import SettingsStore from "./SettingsStore";

export const defaultWatchManager = new WatchManager();

Expand Down Expand Up @@ -94,6 +95,7 @@ export enum Features {
VoiceBroadcastForceSmallChunks = "feature_voice_broadcast_force_small_chunks",
NotificationSettings2 = "feature_notification_settings2",
OidcNativeFlow = "feature_oidc_native_flow",
RustCrypto = "feature_rust_crypto",
}

export const labGroupNames: Record<LabGroup, TranslationKey> = {
Expand Down Expand Up @@ -480,15 +482,22 @@ export const SETTINGS: { [setting: string]: ISetting } = {
description: _td("labs|oidc_native_flow_description"),
default: false,
},
"feature_rust_crypto": {
// use the rust matrix-sdk-crypto-js for crypto.
[Features.RustCrypto]: {
// use the rust matrix-sdk-crypto-wasm for crypto.
isFeature: true,
labsGroup: LabGroup.Developer,
configDisablesSetting: true,
// unlike most features, `configDisablesSetting` is false here.
supportedLevels: LEVELS_DEVICE_ONLY_SETTINGS_WITH_CONFIG,
displayName: _td("labs|rust_crypto"),
description: _td("labs|under_active_development"),
// shouldWarn: true,
description: () => {
if (SettingsStore.getValueAt(SettingLevel.CONFIG, Features.RustCrypto)) {
// It's enabled in the config, so you can't get rid of it even by logging out.
return _t("labs|rust_crypto_in_config_description");
} else {
return _t("labs|rust_crypto_optin_warning");
}
},
shouldWarn: true,
default: false,
controller: new RustCryptoSdkController(),
},
Expand Down
29 changes: 26 additions & 3 deletions src/settings/controllers/RustCryptoSdkController.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,12 +15,35 @@ limitations under the License.
*/

import { _t } from "../../languageHandler";
import SettingsStore from "../SettingsStore";
import { SettingLevel } from "../SettingLevel";
import PlatformPeg from "../../PlatformPeg";
import SettingController from "./SettingController";
import { Features } from "../Settings";
import { MatrixClientPeg } from "../../MatrixClientPeg";
import SdkConfig from "../../SdkConfig";

export default class RustCryptoSdkController extends SettingController {
public onChange(level: SettingLevel, roomId: string | null, newValue: any): void {
// If the crypto stack has already been initialized, we'll need to reload the app to make it take effect.
if (MatrixClientPeg.get()?.getCrypto()) {
PlatformPeg.get()?.reload();

Check warning on line 30 in src/settings/controllers/RustCryptoSdkController.ts

View workflow job for this annotation

GitHub Actions / Jest (2)

RETRY 1: MatrixClientPeg › .start › should show error modal when store database closes

TypeError: _PlatformPeg.default.get(...)?.reload is not a function at RustCryptoSdkController.reload [as onChange] (src/settings/controllers/RustCryptoSdkController.ts:30:32) at Function.onChange (src/settings/SettingsStore.ts:499:29) at MatrixClientPegClass.initClientCrypto (src/MatrixClientPeg.ts:311:9) at MatrixClientPegClass.assign (src/MatrixClientPeg.ts:265:13) at Object.<anonymous> (test/MatrixClientPeg-test.ts:162:13)

Check warning on line 30 in src/settings/controllers/RustCryptoSdkController.ts

View workflow job for this annotation

GitHub Actions / Jest (2)

RETRY 2: MatrixClientPeg › .start › should show error modal when store database closes

TypeError: _PlatformPeg.default.get(...)?.reload is not a function at RustCryptoSdkController.reload [as onChange] (src/settings/controllers/RustCryptoSdkController.ts:30:32) at Function.onChange (src/settings/SettingsStore.ts:499:29) at MatrixClientPegClass.initClientCrypto (src/MatrixClientPeg.ts:311:9) at MatrixClientPegClass.assign (src/MatrixClientPeg.ts:265:13) at Object.<anonymous> (test/MatrixClientPeg-test.ts:162:13)

Check failure on line 30 in src/settings/controllers/RustCryptoSdkController.ts

View workflow job for this annotation

GitHub Actions / Jest (2)

MatrixClientPeg › .start › should show error modal when store database closes

TypeError: _PlatformPeg.default.get(...)?.reload is not a function at RustCryptoSdkController.reload [as onChange] (src/settings/controllers/RustCryptoSdkController.ts:30:32) at Function.onChange (src/settings/SettingsStore.ts:499:29) at MatrixClientPegClass.initClientCrypto (src/MatrixClientPeg.ts:311:9) at MatrixClientPegClass.assign (src/MatrixClientPeg.ts:265:13) at Object.<anonymous> (test/MatrixClientPeg-test.ts:162:13)
}
}

public get settingDisabled(): boolean | string {
// Currently this can only be changed via config.json. In future, we'll allow the user to *enable* this setting
// via labs, which will migrate their existing device to the rust-sdk implementation.
return _t("labs|rust_crypto_disabled_notice");
if (!SettingsStore.getValueAt(SettingLevel.DEVICE, Features.RustCrypto)) {
// If rust crypto has not yet been enabled for this device, you can turn it on, IF YOU DARE
return false;
}

if (SettingsStore.getValueAt(SettingLevel.CONFIG, Features.RustCrypto)) {
// It's enabled in the config, so you can't get rid of it even by logging out.
return _t("labs|rust_crypto_in_config", { brand: SdkConfig.get().brand });
}

// The setting is enabled at the device level, but not mandated at the config level.
// You can only turn it off by logging out and in again.
return _t("labs|rust_crypto_requires_logout");
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -15,11 +15,12 @@ limitations under the License.
*/

import React from "react";
import { render, screen } from "@testing-library/react";
import { fireEvent, render, screen } from "@testing-library/react";

import LabsUserSettingsTab from "../../../../../../src/components/views/settings/tabs/user/LabsUserSettingsTab";
import SettingsStore from "../../../../../../src/settings/SettingsStore";
import SdkConfig from "../../../../../../src/SdkConfig";
import { SettingLevel } from "../../../../../../src/settings/SettingLevel";

describe("<LabsUserSettingsTab />", () => {
const defaultProps = {
Expand All @@ -34,6 +35,7 @@ describe("<LabsUserSettingsTab />", () => {
settingsValueSpy.mockReturnValue(false);
SdkConfig.reset();
SdkConfig.add({ brand: "BrandedClient" });
localStorage.clear();
});

it("renders settings marked as beta as beta cards", () => {
Expand All @@ -60,4 +62,82 @@ describe("<LabsUserSettingsTab />", () => {
const labsSections = container.getElementsByClassName("mx_SettingsSubsection");
expect(labsSections).toHaveLength(9);
});

describe("Rust crypto setting", () => {
const SETTING_NAME = "Rust cryptography implementation";

beforeEach(() => {
SdkConfig.add({ show_labs_settings: true });
});

describe("Not enabled in config", () => {
it("can be turned on if not already", async () => {
// By the time the settings panel is shown, `MatrixClientPeg.initClientCrypto` has saved the current
// value to the settings store.
await SettingsStore.setValue("feature_rust_crypto", null, SettingLevel.DEVICE, false);

const rendered = render(getComponent());
const toggle = rendered.getByRole("switch", { name: SETTING_NAME });
expect(toggle.getAttribute("aria-disabled")).toEqual("false");
expect(toggle.getAttribute("aria-checked")).toEqual("false");

const description = toggle.closest(".mx_SettingsFlag")?.querySelector(".mx_SettingsFlag_microcopy");
expect(description).toHaveTextContent(/To disable you will need to log out and back in/);
});

it("cannot be turned off once enabled", async () => {
await SettingsStore.setValue("feature_rust_crypto", null, SettingLevel.DEVICE, true);

const rendered = render(getComponent());
const toggle = rendered.getByRole("switch", { name: SETTING_NAME });
expect(toggle.getAttribute("aria-disabled")).toEqual("true");
expect(toggle.getAttribute("aria-checked")).toEqual("true");

// Hover over the toggle to make it show the tooltip
fireEvent.mouseOver(toggle);

const tooltip = rendered.getByRole("tooltip");
expect(tooltip).toHaveTextContent(
"Once enabled, Rust cryptography can only be disabled by logging out and in again",
);
});
});

describe("Enabled in config", () => {
beforeEach(() => {
SdkConfig.add({ features: { feature_rust_crypto: true } });
});

it("can be turned on if not already", async () => {
// By the time the settings panel is shown, `MatrixClientPeg.initClientCrypto` has saved the current
// value to the settings store.
await SettingsStore.setValue("feature_rust_crypto", null, SettingLevel.DEVICE, false);

const rendered = render(getComponent());
const toggle = rendered.getByRole("switch", { name: SETTING_NAME });
expect(toggle.getAttribute("aria-disabled")).toEqual("false");
expect(toggle.getAttribute("aria-checked")).toEqual("false");

const description = toggle.closest(".mx_SettingsFlag")?.querySelector(".mx_SettingsFlag_microcopy");
expect(description).toHaveTextContent(/It cannot be disabled/);
});

it("cannot be turned off once enabled", async () => {
await SettingsStore.setValue("feature_rust_crypto", null, SettingLevel.DEVICE, true);

const rendered = render(getComponent());
const toggle = rendered.getByRole("switch", { name: SETTING_NAME });
expect(toggle.getAttribute("aria-disabled")).toEqual("true");
expect(toggle.getAttribute("aria-checked")).toEqual("true");

// Hover over the toggle to make it show the tooltip
fireEvent.mouseOver(toggle);

const tooltip = rendered.getByRole("tooltip");
expect(tooltip).toHaveTextContent(
"Rust cryptography cannot be disabled on this deployment of BrandedClient",
);
});
});
});
});

0 comments on commit 773f5f6

Please sign in to comment.