tao-test/app/tao/scripts/tools/MigrationAction.php

186 lines
5.8 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) 2020 (original work) Open Assessment Technologies SA
*/
declare(strict_types=1);
namespace oat\tao\scripts\tools;
use common_report_Report;
use InvalidArgumentException;
use oat\generis\model\OntologyAwareTrait;
use oat\oatbox\extension\script\ScriptAction;
use oat\tao\model\task\migration\AbstractMigrationTask;
use oat\tao\model\taskQueue\QueueDispatcherInterface;
use oat\tao\model\taskQueue\Task\CallbackTaskInterface;
use oat\tao\model\taskQueue\TaskLogActionTrait;
use oat\tao\model\taskQueue\TaskLogInterface;
use RuntimeException;
use Throwable;
class MigrationAction extends ScriptAction
{
use OntologyAwareTrait;
use TaskLogActionTrait;
protected function provideOptions()
{
return [
'chunkSize' => [
'prefix' => 'c',
'longPrefix' => 'chunkSize',
'required' => false,
'cast' => 'integer',
'defaultValue' => 10000,
'description' => 'Amount of units provided into task queue to proceeded with'
],
'pickSize' => [
'prefix' => 'p',
'longPrefix' => 'pickSize',
'required' => false,
'cast' => 'integer',
'defaultValue' => 0,
'description' => 'Amount of items proceed in chunk'
],
'repeat' => [
'prefix' => 'rp',
'longPrefix' => 'repeat',
'required' => false,
'cast' => 'boolean',
'flag' => true,
'description' => 'Scan all the records to the very end'
],
'customParameters' => [
'prefix' => 'cp',
'longPrefix' => 'customParameters',
'required' => false,
'defaultValue' => [],
'description' => 'Custom parameters for target task ie: “option1=0\&option2=1”'
],
'target' => [
'prefix' => 't',
'longPrefix' => 'target',
'required' => true,
'cast' => 'string',
'description' => sprintf(
'Define unit processing task (must extend "%s")',
AbstractMigrationTask::class
)
],
];
}
protected function provideDescription()
{
}
protected function run()
{
$startedAt = time();
$customParameters = [];
$chunkSize = $this->getOption('chunkSize');
parse_str($this->getOption('customParameters'), $customParameters);
$pickSize = $this->getOption('pickSize');
$repeat = (bool) $this->getOption('repeat');
$taskClass = $this->detectTargetClass($this->getOption('target'));
try {
$task = $this->spawnTask($customParameters, $chunkSize, $pickSize, $taskClass, $repeat);
$taskLogEntity = $this->getTaskLogEntity($task->getId());
$taskReport = $taskLogEntity->getReport();
if (0 === strcasecmp($taskLogEntity->getStatus()->getLabel(), TaskLogInterface::STATUS_FAILED)) {
throw new RuntimeException('task failed please refer logs');
}
} catch (Throwable $e) {
return common_report_Report::createFailure($e->getMessage());
}
$report = common_report_Report::createSuccess(
sprintf(
'Operation took %fsec and %dMb',
time() - $startedAt,
memory_get_peak_usage(true) / 1024 / 1024
)
);
if (null !== $taskReport) {
$report->add($taskReport);
}
return $report;
}
protected function provideUsage()
{
return [
'prefix' => 'h',
'longPrefix' => 'help',
'description' => 'Prints the help.',
];
}
protected function returnJson($data, $httpStatus = 200)
{
}
private function spawnTask(
array $customParameters,
int $chunkSize,
int $pickSize,
string $taskClass,
bool $repeat = true
): CallbackTaskInterface
{
/** @var QueueDispatcherInterface $queueDispatcher */
$queueDispatcher = $this->getServiceLocator()->get(QueueDispatcherInterface::SERVICE_ID);
return $queueDispatcher->createTask(
new $taskClass(),
array_merge(
$customParameters,
[
'chunkSize' => $chunkSize,
'pickSize' => $pickSize,
'repeat' => $repeat,
]
),
sprintf(
'Unit processing by %s with chunk size %s',
$taskClass,
$chunkSize
)
);
}
private function detectTargetClass(string $target): string
{
if (class_exists($target) && is_a($target, AbstractMigrationTask::class, true)) {
return $target;
}
throw new InvalidArgumentException(
sprintf('Task must extend %s', AbstractMigrationTask::class)
);
}
}