tao-test/app/taoQtiTest/models/classes/runner/QtiRunnerService.php

2136 lines
78 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) 2016-2017 (original work) Open Assessment Technologies SA ;
*/
/**
* @author Jean-Sébastien Conan <jean-sebastien.conan@vesperiagroup.com>
*/
namespace oat\taoQtiTest\models\runner;
use common_exception_InvalidArgumentType as InvalidArgumentTypeException;
use common_persistence_AdvKeyValuePersistence;
use common_persistence_KeyValuePersistence;
use League\Flysystem\FileNotFoundException;
use oat\libCat\result\ItemResult;
use oat\libCat\result\ResultVariable;
use oat\oatbox\event\EventManager;
use oat\oatbox\service\ConfigurableService;
use oat\tao\model\theme\ThemeService;
use oat\taoDelivery\model\execution\Delete\DeliveryExecutionDeleteRequest;
use oat\taoDelivery\model\execution\DeliveryExecution;
use oat\taoDelivery\model\execution\DeliveryServerService;
use oat\taoDelivery\model\execution\ServiceProxy;
use oat\taoDelivery\model\RuntimeService;
use oat\taoItems\model\render\ItemAssetsReplacement;
use oat\taoQtiItem\model\portableElement\exception\PortableElementNotFoundException;
use oat\taoQtiItem\model\portableElement\exception\PortableModelMissing;
use oat\taoQtiItem\model\portableElement\PortableElementService;
use oat\taoQtiItem\model\QtiJsonItemCompiler;
use oat\taoQtiTest\models\cat\CatService;
use oat\taoQtiTest\models\cat\GetDeliveryExecutionsItems;
use oat\taoQtiTest\models\event\AfterAssessmentTestSessionClosedEvent;
use oat\taoQtiTest\models\event\QtiContinueInteractionEvent;
use oat\taoQtiTest\models\event\TestExitEvent;
use oat\taoQtiTest\models\event\TestInitEvent;
use oat\taoQtiTest\models\event\TestTimeoutEvent;
use oat\taoQtiTest\models\ExtendedStateService;
use oat\taoQtiTest\models\files\QtiFlysystemFileManager;
use oat\taoQtiTest\models\runner\config\QtiRunnerConfig;
use oat\taoQtiTest\models\runner\config\RunnerConfig;
use oat\taoQtiTest\models\runner\map\QtiRunnerMap;
use oat\taoQtiTest\models\runner\navigation\QtiRunnerNavigation;
use oat\taoQtiTest\models\runner\rubric\QtiRunnerRubric;
use oat\taoQtiTest\models\runner\session\TestSession;
use oat\taoQtiTest\models\runner\toolsStates\ToolsStateStorage;
use oat\taoQtiTest\models\TestSessionService;
use qtism\common\datatypes\QtiString as QtismString;
use qtism\common\enums\BaseType;
use qtism\common\enums\Cardinality;
use qtism\data\AssessmentItemRef;
use qtism\data\NavigationMode;
use qtism\data\SubmissionMode;
use qtism\runtime\common\ResponseVariable;
use qtism\runtime\common\State;
use qtism\runtime\common\Utils;
use qtism\runtime\tests\AssessmentItemSession;
use qtism\runtime\tests\AssessmentItemSessionState;
use qtism\runtime\tests\AssessmentTestSession;
use qtism\runtime\tests\AssessmentTestSessionException;
use qtism\runtime\tests\AssessmentTestSessionState;
use qtism\runtime\tests\RouteItem;
use qtism\runtime\tests\SessionManager;
use tao_models_classes_service_StateStorage;
use taoQtiTest_helpers_TestRunnerUtils as TestRunnerUtils;
/**
* Class QtiRunnerService
*
* QTI implementation service for the test runner
*
* @package oat\taoQtiTest\models
*/
class QtiRunnerService extends ConfigurableService implements RunnerService
{
public const SERVICE_ID = 'taoQtiTest/QtiRunnerService';
/**
* @deprecated use SERVICE_ID
*/
public const CONFIG_ID = 'taoQtiTest/QtiRunnerService';
public const TOOL_ITEM_THEME_SWITCHER = 'itemThemeSwitcher';
public const TOOL_ITEM_THEME_SWITCHER_KEY = 'taoQtiTest/runner/plugins/tools/itemThemeSwitcher/itemThemeSwitcher';
private const TIMEOUT_EXCEPTION_CODES = [
AssessmentTestSessionException::ASSESSMENT_TEST_DURATION_OVERFLOW,
AssessmentTestSessionException::TEST_PART_DURATION_OVERFLOW,
AssessmentTestSessionException::ASSESSMENT_SECTION_DURATION_OVERFLOW,
AssessmentTestSessionException::ASSESSMENT_ITEM_DURATION_OVERFLOW,
];
/**
* The test runner config
* @var RunnerConfig
*/
protected $testConfig;
/**
* Use to store retrieved item data, inside the same request
* @var array
*/
private $dataCache = [];
/**
* Get the data folder from a given item definition
* @param string $itemRef - formatted as itemURI|publicFolderURI|privateFolderURI
* @return array the path
* @throws \common_Exception
*/
private function loadItemData($itemRef, $path)
{
$cacheKey = $itemRef . $path;
if (! empty($cacheKey) && isset($this->dataCache[$itemRef . $path])) {
return $this->dataCache[$itemRef . $path];
}
$directoryIds = explode('|', $itemRef);
if (count($directoryIds) < 3) {
if (is_scalar($itemRef)) {
$itemRefInfo = gettype($itemRef) . ': ' . strval($itemRef);
} elseif (is_object($itemRef)) {
$itemRefInfo = gettype($itemRef) . ': ' . get_class($itemRef);
} else {
$itemRefInfo = gettype($itemRef);
}
throw new \common_exception_InconsistentData("The itemRef (value = '${itemRefInfo}') is not formatted correctly.");
}
$itemUri = $directoryIds[0];
$userDataLang = \common_session_SessionManager::getSession()->getDataLanguage();
$directory = \tao_models_classes_service_FileStorage::singleton()->getDirectoryById($directoryIds[2]);
if ($directory->has($userDataLang)) {
$lang = $userDataLang;
} elseif ($directory->has(DEFAULT_LANG)) {
\common_Logger::d(
$userDataLang . ' is not part of compilation directory for item : ' . $itemUri . ' use ' . DEFAULT_LANG
);
$lang = DEFAULT_LANG;
} else {
throw new \common_Exception(
'item : ' . $itemUri . 'is neither compiled in ' . $userDataLang . ' nor in ' . DEFAULT_LANG
);
}
try {
$content = $directory->read($lang . DIRECTORY_SEPARATOR . $path);
/** @var ItemAssetsReplacement $assetService */
$assetService = $this->getServiceManager()->get(ItemAssetsReplacement::SERVICE_ID);
$jsonContent = json_decode($content, true);
$jsonAssets = [];
if (isset($jsonContent['assets'])) {
foreach ($jsonContent['assets'] as $type => $assets) {
foreach ($assets as $key => $asset) {
$jsonAssets[$type][$key] = $assetService->postProcessAssets($asset);
}
}
$jsonContent["assets"] = $jsonAssets;
}
$this->dataCache[$cacheKey] = $jsonContent;
return $this->dataCache[$cacheKey];
} catch (FileNotFoundException $e) {
throw new \tao_models_classes_FileNotFoundException(
$path . ' for item reference ' . $itemRef
);
}
}
/**
* Gets the test session for a particular delivery execution
*
* This method is called before each action (moveNext, moveBack, pause, ...) call.
*
* @param string $testDefinitionUri The URI of the test
* @param string $testCompilationUri The URI of the compiled delivery
* @param string $testExecutionUri The URI of the delivery execution
* @param string $userUri User identifier. If null current user will be used
* @return QtiRunnerServiceContext
* @throws \common_Exception
*/
public function getServiceContext($testDefinitionUri, $testCompilationUri, $testExecutionUri, $userUri = null)
{
// create a service context based on the provided URI
// initialize the test session and related objects
$serviceContext = new QtiRunnerServiceContext($testDefinitionUri, $testCompilationUri, $testExecutionUri);
$this->propagate($serviceContext);
$serviceContext->setTestConfig($this->getTestConfig());
$serviceContext->setUserUri($userUri);
$sessionService = $this->getServiceManager()->get(TestSessionService::SERVICE_ID);
$sessionService->registerTestSession(
$serviceContext->getTestSession(),
$serviceContext->getStorage(),
$serviceContext->getCompilationDirectory()
);
return $serviceContext;
}
/**
* Checks the created context, then initializes it.
* @param RunnerServiceContext $context
* @return RunnerServiceContext
* @throws \common_Exception
*/
public function initServiceContext(RunnerServiceContext $context)
{
// will throw exception if the test session is not valid
$this->check($context);
// starts the context
$context->init();
return $context;
}
/**
* Persists the AssessmentTestSession into binary data.
* @param QtiRunnerServiceContext $context
*/
public function persist(QtiRunnerServiceContext $context)
{
$testSession = $context->getTestSession();
$sessionId = $testSession->getSessionId();
\common_Logger::d("Persisting QTI Assessment Test Session '${sessionId}'...");
$context->getStorage()->persist($testSession);
if ($this->isTerminated($context)) {
/** @var StorageManager $storageManager */
$storageManager = $this->getServiceManager()->get(StorageManager::SERVICE_ID);
$storageManager->persist();
$userId = \common_session_SessionManager::getSession()->getUser()->getIdentifier();
$eventManager = $this->getServiceManager()->get(EventManager::SERVICE_ID);
$eventManager->trigger(new AfterAssessmentTestSessionClosedEvent($testSession, $userId));
}
}
/**
* Initializes the delivery execution session
*
* This method is called whenever a candidate enters the test. This includes
*
* * Newly launched/instantiated test session.
* * The candidate refreshes the client (F5).
* * Resumed test sessions.
*
* @param RunnerServiceContext $context
* @return boolean
* @throws \common_Exception
*/
public function init(RunnerServiceContext $context)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'init',
0,
QtiRunnerServiceContext::class,
$context
);
}
/* @var TestSession $session */
$session = $context->getTestSession();
// code borrowed from the previous implementation, but the reset timers option has been discarded
if ($session->getState() === AssessmentTestSessionState::INITIAL) {
// The test has just been instantiated.
$session->beginTestSession();
$event = new TestInitEvent($session);
$this->getServiceManager()->get(EventManager::SERVICE_ID)->trigger($event);
\common_Logger::i(sprintf('Assessment Test Session begun. Session id: %s', $session->getSessionId()));
if ($context->isAdaptive()) {
\common_Logger::t("Very first item is adaptive.");
$nextCatItemId = $context->selectAdaptiveNextItem();
$context->persistCurrentCatItemId($nextCatItemId);
$context->persistSeenCatItemIds($nextCatItemId);
}
} elseif ($session->getState() === AssessmentTestSessionState::SUSPENDED) {
$session->resume();
}
$session->initItemTimer();
if ($session->isTimeout() === false) {
TestRunnerUtils::beginCandidateInteraction($session);
}
$this->getServiceManager()->get(ExtendedStateService::SERVICE_ID)->clearEvents($session->getSessionId());
return true;
}
/**
* Gets the test runner config
* @return RunnerConfig
* @throws \common_ext_ExtensionException
*/
public function getTestConfig()
{
if (is_null($this->testConfig)) {
$this->testConfig = $this->getServiceManager()->get(QtiRunnerConfig::SERVICE_ID);
}
return $this->testConfig;
}
/**
* Gets the test definition data
*
* @deprecated the testData is not necessary anymore
* if the config is given directly to the test runner configuration
*
* @param RunnerServiceContext $context
* @return array
* @throws \common_Exception
*/
public function getTestData(RunnerServiceContext $context)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'getTestData',
0,
QtiRunnerServiceContext::class,
$context
);
}
$testDefinition = $context->getTestDefinition();
$response['title'] = $testDefinition->getTitle();
$response['identifier'] = $testDefinition->getIdentifier();
$response['className'] = $testDefinition->getQtiClassName();
$response['toolName'] = $testDefinition->getToolName();
$response['exclusivelyLinear'] = $testDefinition->isExclusivelyLinear();
$response['hasTimeLimits'] = $testDefinition->hasTimeLimits();
//states that can be found in the context
$response['states'] = [
'initial' => AssessmentTestSessionState::INITIAL,
'interacting' => AssessmentTestSessionState::INTERACTING,
'modalFeedback' => AssessmentTestSessionState::MODAL_FEEDBACK,
'suspended' => AssessmentTestSessionState::SUSPENDED,
'closed' => AssessmentTestSessionState::CLOSED
];
$response['itemStates'] = [
'initial' => AssessmentItemSessionState::INITIAL,
'interacting' => AssessmentItemSessionState::INTERACTING,
'modalFeedback' => AssessmentItemSessionState::MODAL_FEEDBACK,
'suspended' => AssessmentItemSessionState::SUSPENDED,
'closed' => AssessmentItemSessionState::CLOSED,
'solution' => AssessmentItemSessionState::SOLUTION,
'review' => AssessmentItemSessionState::REVIEW,
'notSelected' => AssessmentItemSessionState::NOT_SELECTED
];
$timeLimits = $testDefinition->getTimeLimits();
if ($timeLimits) {
if ($timeLimits->hasMinTime()) {
$response['timeLimits']['minTime'] = [
'duration' => TestRunnerUtils::getDurationWithMicroseconds($timeLimits->getMinTime()),
'iso' => $timeLimits->getMinTime()->__toString(),
];
}
if ($timeLimits->hasMaxTime()) {
$response['timeLimits']['maxTime'] = [
'duration' => TestRunnerUtils::getDurationWithMicroseconds($timeLimits->getMaxTime()),
'iso' => $timeLimits->getMaxTime()->__toString(),
];
}
}
$response['config'] = $this->getTestConfig()->getConfig();
if ($this->isThemeSwitcherEnabled()) {
$themeSwitcherPlugin = [
self::TOOL_ITEM_THEME_SWITCHER => [
"activeNamespace" => $this->getCurrentThemeId(),
],
];
$response["config"]["plugins"] = array_merge($response["config"]["plugins"], $themeSwitcherPlugin);
}
return $response;
}
/**
* Gets the test context object
* @param RunnerServiceContext $context
* @return array
* @throws \common_Exception
*/
public function getTestContext(RunnerServiceContext $context)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'getTestContext',
0,
QtiRunnerServiceContext::class,
$context
);
}
/* @var TestSession $session */
$session = $context->getTestSession();
// The state of the test session.
$response['state'] = $session->getState();
// Default values for the test session context.
$response['navigationMode'] = null;
$response['submissionMode'] = null;
$response['remainingAttempts'] = 0;
$response['isAdaptive'] = false;
// Context of interacting test
if ($session->getState() === AssessmentTestSessionState::INTERACTING) {
$config = $this->getTestConfig();
$route = $session->getRoute();
$currentItem = $route->current();
$itemSession = $session->getCurrentAssessmentItemSession();
$itemRef = $context->getCurrentAssessmentItemRef();
$reviewConfig = $config->getConfigValue('review');
$displaySubsectionTitle = isset($reviewConfig['displaySubsectionTitle']) ? (bool) $reviewConfig['displaySubsectionTitle'] : true;
if ($displaySubsectionTitle) {
$currentSection = $session->getCurrentAssessmentSection();
} else {
$sections = $currentItem->getAssessmentSections()->getArrayCopy();
$currentSection = $sections[0];
}
$testOptions = $config->getTestOptions($context);
// The navigation mode.
$response['navigationMode'] = $session->getCurrentNavigationMode();
$response['isLinear'] = $response['navigationMode'] == NavigationMode::LINEAR;
// The submission mode.
$response['submissionMode'] = $session->getCurrentSubmissionMode();
// The number of remaining attempts for the current item.
$response['remainingAttempts'] = $session->getCurrentRemainingAttempts();
// Whether or not the current step is timed out.
$response['isTimeout'] = $session->isTimeout();
// The identifier of the current item.
$response['itemIdentifier'] = $itemRef->getIdentifier();
// The number of current attempt (1 for the first time ...)
$response['attempt'] = ($context->isAdaptive()) ? $context->getCatAttempts($response['itemIdentifier']) + 1 : $itemSession['numAttempts']->getValue();
// The state of the current AssessmentTestSession.
$response['itemSessionState'] = $itemSession->getState();
// Whether the current item is adaptive.
$response['isAdaptive'] = $session->isCurrentAssessmentItemAdaptive();
// Whether the current section is adaptive.
$response['isCatAdaptive'] = $context->isAdaptive();
// Whether the test map must be updated.
// TODO: detect if the map need to be updated and set the flag
$response['needMapUpdate'] = false;
// Whether the current item is the very last one of the test.
$response['isLast'] = (!$context->isAdaptive()) ? $route->isLast() : false;
// The current position in the route.
$response['itemPosition'] = $context->getCurrentPosition();
// The current item flagged state
$response['itemFlagged'] = TestRunnerUtils::getItemFlag($session, $response['itemPosition'], $context);
// The current item answered state
$response['itemAnswered'] = $this->isItemCompleted($context, $currentItem, $itemSession);
// Time constraints.
$response['timeConstraints'] = $this->buildTimeConstraints($context);
// Test Part title.
$response['testPartId'] = $session->getCurrentTestPart()->getIdentifier();
// Current Section title.
$response['sectionId'] = $currentSection->getIdentifier();
$response['sectionTitle'] = $currentSection->getTitle();
// Number of items composing the test session.
$response['numberItems'] = $route->count();
// Number of items completed during the test session.
$response['numberCompleted'] = TestRunnerUtils::testCompletion($session);
// Number of items presented during the test session.
$response['numberPresented'] = $session->numberPresented();
// Whether or not the progress of the test can be inferred.
$response['considerProgress'] = TestRunnerUtils::considerProgress($session, $context->getTestMeta(), $config->getConfig());
// Whether or not the deepest current section is visible.
$response['isDeepestSectionVisible'] = $currentSection->isVisible();
// If the candidate is allowed to move backward e.g. first item of the test.
$response['canMoveBackward'] = $context->canMoveBackward();
//Number of rubric blocks
$response['numberRubrics'] = count($currentItem->getRubricBlockRefs());
//add rubic blocks
if ($response['numberRubrics'] > 0) {
$response['rubrics'] = $this->getRubrics($context, $session->getCurrentAssessmentItemRef());
}
//prevent the user from submitting empty (i.e. default or null) responses, feature availability
$response['enableAllowSkipping'] = $config->getConfigValue('enableAllowSkipping');
//contextual value
$response['allowSkipping'] = $testOptions['allowSkipping'];
//prevent the user from submitting an invalid response
$response['enableValidateResponses'] = $config->getConfigValue('enableValidateResponses');
//contextual value
$response['validateResponses'] = $testOptions['validateResponses'];
//does the item has modal feedbacks ?
$response['hasFeedbacks'] = $this->hasFeedbacks($context, $itemRef->getHref());
// append dynamic options
$response['options'] = $testOptions;
}
return $response;
}
/**
* Gets the map of the test items
* @param RunnerServiceContext $context
* @param bool $partial the full testMap or only the current section
* @return array
* @throws \common_Exception
*/
public function getTestMap(RunnerServiceContext $context, $partial = false)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'getTestMap',
0,
QtiRunnerServiceContext::class,
$context
);
}
$mapService = $this->getServiceLocator()->get(QtiRunnerMap::SERVICE_ID);
if ($partial) {
return $mapService->getScopedMap($context, $this->getTestConfig());
}
return $mapService->getMap($context, $this->getTestConfig());
}
/**
* Gets the rubrics related to the current session state.
* @param RunnerServiceContext $context
* @param AssessmentItemRef $itemRef (optional) otherwise use the current
* @return mixed
* @throws \common_Exception
*/
public function getRubrics(RunnerServiceContext $context, AssessmentItemRef $itemRef = null)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'getRubrics',
0,
QtiRunnerServiceContext::class,
$context
);
}
$rubricHelper = $this->getServiceLocator()->get(QtiRunnerRubric::SERVICE_ID);
return $rubricHelper->getRubrics($context, $itemRef);
}
/**
* Gets AssessmentItemRef's Href by AssessmentItemRef Identifier.
* @param RunnerServiceContext $context
* @param string $itemRef
* @return string
*/
public function getItemHref(RunnerServiceContext $context, $itemRef)
{
$mapService = $this->getServiceLocator()->get(QtiRunnerMap::SERVICE_ID);
return $mapService->getItemHref($context, $itemRef);
}
/**
* Gets definition data of a particular item
* @param RunnerServiceContext $context
* @param $itemRef
* @return mixed
* @throws \common_Exception
*/
public function getItemData(RunnerServiceContext $context, $itemRef)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'getItemData',
0,
QtiRunnerServiceContext::class,
$context
);
}
return $this->loadItemData($itemRef, QtiJsonItemCompiler::ITEM_FILE_NAME);
}
/**
* Gets the state identifier for a particular item
* @param QtiRunnerServiceContext $context
* @param string $itemRef The item identifier
* @return string The state identifier
*/
protected function getStateId(QtiRunnerServiceContext $context, $itemRef)
{
return $this->buildStorageItemKey($context->getTestExecutionUri(), $itemRef);
}
/**
* @param string $deliveryExecutionUri
* @param string $itemRef
* @return string
*/
private function buildStorageItemKey($deliveryExecutionUri, $itemRef)
{
return $deliveryExecutionUri . $itemRef;
}
/**
* Gets the state of a particular item
* @param RunnerServiceContext $context
* @param string $itemRef
* @return array|null
* @throws \common_Exception
*/
public function getItemState(RunnerServiceContext $context, $itemRef)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'getItemState',
0,
QtiRunnerServiceContext::class,
$context
);
}
$serviceService = $this->getServiceManager()->get(StorageManager::SERVICE_ID);
$userUri = \common_session_SessionManager::getSession()->getUserUri();
$stateId = $this->getStateId($context, $itemRef);
$state = is_null($userUri) ? null : $serviceService->get($userUri, $stateId);
if ($state) {
$state = json_decode($state, true);
if (is_null($state)) {
throw new \common_exception_InconsistentData('Unable to decode the state for the item ' . $itemRef);
}
}
return $state;
}
/**
* Sets the state of a particular item
* @param RunnerServiceContext $context
* @param $itemRef
* @param $state
* @return boolean
* @throws \common_Exception
*/
public function setItemState(RunnerServiceContext $context, $itemRef, $state)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'setItemState',
0,
QtiRunnerServiceContext::class,
$context
);
}
$serviceService = $this->getServiceManager()->get(StorageManager::SERVICE_ID);
$userUri = \common_session_SessionManager::getSession()->getUserUri();
$stateId = $this->getStateId($context, $itemRef);
if (!isset($state)) {
$state = '';
}
return is_null($userUri) ? false : $serviceService->set($userUri, $stateId, json_encode($state));
}
/**
* @param RunnerServiceContext $context
* @param $toolStates
* @throws \oat\oatbox\service\exception\InvalidServiceManagerException
*/
public function setToolsStates(RunnerServiceContext $context, $toolStates)
{
if ($context instanceof QtiRunnerServiceContext && is_array($toolStates)) {
/** @var ToolsStateStorage $toolsStateStorage */
$toolsStateStorage = $this->getServiceLocator()->get(ToolsStateStorage::SERVICE_ID);
$toolsStateStorage->storeStates($context->getTestExecutionUri(), $toolStates);
}
}
/**
* @param RunnerServiceContext $context
* @return array
* @throws \oat\oatbox\service\exception\InvalidServiceManagerException
* @throws \common_ext_ExtensionException
*/
public function getToolsStates(RunnerServiceContext $context)
{
$toolsStates = [];
// add those tools missing from the storage but presented on the config
$toolsEnabled = $this->getTestConfig()->getConfigValue('toolStateServerStorage');
if (count($toolsEnabled) === 0) {
return [];
}
if ($context instanceof QtiRunnerServiceContext) {
/** @var ToolsStateStorage $toolsStateStorage */
$toolsStateStorage = $this->getServiceLocator()->get(ToolsStateStorage::SERVICE_ID);
$toolsStates = $toolsStateStorage->getStates($context->getTestExecutionUri());
}
foreach ($toolsEnabled as $toolEnabled) {
if (!array_key_exists($toolEnabled, $toolsStates)) {
$toolsStates[$toolEnabled] = null;
}
}
return $toolsStates;
}
/**
* Parses the responses provided for a particular item
* @param RunnerServiceContext $context
* @param $itemRef
* @param $response
* @return mixed
* @throws \common_Exception
*/
public function parsesItemResponse(RunnerServiceContext $context, $itemRef, $response)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'storeItemResponse',
0,
QtiRunnerServiceContext::class,
$context
);
}
/** @var TestSession $session */
$session = $context->getTestSession();
$currentItem = $context->getCurrentAssessmentItemRef();
$responses = new State();
if ($currentItem === false) {
$msg = "Trying to store item variables but the state of the test session is INITIAL or CLOSED.\n";
$msg .= "Session state value: " . $session->getState() . "\n";
$msg .= "Session ID: " . $session->getSessionId() . "\n";
$msg .= "JSON Payload: " . mb_substr(json_encode($response), 0, 1000);
\common_Logger::e($msg);
}
$filler = new \taoQtiCommon_helpers_PciVariableFiller(
$currentItem,
$this->getServiceManager()->get(QtiFlysystemFileManager::SERVICE_ID)
);
if (is_array($response)) {
foreach ($response as $id => $responseData) {
try {
$var = $filler->fill($id, $responseData);
// Do not take into account QTI File placeholders.
if (\taoQtiCommon_helpers_Utils::isQtiFilePlaceHolder($var) === false) {
$responses->setVariable($var);
}
} catch (\OutOfRangeException $e) {
\common_Logger::d("Could not convert client-side value for variable '${id}'.");
} catch (\OutOfBoundsException $e) {
\common_Logger::d("Could not find variable with identifier '${id}' in current item.");
}
}
} else {
\common_Logger::e('Invalid json payload');
}
return $responses;
}
/**
* Checks if the provided responses are empty
* @param RunnerServiceContext $context
* @param $responses
* @return mixed
* @throws \common_Exception
*/
public function emptyResponse(RunnerServiceContext $context, $responses)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'storeItemResponse',
0,
QtiRunnerServiceContext::class,
$context
);
}
$similar = 0;
/** @var ResponseVariable $responseVariable */
foreach ($responses as $responseVariable) {
$value = $responseVariable->getValue();
$default = $responseVariable->getDefaultValue();
// Similar to default ?
if (TestRunnerUtils::isQtiValueNull($value) === true) {
if (TestRunnerUtils::isQtiValueNull($default) === true) {
$similar++;
}
} elseif ($value->equals($default) === true) {
$similar++;
}
}
$respCount = count($responses);
return $respCount > 0 && $similar === $respCount;
}
/**
* Stores the response of a particular item
* @param RunnerServiceContext $context
* @param $itemRef
* @param $responses
* @return boolean
* @throws \common_Exception
*/
public function storeItemResponse(RunnerServiceContext $context, $itemRef, $responses)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'storeItemResponse',
0,
QtiRunnerServiceContext::class,
$context
);
}
$session = $this->getCurrentAssessmentSession($context);
try {
\common_Logger::t('Responses sent from the client-side. The Response Processing will take place.');
if ($context->isAdaptive()) {
$session->beginItemSession();
$session->beginAttempt();
$session->endAttempt($responses);
$assessmentItem = $session->getAssessmentItem();
$assessmentItemIdentifier = $assessmentItem->getIdentifier();
$score = $session->getVariable('SCORE');
$output = $context->getLastCatItemOutput();
if ($score !== null) {
$output[$assessmentItemIdentifier] = new ItemResult(
$assessmentItemIdentifier,
new ResultVariable(
$score->getIdentifier(),
BaseType::getNameByConstant($score->getBaseType()),
$score->getValue()->getValue(),
null,
$score->getCardinality()
),
microtime(true)
);
} else {
\common_Logger::i("No 'SCORE' outcome variable for item '${assessmentItemIdentifier}' involved in an adaptive section.");
}
$context->persistLastCatItemOutput($output);
// Send results to TAO Results.
$resultTransmitter = new \taoQtiCommon_helpers_ResultTransmitter($context->getSessionManager()->getResultServer());
$hrefParts = explode('|', $assessmentItem->getHref());
$sessionId = $context->getTestSession()->getSessionId();
$itemIdentifier = $assessmentItem->getIdentifier();
// Deal with attempts.
$attempt = $context->getCatAttempts($itemIdentifier);
$transmissionId = "${sessionId}.${itemIdentifier}.${attempt}";
$attempt++;
foreach ($session->getAllVariables() as $var) {
if ($var->getIdentifier() === 'numAttempts') {
$var->setValue(new \qtism\common\datatypes\QtiInteger($attempt));
}
$variables[] = $var;
}
$resultTransmitter->transmitItemVariable($variables, $transmissionId, $hrefParts[0], $hrefParts[2]);
$context->persistCatAttempts($itemIdentifier, $attempt);
$context->getTestSession()->endAttempt(new State(), true);
} else {
// Non adaptive case.
$session->endAttempt($responses, true);
}
return true;
} catch (AssessmentTestSessionException $e) {
\common_Logger::w($e->getMessage());
return false;
}
}
/**
* Should we display feedbacks
* @param RunnerServiceContext $context
* @return boolean
* @throws \common_exception_InvalidArgumentType
*/
public function displayFeedbacks(RunnerServiceContext $context)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'displayFeedbacks',
0,
QtiRunnerServiceContext::class,
$context
);
}
/* @var TestSession $session */
$session = $context->getTestSession();
return $session->getCurrentSubmissionMode() !== SubmissionMode::SIMULTANEOUS;
}
/**
* Get feedback definitions
*
* @param RunnerServiceContext $context
* @param string $itemRef the item reference
* @return array the feedbacks data
* @throws \common_Exception
* @throws \common_exception_InvalidArgumentType
* @deprecated since version 30.7.0, to be removed in 31.0.0. Use getItemVariableElementsData() instead
*/
public function getFeedbacks(RunnerServiceContext $context, $itemRef)
{
return $this->getItemVariableElementsData($context, $itemRef);
}
/**
* @param RunnerServiceContext $context
* @param $itemRef
* @return array
* @throws \common_Exception
* @throws \common_exception_InvalidArgumentType
*/
public function getItemVariableElementsData(RunnerServiceContext $context, $itemRef)
{
$this->assertQtiRunnerServiceContext($context);
return $this->loadItemData($itemRef, QtiJsonItemCompiler::VAR_ELT_FILE_NAME);
}
/**
* Does the given item has feedbacks
*
* @param RunnerServiceContext $context
* @param string $itemRef the item reference
* @return boolean
* @throws \common_Exception
* @throws \common_exception_InconsistentData
* @throws \common_exception_InvalidArgumentType
* @throws \tao_models_classes_FileNotFoundException
*/
public function hasFeedbacks(RunnerServiceContext $context, $itemRef)
{
$hasFeedbacks = false;
$displayFeedbacks = $this->displayFeedbacks($context);
if ($displayFeedbacks) {
$feedbacks = $this->getFeedbacks($context, $itemRef);
foreach ($feedbacks as $entry) {
if (isset($entry['feedbackRules'])) {
if (count($entry['feedbackRules']) > 0) {
$hasFeedbacks = true;
}
break;
}
}
}
return $hasFeedbacks;
}
/**
* Should we display feedbacks
* @param RunnerServiceContext $context
* @return array the item session
* @throws \common_exception_InvalidArgumentType
*/
public function getItemSession(RunnerServiceContext $context)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'getItemSession',
0,
QtiRunnerServiceContext::class,
$context
);
}
/* @var TestSession $session */
$session = $context->getTestSession();
$currentItem = $session->getCurrentAssessmentItemRef();
$currentOccurrence = $session->getCurrentAssessmentItemRefOccurence();
$itemSession = $session->getAssessmentItemSessionStore()->getAssessmentItemSession($currentItem, $currentOccurrence);
$stateOutput = new \taoQtiCommon_helpers_PciStateOutput();
foreach ($itemSession->getAllVariables() as $var) {
$stateOutput->addVariable($var);
}
$output = $stateOutput->getOutput();
// The current item answered state
$route = $session->getRoute();
$position = $route->getPosition();
$output['itemAnswered'] = TestRunnerUtils::isItemCompleted($route->getRouteItemAt($position), $itemSession);
return $output;
}
/**
* Moves the current position to the provided scoped reference.
* @param RunnerServiceContext $context
* @param $direction
* @param $scope
* @param $ref
* @return boolean
* @throws \common_Exception
*/
public function move(RunnerServiceContext $context, $direction, $scope, $ref)
{
$result = true;
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'move',
0,
QtiRunnerServiceContext::class,
$context
);
}
try {
$result = QtiRunnerNavigation::move($direction, $scope, $context, $ref);
} catch (AssessmentTestSessionException $e) {
} finally {
if ($result && (!isset($e) || in_array($e->getCode(), self::TIMEOUT_EXCEPTION_CODES, true))) {
$this->continueInteraction($context);
}
}
return $result;
}
/**
* Skips the current position to the provided scoped reference
* @param RunnerServiceContext $context
* @param $scope
* @param $ref
* @return boolean
* @throws \common_Exception
*/
public function skip(RunnerServiceContext $context, $scope, $ref)
{
return $this->move($context, 'skip', $scope, $ref);
}
/**
* Handles a test timeout
* @param RunnerServiceContext $context
* @param $scope
* @param $ref
* @param $late
* @return boolean
* @throws \common_Exception
*/
public function timeout(RunnerServiceContext $context, $scope, $ref, $late = false)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'timeout',
0,
QtiRunnerServiceContext::class,
$context
);
}
/* @var TestSession $session */
$session = $context->getTestSession();
if ($context->isAdaptive()) {
\common_Logger::t("Select next item before timeout");
$context->selectAdaptiveNextItem();
}
try {
$session->closeTimer($ref, $scope);
if ($late) {
if ($scope == 'assessmentTest') {
$code = AssessmentTestSessionException::ASSESSMENT_TEST_DURATION_OVERFLOW;
} elseif ($scope == 'testPart') {
$code = AssessmentTestSessionException::TEST_PART_DURATION_OVERFLOW;
} elseif ($scope == 'assessmentSection') {
$code = AssessmentTestSessionException::ASSESSMENT_SECTION_DURATION_OVERFLOW;
} else {
$code = AssessmentTestSessionException::ASSESSMENT_ITEM_DURATION_OVERFLOW;
}
throw new AssessmentTestSessionException("Maximum duration of ${scope} '${ref}' not respected.", $code);
} else {
$session->checkTimeLimits(false, true, false);
}
} catch (AssessmentTestSessionException $e) {
$this->onTimeout($context, $e);
}
return true;
}
/**
* Exits the test before its end
* @param RunnerServiceContext $context
* @return boolean
* @throws \common_Exception
*/
public function exitTest(RunnerServiceContext $context)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'exitTest',
0,
QtiRunnerServiceContext::class,
$context
);
}
/* @var TestSession $session */
$session = $context->getTestSession();
$sessionId = $session->getSessionId();
\common_Logger::i("The user has requested termination of the test session '{$sessionId}'");
if ($context->isAdaptive()) {
\common_Logger::t("Select next item before test exit");
$context->selectAdaptiveNextItem();
}
$event = new TestExitEvent($session);
$this->getServiceManager()->get(EventManager::SERVICE_ID)->trigger($event);
$session->endTestSession();
$this->finish($context, $this->getStateAfterExit());
return true;
}
/**
* Finishes the test
* @param RunnerServiceContext $context
* @param string $finalState
* @return boolean
* @throws \common_Exception
*/
public function finish(RunnerServiceContext $context, $finalState = DeliveryExecution::STATE_FINISHED)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'finish',
0,
QtiRunnerServiceContext::class,
$context
);
}
$executionUri = $context->getTestExecutionUri();
$userUri = \common_session_SessionManager::getSession()->getUserUri();
$executionService = ServiceProxy::singleton();
$deliveryExecution = $executionService->getDeliveryExecution($executionUri);
if ($deliveryExecution->getUserIdentifier() == $userUri) {
\common_Logger::i("Finishing the delivery execution {$executionUri}");
$result = $deliveryExecution->setState($finalState);
} else {
\common_Logger::w("Non owner {$userUri} tried to finish deliveryExecution {$executionUri}");
$result = false;
}
$this->getServiceManager()->get(ExtendedStateService::SERVICE_ID)->clearEvents($executionUri);
return $result;
}
/**
* Sets the test to paused state
* @param RunnerServiceContext $context
* @return boolean
* @throws \common_Exception
*/
public function pause(RunnerServiceContext $context)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'pause',
0,
QtiRunnerServiceContext::class,
$context
);
}
$context->getTestSession()->suspend();
$this->persist($context);
return true;
}
/**
* Resumes the test from paused state
* @param RunnerServiceContext $context
* @return boolean
* @throws \common_Exception
*/
public function resume(RunnerServiceContext $context)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'resume',
0,
QtiRunnerServiceContext::class,
$context
);
}
$context->getTestSession()->resume();
$this->persist($context);
return true;
}
/**
* Checks if the test is still valid
* @param RunnerServiceContext $context
* @return boolean
* @throws \common_Exception
* @throws QtiRunnerClosedException
*/
public function check(RunnerServiceContext $context)
{
$state = $context->getTestSession()->getState();
if ($state == AssessmentTestSessionState::CLOSED) {
throw new QtiRunnerClosedException();
}
return true;
}
/**
* Checks if an item has been completed
* @param RunnerServiceContext $context
* @param RouteItem $routeItem
* @param AssessmentItemSession $itemSession
* @param bool $partially (optional) Whether or not consider partially responded sessions as responded.
* @return bool
* @throws \common_Exception
*/
public function isItemCompleted(RunnerServiceContext $context, $routeItem, $itemSession, $partially = true)
{
if ($context instanceof QtiRunnerServiceContext && $context->isAdaptive()) {
$itemIdentifier = $context->getCurrentAssessmentItemRef()->getIdentifier();
$itemState = $this->getItemState($context, $itemIdentifier);
if ($itemState !== null) {
// as the item comes from a CAT section, it is simpler to load the responses from the state
$itemResponse = [];
foreach ($itemState as $key => $value) {
if (isset($value['response'])) {
$itemResponse[$key] = $value['response'];
}
}
$responses = $this->parsesItemResponse($context, $itemIdentifier, $itemResponse);
// fork of AssessmentItemSession::isResponded()
$excludedResponseVariables = ['numAttempts', 'duration'];
foreach ($responses as $var) {
if ($var instanceof ResponseVariable && in_array($var->getIdentifier(), $excludedResponseVariables) === false) {
$value = $var->getValue();
$defaultValue = $var->getDefaultValue();
if (Utils::isNull($value) === true) {
if (Utils::isNull($defaultValue) === (($partially) ? false : true)) {
return (($partially) ? true : false);
}
} else {
if ($value->equals($defaultValue) === (($partially) ? false : true)) {
return (($partially) ? true : false);
}
}
}
}
}
return (($partially) ? false : true);
} else {
return TestRunnerUtils::isItemCompleted($routeItem, $itemSession, $partially);
}
}
/**
* Checks if the test is in paused state
* @param RunnerServiceContext $context
* @return boolean
*/
public function isPaused(RunnerServiceContext $context)
{
return $context->getTestSession()->getState() == AssessmentTestSessionState::SUSPENDED;
}
/**
* Checks if the test is in terminated state
* @param RunnerServiceContext $context
* @return boolean
*/
public function isTerminated(RunnerServiceContext $context)
{
return $context->getTestSession()->getState() == AssessmentTestSessionState::CLOSED;
}
/**
* Get the base url to the item public directory
* @param RunnerServiceContext $context
* @param $itemRef
* @return string
* @throws \common_Exception
* @throws \common_exception_Error
* @throws \common_exception_InvalidArgumentType
*/
public function getItemPublicUrl(RunnerServiceContext $context, $itemRef)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'getItemPublicUrl',
0,
QtiRunnerServiceContext::class,
$context
);
}
$directoryIds = explode('|', $itemRef);
$userDataLang = \common_session_SessionManager::getSession()->getDataLanguage();
$directory = \tao_models_classes_service_FileStorage::singleton()->getDirectoryById($directoryIds[1]);
// do fallback in case userlanguage is not default language
if ($userDataLang != DEFAULT_LANG && !$directory->has($userDataLang) && $directory->has(DEFAULT_LANG)) {
$userDataLang = DEFAULT_LANG;
}
return $directory->getPublicAccessUrl() . $userDataLang . '/';
}
/**
* Comment the test
* @param RunnerServiceContext $context
* @param string $comment
* @return bool
*/
public function comment(RunnerServiceContext $context, $comment)
{
// prepare transmission Id for result server.
$testSession = $context->getTestSession();
$item = $testSession->getCurrentAssessmentItemRef()->getIdentifier();
$occurrence = $testSession->getCurrentAssessmentItemRefOccurence();
$sessionId = $testSession->getSessionId();
$transmissionId = "${sessionId}.${item}.${occurrence}";
/** @var DeliveryServerService $deliveryServerService */
$deliveryServerService = $this->getServiceManager()->get(DeliveryServerService::SERVICE_ID);
$resultStore = $deliveryServerService->getResultStoreWrapper($sessionId);
$transmitter = new \taoQtiCommon_helpers_ResultTransmitter($resultStore);
// build variable and send it.
$itemUri = TestRunnerUtils::getCurrentItemUri($testSession);
$testUri = $testSession->getTest()->getUri();
$variable = new ResponseVariable('comment', Cardinality::SINGLE, BaseType::STRING, new QtismString($comment));
$transmitter->transmitItemVariable($variable, $transmissionId, $itemUri, $testUri);
return true;
}
/**
* Continue the test interaction if possible
* @param RunnerServiceContext $context
* @return bool
*/
protected function continueInteraction(RunnerServiceContext $context)
{
$continue = false;
/* @var TestSession $session */
$session = $context->getTestSession();
if ($session->isRunning() === true && $session->isTimeout() === false) {
$event = new QtiContinueInteractionEvent($context, $this);
$this->getServiceManager()->get(EventManager::SERVICE_ID)->trigger($event);
TestRunnerUtils::beginCandidateInteraction($session);
$continue = true;
} else {
$this->finish($context);
}
return $continue;
}
/**
* Stuff to be undertaken when the Assessment Item presented to the candidate
* times out.
*
* @param RunnerServiceContext $context
* @param AssessmentTestSessionException $timeOutException The AssessmentTestSessionException object thrown to indicate the timeout.
*/
protected function onTimeout(RunnerServiceContext $context, AssessmentTestSessionException $timeOutException)
{
/* @var TestSession $session */
$session = $context->getTestSession();
$event = new TestTimeoutEvent($session, $timeOutException->getCode(), true);
$this->getServiceManager()->get(EventManager::SERVICE_ID)->trigger($event);
$isLinear = $session->getCurrentNavigationMode() === NavigationMode::LINEAR;
switch ($timeOutException->getCode()) {
case AssessmentTestSessionException::ASSESSMENT_TEST_DURATION_OVERFLOW:
\common_Logger::i('TIMEOUT: closing the assessment test session');
$session->moveThroughAndEndTestSession();
break;
case AssessmentTestSessionException::TEST_PART_DURATION_OVERFLOW:
if ($isLinear) {
\common_Logger::i('TIMEOUT: moving to the next test part');
$session->moveNextTestPart();
} else {
\common_Logger::i('TIMEOUT: closing the assessment test part');
$session->closeTestPart();
}
break;
case AssessmentTestSessionException::ASSESSMENT_SECTION_DURATION_OVERFLOW:
if ($isLinear) {
\common_Logger::i('TIMEOUT: moving to the next assessment section');
$session->moveNextAssessmentSection();
} else {
\common_Logger::i('TIMEOUT: closing the assessment section session');
$session->closeAssessmentSection();
}
break;
case AssessmentTestSessionException::ASSESSMENT_ITEM_DURATION_OVERFLOW:
if ($isLinear) {
\common_Logger::i('TIMEOUT: moving to the next item');
$session->moveNextAssessmentItem();
} else {
\common_Logger::i('TIMEOUT: closing the assessment item session');
$session->closeAssessmentItem();
}
break;
}
$event = new TestTimeoutEvent($session, $timeOutException->getCode(), false);
$this->getServiceManager()->get(EventManager::SERVICE_ID)->trigger($event);
$this->continueInteraction($context);
}
/**
* Build an array where each cell represent a time constraint (a.k.a. time limits)
* in force. Each cell is actually an array with two keys:
*
* * 'source': The identifier of the QTI component emitting the constraint (e.g. AssessmentTest, TestPart, AssessmentSection, AssessmentItemRef).
* * 'seconds': The number of remaining seconds until it times out.
*
* @param RunnerServiceContext $context
* @return array
*/
protected function buildTimeConstraints(RunnerServiceContext $context)
{
$constraints = [];
$session = $context->getTestSession();
foreach ($session->getRegularTimeConstraints() as $constraint) {
if ($constraint->getMaximumRemainingTime() != false || $constraint->getMinimumRemainingTime() != false) {
$constraints[] = $constraint;
}
}
return $constraints;
}
/**
* Stores trace variable related to an item, a test or a section
*
* @param RunnerServiceContext $context
* @param $itemUri
* @param $variableIdentifier
* @param $variableValue
* @return boolean
* @throws \common_Exception
*/
public function storeTraceVariable(RunnerServiceContext $context, $itemUri, $variableIdentifier, $variableValue)
{
$this->assertQtiRunnerServiceContext($context);
$metaVariable = $this->getTraceVariable($variableIdentifier, $variableValue);
return $this->storeVariable($context, $itemUri, $metaVariable);
}
/**
* Create a trace variable from variable identifier and value
*
* @param $variableIdentifier
* @param $variableValue
* @return \taoResultServer_models_classes_TraceVariable
* @throws \common_exception_InvalidArgumentType
*/
public function getTraceVariable($variableIdentifier, $variableValue)
{
if (!is_string($variableValue) && !is_numeric($variableValue)) {
$variableValue = json_encode($variableValue);
}
$metaVariable = new \taoResultServer_models_classes_TraceVariable();
$metaVariable->setIdentifier($variableIdentifier);
$metaVariable->setBaseType('string');
$metaVariable->setCardinality(Cardinality::getNameByConstant(Cardinality::SINGLE));
$metaVariable->setTrace($variableValue);
return $metaVariable;
}
/**
* Stores outcome variable related to an item, a test or a section
*
* @param RunnerServiceContext $context
* @param $itemUri
* @param $variableIdentifier
* @param $variableValue
* @return boolean
* @throws \common_Exception
*/
public function storeOutcomeVariable(RunnerServiceContext $context, $itemUri, $variableIdentifier, $variableValue)
{
$this->assertQtiRunnerServiceContext($context);
$metaVariable = $this->getOutcomeVariable($variableIdentifier, $variableValue);
return $this->storeVariable($context, $itemUri, $metaVariable);
}
/**
* Create an outcome variable from variable identifier and value
*
* @param $variableIdentifier
* @param $variableValue
* @return \taoResultServer_models_classes_OutcomeVariable
* @throws \common_exception_InvalidArgumentType
*/
public function getOutcomeVariable($variableIdentifier, $variableValue)
{
if (!is_string($variableValue) && !is_numeric($variableValue)) {
$variableValue = json_encode($variableValue);
}
$metaVariable = new \taoResultServer_models_classes_OutcomeVariable();
$metaVariable->setIdentifier($variableIdentifier);
$metaVariable->setBaseType('string');
$metaVariable->setCardinality(Cardinality::getNameByConstant(Cardinality::SINGLE));
$metaVariable->setValue($variableValue);
return $metaVariable;
}
/**
* Stores response variable related to an item, a test or a section
*
* @param RunnerServiceContext $context
* @param $itemUri
* @param $variableIdentifier
* @param $variableValue
* @return boolean
* @throws \common_Exception
*/
public function storeResponseVariable(RunnerServiceContext $context, $itemUri, $variableIdentifier, $variableValue)
{
$this->assertQtiRunnerServiceContext($context);
$metaVariable = $this->getResponseVariable($variableIdentifier, $variableValue);
return $this->storeVariable($context, $itemUri, $metaVariable);
}
/**
* Create a response variable from variable identifier and value
*
* @param $variableIdentifier
* @param $variableValue
* @return \taoResultServer_models_classes_ResponseVariable
* @throws \common_exception_InvalidArgumentType
*/
public function getResponseVariable($variableIdentifier, $variableValue)
{
if (!is_string($variableValue) && !is_numeric($variableValue)) {
$variableValue = json_encode($variableValue);
}
$metaVariable = new \taoResultServer_models_classes_ResponseVariable();
$metaVariable->setIdentifier($variableIdentifier);
$metaVariable->setBaseType('string');
$metaVariable->setCardinality(Cardinality::getNameByConstant(Cardinality::SINGLE));
$metaVariable->setValue($variableValue);
return $metaVariable;
}
/**
* Store a set of result variables to the result server
*
* @param QtiRunnerServiceContext $context
* @param string $itemUri This is the item uri
* @param \taoResultServer_models_classes_Variable[] $metaVariables
* @param null $itemId The assessment item ref id (optional)
* @return bool
* @throws \Exception
* @throws \common_exception_NotImplemented If the given $itemId is not the current assessment item ref
*/
public function storeVariables(
QtiRunnerServiceContext $context,
$itemUri,
$metaVariables,
$itemId = null
) {
$sessionId = $context->getTestSession()->getSessionId();
/** @var DeliveryServerService $deliveryServerService */
$deliveryServerService = $this->getServiceManager()->get(DeliveryServerService::SERVICE_ID);
$resultStore = $deliveryServerService->getResultStoreWrapper($sessionId);
$testUri = $context->getTestDefinitionUri();
if (!is_null($itemUri)) {
$resultStore->storeItemVariables($testUri, $itemUri, $metaVariables, $this->getTransmissionId($context, $itemId));
} else {
$resultStore->storeTestVariables($testUri, $metaVariables, $sessionId);
}
return true;
}
/**
* Store a result variable to the result server
*
* @param QtiRunnerServiceContext $context
* @param string $itemUri This is the item identifier
* @param \taoResultServer_models_classes_Variable $metaVariable
* @param null $itemId The assessment item ref id (optional)
* @return bool
* @throws \common_exception_NotImplemented If the given $itemId is not the current assessment item ref
*/
protected function storeVariable(
QtiRunnerServiceContext $context,
$itemUri,
\taoResultServer_models_classes_Variable $metaVariable,
$itemId = null
) {
$sessionId = $context->getTestSession()->getSessionId();
$testUri = $context->getTestDefinitionUri();
/** @var DeliveryServerService $deliveryServerService */
$deliveryServerService = $this->getServiceManager()->get(DeliveryServerService::SERVICE_ID);
$resultStore = $deliveryServerService->getResultStoreWrapper($sessionId);
if (!is_null($itemUri)) {
$resultStore->storeItemVariable($testUri, $itemUri, $metaVariable, $this->getTransmissionId($context, $itemId));
} else {
$resultStore->storeTestVariable($testUri, $metaVariable, $sessionId);
}
return true;
}
/**
* Build the transmission based on context and item ref id to store Item variables
*
* @param QtiRunnerServiceContext $context
* @param null $itemId The item ref identifier
* @return string The transmission id to store item variables
* @throws \common_exception_NotImplemented If the given $itemId is not the current assessment item ref
*/
protected function getTransmissionId(QtiRunnerServiceContext $context, $itemId = null)
{
if (is_null($itemId)) {
$itemId = $context->getCurrentAssessmentItemRef();
} elseif ($itemId != $context->getCurrentAssessmentItemRef()) {
throw new \common_exception_NotImplemented('Item variables can be stored only for the current item');
}
$sessionId = $context->getTestSession()->getSessionId();
$currentOccurrence = $context->getTestSession()->getCurrentAssessmentItemRefOccurence();
return $sessionId . '.' . $itemId . '.' . $currentOccurrence;
}
/**
* Check if the given RunnerServiceContext is a QtiRunnerServiceContext
*
* @param RunnerServiceContext $context
* @throws \common_exception_InvalidArgumentType
*/
public function assertQtiRunnerServiceContext(RunnerServiceContext $context)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
__CLASS__,
__FUNCTION__,
0,
QtiRunnerServiceContext::class,
$context
);
}
}
/**
* Starts the timer for the current item in the TestSession
*
* @param RunnerServiceContext $context
* @param float $timestamp allow to start the timer at a specific time, or use current when it's null
* @return bool
* @throws \common_exception_InvalidArgumentType
*/
public function startTimer(RunnerServiceContext $context, $timestamp = null)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'startTimer',
0,
QtiRunnerServiceContext::class,
$context
);
}
/* @var TestSession $session */
$session = $context->getTestSession();
if ($session->getState() === AssessmentTestSessionState::INTERACTING) {
$session->startItemTimer($timestamp);
}
return true;
}
/**
* Ends the timer for the current item in the TestSession
*
* @param RunnerServiceContext $context
* @param float $duration The client side duration to adjust the timer
* @param float $timestamp allow to end the timer at a specific time, or use current when it's null
* @return bool
* @throws \common_exception_InvalidArgumentType
*/
public function endTimer(RunnerServiceContext $context, $duration = null, $timestamp = null)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'endTimer',
0,
QtiRunnerServiceContext::class,
$context
);
}
/* @var TestSession $session */
$session = $context->getTestSession();
$session->endItemTimer($duration, $timestamp);
return true;
}
/**
* Switch the received client store ids. Put the received id if different from the last stored.
* This enables us to check wether the stores has been changed during a test session.
* @param RunnerServiceContext $context
* @param string $receivedStoreId The identifier of the client side store
* @return string the identifier of the LAST saved client side store
* @throws \common_exception_InvalidArgumentType
*/
public function switchClientStoreId(RunnerServiceContext $context, $receivedStoreId)
{
if (!$context instanceof QtiRunnerServiceContext) {
throw new InvalidArgumentTypeException(
'QtiRunnerService',
'switchClientStoreId',
0,
QtiRunnerServiceContext::class,
$context
);
}
/* @var TestSession $session */
$session = $context->getTestSession();
$sessionId = $session->getSessionId();
$stateService = $this->getServiceManager()->get(ExtendedStateService::SERVICE_ID);
$lastStoreId = $stateService->getStoreId($sessionId);
if ($lastStoreId == false || $lastStoreId != $receivedStoreId) {
$stateService->setStoreId($sessionId, $receivedStoreId);
}
return $lastStoreId;
}
/**
* Get Current Assessment Session.
*
* Depending on the context (adaptive or not), it will return an appropriate Assessment Object to deal with.
*
* In case of the context is not adaptive, an AssessmentTestSession corresponding to the current test $context is returned.
*
* Otherwise, an AssessmentItemSession to deal with is returned.
*
* @param \oat\taoQtiTest\models\runner\RunnerServiceContext $context
* @return \qtism\runtime\tests\AssessmentTestSession|\qtism\runtime\tests\AssessmentItemSession
*/
public function getCurrentAssessmentSession(RunnerServiceContext $context)
{
if ($context->isAdaptive()) {
return new AssessmentItemSession($context->getCurrentAssessmentItemRef(), new SessionManager());
} else {
return $context->getTestSession();
}
}
/**
* @param TestSession $session
* @param string $qtiClassName
* @return null|string
*/
public function getTimeLimitsFromSession(TestSession $session, $qtiClassName)
{
$maxTimeSeconds = null;
$item = null;
switch ($qtiClassName) {
case 'assessmentTest':
$item = $session->getAssessmentTest();
break;
case 'testPart':
$item = $session->getCurrentTestPart();
break;
case 'assessmentSection':
$item = $session->getCurrentAssessmentSection();
break;
case 'assessmentItemRef':
$item = $session->getCurrentAssessmentItemRef();
break;
}
if ($item && $limits = $item->getTimeLimits()) {
$maxTimeSeconds = $limits->hasMaxTime()
? $limits->getMaxTime()->getSeconds(true)
: $maxTimeSeconds;
}
return $maxTimeSeconds;
}
/**
* @inheritdoc
*/
public function deleteDeliveryExecutionData(DeliveryExecutionDeleteRequest $request)
{
/** @var StorageManager $storage */
$storage = $this->getServiceLocator()->get(StorageManager::SERVICE_ID);
$userUri = $request->getDeliveryExecution()->getUserIdentifier();
/** @var TestSessionService $testSessionService */
$testSessionService = $this->getServiceLocator()->get(TestSessionService::SERVICE_ID);
$session = $testSessionService->getTestSession($request->getDeliveryExecution(), false);
if ($session === null) {
$status = $this->deleteExecutionStates(
$request->getDeliveryExecution()->getIdentifier(),
$userUri,
$storage
);
} else {
$status = $this->deleteExecutionStatesBasedOnSession($request, $storage, $userUri, $session);
}
/** @var ToolsStateStorage $toolsStateStorage */
$toolsStateStorage = $this->getServiceLocator()->get(ToolsStateStorage::SERVICE_ID);
$toolsStateStorage->deleteStates($request->getDeliveryExecution()->getIdentifier());
return $status;
}
/**
* @param RunnerServiceContext $context
* @param $itemRef
* @return array|string
* @throws \common_Exception
* @throws \common_exception_InconsistentData
*/
public function getItemPortableElements(RunnerServiceContext $context, $itemRef)
{
$portableElementService = new PortableElementService();
$portableElementService->setServiceLocator($this->getServiceLocator());
$portableElements = [];
try {
$portableElements = $this->loadItemData($itemRef, QtiJsonItemCompiler::PORTABLE_ELEMENT_FILE_NAME);
foreach ($portableElements as $portableModel => &$elements) {
foreach ($elements as $typeIdentifier => &$versions) {
foreach ($versions as &$portableData) {
try {
$portableElementService->setBaseUrlToPortableData($portableData);
} catch (PortableElementNotFoundException $e) {
\common_Logger::w('the portable element version does not exist in delivery server');
} catch (PortableModelMissing $e) {
\common_Logger::w('the portable element model does not exist in delivery server');
}
}
}
}
} catch (\tao_models_classes_FileNotFoundException $e) {
\common_Logger::i('old delivery that does not contain the compiled portable element data in the item ' . $itemRef);
}
return $portableElements;
}
/**
* @param $itemRef
* @return array|mixed|string
* @throws \common_Exception
*/
public function getItemMetadataElements($itemRef)
{
$metadataElements = [];
try {
$metadataElements = $this->loadItemData($itemRef, QtiJsonItemCompiler::METADATA_FILE_NAME);
} catch (\tao_models_classes_FileNotFoundException $e) {
\common_Logger::i('Old delivery that does not contain the compiled portable element data in the item ' . $itemRef . '. Original message: ' . $e->getMessage());
} catch (\Exception $e) {
\common_Logger::w('An exception caught during fetching item metadata elements. Original message: ' . $e->getMessage());
}
return $metadataElements;
}
/**
* @param $deUri
* @param $userUri
* @param $storage
* @return mixed
*/
protected function deleteExecutionStates($deUri, $userUri, StorageManager $storage)
{
$stateStorage = $storage->getStorage();
$persistence = common_persistence_KeyValuePersistence::getPersistence(
$stateStorage->getOption(tao_models_classes_service_StateStorage::OPTION_PERSISTENCE)
);
$driver = $persistence->getDriver();
if ($driver instanceof common_persistence_AdvKeyValuePersistence) {
$keys = $driver->keys(tao_models_classes_service_StateStorage::KEY_NAMESPACE . '*' . $deUri . '*');
foreach ($keys as $key) {
$driver->del($key);
}
return $storage->persist($userUri);
}
return false;
}
/**
* @param DeliveryExecutionDeleteRequest $request
* @param StorageManager $storage
* @param $userUri
* @param AssessmentTestSession $session
* @return bool
* @throws \common_exception_NotFound
*/
protected function deleteExecutionStatesBasedOnSession(DeliveryExecutionDeleteRequest $request, StorageManager $storage, $userUri, AssessmentTestSession $session)
{
$itemsRefs = $this->getItemsRefs($request, $session);
foreach ($itemsRefs as $itemRef) {
$stateId = $this->buildStorageItemKey(
$request->getDeliveryExecution()->getIdentifier(),
$itemRef
);
if ($storage->has($userUri, $stateId)) {
$storage->del($userUri, $stateId);
}
}
return $storage->persist($userUri);
}
/**
* @param DeliveryExecutionDeleteRequest $request
* @param AssessmentTestSession $session
* @return array
*/
protected function getItemsRefs(DeliveryExecutionDeleteRequest $request, AssessmentTestSession $session)
{
try {
$itemsRefs = (new GetDeliveryExecutionsItems(
$this->getServiceLocator()->get(RuntimeService::SERVICE_ID),
$this->getServiceLocator()->get(CatService::SERVICE_ID),
\tao_models_classes_service_FileStorage::singleton(),
$request->getDeliveryExecution(),
$session
))->getItemsRefs();
} catch (\Exception $exception) {
$itemsRefs = [];
}
return $itemsRefs;
}
/**
* Get state of delivery execution after exit triggered by test taker
* @return string
*/
protected function getStateAfterExit()
{
return DeliveryExecution::STATE_FINISHED;
}
/**
* Returns that the Theme Switcher Plugin is enabled or not
*
* @return bool
* @throws \common_ext_ExtensionException
*/
private function isThemeSwitcherEnabled()
{
/** @var \common_ext_ExtensionsManager $extensionsManager */
$extensionsManager = $this->getServiceLocator()->get(\common_ext_ExtensionsManager::SERVICE_ID);
$config = $extensionsManager->getExtensionById("taoTests")->getConfig("test_runner_plugin_registry");
return array_key_exists(self::TOOL_ITEM_THEME_SWITCHER_KEY, $config)
&& $config[self::TOOL_ITEM_THEME_SWITCHER_KEY]["active"] === true;
}
/**
* Returns the ID of the current theme
*
* @return string
* @throws \common_exception_InconsistentData
*/
private function getCurrentThemeId()
{
/** @var ThemeService $themeService */
$themeService = $this->getServiceLocator()->get(ThemeService::SERVICE_ID);
return $themeService->getTheme()->getId();
}
}