You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
135 lines
4.9 KiB
TypeScript
135 lines
4.9 KiB
TypeScript
/* eslint-disable no-await-in-loop */
|
|
/* eslint-disable no-unused-expressions */
|
|
import chai, { expect } from 'chai';
|
|
import Sinon, { useFakeTimers } from 'sinon';
|
|
import { noop } from 'lodash';
|
|
import chaiAsPromised from 'chai-as-promised';
|
|
|
|
import { Reactions } from '../../../../util/reactions';
|
|
import { Data } from '../../../../data/data';
|
|
import * as Storage from '../../../../util/storage';
|
|
import { generateFakeIncomingPrivateMessage, stubWindowLog } from '../../../test-utils/utils';
|
|
import { DEFAULT_RECENT_REACTS } from '../../../../session/constants';
|
|
|
|
import { UserUtils } from '../../../../session/utils';
|
|
import { SignalService } from '../../../../protobuf';
|
|
import { MessageCollection } from '../../../../models/message';
|
|
|
|
chai.use(chaiAsPromised as any);
|
|
|
|
describe('ReactionMessage', () => {
|
|
stubWindowLog();
|
|
|
|
let clock: Sinon.SinonFakeTimers;
|
|
const ourNumber = '0123456789abcdef';
|
|
const originalMessage = generateFakeIncomingPrivateMessage();
|
|
originalMessage.set('sent_at', Date.now());
|
|
|
|
beforeEach(() => {
|
|
Sinon.stub(originalMessage, 'getConversation').returns({
|
|
hasReactions: () => true,
|
|
sendReaction: noop,
|
|
} as any);
|
|
|
|
// sendMessageReaction stubs
|
|
Sinon.stub(Data, 'getMessageById').resolves(originalMessage);
|
|
Sinon.stub(Storage, 'getRecentReactions').returns(DEFAULT_RECENT_REACTS);
|
|
Sinon.stub(Storage, 'saveRecentReations').resolves();
|
|
Sinon.stub(UserUtils, 'getOurPubKeyStrFromCache').returns(ourNumber);
|
|
|
|
// handleMessageReaction stubs
|
|
Sinon.stub(Data, 'getMessagesBySentAt').resolves(new MessageCollection([originalMessage]));
|
|
Sinon.stub(originalMessage, 'commit').resolves();
|
|
});
|
|
|
|
it('can react to a message', async () => {
|
|
// Send reaction
|
|
const reaction = await Reactions.sendMessageReaction(originalMessage.get('id'), '😄');
|
|
|
|
expect(reaction?.id, 'id should match the original message timestamp').to.be.equal(
|
|
Number(originalMessage.get('sent_at'))
|
|
);
|
|
expect(reaction?.author, 'author should match the original message author').to.be.equal(
|
|
originalMessage.get('source')
|
|
);
|
|
expect(reaction?.emoji, 'emoji should be 😄').to.be.equal('😄');
|
|
expect(reaction?.action, 'action should be 0').to.be.equal(0);
|
|
|
|
// Handling reaction
|
|
const updatedMessage = await Reactions.handleMessageReaction({
|
|
reaction: reaction as SignalService.DataMessage.IReaction,
|
|
sender: ourNumber,
|
|
you: true,
|
|
});
|
|
|
|
expect(updatedMessage?.get('reacts'), 'original message should have reacts').to.not.be
|
|
.undefined;
|
|
|
|
expect(updatedMessage?.get('reacts')!['😄'], 'reacts should have 😄 key').to.not.be.undefined;
|
|
|
|
expect(
|
|
updatedMessage!.get('reacts')!['😄'].senders[0],
|
|
'sender pubkey should match'
|
|
).to.be.equal(ourNumber);
|
|
expect(updatedMessage!.get('reacts')!['😄'].count, 'count should be 1').to.be.equal(1);
|
|
});
|
|
|
|
it('can remove a reaction from a message', async () => {
|
|
// Send reaction
|
|
const reaction = await Reactions.sendMessageReaction(originalMessage.get('id'), '😄');
|
|
|
|
expect(reaction?.id, 'id should match the original message timestamp').to.be.equal(
|
|
Number(originalMessage.get('sent_at'))
|
|
);
|
|
expect(reaction?.author, 'author should match the original message author').to.be.equal(
|
|
originalMessage.get('source')
|
|
);
|
|
expect(reaction?.emoji, 'emoji should be 😄').to.be.equal('😄');
|
|
expect(reaction?.action, 'action should be 1').to.be.equal(1);
|
|
|
|
// Handling reaction
|
|
const updatedMessage = await Reactions.handleMessageReaction({
|
|
reaction: reaction as SignalService.DataMessage.IReaction,
|
|
sender: ourNumber,
|
|
you: true,
|
|
});
|
|
|
|
expect(updatedMessage?.get('reacts'), 'original message reacts should be undefined').to.be
|
|
.undefined;
|
|
});
|
|
|
|
it('reactions are rate limited to 20 reactions per minute', async () => {
|
|
// we have already sent 2 messages when this test runs
|
|
for (let i = 0; i < 18; i++) {
|
|
// Send reaction
|
|
await Reactions.sendMessageReaction(originalMessage.get('id'), '👍');
|
|
}
|
|
|
|
let reaction = await Reactions.sendMessageReaction(originalMessage.get('id'), '👎');
|
|
|
|
expect(reaction, 'no reaction should be returned since we are over the rate limit').to.be
|
|
.undefined;
|
|
|
|
clock = useFakeTimers({ now: Date.now(), shouldAdvanceTime: true });
|
|
|
|
// Wait a miniute for the rate limit to clear
|
|
clock.tick(1 * 60 * 1000);
|
|
|
|
reaction = await Reactions.sendMessageReaction(originalMessage.get('id'), '👋');
|
|
|
|
expect(reaction?.id, 'id should match the original message timestamp').to.be.equal(
|
|
Number(originalMessage.get('sent_at'))
|
|
);
|
|
expect(reaction?.author, 'author should match the original message author').to.be.equal(
|
|
originalMessage.get('source')
|
|
);
|
|
expect(reaction?.emoji, 'emoji should be 👋').to.be.equal('👋');
|
|
expect(reaction?.action, 'action should be 0').to.be.equal(0);
|
|
clock.restore();
|
|
});
|
|
|
|
afterEach(() => {
|
|
Sinon.restore();
|
|
});
|
|
});
|