tao-test/app/tao/actions/class.TaskQueueWebApi.php

295 lines
9.7 KiB
PHP

<?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) 2019-2021 (original work) Open Assessment Technologies SA (under the project TAO-PRODUCT);
*
*/
use oat\generis\model\fileReference\FileReferenceSerializer;
use oat\oatbox\filesystem\FileSystemService;
use oat\tao\model\http\HttpJsonResponseTrait;
use oat\tao\model\taskQueue\Task\FileReferenceSerializerAwareTrait;
use oat\tao\model\taskQueue\Task\FilesystemAwareTrait;
use oat\tao\model\taskQueue\TaskLog\Broker\TaskLogBrokerInterface;
use oat\tao\model\taskQueue\TaskLog\Decorator\CategoryEntityDecorator;
use oat\tao\model\taskQueue\TaskLog\Decorator\HasFileEntityDecorator;
use oat\tao\model\taskQueue\TaskLog\Decorator\RedirectUrlEntityDecorator;
use oat\tao\model\taskQueue\TaskLog\Decorator\SimpleManagementCollectionDecorator;
use oat\tao\model\taskQueue\TaskLog\TaskLogFilter;
use oat\tao\model\taskQueue\TaskLogInterface;
/**
* API controller to get task queue data by our WEB front-end.
*
* @author Gyula Szucs <gyula@taotesting.com>
*/
class tao_actions_TaskQueueWebApi extends tao_actions_CommonModule
{
use FilesystemAwareTrait;
use FileReferenceSerializerAwareTrait;
use HttpJsonResponseTrait;
private const PARAMETER_TASK_ID = 'taskId';
private const PARAMETER_LIMIT = 'limit';
private const PARAMETER_OFFSET = 'offset';
private const ALL = 'all';
/**
* @throws common_exception_NotImplemented
* @throws Exception
*/
public function getAll(): void
{
$this->checkIfIsXmlHttpRequest();
$taskLogService = $this->getTaskLogService();
$limit = $offset = null;
if ($this->hasRequestParameter(self::PARAMETER_LIMIT)) {
$limit = (int) $this->getRequestParameter(self::PARAMETER_LIMIT);
}
if ($this->hasRequestParameter(self::PARAMETER_OFFSET)) {
$offset = (int) $this->getRequestParameter(self::PARAMETER_OFFSET);
}
$collection = new SimpleManagementCollectionDecorator(
$taskLogService->findAvailableByUser($this->getSessionUserUri(), $limit, $offset),
$taskLogService,
$this->getFileSystemService(),
$this->getFileReferenceSerializer(),
false
);
$this->setSuccessJsonResponse($collection->toArray());
}
/**
* @throws common_exception_NotImplemented
* @throws Exception
*/
public function get(): void
{
$this->checkIfIsXmlHttpRequest();
try {
$this->checkIfTaskIdExists();
$taskLogService = $this->getTaskLogService();
$entity = $taskLogService->getByIdAndUser(
$this->getRequestParameter(self::PARAMETER_TASK_ID),
$this->getSessionUserUri()
);
$this->setSuccessJsonResponse((new RedirectUrlEntityDecorator(
new HasFileEntityDecorator(
new CategoryEntityDecorator($entity, $taskLogService),
$this->getFileSystemService(),
$this->getFileReferenceSerializer()
),
$taskLogService,
common_session_SessionManager::getSession()->getUser()
))->toArray());
} catch (Exception $e) {
$this->setErrorJsonResponse(
$e instanceof common_exception_UserReadableException ? $e->getUserMessage() : $e->getMessage(),
$e->getCode()
);
}
}
/**
* @throws common_exception_NotImplemented
* @throws Exception
*/
public function stats(): void
{
$this->checkIfIsXmlHttpRequest();
$this->setSuccessJsonResponse(
$this->getTaskLogService()->getStats($this->getSessionUserUri())->toArray()
);
}
/**
* @throws common_exception_NotImplemented
* @throws Exception
*/
public function archive()
{
$this->checkIfIsXmlHttpRequest();
try {
$this->checkIfTaskIdExists();
$taskIds = $this->detectTaskIds();
$taskLogService = $this->getTaskLogService();
$filter = $taskIds === static::ALL
? (new TaskLogFilter())->availableForArchived($this->getSessionUserUri())
: (new TaskLogFilter())->addAvailableFilters($this->getSessionUserUri())->in(TaskLogBrokerInterface::COLUMN_ID, $taskIds);
return $this->returnJson([
'success' => (bool) $taskLogService->archiveCollection($taskLogService->search($filter))
]);
} catch (Exception $e) {
$this->setErrorJsonResponse(
$e instanceof common_exception_UserReadableException ? $e->getUserMessage() : $e->getMessage(),
$e instanceof \common_exception_NotFound ? 404 : $e->getCode()
);
}
}
/**
* @throws Exception
*/
public function cancel()
{
$this->checkIfIsXmlHttpRequest();
try {
$this->checkIfTaskIdExists();
$taskIds = $this->detectTaskIds();
$taskLogService = $this->getTaskLogService();
$filter = $taskIds === static::ALL
? (new TaskLogFilter())->availableForCancelled($this->getSessionUserUri())
: (new TaskLogFilter())->addAvailableFilters($this->getSessionUserUri())->in(TaskLogBrokerInterface::COLUMN_ID, $taskIds);
return $this->returnJson([
'success' => (bool) $taskLogService->cancelCollection($taskLogService->search($filter))
]);
} catch (Exception $e) {
$this->setErrorJsonResponse(
$e instanceof common_exception_UserReadableException ? $e->getUserMessage() : $e->getMessage(),
$e instanceof \common_exception_NotFound ? 404 : $e->getCode()
);
}
}
/**
* Download the file created by task.
*/
public function download()
{
try {
$this->checkIfTaskIdExists();
$taskLogEntity = $this->getTaskLogService()->getByIdAndUser(
$this->getRequestParameter(self::PARAMETER_TASK_ID),
$this->getSessionUserUri()
);
if (!$taskLogEntity->getStatus()->isCompleted()) {
throw new \common_Exception('Task "' . $taskLogEntity->getId() . '" is not downloadable.');
}
$fileNameOrSerial = $taskLogEntity->getFileNameFromReport();
if (empty($fileNameOrSerial)) {
throw new \common_Exception('Filename not found in report.');
}
// first try to get a referenced file is it is a serial
$file = $this->getReferencedFile($fileNameOrSerial);
// if no file let's try the task queue storage
if (is_null($file)) {
$file = $this->getQueueStorageFile($fileNameOrSerial);
}
if (!$file) {
throw new \common_exception_NotFound('File not found.');
}
header('Set-Cookie: fileDownload=true');
setcookie('fileDownload', 'true', 0, '/');
header('Content-Disposition: attachment; filename="' . $file->getBasename() . '"');
header('Content-Type: ' . $file->getMimeType());
\tao_helpers_Http::returnStream($file->readPsrStream());
exit();
} catch (Exception $e) {
$this->setErrorJsonResponse(
$e instanceof common_exception_UserReadableException ? $e->getUserMessage() : $e->getMessage(),
$e->getCode()
);
}
}
/**
* @throws common_exception_MissingParameter
*/
protected function checkIfTaskIdExists(): void
{
if (!$this->hasRequestParameter(self::PARAMETER_TASK_ID)) {
throw new common_exception_MissingParameter(self::PARAMETER_TASK_ID, $this->getRequestURI());
}
}
/**
* @throws Exception
*/
protected function checkIfIsXmlHttpRequest(): void
{
if (!$this->isXmlHttpRequest()) {
throw new Exception('Only ajax call allowed.');
}
}
/**
* @return array|string
*/
protected function detectTaskIds()
{
$taskIdsParams = $this->getRequestParameter(self::PARAMETER_TASK_ID);
if (is_array($taskIdsParams)) {
return $taskIdsParams;
} elseif ($taskIdsParams === static::ALL) {
return static::ALL;
} else {
return [$taskIdsParams];
}
}
/**
* @throws common_exception_Error
*/
protected function getSessionUserUri(): string
{
return $this->getSession()->getUserUri();
}
protected function getFileReferenceSerializer(): FileReferenceSerializer
{
return $this->getServiceLocator()->get(FileReferenceSerializer::SERVICE_ID);
}
protected function getFileSystemService(): FileSystemService
{
return $this->getServiceLocator()->get(FileSystemService::SERVICE_ID);
}
protected function getTaskLogService(): TaskLogInterface
{
return $this->getServiceLocator()->get(TaskLogInterface::SERVICE_ID);
}
}