signal-desktop/ts/test-both/reactions/util_test.ts
2022-04-28 18:06:28 -04:00

308 lines
8.9 KiB
TypeScript

// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import { assert } from 'chai';
import { v4 as uuid } from 'uuid';
import { omit } from 'lodash';
import type {
MessageAttributesType,
MessageReactionType,
} from '../../model-types.d';
import { isEmpty } from '../../util/iterables';
import {
addOutgoingReaction,
getNewestPendingOutgoingReaction,
getUnsentConversationIds,
markOutgoingReactionFailed,
markOutgoingReactionSent,
} from '../../reactions/util';
describe('reaction utilities', () => {
const OUR_CONVO_ID = uuid();
const rxn = (
emoji: undefined | string,
{ isPending = false }: Readonly<{ isPending?: boolean }> = {}
): MessageReactionType => ({
emoji,
fromId: OUR_CONVO_ID,
targetAuthorUuid: uuid(),
targetTimestamp: Date.now(),
timestamp: Date.now(),
...(isPending ? { isSentByConversationId: { [uuid()]: false } } : {}),
});
describe('addOutgoingReaction', () => {
it('adds the reaction to the end of an empty list', () => {
const reaction = rxn('💅');
const result = addOutgoingReaction([], reaction);
assert.deepStrictEqual(result, [reaction]);
});
it('removes all pending reactions', () => {
const oldReactions = [
{ ...rxn('😭', { isPending: true }), timestamp: 3 },
{ ...rxn('💬'), fromId: uuid() },
{ ...rxn('🥀', { isPending: true }), timestamp: 1 },
{ ...rxn('🌹', { isPending: true }), timestamp: 2 },
];
const reaction = rxn('😀');
const newReactions = addOutgoingReaction(oldReactions, reaction);
assert.deepStrictEqual(newReactions, [oldReactions[1], reaction]);
});
it('does not remove any pending reactions if its a story', () => {
const oldReactions = [
{ ...rxn('😭', { isPending: true }), timestamp: 3 },
{ ...rxn('💬'), fromId: uuid() },
{ ...rxn('🥀', { isPending: true }), timestamp: 1 },
{ ...rxn('🌹', { isPending: true }), timestamp: 2 },
];
const reaction = rxn('😀');
const newReactions = addOutgoingReaction(oldReactions, reaction, true);
assert.deepStrictEqual(newReactions, [...oldReactions, reaction]);
});
});
describe('getNewestPendingOutgoingReaction', () => {
it('returns undefined if there are no pending outgoing reactions', () => {
[[], [rxn('🔔')], [rxn('😭'), { ...rxn('💬'), fromId: uuid() }]].forEach(
oldReactions => {
assert.deepStrictEqual(
getNewestPendingOutgoingReaction(oldReactions, OUR_CONVO_ID),
{}
);
}
);
});
it("returns undefined if there's a pending reaction before a fully sent one", () => {
const oldReactions = [
{ ...rxn('⭐️'), timestamp: 2 },
{ ...rxn('🔥', { isPending: true }), timestamp: 1 },
];
const { pendingReaction, emojiToRemove } =
getNewestPendingOutgoingReaction(oldReactions, OUR_CONVO_ID);
assert.isUndefined(pendingReaction);
assert.isUndefined(emojiToRemove);
});
it('returns the newest pending reaction', () => {
[
[rxn('⭐️', { isPending: true })],
[
{ ...rxn('🥀', { isPending: true }), timestamp: 1 },
{ ...rxn('⭐️', { isPending: true }), timestamp: 2 },
],
].forEach(oldReactions => {
const { pendingReaction, emojiToRemove } =
getNewestPendingOutgoingReaction(oldReactions, OUR_CONVO_ID);
assert.strictEqual(pendingReaction?.emoji, '⭐️');
assert.isUndefined(emojiToRemove);
});
});
it('makes its best guess of an emoji to remove, if applicable', () => {
const oldReactions = [
{ ...rxn('⭐️'), timestamp: 1 },
{ ...rxn(undefined, { isPending: true }), timestamp: 3 },
{ ...rxn('🔥', { isPending: true }), timestamp: 2 },
];
const { pendingReaction, emojiToRemove } =
getNewestPendingOutgoingReaction(oldReactions, OUR_CONVO_ID);
assert.isDefined(pendingReaction);
assert.isUndefined(pendingReaction?.emoji);
assert.strictEqual(emojiToRemove, '⭐️');
});
});
describe('getUnsentConversationIds', () => {
it("returns an empty iterable if there's nothing to send", () => {
assert(isEmpty(getUnsentConversationIds({})));
assert(
isEmpty(
getUnsentConversationIds({
isSentByConversationId: { [uuid()]: true },
})
)
);
});
it('returns an iterable of all unsent conversation IDs', () => {
const unsent1 = uuid();
const unsent2 = uuid();
const fakeReaction = {
isSentByConversationId: {
[unsent1]: false,
[unsent2]: false,
[uuid()]: true,
[uuid()]: true,
},
};
assert.sameMembers(
[...getUnsentConversationIds(fakeReaction)],
[unsent1, unsent2]
);
});
});
describe('markReactionFailed', () => {
const fullySent = rxn('⭐️');
const partiallySent = {
...rxn('🔥'),
isSentByConversationId: { [uuid()]: true, [uuid()]: false },
};
const unsent = rxn('🤫', { isPending: true });
const reactions = [fullySent, partiallySent, unsent];
it('removes the pending state if the reaction, with emoji, was partially sent', () => {
assert.deepStrictEqual(
markOutgoingReactionFailed(reactions, partiallySent),
[fullySent, omit(partiallySent, 'isSentByConversationId'), unsent]
);
});
it('removes the removal reaction', () => {
const none = rxn(undefined, { isPending: true });
assert.isEmpty(markOutgoingReactionFailed([none], none));
});
it('does nothing if the reaction is not in the list', () => {
assert.deepStrictEqual(
markOutgoingReactionFailed(reactions, rxn('🥀', { isPending: true })),
reactions
);
});
it('removes the completely-unsent emoji reaction', () => {
assert.deepStrictEqual(markOutgoingReactionFailed(reactions, unsent), [
fullySent,
partiallySent,
]);
});
});
describe('markOutgoingReactionSent', () => {
const uuid1 = uuid();
const uuid2 = uuid();
const uuid3 = uuid();
const star = {
...rxn('⭐️'),
timestamp: 2,
isSentByConversationId: {
[uuid1]: false,
[uuid2]: false,
[uuid3]: false,
},
};
const none = {
...rxn(undefined),
timestamp: 3,
isSentByConversationId: {
[uuid1]: false,
[uuid2]: false,
[uuid3]: false,
},
};
const reactions = [star, none, { ...rxn('🔕'), timestamp: 1 }];
function getMessage(): MessageAttributesType {
const now = Date.now();
return {
conversationId: uuid(),
id: uuid(),
received_at: now,
sent_at: now,
timestamp: now,
type: 'incoming',
};
}
it("does nothing if the reaction isn't in the list", () => {
const result = markOutgoingReactionSent(
reactions,
rxn('🥀', { isPending: true }),
[uuid()],
getMessage()
);
assert.deepStrictEqual(result, reactions);
});
it('updates reactions to be partially sent', () => {
[star, none].forEach(reaction => {
const result = markOutgoingReactionSent(
reactions,
reaction,
[uuid1, uuid2],
getMessage()
);
assert.deepStrictEqual(
result.find(re => re.emoji === reaction.emoji)
?.isSentByConversationId,
{
[uuid1]: true,
[uuid2]: true,
[uuid3]: false,
}
);
});
});
it('removes sent state if a reaction with emoji is fully sent', () => {
const result = markOutgoingReactionSent(
reactions,
star,
[uuid1, uuid2, uuid3],
getMessage()
);
const newReaction = result.find(re => re.emoji === '⭐️');
assert.isDefined(newReaction);
assert.isUndefined(newReaction?.isSentByConversationId);
});
it('removes a fully-sent reaction removal', () => {
const result = markOutgoingReactionSent(
reactions,
none,
[uuid1, uuid2, uuid3],
getMessage()
);
assert(
result.every(({ emoji }) => typeof emoji === 'string'),
'Expected the emoji removal to be gone'
);
});
it('removes older reactions of mine', () => {
const result = markOutgoingReactionSent(
reactions,
star,
[uuid1, uuid2, uuid3],
getMessage()
);
assert.isUndefined(result.find(re => re.emoji === '🔕'));
});
it('does not remove my older reactions if they are on a story', () => {
const result = markOutgoingReactionSent(
reactions,
star,
[uuid1, uuid2, uuid3],
{ ...getMessage(), type: 'story' }
);
assert.isDefined(result.find(re => re.emoji === '🔕'));
});
});
});