12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376 |
- <?php
- 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\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;
- protected function setUp(): void {
- parent::setUp();
- $this->connection = \OC::$server->getDatabaseConnection();
- $sql = $this->connection->getDatabasePlatform()->getTruncateTableSQL('`*PREFIX*comments`');
- $this->connection->prepare($sql)->execute();
- $sql = $this->connection->getDatabasePlatform()->getTruncateTableSQL('`*PREFIX*reactions`');
- $this->connection->prepare($sql)->execute();
- }
- protected function addDatabaseEntry($parentId, $topmostParentId, $creationDT = null, $latestChildDT = null, $objectId = null, $expireDate = null) {
- if (is_null($creationDT)) {
- $creationDT = new \DateTime();
- }
- if (is_null($latestChildDT)) {
- $latestChildDT = new \DateTime('yesterday');
- }
- if (is_null($objectId)) {
- $objectId = 'file64';
- }
- $qb = $this->connection->getQueryBuilder();
- $qb
- ->insert('comments')
- ->values([
- 'parent_id' => $qb->createNamedParameter($parentId),
- 'topmost_parent_id' => $qb->createNamedParameter($topmostParentId),
- 'children_count' => $qb->createNamedParameter(2),
- 'actor_type' => $qb->createNamedParameter('users'),
- 'actor_id' => $qb->createNamedParameter('alice'),
- 'message' => $qb->createNamedParameter('nice one'),
- 'verb' => $qb->createNamedParameter('comment'),
- 'creation_timestamp' => $qb->createNamedParameter($creationDT, 'datetime'),
- 'latest_child_timestamp' => $qb->createNamedParameter($latestChildDT, 'datetime'),
- 'object_type' => $qb->createNamedParameter('files'),
- 'object_id' => $qb->createNamedParameter($objectId),
- 'expire_date' => $qb->createNamedParameter($expireDate, 'datetime'),
- ])
- ->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)
- );
- }
- public function testGetCommentNotFound() {
- $this->expectException(\OCP\Comments\NotFoundException::class);
- $manager = $this->getManager();
- $manager->get('22');
- }
- public function testGetCommentNotFoundInvalidInput() {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->get('unexisting22');
- }
- public function testGetComment() {
- $manager = $this->getManager();
- $creationDT = new \DateTime();
- $latestChildDT = new \DateTime('yesterday');
- $qb = \OC::$server->getDatabaseConnection()->getQueryBuilder();
- $qb
- ->insert('comments')
- ->values([
- 'parent_id' => $qb->createNamedParameter('2'),
- 'topmost_parent_id' => $qb->createNamedParameter('1'),
- 'children_count' => $qb->createNamedParameter(2),
- 'actor_type' => $qb->createNamedParameter('users'),
- 'actor_id' => $qb->createNamedParameter('alice'),
- 'message' => $qb->createNamedParameter('nice one'),
- 'verb' => $qb->createNamedParameter('comment'),
- 'creation_timestamp' => $qb->createNamedParameter($creationDT, 'datetime'),
- 'latest_child_timestamp' => $qb->createNamedParameter($latestChildDT, 'datetime'),
- 'object_type' => $qb->createNamedParameter('files'),
- 'object_id' => $qb->createNamedParameter('file64'),
- ])
- ->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);
- }
- public function testGetTreeNotFound() {
- $this->expectException(\OCP\Comments\NotFoundException::class);
- $manager = $this->getManager();
- $manager->getTree('22');
- }
- public function testGetTreeNotFoundInvalidIpnut() {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->getTree('unexisting22');
- }
- public function testGetTree() {
- $headId = $this->addDatabaseEntry(0, 0);
- $this->addDatabaseEntry($headId, $headId, new \DateTime('-3 hours'));
- $this->addDatabaseEntry($headId, $headId, new \DateTime('-2 hours'));
- $id = $this->addDatabaseEntry($headId, $headId, new \DateTime('-1 hour'));
- $manager = $this->getManager();
- $tree = $manager->getTree($headId);
- // Verifying the root comment
- $this->assertTrue(isset($tree['comment']));
- $this->assertTrue($tree['comment'] instanceof IComment);
- $this->assertSame($tree['comment']->getId(), strval($headId));
- $this->assertTrue(isset($tree['replies']));
- $this->assertSame(count($tree['replies']), 3);
- // one level deep
- foreach ($tree['replies'] as $reply) {
- $this->assertTrue($reply['comment'] instanceof IComment);
- $this->assertSame($reply['comment']->getId(), strval($id));
- $this->assertSame(count($reply['replies']), 0);
- $id--;
- }
- }
- public function testGetTreeNoReplies() {
- $id = $this->addDatabaseEntry(0, 0);
- $manager = $this->getManager();
- $tree = $manager->getTree($id);
- // Verifying the root comment
- $this->assertTrue(isset($tree['comment']));
- $this->assertTrue($tree['comment'] instanceof IComment);
- $this->assertSame($tree['comment']->getId(), strval($id));
- $this->assertTrue(isset($tree['replies']));
- $this->assertSame(count($tree['replies']), 0);
- // one level deep
- foreach ($tree['replies'] as $reply) {
- throw new \Exception('This ain`t happen');
- }
- }
- public function testGetTreeWithLimitAndOffset() {
- $headId = $this->addDatabaseEntry(0, 0);
- $this->addDatabaseEntry($headId, $headId, new \DateTime('-3 hours'));
- $this->addDatabaseEntry($headId, $headId, new \DateTime('-2 hours'));
- $this->addDatabaseEntry($headId, $headId, new \DateTime('-1 hour'));
- $idToVerify = $this->addDatabaseEntry($headId, $headId, new \DateTime());
- $manager = $this->getManager();
- for ($offset = 0; $offset < 3; $offset += 2) {
- $tree = $manager->getTree(strval($headId), 2, $offset);
- // Verifying the root comment
- $this->assertTrue(isset($tree['comment']));
- $this->assertTrue($tree['comment'] instanceof IComment);
- $this->assertSame($tree['comment']->getId(), strval($headId));
- $this->assertTrue(isset($tree['replies']));
- $this->assertSame(count($tree['replies']), 2);
- // one level deep
- foreach ($tree['replies'] as $reply) {
- $this->assertTrue($reply['comment'] instanceof IComment);
- $this->assertSame($reply['comment']->getId(), strval($idToVerify));
- $this->assertSame(count($reply['replies']), 0);
- $idToVerify--;
- }
- }
- }
- public function testGetForObject() {
- $this->addDatabaseEntry(0, 0);
- $manager = $this->getManager();
- $comments = $manager->getForObject('files', 'file64');
- $this->assertTrue(is_array($comments));
- $this->assertSame(count($comments), 1);
- $this->assertTrue($comments[0] instanceof IComment);
- $this->assertSame($comments[0]->getMessage(), 'nice one');
- }
- public function testGetForObjectWithLimitAndOffset() {
- $this->addDatabaseEntry(0, 0, new \DateTime('-6 hours'));
- $this->addDatabaseEntry(0, 0, new \DateTime('-5 hours'));
- $this->addDatabaseEntry(1, 1, new \DateTime('-4 hours'));
- $this->addDatabaseEntry(0, 0, new \DateTime('-3 hours'));
- $this->addDatabaseEntry(2, 2, new \DateTime('-2 hours'));
- $this->addDatabaseEntry(2, 2, new \DateTime('-1 hours'));
- $idToVerify = $this->addDatabaseEntry(3, 1, new \DateTime());
- $manager = $this->getManager();
- $offset = 0;
- do {
- $comments = $manager->getForObject('files', 'file64', 3, $offset);
- $this->assertTrue(is_array($comments));
- foreach ($comments as $comment) {
- $this->assertTrue($comment instanceof IComment);
- $this->assertSame($comment->getMessage(), 'nice one');
- $this->assertSame($comment->getId(), strval($idToVerify));
- $idToVerify--;
- }
- $offset += 3;
- } while (count($comments) > 0);
- }
- public function testGetForObjectWithDateTimeConstraint() {
- $this->addDatabaseEntry(0, 0, new \DateTime('-6 hours'));
- $this->addDatabaseEntry(0, 0, new \DateTime('-5 hours'));
- $id1 = $this->addDatabaseEntry(0, 0, new \DateTime('-3 hours'));
- $id2 = $this->addDatabaseEntry(2, 2, new \DateTime('-2 hours'));
- $manager = $this->getManager();
- $comments = $manager->getForObject('files', 'file64', 0, 0, new \DateTime('-4 hours'));
- $this->assertSame(count($comments), 2);
- $this->assertSame($comments[0]->getId(), strval($id2));
- $this->assertSame($comments[1]->getId(), strval($id1));
- }
- public function testGetForObjectWithLimitAndOffsetAndDateTimeConstraint() {
- $this->addDatabaseEntry(0, 0, new \DateTime('-7 hours'));
- $this->addDatabaseEntry(0, 0, new \DateTime('-6 hours'));
- $this->addDatabaseEntry(1, 1, new \DateTime('-5 hours'));
- $this->addDatabaseEntry(0, 0, new \DateTime('-3 hours'));
- $this->addDatabaseEntry(2, 2, new \DateTime('-2 hours'));
- $this->addDatabaseEntry(2, 2, new \DateTime('-1 hours'));
- $idToVerify = $this->addDatabaseEntry(3, 1, new \DateTime());
- $manager = $this->getManager();
- $offset = 0;
- do {
- $comments = $manager->getForObject('files', 'file64', 3, $offset, new \DateTime('-4 hours'));
- $this->assertTrue(is_array($comments));
- foreach ($comments as $comment) {
- $this->assertTrue($comment instanceof IComment);
- $this->assertSame($comment->getMessage(), 'nice one');
- $this->assertSame($comment->getId(), strval($idToVerify));
- $this->assertTrue(intval($comment->getId()) >= 4);
- $idToVerify--;
- }
- $offset += 3;
- } while (count($comments) > 0);
- }
- public function testGetNumberOfCommentsForObject() {
- for ($i = 1; $i < 5; $i++) {
- $this->addDatabaseEntry(0, 0);
- }
- $manager = $this->getManager();
- $amount = $manager->getNumberOfCommentsForObject('untype', '00');
- $this->assertSame($amount, 0);
- $amount = $manager->getNumberOfCommentsForObject('files', 'file64');
- $this->assertSame($amount, 4);
- }
- public function testGetNumberOfUnreadCommentsForFolder() {
- $query = $this->connection->getQueryBuilder();
- $query->insert('filecache')
- ->values([
- 'parent' => $query->createNamedParameter(1000),
- 'size' => $query->createNamedParameter(10),
- 'mtime' => $query->createNamedParameter(10),
- 'storage_mtime' => $query->createNamedParameter(10),
- 'path' => $query->createParameter('path'),
- 'path_hash' => $query->createParameter('path'),
- ]);
- $fileIds = [];
- for ($i = 0; $i < 4; $i++) {
- $query->setParameter('path', 'path_' . $i);
- $query->execute();
- $fileIds[] = $query->getLastInsertId();
- }
- // 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(1000, $user);
- $this->assertEquals([
- $fileIds[0] => 1,
- $fileIds[1] => 2,
- $fileIds[3] => 1,
- ], $amount);
- }
- /**
- * @dataProvider dataGetForObjectSince
- * @param $lastKnown
- * @param $order
- * @param $limit
- * @param $resultFrom
- * @param $resultTo
- */
- public function testGetForObjectSince($lastKnown, $order, $limit, $resultFrom, $resultTo) {
- $ids = [];
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $manager = $this->getManager();
- $comments = $manager->getForObjectSince('files', 'file64', ($lastKnown === null ? 0 : $ids[$lastKnown]), $order, $limit);
- $expected = array_slice($ids, $resultFrom, $resultTo - $resultFrom + 1);
- if ($order === 'desc') {
- $expected = array_reverse($expected);
- }
- $this->assertSame($expected, array_map(function (IComment $c) {
- return (int) $c->getId();
- }, $comments));
- }
- public function dataGetForObjectSince() {
- return [
- [null, 'asc', 20, 0, 4],
- [null, 'asc', 2, 0, 1],
- [null, 'desc', 20, 0, 4],
- [null, 'desc', 2, 3, 4],
- [1, 'asc', 20, 2, 4],
- [1, 'asc', 2, 2, 3],
- [3, 'desc', 20, 0, 2],
- [3, 'desc', 2, 1, 2],
- ];
- }
- public function invalidCreateArgsProvider() {
- return [
- ['', 'aId-1', 'oType-1', 'oId-1'],
- ['aType-1', '', 'oType-1', 'oId-1'],
- ['aType-1', 'aId-1', '', 'oId-1'],
- ['aType-1', 'aId-1', 'oType-1', ''],
- [1, 'aId-1', 'oType-1', 'oId-1'],
- ['aType-1', 1, 'oType-1', 'oId-1'],
- ['aType-1', 'aId-1', 1, 'oId-1'],
- ['aType-1', 'aId-1', 'oType-1', 1],
- ];
- }
- /**
- * @dataProvider invalidCreateArgsProvider
- * @param string $aType
- * @param string $aId
- * @param string $oType
- * @param string $oId
- */
- public function testCreateCommentInvalidArguments($aType, $aId, $oType, $oId) {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->create($aType, $aId, $oType, $oId);
- }
- public function testCreateComment() {
- $actorType = 'bot';
- $actorId = 'bob';
- $objectType = 'weather';
- $objectId = 'bielefeld';
- $comment = $this->getManager()->create($actorType, $actorId, $objectType, $objectId);
- $this->assertTrue($comment instanceof IComment);
- $this->assertSame($comment->getActorType(), $actorType);
- $this->assertSame($comment->getActorId(), $actorId);
- $this->assertSame($comment->getObjectType(), $objectType);
- $this->assertSame($comment->getObjectId(), $objectId);
- }
- public function testDelete() {
- $this->expectException(\OCP\Comments\NotFoundException::class);
- $manager = $this->getManager();
- $done = $manager->delete('404');
- $this->assertFalse($done);
- $done = $manager->delete('%');
- $this->assertFalse($done);
- $done = $manager->delete('');
- $this->assertFalse($done);
- $id = strval($this->addDatabaseEntry(0, 0));
- $comment = $manager->get($id);
- $this->assertTrue($comment instanceof IComment);
- $done = $manager->delete($id);
- $this->assertTrue($done);
- $manager->get($id);
- }
- /**
- * @dataProvider providerTestSave
- */
- public function testSave(string $message, string $actorId, string $verb, ?string $parentId, ?string $id = ''): IComment {
- $manager = $this->getManager();
- $comment = new Comment();
- $comment
- ->setId($id)
- ->setActor('users', $actorId)
- ->setObject('files', 'file64')
- ->setMessage($message)
- ->setVerb($verb);
- if ($parentId) {
- $comment->setParentId($parentId);
- }
- $saveSuccessful = $manager->save($comment);
- $this->assertTrue($saveSuccessful);
- $this->assertTrue($comment->getId() !== '');
- $this->assertTrue($comment->getId() !== '0');
- $this->assertTrue(!is_null($comment->getCreationDateTime()));
- $loadedComment = $manager->get($comment->getId());
- $this->assertSame($comment->getMessage(), $loadedComment->getMessage());
- $this->assertEquals($comment->getCreationDateTime()->getTimestamp(), $loadedComment->getCreationDateTime()->getTimestamp());
- return $comment;
- }
- public function providerTestSave(): array {
- return [
- ['very beautiful, I am impressed!', 'alice', 'comment', null]
- ];
- }
- public function testSaveUpdate() {
- $manager = $this->getManager();
- $comment = new Comment();
- $comment
- ->setActor('users', 'alice')
- ->setObject('files', 'file64')
- ->setMessage('very beautiful, I am impressed!')
- ->setVerb('comment')
- ->setExpireDate(new \DateTime('+2 hours'));
- $manager->save($comment);
- $loadedComment = $manager->get($comment->getId());
- // Compare current object with database values
- $this->assertSame($comment->getMessage(), $loadedComment->getMessage());
- $this->assertSame(
- $comment->getExpireDate()->format('Y-m-d H:i:s'),
- $loadedComment->getExpireDate()->format('Y-m-d H:i:s')
- );
- // Preserve the original comment to compare after update
- $original = clone $comment;
- // Update values
- $comment->setMessage('very beautiful, I am really so much impressed!')
- ->setExpireDate(new \DateTime('+1 hours'));
- $manager->save($comment);
- $loadedComment = $manager->get($comment->getId());
- // Compare current object with database values
- $this->assertSame($comment->getMessage(), $loadedComment->getMessage());
- $this->assertSame(
- $comment->getExpireDate()->format('Y-m-d H:i:s'),
- $loadedComment->getExpireDate()->format('Y-m-d H:i:s')
- );
- // Compare original object with database values
- $this->assertNotSame($original->getMessage(), $loadedComment->getMessage());
- $this->assertNotSame(
- $original->getExpireDate()->format('Y-m-d H:i:s'),
- $loadedComment->getExpireDate()->format('Y-m-d H:i:s')
- );
- }
- public function testSaveUpdateException() {
- $this->expectException(\OCP\Comments\NotFoundException::class);
- $manager = $this->getManager();
- $comment = new Comment();
- $comment
- ->setActor('users', 'alice')
- ->setObject('files', 'file64')
- ->setMessage('very beautiful, I am impressed!')
- ->setVerb('comment');
- $manager->save($comment);
- $manager->delete($comment->getId());
- $comment->setMessage('very beautiful, I am really so much impressed!');
- $manager->save($comment);
- }
- public function testSaveIncomplete() {
- $this->expectException(\UnexpectedValueException::class);
- $manager = $this->getManager();
- $comment = new Comment();
- $comment->setMessage('from no one to nothing');
- $manager->save($comment);
- }
- public function testSaveAsChild() {
- $id = $this->addDatabaseEntry(0, 0);
- $manager = $this->getManager();
- for ($i = 0; $i < 3; $i++) {
- $comment = new Comment();
- $comment
- ->setActor('users', 'alice')
- ->setObject('files', 'file64')
- ->setParentId(strval($id))
- ->setMessage('full ack')
- ->setVerb('comment')
- // setting the creation time avoids using sleep() while making sure to test with different timestamps
- ->setCreationDateTime(new \DateTime('+' . $i . ' minutes'));
- $manager->save($comment);
- $this->assertSame($comment->getTopmostParentId(), strval($id));
- $parentComment = $manager->get(strval($id));
- $this->assertSame($parentComment->getChildrenCount(), $i + 1);
- $this->assertEquals($parentComment->getLatestChildDateTime()->getTimestamp(), $comment->getCreationDateTime()->getTimestamp());
- }
- }
- public function invalidActorArgsProvider() {
- return
- [
- ['', ''],
- [1, 'alice'],
- ['users', 1],
- ];
- }
- /**
- * @dataProvider invalidActorArgsProvider
- * @param string $type
- * @param string $id
- */
- public function testDeleteReferencesOfActorInvalidInput($type, $id) {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->deleteReferencesOfActor($type, $id);
- }
- public function testDeleteReferencesOfActor() {
- $ids = [];
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $manager = $this->getManager();
- // just to make sure they are really set, with correct actor data
- $comment = $manager->get(strval($ids[1]));
- $this->assertSame($comment->getActorType(), 'users');
- $this->assertSame($comment->getActorId(), 'alice');
- $wasSuccessful = $manager->deleteReferencesOfActor('users', 'alice');
- $this->assertTrue($wasSuccessful);
- foreach ($ids as $id) {
- $comment = $manager->get(strval($id));
- $this->assertSame($comment->getActorType(), ICommentsManager::DELETED_USER);
- $this->assertSame($comment->getActorId(), ICommentsManager::DELETED_USER);
- }
- // actor info is gone from DB, but when database interaction is alright,
- // we still expect to get true back
- $wasSuccessful = $manager->deleteReferencesOfActor('users', 'alice');
- $this->assertTrue($wasSuccessful);
- }
- public function testDeleteReferencesOfActorWithUserManagement() {
- $user = \OC::$server->getUserManager()->createUser('xenia', '123456');
- $this->assertTrue($user instanceof IUser);
- $manager = \OC::$server->getCommentsManager();
- $comment = $manager->create('users', $user->getUID(), 'files', 'file64');
- $comment
- ->setMessage('Most important comment I ever left on the Internet.')
- ->setVerb('comment');
- $status = $manager->save($comment);
- $this->assertTrue($status);
- $commentID = $comment->getId();
- $user->delete();
- $comment = $manager->get($commentID);
- $this->assertSame($comment->getActorType(), ICommentsManager::DELETED_USER);
- $this->assertSame($comment->getActorId(), ICommentsManager::DELETED_USER);
- }
- public function invalidObjectArgsProvider() {
- return
- [
- ['', ''],
- [1, 'file64'],
- ['files', 1],
- ];
- }
- /**
- * @dataProvider invalidObjectArgsProvider
- * @param string $type
- * @param string $id
- */
- public function testDeleteCommentsAtObjectInvalidInput($type, $id) {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $manager->deleteCommentsAtObject($type, $id);
- }
- public function testDeleteCommentsAtObject() {
- $ids = [];
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $ids[] = $this->addDatabaseEntry(0, 0);
- $manager = $this->getManager();
- // just to make sure they are really set, with correct actor data
- $comment = $manager->get(strval($ids[1]));
- $this->assertSame($comment->getObjectType(), 'files');
- $this->assertSame($comment->getObjectId(), 'file64');
- $wasSuccessful = $manager->deleteCommentsAtObject('files', 'file64');
- $this->assertTrue($wasSuccessful);
- $verified = 0;
- foreach ($ids as $id) {
- try {
- $manager->get(strval($id));
- } catch (NotFoundException $e) {
- $verified++;
- }
- }
- $this->assertSame($verified, 3);
- // actor info is gone from DB, but when database interaction is alright,
- // we still expect to get true back
- $wasSuccessful = $manager->deleteCommentsAtObject('files', 'file64');
- $this->assertTrue($wasSuccessful);
- }
- public function testDeleteCommentsExpiredAtObjectTypeAndId(): void {
- $ids = [];
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, null, new \DateTime('+2 hours'));
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, null, new \DateTime('+2 hours'));
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, null, new \DateTime('+2 hours'));
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, null, new \DateTime('-2 hours'));
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, null, new \DateTime('-2 hours'));
- $ids[] = $this->addDatabaseEntry(0, 0, null, null, null, new \DateTime('-2 hours'));
- $manager = new Manager(
- $this->connection,
- $this->createMock(LoggerInterface::class),
- $this->createMock(IConfig::class),
- Server::get(ITimeFactory::class),
- new EmojiHelper($this->connection),
- $this->createMock(IInitialStateService::class)
- );
- // 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)
- );
- $deleted = $manager->deleteCommentsExpiredAtObject('files');
- $this->assertTrue($deleted);
- $deleted = 0;
- $exists = 0;
- foreach ($ids as $id) {
- try {
- $manager->get((string) $id);
- $exists++;
- } catch (NotFoundException $e) {
- $deleted++;
- }
- }
- $this->assertSame($exists, 0);
- $this->assertSame($deleted, 6);
- // actor info is gone from DB, but when database interaction is alright,
- // we still expect to get true back
- $deleted = $manager->deleteCommentsExpiredAtObject('files');
- $this->assertFalse($deleted);
- }
- public function testSetMarkRead() {
- /** @var IUser|\PHPUnit\Framework\MockObject\MockObject $user */
- $user = $this->createMock(IUser::class);
- $user->expects($this->any())
- ->method('getUID')
- ->willReturn('alice');
- $dateTimeSet = new \DateTime();
- $manager = $this->getManager();
- $manager->setReadMark('robot', '36', $dateTimeSet, $user);
- $dateTimeGet = $manager->getReadMark('robot', '36', $user);
- $this->assertEquals($dateTimeGet->getTimestamp(), $dateTimeSet->getTimestamp());
- }
- public function testSetMarkReadUpdate() {
- /** @var IUser|\PHPUnit\Framework\MockObject\MockObject $user */
- $user = $this->createMock(IUser::class);
- $user->expects($this->any())
- ->method('getUID')
- ->willReturn('alice');
- $dateTimeSet = new \DateTime('yesterday');
- $manager = $this->getManager();
- $manager->setReadMark('robot', '36', $dateTimeSet, $user);
- $dateTimeSet = new \DateTime('today');
- $manager->setReadMark('robot', '36', $dateTimeSet, $user);
- $dateTimeGet = $manager->getReadMark('robot', '36', $user);
- $this->assertEquals($dateTimeGet, $dateTimeSet);
- }
- public function testReadMarkDeleteUser() {
- /** @var IUser|\PHPUnit\Framework\MockObject\MockObject $user */
- $user = $this->createMock(IUser::class);
- $user->expects($this->any())
- ->method('getUID')
- ->willReturn('alice');
- $dateTimeSet = new \DateTime();
- $manager = $this->getManager();
- $manager->setReadMark('robot', '36', $dateTimeSet, $user);
- $manager->deleteReadMarksFromUser($user);
- $dateTimeGet = $manager->getReadMark('robot', '36', $user);
- $this->assertNull($dateTimeGet);
- }
- public function testReadMarkDeleteObject() {
- /** @var IUser|\PHPUnit\Framework\MockObject\MockObject $user */
- $user = $this->createMock(IUser::class);
- $user->expects($this->any())
- ->method('getUID')
- ->willReturn('alice');
- $dateTimeSet = new \DateTime();
- $manager = $this->getManager();
- $manager->setReadMark('robot', '36', $dateTimeSet, $user);
- $manager->deleteReadMarksOnObject('robot', '36');
- $dateTimeGet = $manager->getReadMark('robot', '36', $user);
- $this->assertNull($dateTimeGet);
- }
- public function testSendEvent() {
- $handler1 = $this->getMockBuilder(ICommentsEventHandler::class)->getMock();
- $handler1->expects($this->exactly(4))
- ->method('handle');
- $handler2 = $this->getMockBuilder(ICommentsEventHandler::class)->getMock();
- $handler1->expects($this->exactly(4))
- ->method('handle');
- $manager = $this->getManager();
- $manager->registerEventHandler(function () use ($handler1) {
- return $handler1;
- });
- $manager->registerEventHandler(function () use ($handler2) {
- return $handler2;
- });
- $comment = new Comment();
- $comment
- ->setActor('users', 'alice')
- ->setObject('files', 'file64')
- ->setMessage('very beautiful, I am impressed!')
- ->setVerb('comment');
- // Add event
- $manager->save($comment);
- // Update event
- $comment->setMessage('Different topic');
- $manager->save($comment);
- // Delete event
- $manager->delete($comment->getId());
- }
- public function testResolveDisplayName() {
- $manager = $this->getManager();
- $planetClosure = function ($name) {
- return ucfirst($name);
- };
- $galaxyClosure = function ($name) {
- return strtoupper($name);
- };
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- $manager->registerDisplayNameResolver('galaxy', $galaxyClosure);
- $this->assertSame('Neptune', $manager->resolveDisplayName('planet', 'neptune'));
- $this->assertSame('SOMBRERO', $manager->resolveDisplayName('galaxy', 'sombrero'));
- }
- public function testRegisterResolverDuplicate() {
- $this->expectException(\OutOfBoundsException::class);
- $manager = $this->getManager();
- $planetClosure = function ($name) {
- return ucfirst($name);
- };
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- }
- public function testRegisterResolverInvalidType() {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $planetClosure = function ($name) {
- return ucfirst($name);
- };
- $manager->registerDisplayNameResolver(1337, $planetClosure);
- }
- public function testResolveDisplayNameUnregisteredType() {
- $this->expectException(\OutOfBoundsException::class);
- $manager = $this->getManager();
- $planetClosure = function ($name) {
- return ucfirst($name);
- };
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- $manager->resolveDisplayName('galaxy', 'sombrero');
- }
- public function testResolveDisplayNameDirtyResolver() {
- $manager = $this->getManager();
- $planetClosure = function () {
- return null;
- };
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- $this->assertTrue(is_string($manager->resolveDisplayName('planet', 'neptune')));
- }
- private function skipIfNotSupport4ByteUTF() {
- if (!$this->getManager()->supportReactions()) {
- $this->markTestSkipped('MySQL doesn\'t support 4 byte UTF-8');
- }
- }
- /**
- * @dataProvider providerTestReactionAddAndDelete
- *
- * @param IComment[] $comments
- * @param array $reactionsExpected
- * @return void
- */
- public function testReactionAddAndDelete(array $comments, array $reactionsExpected) {
- $this->skipIfNotSupport4ByteUTF();
- $manager = $this->getManager();
- $processedComments = $this->proccessComments($comments);
- $comment = end($processedComments);
- if ($comment->getParentId()) {
- $parent = $manager->get($comment->getParentId());
- $this->assertEqualsCanonicalizing($reactionsExpected, $parent->getReactions());
- }
- }
- public function providerTestReactionAddAndDelete(): array {
- return[
- [
- [
- ['message', 'alice', 'comment', null],
- ], [],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['๐', 'alice', 'reaction', 'message#alice'],
- ], ['๐' => 1],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['๐', 'alice', 'reaction', 'message#alice'],
- ['๐', 'alice', 'reaction', 'message#alice'],
- ], ['๐' => 1],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['๐', 'alice', 'reaction', 'message#alice'],
- ['๐', 'frank', 'reaction', 'message#alice'],
- ], ['๐' => 2],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['๐', 'alice', 'reaction', 'message#alice'],
- ['๐', 'frank', 'reaction', 'message#alice'],
- ['๐', 'frank', 'reaction_deleted', 'message#alice'],
- ], ['๐' => 1],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['๐', 'alice', 'reaction', 'message#alice'],
- ['๐', 'frank', 'reaction', 'message#alice'],
- ['๐', 'alice', 'reaction_deleted', 'message#alice'],
- ['๐', 'frank', 'reaction_deleted', 'message#alice'],
- ], [],
- ],
- ];
- }
- public function testResolveDisplayNameInvalidType() {
- $this->expectException(\InvalidArgumentException::class);
- $manager = $this->getManager();
- $planetClosure = function () {
- return null;
- };
- $manager->registerDisplayNameResolver('planet', $planetClosure);
- $this->assertTrue(is_string($manager->resolveDisplayName(1337, 'neptune')));
- }
- /**
- * @param array $data
- * @return IComment[]
- */
- private function proccessComments(array $data): array {
- /** @var IComment[] */
- $comments = [];
- foreach ($data as $comment) {
- [$message, $actorId, $verb, $parentText] = $comment;
- $parentId = null;
- if ($parentText) {
- $parentId = (string) $comments[$parentText]->getId();
- }
- $id = '';
- if ($verb === 'reaction_deleted') {
- $id = $comments[$message . '#' . $actorId]->getId();
- }
- $comment = $this->testSave($message, $actorId, $verb, $parentId, $id);
- $comments[$comment->getMessage() . '#' . $comment->getActorId()] = $comment;
- }
- return $comments;
- }
- /**
- * @dataProvider providerTestRetrieveAllReactions
- */
- public function testRetrieveAllReactions(array $comments, array $expected) {
- $this->skipIfNotSupport4ByteUTF();
- $manager = $this->getManager();
- $processedComments = $this->proccessComments($comments);
- $comment = reset($processedComments);
- $all = $manager->retrieveAllReactions($comment->getId());
- $actual = array_map(function ($row) {
- return [
- 'message' => $row->getMessage(),
- 'actorId' => $row->getActorId(),
- ];
- }, $all);
- $this->assertEqualsCanonicalizing($expected, $actual);
- }
- public function providerTestRetrieveAllReactions(): array {
- return [
- [
- [
- ['message', 'alice', 'comment', null],
- ],
- [],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['๐', 'alice', 'reaction', 'message#alice'],
- ['๐', 'frank', 'reaction', 'message#alice'],
- ],
- [
- ['๐', 'alice'],
- ['๐', 'frank'],
- ],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['๐', 'alice', 'reaction', 'message#alice'],
- ['๐', 'alice', 'reaction', 'message#alice'],
- ['๐', 'frank', 'reaction', 'message#alice'],
- ],
- [
- ['๐', 'alice'],
- ['๐', 'frank'],
- ],
- ],
- ];
- }
- /**
- * @dataProvider providerTestRetrieveAllReactionsWithSpecificReaction
- */
- public function testRetrieveAllReactionsWithSpecificReaction(array $comments, string $reaction, array $expected) {
- $this->skipIfNotSupport4ByteUTF();
- $manager = $this->getManager();
- $processedComments = $this->proccessComments($comments);
- $comment = reset($processedComments);
- $all = $manager->retrieveAllReactionsWithSpecificReaction($comment->getId(), $reaction);
- $actual = array_map(function ($row) {
- return [
- 'message' => $row->getMessage(),
- 'actorId' => $row->getActorId(),
- ];
- }, $all);
- $this->assertEqualsCanonicalizing($expected, $actual);
- }
- public function providerTestRetrieveAllReactionsWithSpecificReaction(): array {
- return [
- [
- [
- ['message', 'alice', 'comment', null],
- ],
- '๐',
- [],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['๐', 'alice', 'reaction', 'message#alice'],
- ['๐', 'frank', 'reaction', 'message#alice'],
- ],
- '๐',
- [
- ['๐', 'alice'],
- ['๐', 'frank'],
- ],
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['๐', 'alice', 'reaction', 'message#alice'],
- ['๐', 'alice', 'reaction', 'message#alice'],
- ['๐', 'frank', 'reaction', 'message#alice'],
- ],
- '๐',
- [
- ['๐', 'alice'],
- ],
- ],
- ];
- }
- /**
- * @dataProvider providerTestGetReactionComment
- */
- public function testGetReactionComment(array $comments, array $expected, bool $notFound) {
- $this->skipIfNotSupport4ByteUTF();
- $manager = $this->getManager();
- $processedComments = $this->proccessComments($comments);
- $keys = ['message', 'actorId', 'verb', 'parent'];
- $expected = array_combine($keys, $expected);
- if ($notFound) {
- $this->expectException(\OCP\Comments\NotFoundException::class);
- }
- $comment = $processedComments[$expected['message'] . '#' . $expected['actorId']];
- $actual = $manager->getReactionComment($comment->getParentId(), $comment->getActorType(), $comment->getActorId(), $comment->getMessage());
- if (!$notFound) {
- $this->assertEquals($expected['message'], $actual->getMessage());
- $this->assertEquals($expected['actorId'], $actual->getActorId());
- $this->assertEquals($expected['verb'], $actual->getVerb());
- $this->assertEquals($processedComments[$expected['parent']]->getId(), $actual->getParentId());
- }
- }
- public function providerTestGetReactionComment(): array {
- return [
- [
- [
- ['message', 'Matthew', 'comment', null],
- ['๐', 'Matthew', 'reaction', 'message#Matthew'],
- ['๐', 'Mark', 'reaction', 'message#Matthew'],
- ['๐', 'Luke', 'reaction', 'message#Matthew'],
- ['๐', 'John', 'reaction', 'message#Matthew'],
- ],
- ['๐', 'Matthew', 'reaction', 'message#Matthew'],
- false,
- ],
- [
- [
- ['message', 'Matthew', 'comment', null],
- ['๐', 'Matthew', 'reaction', 'message#Matthew'],
- ['๐', 'Mark', 'reaction', 'message#Matthew'],
- ['๐', 'Luke', 'reaction', 'message#Matthew'],
- ['๐', 'John', 'reaction', 'message#Matthew'],
- ],
- ['๐', 'Mark', 'reaction', 'message#Matthew'],
- false,
- ],
- [
- [
- ['message', 'Matthew', 'comment', null],
- ['๐', 'Matthew', 'reaction', 'message#Matthew'],
- ],
- ['๐', 'Matthew', 'reaction', 'message#Matthew'],
- false,
- ],
- [
- [
- ['message', 'Matthew', 'comment', null],
- ['๐', 'Matthew', 'reaction', 'message#Matthew'],
- ['๐', 'Matthew', 'reaction_deleted', 'message#Matthew'],
- ],
- ['๐', 'Matthew', 'reaction', 'message#Matthew'],
- true,
- ],
- ];
- }
- /**
- * @dataProvider providerTestReactionMessageSize
- */
- public function testReactionMessageSize($reactionString, $valid) {
- $this->skipIfNotSupport4ByteUTF();
- if (!$valid) {
- $this->expectException(\UnexpectedValueException::class);
- }
- $manager = $this->getManager();
- $comment = new Comment();
- $comment->setMessage($reactionString)
- ->setVerb('reaction')
- ->setActor('users', 'alice')
- ->setObject('files', 'file64');
- $status = $manager->save($comment);
- $this->assertTrue($status);
- }
- public function providerTestReactionMessageSize(): array {
- return [
- ['a', false],
- ['1', false],
- ['๐', true],
- ['๐๐', false],
- ['๐๐ฝ', true],
- ['๐จ๐ฝโ๐ป', true],
- ['๐จ๐ฝโ๐ป๐', false],
- ];
- }
- /**
- * @dataProvider providerTestReactionsSummarizeOrdered
- */
- public function testReactionsSummarizeOrdered(array $comments, array $expected, bool $isFullMatch) {
- $this->skipIfNotSupport4ByteUTF();
- $manager = $this->getManager();
- $processedComments = $this->proccessComments($comments);
- $comment = end($processedComments);
- $actual = $manager->get($comment->getParentId());
- if ($isFullMatch) {
- $this->assertSame($expected, $actual->getReactions());
- } else {
- $subResult = array_slice($actual->getReactions(), 0, count($expected));
- $this->assertSame($expected, $subResult);
- }
- }
- public function providerTestReactionsSummarizeOrdered(): array {
- return [
- [
- [
- ['message', 'alice', 'comment', null],
- ['๐', 'alice', 'reaction', 'message#alice'],
- ],
- ['๐' => 1],
- true,
- ],
- [
- [
- ['message', 'alice', 'comment', null],
- ['๐', 'John', 'reaction', 'message#alice'],
- ['๐ผ', 'Luke', 'reaction', 'message#alice'],
- ['๐', 'Luke', 'reaction', 'message#alice'],
- ['๐', 'Luke', 'reaction', 'message#alice'],
- ['๐ค', 'Luke', 'reaction', 'message#alice'],
- ['๐', 'Luke', 'reaction', 'message#alice'],
- ['๐', 'Luke', 'reaction', 'message#alice'],
- ['๐', 'Luke', 'reaction', 'message#alice'],
- ['๐ฅ', 'Luke', 'reaction', 'message#alice'],
- ['๐', 'Luke', 'reaction', 'message#alice'],
- ['โ', 'Luke', 'reaction', 'message#alice'],
- ['๐', 'Luke', 'reaction', 'message#alice'],
- ['๐', 'Luke', 'reaction', 'message#alice'],
- ['๐', 'Luke', 'reaction', 'message#alice'],
- ['๐', 'Luke', 'reaction', 'message#alice'],
- ['๐ธ', 'Luke', 'reaction', 'message#alice'],
- ['๐ฐ', 'Luke', 'reaction', 'message#alice'],
- ['โ๏ธ', 'Luke', 'reaction', 'message#alice'],
- ['๐จ', 'Luke', 'reaction', 'message#alice'],
- ['๐ฅ', 'Luke', 'reaction', 'message#alice'],
- ['๐', 'Paul', 'reaction', 'message#alice'],
- ['๐', 'Peter', 'reaction', 'message#alice'],
- ['๐', 'Matthew', 'reaction', 'message#alice'],
- ['๐', 'Mark', 'reaction', 'message#alice'],
- ['๐', 'Luke', 'reaction', 'message#alice'],
- ],
- [
- '๐' => 3,
- '๐' => 2,
- ],
- false,
- ],
- ];
- }
- }
|