From 34b8a7bd92e56c6e85c1376e692e68f91580e6ed Mon Sep 17 00:00:00 2001 From: Michael Weimann Date: Mon, 12 Dec 2022 15:56:06 +0100 Subject: [PATCH] When deleting a voice broadcast, also delete related events --- .../views/dialogs/ConfirmRedactDialog.tsx | 22 ++++- src/voice-broadcast/index.ts | 1 + .../utils/isVoiceBroadcastStartedEvent.ts | 24 +++++ .../dialogs/ConfirmRedactDialog-test.tsx | 98 +++++++++++++++++++ test/test-utils/test-utils.ts | 1 + 5 files changed, 144 insertions(+), 2 deletions(-) create mode 100644 src/voice-broadcast/utils/isVoiceBroadcastStartedEvent.ts create mode 100644 test/components/views/dialogs/ConfirmRedactDialog-test.tsx diff --git a/src/components/views/dialogs/ConfirmRedactDialog.tsx b/src/components/views/dialogs/ConfirmRedactDialog.tsx index 1f704fbe73e..73733a4bc5c 100644 --- a/src/components/views/dialogs/ConfirmRedactDialog.tsx +++ b/src/components/views/dialogs/ConfirmRedactDialog.tsx @@ -14,12 +14,14 @@ See the License for the specific language governing permissions and limitations under the License. */ -import { MatrixEvent } from 'matrix-js-sdk/src/models/event'; +import { Feature, ServerSupport } from 'matrix-js-sdk/src/feature'; +import { MatrixEvent, RelationType } from 'matrix-js-sdk/src/matrix'; import React from 'react'; import { _t } from '../../../languageHandler'; import { MatrixClientPeg } from '../../../MatrixClientPeg'; import Modal from '../../../Modal'; +import { isVoiceBroadcastStartedEvent } from '../../../voice-broadcast'; import ErrorDialog from './ErrorDialog'; import TextInputDialog from "./TextInputDialog"; @@ -58,13 +60,29 @@ export function createRedactEventDialog({ if (!proceed) return; const cli = MatrixClientPeg.get(); + const withRelations: { with_relations?: RelationType[] } = {}; + + // redact related events if this is a voice broadcast started event and + // server has support for relation based redactions + if (isVoiceBroadcastStartedEvent(mxEvent)) { + const relationBasedRedactionsSupport = cli.canSupport.get(Feature.RelationBasedRedactions); + if (relationBasedRedactionsSupport && relationBasedRedactionsSupport !== ServerSupport.Unsupported) { + withRelations.with_relations = [RelationType.Reference]; + } + } + try { onCloseDialog?.(); await cli.redactEvent( mxEvent.getRoomId(), mxEvent.getId(), undefined, - reason ? { reason } : {}, + { + ...( + reason ? { reason } : {} + ), + ...withRelations, + }, ); } catch (e) { const code = e.errcode || e.statusCode; diff --git a/src/voice-broadcast/index.ts b/src/voice-broadcast/index.ts index f71ce077ad8..49a4510cc36 100644 --- a/src/voice-broadcast/index.ts +++ b/src/voice-broadcast/index.ts @@ -47,6 +47,7 @@ export * from "./utils/doMaybeSetCurrentVoiceBroadcastPlayback"; export * from "./utils/getChunkLength"; export * from "./utils/getMaxBroadcastLength"; export * from "./utils/hasRoomLiveVoiceBroadcast"; +export * from "./utils/isVoiceBroadcastStartedEvent"; export * from "./utils/findRoomLiveVoiceBroadcastFromUserAndDevice"; export * from "./utils/shouldDisplayAsVoiceBroadcastRecordingTile"; export * from "./utils/shouldDisplayAsVoiceBroadcastTile"; diff --git a/src/voice-broadcast/utils/isVoiceBroadcastStartedEvent.ts b/src/voice-broadcast/utils/isVoiceBroadcastStartedEvent.ts new file mode 100644 index 00000000000..ddac137ea1b --- /dev/null +++ b/src/voice-broadcast/utils/isVoiceBroadcastStartedEvent.ts @@ -0,0 +1,24 @@ +/* +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. +*/ + +import { MatrixEvent } from "matrix-js-sdk/src/matrix"; + +import { VoiceBroadcastInfoEventType, VoiceBroadcastInfoState } from ".."; + +export const isVoiceBroadcastStartedEvent = (event: MatrixEvent): boolean => { + return event.getType() === VoiceBroadcastInfoEventType + && event.getContent()?.state === VoiceBroadcastInfoState.Started; +}; diff --git a/test/components/views/dialogs/ConfirmRedactDialog-test.tsx b/test/components/views/dialogs/ConfirmRedactDialog-test.tsx new file mode 100644 index 00000000000..0eebbf8a448 --- /dev/null +++ b/test/components/views/dialogs/ConfirmRedactDialog-test.tsx @@ -0,0 +1,98 @@ +/* +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. +*/ + +import { Feature, ServerSupport } from "matrix-js-sdk/src/feature"; +import { MatrixClient, MatrixEvent, RelationType } from "matrix-js-sdk/src/matrix"; +import { screen } from "@testing-library/react"; +import userEvent from "@testing-library/user-event"; + +import { flushPromises, stubClient } from "../../../test-utils"; +import { mkVoiceBroadcastInfoStateEvent } from "../../../voice-broadcast/utils/test-utils"; +import { VoiceBroadcastInfoState } from "../../../../src/voice-broadcast"; +import { createRedactEventDialog } from "../../../../src/components/views/dialogs/ConfirmRedactDialog"; + +describe("ConfirmRedactDialog", () => { + const roomId = "!room:example.com"; + let client: MatrixClient; + let mxEvent: MatrixEvent; + + const setUpVoiceBroadcastStartedEvent = () => { + mxEvent = mkVoiceBroadcastInfoStateEvent( + roomId, + VoiceBroadcastInfoState.Started, + client.getUserId()!, + client.deviceId!, + ); + }; + + const confirmDeleteVoiceBroadcastStartedEvent = async () => { + setUpVoiceBroadcastStartedEvent(); + createRedactEventDialog({ mxEvent }); + // double-flush promises required for the dialog to show up + await flushPromises(); + await flushPromises(); + + await userEvent.click(screen.getByTestId("dialog-primary-button")); + }; + + beforeEach(() => { + client = stubClient(); + }); + + describe("when the server does not support relation based redactions", () => { + beforeEach(() => { + client.canSupport.set(Feature.RelationBasedRedactions, ServerSupport.Unsupported); + }); + + describe("and displaying and confirm the dialog for a voice broadcast", () => { + beforeEach(async () => { + await confirmDeleteVoiceBroadcastStartedEvent(); + }); + + it("should call redact without `with_relations`", () => { + expect(client.redactEvent).toHaveBeenCalledWith( + roomId, + mxEvent.getId(), + undefined, + {}, + ); + }); + }); + }); + + describe("when the server supports relation based redactions", () => { + beforeEach(() => { + client.canSupport.set(Feature.RelationBasedRedactions, ServerSupport.Unstable); + }); + + describe("and displaying and confirm the dialog for a voice broadcast", () => { + beforeEach(async () => { + await confirmDeleteVoiceBroadcastStartedEvent(); + }); + + it("should call redact with `with_relations`", () => { + expect(client.redactEvent).toHaveBeenCalledWith( + roomId, + mxEvent.getId(), + undefined, + { + with_relations: [RelationType.Reference], + }, + ); + }); + }); + }); +}); diff --git a/test/test-utils/test-utils.ts b/test/test-utils/test-utils.ts index 69b626dfd5a..769cfef6e17 100644 --- a/test/test-utils/test-utils.ts +++ b/test/test-utils/test-utils.ts @@ -201,6 +201,7 @@ export function createTestClient(): MatrixClient { requestPasswordEmailToken: jest.fn().mockRejectedValue({}), setPassword: jest.fn().mockRejectedValue({}), groupCallEventHandler: { groupCalls: new Map() }, + redactEvent: jest.fn(), } as unknown as MatrixClient; client.reEmitter = new ReEmitter(client);