Files
sysPass/tests/SP/Domain/Account/Services/AccountAclTest.php
Rubén D 40c6cb1dba chore: Use strict types
Signed-off-by: Rubén D <nuxsmin@syspass.org>
2024-05-01 12:03:55 +02:00

864 lines
28 KiB
PHP

<?php
declare(strict_types=1);
/*
* sysPass
*
* @author nuxsmin
* @link https://syspass.org
* @copyright 2012-2024, Rubén Domínguez nuxsmin@$syspass.org
*
* This file is part of sysPass.
*
* sysPass is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* sysPass 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with sysPass. If not, see <http://www.gnu.org/licenses/>.
*/
namespace SP\Tests\Domain\Account\Services;
use DateTime;
use PHPUnit\Framework\Attributes\DataProvider;
use PHPUnit\Framework\Attributes\Group;
use PHPUnit\Framework\MockObject\Exception;
use PHPUnit\Framework\MockObject\MockObject;
use SP\Core\Acl\Acl;
use SP\Domain\Account\Adapters\AccountPermission;
use SP\Domain\Account\Dtos\AccountAclDto;
use SP\Domain\Account\Services\AccountAcl;
use SP\Domain\Common\Models\Item;
use SP\Domain\Common\Models\Simple;
use SP\Domain\Core\Acl\AclActionsInterface;
use SP\Domain\Core\Acl\ActionsInterface;
use SP\Domain\Core\Exceptions\ConstraintException;
use SP\Domain\Core\Exceptions\QueryException;
use SP\Domain\Storage\Ports\FileCacheService;
use SP\Domain\User\Dtos\UserDataDto;
use SP\Domain\User\Models\User;
use SP\Domain\User\Ports\UserToUserGroupService;
use SP\Infrastructure\File\FileException;
use SP\Tests\Generators\UserDataGenerator;
use SP\Tests\UnitaryTestCase;
/**
* Class AccountAclServiceTest
*
*/
#[Group('unitary')]
class AccountAclTest extends UnitaryTestCase
{
private const ACTIONS = [
AclActionsInterface::ACCOUNT_SEARCH,
AclActionsInterface::ACCOUNT_VIEW,
AclActionsInterface::ACCOUNT_VIEW_PASS,
AclActionsInterface::ACCOUNT_HISTORY_VIEW,
AclActionsInterface::ACCOUNT_CREATE,
AclActionsInterface::ACCOUNT_EDIT,
AclActionsInterface::ACCOUNT_EDIT_PASS,
AclActionsInterface::ACCOUNT_EDIT_RESTORE,
AclActionsInterface::ACCOUNT_COPY,
AclActionsInterface::ACCOUNT_COPY_PASS,
AclActionsInterface::ACCOUNT_DELETE,
];
private static array $accounts;
private Acl $acl;
private UserToUserGroupService|MockObject $userToUserGroupService;
public static function setUpBeforeClass(): void
{
parent::setUpBeforeClass();
self::$accounts = [
1 => [
'userGroupId' => 1,
'userId' => 1,
'isPrivate' => 0,
'isPrivateGroup' => 0,
'otherUserGroupEdit' => 0,
'otherUserEdit' => 0,
'users' => [new Item(['id' => 3, 'isEdit' => 1])],
'groups' => [new Item(['id' => 2, 'isEdit' => 1])],
],
2 => [
'userGroupId' => 1,
'userId' => 1,
'isPrivate' => 0,
'isPrivateGroup' => 0,
'otherUserGroupEdit' => 0,
'otherUserEdit' => 0,
'users' => [],
'groups' => [
new Item(['id' => 2, 'isEdit' => 1]),
new Item(['id' => 3, 'isEdit' => 1]),
],
],
3 => [
'userGroupId' => 3,
'userId' => 3,
'isPrivate' => 1,
'isPrivateGroup' => 0,
'otherUserGroupEdit' => 0,
'otherUserEdit' => 0,
'users' => [],
'groups' => [],
],
4 => [
'userGroupId' => 3,
'userId' => 3,
'isPrivate' => 0,
'isPrivateGroup' => 1,
'otherUserGroupEdit' => 0,
'otherUserEdit' => 0,
'users' => [],
'groups' => [],
],
];
}
public static function accountPropertiesProvider(): array
{
/**
* Account |View |Edit
* 1 |u=3 g=1,2 |u=3 g=1,2
* 2 |g=1,2,3 |g=1,2,3
* 3 |u=3 g=3 |u=3 g=3
* 4 |u=3 |u=3
*
* User | Group
* 1 | 2
* 2 | 1
* 3 | 2
* 4 | None
* Matrix: Account | UserId | GroupId | ShouldView | ShouldEdit
*/
return [
[1, 2, 2, true, true],
[1, 3, 0, true, true],
[1, 3, 1, true, true],
[1, 3, 2, true, true],
[1, 4, 0, false, false],
[1, 4, 3, false, false],
[1, 4, 4, false, false],
[2, 2, 1, true, true],
[2, 2, 2, true, true],
[2, 2, 3, true, true],
[2, 3, 0, false, false],
[2, 3, 3, true, true],
[2, 4, 0, false, false],
[2, 4, 1, true, true],
[2, 4, 2, true, true],
[2, 4, 3, true, true],
[2, 4, 4, false, false],
[3, 1, 1, false, false],
[3, 1, 2, false, false],
[3, 1, 3, true, true],
[3, 2, 1, false, false],
[3, 2, 2, false, false],
[3, 2, 3, true, true],
[3, 3, 0, true, true],
[3, 3, 1, true, true],
[3, 3, 2, true, true],
[3, 3, 3, true, true],
[4, 3, 0, true, true],
[4, 3, 1, true, true],
[4, 3, 2, true, true],
[4, 3, 3, true, true],
];
}
/**
* @throws ConstraintException
* @throws QueryException
*/
#[Group('acl:admin')]
public function testGetAclForAdminApp(): void
{
$this->context
->getUserProfile()
->setAccAdd(true)
->setAccView(true)
->setAccViewPass(true)
->setAccEdit(true)
->setAccEditPass(true)
->setAccFiles(true)
->setAccDelete(true)
->setAccPermission(true)
->setAccViewHistory(true);
$this->checkForUserByExample(
$this->setUpAccountEnvironment(
self::$faker->numberBetween(1, 4),
self::$faker->randomNumber(),
self::$faker->randomNumber(),
true
),
$this->getExampleAclForAdmin()
);
}
/**
* @param AccountAclDto $accountAclDto The ACL dto to compile the ACL for the user
* @param AccountPermission $example An example ACL to test against the compiled ACL
*
* @throws ConstraintException
* @throws QueryException
*/
private function checkForUserByExample(
AccountAclDto $accountAclDto,
AccountPermission $example
): void {
$accountAcl = new AccountAcl(
$this->application,
$this->acl,
$this->userToUserGroupService
);
foreach (self::ACTIONS as $action) {
$example->setActionId($action);
$aclUnderTest = $accountAcl->getAcl($action, $accountAclDto);
$this->assertTrue($aclUnderTest->isCompiledAccountAccess());
$this->assertTrue($aclUnderTest->isCompiledShowAccess());
$this->assertEquals($example->isResultView(), $aclUnderTest->isResultView());
$this->assertEquals($example->isResultEdit(), $aclUnderTest->isResultEdit());
$this->assertEquals($example->isShowPermission(), $aclUnderTest->isShowPermission());
if ($action !== AclActionsInterface::ACCOUNT_CREATE
&& $action !== AclActionsInterface::ACCOUNT_COPY_PASS
) {
$this->assertEquals($example->checkAccountAccess($action), $aclUnderTest->checkAccountAccess($action));
}
if ($action === AclActionsInterface::ACCOUNT_VIEW
|| $action === AclActionsInterface::ACCOUNT_HISTORY_VIEW
|| $action === AclActionsInterface::ACCOUNT_DELETE
) {
$this->assertEquals($example->isShowDetails(), $aclUnderTest->isShowDetails());
}
if ($action === AclActionsInterface::ACCOUNT_CREATE
|| $action === AclActionsInterface::ACCOUNT_COPY
) {
$this->assertEquals($example->isShowPass(), $aclUnderTest->isShowPass());
}
if ($action === AclActionsInterface::ACCOUNT_EDIT
|| $action === AclActionsInterface::ACCOUNT_VIEW
|| $action === AclActionsInterface::ACCOUNT_HISTORY_VIEW
) {
$this->assertEquals($example->isShowFiles(), $aclUnderTest->isShowFiles());
}
if ($action === AclActionsInterface::ACCOUNT_SEARCH
|| $action === AclActionsInterface::ACCOUNT_VIEW
|| $action === AclActionsInterface::ACCOUNT_VIEW_PASS
|| $action === AclActionsInterface::ACCOUNT_HISTORY_VIEW
|| $action === AclActionsInterface::ACCOUNT_EDIT
) {
$this->assertEquals($example->isShowViewPass(), $aclUnderTest->isShowViewPass());
}
if ($action === AclActionsInterface::ACCOUNT_EDIT
|| $action === AclActionsInterface::ACCOUNT_CREATE
|| $action === AclActionsInterface::ACCOUNT_COPY
) {
$this->assertEquals($example->isShowSave(), $aclUnderTest->isShowSave());
}
if ($action === AclActionsInterface::ACCOUNT_SEARCH
|| $action === AclActionsInterface::ACCOUNT_VIEW
) {
$this->assertEquals($example->isShowEdit(), $aclUnderTest->isShowEdit());
}
if ($action === AclActionsInterface::ACCOUNT_EDIT
|| $action === AclActionsInterface::ACCOUNT_VIEW
) {
$this->assertEquals($example->isShowEditPass(), $aclUnderTest->isShowEditPass());
}
if ($action === AclActionsInterface::ACCOUNT_SEARCH
|| $action === AclActionsInterface::ACCOUNT_DELETE
|| $action === AclActionsInterface::ACCOUNT_EDIT
) {
$this->assertEquals($example->isShowDelete(), $aclUnderTest->isShowDelete());
}
if ($action === AclActionsInterface::ACCOUNT_HISTORY_VIEW) {
$this->assertEquals($example->isShowRestore(), $aclUnderTest->isShowRestore());
}
$this->assertEquals($example->isShowLink(), $aclUnderTest->isShowLink());
if ($action === AclActionsInterface::ACCOUNT_VIEW
|| $action === AclActionsInterface::ACCOUNT_HISTORY_VIEW
) {
$this->assertEquals($example->isShowHistory(), $aclUnderTest->isShowHistory());
}
if ($action === AclActionsInterface::ACCOUNT_SEARCH
|| $action === AclActionsInterface::ACCOUNT_VIEW
|| $action === AclActionsInterface::ACCOUNT_EDIT
) {
$this->assertEquals($example->isShowCopy(), $aclUnderTest->isShowCopy());
}
}
}
/**
* @param int $accountId
* @param int $userId
* @param int $groupId
*
* @param bool $isAdminApp
* @param bool|int $isAdminAcc
*
* @return AccountAclDto
*/
private function setUpAccountEnvironment(
int $accountId,
int $userId,
int $groupId,
bool $isAdminApp = false,
bool $isAdminAcc = false
): AccountAclDto {
$this->context->setUserData(
new UserDataDto(
new User(
[
'id' => $userId,
'userGroupId' => $groupId,
'isAdminApp' => $isAdminApp,
'isAdminAcc' => $isAdminAcc
]
)
)
);
return new AccountAclDto(
$accountId,
self::$accounts[$accountId]['userId'],
self::$accounts[$accountId]['users'],
self::$accounts[$accountId]['userGroupId'],
self::$accounts[$accountId]['groups'],
self::$faker->unixTime
);
}
/**
* @return AccountPermission
*/
#[Group('acl:admin')]
private function getExampleAclForAdmin(): AccountPermission
{
return (new AccountPermission(0))
->setCompiledAccountAccess(true)
->setCompiledShowAccess(true)
->setResultView(true)
->setResultEdit(true)
->setResultView(true)
->setResultEdit(true)
->setShowCopy(true)
->setShowPermission(true)
->setShowLink(true)
->setShowView(true)
->setShowViewPass(true)
->setShowRestore(true)
->setShowHistory(true)
->setShowDelete(true)
->setShowEdit(true)
->setShowEditPass(true)
->setShowFiles(true)
->setShowDetails(true)
->setShowPass(true);
}
/**
* @throws QueryException
* @throws ConstraintException
*/
#[Group('acl:admin')]
public function testGetAclForAdminAcc(): void
{
$this->context
->getUserProfile()
->setAccAdd(true)
->setAccView(true)
->setAccViewPass(true)
->setAccEdit(true)
->setAccEditPass(true)
->setAccFiles(true)
->setAccDelete(true)
->setAccPermission(true)
->setAccViewHistory(true);
$exampleAcl = $this->getExampleAclForAdmin()->setShowLink(false);
$this->checkForUserByExample(
$this->setUpAccountEnvironment(
self::$faker->numberBetween(1, 4),
self::$faker->randomNumber(),
self::$faker->randomNumber(),
false,
true
),
$exampleAcl
);
}
/**
* @param int $accountId
* @param int $userId
* @param int $groupId
* @param bool $shouldView
* @param bool $shouldEdit
*
* @throws ConstraintException
* @throws QueryException
*/
#[Group('acl:admin')]
#[DataProvider('accountPropertiesProvider')]
public function testCheckViewPass(
int $accountId,
int $userId,
int $groupId,
bool $shouldView = true,
bool $shouldEdit = true
): void {
$shouldViewOrEdit = $shouldView || $shouldEdit;
$this->context
->getUserProfile()
->setAccViewPass($shouldView);
$example = (new AccountPermission(0))
->setCompiledAccountAccess(true)
->setCompiledShowAccess(true)
->setResultView($shouldViewOrEdit)
->setResultEdit($shouldEdit)
->setShowViewPass($shouldView);
$this->checkForUserByExample($this->setUpAccountEnvironment($accountId, $userId, $groupId), $example);
}
/**
* @param int $accountId
* @param int $userId
* @param int $groupId
* @param bool $shouldView
* @param bool $shouldEdit
*
* @throws ConstraintException
* @throws QueryException
*/
#[Group('acl:admin')]
#[DataProvider('accountPropertiesProvider')]
public function testCheckDelete(
int $accountId,
int $userId,
int $groupId,
bool $shouldView = true,
bool $shouldEdit = true
): void {
$shouldViewOrEdit = $shouldView || $shouldEdit;
$this->context
->getUserProfile()
->setAccDelete($shouldView);
$example = (new AccountPermission(0))
->setCompiledAccountAccess(true)
->setCompiledShowAccess(true)
->setResultView($shouldViewOrEdit)
->setResultEdit($shouldEdit)
->setShowDelete($shouldEdit);
$this->checkForUserByExample($this->setUpAccountEnvironment($accountId, $userId, $groupId), $example);
}
/**
* @param int $accountId
* @param int $userId
* @param int $groupId
* @param bool $shouldView
* @param bool $shouldEdit
*
* @throws ConstraintException
* @throws QueryException
*/
#[Group('acl:admin')]
#[DataProvider('accountPropertiesProvider')]
public function testEditPass(
int $accountId,
int $userId,
int $groupId,
bool $shouldView = true,
bool $shouldEdit = true
): void {
$shouldViewOrEdit = $shouldView || $shouldEdit;
$this->context
->getUserProfile()
->setAccEditPass($shouldEdit);
$example = (new AccountPermission(0))
->setCompiledAccountAccess(true)
->setCompiledShowAccess(true)
->setResultView($shouldViewOrEdit)
->setResultEdit($shouldEdit)
->setShowEditPass($shouldEdit);
$this->checkForUserByExample($this->setUpAccountEnvironment($accountId, $userId, $groupId), $example);
}
/**
* @param int $accountId
* @param int $userId
* @param int $groupId
* @param bool $shouldView
* @param bool $shouldEdit
*
* @throws ConstraintException
* @throws QueryException
*/
#[Group('acl:admin')]
#[DataProvider('accountPropertiesProvider')]
public function testEditAndRestore(
int $accountId,
int $userId,
int $groupId,
bool $shouldView = true,
bool $shouldEdit = true
): void {
$shouldViewOrEdit = $shouldView || $shouldEdit;
$this->context
->getUserProfile()
->setAccEdit($shouldEdit);
$example = (new AccountPermission(0))
->setCompiledAccountAccess(true)
->setCompiledShowAccess(true)
->setResultView($shouldViewOrEdit)
->setResultEdit($shouldEdit)
->setShowEdit($shouldEdit)
->setShowRestore($shouldEdit);
$this->checkForUserByExample($this->setUpAccountEnvironment($accountId, $userId, $groupId), $example);
}
/**
* @param int $accountId
* @param int $userId
* @param int $groupId
* @param bool $shouldView
* @param bool $shouldEdit
*
* @throws ConstraintException
* @throws QueryException
*/
#[Group('acl:admin')]
#[DataProvider('accountPropertiesProvider')]
public function testCheckPermission(
int $accountId,
int $userId,
int $groupId,
bool $shouldView = true,
bool $shouldEdit = true
): void {
$shouldViewOrEdit = $shouldView || $shouldEdit;
$this->context
->getUserProfile()
->setAccPermission($shouldEdit);
$example = (new AccountPermission(0))
->setCompiledAccountAccess(true)
->setCompiledShowAccess(true)
->setResultView($shouldViewOrEdit)
->setResultEdit($shouldEdit)
->setShowPermission($shouldEdit);
$this->checkForUserByExample($this->setUpAccountEnvironment($accountId, $userId, $groupId), $example);
}
/**
* @param int $accountId
* @param int $userId
* @param int $groupId
* @param bool $shouldView
* @param bool $shouldEdit
*
* @throws ConstraintException
* @throws QueryException
*/
#[Group('acl:admin')]
#[DataProvider('accountPropertiesProvider')]
public function testViewFiles(
int $accountId,
int $userId,
int $groupId,
bool $shouldView = true,
bool $shouldEdit = true
): void {
$shouldViewOrEdit = $shouldView || $shouldEdit;
$this->context
->getUserProfile()
->setAccFiles($shouldEdit);
$example = (new AccountPermission(0))
->setCompiledAccountAccess(true)
->setCompiledShowAccess(true)
->setResultView($shouldViewOrEdit)
->setResultEdit($shouldEdit)
->setShowFiles($shouldView);
$this->checkForUserByExample($this->setUpAccountEnvironment($accountId, $userId, $groupId), $example);
}
/**
* @param int $accountId
* @param int $userId
* @param int $groupId
* @param bool $shouldView
* @param bool $shouldEdit
*
* @throws ConstraintException
* @throws QueryException
*/
#[Group('acl:admin')]
#[DataProvider('accountPropertiesProvider')]
public function testCheckView(
int $accountId,
int $userId,
int $groupId,
bool $shouldView = true,
bool $shouldEdit = true
): void {
$shouldViewOrEdit = $shouldView || $shouldEdit;
$this->context
->getUserProfile()
->setAccView($shouldViewOrEdit)
->setAccEdit($shouldEdit);
$example = (new AccountPermission(0))
->setCompiledAccountAccess(true)
->setCompiledShowAccess(true)
->setResultView($shouldViewOrEdit)
->setResultEdit($shouldEdit)
->setShowView($shouldViewOrEdit)
->setShowEdit($shouldEdit)
->setShowRestore($shouldEdit)
->setShowDetails($shouldViewOrEdit)
->setShowPermission(
AccountAcl::getShowPermission($this->context->getUserData(), $this->context->getUserProfile())
);
$this->checkForUserByExample($this->setUpAccountEnvironment($accountId, $userId, $groupId), $example);
}
/**
* @throws Exception
* @throws ConstraintException
* @throws QueryException
*/
public function testCacheIsUsedWithHit(): void
{
$dto = new AccountAclDto(
self::$faker->randomNumber(),
1,
[],
self::$faker->randomNumber(),
[],
self::$faker->unixTime
);
$userToUserGroupService = $this->createMock(UserToUserGroupService::class);
$userToUserGroupService->method('getGroupsForUser')->willReturn([]);
$fileCache = $this->createMock(FileCacheService::class);
$actions = $this->createMock(ActionsInterface::class);
$lastUpdate = DateTime::createFromFormat('U', (string)($dto->getDateEdit() + 10))->format('Y-m-d H:i:s');
$this->context->setUserData(
new UserDataDto(
UserDataGenerator::factory()
->buildUserData()
->mutate(['lastUpdate' => $lastUpdate])
)
);
$accountAclService = new AccountAcl(
$this->application,
new Acl($this->context, $this->application->getEventDispatcher(), $actions),
$userToUserGroupService,
$fileCache
);
$acl = new AccountPermission(self::$faker->randomNumber());
$acl->setTime($dto->getDateEdit() + 10);
$fileCache->expects(self::once())
->method('load')
->with(self::callback((static fn($path) => is_string($path))))
->willReturn($acl);
$accountAclService->getAcl(self::$faker->randomNumber(), $dto);
}
/**
* @throws Exception
* @throws ConstraintException
* @throws QueryException
*/
public function testCacheIsUsedWithMiss(): void
{
$dto = new AccountAclDto(
self::$faker->randomNumber(),
1,
[],
self::$faker->randomNumber(),
[],
self::$faker->unixTime
);
$userToUserGroupService = $this->createMock(UserToUserGroupService::class);
$userToUserGroupService->method('getGroupsForUser')->willReturn([]);
$fileCache = $this->createMock(FileCacheService::class);
$actions = $this->createMock(ActionsInterface::class);
$accountAclService = new AccountAcl(
$this->application,
new Acl($this->context, $this->application->getEventDispatcher(), $actions),
$userToUserGroupService,
$fileCache
);
$acl = new AccountPermission(self::$faker->randomNumber());
$fileCache->expects(self::once())
->method('load')
->with(self::callback((static fn($path) => is_string($path))))
->willReturn($acl);
$fileCache->expects(self::once())
->method('save')
->with(
self::callback(
(static fn($acl) => $acl instanceof AccountPermission)
),
self::callback((static fn($path) => is_string($path)))
);
$accountAclService->getAcl(self::$faker->randomNumber(), $dto);
}
/**
* @throws Exception
* @throws ConstraintException
* @throws QueryException
*/
public function testCacheLoadThrowsExceptionAndLogged(): void
{
$dto = new AccountAclDto(
self::$faker->randomNumber(),
1,
[],
self::$faker->randomNumber(),
[],
self::$faker->unixTime
);
$userToUserGroupService = $this->createMock(UserToUserGroupService::class);
$userToUserGroupService->method('getGroupsForUser')->willReturn([]);
$fileCache = $this->createMock(FileCacheService::class);
$actions = $this->createMock(ActionsInterface::class);
$accountAclService = new AccountAcl(
$this->application,
new Acl($this->context, $this->application->getEventDispatcher(), $actions),
$userToUserGroupService,
$fileCache
);
$fileCache->expects(self::once())
->method('load')
->with(self::callback((static fn($path) => is_string($path))))
->willThrowException(new FileException('test'));
$accountAclService->getAcl(self::$faker->randomNumber(), $dto);
}
/**
* @throws Exception
* @throws ConstraintException
* @throws QueryException
*/
public function testCacheSaveThrowsExceptionAndLogged(): void
{
$dto = new AccountAclDto(
self::$faker->randomNumber(),
1,
[],
self::$faker->randomNumber(),
[],
self::$faker->unixTime
);
$userToUserGroupService = $this->createMock(UserToUserGroupService::class);
$userToUserGroupService->method('getGroupsForUser')->willReturn([]);
$fileCache = $this->createMock(FileCacheService::class);
$actions = $this->createMock(ActionsInterface::class);
$accountAclService = new AccountAcl(
$this->application,
new Acl($this->context, $this->application->getEventDispatcher(), $actions),
$userToUserGroupService,
$fileCache
);
$fileCache->expects(self::once())
->method('save')
->with(
self::callback(
(static fn($acl) => $acl instanceof AccountPermission)
),
self::callback((static fn($path) => is_string($path)))
)
->willThrowException(new FileException('test'));
$accountAclService->getAcl(self::$faker->randomNumber(), $dto);
}
protected function setUp(): void
{
parent::setUp();
$actions = $this->createMock(ActionsInterface::class);
$this->acl = new Acl($this->context, $this->application->getEventDispatcher(), $actions);
$this->userToUserGroupService = $this->createMock(UserToUserGroupService::class);
$this->userToUserGroupService->method('getGroupsForUser')
->willReturnMap([
[1, [new Simple(['userGroupId' => 2])]],
[2, [new Simple(['userGroupId' => 1])]],
[3, [new Simple(['userGroupId' => 2])]],
[4, []],
]);
}
}