Add stable unstable version for jump to date before `v1.6` is fully supported on a homeserver (#10398)

Add stable unstable version (`org.matrix.msc3030.stable`) for jump to date [before `v1.6` is fully supported on a homeserver](https://github.com/matrix-org/synapse/issues/15089).

Related to https://github.com/vector-im/element-web/issues/24362 but does not solve immediately because Synapse does not supply `org.matrix.msc3030.stable` yet

Also refactored `ServerSupportUnstableFeatureController` to support multiple feature groups where any one of them will enable the setting. All features in a feature group are required. This way having either `org.matrix.msc3030` or `org.matrix.msc3030.stable` will enable the jump to date feature flag with a config of `[["org.matrix.msc3030"], ["org.matrix.msc3030.stable"]]`
pull/28788/head^2
Eric Eastwood 2023-03-17 15:08:17 -05:00 committed by GitHub
parent f3f87054b4
commit bc60e59eda
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 74 additions and 24 deletions

View File

@ -230,7 +230,7 @@ export const SETTINGS: { [setting: string]: ISetting } = {
controller: new ServerSupportUnstableFeatureController( controller: new ServerSupportUnstableFeatureController(
"feature_exploring_public_spaces", "feature_exploring_public_spaces",
defaultWatchManager, defaultWatchManager,
["org.matrix.msc3827.stable"], [["org.matrix.msc3827.stable"]],
undefined, undefined,
_td("Requires your server to support the stable version of MSC3827"), _td("Requires your server to support the stable version of MSC3827"),
), ),
@ -372,7 +372,7 @@ export const SETTINGS: { [setting: string]: ISetting } = {
controller: new ServerSupportUnstableFeatureController( controller: new ServerSupportUnstableFeatureController(
"feature_jump_to_date", "feature_jump_to_date",
defaultWatchManager, defaultWatchManager,
["org.matrix.msc3030"], [["org.matrix.msc3030"], ["org.matrix.msc3030.stable"]],
"v1.6", "v1.6",
_td("Requires your server to support MSC3030"), _td("Requires your server to support MSC3030"),
), ),
@ -388,7 +388,7 @@ export const SETTINGS: { [setting: string]: ISetting } = {
controller: new ServerSupportUnstableFeatureController( controller: new ServerSupportUnstableFeatureController(
"sendReadReceipts", "sendReadReceipts",
defaultWatchManager, defaultWatchManager,
["org.matrix.msc2285.stable"], [["org.matrix.msc2285.stable"]],
"v1.4", "v1.4",
_td("Your server doesn't support disabling sending read receipts."), _td("Your server doesn't support disabling sending read receipts."),
true, true,

View File

@ -26,12 +26,21 @@ import SettingsStore from "../SettingsStore";
* When a setting gets disabled or enabled from this controller it notifies the given WatchManager * When a setting gets disabled or enabled from this controller it notifies the given WatchManager
*/ */
export default class ServerSupportUnstableFeatureController extends MatrixClientBackedController { export default class ServerSupportUnstableFeatureController extends MatrixClientBackedController {
// Starts off as `undefined` so when we first compare the `newDisabledValue`, it sees
// it as a change and updates the watchers.
private enabled: boolean | undefined; private enabled: boolean | undefined;
/**
* Construct a new ServerSupportUnstableFeatureController.
*
* @param unstableFeatureGroups - If any one of the feature groups is satisfied,
* then the setting is considered enabled. A feature group is satisfied if all of
* the features in the group are supported (all features in a group are required).
*/
public constructor( public constructor(
private readonly settingName: string, private readonly settingName: string,
private readonly watchers: WatchManager, private readonly watchers: WatchManager,
private readonly unstableFeatures: string[], private readonly unstableFeatureGroups: string[][],
private readonly stableVersion?: string, private readonly stableVersion?: string,
private readonly disabledMessage?: string, private readonly disabledMessage?: string,
private readonly forcedValue: any = false, private readonly forcedValue: any = false,
@ -43,9 +52,9 @@ export default class ServerSupportUnstableFeatureController extends MatrixClient
return !this.enabled; return !this.enabled;
} }
public set disabled(v: boolean) { public set disabled(newDisabledValue: boolean) {
if (!v === this.enabled) return; if (!newDisabledValue === this.enabled) return;
this.enabled = !v; this.enabled = !newDisabledValue;
const level = SettingsStore.firstSupportedLevel(this.settingName); const level = SettingsStore.firstSupportedLevel(this.settingName);
if (!level) return; if (!level) return;
const settingValue = SettingsStore.getValue(this.settingName, null); const settingValue = SettingsStore.getValue(this.settingName, null);
@ -53,19 +62,33 @@ export default class ServerSupportUnstableFeatureController extends MatrixClient
} }
protected async initMatrixClient(oldClient: MatrixClient, newClient: MatrixClient): Promise<void> { protected async initMatrixClient(oldClient: MatrixClient, newClient: MatrixClient): Promise<void> {
this.disabled = true; // Check for stable version support first
let supported = true;
if (this.stableVersion && (await this.client.isVersionSupported(this.stableVersion))) { if (this.stableVersion && (await this.client.isVersionSupported(this.stableVersion))) {
this.disabled = false; this.disabled = false;
return; return;
} }
for (const feature of this.unstableFeatures) {
supported = await this.client.doesServerSupportUnstableFeature(feature); // Otherwise, only one of the unstable feature groups needs to be satisfied in
if (!supported) break; // order for this setting overall to be enabled
let isEnabled = false;
for (const featureGroup of this.unstableFeatureGroups) {
const featureSupportList = await Promise.all(
featureGroup.map(async (feature) => {
const isFeatureSupported = await this.client.doesServerSupportUnstableFeature(feature);
return isFeatureSupported;
}),
);
// Every feature in a feature group is required in order
// for this setting overall to be enabled.
const isFeatureGroupSatisfied = featureSupportList.every((isFeatureSupported) => isFeatureSupported);
if (isFeatureGroupSatisfied) {
isEnabled = true;
break;
}
} }
this.disabled = !supported; this.disabled = !isEnabled;
} }
public getValueOverride( public getValueOverride(

View File

@ -55,7 +55,7 @@ describe("ServerSupportUnstableFeatureController", () => {
const controller = new ServerSupportUnstableFeatureController( const controller = new ServerSupportUnstableFeatureController(
setting, setting,
watchers, watchers,
["feature"], [["feature"]],
undefined, undefined,
undefined, undefined,
"other_value", "other_value",
@ -74,7 +74,7 @@ describe("ServerSupportUnstableFeatureController", () => {
const controller = new ServerSupportUnstableFeatureController( const controller = new ServerSupportUnstableFeatureController(
setting, setting,
watchers, watchers,
["feature"], [["feature"]],
"other_value", "other_value",
); );
await prepareSetting(cli, controller); await prepareSetting(cli, controller);
@ -84,38 +84,65 @@ describe("ServerSupportUnstableFeatureController", () => {
}); });
describe("settingDisabled()", () => { describe("settingDisabled()", () => {
it("returns true if there is no matrix client", () => { it("considered disabled if there is no matrix client", () => {
const controller = new ServerSupportUnstableFeatureController(setting, watchers, ["org.matrix.msc3030"]); const controller = new ServerSupportUnstableFeatureController(setting, watchers, [["org.matrix.msc3030"]]);
expect(controller.settingDisabled).toEqual(true); expect(controller.settingDisabled).toEqual(true);
}); });
it("returns true if not all required features are supported", async () => { it("considered disabled if not all required features in the only feature group are supported", async () => {
const cli = stubClient(); const cli = stubClient();
cli.doesServerSupportUnstableFeature = jest.fn(async (featureName) => { cli.doesServerSupportUnstableFeature = jest.fn(async (featureName) => {
return featureName === "org.matrix.msc3827.stable"; return featureName === "org.matrix.msc3827.stable";
}); });
const controller = new ServerSupportUnstableFeatureController(setting, watchers, [ const controller = new ServerSupportUnstableFeatureController(setting, watchers, [
"org.matrix.msc3827.stable", ["org.matrix.msc3827.stable", "org.matrix.msc3030"],
"org.matrix.msc3030",
]); ]);
await prepareSetting(cli, controller); await prepareSetting(cli, controller);
expect(controller.settingDisabled).toEqual(true); expect(controller.settingDisabled).toEqual(true);
}); });
it("returns false if all required features are supported", async () => { it("considered enabled if all required features in the only feature group are supported", async () => {
const cli = stubClient(); const cli = stubClient();
cli.doesServerSupportUnstableFeature = jest.fn(async (featureName) => { cli.doesServerSupportUnstableFeature = jest.fn(async (featureName) => {
return featureName === "org.matrix.msc3827.stable" || featureName === "org.matrix.msc3030"; return featureName === "org.matrix.msc3827.stable" || featureName === "org.matrix.msc3030";
}); });
const controller = new ServerSupportUnstableFeatureController(setting, watchers, [ const controller = new ServerSupportUnstableFeatureController(setting, watchers, [
"org.matrix.msc3827.stable", ["org.matrix.msc3827.stable", "org.matrix.msc3030"],
"org.matrix.msc3030",
]); ]);
await prepareSetting(cli, controller); await prepareSetting(cli, controller);
expect(controller.settingDisabled).toEqual(false); expect(controller.settingDisabled).toEqual(false);
}); });
it("considered enabled if all required features in one of the feature groups are supported", async () => {
const cli = stubClient();
cli.doesServerSupportUnstableFeature = jest.fn(async (featureName) => {
return featureName === "org.matrix.msc3827.stable" || featureName === "org.matrix.msc3030";
});
const controller = new ServerSupportUnstableFeatureController(setting, watchers, [
["foo-unsupported", "bar-unsupported"],
["org.matrix.msc3827.stable", "org.matrix.msc3030"],
]);
await prepareSetting(cli, controller);
expect(controller.settingDisabled).toEqual(false);
});
it("considered disabled if not all required features in one of the feature groups are supported", async () => {
const cli = stubClient();
cli.doesServerSupportUnstableFeature = jest.fn(async (featureName) => {
return featureName === "org.matrix.msc3827.stable";
});
const controller = new ServerSupportUnstableFeatureController(setting, watchers, [
["foo-unsupported", "bar-unsupported"],
["org.matrix.msc3827.stable", "org.matrix.msc3030"],
]);
await prepareSetting(cli, controller);
expect(controller.settingDisabled).toEqual(true);
});
}); });
}); });