mirror of
https://github.com/jbcr/ezmigrationbundle.git
synced 2026-03-24 08:52:08 +01:00
171 lines
5.3 KiB
PHP
171 lines
5.3 KiB
PHP
<?php
|
|
|
|
namespace Kaliop\eZMigrationBundle\Core\Executor;
|
|
|
|
use Symfony\Component\DependencyInjection\ContainerInterface;
|
|
use Kaliop\eZMigrationBundle\API\Value\MigrationStep;
|
|
use Kaliop\eZMigrationBundle\API\ReferenceResolverBagInterface;
|
|
|
|
class ServiceExecutor extends AbstractExecutor
|
|
{
|
|
use IgnorableStepExecutorTrait;
|
|
|
|
protected $supportedStepTypes = array('service');
|
|
protected $supportedActions = array('call');
|
|
|
|
/** @var ReferenceResolverBagInterface $referenceResolver */
|
|
protected $referenceResolver;
|
|
|
|
protected $container;
|
|
|
|
public function __construct(ContainerInterface $container, ReferenceResolverBagInterface $referenceResolver)
|
|
{
|
|
$this->referenceResolver = $referenceResolver;
|
|
$this->container = $container;
|
|
}
|
|
|
|
/**
|
|
* @param MigrationStep $step
|
|
* @return mixed
|
|
* @throws \Exception
|
|
*/
|
|
public function execute(MigrationStep $step)
|
|
{
|
|
parent::execute($step);
|
|
|
|
if (!isset($step->dsl['mode'])) {
|
|
throw new \Exception("Invalid step definition: missing 'mode'");
|
|
}
|
|
|
|
$action = $step->dsl['mode'];
|
|
|
|
if (!in_array($action, $this->supportedActions)) {
|
|
throw new \Exception("Invalid step definition: value '$action' is not allowed for 'mode'");
|
|
}
|
|
|
|
$this->skipStepIfNeeded($step);
|
|
|
|
return $this->$action($step->dsl, $step->context);
|
|
}
|
|
|
|
/**
|
|
* @param $dsl
|
|
* @param $context
|
|
* @return mixed
|
|
* @throws \Exception
|
|
*/
|
|
protected function call($dsl, $context)
|
|
{
|
|
if (!isset($dsl['service'])) {
|
|
throw new \Exception("Can not call service method: 'service' missing");
|
|
}
|
|
if (!isset($dsl['method'])) {
|
|
throw new \Exception("Can not call service method: 'method' missing");
|
|
}
|
|
if (isset($dsl['arguments']) && !is_array($dsl['arguments'])) {
|
|
throw new \Exception("Can not call service method: 'arguments' is not an array");
|
|
}
|
|
|
|
$service = $this->container->get($this->referenceResolver->resolveReference($dsl['service']));
|
|
$method = $this->referenceResolver->resolveReference($dsl['method']);
|
|
$callable = array($service, $method);
|
|
if (!is_callable($callable)) {
|
|
throw new \Exception("Can not call service method: $method is not a method of " . get_class($service));
|
|
}
|
|
|
|
if (isset($dsl['arguments'])) {
|
|
$args = $dsl['arguments'];
|
|
} else {
|
|
$args = array();
|
|
}
|
|
foreach($args as &$val) {
|
|
$val = $this->resolveReferencesRecursively($val);
|
|
}
|
|
|
|
$exception = null;
|
|
$result = null;
|
|
try {
|
|
$result = call_user_func_array($callable, $args);
|
|
} catch (\Exception $exception) {
|
|
$catch = false;
|
|
|
|
// allow to specify a set of exceptions to tolerate
|
|
if (isset($dsl['catch'])) {
|
|
if (is_array($dsl['catch'])) {
|
|
$caught = $dsl['catch'];
|
|
} else {
|
|
$caught = array($dsl['catch']);
|
|
}
|
|
|
|
foreach ($caught as $baseException) {
|
|
if (is_subclass_of($exception, $baseException)) {
|
|
$catch = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!$catch) {
|
|
throw $exception;
|
|
}
|
|
}
|
|
|
|
$this->setReferences($result, $exception, $dsl);
|
|
|
|
return $result;
|
|
}
|
|
|
|
protected function setReferences($result, \Exception $exception = null, $dsl)
|
|
{
|
|
if (!array_key_exists('references', $dsl)) {
|
|
return false;
|
|
}
|
|
|
|
foreach ($dsl['references'] as $reference) {
|
|
switch ($reference['attribute']) {
|
|
case 'result':
|
|
$value = $result;
|
|
break;
|
|
case 'exception_code':
|
|
$value = $exception ? $exception->getCode() : null;
|
|
break;
|
|
case 'exception_message':
|
|
$value = $exception ? $exception->getMessage() : null;
|
|
break;
|
|
case 'exception_file':
|
|
$value = $exception ? $exception->getFile() : null;
|
|
break;
|
|
case 'exception_line':
|
|
$value = $exception ? $exception->getLine() : null;
|
|
break;
|
|
|
|
default:
|
|
throw new \InvalidArgumentException('Service executor does not support setting references for attribute ' . $reference['attribute']);
|
|
}
|
|
|
|
$overwrite = false;
|
|
if (isset($reference['overwrite'])) {
|
|
$overwrite = $reference['overwrite'];
|
|
}
|
|
$this->referenceResolver->addReference($reference['identifier'], $value, $overwrite);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* @todo should be moved into the reference resolver classes
|
|
*/
|
|
protected function resolveReferencesRecursively($match)
|
|
{
|
|
if (is_array($match)) {
|
|
foreach ($match as $condition => $values) {
|
|
$match[$condition] = $this->resolveReferencesRecursively($values);
|
|
}
|
|
return $match;
|
|
} else {
|
|
return $this->referenceResolver->resolveReference($match);
|
|
}
|
|
}
|
|
}
|