1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602 |
- <?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\DB\QueryBuilder\IQueryBuilder;
- 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, IQueryBuilder::PARAM_DATETIME_MUTABLE),
- 'latest_child_timestamp' => $qb->createNamedParameter($latestChildDT, IQueryBuilder::PARAM_DATETIME_MUTABLE),
- 'object_type' => $qb->createNamedParameter('files'),
- 'object_id' => $qb->createNamedParameter($objectId),
- 'expire_date' => $qb->createNamedParameter($expireDate, IQueryBuilder::PARAM_DATETIME_MUTABLE),
- '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,
- ],
- ];
- }
- }
|