123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601 |
- <?php
- /**
- * SPDX-FileCopyrightText: 2016-2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-FileCopyrightText: 2016 ownCloud, Inc.
- * SPDX-License-Identifier: AGPL-3.0-only
- */
- namespace Test\Comments;
- use OC\Comments\Comment;
- use OC\Comments\Manager;
- use OC\EmojiHelper;
- use OCP\AppFramework\Utility\ITimeFactory;
- use OCP\Comments\IComment;
- use OCP\Comments\ICommentsEventHandler;
- use OCP\Comments\ICommentsManager;
- use OCP\Comments\NotFoundException;
- use OCP\EventDispatcher\IEventDispatcher;
- use OCP\Files\Folder;
- use OCP\Files\IRootFolder;
- use OCP\IConfig;
- use OCP\IDBConnection;
- use OCP\IInitialStateService;
- use OCP\IUser;
- use OCP\Server;
- use Psr\Log\LoggerInterface;
- use Test\TestCase;
- /**
- * Class ManagerTest
- *
- * @group DB
- */
- class ManagerTest extends TestCase {
- /** @var IDBConnection */
- private $connection;
- /** @var \PHPUnit\Framework\MockObject\MockObject|IRootFolder */
- private $rootFolder;
- protected function setUp(): void {
- parent::setUp();
- $this->connection = \OC::$server->getDatabaseConnection();
- $this->rootFolder = $this->createMock(IRootFolder::class);
- $sql = $this->connection->getDatabasePlatform()->getTruncateTableSQL('`*PREFIX*comments`');
- $this->connection->prepare($sql)->execute();
- $sql = $this->connection->getDatabasePlatform()->getTruncateTableSQL('`*PREFIX*reactions`');
- $this->connection->prepare($sql)->execute();
- }
- protected function addDatabaseEntry($parentId, $topmostParentId, $creationDT = null, $latestChildDT = null, $objectId = null, $expireDate = null) {
- if (is_null($creationDT)) {
- $creationDT = new \DateTime();
- }
- if (is_null($latestChildDT)) {
- $latestChildDT = new \DateTime('yesterday');
- }
- if (is_null($objectId)) {
- $objectId = 'file64';
- }
- $qb = $this->connection->getQueryBuilder();
- $qb
- ->insert('comments')
- ->values([
- 'parent_id' => $qb->createNamedParameter($parentId),
- 'topmost_parent_id' => $qb->createNamedParameter($topmostParentId),
- 'children_count' => $qb->createNamedParameter(2),
- 'actor_type' => $qb->createNamedParameter('users'),
- 'actor_id' => $qb->createNamedParameter('alice'),
- 'message' => $qb->createNamedParameter('nice one'),
- 'verb' => $qb->createNamedParameter('comment'),
- 'creation_timestamp' => $qb->createNamedParameter($creationDT, 'datetime'),
- 'latest_child_timestamp' => $qb->createNamedParameter($latestChildDT, 'datetime'),
- 'object_type' => $qb->createNamedParameter('files'),
- 'object_id' => $qb->createNamedParameter($objectId),
- 'expire_date' => $qb->createNamedParameter($expireDate, 'datetime'),
- 'reference_id' => $qb->createNamedParameter('referenceId'),
- 'meta_data' => $qb->createNamedParameter(json_encode(['last_edit_actor_id' => 'admin'])),
- ])
- ->execute();
- return $qb->getLastInsertId();
- }
- protected function getManager() {
- return new Manager(
- $this->connection,
- $this->createMock(LoggerInterface::class),
- $this->createMock(IConfig::class),
- $this->createMock(ITimeFactory::class),
- new EmojiHelper($this->connection),
- $this->createMock(IInitialStateService::class),
- $this->rootFolder,
- $this->createMock(IEventDispatcher::class),
- );
- }
- public function testGetCommentNotFound(): void {
- $this->expectException(\OCP\Comments\NotFoundException::class);
- $manager = $this->getManager();
- $manager->get('22');
- }
- public function testGetCommentNotFoundInvalidInput(): void {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->get('unexisting22');
- }
- public function testGetComment(): void {
- $manager = $this->getManager();
- $creationDT = new \DateTime();
- $latestChildDT = new \DateTime('yesterday');
- $qb = \OC::$server->getDatabaseConnection()->getQueryBuilder();
- $qb
- ->insert('comments')
- ->values([
- 'parent_id' => $qb->createNamedParameter('2'),
- 'topmost_parent_id' => $qb->createNamedParameter('1'),
- 'children_count' => $qb->createNamedParameter(2),
- 'actor_type' => $qb->createNamedParameter('users'),
- 'actor_id' => $qb->createNamedParameter('alice'),
- 'message' => $qb->createNamedParameter('nice one'),
- 'verb' => $qb->createNamedParameter('comment'),
- 'creation_timestamp' => $qb->createNamedParameter($creationDT, 'datetime'),
- 'latest_child_timestamp' => $qb->createNamedParameter($latestChildDT, 'datetime'),
- 'object_type' => $qb->createNamedParameter('files'),
- 'object_id' => $qb->createNamedParameter('file64'),
- 'reference_id' => $qb->createNamedParameter('referenceId'),
- 'meta_data' => $qb->createNamedParameter(json_encode(['last_edit_actor_id' => 'admin'])),
- ])
- ->execute();
- $id = strval($qb->getLastInsertId());
- $comment = $manager->get($id);
- $this->assertTrue($comment instanceof IComment);
- $this->assertSame($comment->getId(), $id);
- $this->assertSame($comment->getParentId(), '2');
- $this->assertSame($comment->getTopmostParentId(), '1');
- $this->assertSame($comment->getChildrenCount(), 2);
- $this->assertSame($comment->getActorType(), 'users');
- $this->assertSame($comment->getActorId(), 'alice');
- $this->assertSame($comment->getMessage(), 'nice one');
- $this->assertSame($comment->getVerb(), 'comment');
- $this->assertSame($comment->getObjectType(), 'files');
- $this->assertSame($comment->getObjectId(), 'file64');
- $this->assertEquals($comment->getCreationDateTime()->getTimestamp(), $creationDT->getTimestamp());
- $this->assertEquals($comment->getLatestChildDateTime(), $latestChildDT);
- $this->assertEquals($comment->getReferenceId(), 'referenceId');
- $this->assertEquals($comment->getMetaData(), ['last_edit_actor_id' => 'admin']);
- }
- public function testGetTreeNotFound(): void {
- $this->expectException(\OCP\Comments\NotFoundException::class);
- $manager = $this->getManager();
- $manager->getTree('22');
- }
- public function testGetTreeNotFoundInvalidIpnut(): void {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->getTree('unexisting22');
- }
- public function testGetTree(): void {
- $headId = $this->addDatabaseEntry(0, 0);
- $this->addDatabaseEntry($headId, $headId, new \DateTime('-3 hours'));
- $this->addDatabaseEntry($headId, $headId, new \DateTime('-2 hours'));
- $id = $this->addDatabaseEntry($headId, $headId, new \DateTime('-1 hour'));
- $manager = $this->getManager();
- $tree = $manager->getTree($headId);
- // Verifying the root comment
- $this->assertTrue(isset($tree['comment']));
- $this->assertTrue($tree['comment'] instanceof IComment);
- $this->assertSame($tree['comment']->getId(), strval($headId));
- $this->assertTrue(isset($tree['replies']));
- $this->assertSame(count($tree['replies']), 3);
- // one level deep
- foreach ($tree['replies'] as $reply) {
- $this->assertTrue($reply['comment'] instanceof IComment);
- $this->assertSame($reply['comment']->getId(), strval($id));
- $this->assertSame(count($reply['replies']), 0);
- $id--;
- }
- }
- public function testGetTreeNoReplies(): void {
- $id = $this->addDatabaseEntry(0, 0);
- $manager = $this->getManager();
- $tree = $manager->getTree($id);
- // Verifying the root comment
- $this->assertTrue(isset($tree['comment']));
- $this->assertTrue($tree['comment'] instanceof IComment);
- $this->assertSame($tree['comment']->getId(), strval($id));
- $this->assertTrue(isset($tree['replies']));
- $this->assertSame(count($tree['replies']), 0);
- // one level deep
- foreach ($tree['replies'] as $reply) {
- throw new \Exception('This ain`t happen');
- }
- }
- public function testGetTreeWithLimitAndOffset(): void {
- $headId = $this->addDatabaseEntry(0, 0);
- $this->addDatabaseEntry($headId, $headId, new \DateTime('-3 hours'));
- $this->addDatabaseEntry($headId, $headId, new \DateTime('-2 hours'));
- $this->addDatabaseEntry($headId, $headId, new \DateTime('-1 hour'));
- $idToVerify = $this->addDatabaseEntry($headId, $headId, new \DateTime());
- $manager = $this->getManager();
- for ($offset = 0; $offset < 3; $offset += 2) {
- $tree = $manager->getTree(strval($headId), 2, $offset);
- // Verifying the root comment
- $this->assertTrue(isset($tree['comment']));
- $this->assertTrue($tree['comment'] instanceof IComment);
- $this->assertSame($tree['comment']->getId(), strval($headId));
- $this->assertTrue(isset($tree['replies']));
- $this->assertSame(count($tree['replies']), 2);
- // one level deep
- foreach ($tree['replies'] as $reply) {
- $this->assertTrue($reply['comment'] instanceof IComment);
- $this->assertSame($reply['comment']->getId(), strval($idToVerify));
- $this->assertSame(count($reply['replies']), 0);
- $idToVerify--;
- }
- }
- }
- public function testGetForObject(): void {
- $this->addDatabaseEntry(0, 0);
- $manager = $this->getManager();
- $comments = $manager->getForObject('files', 'file64');
- $this->assertTrue(is_array($comments));
- $this->assertSame(count($comments), 1);
- $this->assertTrue($comments[0] instanceof IComment);
- $this->assertSame($comments[0]->getMessage(), 'nice one');
- }
- public function testGetForObjectWithLimitAndOffset(): void {
- $this->addDatabaseEntry(0, 0, new \DateTime('-6 hours'));
- $this->addDatabaseEntry(0, 0, new \DateTime('-5 hours'));
- $this->addDatabaseEntry(1, 1, new \DateTime('-4 hours'));
- $this->addDatabaseEntry(0, 0, new \DateTime('-3 hours'));
- $this->addDatabaseEntry(2, 2, new \DateTime('-2 hours'));
- $this->addDatabaseEntry(2, 2, new \DateTime('-1 hours'));
- $idToVerify = $this->addDatabaseEntry(3, 1, new \DateTime());
- $manager = $this->getManager();
- $offset = 0;
- do {
- $comments = $manager->getForObject('files', 'file64', 3, $offset);
- $this->assertTrue(is_array($comments));
- foreach ($comments as $comment) {
- $this->assertTrue($comment instanceof IComment);
- $this->assertSame($comment->getMessage(), 'nice one');
- $this->assertSame($comment->getId(), strval($idToVerify));
- $idToVerify--;
- }
- $offset += 3;
- } while (count($comments) > 0);
- }
- public function testGetForObjectWithDateTimeConstraint(): void {
- $this->addDatabaseEntry(0, 0, new \DateTime('-6 hours'));
- $this->addDatabaseEntry(0, 0, new \DateTime('-5 hours'));
- $id1 = $this->addDatabaseEntry(0, 0, new \DateTime('-3 hours'));
- $id2 = $this->addDatabaseEntry(2, 2, new \DateTime('-2 hours'));
- $manager = $this->getManager();
- $comments = $manager->getForObject('files', 'file64', 0, 0, new \DateTime('-4 hours'));
- $this->assertSame(count($comments), 2);
- $this->assertSame($comments[0]->getId(), strval($id2));
- $this->assertSame($comments[1]->getId(), strval($id1));
- }
- public function testGetForObjectWithLimitAndOffsetAndDateTimeConstraint(): void {
- $this->addDatabaseEntry(0, 0, new \DateTime('-7 hours'));
- $this->addDatabaseEntry(0, 0, new \DateTime('-6 hours'));
- $this->addDatabaseEntry(1, 1, new \DateTime('-5 hours'));
- $this->addDatabaseEntry(0, 0, new \DateTime('-3 hours'));
- $this->addDatabaseEntry(2, 2, new \DateTime('-2 hours'));
- $this->addDatabaseEntry(2, 2, new \DateTime('-1 hours'));
- $idToVerify = $this->addDatabaseEntry(3, 1, new \DateTime());
- $manager = $this->getManager();
- $offset = 0;
- do {
- $comments = $manager->getForObject('files', 'file64', 3, $offset, new \DateTime('-4 hours'));
- $this->assertTrue(is_array($comments));
- foreach ($comments as $comment) {
- $this->assertTrue($comment instanceof IComment);
- $this->assertSame($comment->getMessage(), 'nice one');
- $this->assertSame($comment->getId(), strval($idToVerify));
- $this->assertTrue(intval($comment->getId()) >= 4);
- $idToVerify--;
- }
- $offset += 3;
- } while (count($comments) > 0);
- }
- public function testGetNumberOfCommentsForObject(): void {
- for ($i = 1; $i < 5; $i++) {
- $this->addDatabaseEntry(0, 0);
- }
- $manager = $this->getManager();
- $amount = $manager->getNumberOfCommentsForObject('untype', '00');
- $this->assertSame($amount, 0);
- $amount = $manager->getNumberOfCommentsForObject('files', 'file64');
- $this->assertSame($amount, 4);
- }
- public function testGetNumberOfUnreadCommentsForFolder(): void {
- $folder = $this->createMock(Folder::class);
- $fileIds = range(1111, 1114);
- $children = array_map(function (int $id) {
- $file = $this->createMock(Folder::class);
- $file->method('getId')
- ->willReturn($id);
- return $file;
- }, $fileIds);
- $folder->method('getId')->willReturn(1000);
- $folder->method('getDirectoryListing')->willReturn($children);
- $this->rootFolder->method('getFirstNodeById')
- ->with($folder->getId())
- ->willReturn($folder);
- // 2 comment for 1111 with 1 before read marker
- // 2 comments for 1112 with no read marker
- // 1 comment for 1113 before read marker
- // 1 comment for 1114 with no read marker
- $this->addDatabaseEntry(0, 0, null, null, $fileIds[1]);
- for ($i = 0; $i < 4; $i++) {
- $this->addDatabaseEntry(0, 0, null, null, $fileIds[$i]);
- }
- $this->addDatabaseEntry(0, 0, (new \DateTime())->modify('-2 days'), null, $fileIds[0]);
- /** @var IUser|\PHPUnit\Framework\MockObject\MockObject $user */
- $user = $this->createMock(IUser::class);
- $user->expects($this->any())
- ->method('getUID')
- ->willReturn('comment_test');
- $manager = $this->getManager();
- $manager->setReadMark('files', (string)$fileIds[0], (new \DateTime())->modify('-1 days'), $user);
- $manager->setReadMark('files', (string)$fileIds[2], (new \DateTime()), $user);
- $amount = $manager->getNumberOfUnreadCommentsForFolder($folder->getId(), $user);
- $this->assertEquals([
- $fileIds[0] => 1,
- $fileIds[1] => 2,
- $fileIds[3] => 1,
- ], $amount);
- }
- /**
- * @dataProvider dataGetForObjectSince
- * @param $lastKnown
- * @param $order
- * @param $limit
- * @param $resultFrom
- * @param $resultTo
- */
- public function testGetForObjectSince($lastKnown, $order, $limit, $resultFrom, $resultTo): void {
- $ids = [];
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $manager = $this->getManager();
- $comments = $manager->getForObjectSince('files', 'file64', ($lastKnown === null ? 0 : $ids[$lastKnown]), $order, $limit);
- $expected = array_slice($ids, $resultFrom, $resultTo - $resultFrom + 1);
- if ($order === 'desc') {
- $expected = array_reverse($expected);
- }
- $this->assertSame($expected, array_map(function (IComment $c) {
- return (int)$c->getId();
- }, $comments));
- }
- public function dataGetForObjectSince() {
- return [
- [null, 'asc', 20, 0, 4],
- [null, 'asc', 2, 0, 1],
- [null, 'desc', 20, 0, 4],
- [null, 'desc', 2, 3, 4],
- [1, 'asc', 20, 2, 4],
- [1, 'asc', 2, 2, 3],
- [3, 'desc', 20, 0, 2],
- [3, 'desc', 2, 1, 2],
- ];
- }
- public function invalidCreateArgsProvider() {
- return [
- ['', 'aId-1', 'oType-1', 'oId-1'],
- ['aType-1', '', 'oType-1', 'oId-1'],
- ['aType-1', 'aId-1', '', 'oId-1'],
- ['aType-1', 'aId-1', 'oType-1', ''],
- [1, 'aId-1', 'oType-1', 'oId-1'],
- ['aType-1', 1, 'oType-1', 'oId-1'],
- ['aType-1', 'aId-1', 1, 'oId-1'],
- ['aType-1', 'aId-1', 'oType-1', 1],
- ];
- }
- /**
- * @dataProvider invalidCreateArgsProvider
- * @param string $aType
- * @param string $aId
- * @param string $oType
- * @param string $oId
- */
- public function testCreateCommentInvalidArguments($aType, $aId, $oType, $oId): void {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->create($aType, $aId, $oType, $oId);
- }
- public function testCreateComment(): void {
- $actorType = 'bot';
- $actorId = 'bob';
- $objectType = 'weather';
- $objectId = 'bielefeld';
- $comment = $this->getManager()->create($actorType, $actorId, $objectType, $objectId);
- $this->assertTrue($comment instanceof IComment);
- $this->assertSame($comment->getActorType(), $actorType);
- $this->assertSame($comment->getActorId(), $actorId);
- $this->assertSame($comment->getObjectType(), $objectType);
- $this->assertSame($comment->getObjectId(), $objectId);
- }
- public function testDelete(): void {
- $this->expectException(\OCP\Comments\NotFoundException::class);
- $manager = $this->getManager();
- $done = $manager->delete('404');
- $this->assertFalse($done);
- $done = $manager->delete('%');
- $this->assertFalse($done);
- $done = $manager->delete('');
- $this->assertFalse($done);
- $id = strval($this->addDatabaseEntry(0, 0));
- $comment = $manager->get($id);
- $this->assertTrue($comment instanceof IComment);
- $done = $manager->delete($id);
- $this->assertTrue($done);
- $manager->get($id);
- }
- /**
- * @dataProvider providerTestSave
- */
- public function testSave(string $message, string $actorId, string $verb, ?string $parentId, ?string $id = ''): IComment {
- $manager = $this->getManager();
- $comment = new Comment();
- $comment
- ->setId($id)
- ->setActor('users', $actorId)
- ->setObject('files', 'file64')
- ->setMessage($message)
- ->setVerb($verb);
- if ($parentId) {
- $comment->setParentId($parentId);
- }
- $saveSuccessful = $manager->save($comment);
- $this->assertTrue($saveSuccessful);
- $this->assertTrue($comment->getId() !== '');
- $this->assertTrue($comment->getId() !== '0');
- $this->assertTrue(!is_null($comment->getCreationDateTime()));
- $loadedComment = $manager->get($comment->getId());
- $this->assertSame($comment->getMessage(), $loadedComment->getMessage());
- $this->assertEquals($comment->getCreationDateTime()->getTimestamp(), $loadedComment->getCreationDateTime()->getTimestamp());
- return $comment;
- }
- public function providerTestSave(): array {
- return [
- ['very beautiful, I am impressed!', 'alice', 'comment', null]
- ];
- }
- public function testSaveUpdate(): void {
- $manager = $this->getManager();
- $comment = new Comment();
- $comment
- ->setActor('users', 'alice')
- ->setObject('files', 'file64')
- ->setMessage('very beautiful, I am impressed!')
- ->setVerb('comment')
- ->setExpireDate(new \DateTime('+2 hours'));
- $manager->save($comment);
- $loadedComment = $manager->get($comment->getId());
- // Compare current object with database values
- $this->assertSame($comment->getMessage(), $loadedComment->getMessage());
- $this->assertSame(
- $comment->getExpireDate()->format('Y-m-d H:i:s'),
- $loadedComment->getExpireDate()->format('Y-m-d H:i:s')
- );
- // Preserve the original comment to compare after update
- $original = clone $comment;
- // Update values
- $comment->setMessage('very beautiful, I am really so much impressed!')
- ->setExpireDate(new \DateTime('+1 hours'));
- $manager->save($comment);
- $loadedComment = $manager->get($comment->getId());
- // Compare current object with database values
- $this->assertSame($comment->getMessage(), $loadedComment->getMessage());
- $this->assertSame(
- $comment->getExpireDate()->format('Y-m-d H:i:s'),
- $loadedComment->getExpireDate()->format('Y-m-d H:i:s')
- );
- // Compare original object with database values
- $this->assertNotSame($original->getMessage(), $loadedComment->getMessage());
- $this->assertNotSame(
- $original->getExpireDate()->format('Y-m-d H:i:s'),
- $loadedComment->getExpireDate()->format('Y-m-d H:i:s')
- );
- }
- public function testSaveUpdateException(): void {
- $this->expectException(\OCP\Comments\NotFoundException::class);
- $manager = $this->getManager();
- $comment = new Comment();
- $comment
- ->setActor('users', 'alice')
- ->setObject('files', 'file64')
- ->setMessage('very beautiful, I am impressed!')
- ->setVerb('comment');
- $manager->save($comment);
- $manager->delete($comment->getId());
- $comment->setMessage('very beautiful, I am really so much impressed!');
- $manager->save($comment);
- }
- public function testSaveIncomplete(): void {
- $this->expectException(\UnexpectedValueException::class);
- $manager = $this->getManager();
- $comment = new Comment();
- $comment->setMessage('from no one to nothing');
- $manager->save($comment);
- }
- public function testSaveAsChild(): void {
- $id = $this->addDatabaseEntry(0, 0);
- $manager = $this->getManager();
- for ($i = 0; $i < 3; $i++) {
- $comment = new Comment();
- $comment
- ->setActor('users', 'alice')
- ->setObject('files', 'file64')
- ->setParentId(strval($id))
- ->setMessage('full ack')
- ->setVerb('comment')
- // setting the creation time avoids using sleep() while making sure to test with different timestamps
- ->setCreationDateTime(new \DateTime('+' . $i . ' minutes'));
- $manager->save($comment);
- $this->assertSame($comment->getTopmostParentId(), strval($id));
- $parentComment = $manager->get(strval($id));
- $this->assertSame($parentComment->getChildrenCount(), $i + 1);
- $this->assertEquals($parentComment->getLatestChildDateTime()->getTimestamp(), $comment->getCreationDateTime()->getTimestamp());
- }
- }
- public function invalidActorArgsProvider() {
- return
- [
- ['', ''],
- [1, 'alice'],
- ['users', 1],
- ];
- }
- /**
- * @dataProvider invalidActorArgsProvider
- * @param string $type
- * @param string $id
- */
- public function testDeleteReferencesOfActorInvalidInput($type, $id): void {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->deleteReferencesOfActor($type, $id);
- }
- public function testDeleteReferencesOfActor(): void {
- $ids = [];
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $manager = $this->getManager();
- // just to make sure they are really set, with correct actor data
- $comment = $manager->get(strval($ids[1]));
- $this->assertSame($comment->getActorType(), 'users');
- $this->assertSame($comment->getActorId(), 'alice');
- $wasSuccessful = $manager->deleteReferencesOfActor('users', 'alice');
- $this->assertTrue($wasSuccessful);
- foreach ($ids as $id) {
- $comment = $manager->get(strval($id));
- $this->assertSame($comment->getActorType(), ICommentsManager::DELETED_USER);
- $this->assertSame($comment->getActorId(), ICommentsManager::DELETED_USER);
- }
- // actor info is gone from DB, but when database interaction is alright,
- // we still expect to get true back
- $wasSuccessful = $manager->deleteReferencesOfActor('users', 'alice');
- $this->assertTrue($wasSuccessful);
- }
- public function testDeleteReferencesOfActorWithUserManagement(): void {
- $user = \OC::$server->getUserManager()->createUser('xenia', 'NotAnEasyPassword123456+');
- $this->assertTrue($user instanceof IUser);
- $manager = \OC::$server->get(ICommentsManager::class);
- $comment = $manager->create('users', $user->getUID(), 'files', 'file64');
- $comment
- ->setMessage('Most important comment I ever left on the Internet.')
- ->setVerb('comment');
- $status = $manager->save($comment);
- $this->assertTrue($status);
- $commentID = $comment->getId();
- $user->delete();
- $comment = $manager->get($commentID);
- $this->assertSame($comment->getActorType(), ICommentsManager::DELETED_USER);
- $this->assertSame($comment->getActorId(), ICommentsManager::DELETED_USER);
- }
- public function invalidObjectArgsProvider() {
- return
- [
- ['', ''],
- [1, 'file64'],
- ['files', 1],
- ];
- }
- /**
- * @dataProvider invalidObjectArgsProvider
- * @param string $type
- * @param string $id
- */
- public function testDeleteCommentsAtObjectInvalidInput($type, $id): void {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->deleteCommentsAtObject($type, $id);
- }
- public function testDeleteCommentsAtObject(): void {
- $ids = [];
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $manager = $this->getManager();
- // just to make sure they are really set, with correct actor data
- $comment = $manager->get(strval($ids[1]));
- $this->assertSame($comment->getObjectType(), 'files');
- $this->assertSame($comment->getObjectId(), 'file64');
- $wasSuccessful = $manager->deleteCommentsAtObject('files', 'file64');
- $this->assertTrue($wasSuccessful);
- $verified = 0;
- foreach ($ids as $id) {
- try {
- $manager->get(strval($id));
- } catch (NotFoundException $e) {
- $verified++;
- }
- }
- $this->assertSame($verified, 3);
- // actor info is gone from DB, but when database interaction is alright,
- // we still expect to get true back
- $wasSuccessful = $manager->deleteCommentsAtObject('files', 'file64');
- $this->assertTrue($wasSuccessful);
- }
- public function testDeleteCommentsExpiredAtObjectTypeAndId(): void {
- $ids = [];
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, null, new \DateTime('+2 hours'));
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, null, new \DateTime('+2 hours'));
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, null, new \DateTime('+2 hours'));
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, null, new \DateTime('-2 hours'));
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, null, new \DateTime('-2 hours'));
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, null, new \DateTime('-2 hours'));
- $manager = new Manager(
- $this->connection,
- $this->createMock(LoggerInterface::class),
- $this->createMock(IConfig::class),
- Server::get(ITimeFactory::class),
- new EmojiHelper($this->connection),
- $this->createMock(IInitialStateService::class),
- $this->rootFolder,
- $this->createMock(IEventDispatcher::class)
- );
- // just to make sure they are really set, with correct actor data
- $comment = $manager->get((string)$ids[1]);
- $this->assertSame($comment->getObjectType(), 'files');
- $this->assertSame($comment->getObjectId(), 'file64');
- $deleted = $manager->deleteCommentsExpiredAtObject('files', 'file64');
- $this->assertTrue($deleted);
- $deleted = 0;
- $exists = 0;
- foreach ($ids as $id) {
- try {
- $manager->get((string)$id);
- $exists++;
- } catch (NotFoundException $e) {
- $deleted++;
- }
- }
- $this->assertSame($exists, 3);
- $this->assertSame($deleted, 3);
- // actor info is gone from DB, but when database interaction is alright,
- // we still expect to get true back
- $deleted = $manager->deleteCommentsExpiredAtObject('files', 'file64');
- $this->assertFalse($deleted);
- }
- public function testDeleteCommentsExpiredAtObjectType(): void {
- $ids = [];
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, 'file1', new \DateTime('-2 hours'));
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, 'file2', new \DateTime('-2 hours'));
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, 'file3', new \DateTime('-2 hours'));
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, 'file3', new \DateTime());
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, 'file3', new \DateTime());
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, 'file3', new \DateTime());
- $manager = new Manager(
- $this->connection,
- $this->createMock(LoggerInterface::class),
- $this->createMock(IConfig::class),
- Server::get(ITimeFactory::class),
- new EmojiHelper($this->connection),
- $this->createMock(IInitialStateService::class),
- $this->rootFolder,
- $this->createMock(IEventDispatcher::class)
- );
- $deleted = $manager->deleteCommentsExpiredAtObject('files');
- $this->assertTrue($deleted);
- $deleted = 0;
- $exists = 0;
- foreach ($ids as $id) {
- try {
- $manager->get((string)$id);
- $exists++;
- } catch (NotFoundException $e) {
- $deleted++;
- }
- }
- $this->assertSame($exists, 0);
- $this->assertSame($deleted, 6);
- // actor info is gone from DB, but when database interaction is alright,
- // we still expect to get true back
- $deleted = $manager->deleteCommentsExpiredAtObject('files');
- $this->assertFalse($deleted);
- }
- public function testSetMarkRead(): void {
- /** @var IUser|\PHPUnit\Framework\MockObject\MockObject $user */
- $user = $this->createMock(IUser::class);
- $user->expects($this->any())
- ->method('getUID')
- ->willReturn('alice');
- $dateTimeSet = new \DateTime();
- $manager = $this->getManager();
- $manager->setReadMark('robot', '36', $dateTimeSet, $user);
- $dateTimeGet = $manager->getReadMark('robot', '36', $user);
- $this->assertEquals($dateTimeGet->getTimestamp(), $dateTimeSet->getTimestamp());
- }
- public function testSetMarkReadUpdate(): void {
- /** @var IUser|\PHPUnit\Framework\MockObject\MockObject $user */
- $user = $this->createMock(IUser::class);
- $user->expects($this->any())
- ->method('getUID')
- ->willReturn('alice');
- $dateTimeSet = new \DateTime('yesterday');
- $manager = $this->getManager();
- $manager->setReadMark('robot', '36', $dateTimeSet, $user);
- $dateTimeSet = new \DateTime('today');
- $manager->setReadMark('robot', '36', $dateTimeSet, $user);
- $dateTimeGet = $manager->getReadMark('robot', '36', $user);
- $this->assertEquals($dateTimeGet, $dateTimeSet);
- }
- public function testReadMarkDeleteUser(): void {
- /** @var IUser|\PHPUnit\Framework\MockObject\MockObject $user */
- $user = $this->createMock(IUser::class);
- $user->expects($this->any())
- ->method('getUID')
- ->willReturn('alice');
- $dateTimeSet = new \DateTime();
- $manager = $this->getManager();
- $manager->setReadMark('robot', '36', $dateTimeSet, $user);
- $manager->deleteReadMarksFromUser($user);
- $dateTimeGet = $manager->getReadMark('robot', '36', $user);
- $this->assertNull($dateTimeGet);
- }
- public function testReadMarkDeleteObject(): void {
- /** @var IUser|\PHPUnit\Framework\MockObject\MockObject $user */
- $user = $this->createMock(IUser::class);
- $user->expects($this->any())
- ->method('getUID')
- ->willReturn('alice');
- $dateTimeSet = new \DateTime();
- $manager = $this->getManager();
- $manager->setReadMark('robot', '36', $dateTimeSet, $user);
- $manager->deleteReadMarksOnObject('robot', '36');
- $dateTimeGet = $manager->getReadMark('robot', '36', $user);
- $this->assertNull($dateTimeGet);
- }
- public function testSendEvent(): void {
- $handler1 = $this->getMockBuilder(ICommentsEventHandler::class)->getMock();
- $handler1->expects($this->exactly(4))
- ->method('handle');
- $handler2 = $this->getMockBuilder(ICommentsEventHandler::class)->getMock();
- $handler1->expects($this->exactly(4))
- ->method('handle');
- $manager = $this->getManager();
- $manager->registerEventHandler(function () use ($handler1) {
- return $handler1;
- });
- $manager->registerEventHandler(function () use ($handler2) {
- return $handler2;
- });
- $comment = new Comment();
- $comment
- ->setActor('users', 'alice')
- ->setObject('files', 'file64')
- ->setMessage('very beautiful, I am impressed!')
- ->setVerb('comment');
- // Add event
- $manager->save($comment);
- // Update event
- $comment->setMessage('Different topic');
- $manager->save($comment);
- // Delete event
- $manager->delete($comment->getId());
- }
- public function testResolveDisplayName(): void {
- $manager = $this->getManager();
- $planetClosure = function ($name) {
- return ucfirst($name);
- };
- $galaxyClosure = function ($name) {
- return strtoupper($name);
- };
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- $manager->registerDisplayNameResolver('galaxy', $galaxyClosure);
- $this->assertSame('Neptune', $manager->resolveDisplayName('planet', 'neptune'));
- $this->assertSame('SOMBRERO', $manager->resolveDisplayName('galaxy', 'sombrero'));
- }
- public function testRegisterResolverDuplicate(): void {
- $this->expectException(\OutOfBoundsException::class);
- $manager = $this->getManager();
- $planetClosure = function ($name) {
- return ucfirst($name);
- };
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- }
- public function testRegisterResolverInvalidType(): void {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $planetClosure = function ($name) {
- return ucfirst($name);
- };
- $manager->registerDisplayNameResolver(1337, $planetClosure);
- }
- public function testResolveDisplayNameUnregisteredType(): void {
- $this->expectException(\OutOfBoundsException::class);
- $manager = $this->getManager();
- $planetClosure = function ($name) {
- return ucfirst($name);
- };
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- $manager->resolveDisplayName('galaxy', 'sombrero');
- }
- public function testResolveDisplayNameDirtyResolver(): void {
- $manager = $this->getManager();
- $planetClosure = function () {
- return null;
- };
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- $this->assertTrue(is_string($manager->resolveDisplayName('planet', 'neptune')));
- }
- private function skipIfNotSupport4ByteUTF() {
- if (!$this->getManager()->supportReactions()) {
- $this->markTestSkipped('MySQL doesn\'t support 4 byte UTF-8');
- }
- }
- /**
- * @dataProvider providerTestReactionAddAndDelete
- *
- * @param IComment[] $comments
- * @param array $reactionsExpected
- * @return void
- */
- public function testReactionAddAndDelete(array $comments, array $reactionsExpected): void {
- $this->skipIfNotSupport4ByteUTF();
- $manager = $this->getManager();
- $processedComments = $this->proccessComments($comments);
- $comment = end($processedComments);
- if ($comment->getParentId()) {
- $parent = $manager->get($comment->getParentId());
- $this->assertEqualsCanonicalizing($reactionsExpected, $parent->getReactions());
- }
- }
- public function providerTestReactionAddAndDelete(): array {
- return[
- [
- [
- ['message', 'alice', 'comment', null],
- ], [],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['👍', 'alice', 'reaction', 'message#alice'],
- ], ['👍' => 1],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['👍', 'alice', 'reaction', 'message#alice'],
- ['👍', 'alice', 'reaction', 'message#alice'],
- ], ['👍' => 1],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['👍', 'alice', 'reaction', 'message#alice'],
- ['👍', 'frank', 'reaction', 'message#alice'],
- ], ['👍' => 2],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['👍', 'alice', 'reaction', 'message#alice'],
- ['👍', 'frank', 'reaction', 'message#alice'],
- ['👍', 'frank', 'reaction_deleted', 'message#alice'],
- ], ['👍' => 1],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['👍', 'alice', 'reaction', 'message#alice'],
- ['👍', 'frank', 'reaction', 'message#alice'],
- ['👍', 'alice', 'reaction_deleted', 'message#alice'],
- ['👍', 'frank', 'reaction_deleted', 'message#alice'],
- ], [],
- ],
- ];
- }
- public function testResolveDisplayNameInvalidType(): void {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $planetClosure = function () {
- return null;
- };
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- $this->assertTrue(is_string($manager->resolveDisplayName(1337, 'neptune')));
- }
- /**
- * @param array $data
- * @return IComment[]
- */
- private function proccessComments(array $data): array {
- /** @var IComment[] */
- $comments = [];
- foreach ($data as $comment) {
- [$message, $actorId, $verb, $parentText] = $comment;
- $parentId = null;
- if ($parentText) {
- $parentId = (string)$comments[$parentText]->getId();
- }
- $id = '';
- if ($verb === 'reaction_deleted') {
- $id = $comments[$message . '#' . $actorId]->getId();
- }
- $comment = $this->testSave($message, $actorId, $verb, $parentId, $id);
- $comments[$comment->getMessage() . '#' . $comment->getActorId()] = $comment;
- }
- return $comments;
- }
- /**
- * @dataProvider providerTestRetrieveAllReactions
- */
- public function testRetrieveAllReactions(array $comments, array $expected): void {
- $this->skipIfNotSupport4ByteUTF();
- $manager = $this->getManager();
- $processedComments = $this->proccessComments($comments);
- $comment = reset($processedComments);
- $all = $manager->retrieveAllReactions($comment->getId());
- $actual = array_map(function ($row) {
- return [
- 'message' => $row->getMessage(),
- 'actorId' => $row->getActorId(),
- ];
- }, $all);
- $this->assertEqualsCanonicalizing($expected, $actual);
- }
- public function providerTestRetrieveAllReactions(): array {
- return [
- [
- [
- ['message', 'alice', 'comment', null],
- ],
- [],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['👍', 'alice', 'reaction', 'message#alice'],
- ['👍', 'frank', 'reaction', 'message#alice'],
- ],
- [
- ['👍', 'alice'],
- ['👍', 'frank'],
- ],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['👍', 'alice', 'reaction', 'message#alice'],
- ['👍', 'alice', 'reaction', 'message#alice'],
- ['👍', 'frank', 'reaction', 'message#alice'],
- ],
- [
- ['👍', 'alice'],
- ['👍', 'frank'],
- ],
- ],
- [# 600 reactions to cover chunk size when retrieve comments of reactions.
- [
- ['message', 'alice', 'comment', null],
- ['😀', 'alice', 'reaction', 'message#alice'],
- ['😃', 'alice', 'reaction', 'message#alice'],
- ['😄', 'alice', 'reaction', 'message#alice'],
- ['😁', 'alice', 'reaction', 'message#alice'],
- ['😆', 'alice', 'reaction', 'message#alice'],
- ['😅', 'alice', 'reaction', 'message#alice'],
- ['😂', 'alice', 'reaction', 'message#alice'],
- ['🤣', 'alice', 'reaction', 'message#alice'],
- ['🥲', 'alice', 'reaction', 'message#alice'],
- ['🥹', 'alice', 'reaction', 'message#alice'],
- ['☺️', 'alice', 'reaction', 'message#alice'],
- ['😊', 'alice', 'reaction', 'message#alice'],
- ['😇', 'alice', 'reaction', 'message#alice'],
- ['🙂', 'alice', 'reaction', 'message#alice'],
- ['🙃', 'alice', 'reaction', 'message#alice'],
- ['😉', 'alice', 'reaction', 'message#alice'],
- ['😌', 'alice', 'reaction', 'message#alice'],
- ['😍', 'alice', 'reaction', 'message#alice'],
- ['🥰', 'alice', 'reaction', 'message#alice'],
- ['😘', 'alice', 'reaction', 'message#alice'],
- ['😗', 'alice', 'reaction', 'message#alice'],
- ['😙', 'alice', 'reaction', 'message#alice'],
- ['😚', 'alice', 'reaction', 'message#alice'],
- ['😋', 'alice', 'reaction', 'message#alice'],
- ['😛', 'alice', 'reaction', 'message#alice'],
- ['😝', 'alice', 'reaction', 'message#alice'],
- ['😜', 'alice', 'reaction', 'message#alice'],
- ['🤪', 'alice', 'reaction', 'message#alice'],
- ['🤨', 'alice', 'reaction', 'message#alice'],
- ['🧐', 'alice', 'reaction', 'message#alice'],
- ['🤓', 'alice', 'reaction', 'message#alice'],
- ['😎', 'alice', 'reaction', 'message#alice'],
- ['🥸', 'alice', 'reaction', 'message#alice'],
- ['🤩', 'alice', 'reaction', 'message#alice'],
- ['🥳', 'alice', 'reaction', 'message#alice'],
- ['😏', 'alice', 'reaction', 'message#alice'],
- ['😒', 'alice', 'reaction', 'message#alice'],
- ['😞', 'alice', 'reaction', 'message#alice'],
- ['😔', 'alice', 'reaction', 'message#alice'],
- ['😟', 'alice', 'reaction', 'message#alice'],
- ['😕', 'alice', 'reaction', 'message#alice'],
- ['🙁', 'alice', 'reaction', 'message#alice'],
- ['☹️', 'alice', 'reaction', 'message#alice'],
- ['😣', 'alice', 'reaction', 'message#alice'],
- ['😖', 'alice', 'reaction', 'message#alice'],
- ['😫', 'alice', 'reaction', 'message#alice'],
- ['😩', 'alice', 'reaction', 'message#alice'],
- ['🥺', 'alice', 'reaction', 'message#alice'],
- ['😢', 'alice', 'reaction', 'message#alice'],
- ['😭', 'alice', 'reaction', 'message#alice'],
- ['😮💨', 'alice', 'reaction', 'message#alice'],
- ['😤', 'alice', 'reaction', 'message#alice'],
- ['😠', 'alice', 'reaction', 'message#alice'],
- ['😡', 'alice', 'reaction', 'message#alice'],
- ['🤬', 'alice', 'reaction', 'message#alice'],
- ['🤯', 'alice', 'reaction', 'message#alice'],
- ['😳', 'alice', 'reaction', 'message#alice'],
- ['🥵', 'alice', 'reaction', 'message#alice'],
- ['🥶', 'alice', 'reaction', 'message#alice'],
- ['😱', 'alice', 'reaction', 'message#alice'],
- ['😨', 'alice', 'reaction', 'message#alice'],
- ['😰', 'alice', 'reaction', 'message#alice'],
- ['😥', 'alice', 'reaction', 'message#alice'],
- ['😓', 'alice', 'reaction', 'message#alice'],
- ['🫣', 'alice', 'reaction', 'message#alice'],
- ['🤗', 'alice', 'reaction', 'message#alice'],
- ['🫡', 'alice', 'reaction', 'message#alice'],
- ['🤔', 'alice', 'reaction', 'message#alice'],
- ['🫢', 'alice', 'reaction', 'message#alice'],
- ['🤭', 'alice', 'reaction', 'message#alice'],
- ['🤫', 'alice', 'reaction', 'message#alice'],
- ['🤥', 'alice', 'reaction', 'message#alice'],
- ['😶', 'alice', 'reaction', 'message#alice'],
- ['😶🌫️', 'alice', 'reaction', 'message#alice'],
- ['😐', 'alice', 'reaction', 'message#alice'],
- ['😑', 'alice', 'reaction', 'message#alice'],
- ['😬', 'alice', 'reaction', 'message#alice'],
- ['🫠', 'alice', 'reaction', 'message#alice'],
- ['🙄', 'alice', 'reaction', 'message#alice'],
- ['😯', 'alice', 'reaction', 'message#alice'],
- ['😦', 'alice', 'reaction', 'message#alice'],
- ['😧', 'alice', 'reaction', 'message#alice'],
- ['😮', 'alice', 'reaction', 'message#alice'],
- ['😲', 'alice', 'reaction', 'message#alice'],
- ['🥱', 'alice', 'reaction', 'message#alice'],
- ['😴', 'alice', 'reaction', 'message#alice'],
- ['🤤', 'alice', 'reaction', 'message#alice'],
- ['😪', 'alice', 'reaction', 'message#alice'],
- ['😵', 'alice', 'reaction', 'message#alice'],
- ['😵💫', 'alice', 'reaction', 'message#alice'],
- ['🫥', 'alice', 'reaction', 'message#alice'],
- ['🤐', 'alice', 'reaction', 'message#alice'],
- ['🥴', 'alice', 'reaction', 'message#alice'],
- ['🤢', 'alice', 'reaction', 'message#alice'],
- ['🤮', 'alice', 'reaction', 'message#alice'],
- ['🤧', 'alice', 'reaction', 'message#alice'],
- ['😷', 'alice', 'reaction', 'message#alice'],
- ['🤒', 'alice', 'reaction', 'message#alice'],
- ['🤕', 'alice', 'reaction', 'message#alice'],
- ['🤑', 'alice', 'reaction', 'message#alice'],
- ['🤠', 'alice', 'reaction', 'message#alice'],
- ['😈', 'alice', 'reaction', 'message#alice'],
- ['👿', 'alice', 'reaction', 'message#alice'],
- ['👹', 'alice', 'reaction', 'message#alice'],
- ['👺', 'alice', 'reaction', 'message#alice'],
- ['🤡', 'alice', 'reaction', 'message#alice'],
- ['💩', 'alice', 'reaction', 'message#alice'],
- ['👻', 'alice', 'reaction', 'message#alice'],
- ['💀', 'alice', 'reaction', 'message#alice'],
- ['☠️', 'alice', 'reaction', 'message#alice'],
- ['👽', 'alice', 'reaction', 'message#alice'],
- ['👾', 'alice', 'reaction', 'message#alice'],
- ['🤖', 'alice', 'reaction', 'message#alice'],
- ['🎃', 'alice', 'reaction', 'message#alice'],
- ['😺', 'alice', 'reaction', 'message#alice'],
- ['😸', 'alice', 'reaction', 'message#alice'],
- ['😹', 'alice', 'reaction', 'message#alice'],
- ['😻', 'alice', 'reaction', 'message#alice'],
- ['😼', 'alice', 'reaction', 'message#alice'],
- ['😽', 'alice', 'reaction', 'message#alice'],
- ['🙀', 'alice', 'reaction', 'message#alice'],
- ['😿', 'alice', 'reaction', 'message#alice'],
- ['😾', 'alice', 'reaction', 'message#alice'],
- ['👶', 'alice', 'reaction', 'message#alice'],
- ['👧', 'alice', 'reaction', 'message#alice'],
- ['🧒', 'alice', 'reaction', 'message#alice'],
- ['👦', 'alice', 'reaction', 'message#alice'],
- ['👩', 'alice', 'reaction', 'message#alice'],
- ['🧑', 'alice', 'reaction', 'message#alice'],
- ['👨', 'alice', 'reaction', 'message#alice'],
- ['👩🦱', 'alice', 'reaction', 'message#alice'],
- ['🧑🦱', 'alice', 'reaction', 'message#alice'],
- ['👨🦱', 'alice', 'reaction', 'message#alice'],
- ['👩🦰', 'alice', 'reaction', 'message#alice'],
- ['🧑🦰', 'alice', 'reaction', 'message#alice'],
- ['👨🦰', 'alice', 'reaction', 'message#alice'],
- ['👱♀️', 'alice', 'reaction', 'message#alice'],
- ['👱', 'alice', 'reaction', 'message#alice'],
- ['👱♂️', 'alice', 'reaction', 'message#alice'],
- ['👩🦳', 'alice', 'reaction', 'message#alice'],
- ['🧑🦳', 'alice', 'reaction', 'message#alice'],
- ['👨🦳', 'alice', 'reaction', 'message#alice'],
- ['👩🦲', 'alice', 'reaction', 'message#alice'],
- ['🧑🦲', 'alice', 'reaction', 'message#alice'],
- ['👨🦲', 'alice', 'reaction', 'message#alice'],
- ['🧔♀️', 'alice', 'reaction', 'message#alice'],
- ['🧔', 'alice', 'reaction', 'message#alice'],
- ['🧔♂️', 'alice', 'reaction', 'message#alice'],
- ['👵', 'alice', 'reaction', 'message#alice'],
- ['🧓', 'alice', 'reaction', 'message#alice'],
- ['👴', 'alice', 'reaction', 'message#alice'],
- ['👲', 'alice', 'reaction', 'message#alice'],
- ['👳♀️', 'alice', 'reaction', 'message#alice'],
- ['👳', 'alice', 'reaction', 'message#alice'],
- ['👳♂️', 'alice', 'reaction', 'message#alice'],
- ['🧕', 'alice', 'reaction', 'message#alice'],
- ['👮♀️', 'alice', 'reaction', 'message#alice'],
- ['👮', 'alice', 'reaction', 'message#alice'],
- ['👮♂️', 'alice', 'reaction', 'message#alice'],
- ['👷♀️', 'alice', 'reaction', 'message#alice'],
- ['👷', 'alice', 'reaction', 'message#alice'],
- ['👷♂️', 'alice', 'reaction', 'message#alice'],
- ['💂♀️', 'alice', 'reaction', 'message#alice'],
- ['💂', 'alice', 'reaction', 'message#alice'],
- ['💂♂️', 'alice', 'reaction', 'message#alice'],
- ['🕵️♀️', 'alice', 'reaction', 'message#alice'],
- ['🕵️', 'alice', 'reaction', 'message#alice'],
- ['🕵️♂️', 'alice', 'reaction', 'message#alice'],
- ['👩⚕️', 'alice', 'reaction', 'message#alice'],
- ['🧑⚕️', 'alice', 'reaction', 'message#alice'],
- ['👨⚕️', 'alice', 'reaction', 'message#alice'],
- ['👩🌾', 'alice', 'reaction', 'message#alice'],
- ['🧑🌾', 'alice', 'reaction', 'message#alice'],
- ['👨🌾', 'alice', 'reaction', 'message#alice'],
- ['👩🍳', 'alice', 'reaction', 'message#alice'],
- ['🧑🍳', 'alice', 'reaction', 'message#alice'],
- ['👨🍳', 'alice', 'reaction', 'message#alice'],
- ['👩🎓', 'alice', 'reaction', 'message#alice'],
- ['🧑🎓', 'alice', 'reaction', 'message#alice'],
- ['👨🎓', 'alice', 'reaction', 'message#alice'],
- ['👩🎤', 'alice', 'reaction', 'message#alice'],
- ['🧑🎤', 'alice', 'reaction', 'message#alice'],
- ['👨🎤', 'alice', 'reaction', 'message#alice'],
- ['👩🏫', 'alice', 'reaction', 'message#alice'],
- ['🧑🏫', 'alice', 'reaction', 'message#alice'],
- ['👨🏫', 'alice', 'reaction', 'message#alice'],
- ['👩🏭', 'alice', 'reaction', 'message#alice'],
- ['🧑🏭', 'alice', 'reaction', 'message#alice'],
- ['👨🏭', 'alice', 'reaction', 'message#alice'],
- ['👩💻', 'alice', 'reaction', 'message#alice'],
- ['🧑💻', 'alice', 'reaction', 'message#alice'],
- ['👨💻', 'alice', 'reaction', 'message#alice'],
- ['👩💼', 'alice', 'reaction', 'message#alice'],
- ['🧑💼', 'alice', 'reaction', 'message#alice'],
- ['👨💼', 'alice', 'reaction', 'message#alice'],
- ['👩🔧', 'alice', 'reaction', 'message#alice'],
- ['🧑🔧', 'alice', 'reaction', 'message#alice'],
- ['👨🔧', 'alice', 'reaction', 'message#alice'],
- ['👩🔬', 'alice', 'reaction', 'message#alice'],
- ['🧑🔬', 'alice', 'reaction', 'message#alice'],
- ['👨🔬', 'alice', 'reaction', 'message#alice'],
- ['👩🎨', 'alice', 'reaction', 'message#alice'],
- ['🧑🎨', 'alice', 'reaction', 'message#alice'],
- ['👨🎨', 'alice', 'reaction', 'message#alice'],
- ['👩🚒', 'alice', 'reaction', 'message#alice'],
- ['🧑🚒', 'alice', 'reaction', 'message#alice'],
- ['👨🚒', 'alice', 'reaction', 'message#alice'],
- ['👩✈️', 'alice', 'reaction', 'message#alice'],
- ['🧑✈️', 'alice', 'reaction', 'message#alice'],
- ['👨✈️', 'alice', 'reaction', 'message#alice'],
- ['👩🚀', 'alice', 'reaction', 'message#alice'],
- ['🧑🚀', 'alice', 'reaction', 'message#alice'],
- ['👨🚀', 'alice', 'reaction', 'message#alice'],
- ['👩⚖️', 'alice', 'reaction', 'message#alice'],
- ['🧑⚖️', 'alice', 'reaction', 'message#alice'],
- ['👨⚖️', 'alice', 'reaction', 'message#alice'],
- ['👰♀️', 'alice', 'reaction', 'message#alice'],
- ['👰', 'alice', 'reaction', 'message#alice'],
- ['👰♂️', 'alice', 'reaction', 'message#alice'],
- ['🤵♀️', 'alice', 'reaction', 'message#alice'],
- ['🤵', 'alice', 'reaction', 'message#alice'],
- ['🤵♂️', 'alice', 'reaction', 'message#alice'],
- ['👸', 'alice', 'reaction', 'message#alice'],
- ['🫅', 'alice', 'reaction', 'message#alice'],
- ['🤴', 'alice', 'reaction', 'message#alice'],
- ['🥷', 'alice', 'reaction', 'message#alice'],
- ['🦸♀️', 'alice', 'reaction', 'message#alice'],
- ['🦸', 'alice', 'reaction', 'message#alice'],
- ['🦸♂️', 'alice', 'reaction', 'message#alice'],
- ['🦹♀️', 'alice', 'reaction', 'message#alice'],
- ['🦹', 'alice', 'reaction', 'message#alice'],
- ['🦹♂️', 'alice', 'reaction', 'message#alice'],
- ['🤶', 'alice', 'reaction', 'message#alice'],
- ['🧑🎄', 'alice', 'reaction', 'message#alice'],
- ['🎅', 'alice', 'reaction', 'message#alice'],
- ['🧙♀️', 'alice', 'reaction', 'message#alice'],
- ['🧙', 'alice', 'reaction', 'message#alice'],
- ['🧙♂️', 'alice', 'reaction', 'message#alice'],
- ['🧝♀️', 'alice', 'reaction', 'message#alice'],
- ['🧝', 'alice', 'reaction', 'message#alice'],
- ['🧝♂️', 'alice', 'reaction', 'message#alice'],
- ['🧛♀️', 'alice', 'reaction', 'message#alice'],
- ['🧛', 'alice', 'reaction', 'message#alice'],
- ['🧛♂️', 'alice', 'reaction', 'message#alice'],
- ['🧟♀️', 'alice', 'reaction', 'message#alice'],
- ['🧟', 'alice', 'reaction', 'message#alice'],
- ['🧟♂️', 'alice', 'reaction', 'message#alice'],
- ['🧞♀️', 'alice', 'reaction', 'message#alice'],
- ['🧞', 'alice', 'reaction', 'message#alice'],
- ['🧞♂️', 'alice', 'reaction', 'message#alice'],
- ['🧜♀️', 'alice', 'reaction', 'message#alice'],
- ['🧜', 'alice', 'reaction', 'message#alice'],
- ['🧜♂️', 'alice', 'reaction', 'message#alice'],
- ['🧚♀️', 'alice', 'reaction', 'message#alice'],
- ['🧚', 'alice', 'reaction', 'message#alice'],
- ['🧚♂️', 'alice', 'reaction', 'message#alice'],
- ['🧌', 'alice', 'reaction', 'message#alice'],
- ['👼', 'alice', 'reaction', 'message#alice'],
- ['🤰', 'alice', 'reaction', 'message#alice'],
- ['🫄', 'alice', 'reaction', 'message#alice'],
- ['🫃', 'alice', 'reaction', 'message#alice'],
- ['🤱', 'alice', 'reaction', 'message#alice'],
- ['👩🍼', 'alice', 'reaction', 'message#alice'],
- ['🧑🍼', 'alice', 'reaction', 'message#alice'],
- ['👨🍼', 'alice', 'reaction', 'message#alice'],
- ['🙇♀️', 'alice', 'reaction', 'message#alice'],
- ['🙇', 'alice', 'reaction', 'message#alice'],
- ['🙇♂️', 'alice', 'reaction', 'message#alice'],
- ['💁♀️', 'alice', 'reaction', 'message#alice'],
- ['💁', 'alice', 'reaction', 'message#alice'],
- ['💁♂️', 'alice', 'reaction', 'message#alice'],
- ['🙅♀️', 'alice', 'reaction', 'message#alice'],
- ['🙅', 'alice', 'reaction', 'message#alice'],
- ['🙅♂️', 'alice', 'reaction', 'message#alice'],
- ['🙆♀️', 'alice', 'reaction', 'message#alice'],
- ['🙆', 'alice', 'reaction', 'message#alice'],
- ['🙆♂️', 'alice', 'reaction', 'message#alice'],
- ['🙋♀️', 'alice', 'reaction', 'message#alice'],
- ['🙋', 'alice', 'reaction', 'message#alice'],
- ['🙋♂️', 'alice', 'reaction', 'message#alice'],
- ['🧏♀️', 'alice', 'reaction', 'message#alice'],
- ['🧏', 'alice', 'reaction', 'message#alice'],
- ['🧏♂️', 'alice', 'reaction', 'message#alice'],
- ['🤦♀️', 'alice', 'reaction', 'message#alice'],
- ['🤦', 'alice', 'reaction', 'message#alice'],
- ['🤦♂️', 'alice', 'reaction', 'message#alice'],
- ['🤷♀️', 'alice', 'reaction', 'message#alice'],
- ['🤷', 'alice', 'reaction', 'message#alice'],
- ['🤷♂️', 'alice', 'reaction', 'message#alice'],
- ['🙎♀️', 'alice', 'reaction', 'message#alice'],
- ['🙎', 'alice', 'reaction', 'message#alice'],
- ['🙎♂️', 'alice', 'reaction', 'message#alice'],
- ['🙍♀️', 'alice', 'reaction', 'message#alice'],
- ['🙍', 'alice', 'reaction', 'message#alice'],
- ['🙍♂️', 'alice', 'reaction', 'message#alice'],
- ['💇♀️', 'alice', 'reaction', 'message#alice'],
- ['💇', 'alice', 'reaction', 'message#alice'],
- ['💇♂️', 'alice', 'reaction', 'message#alice'],
- ['💆♀️', 'alice', 'reaction', 'message#alice'],
- ['💆', 'alice', 'reaction', 'message#alice'],
- ['💆♂️', 'alice', 'reaction', 'message#alice'],
- ['🧖♀️', 'alice', 'reaction', 'message#alice'],
- ['🧖', 'alice', 'reaction', 'message#alice'],
- ['🧖♂️', 'alice', 'reaction', 'message#alice'],
- ['💅', 'alice', 'reaction', 'message#alice'],
- ['🤳', 'alice', 'reaction', 'message#alice'],
- ['💃', 'alice', 'reaction', 'message#alice'],
- ['🕺', 'alice', 'reaction', 'message#alice'],
- ['👯♀️', 'alice', 'reaction', 'message#alice'],
- ['👯', 'alice', 'reaction', 'message#alice'],
- ['👯♂️', 'alice', 'reaction', 'message#alice'],
- ['🕴', 'alice', 'reaction', 'message#alice'],
- ['👩🦽', 'alice', 'reaction', 'message#alice'],
- ['🧑🦽', 'alice', 'reaction', 'message#alice'],
- ['👨🦽', 'alice', 'reaction', 'message#alice'],
- ['👩🦼', 'alice', 'reaction', 'message#alice'],
- ['🧑🦼', 'alice', 'reaction', 'message#alice'],
- ['👨🦼', 'alice', 'reaction', 'message#alice'],
- ['🚶♀️', 'alice', 'reaction', 'message#alice'],
- ['🚶', 'alice', 'reaction', 'message#alice'],
- ['🚶♂️', 'alice', 'reaction', 'message#alice'],
- ['👩🦯', 'alice', 'reaction', 'message#alice'],
- ['🧑🦯', 'alice', 'reaction', 'message#alice'],
- ['👨🦯', 'alice', 'reaction', 'message#alice'],
- ['🧎♀️', 'alice', 'reaction', 'message#alice'],
- ['🧎', 'alice', 'reaction', 'message#alice'],
- ['🧎♂️', 'alice', 'reaction', 'message#alice'],
- ['🏃♀️', 'alice', 'reaction', 'message#alice'],
- ['🏃', 'alice', 'reaction', 'message#alice'],
- ['🏃♂️', 'alice', 'reaction', 'message#alice'],
- ['🧍♀️', 'alice', 'reaction', 'message#alice'],
- ['🧍', 'alice', 'reaction', 'message#alice'],
- ['🧍♂️', 'alice', 'reaction', 'message#alice'],
- ['👭', 'alice', 'reaction', 'message#alice'],
- ['🧑🤝🧑', 'alice', 'reaction', 'message#alice'],
- ['👬', 'alice', 'reaction', 'message#alice'],
- ['👫', 'alice', 'reaction', 'message#alice'],
- ['👩❤️👩', 'alice', 'reaction', 'message#alice'],
- ['💑', 'alice', 'reaction', 'message#alice'],
- ['👨❤️👨', 'alice', 'reaction', 'message#alice'],
- ['👩❤️👨', 'alice', 'reaction', 'message#alice'],
- ['👩❤️💋👩', 'alice', 'reaction', 'message#alice'],
- ['💏', 'alice', 'reaction', 'message#alice'],
- ['👨❤️💋👨', 'alice', 'reaction', 'message#alice'],
- ['👩❤️💋👨', 'alice', 'reaction', 'message#alice'],
- ['👪', 'alice', 'reaction', 'message#alice'],
- ['👨👩👦', 'alice', 'reaction', 'message#alice'],
- ['👨👩👧', 'alice', 'reaction', 'message#alice'],
- ['👨👩👧👦', 'alice', 'reaction', 'message#alice'],
- ['👨👩👦👦', 'alice', 'reaction', 'message#alice'],
- ['👨👩👧👧', 'alice', 'reaction', 'message#alice'],
- ['👨👨👦', 'alice', 'reaction', 'message#alice'],
- ['👨👨👧', 'alice', 'reaction', 'message#alice'],
- ['👨👨👧👦', 'alice', 'reaction', 'message#alice'],
- ['👨👨👦👦', 'alice', 'reaction', 'message#alice'],
- ['👨👨👧👧', 'alice', 'reaction', 'message#alice'],
- ['👩👩👦', 'alice', 'reaction', 'message#alice'],
- ['👩👩👧', 'alice', 'reaction', 'message#alice'],
- ['👩👩👧👦', 'alice', 'reaction', 'message#alice'],
- ['👩👩👦👦', 'alice', 'reaction', 'message#alice'],
- ['👩👩👧👧', 'alice', 'reaction', 'message#alice'],
- ['👨👦', 'alice', 'reaction', 'message#alice'],
- ['👨👦👦', 'alice', 'reaction', 'message#alice'],
- ['👨👧', 'alice', 'reaction', 'message#alice'],
- ['👨👧👦', 'alice', 'reaction', 'message#alice'],
- ['👨👧👧', 'alice', 'reaction', 'message#alice'],
- ['👩👦', 'alice', 'reaction', 'message#alice'],
- ['👩👦👦', 'alice', 'reaction', 'message#alice'],
- ['👩👧', 'alice', 'reaction', 'message#alice'],
- ['👩👧👦', 'alice', 'reaction', 'message#alice'],
- ['👩👧👧', 'alice', 'reaction', 'message#alice'],
- ['🗣', 'alice', 'reaction', 'message#alice'],
- ['👤', 'alice', 'reaction', 'message#alice'],
- ['👥', 'alice', 'reaction', 'message#alice'],
- ['🫂', 'alice', 'reaction', 'message#alice'],
- ['👋🏽', 'alice', 'reaction', 'message#alice'],
- ['🤚🏽', 'alice', 'reaction', 'message#alice'],
- ['🖐🏽', 'alice', 'reaction', 'message#alice'],
- ['✋🏽', 'alice', 'reaction', 'message#alice'],
- ['🖖🏽', 'alice', 'reaction', 'message#alice'],
- ['👌🏽', 'alice', 'reaction', 'message#alice'],
- ['🤌🏽', 'alice', 'reaction', 'message#alice'],
- ['🤏🏽', 'alice', 'reaction', 'message#alice'],
- ['✌🏽', 'alice', 'reaction', 'message#alice'],
- ['🤞🏽', 'alice', 'reaction', 'message#alice'],
- ['🫰🏽', 'alice', 'reaction', 'message#alice'],
- ['🤟🏽', 'alice', 'reaction', 'message#alice'],
- ['🤘🏽', 'alice', 'reaction', 'message#alice'],
- ['🤙🏽', 'alice', 'reaction', 'message#alice'],
- ['🫵🏽', 'alice', 'reaction', 'message#alice'],
- ['🫱🏽', 'alice', 'reaction', 'message#alice'],
- ['🫲🏽', 'alice', 'reaction', 'message#alice'],
- ['🫳🏽', 'alice', 'reaction', 'message#alice'],
- ['🫴🏽', 'alice', 'reaction', 'message#alice'],
- ['👈🏽', 'alice', 'reaction', 'message#alice'],
- ['👉🏽', 'alice', 'reaction', 'message#alice'],
- ['👆🏽', 'alice', 'reaction', 'message#alice'],
- ['🖕🏽', 'alice', 'reaction', 'message#alice'],
- ['👇🏽', 'alice', 'reaction', 'message#alice'],
- ['☝🏽', 'alice', 'reaction', 'message#alice'],
- ['👍🏽', 'alice', 'reaction', 'message#alice'],
- ['👎🏽', 'alice', 'reaction', 'message#alice'],
- ['✊🏽', 'alice', 'reaction', 'message#alice'],
- ['👊🏽', 'alice', 'reaction', 'message#alice'],
- ['🤛🏽', 'alice', 'reaction', 'message#alice'],
- ['🤜🏽', 'alice', 'reaction', 'message#alice'],
- ['👏🏽', 'alice', 'reaction', 'message#alice'],
- ['🫶🏽', 'alice', 'reaction', 'message#alice'],
- ['🙌🏽', 'alice', 'reaction', 'message#alice'],
- ['👐🏽', 'alice', 'reaction', 'message#alice'],
- ['🤲🏽', 'alice', 'reaction', 'message#alice'],
- ['🙏🏽', 'alice', 'reaction', 'message#alice'],
- ['✍🏽', 'alice', 'reaction', 'message#alice'],
- ['💅🏽', 'alice', 'reaction', 'message#alice'],
- ['🤳🏽', 'alice', 'reaction', 'message#alice'],
- ['💪🏽', 'alice', 'reaction', 'message#alice'],
- ['🦵🏽', 'alice', 'reaction', 'message#alice'],
- ['🦶🏽', 'alice', 'reaction', 'message#alice'],
- ['👂🏽', 'alice', 'reaction', 'message#alice'],
- ['🦻🏽', 'alice', 'reaction', 'message#alice'],
- ['👃🏽', 'alice', 'reaction', 'message#alice'],
- ['👶🏽', 'alice', 'reaction', 'message#alice'],
- ['👧🏽', 'alice', 'reaction', 'message#alice'],
- ['🧒🏽', 'alice', 'reaction', 'message#alice'],
- ['👦🏽', 'alice', 'reaction', 'message#alice'],
- ['👩🏽', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽', 'alice', 'reaction', 'message#alice'],
- ['👨🏽', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🦱', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🦱', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🦱', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🦰', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🦰', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🦰', 'alice', 'reaction', 'message#alice'],
- ['👱🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['👱🏽', 'alice', 'reaction', 'message#alice'],
- ['👱🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🦳', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🦳', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🦳', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🦲', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🦲', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🦲', 'alice', 'reaction', 'message#alice'],
- ['🧔🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🧔🏽', 'alice', 'reaction', 'message#alice'],
- ['🧔🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['👵🏽', 'alice', 'reaction', 'message#alice'],
- ['🧓🏽', 'alice', 'reaction', 'message#alice'],
- ['👴🏽', 'alice', 'reaction', 'message#alice'],
- ['👲🏽', 'alice', 'reaction', 'message#alice'],
- ['👳🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['👳🏽', 'alice', 'reaction', 'message#alice'],
- ['👳🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🧕🏽', 'alice', 'reaction', 'message#alice'],
- ['👮🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['👮🏽', 'alice', 'reaction', 'message#alice'],
- ['👮🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['👷🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['👷🏽', 'alice', 'reaction', 'message#alice'],
- ['👷🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['💂🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['💂🏽', 'alice', 'reaction', 'message#alice'],
- ['💂🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🕵🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🕵🏽', 'alice', 'reaction', 'message#alice'],
- ['🕵🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['👩🏽⚕️', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽⚕️', 'alice', 'reaction', 'message#alice'],
- ['👨🏽⚕️', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🌾', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🌾', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🌾', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🍳', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🍳', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🍳', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🎓', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🎓', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🎓', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🎤', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🎤', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🎤', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🏫', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🏫', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🏫', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🏭', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🏭', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🏭', 'alice', 'reaction', 'message#alice'],
- ['👩🏽💻', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽💻', 'alice', 'reaction', 'message#alice'],
- ['👨🏽💻', 'alice', 'reaction', 'message#alice'],
- ['👩🏽💼', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽💼', 'alice', 'reaction', 'message#alice'],
- ['👨🏽💼', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🔧', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🔧', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🔧', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🔬', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🔬', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🔬', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🎨', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🎨', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🎨', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🚒', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🚒', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🚒', 'alice', 'reaction', 'message#alice'],
- ['👩🏽✈️', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽✈️', 'alice', 'reaction', 'message#alice'],
- ['👨🏽✈️', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🚀', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🚀', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🚀', 'alice', 'reaction', 'message#alice'],
- ['👩🏽⚖️', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽⚖️', 'alice', 'reaction', 'message#alice'],
- ['👨🏽⚖️', 'alice', 'reaction', 'message#alice'],
- ['👰🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['👰🏽', 'alice', 'reaction', 'message#alice'],
- ['👰🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🤵🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🤵🏽', 'alice', 'reaction', 'message#alice'],
- ['🤵🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['👸🏽', 'alice', 'reaction', 'message#alice'],
- ['🫅🏽', 'alice', 'reaction', 'message#alice'],
- ['🤴🏽', 'alice', 'reaction', 'message#alice'],
- ['🥷🏽', 'alice', 'reaction', 'message#alice'],
- ['🦸🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🦸🏽', 'alice', 'reaction', 'message#alice'],
- ['🦸🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🦹🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🦹🏽', 'alice', 'reaction', 'message#alice'],
- ['🦹🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🤶🏽', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🎄', 'alice', 'reaction', 'message#alice'],
- ['🎅🏽', 'alice', 'reaction', 'message#alice'],
- ['🧙🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🧙🏽', 'alice', 'reaction', 'message#alice'],
- ['🧙🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🧝🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🧝🏽', 'alice', 'reaction', 'message#alice'],
- ['🧝🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🧛🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🧛🏽', 'alice', 'reaction', 'message#alice'],
- ['🧛🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🧜🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🧜🏽', 'alice', 'reaction', 'message#alice'],
- ['🧜🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🧚🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🧚🏽', 'alice', 'reaction', 'message#alice'],
- ['🧚🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['👼🏽', 'alice', 'reaction', 'message#alice'],
- ['🤰🏽', 'alice', 'reaction', 'message#alice'],
- ['🫄🏽', 'alice', 'reaction', 'message#alice'],
- ['🫃🏽', 'alice', 'reaction', 'message#alice'],
- ['🤱🏽', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🍼', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🍼', 'alice', 'reaction', 'message#alice'],
- ['👨🏽🍼', 'alice', 'reaction', 'message#alice'],
- ['🙇🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🙇🏽', 'alice', 'reaction', 'message#alice'],
- ['🙇🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['💁🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['💁🏽', 'alice', 'reaction', 'message#alice'],
- ['💁🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🙅🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🙅🏽', 'alice', 'reaction', 'message#alice'],
- ['🙅🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🙆🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🙆🏽', 'alice', 'reaction', 'message#alice'],
- ['🙆🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🙋🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🙋🏽', 'alice', 'reaction', 'message#alice'],
- ['🙋🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🧏🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🧏🏽', 'alice', 'reaction', 'message#alice'],
- ['🧏🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🤦🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🤦🏽', 'alice', 'reaction', 'message#alice'],
- ['🤦🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🤷🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🤷🏽', 'alice', 'reaction', 'message#alice'],
- ['🤷🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🙎🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🙎🏽', 'alice', 'reaction', 'message#alice'],
- ['🙎🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🙍🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🙍🏽', 'alice', 'reaction', 'message#alice'],
- ['🙍🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['💇🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['💇🏽', 'alice', 'reaction', 'message#alice'],
- ['💇🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['💆🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['💆🏽', 'alice', 'reaction', 'message#alice'],
- ['💆🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['🧖🏽♀️', 'alice', 'reaction', 'message#alice'],
- ['🧖🏽', 'alice', 'reaction', 'message#alice'],
- ['🧖🏽♂️', 'alice', 'reaction', 'message#alice'],
- ['💃🏽', 'alice', 'reaction', 'message#alice'],
- ['🕺🏽', 'alice', 'reaction', 'message#alice'],
- ['🕴🏽', 'alice', 'reaction', 'message#alice'],
- ['👩🏽🦽', 'alice', 'reaction', 'message#alice'],
- ['🧑🏽🦽', 'alice', 'reaction', 'message#alice'],
- ],
- [
- ['😀', 'alice'],
- ['😃', 'alice'],
- ['😄', 'alice'],
- ['😁', 'alice'],
- ['😆', 'alice'],
- ['😅', 'alice'],
- ['😂', 'alice'],
- ['🤣', 'alice'],
- ['🥲', 'alice'],
- ['🥹', 'alice'],
- ['☺️', 'alice'],
- ['😊', 'alice'],
- ['😇', 'alice'],
- ['🙂', 'alice'],
- ['🙃', 'alice'],
- ['😉', 'alice'],
- ['😌', 'alice'],
- ['😍', 'alice'],
- ['🥰', 'alice'],
- ['😘', 'alice'],
- ['😗', 'alice'],
- ['😙', 'alice'],
- ['😚', 'alice'],
- ['😋', 'alice'],
- ['😛', 'alice'],
- ['😝', 'alice'],
- ['😜', 'alice'],
- ['🤪', 'alice'],
- ['🤨', 'alice'],
- ['🧐', 'alice'],
- ['🤓', 'alice'],
- ['😎', 'alice'],
- ['🥸', 'alice'],
- ['🤩', 'alice'],
- ['🥳', 'alice'],
- ['😏', 'alice'],
- ['😒', 'alice'],
- ['😞', 'alice'],
- ['😔', 'alice'],
- ['😟', 'alice'],
- ['😕', 'alice'],
- ['🙁', 'alice'],
- ['☹️', 'alice'],
- ['😣', 'alice'],
- ['😖', 'alice'],
- ['😫', 'alice'],
- ['😩', 'alice'],
- ['🥺', 'alice'],
- ['😢', 'alice'],
- ['😭', 'alice'],
- ['😮💨', 'alice'],
- ['😤', 'alice'],
- ['😠', 'alice'],
- ['😡', 'alice'],
- ['🤬', 'alice'],
- ['🤯', 'alice'],
- ['😳', 'alice'],
- ['🥵', 'alice'],
- ['🥶', 'alice'],
- ['😱', 'alice'],
- ['😨', 'alice'],
- ['😰', 'alice'],
- ['😥', 'alice'],
- ['😓', 'alice'],
- ['🫣', 'alice'],
- ['🤗', 'alice'],
- ['🫡', 'alice'],
- ['🤔', 'alice'],
- ['🫢', 'alice'],
- ['🤭', 'alice'],
- ['🤫', 'alice'],
- ['🤥', 'alice'],
- ['😶', 'alice'],
- ['😶🌫️', 'alice'],
- ['😐', 'alice'],
- ['😑', 'alice'],
- ['😬', 'alice'],
- ['🫠', 'alice'],
- ['🙄', 'alice'],
- ['😯', 'alice'],
- ['😦', 'alice'],
- ['😧', 'alice'],
- ['😮', 'alice'],
- ['😲', 'alice'],
- ['🥱', 'alice'],
- ['😴', 'alice'],
- ['🤤', 'alice'],
- ['😪', 'alice'],
- ['😵', 'alice'],
- ['😵💫', 'alice'],
- ['🫥', 'alice'],
- ['🤐', 'alice'],
- ['🥴', 'alice'],
- ['🤢', 'alice'],
- ['🤮', 'alice'],
- ['🤧', 'alice'],
- ['😷', 'alice'],
- ['🤒', 'alice'],
- ['🤕', 'alice'],
- ['🤑', 'alice'],
- ['🤠', 'alice'],
- ['😈', 'alice'],
- ['👿', 'alice'],
- ['👹', 'alice'],
- ['👺', 'alice'],
- ['🤡', 'alice'],
- ['💩', 'alice'],
- ['👻', 'alice'],
- ['💀', 'alice'],
- ['☠️', 'alice'],
- ['👽', 'alice'],
- ['👾', 'alice'],
- ['🤖', 'alice'],
- ['🎃', 'alice'],
- ['😺', 'alice'],
- ['😸', 'alice'],
- ['😹', 'alice'],
- ['😻', 'alice'],
- ['😼', 'alice'],
- ['😽', 'alice'],
- ['🙀', 'alice'],
- ['😿', 'alice'],
- ['😾', 'alice'],
- ['👶', 'alice'],
- ['👧', 'alice'],
- ['🧒', 'alice'],
- ['👦', 'alice'],
- ['👩', 'alice'],
- ['🧑', 'alice'],
- ['👨', 'alice'],
- ['👩🦱', 'alice'],
- ['🧑🦱', 'alice'],
- ['👨🦱', 'alice'],
- ['👩🦰', 'alice'],
- ['🧑🦰', 'alice'],
- ['👨🦰', 'alice'],
- ['👱♀️', 'alice'],
- ['👱', 'alice'],
- ['👱♂️', 'alice'],
- ['👩🦳', 'alice'],
- ['🧑🦳', 'alice'],
- ['👨🦳', 'alice'],
- ['👩🦲', 'alice'],
- ['🧑🦲', 'alice'],
- ['👨🦲', 'alice'],
- ['🧔♀️', 'alice'],
- ['🧔', 'alice'],
- ['🧔♂️', 'alice'],
- ['👵', 'alice'],
- ['🧓', 'alice'],
- ['👴', 'alice'],
- ['👲', 'alice'],
- ['👳♀️', 'alice'],
- ['👳', 'alice'],
- ['👳♂️', 'alice'],
- ['🧕', 'alice'],
- ['👮♀️', 'alice'],
- ['👮', 'alice'],
- ['👮♂️', 'alice'],
- ['👷♀️', 'alice'],
- ['👷', 'alice'],
- ['👷♂️', 'alice'],
- ['💂♀️', 'alice'],
- ['💂', 'alice'],
- ['💂♂️', 'alice'],
- ['🕵️♀️', 'alice'],
- ['🕵️', 'alice'],
- ['🕵️♂️', 'alice'],
- ['👩⚕️', 'alice'],
- ['🧑⚕️', 'alice'],
- ['👨⚕️', 'alice'],
- ['👩🌾', 'alice'],
- ['🧑🌾', 'alice'],
- ['👨🌾', 'alice'],
- ['👩🍳', 'alice'],
- ['🧑🍳', 'alice'],
- ['👨🍳', 'alice'],
- ['👩🎓', 'alice'],
- ['🧑🎓', 'alice'],
- ['👨🎓', 'alice'],
- ['👩🎤', 'alice'],
- ['🧑🎤', 'alice'],
- ['👨🎤', 'alice'],
- ['👩🏫', 'alice'],
- ['🧑🏫', 'alice'],
- ['👨🏫', 'alice'],
- ['👩🏭', 'alice'],
- ['🧑🏭', 'alice'],
- ['👨🏭', 'alice'],
- ['👩💻', 'alice'],
- ['🧑💻', 'alice'],
- ['👨💻', 'alice'],
- ['👩💼', 'alice'],
- ['🧑💼', 'alice'],
- ['👨💼', 'alice'],
- ['👩🔧', 'alice'],
- ['🧑🔧', 'alice'],
- ['👨🔧', 'alice'],
- ['👩🔬', 'alice'],
- ['🧑🔬', 'alice'],
- ['👨🔬', 'alice'],
- ['👩🎨', 'alice'],
- ['🧑🎨', 'alice'],
- ['👨🎨', 'alice'],
- ['👩🚒', 'alice'],
- ['🧑🚒', 'alice'],
- ['👨🚒', 'alice'],
- ['👩✈️', 'alice'],
- ['🧑✈️', 'alice'],
- ['👨✈️', 'alice'],
- ['👩🚀', 'alice'],
- ['🧑🚀', 'alice'],
- ['👨🚀', 'alice'],
- ['👩⚖️', 'alice'],
- ['🧑⚖️', 'alice'],
- ['👨⚖️', 'alice'],
- ['👰♀️', 'alice'],
- ['👰', 'alice'],
- ['👰♂️', 'alice'],
- ['🤵♀️', 'alice'],
- ['🤵', 'alice'],
- ['🤵♂️', 'alice'],
- ['👸', 'alice'],
- ['🫅', 'alice'],
- ['🤴', 'alice'],
- ['🥷', 'alice'],
- ['🦸♀️', 'alice'],
- ['🦸', 'alice'],
- ['🦸♂️', 'alice'],
- ['🦹♀️', 'alice'],
- ['🦹', 'alice'],
- ['🦹♂️', 'alice'],
- ['🤶', 'alice'],
- ['🧑🎄', 'alice'],
- ['🎅', 'alice'],
- ['🧙♀️', 'alice'],
- ['🧙', 'alice'],
- ['🧙♂️', 'alice'],
- ['🧝♀️', 'alice'],
- ['🧝', 'alice'],
- ['🧝♂️', 'alice'],
- ['🧛♀️', 'alice'],
- ['🧛', 'alice'],
- ['🧛♂️', 'alice'],
- ['🧟♀️', 'alice'],
- ['🧟', 'alice'],
- ['🧟♂️', 'alice'],
- ['🧞♀️', 'alice'],
- ['🧞', 'alice'],
- ['🧞♂️', 'alice'],
- ['🧜♀️', 'alice'],
- ['🧜', 'alice'],
- ['🧜♂️', 'alice'],
- ['🧚♀️', 'alice'],
- ['🧚', 'alice'],
- ['🧚♂️', 'alice'],
- ['🧌', 'alice'],
- ['👼', 'alice'],
- ['🤰', 'alice'],
- ['🫄', 'alice'],
- ['🫃', 'alice'],
- ['🤱', 'alice'],
- ['👩🍼', 'alice'],
- ['🧑🍼', 'alice'],
- ['👨🍼', 'alice'],
- ['🙇♀️', 'alice'],
- ['🙇', 'alice'],
- ['🙇♂️', 'alice'],
- ['💁♀️', 'alice'],
- ['💁', 'alice'],
- ['💁♂️', 'alice'],
- ['🙅♀️', 'alice'],
- ['🙅', 'alice'],
- ['🙅♂️', 'alice'],
- ['🙆♀️', 'alice'],
- ['🙆', 'alice'],
- ['🙆♂️', 'alice'],
- ['🙋♀️', 'alice'],
- ['🙋', 'alice'],
- ['🙋♂️', 'alice'],
- ['🧏♀️', 'alice'],
- ['🧏', 'alice'],
- ['🧏♂️', 'alice'],
- ['🤦♀️', 'alice'],
- ['🤦', 'alice'],
- ['🤦♂️', 'alice'],
- ['🤷♀️', 'alice'],
- ['🤷', 'alice'],
- ['🤷♂️', 'alice'],
- ['🙎♀️', 'alice'],
- ['🙎', 'alice'],
- ['🙎♂️', 'alice'],
- ['🙍♀️', 'alice'],
- ['🙍', 'alice'],
- ['🙍♂️', 'alice'],
- ['💇♀️', 'alice'],
- ['💇', 'alice'],
- ['💇♂️', 'alice'],
- ['💆♀️', 'alice'],
- ['💆', 'alice'],
- ['💆♂️', 'alice'],
- ['🧖♀️', 'alice'],
- ['🧖', 'alice'],
- ['🧖♂️', 'alice'],
- ['💅', 'alice'],
- ['🤳', 'alice'],
- ['💃', 'alice'],
- ['🕺', 'alice'],
- ['👯♀️', 'alice'],
- ['👯', 'alice'],
- ['👯♂️', 'alice'],
- ['🕴', 'alice'],
- ['👩🦽', 'alice'],
- ['🧑🦽', 'alice'],
- ['👨🦽', 'alice'],
- ['👩🦼', 'alice'],
- ['🧑🦼', 'alice'],
- ['👨🦼', 'alice'],
- ['🚶♀️', 'alice'],
- ['🚶', 'alice'],
- ['🚶♂️', 'alice'],
- ['👩🦯', 'alice'],
- ['🧑🦯', 'alice'],
- ['👨🦯', 'alice'],
- ['🧎♀️', 'alice'],
- ['🧎', 'alice'],
- ['🧎♂️', 'alice'],
- ['🏃♀️', 'alice'],
- ['🏃', 'alice'],
- ['🏃♂️', 'alice'],
- ['🧍♀️', 'alice'],
- ['🧍', 'alice'],
- ['🧍♂️', 'alice'],
- ['👭', 'alice'],
- ['🧑🤝🧑', 'alice'],
- ['👬', 'alice'],
- ['👫', 'alice'],
- ['👩❤️👩', 'alice'],
- ['💑', 'alice'],
- ['👨❤️👨', 'alice'],
- ['👩❤️👨', 'alice'],
- ['👩❤️💋👩', 'alice'],
- ['💏', 'alice'],
- ['👨❤️💋👨', 'alice'],
- ['👩❤️💋👨', 'alice'],
- ['👪', 'alice'],
- ['👨👩👦', 'alice'],
- ['👨👩👧', 'alice'],
- ['👨👩👧👦', 'alice'],
- ['👨👩👦👦', 'alice'],
- ['👨👩👧👧', 'alice'],
- ['👨👨👦', 'alice'],
- ['👨👨👧', 'alice'],
- ['👨👨👧👦', 'alice'],
- ['👨👨👦👦', 'alice'],
- ['👨👨👧👧', 'alice'],
- ['👩👩👦', 'alice'],
- ['👩👩👧', 'alice'],
- ['👩👩👧👦', 'alice'],
- ['👩👩👦👦', 'alice'],
- ['👩👩👧👧', 'alice'],
- ['👨👦', 'alice'],
- ['👨👦👦', 'alice'],
- ['👨👧', 'alice'],
- ['👨👧👦', 'alice'],
- ['👨👧👧', 'alice'],
- ['👩👦', 'alice'],
- ['👩👦👦', 'alice'],
- ['👩👧', 'alice'],
- ['👩👧👦', 'alice'],
- ['👩👧👧', 'alice'],
- ['🗣', 'alice'],
- ['👤', 'alice'],
- ['👥', 'alice'],
- ['🫂', 'alice'],
- ['👋🏽', 'alice'],
- ['🤚🏽', 'alice'],
- ['🖐🏽', 'alice'],
- ['✋🏽', 'alice'],
- ['🖖🏽', 'alice'],
- ['👌🏽', 'alice'],
- ['🤌🏽', 'alice'],
- ['🤏🏽', 'alice'],
- ['✌🏽', 'alice'],
- ['🤞🏽', 'alice'],
- ['🫰🏽', 'alice'],
- ['🤟🏽', 'alice'],
- ['🤘🏽', 'alice'],
- ['🤙🏽', 'alice'],
- ['🫵🏽', 'alice'],
- ['🫱🏽', 'alice'],
- ['🫲🏽', 'alice'],
- ['🫳🏽', 'alice'],
- ['🫴🏽', 'alice'],
- ['👈🏽', 'alice'],
- ['👉🏽', 'alice'],
- ['👆🏽', 'alice'],
- ['🖕🏽', 'alice'],
- ['👇🏽', 'alice'],
- ['☝🏽', 'alice'],
- ['👍🏽', 'alice'],
- ['👎🏽', 'alice'],
- ['✊🏽', 'alice'],
- ['👊🏽', 'alice'],
- ['🤛🏽', 'alice'],
- ['🤜🏽', 'alice'],
- ['👏🏽', 'alice'],
- ['🫶🏽', 'alice'],
- ['🙌🏽', 'alice'],
- ['👐🏽', 'alice'],
- ['🤲🏽', 'alice'],
- ['🙏🏽', 'alice'],
- ['✍🏽', 'alice'],
- ['💅🏽', 'alice'],
- ['🤳🏽', 'alice'],
- ['💪🏽', 'alice'],
- ['🦵🏽', 'alice'],
- ['🦶🏽', 'alice'],
- ['👂🏽', 'alice'],
- ['🦻🏽', 'alice'],
- ['👃🏽', 'alice'],
- ['👶🏽', 'alice'],
- ['👧🏽', 'alice'],
- ['🧒🏽', 'alice'],
- ['👦🏽', 'alice'],
- ['👩🏽', 'alice'],
- ['🧑🏽', 'alice'],
- ['👨🏽', 'alice'],
- ['👩🏽🦱', 'alice'],
- ['🧑🏽🦱', 'alice'],
- ['👨🏽🦱', 'alice'],
- ['👩🏽🦰', 'alice'],
- ['🧑🏽🦰', 'alice'],
- ['👨🏽🦰', 'alice'],
- ['👱🏽♀️', 'alice'],
- ['👱🏽', 'alice'],
- ['👱🏽♂️', 'alice'],
- ['👩🏽🦳', 'alice'],
- ['🧑🏽🦳', 'alice'],
- ['👨🏽🦳', 'alice'],
- ['👩🏽🦲', 'alice'],
- ['🧑🏽🦲', 'alice'],
- ['👨🏽🦲', 'alice'],
- ['🧔🏽♀️', 'alice'],
- ['🧔🏽', 'alice'],
- ['🧔🏽♂️', 'alice'],
- ['👵🏽', 'alice'],
- ['🧓🏽', 'alice'],
- ['👴🏽', 'alice'],
- ['👲🏽', 'alice'],
- ['👳🏽♀️', 'alice'],
- ['👳🏽', 'alice'],
- ['👳🏽♂️', 'alice'],
- ['🧕🏽', 'alice'],
- ['👮🏽♀️', 'alice'],
- ['👮🏽', 'alice'],
- ['👮🏽♂️', 'alice'],
- ['👷🏽♀️', 'alice'],
- ['👷🏽', 'alice'],
- ['👷🏽♂️', 'alice'],
- ['💂🏽♀️', 'alice'],
- ['💂🏽', 'alice'],
- ['💂🏽♂️', 'alice'],
- ['🕵🏽♀️', 'alice'],
- ['🕵🏽', 'alice'],
- ['🕵🏽♂️', 'alice'],
- ['👩🏽⚕️', 'alice'],
- ['🧑🏽⚕️', 'alice'],
- ['👨🏽⚕️', 'alice'],
- ['👩🏽🌾', 'alice'],
- ['🧑🏽🌾', 'alice'],
- ['👨🏽🌾', 'alice'],
- ['👩🏽🍳', 'alice'],
- ['🧑🏽🍳', 'alice'],
- ['👨🏽🍳', 'alice'],
- ['👩🏽🎓', 'alice'],
- ['🧑🏽🎓', 'alice'],
- ['👨🏽🎓', 'alice'],
- ['👩🏽🎤', 'alice'],
- ['🧑🏽🎤', 'alice'],
- ['👨🏽🎤', 'alice'],
- ['👩🏽🏫', 'alice'],
- ['🧑🏽🏫', 'alice'],
- ['👨🏽🏫', 'alice'],
- ['👩🏽🏭', 'alice'],
- ['🧑🏽🏭', 'alice'],
- ['👨🏽🏭', 'alice'],
- ['👩🏽💻', 'alice'],
- ['🧑🏽💻', 'alice'],
- ['👨🏽💻', 'alice'],
- ['👩🏽💼', 'alice'],
- ['🧑🏽💼', 'alice'],
- ['👨🏽💼', 'alice'],
- ['👩🏽🔧', 'alice'],
- ['🧑🏽🔧', 'alice'],
- ['👨🏽🔧', 'alice'],
- ['👩🏽🔬', 'alice'],
- ['🧑🏽🔬', 'alice'],
- ['👨🏽🔬', 'alice'],
- ['👩🏽🎨', 'alice'],
- ['🧑🏽🎨', 'alice'],
- ['👨🏽🎨', 'alice'],
- ['👩🏽🚒', 'alice'],
- ['🧑🏽🚒', 'alice'],
- ['👨🏽🚒', 'alice'],
- ['👩🏽✈️', 'alice'],
- ['🧑🏽✈️', 'alice'],
- ['👨🏽✈️', 'alice'],
- ['👩🏽🚀', 'alice'],
- ['🧑🏽🚀', 'alice'],
- ['👨🏽🚀', 'alice'],
- ['👩🏽⚖️', 'alice'],
- ['🧑🏽⚖️', 'alice'],
- ['👨🏽⚖️', 'alice'],
- ['👰🏽♀️', 'alice'],
- ['👰🏽', 'alice'],
- ['👰🏽♂️', 'alice'],
- ['🤵🏽♀️', 'alice'],
- ['🤵🏽', 'alice'],
- ['🤵🏽♂️', 'alice'],
- ['👸🏽', 'alice'],
- ['🫅🏽', 'alice'],
- ['🤴🏽', 'alice'],
- ['🥷🏽', 'alice'],
- ['🦸🏽♀️', 'alice'],
- ['🦸🏽', 'alice'],
- ['🦸🏽♂️', 'alice'],
- ['🦹🏽♀️', 'alice'],
- ['🦹🏽', 'alice'],
- ['🦹🏽♂️', 'alice'],
- ['🤶🏽', 'alice'],
- ['🧑🏽🎄', 'alice'],
- ['🎅🏽', 'alice'],
- ['🧙🏽♀️', 'alice'],
- ['🧙🏽', 'alice'],
- ['🧙🏽♂️', 'alice'],
- ['🧝🏽♀️', 'alice'],
- ['🧝🏽', 'alice'],
- ['🧝🏽♂️', 'alice'],
- ['🧛🏽♀️', 'alice'],
- ['🧛🏽', 'alice'],
- ['🧛🏽♂️', 'alice'],
- ['🧜🏽♀️', 'alice'],
- ['🧜🏽', 'alice'],
- ['🧜🏽♂️', 'alice'],
- ['🧚🏽♀️', 'alice'],
- ['🧚🏽', 'alice'],
- ['🧚🏽♂️', 'alice'],
- ['👼🏽', 'alice'],
- ['🤰🏽', 'alice'],
- ['🫄🏽', 'alice'],
- ['🫃🏽', 'alice'],
- ['🤱🏽', 'alice'],
- ['👩🏽🍼', 'alice'],
- ['🧑🏽🍼', 'alice'],
- ['👨🏽🍼', 'alice'],
- ['🙇🏽♀️', 'alice'],
- ['🙇🏽', 'alice'],
- ['🙇🏽♂️', 'alice'],
- ['💁🏽♀️', 'alice'],
- ['💁🏽', 'alice'],
- ['💁🏽♂️', 'alice'],
- ['🙅🏽♀️', 'alice'],
- ['🙅🏽', 'alice'],
- ['🙅🏽♂️', 'alice'],
- ['🙆🏽♀️', 'alice'],
- ['🙆🏽', 'alice'],
- ['🙆🏽♂️', 'alice'],
- ['🙋🏽♀️', 'alice'],
- ['🙋🏽', 'alice'],
- ['🙋🏽♂️', 'alice'],
- ['🧏🏽♀️', 'alice'],
- ['🧏🏽', 'alice'],
- ['🧏🏽♂️', 'alice'],
- ['🤦🏽♀️', 'alice'],
- ['🤦🏽', 'alice'],
- ['🤦🏽♂️', 'alice'],
- ['🤷🏽♀️', 'alice'],
- ['🤷🏽', 'alice'],
- ['🤷🏽♂️', 'alice'],
- ['🙎🏽♀️', 'alice'],
- ['🙎🏽', 'alice'],
- ['🙎🏽♂️', 'alice'],
- ['🙍🏽♀️', 'alice'],
- ['🙍🏽', 'alice'],
- ['🙍🏽♂️', 'alice'],
- ['💇🏽♀️', 'alice'],
- ['💇🏽', 'alice'],
- ['💇🏽♂️', 'alice'],
- ['💆🏽♀️', 'alice'],
- ['💆🏽', 'alice'],
- ['💆🏽♂️', 'alice'],
- ['🧖🏽♀️', 'alice'],
- ['🧖🏽', 'alice'],
- ['🧖🏽♂️', 'alice'],
- ['💃🏽', 'alice'],
- ['🕺🏽', 'alice'],
- ['🕴🏽', 'alice'],
- ['👩🏽🦽', 'alice'],
- ['🧑🏽🦽', 'alice'],
- ],
- ],
- ];
- }
- /**
- * @dataProvider providerTestRetrieveAllReactionsWithSpecificReaction
- */
- public function testRetrieveAllReactionsWithSpecificReaction(array $comments, string $reaction, array $expected): void {
- $this->skipIfNotSupport4ByteUTF();
- $manager = $this->getManager();
- $processedComments = $this->proccessComments($comments);
- $comment = reset($processedComments);
- $all = $manager->retrieveAllReactionsWithSpecificReaction($comment->getId(), $reaction);
- $actual = array_map(function ($row) {
- return [
- 'message' => $row->getMessage(),
- 'actorId' => $row->getActorId(),
- ];
- }, $all);
- $this->assertEqualsCanonicalizing($expected, $actual);
- }
- public function providerTestRetrieveAllReactionsWithSpecificReaction(): array {
- return [
- [
- [
- ['message', 'alice', 'comment', null],
- ],
- '👎',
- [],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['👍', 'alice', 'reaction', 'message#alice'],
- ['👍', 'frank', 'reaction', 'message#alice'],
- ],
- '👍',
- [
- ['👍', 'alice'],
- ['👍', 'frank'],
- ],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['👍', 'alice', 'reaction', 'message#alice'],
- ['👎', 'alice', 'reaction', 'message#alice'],
- ['👍', 'frank', 'reaction', 'message#alice'],
- ],
- '👎',
- [
- ['👎', 'alice'],
- ],
- ],
- ];
- }
- /**
- * @dataProvider providerTestGetReactionComment
- */
- public function testGetReactionComment(array $comments, array $expected, bool $notFound): void {
- $this->skipIfNotSupport4ByteUTF();
- $manager = $this->getManager();
- $processedComments = $this->proccessComments($comments);
- $keys = ['message', 'actorId', 'verb', 'parent'];
- $expected = array_combine($keys, $expected);
- if ($notFound) {
- $this->expectException(\OCP\Comments\NotFoundException::class);
- }
- $comment = $processedComments[$expected['message'] . '#' . $expected['actorId']];
- $actual = $manager->getReactionComment($comment->getParentId(), $comment->getActorType(), $comment->getActorId(), $comment->getMessage());
- if (!$notFound) {
- $this->assertEquals($expected['message'], $actual->getMessage());
- $this->assertEquals($expected['actorId'], $actual->getActorId());
- $this->assertEquals($expected['verb'], $actual->getVerb());
- $this->assertEquals($processedComments[$expected['parent']]->getId(), $actual->getParentId());
- }
- }
- public function providerTestGetReactionComment(): array {
- return [
- [
- [
- ['message', 'Matthew', 'comment', null],
- ['👍', 'Matthew', 'reaction', 'message#Matthew'],
- ['👍', 'Mark', 'reaction', 'message#Matthew'],
- ['👍', 'Luke', 'reaction', 'message#Matthew'],
- ['👍', 'John', 'reaction', 'message#Matthew'],
- ],
- ['👍', 'Matthew', 'reaction', 'message#Matthew'],
- false,
- ],
- [
- [
- ['message', 'Matthew', 'comment', null],
- ['👍', 'Matthew', 'reaction', 'message#Matthew'],
- ['👍', 'Mark', 'reaction', 'message#Matthew'],
- ['👍', 'Luke', 'reaction', 'message#Matthew'],
- ['👍', 'John', 'reaction', 'message#Matthew'],
- ],
- ['👍', 'Mark', 'reaction', 'message#Matthew'],
- false,
- ],
- [
- [
- ['message', 'Matthew', 'comment', null],
- ['👎', 'Matthew', 'reaction', 'message#Matthew'],
- ],
- ['👎', 'Matthew', 'reaction', 'message#Matthew'],
- false,
- ],
- [
- [
- ['message', 'Matthew', 'comment', null],
- ['👎', 'Matthew', 'reaction', 'message#Matthew'],
- ['👎', 'Matthew', 'reaction_deleted', 'message#Matthew'],
- ],
- ['👎', 'Matthew', 'reaction', 'message#Matthew'],
- true,
- ],
- ];
- }
- /**
- * @dataProvider providerTestReactionMessageSize
- */
- public function testReactionMessageSize($reactionString, $valid): void {
- $this->skipIfNotSupport4ByteUTF();
- if (!$valid) {
- $this->expectException(\UnexpectedValueException::class);
- }
- $manager = $this->getManager();
- $comment = new Comment();
- $comment->setMessage($reactionString)
- ->setVerb('reaction')
- ->setActor('users', 'alice')
- ->setObject('files', 'file64');
- $status = $manager->save($comment);
- $this->assertTrue($status);
- }
- public function providerTestReactionMessageSize(): array {
- return [
- ['a', false],
- ['1', false],
- ['👍', true],
- ['👍👍', false],
- ['👍🏽', true],
- ['👨🏽💻', true],
- ['👨🏽💻👍', false],
- ];
- }
- /**
- * @dataProvider providerTestReactionsSummarizeOrdered
- */
- public function testReactionsSummarizeOrdered(array $comments, array $expected, bool $isFullMatch): void {
- $this->skipIfNotSupport4ByteUTF();
- $manager = $this->getManager();
- $processedComments = $this->proccessComments($comments);
- $comment = end($processedComments);
- $actual = $manager->get($comment->getParentId());
- if ($isFullMatch) {
- $this->assertSame($expected, $actual->getReactions());
- } else {
- $subResult = array_slice($actual->getReactions(), 0, count($expected));
- $this->assertSame($expected, $subResult);
- }
- }
- public function providerTestReactionsSummarizeOrdered(): array {
- return [
- [
- [
- ['message', 'alice', 'comment', null],
- ['👍', 'alice', 'reaction', 'message#alice'],
- ],
- ['👍' => 1],
- true,
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['👎', 'John', 'reaction', 'message#alice'],
- ['💼', 'Luke', 'reaction', 'message#alice'],
- ['📋', 'Luke', 'reaction', 'message#alice'],
- ['🚀', 'Luke', 'reaction', 'message#alice'],
- ['🖤', 'Luke', 'reaction', 'message#alice'],
- ['😜', 'Luke', 'reaction', 'message#alice'],
- ['🌖', 'Luke', 'reaction', 'message#alice'],
- ['💖', 'Luke', 'reaction', 'message#alice'],
- ['📥', 'Luke', 'reaction', 'message#alice'],
- ['🐉', 'Luke', 'reaction', 'message#alice'],
- ['☕', 'Luke', 'reaction', 'message#alice'],
- ['🐄', 'Luke', 'reaction', 'message#alice'],
- ['🐕', 'Luke', 'reaction', 'message#alice'],
- ['🐈', 'Luke', 'reaction', 'message#alice'],
- ['🛂', 'Luke', 'reaction', 'message#alice'],
- ['🕸', 'Luke', 'reaction', 'message#alice'],
- ['🏰', 'Luke', 'reaction', 'message#alice'],
- ['⚙️', 'Luke', 'reaction', 'message#alice'],
- ['🚨', 'Luke', 'reaction', 'message#alice'],
- ['👥', 'Luke', 'reaction', 'message#alice'],
- ['👍', 'Paul', 'reaction', 'message#alice'],
- ['👍', 'Peter', 'reaction', 'message#alice'],
- ['💜', 'Matthew', 'reaction', 'message#alice'],
- ['💜', 'Mark', 'reaction', 'message#alice'],
- ['💜', 'Luke', 'reaction', 'message#alice'],
- ],
- [
- '💜' => 3,
- '👍' => 2,
- ],
- false,
- ],
- ];
- }
- }
|