You've already forked matrix-js-sdk
mirror of
https://github.com/matrix-org/matrix-js-sdk.git
synced 2025-08-07 23:02:56 +03:00
This reverts commit bdd4d82cb3
.
This commit is contained in:
@@ -28,6 +28,7 @@ import {
|
|||||||
WidgetApiToWidgetAction,
|
WidgetApiToWidgetAction,
|
||||||
MatrixCapabilities,
|
MatrixCapabilities,
|
||||||
ITurnServer,
|
ITurnServer,
|
||||||
|
IRoomEvent,
|
||||||
IOpenIDCredentials,
|
IOpenIDCredentials,
|
||||||
ISendEventFromWidgetResponseData,
|
ISendEventFromWidgetResponseData,
|
||||||
WidgetApiResponseError,
|
WidgetApiResponseError,
|
||||||
@@ -634,20 +635,12 @@ describe("RoomWidgetClient", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it("receives", async () => {
|
it("receives", async () => {
|
||||||
const init = makeClient({ receiveState: [{ eventType: "org.example.foo", stateKey: "bar" }] });
|
await makeClient({ receiveState: [{ eventType: "org.example.foo", stateKey: "bar" }] });
|
||||||
expect(widgetApi.requestCapabilityForRoomTimeline).toHaveBeenCalledWith("!1:example.org");
|
expect(widgetApi.requestCapabilityForRoomTimeline).toHaveBeenCalledWith("!1:example.org");
|
||||||
expect(widgetApi.requestCapabilityToReceiveState).toHaveBeenCalledWith("org.example.foo", "bar");
|
expect(widgetApi.requestCapabilityToReceiveState).toHaveBeenCalledWith("org.example.foo", "bar");
|
||||||
// Client needs to be told that the room state is loaded
|
|
||||||
widgetApi.emit(
|
|
||||||
`action:${WidgetApiToWidgetAction.UpdateState}`,
|
|
||||||
new CustomEvent(`action:${WidgetApiToWidgetAction.UpdateState}`, { detail: { data: { state: [] } } }),
|
|
||||||
);
|
|
||||||
await init;
|
|
||||||
|
|
||||||
const emittedEvent = new Promise<MatrixEvent>((resolve) => client.once(ClientEvent.Event, resolve));
|
const emittedEvent = new Promise<MatrixEvent>((resolve) => client.once(ClientEvent.Event, resolve));
|
||||||
const emittedSync = new Promise<SyncState>((resolve) => client.once(ClientEvent.Sync, resolve));
|
const emittedSync = new Promise<SyncState>((resolve) => client.once(ClientEvent.Sync, resolve));
|
||||||
// Let's assume that a state event comes in but it doesn't actually
|
|
||||||
// update the state of the room just yet (maybe it's unauthorized)
|
|
||||||
widgetApi.emit(
|
widgetApi.emit(
|
||||||
`action:${WidgetApiToWidgetAction.SendEvent}`,
|
`action:${WidgetApiToWidgetAction.SendEvent}`,
|
||||||
new CustomEvent(`action:${WidgetApiToWidgetAction.SendEvent}`, { detail: { data: event } }),
|
new CustomEvent(`action:${WidgetApiToWidgetAction.SendEvent}`, { detail: { data: event } }),
|
||||||
@@ -656,43 +649,26 @@ describe("RoomWidgetClient", () => {
|
|||||||
// The client should've emitted about the received event
|
// The client should've emitted about the received event
|
||||||
expect((await emittedEvent).getEffectiveEvent()).toEqual(event);
|
expect((await emittedEvent).getEffectiveEvent()).toEqual(event);
|
||||||
expect(await emittedSync).toEqual(SyncState.Syncing);
|
expect(await emittedSync).toEqual(SyncState.Syncing);
|
||||||
// However it should not have changed the room state
|
// It should've also inserted the event into the room object
|
||||||
const room = client.getRoom("!1:example.org");
|
const room = client.getRoom("!1:example.org");
|
||||||
expect(room!.currentState.getStateEvents("org.example.foo", "bar")).toBe(null);
|
expect(room).not.toBeNull();
|
||||||
|
|
||||||
// Now assume that the state event becomes favored by state
|
|
||||||
// resolution for whatever reason and enters into the current state
|
|
||||||
// of the room
|
|
||||||
widgetApi.emit(
|
|
||||||
`action:${WidgetApiToWidgetAction.UpdateState}`,
|
|
||||||
new CustomEvent(`action:${WidgetApiToWidgetAction.UpdateState}`, {
|
|
||||||
detail: { data: { state: [event] } },
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
// It should now have changed the room state
|
|
||||||
expect(room!.currentState.getStateEvents("org.example.foo", "bar")?.getEffectiveEvent()).toEqual(event);
|
expect(room!.currentState.getStateEvents("org.example.foo", "bar")?.getEffectiveEvent()).toEqual(event);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("ignores state updates for other rooms", async () => {
|
it("backfills", async () => {
|
||||||
const init = makeClient({ receiveState: [{ eventType: "org.example.foo", stateKey: "bar" }] });
|
widgetApi.readStateEvents.mockImplementation(async (eventType, limit, stateKey) =>
|
||||||
// Client needs to be told that the room state is loaded
|
eventType === "org.example.foo" && (limit ?? Infinity) > 0 && stateKey === "bar"
|
||||||
widgetApi.emit(
|
? [event as IRoomEvent]
|
||||||
`action:${WidgetApiToWidgetAction.UpdateState}`,
|
: [],
|
||||||
new CustomEvent(`action:${WidgetApiToWidgetAction.UpdateState}`, { detail: { data: { state: [] } } }),
|
|
||||||
);
|
);
|
||||||
await init;
|
|
||||||
|
|
||||||
// Now a room we're not interested in receives a state update
|
await makeClient({ receiveState: [{ eventType: "org.example.foo", stateKey: "bar" }] });
|
||||||
widgetApi.emit(
|
expect(widgetApi.requestCapabilityForRoomTimeline).toHaveBeenCalledWith("!1:example.org");
|
||||||
`action:${WidgetApiToWidgetAction.UpdateState}`,
|
expect(widgetApi.requestCapabilityToReceiveState).toHaveBeenCalledWith("org.example.foo", "bar");
|
||||||
new CustomEvent(`action:${WidgetApiToWidgetAction.UpdateState}`, {
|
|
||||||
detail: { data: { state: [{ ...event, room_id: "!other-room:example.org" }] } },
|
const room = client.getRoom("!1:example.org");
|
||||||
}),
|
expect(room).not.toBeNull();
|
||||||
);
|
expect(room!.currentState.getStateEvents("org.example.foo", "bar")?.getEffectiveEvent()).toEqual(event);
|
||||||
// No change to the room state
|
|
||||||
for (const room of client.getRooms()) {
|
|
||||||
expect(room.currentState.getStateEvents("org.example.foo", "bar")).toBe(null);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@@ -28,7 +28,6 @@ import {
|
|||||||
WidgetApiAction,
|
WidgetApiAction,
|
||||||
IWidgetApiResponse,
|
IWidgetApiResponse,
|
||||||
IWidgetApiResponseData,
|
IWidgetApiResponseData,
|
||||||
IUpdateStateToWidgetActionRequest,
|
|
||||||
} from "matrix-widget-api";
|
} from "matrix-widget-api";
|
||||||
|
|
||||||
import { MatrixEvent, IEvent, IContent, EventStatus } from "./models/event.ts";
|
import { MatrixEvent, IEvent, IContent, EventStatus } from "./models/event.ts";
|
||||||
@@ -137,7 +136,6 @@ export type EventHandlerMap = { [RoomWidgetClientEvent.PendingEventsChanged]: ()
|
|||||||
export class RoomWidgetClient extends MatrixClient {
|
export class RoomWidgetClient extends MatrixClient {
|
||||||
private room?: Room;
|
private room?: Room;
|
||||||
private readonly widgetApiReady: Promise<void>;
|
private readonly widgetApiReady: Promise<void>;
|
||||||
private readonly roomStateSynced: Promise<void>;
|
|
||||||
private lifecycle?: AbortController;
|
private lifecycle?: AbortController;
|
||||||
private syncState: SyncState | null = null;
|
private syncState: SyncState | null = null;
|
||||||
|
|
||||||
@@ -191,11 +189,6 @@ export class RoomWidgetClient extends MatrixClient {
|
|||||||
};
|
};
|
||||||
|
|
||||||
this.widgetApiReady = new Promise<void>((resolve) => this.widgetApi.once("ready", resolve));
|
this.widgetApiReady = new Promise<void>((resolve) => this.widgetApi.once("ready", resolve));
|
||||||
this.roomStateSynced = capabilities.receiveState?.length
|
|
||||||
? new Promise<void>((resolve) =>
|
|
||||||
this.widgetApi.once(`action:${WidgetApiToWidgetAction.UpdateState}`, resolve),
|
|
||||||
)
|
|
||||||
: Promise.resolve();
|
|
||||||
|
|
||||||
// Request capabilities for the functionality this client needs to support
|
// Request capabilities for the functionality this client needs to support
|
||||||
if (
|
if (
|
||||||
@@ -248,7 +241,6 @@ export class RoomWidgetClient extends MatrixClient {
|
|||||||
|
|
||||||
widgetApi.on(`action:${WidgetApiToWidgetAction.SendEvent}`, this.onEvent);
|
widgetApi.on(`action:${WidgetApiToWidgetAction.SendEvent}`, this.onEvent);
|
||||||
widgetApi.on(`action:${WidgetApiToWidgetAction.SendToDevice}`, this.onToDevice);
|
widgetApi.on(`action:${WidgetApiToWidgetAction.SendToDevice}`, this.onToDevice);
|
||||||
widgetApi.on(`action:${WidgetApiToWidgetAction.UpdateState}`, this.onStateUpdate);
|
|
||||||
|
|
||||||
// Open communication with the host
|
// Open communication with the host
|
||||||
widgetApi.start();
|
widgetApi.start();
|
||||||
@@ -284,6 +276,28 @@ export class RoomWidgetClient extends MatrixClient {
|
|||||||
|
|
||||||
await this.widgetApiReady;
|
await this.widgetApiReady;
|
||||||
|
|
||||||
|
// Backfill the requested events
|
||||||
|
// We only get the most recent event for every type + state key combo,
|
||||||
|
// so it doesn't really matter what order we inject them in
|
||||||
|
await Promise.all(
|
||||||
|
this.capabilities.receiveState?.map(async ({ eventType, stateKey }) => {
|
||||||
|
const rawEvents = await this.widgetApi.readStateEvents(eventType, undefined, stateKey, [this.roomId]);
|
||||||
|
const events = rawEvents.map((rawEvent) => new MatrixEvent(rawEvent as Partial<IEvent>));
|
||||||
|
|
||||||
|
if (this.syncApi instanceof SyncApi) {
|
||||||
|
// Passing undefined for `stateAfterEventList` allows will make `injectRoomEvents` run in legacy mode
|
||||||
|
// -> state events in `timelineEventList` will update the state.
|
||||||
|
await this.syncApi.injectRoomEvents(this.room!, undefined, events);
|
||||||
|
} else {
|
||||||
|
await this.syncApi!.injectRoomEvents(this.room!, events); // Sliding Sync
|
||||||
|
}
|
||||||
|
events.forEach((event) => {
|
||||||
|
this.emit(ClientEvent.Event, event);
|
||||||
|
logger.info(`Backfilled event ${event.getId()} ${event.getType()} ${event.getStateKey()}`);
|
||||||
|
});
|
||||||
|
}) ?? [],
|
||||||
|
);
|
||||||
|
|
||||||
if (opts.clientWellKnownPollPeriod !== undefined) {
|
if (opts.clientWellKnownPollPeriod !== undefined) {
|
||||||
this.clientWellKnownIntervalID = setInterval(() => {
|
this.clientWellKnownIntervalID = setInterval(() => {
|
||||||
this.fetchClientWellKnown();
|
this.fetchClientWellKnown();
|
||||||
@@ -291,9 +305,8 @@ export class RoomWidgetClient extends MatrixClient {
|
|||||||
this.fetchClientWellKnown();
|
this.fetchClientWellKnown();
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.roomStateSynced;
|
|
||||||
this.setSyncState(SyncState.Syncing);
|
this.setSyncState(SyncState.Syncing);
|
||||||
logger.info("Finished initial sync");
|
logger.info("Finished backfilling events");
|
||||||
|
|
||||||
this.matrixRTC.start();
|
this.matrixRTC.start();
|
||||||
|
|
||||||
@@ -304,7 +317,6 @@ export class RoomWidgetClient extends MatrixClient {
|
|||||||
public stopClient(): void {
|
public stopClient(): void {
|
||||||
this.widgetApi.off(`action:${WidgetApiToWidgetAction.SendEvent}`, this.onEvent);
|
this.widgetApi.off(`action:${WidgetApiToWidgetAction.SendEvent}`, this.onEvent);
|
||||||
this.widgetApi.off(`action:${WidgetApiToWidgetAction.SendToDevice}`, this.onToDevice);
|
this.widgetApi.off(`action:${WidgetApiToWidgetAction.SendToDevice}`, this.onToDevice);
|
||||||
this.widgetApi.off(`action:${WidgetApiToWidgetAction.UpdateState}`, this.onStateUpdate);
|
|
||||||
|
|
||||||
super.stopClient();
|
super.stopClient();
|
||||||
this.lifecycle!.abort(); // Signal to other async tasks that the client has stopped
|
this.lifecycle!.abort(); // Signal to other async tasks that the client has stopped
|
||||||
@@ -562,15 +574,36 @@ export class RoomWidgetClient extends MatrixClient {
|
|||||||
// Only inject once we have update the txId
|
// Only inject once we have update the txId
|
||||||
await this.updateTxId(event);
|
await this.updateTxId(event);
|
||||||
|
|
||||||
|
// The widget API does not tell us whether a state event came from `state_after` or not so we assume legacy behaviour for now.
|
||||||
if (this.syncApi instanceof SyncApi) {
|
if (this.syncApi instanceof SyncApi) {
|
||||||
await this.syncApi.injectRoomEvents(this.room!, undefined, [], [event]);
|
// The code will want to be something like:
|
||||||
|
// ```
|
||||||
|
// if (!params.addToTimeline && !params.addToState) {
|
||||||
|
// // Passing undefined for `stateAfterEventList` makes `injectRoomEvents` run in "legacy mode"
|
||||||
|
// // -> state events part of the `timelineEventList` parameter will update the state.
|
||||||
|
// this.injectRoomEvents(this.room!, [], undefined, [event]);
|
||||||
|
// } else {
|
||||||
|
// this.injectRoomEvents(this.room!, undefined, params.addToState ? [event] : [], params.addToTimeline ? [event] : []);
|
||||||
|
// }
|
||||||
|
// ```
|
||||||
|
|
||||||
|
// Passing undefined for `stateAfterEventList` allows will make `injectRoomEvents` run in legacy mode
|
||||||
|
// -> state events in `timelineEventList` will update the state.
|
||||||
|
await this.syncApi.injectRoomEvents(this.room!, [], undefined, [event]);
|
||||||
} else {
|
} else {
|
||||||
// Sliding Sync
|
// The code will want to be something like:
|
||||||
await this.syncApi!.injectRoomEvents(this.room!, [], [event]);
|
// ```
|
||||||
|
// if (!params.addToTimeline && !params.addToState) {
|
||||||
|
// this.injectRoomEvents(this.room!, [], [event]);
|
||||||
|
// } else {
|
||||||
|
// this.injectRoomEvents(this.room!, params.addToState ? [event] : [], params.addToTimeline ? [event] : []);
|
||||||
|
// }
|
||||||
|
// ```
|
||||||
|
await this.syncApi!.injectRoomEvents(this.room!, [], [event]); // Sliding Sync
|
||||||
}
|
}
|
||||||
this.emit(ClientEvent.Event, event);
|
this.emit(ClientEvent.Event, event);
|
||||||
this.setSyncState(SyncState.Syncing);
|
this.setSyncState(SyncState.Syncing);
|
||||||
logger.info(`Received event ${event.getId()} ${event.getType()}`);
|
logger.info(`Received event ${event.getId()} ${event.getType()} ${event.getStateKey()}`);
|
||||||
} else {
|
} else {
|
||||||
const { event_id: eventId, room_id: roomId } = ev.detail.data;
|
const { event_id: eventId, room_id: roomId } = ev.detail.data;
|
||||||
logger.info(`Received event ${eventId} for a different room ${roomId}; discarding`);
|
logger.info(`Received event ${eventId} for a different room ${roomId}; discarding`);
|
||||||
@@ -595,32 +628,6 @@ export class RoomWidgetClient extends MatrixClient {
|
|||||||
await this.ack(ev);
|
await this.ack(ev);
|
||||||
};
|
};
|
||||||
|
|
||||||
private onStateUpdate = async (ev: CustomEvent<IUpdateStateToWidgetActionRequest>): Promise<void> => {
|
|
||||||
ev.preventDefault();
|
|
||||||
|
|
||||||
for (const rawEvent of ev.detail.data.state) {
|
|
||||||
// Verify the room ID matches, since it's possible for the client to
|
|
||||||
// send us state updates from other rooms if this widget is always
|
|
||||||
// on screen
|
|
||||||
if (rawEvent.room_id === this.roomId) {
|
|
||||||
const event = new MatrixEvent(rawEvent as Partial<IEvent>);
|
|
||||||
|
|
||||||
if (this.syncApi instanceof SyncApi) {
|
|
||||||
await this.syncApi.injectRoomEvents(this.room!, undefined, [event]);
|
|
||||||
} else {
|
|
||||||
// Sliding Sync
|
|
||||||
await this.syncApi!.injectRoomEvents(this.room!, [event]);
|
|
||||||
}
|
|
||||||
logger.info(`Updated state entry ${event.getType()} ${event.getStateKey()} to ${event.getId()}`);
|
|
||||||
} else {
|
|
||||||
const { event_id: eventId, room_id: roomId } = ev.detail.data;
|
|
||||||
logger.info(`Received state entry ${eventId} for a different room ${roomId}; discarding`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
await this.ack(ev);
|
|
||||||
};
|
|
||||||
|
|
||||||
private async watchTurnServers(): Promise<void> {
|
private async watchTurnServers(): Promise<void> {
|
||||||
const servers = this.widgetApi.getTurnServers();
|
const servers = this.widgetApi.getTurnServers();
|
||||||
const onClientStopped = (): void => {
|
const onClientStopped = (): void => {
|
||||||
|
@@ -4875,9 +4875,9 @@ matrix-mock-request@^2.5.0:
|
|||||||
expect "^28.1.0"
|
expect "^28.1.0"
|
||||||
|
|
||||||
matrix-widget-api@^1.10.0:
|
matrix-widget-api@^1.10.0:
|
||||||
version "1.12.0"
|
version "1.10.0"
|
||||||
resolved "https://registry.yarnpkg.com/matrix-widget-api/-/matrix-widget-api-1.12.0.tgz#b3d22bab1670051c8eeee66bb96d08b33148bc99"
|
resolved "https://registry.yarnpkg.com/matrix-widget-api/-/matrix-widget-api-1.10.0.tgz#d31ea073a5871a1fb1a511ef900b0c125a37bf55"
|
||||||
integrity sha512-6JRd9fJGGvuBRhcTg9wX+Skn/Q1wox3jdp5yYQKJ6pPw4urW9bkTR90APBKVDB1vorJKT44jml+lCzkDMRBjww==
|
integrity sha512-rkAJ29briYV7TJnfBVLVSKtpeBrBju15JZFSDP6wj8YdbCu1bdmlplJayQ+vYaw1x4fzI49Q+Nz3E85s46sRDw==
|
||||||
dependencies:
|
dependencies:
|
||||||
"@types/events" "^3.0.0"
|
"@types/events" "^3.0.0"
|
||||||
events "^3.2.0"
|
events "^3.2.0"
|
||||||
|
Reference in New Issue
Block a user