mirror of
https://github.com/jbcr/ezmigrationbundle.git
synced 2026-03-24 08:52:08 +01:00
173 lines
5.3 KiB
PHP
173 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\EmbeddedReferenceResolverBagInterface;
|
|
use Psr\Http\Message\ResponseInterface;
|
|
|
|
class HTTPExecutor extends AbstractExecutor
|
|
{
|
|
use IgnorableStepExecutorTrait;
|
|
|
|
protected $supportedStepTypes = array('http');
|
|
protected $supportedActions = array('call');
|
|
|
|
/** @var EmbeddedReferenceResolverBagInterface $referenceResolver */
|
|
protected $referenceResolver;
|
|
|
|
protected $container;
|
|
|
|
/**
|
|
* @param ContainerInterface $container
|
|
* @param EmbeddedReferenceResolverBagInterface $referenceResolver has to implement EmbeddedReferenceResolverInterface as well!
|
|
*/
|
|
public function __construct(ContainerInterface $container, EmbeddedReferenceResolverBagInterface $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 array $dsl
|
|
* @param array $context
|
|
* @return true
|
|
* @throws \Exception
|
|
*/
|
|
protected function call($dsl, $context)
|
|
{
|
|
if (!isset($dsl['uri'])) {
|
|
throw new \Exception("Can not execute http call without 'uri' in the step definition");
|
|
}
|
|
|
|
$method = isset($dsl['method']) ? $dsl['method'] : 'GET';
|
|
|
|
$uri = $this->resolveReferencesInText($dsl['uri']);
|
|
|
|
$headers = isset($dsl['headers']) ? $this->resolveReferencesInTextRecursively($dsl['headers']) : array();
|
|
|
|
$body = isset($dsl['body']) ? $this->resolveReferencesInText($dsl['body']) : null;
|
|
|
|
if (isset($dsl['client'])) {
|
|
$client = $this->container->get('httplug.client.'.$dsl['client']);
|
|
} else {
|
|
$client = $this->container->get('httplug.client');
|
|
}
|
|
|
|
$request = $this->container->get('httplug.message_factory')->createRequest($method, $uri, $headers, $body);
|
|
|
|
$response = $client->sendRequest($request);
|
|
|
|
$this->setReferences($response, $dsl);
|
|
|
|
return $response;
|
|
}
|
|
|
|
/**
|
|
* @param ResponseInterface $response
|
|
* @param array $dsl
|
|
* @return bool
|
|
* @todo use jmespath syntax to allow setting refs to response headers
|
|
*/
|
|
protected function setReferences(ResponseInterface $response, $dsl)
|
|
{
|
|
if (!array_key_exists('references', $dsl)) {
|
|
return false;
|
|
}
|
|
|
|
foreach ($dsl['references'] as $reference) {
|
|
switch ($reference['attribute']) {
|
|
case 'status_code':
|
|
$value = $response->getStatusCode();
|
|
break;
|
|
case 'reason_phrase':
|
|
$value = $response->getReasonPhrase();
|
|
break;
|
|
case 'protocol_version':
|
|
$value = $response->getProtocolVersion();
|
|
break;
|
|
case 'body':
|
|
$value = $response->getBody()->__toString();
|
|
break;
|
|
case 'body_size':
|
|
$value = $response->getBody()->getSize();
|
|
break;
|
|
default:
|
|
throw new \InvalidArgumentException('HTTP 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);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Replaces any references inside a string
|
|
*
|
|
* @param string
|
|
* @return string
|
|
* @throws \Exception
|
|
*/
|
|
protected function resolveReferencesInText($text)
|
|
{
|
|
return $this->referenceResolver->ResolveEmbeddedReferences($text);
|
|
}
|
|
|
|
protected function resolveReferencesInTextRecursively($textOrArray)
|
|
{
|
|
if (is_array($textOrArray)) {
|
|
foreach ($textOrArray as $condition => $values) {
|
|
$textOrArray[$condition] = $this->resolveReferencesInTextRecursively($values);
|
|
}
|
|
return $textOrArray;
|
|
} else {
|
|
return $this->resolveReferencesInText($textOrArray);
|
|
}
|
|
}
|
|
}
|