12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597 |
- <?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\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,
- );
- }
- public function testGetCommentNotFound() {
- $this->expectException(\OCP\Comments\NotFoundException::class);
- $manager = $this->getManager();
- $manager->get('22');
- }
- public function testGetCommentNotFoundInvalidInput() {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->get('unexisting22');
- }
- public function testGetComment() {
- $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() {
- $this->expectException(\OCP\Comments\NotFoundException::class);
- $manager = $this->getManager();
- $manager->getTree('22');
- }
- public function testGetTreeNotFoundInvalidIpnut() {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->getTree('unexisting22');
- }
- public function testGetTree() {
- $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() {
- $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() {
- $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() {
- $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() {
- $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() {
- $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() {
- $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() {
- 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() {
- $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) {
- $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) {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->create($aType, $aId, $oType, $oId);
- }
- public function testCreateComment() {
- $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() {
- $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() {
- $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() {
- $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() {
- $this->expectException(\UnexpectedValueException::class);
- $manager = $this->getManager();
- $comment = new Comment();
- $comment->setMessage('from no one to nothing');
- $manager->save($comment);
- }
- public function testSaveAsChild() {
- $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) {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->deleteReferencesOfActor($type, $id);
- }
- public function testDeleteReferencesOfActor() {
- $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() {
- $user = \OC::$server->getUserManager()->createUser('xenia', '123456');
- $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) {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->deleteCommentsAtObject($type, $id);
- }
- public function testDeleteCommentsAtObject() {
- $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,
- );
- // 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,
- );
- $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() {
- /** @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() {
- /** @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() {
- /** @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() {
- /** @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() {
- $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() {
- $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() {
- $this->expectException(\OutOfBoundsException::class);
- $manager = $this->getManager();
- $planetClosure = function ($name) {
- return ucfirst($name);
- };
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- }
- public function testRegisterResolverInvalidType() {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $planetClosure = function ($name) {
- return ucfirst($name);
- };
- $manager->registerDisplayNameResolver(1337, $planetClosure);
- }
- public function testResolveDisplayNameUnregisteredType() {
- $this->expectException(\OutOfBoundsException::class);
- $manager = $this->getManager();
- $planetClosure = function ($name) {
- return ucfirst($name);
- };
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- $manager->resolveDisplayName('galaxy', 'sombrero');
- }
- public function testResolveDisplayNameDirtyResolver() {
- $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) {
- $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() {
- $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) {
- $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) {
- $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) {
- $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) {
- $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) {
- $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,
- ],
- ];
- }
- }
|