tao-test/app/taoProctoring/test/unit/model/execution/DeliveryExecutionManagerServiceTest.php

464 lines
17 KiB
PHP
Raw Normal View History

2022-08-29 20:14:13 +02:00
<?php
/**
* This program 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; under version 2
* of the License (non-upgradable).
*
* 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
* Copyright (c) 2020 (original work) Open Assessment Technologies SA;
*/
declare(strict_types=1);
namespace oat\taoProctoring\test\unit\model\execution;
use common_Exception;
use common_exception_Error;
use common_exception_NotFound;
use common_ext_ExtensionException;
use common_session_Session;
use core_kernel_classes_Resource;
use oat\generis\test\MockObject;
use oat\generis\test\TestCase;
use oat\oatbox\event\EventManager;
use oat\oatbox\log\LoggerService;
use oat\oatbox\service\exception\InvalidServiceManagerException;
use oat\oatbox\session\SessionService;
use oat\oatbox\user\User;
use oat\taoDelivery\model\execution\DeliveryExecution;
use oat\taoDelivery\model\execution\DeliveryExecutionInterface;
use oat\taoDelivery\model\execution\ServiceProxy;
use oat\taoProctoring\model\event\DeliveryExecutionTimerAdjusted;
use oat\taoProctoring\model\execution\DeliveryExecution as DeliveryExecutionProctoring;
use oat\taoProctoring\model\execution\DeliveryExecutionManagerService;
use oat\taoProctoring\model\implementation\TestSessionService;
use oat\taoProctoring\model\monitorCache\DeliveryMonitoringData;
use oat\taoProctoring\model\monitorCache\DeliveryMonitoringService;
use oat\taoQtiTest\models\QtiTestExtractionFailedException;
use oat\taoQtiTest\models\runner\session\TestSession;
use oat\taoQtiTest\models\runner\time\QtiTimeConstraint;
use oat\taoQtiTest\models\runner\time\QtiTimer;
use oat\taoQtiTest\models\runner\time\TimerAdjustmentService;
use oat\taoQtiTest\models\runner\time\TimerAdjustmentServiceInterface;
use qtism\common\datatypes\QtiDuration;
use qtism\data\QtiIdentifiable;
class DeliveryExecutionManagerServiceTest extends TestCase
{
/**
* @var DeliveryExecutionManagerService
*/
private $subject;
/**
* @var ServiceProxy|MockObject
*/
private $serviceProxyMock;
/**
* @var TestSessionService|MockObject
*/
private $testSessionServiceMock;
/**
* @var TimerAdjustmentServiceInterface|MockObject
*/
private $timerAdjustmentServiceMock;
/**
* @var DeliveryMonitoringData|MockObject
*/
private $deliveryMonitoringDataMock;
/**
* @var DeliveryMonitoringService|MockObject
*/
private $deliveryMonitoringServiceMock;
/**
* @var User|MockObject
*/
private $userMock;
/**
* @var common_session_Session|MockObject
*/
private $sessionMock;
/**
* @var SessionService|MockObject
*/
private $sessionServiceMock;
/**
* @var EventManager|MockObject
*/
private $eventManagerMock;
/**
* @var TestSession|MockObject
*/
private $testSessionMock;
/**
* @var QtiTimer|MockObject
*/
private $qtiTimerMock;
private $loggerServiceMock;
protected function setUp(): void
{
parent::setUp();
$this->timerAdjustmentServiceMock = $this->createMock(TimerAdjustmentServiceInterface::class);
$this->deliveryMonitoringDataMock = $this->createMock(DeliveryMonitoringData::class);
$this->deliveryMonitoringServiceMock = $this->createMock(DeliveryMonitoringService::class);
$this->userMock = $this->createMock(User::class);
$this->sessionMock = $this->createMock(common_session_Session::class);
$this->sessionServiceMock = $this->createMock(SessionService::class);
$this->eventManagerMock = $this->createMock(EventManager::class);
$this->testSessionMock = $this->createMock(TestSession::class);
$this->testSessionServiceMock = $this->createMock(TestSessionService::class);
$this->serviceProxyMock = $this->createMock(ServiceProxy::class);
$this->qtiTimerMock = $this->createMock(QtiTimer::class);
$this->loggerServiceMock = $this->createMock(LoggerService::class);
$serviceLocatorMock = $this->getServiceLocatorMock([
ServiceProxy::SERVICE_ID => $this->serviceProxyMock,
TestSessionService::SERVICE_ID => $this->testSessionServiceMock,
LoggerService::SERVICE_ID => $this->loggerServiceMock,
TimerAdjustmentService::SERVICE_ID => $this->timerAdjustmentServiceMock,
EventManager::SERVICE_ID => $this->eventManagerMock,
SessionService::SERVICE_ID => $this->sessionServiceMock,
DeliveryMonitoringService::SERVICE_ID => $this->deliveryMonitoringServiceMock,
]);
$this->subject = new DeliveryExecutionManagerService();
$this->subject->setServiceLocator($serviceLocatorMock);
}
/**
* @throws common_Exception
* @throws common_exception_Error
* @throws common_exception_NotFound
* @throws common_ext_ExtensionException
* @throws InvalidServiceManagerException
* @throws QtiTestExtractionFailedException
*/
public function testAdjustTimers(): void
{
$expectedResult = [
'processed' => [
'AWAITING_URI' => true
],
'unprocessed' => [
'CANCELED_URI' => false,
'PAUSED_URI' => false,
]
];
// Mock input data
$awaitingExecution = $this->getDeliveryExecutionMockWithState(
DeliveryExecutionProctoring::STATE_AWAITING,
'AWAITING_URI'
);
$canceledExecution = $this->getDeliveryExecutionMockWithState(
DeliveryExecutionProctoring::STATE_CANCELED,
'CANCELED_URI'
);
$pausedExecution = $this->getDeliveryExecutionMockWithState(
DeliveryExecutionProctoring::STATE_PAUSED,
'PAUSED_URI'
);
$deliveryExecutions = [$awaitingExecution,$canceledExecution, $pausedExecution];
// Mock user session
$this->sessionMock->method('getUser')->willReturn($this->userMock);
$this->sessionServiceMock->method('getCurrentSession')->willReturn($this->sessionMock);
// Mock time adjustment
$timeConstraintMock = $this->createMock(QtiTimeConstraint::class);
$this->testSessionServiceMock
->method('getTestSession')
->willReturn($this->testSessionMock);
$this->testSessionServiceMock
->method('getSmallestMaxTimeConstraint')
->willReturn($timeConstraintMock);
$this->timerAdjustmentServiceMock
->expects(self::once())
->method('increase')
->willReturn(true);
$this->deliveryMonitoringServiceMock
->method('getData')
->willReturn($this->deliveryMonitoringDataMock);
$this->deliveryMonitoringDataMock
->method('updateData')
->with([DeliveryMonitoringService::REMAINING_TIME]);
$this->eventManagerMock
->expects(self::once())
->method('trigger');
$adjustmentValue = 10;
$adjustmentResult = $this->subject->adjustTimers($deliveryExecutions, $adjustmentValue, ['reasons']);
self::assertSame($expectedResult, $adjustmentResult, 'Method must return response in correct format.');
}
public function testGetTimerAdjustmentDecreaseLimit_CalculationFailedNoDecreaseLimit(): void
{
$expectedLimit = 300;
$deliveryExecutionId = 'FAKE_ID';
$deliveryExecutionMock = $this->createMock(DeliveryExecution::class);
$this->serviceProxyMock->method('getDeliveryExecution')
->willReturn($deliveryExecutionMock);
// Setup TestSessionService mock
$durationMock = $this->createMock(QtiDuration::class);
$durationMock->method('getSeconds')
->willReturn($expectedLimit);
$qtiTimeConstraintMock = $this->createMock(QtiTimeConstraint::class);
$qtiTimeConstraintMock->method('getMaximumRemainingTime')
->willReturn($durationMock);
$testSessionMock = $this->createMock(TestSession::class);
$this->testSessionServiceMock->method('getTestSession')
->willReturn($testSessionMock);
$this->testSessionServiceMock->method('getSmallestMaxTimeConstraint')
->willReturn($qtiTimeConstraintMock);
self::assertSame(
$expectedLimit,
$this->subject->getTimerAdjustmentDecreaseLimit($deliveryExecutionId),
'Method must return correct value of maximum possible time decrease.'
);
}
public function testGetTimerAdjustmentDecreaseLimit_NullSmallestMaxTime(): void
{
$expectedLimit = -1;
$deliveryExecutionId = 'FAKE_ID';
$deliveryExecutionMock = $this->createMock(DeliveryExecution::class);
$this->serviceProxyMock->method('getDeliveryExecution')
->willReturn($deliveryExecutionMock);
$testSessionMock = $this->createMock(TestSession::class);
$this->testSessionServiceMock->method('getTestSession')
->willReturn($testSessionMock);
$this->testSessionServiceMock->method('getSmallestMaxTimeConstraint')
->willReturn(null);
self::assertSame(
$expectedLimit,
$this->subject->getTimerAdjustmentDecreaseLimit($deliveryExecutionId),
'Method must return correct value of maximum possible time decrease.'
);
}
public function testGetTimerAdjustmentDecreaseLimit(): void
{
$expectedLimit = -1;
$deliveryExecutionId = 'FAKE_ID';
$this->serviceProxyMock->method('getDeliveryExecution')
->willThrowException(new common_Exception('FAKE ERROR MESSAGE'));
self::assertSame(
$expectedLimit,
$this->subject->getTimerAdjustmentDecreaseLimit($deliveryExecutionId),
'Method must return correct value in case when limit calculation failed.'
);
}
public function testGetTimerAdjustmentIncreaseLimit(): void
{
$expectedLimit = -1;
$deliveryExecutionId = 'FAKE_ID';
self::assertSame(
$expectedLimit,
$this->subject->getTimerAdjustmentIncreaseLimit($deliveryExecutionId),
'Method must return correct maximum limit for timer increase.'
);
}
public function testAdjustedTimeWithoutTestSession(): void {
$pausedExecution = $this->getDeliveryExecutionMockWithState(
DeliveryExecutionProctoring::STATE_PAUSED,
'paused'
);
$this->serviceProxyMock
->expects($this->once())
->method('getDeliveryExecution')
->willReturn($pausedExecution);
$this->assertSame(0, $this->subject->getAdjustedTime('PHPUnitDeliveryExecutionId'));
}
public function testAdjustedTimeWithoutTimer(): void
{
$pausedExecution = $this->getDeliveryExecutionMockWithState(
DeliveryExecutionProctoring::STATE_PAUSED,
'paused'
);
$this->serviceProxyMock
->expects($this->once())
->method('getDeliveryExecution')
->willReturn($pausedExecution);
$this->testSessionServiceMock
->expects($this->once())
->method('getTestSession')
->willReturn($this->testSessionMock);
$this->assertSame(0, $this->subject->getAdjustedTime('PHPUnitDeliveryExecutionId'));
}
/**
* @throws QtiTestExtractionFailedException
*/
public function testAdjustedTime(): void
{
$item = $this->createMock(QtiIdentifiable::class);
$pausedExecution = $this->getDeliveryExecutionMockWithState(
DeliveryExecutionProctoring::STATE_PAUSED,
'paused'
);
$this->serviceProxyMock
->expects($this->once())
->method('getDeliveryExecution')
->willReturn($pausedExecution);
$this->timerAdjustmentServiceMock
->expects($this->once())
->method('getAdjustmentByType')
->willReturn(9);
$this->testSessionServiceMock
->expects($this->once())
->method('getTestSession')
->willReturn($this->testSessionMock);
$qtiTimeConstrainMock = $this->createMock(QtiTimeConstraint::class);
$qtiTimeConstrainMock
->expects($this->once())
->method('getSource')
->willReturn($item);
$qtiTimeConstrainMock
->expects($this->once())
->method('getTimer')
->willReturn($this->qtiTimerMock);
$this->testSessionServiceMock
->expects($this->once())
->method('getSmallestMaxTimeConstraint')
->willReturn($qtiTimeConstrainMock);
$this->assertSame(9, $this->subject->getAdjustedTime('PHPUnitDeliveryExecutionId'));
}
public function isTimerAdjustmentAllowed_WhenSessionHasTimer_ThenReturnTrue(): void
{
$executionState = 'http://www.tao.lu/Ontologies/TAODelivery.rdf#DeliveryExecutionStatusAwaiting';
$deliveryExecution = $this->getDeliveryExecutionMock($executionState);
$this->testSessionServiceMock
->method('getTestSession')
->willReturn($this->testSessionMock);
$timeConstraintMock = $this->createMock(QtiTimeConstraint::class);
$this->testSessionServiceMock
->method('getSmallestMaxTimeConstraint')
->willReturn($timeConstraintMock);
$adjustmentAllowed = $this->subject->isTimerAdjustmentAllowed($deliveryExecution);
self::assertTrue($adjustmentAllowed, 'Timer adjustment must be allowed if test session has time constraint.');
}
public function isTimerAdjustmentAllowed_WhenIncorrectSessionStatus_ThenReturnFalse(): void
{
$executionState = 'INVALID_URI';
$deliveryExecution = $this->getDeliveryExecutionMock($executionState);
$adjustmentAllowed = $this->subject->isTimerAdjustmentAllowed($deliveryExecution);
self::assertFalse($adjustmentAllowed, 'Timer adjustment should not be allowed if test session is in incorrect state.');
}
public function isTimerAdjustmentAllowed_WhenSessionNotLoaded_ThenReturnFalse(): void
{
$executionState = 'http://www.tao.lu/Ontologies/TAODelivery.rdf#DeliveryExecutionStatusAwaiting';
$deliveryExecution = $this->getDeliveryExecutionMock($executionState);
$this->testSessionServiceMock
->method('getTestSession')
->willReturn(null);
$adjustmentAllowed = $this->subject->isTimerAdjustmentAllowed($deliveryExecution);
self::assertFalse($adjustmentAllowed, 'Timer adjustment should not be allowed if test session cannot be loaded.');
}
public function isTimerAdjustmentAllowed_WhenSessionWithoutTimer_ThenReturnFalse(): void
{
$executionState = 'http://www.tao.lu/Ontologies/TAODelivery.rdf#DeliveryExecutionStatusAwaiting';
$deliveryExecution = $this->getDeliveryExecutionMock($executionState);
$this->testSessionServiceMock
->method('getTestSession')
->willReturn($this->testSessionMock);
$this->testSessionServiceMock
->method('getSmallestMaxTimeConstraint')
->willReturn(null);
$adjustmentAllowed = $this->subject->isTimerAdjustmentAllowed($deliveryExecution);
self::assertFalse($adjustmentAllowed, 'Timer adjustment should not be allowed if test session has no time constraints.');
}
/**
* @param string $stateUri
* @param string $deliveryIdentifier
* @return DeliveryExecution|MockObject
*/
private function getDeliveryExecutionMockWithState(string $stateUri, string $deliveryIdentifier): DeliveryExecution
{
$executionStateMock = $this->createMock(core_kernel_classes_Resource::class);
$executionStateMock->method('getUri')->willReturn($stateUri);
$deliveryExecutionMock = $this->createMock(DeliveryExecution::class);
$deliveryExecutionMock->method('getState')->willReturn($executionStateMock);
$deliveryExecutionMock->method('getIdentifier')->willReturn($deliveryIdentifier);
return $deliveryExecutionMock;
}
/**
* @param string $state
* @return DeliveryExecutionInterface|MockObject
*/
private function getDeliveryExecutionMock(string $state): DeliveryExecutionInterface
{
$stateMock = $this->createMock(core_kernel_classes_Resource::class);
$stateMock->method('getUri')
->willReturn($state);
$deliveryExecution = $this->createMock(DeliveryExecutionInterface::class);
$deliveryExecution->method('getState')
->willReturn($stateMock);
return $deliveryExecution;
}
}