* * @copyright Copyright (c) 2016, ownCloud, Inc. * @license AGPL-3.0 * * This code is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License, version 3, * as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License, version 3, * along with this program. If not, see * */ namespace Test\Accounts; use OC\Accounts\Account; use OC\Accounts\AccountManager; use OCA\Settings\BackgroundJobs\VerifyUserData; use OCP\Accounts\IAccountManager; use OCP\BackgroundJob\IJobList; use OCP\Defaults; use OCP\IConfig; use OCP\IDBConnection; use OCP\IURLGenerator; use OCP\IUser; use OCP\L10N\IFactory; use OCP\Mail\IMailer; use OCP\Security\ICrypto; use OCP\Security\VerificationToken\IVerificationToken; use PHPUnit\Framework\MockObject\MockObject; use Psr\Log\LoggerInterface; use Symfony\Component\EventDispatcher\EventDispatcherInterface; use Symfony\Component\EventDispatcher\GenericEvent; use Test\TestCase; /** * Class AccountManagerTest * * @group DB * @package Test\Accounts */ class AccountManagerTest extends TestCase { /** @var IVerificationToken|MockObject */ protected $verificationToken; /** @var IMailer|MockObject */ protected $mailer; /** @var ICrypto|MockObject */ protected $crypto; /** @var IURLGenerator|MockObject */ protected $urlGenerator; /** @var Defaults|MockObject */ protected $defaults; /** @var IFactory|MockObject */ protected $l10nFactory; /** @var \OCP\IDBConnection */ private $connection; /** @var IConfig|MockObject */ private $config; /** @var EventDispatcherInterface|MockObject */ private $eventDispatcher; /** @var IJobList|MockObject */ private $jobList; /** @var string accounts table name */ private $table = 'accounts'; /** @var LoggerInterface|MockObject */ private $logger; /** @var AccountManager */ private $accountManager; protected function setUp(): void { parent::setUp(); $this->eventDispatcher = $this->createMock(EventDispatcherInterface::class); $this->connection = \OC::$server->get(IDBConnection::class); $this->config = $this->createMock(IConfig::class); $this->jobList = $this->createMock(IJobList::class); $this->logger = $this->createMock(LoggerInterface::class); $this->verificationToken = $this->createMock(IVerificationToken::class); $this->mailer = $this->createMock(IMailer::class); $this->defaults = $this->createMock(Defaults::class); $this->l10nFactory = $this->createMock(IFactory::class); $this->urlGenerator = $this->createMock(IURLGenerator::class); $this->crypto = $this->createMock(ICrypto::class); $this->accountManager = new AccountManager( $this->connection, $this->config, $this->eventDispatcher, $this->jobList, $this->logger, $this->verificationToken, $this->mailer, $this->defaults, $this->l10nFactory, $this->urlGenerator, $this->crypto ); } protected function tearDown(): void { parent::tearDown(); $query = $this->connection->getQueryBuilder(); $query->delete($this->table)->execute(); } protected function makeUser(string $uid, string $name, string $email = null): IUser { $user = $this->createMock(IUser::class); $user->expects($this->any()) ->method('getUid') ->willReturn($uid); $user->expects($this->any()) ->method('getDisplayName') ->willReturn($name); if ($email !== null) { $user->expects($this->any()) ->method('getEMailAddress') ->willReturn($email); } return $user; } protected function populateOrUpdate(): void { $users = [ [ 'user' => $this->makeUser('j.doe', 'Jane Doe', 'jane.doe@acme.com'), 'data' => [ [ 'name' => IAccountManager::PROPERTY_DISPLAYNAME, 'value' => 'Jane Doe', 'scope' => IAccountManager::SCOPE_PUBLISHED ], [ 'name' => IAccountManager::PROPERTY_EMAIL, 'value' => 'jane.doe@acme.com', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_TWITTER, 'value' => '@sometwitter', 'scope' => IAccountManager::SCOPE_PUBLISHED ], [ 'name' => IAccountManager::PROPERTY_PHONE, 'value' => '+491601231212', 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_ADDRESS, 'value' => 'some street', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_WEBSITE, 'value' => 'https://acme.com', 'scope' => IAccountManager::SCOPE_PRIVATE ], [ 'name' => IAccountManager::PROPERTY_ORGANISATION, 'value' => 'Some organisation', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_ROLE, 'value' => 'Human', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_HEADLINE, 'value' => 'Hi', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_BIOGRAPHY, 'value' => 'Biography', 'scope' => IAccountManager::SCOPE_LOCAL ], ], ], [ 'user' => $this->makeUser('a.allison', 'Alice Allison', 'a.allison@example.org'), 'data' => [ [ 'name' => IAccountManager::PROPERTY_DISPLAYNAME, 'value' => 'Alice Allison', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_EMAIL, 'value' => 'a.allison@example.org', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_TWITTER, 'value' => '@a_alice', 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_PHONE, 'value' => '+491602312121', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_ADDRESS, 'value' => 'Dundee Road 45', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_WEBSITE, 'value' => 'https://example.org', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_ORGANISATION, 'value' => 'Another organisation', 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_ROLE, 'value' => 'Alien', 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_HEADLINE, 'value' => 'Hello', 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_BIOGRAPHY, 'value' => 'Different biography', 'scope' => IAccountManager::SCOPE_FEDERATED ], ], ], [ 'user' => $this->makeUser('b32c5a5b-1084-4380-8856-e5223b16de9f', 'Armel Oliseh', 'oliseh@example.com'), 'data' => [ [ 'name' => IAccountManager::PROPERTY_DISPLAYNAME, 'value' => 'Armel Oliseh', 'scope' => IAccountManager::SCOPE_PUBLISHED ], [ 'name' => IAccountManager::PROPERTY_EMAIL, 'value' => 'oliseh@example.com', 'scope' => IAccountManager::SCOPE_PUBLISHED ], [ 'name' => IAccountManager::PROPERTY_TWITTER, 'value' => '', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_PHONE, 'value' => '+491603121212', 'scope' => IAccountManager::SCOPE_PUBLISHED ], [ 'name' => IAccountManager::PROPERTY_ADDRESS, 'value' => 'Sunflower Blvd. 77', 'scope' => IAccountManager::SCOPE_PUBLISHED ], [ 'name' => IAccountManager::PROPERTY_WEBSITE, 'value' => 'https://example.com', 'scope' => IAccountManager::SCOPE_PUBLISHED ], [ 'name' => IAccountManager::PROPERTY_ORGANISATION, 'value' => 'Yet another organisation', 'scope' => IAccountManager::SCOPE_PUBLISHED ], [ 'name' => IAccountManager::PROPERTY_ROLE, 'value' => 'Being', 'scope' => IAccountManager::SCOPE_PUBLISHED ], [ 'name' => IAccountManager::PROPERTY_HEADLINE, 'value' => 'This is a headline', 'scope' => IAccountManager::SCOPE_PUBLISHED ], [ 'name' => IAccountManager::PROPERTY_BIOGRAPHY, 'value' => 'Some long biography', 'scope' => IAccountManager::SCOPE_PUBLISHED ], ], ], [ 'user' => $this->makeUser('31b5316a-9b57-4b17-970a-315a4cbe73eb', 'K. Cheng', 'cheng@emca.com'), 'data' => [ [ 'name' => IAccountManager::PROPERTY_DISPLAYNAME, 'value' => 'K. Cheng', 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_EMAIL, 'value' => 'cheng@emca.com', 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_TWITTER, 'value' => '', ' scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_PHONE, 'value' => '+71601212123', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_ADDRESS, 'value' => 'Pinapple Street 22', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_WEBSITE, 'value' => 'https://emca.com', 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_ORGANISATION, 'value' => 'Organisation A', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_ROLE, 'value' => 'Animal', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_HEADLINE, 'value' => 'My headline', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_BIOGRAPHY, 'value' => 'Short biography', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::COLLECTION_EMAIL, 'value' => 'k.cheng@emca.com', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::COLLECTION_EMAIL, 'value' => 'kai.cheng@emca.com', 'scope' => IAccountManager::SCOPE_LOCAL ], ], ], [ 'user' => $this->makeUser('goodpal@elpmaxe.org', 'Goodpal, Kim', 'goodpal@elpmaxe.org'), 'data' => [ [ 'name' => IAccountManager::PROPERTY_DISPLAYNAME, 'value' => 'Goodpal, Kim', 'scope' => IAccountManager::SCOPE_PUBLISHED ], [ 'name' => IAccountManager::PROPERTY_EMAIL, 'value' => 'goodpal@elpmaxe.org', 'scope' => IAccountManager::SCOPE_PUBLISHED ], [ 'name' => IAccountManager::PROPERTY_TWITTER, 'value' => '', 'scope' => IAccountManager::SCOPE_LOCAL ], [ 'name' => IAccountManager::PROPERTY_PHONE, 'value' => '+71602121231', 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_ADDRESS, 'value' => 'Octopus Ave 17', 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_WEBSITE, 'value' => 'https://elpmaxe.org', 'scope' => IAccountManager::SCOPE_PUBLISHED ], [ 'name' => IAccountManager::PROPERTY_ORGANISATION, 'value' => 'Organisation B', 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_ROLE, 'value' => 'Organism', 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_HEADLINE, 'value' => 'Best headline', 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_BIOGRAPHY, 'value' => 'Autobiography', 'scope' => IAccountManager::SCOPE_FEDERATED ], ], ], ]; foreach ($users as $userInfo) { $this->invokePrivate($this->accountManager, 'updateUser', [$userInfo['user'], $userInfo['data'], null, false]); } } /** * get a instance of the accountManager * * @param array $mockedMethods list of methods which should be mocked * @return MockObject | AccountManager */ public function getInstance($mockedMethods = null) { return $this->getMockBuilder(AccountManager::class) ->setConstructorArgs([ $this->connection, $this->config, $this->eventDispatcher, $this->jobList, $this->logger, $this->verificationToken, $this->mailer, $this->defaults, $this->l10nFactory, $this->urlGenerator, $this->crypto ]) ->setMethods($mockedMethods) ->getMock(); } /** * @dataProvider dataTrueFalse * * @param array $newData * @param array $oldData * @param bool $insertNew * @param bool $updateExisting */ public function testUpdateUser($newData, $oldData, $insertNew, $updateExisting) { $accountManager = $this->getInstance(['getUser', 'insertNewUser', 'updateExistingUser']); /** @var IUser $user */ $user = $this->createMock(IUser::class); if ($updateExisting) { $accountManager->expects($this->once())->method('updateExistingUser') ->with($user, $newData); $accountManager->expects($this->never())->method('insertNewUser'); } if ($insertNew) { $accountManager->expects($this->once())->method('insertNewUser') ->with($user, $newData); $accountManager->expects($this->never())->method('updateExistingUser'); } if (!$insertNew && !$updateExisting) { $accountManager->expects($this->never())->method('updateExistingUser'); $accountManager->expects($this->never())->method('insertNewUser'); $this->eventDispatcher->expects($this->never())->method('dispatch'); } else { $this->eventDispatcher->expects($this->once())->method('dispatch') ->willReturnCallback( function ($eventName, $event) use ($user, $newData) { $this->assertSame('OC\AccountManager::userUpdated', $eventName); $this->assertInstanceOf(GenericEvent::class, $event); /** @var GenericEvent $event */ $this->assertSame($user, $event->getSubject()); $this->assertSame($newData, $event->getArguments()); } ); } $this->invokePrivate($accountManager, 'updateUser', [$user, $newData, $oldData]); } public function dataTrueFalse(): array { return [ #$newData | $oldData | $insertNew | $updateExisting [['myProperty' => ['value' => 'newData']], ['myProperty' => ['value' => 'oldData']], false, true], [['myProperty' => ['value' => 'oldData']], ['myProperty' => ['value' => 'oldData']], false, false] ]; } public function testAddMissingDefaults() { $user = $this->createMock(IUser::class); $this->config ->expects($this->once()) ->method('getAppValue') ->with('settings', 'profile_enabled_by_default', '1') ->willReturn('1'); $input = [ [ 'name' => IAccountManager::PROPERTY_DISPLAYNAME, 'value' => 'bob', 'verified' => IAccountManager::NOT_VERIFIED, ], [ 'name' => IAccountManager::PROPERTY_EMAIL, 'value' => 'bob@bob.bob', ], ]; $expected = [ [ 'name' => IAccountManager::PROPERTY_DISPLAYNAME, 'value' => 'bob', 'scope' => IAccountManager::SCOPE_FEDERATED, 'verified' => IAccountManager::NOT_VERIFIED, ], [ 'name' => IAccountManager::PROPERTY_EMAIL, 'value' => 'bob@bob.bob', 'scope' => IAccountManager::SCOPE_FEDERATED, 'verified' => IAccountManager::NOT_VERIFIED, ], [ 'name' => IAccountManager::PROPERTY_ADDRESS, 'value' => '', 'scope' => IAccountManager::SCOPE_LOCAL, 'verified' => IAccountManager::NOT_VERIFIED, ], [ 'name' => IAccountManager::PROPERTY_WEBSITE, 'value' => '', 'scope' => IAccountManager::SCOPE_LOCAL, 'verified' => IAccountManager::NOT_VERIFIED, ], [ 'name' => IAccountManager::PROPERTY_AVATAR, 'scope' => IAccountManager::SCOPE_FEDERATED ], [ 'name' => IAccountManager::PROPERTY_PHONE, 'value' => '', 'scope' => IAccountManager::SCOPE_LOCAL, 'verified' => IAccountManager::NOT_VERIFIED, ], [ 'name' => IAccountManager::PROPERTY_TWITTER, 'value' => '', 'scope' => IAccountManager::SCOPE_LOCAL, 'verified' => IAccountManager::NOT_VERIFIED, ], [ 'name' => IAccountManager::PROPERTY_ORGANISATION, 'value' => '', 'scope' => IAccountManager::SCOPE_LOCAL, ], [ 'name' => IAccountManager::PROPERTY_ROLE, 'value' => '', 'scope' => IAccountManager::SCOPE_LOCAL, ], [ 'name' => IAccountManager::PROPERTY_HEADLINE, 'value' => '', 'scope' => IAccountManager::SCOPE_LOCAL, ], [ 'name' => IAccountManager::PROPERTY_BIOGRAPHY, 'value' => '', 'scope' => IAccountManager::SCOPE_LOCAL, ], [ 'name' => IAccountManager::PROPERTY_PROFILE_ENABLED, 'value' => '1', ], ]; $defaultUserRecord = $this->invokePrivate($this->accountManager, 'buildDefaultUserRecord', [$user]); $result = $this->invokePrivate($this->accountManager, 'addMissingDefaultValues', [$input, $defaultUserRecord]); $this->assertSame($expected, $result); } private function addDummyValuesToTable($uid, $data) { $query = $this->connection->getQueryBuilder(); $query->insert($this->table) ->values( [ 'uid' => $query->createNamedParameter($uid), 'data' => $query->createNamedParameter(json_encode($data)), ] ) ->execute(); } public function testGetAccount() { $accountManager = $this->getInstance(['getUser']); /** @var IUser $user */ $user = $this->createMock(IUser::class); $data = [ [ 'value' => '@twitterhandle', 'scope' => IAccountManager::SCOPE_LOCAL, 'verified' => IAccountManager::NOT_VERIFIED, 'name' => IAccountManager::PROPERTY_TWITTER, ], [ 'value' => 'test@example.com', 'scope' => IAccountManager::SCOPE_PUBLISHED, 'verified' => IAccountManager::VERIFICATION_IN_PROGRESS, 'name' => IAccountManager::PROPERTY_EMAIL, ], [ 'value' => 'https://example.com', 'scope' => IAccountManager::SCOPE_FEDERATED, 'verified' => IAccountManager::VERIFIED, 'name' => IAccountManager::PROPERTY_WEBSITE, ], ]; $expected = new Account($user); $expected->setProperty(IAccountManager::PROPERTY_TWITTER, '@twitterhandle', IAccountManager::SCOPE_LOCAL, IAccountManager::NOT_VERIFIED); $expected->setProperty(IAccountManager::PROPERTY_EMAIL, 'test@example.com', IAccountManager::SCOPE_PUBLISHED, IAccountManager::VERIFICATION_IN_PROGRESS); $expected->setProperty(IAccountManager::PROPERTY_WEBSITE, 'https://example.com', IAccountManager::SCOPE_FEDERATED, IAccountManager::VERIFIED); $accountManager->expects($this->once()) ->method('getUser') ->willReturn($data); $this->assertEquals($expected, $accountManager->getAccount($user)); } public function dataParsePhoneNumber(): array { return [ ['0711 / 25 24 28-90', 'DE', '+4971125242890'], ['0711 / 25 24 28-90', '', null], ['+49 711 / 25 24 28-90', '', '+4971125242890'], ]; } /** * @dataProvider dataParsePhoneNumber * @param string $phoneInput * @param string $defaultRegion * @param string|null $phoneNumber */ public function testParsePhoneNumber(string $phoneInput, string $defaultRegion, ?string $phoneNumber): void { $this->config->method('getSystemValueString') ->willReturn($defaultRegion); if ($phoneNumber === null) { $this->expectException(\InvalidArgumentException::class); self::invokePrivate($this->accountManager, 'parsePhoneNumber', [$phoneInput]); } else { self::assertEquals($phoneNumber, self::invokePrivate($this->accountManager, 'parsePhoneNumber', [$phoneInput])); } } public function dataParseWebsite(): array { return [ ['https://nextcloud.com', 'https://nextcloud.com'], ['http://nextcloud.com', 'http://nextcloud.com'], ['ftp://nextcloud.com', null], ['//nextcloud.com/', null], ['https:///?query', null], ]; } /** * @dataProvider dataParseWebsite * @param string $websiteInput * @param string|null $websiteOutput */ public function testParseWebsite(string $websiteInput, ?string $websiteOutput): void { if ($websiteOutput === null) { $this->expectException(\InvalidArgumentException::class); self::invokePrivate($this->accountManager, 'parseWebsite', [$websiteInput]); } else { self::assertEquals($websiteOutput, self::invokePrivate($this->accountManager, 'parseWebsite', [$websiteInput])); } } /** * @dataProvider searchDataProvider */ public function testSearchUsers(string $property, array $values, array $expected): void { $this->populateOrUpdate(); $matchedUsers = $this->accountManager->searchUsers($property, $values); foreach ($expected as $expectedEntry) { $this->assertContains($expectedEntry, $matchedUsers); } if (empty($expected)) { $this->assertEmpty($matchedUsers); } } public function searchDataProvider(): array { return [ [ #0 Search for an existing name IAccountManager::PROPERTY_DISPLAYNAME, ['Jane Doe'], ['Jane Doe' => 'j.doe'] ], [ #1 Search for part of a name (no result) IAccountManager::PROPERTY_DISPLAYNAME, ['Jane'], [] ], [ #2 Search for part of a name (no result, test wildcard) IAccountManager::PROPERTY_DISPLAYNAME, ['Jane%'], [] ], [ #3 Search for phone IAccountManager::PROPERTY_PHONE, ['+491603121212'], ['+491603121212' => 'b32c5a5b-1084-4380-8856-e5223b16de9f'], ], [ #4 Search for twitter handles IAccountManager::PROPERTY_TWITTER, ['@sometwitter', '@a_alice', '@unseen'], ['@sometwitter' => 'j.doe', '@a_alice' => 'a.allison'], ], [ #5 Search for email IAccountManager::PROPERTY_EMAIL, ['cheng@emca.com'], ['cheng@emca.com' => '31b5316a-9b57-4b17-970a-315a4cbe73eb'], ], [ #6 Search for email by additional email IAccountManager::PROPERTY_EMAIL, ['kai.cheng@emca.com'], ['kai.cheng@emca.com' => '31b5316a-9b57-4b17-970a-315a4cbe73eb'], ], [ #7 Search for additional email IAccountManager::COLLECTION_EMAIL, ['kai.cheng@emca.com', 'cheng@emca.com'], ['kai.cheng@emca.com' => '31b5316a-9b57-4b17-970a-315a4cbe73eb'], ], [ #8 Search for email by additional email (two valid search values, but the same user) IAccountManager::PROPERTY_EMAIL, ['kai.cheng@emca.com', 'cheng@emca.com'], [ 'kai.cheng@emca.com' => '31b5316a-9b57-4b17-970a-315a4cbe73eb', ], ], ]; } public function dataCheckEmailVerification(): array { return [ [$this->makeUser('steve', 'Steve Smith', 'steve@steve.steve'), null], [$this->makeUser('emma', 'Emma Morales', 'emma@emma.com'), 'emma@morales.com'], [$this->makeUser('sarah@web.org', 'Sarah Foster', 'sarah@web.org'), null], [$this->makeUser('cole@web.org', 'Cole Harrison', 'cole@web.org'), 'cole@example.com'], [$this->makeUser('8d29e358-cf69-4849-bbf9-28076c0b908b', 'Alice McPherson', 'alice@example.com'), 'alice@mcpherson.com'], [$this->makeUser('11da2744-3f4d-4c17-8c13-4c057a379237', 'James Loranger', 'james@example.com'), ''], ]; } /** * @dataProvider dataCheckEmailVerification */ public function testCheckEmailVerification(IUser $user, ?string $newEmail): void { $account = $this->accountManager->getAccount($user); $emailUpdated = false; if (!empty($newEmail)) { $account->getProperty(IAccountManager::PROPERTY_EMAIL)->setValue($newEmail); $emailUpdated = true; } if ($emailUpdated) { $this->jobList->expects($this->once()) ->method('add') ->with(VerifyUserData::class); } else { $this->jobList->expects($this->never()) ->method('add') ->with(VerifyUserData::class); } /** @var array $oldData */ $oldData = $this->invokePrivate($this->accountManager, 'getUser', [$user, false]); $this->invokePrivate($this->accountManager, 'checkEmailVerification', [$account, $oldData]); } }