Skip to content

Commit

Permalink
Make persist opts optional, refactor and changeset
Browse files Browse the repository at this point in the history
Signed-off-by: Milton Moura <[email protected]>
  • Loading branch information
mgcm committed Nov 18, 2024
1 parent a53a27d commit f3e0c69
Show file tree
Hide file tree
Showing 5 changed files with 99 additions and 55 deletions.
6 changes: 6 additions & 0 deletions .changeset/eleven-islands-talk.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
---
'@nordeck/matrix-neoboard-widget': minor
'@nordeck/matrix-neoboard-react-sdk': minor
---

Send snapshot on membership changes that don't have access to room history
Original file line number Diff line number Diff line change
Expand Up @@ -208,7 +208,7 @@ describe('<FallbackSnapshotProvider />', () => {
['unencrypted', undefined],
['encrypted', { event: mockRoomEncryption() }],
])(
'should persist snapshot immediatly if the last event is an invite and inviter is the current user',
'should persist snapshot immediatly if the last event is an invite and inviter is the current user (%s room)',
(_, encryptionState) => {
(useGetRoomMembersQuery as Mock).mockReturnValue({
data: mockRoomMembersOwnUser,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ import {
} from '@matrix-widget-toolkit/api';
import { useWidgetApi } from '@matrix-widget-toolkit/react';
import { PropsWithChildren, useEffect } from 'react';
import { RoomEncryptionEvent, RoomHistoryVisibilityEvent } from '../../model';
import { useActiveWhiteboardInstance } from '../../state';
import {
useGetRoomEncryptionQuery,
Expand All @@ -35,37 +36,23 @@ export function FallbackSnapshotProvider({ children }: PropsWithChildren<{}>) {
const ownUserId = useWidgetApi().widgetParameters.userId;

useEffect(() => {
const isRoomEncrypted =
(roomEncryption && roomEncryption.event?.content?.algorithm) !==
undefined;
const roomHistoryVisibilityValue =
roomHistoryVisibility &&
roomHistoryVisibility?.event?.content?.history_visibility;
const isRoomHistoryVisibilityShared =
roomHistoryVisibilityValue && roomHistoryVisibilityValue === 'shared';
const sendFallbackSnapshot =
isRoomEncrypted ||
(!isRoomHistoryVisibilityShared &&
roomHistoryVisibilityValue !== undefined);
const isRoomEncrypted = checkIfRoomEncrypted(roomEncryption);
const roomHistoryVisibilityValue = getRoomHistoryVisibilityValue(
roomHistoryVisibility,
);
const isRoomHistoryVisibilityShared = checkIfRoomHistoryVisibilityShared(
roomHistoryVisibilityValue,
);
const sendFallbackSnapshot = shouldSendFallbackSnapshot(
isRoomEncrypted,
isRoomHistoryVisibilityShared,
roomHistoryVisibilityValue,
);

if (sendFallbackSnapshot) {
let membershipFilter = (
member: StateEvent<RoomMemberStateEventContent>,
) => {
return (
member.content.membership == 'invite' ||
member.content.membership == 'join'
);
};
if (roomHistoryVisibilityValue == 'joined') {
membershipFilter = (
member: StateEvent<RoomMemberStateEventContent>,
) => {
return member.content.membership == 'join';
};
}

const membershipFilter = getMembershipFilter(roomHistoryVisibilityValue);
const members = roomMembers?.entities;

if (members) {
const invitesOrJoins = Object.values(members).filter(membershipFilter);
const sortedInvitesOrJoins = invitesOrJoins.sort((a, b) => {
Expand All @@ -74,33 +61,18 @@ export function FallbackSnapshotProvider({ children }: PropsWithChildren<{}>) {
const lastInviteOrJoin = sortedInvitesOrJoins[0];

if (
lastInviteOrJoin.content.membership == 'join' &&
roomHistoryVisibilityValue == 'invited'
shouldUseInviteTimestamp(lastInviteOrJoin, roomHistoryVisibilityValue)
) {
// get the invite instead
const joinEventObject = Object(lastInviteOrJoin);
const prevEventWasInvite =
joinEventObject['unsigned'].prev_content?.membership == 'invite';
if (prevEventWasInvite) {
const _inviteEventId = joinEventObject['unsigned']?.replaces_state;
// TODO: we now need to get the timestamp of the invite event and use that instead
// but the Widget API doesn't have a way to get an event by event ID
// and the `readRoomEvents` is filtering out events with state_key set
}
// TODO: Implement logic to get the timestamp of the invite event.
// There is currently no way to get an event by id or
// fetch an older state event from the room timeline using the Widget API
}

let immediate = false;
if (
lastInviteOrJoin.sender == ownUserId &&
lastInviteOrJoin.content.membership == 'invite'
) {
// if the sender is the current user, try to persist immediatly
immediate = true;
}
const immediate = shouldPersistImmediately(lastInviteOrJoin, ownUserId);

whiteboardInstance.persist({
timestamp: lastInviteOrJoin.origin_server_ts,
immediate: immediate,
immediate,
});
}
}
Expand All @@ -114,3 +86,69 @@ export function FallbackSnapshotProvider({ children }: PropsWithChildren<{}>) {

return <>{children}</>;
}

function checkIfRoomEncrypted(roomEncryption?: {
event: StateEvent<RoomEncryptionEvent> | undefined;
}): boolean {
return (
(roomEncryption && roomEncryption.event?.content?.algorithm) !== undefined
);
}

function getRoomHistoryVisibilityValue(roomHistoryVisibility?: {
event: StateEvent<RoomHistoryVisibilityEvent> | undefined;
}): string | undefined {
return roomHistoryVisibility?.event?.content?.history_visibility;
}

function checkIfRoomHistoryVisibilityShared(
roomHistoryVisibilityValue: string | undefined,
): boolean {
return roomHistoryVisibilityValue === 'shared';
}

function shouldSendFallbackSnapshot(
isRoomEncrypted: boolean,
isRoomHistoryVisibilityShared: boolean,
roomHistoryVisibilityValue: string | undefined,
): boolean {
return (
isRoomEncrypted ||
(!isRoomHistoryVisibilityShared && roomHistoryVisibilityValue !== undefined)
);
}

function getMembershipFilter(
roomHistoryVisibilityValue: string | undefined,
): (member: StateEvent<RoomMemberStateEventContent>) => boolean {
if (roomHistoryVisibilityValue === 'joined') {
return (member) => member.content.membership === 'join';
}
return (member) =>
member.content.membership === 'invite' ||
member.content.membership === 'join';
}

function shouldUseInviteTimestamp(
lastInviteOrJoin: StateEvent<RoomMemberStateEventContent>,
roomHistoryVisibilityValue: string | undefined,
) {
if (
lastInviteOrJoin.content.membership === 'join' &&
roomHistoryVisibilityValue === 'invited'
) {
const joinEventObject = Object(lastInviteOrJoin);
return joinEventObject['unsigned'].prev_content?.membership === 'invite';
}
return false;
}

function shouldPersistImmediately(
lastInviteOrJoin: StateEvent<RoomMemberStateEventContent>,
ownUserId: string | undefined,
) {
return (
lastInviteOrJoin.sender === ownUserId &&
lastInviteOrJoin.content.membership === 'invite'
);
}
6 changes: 3 additions & 3 deletions packages/react-sdk/src/state/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -56,8 +56,8 @@ export type WhiteboardStatistics = {
};

export type PersistOptions = {
timestamp?: number;
immediate?: boolean;
timestamp: number;
immediate: boolean;
};

/** An instance of a whiteboard that can be used to read and manipulate it. */
Expand Down Expand Up @@ -153,7 +153,7 @@ export type WhiteboardInstance = {
destroy(): void;

/** Persist the whiteboard state. */
persist(options: PersistOptions): Promise<void>;
persist(options?: PersistOptions): Promise<void>;
};

export type ElementUpdate = {
Expand Down
4 changes: 2 additions & 2 deletions packages/react-sdk/src/state/whiteboardInstanceImpl.ts
Original file line number Diff line number Diff line change
Expand Up @@ -453,8 +453,8 @@ export class WhiteboardInstanceImpl implements WhiteboardInstance {
this.communicationChannel.destroy();
}

async persist(options: PersistOptions): Promise<void> {
if (options.timestamp !== undefined && options.immediate !== undefined) {
async persist(options?: PersistOptions): Promise<void> {
if (options !== undefined) {
const snapshot = this.synchronizedDocument.getLatestDocumentSnapshot();
if (snapshot && snapshot.origin_server_ts < options.timestamp) {
if (options.immediate) {
Expand Down

0 comments on commit f3e0c69

Please sign in to comment.