local/modules/OpenApi/Model/Api/BaseApiModel.php line 123

  1. <?php
  2. namespace OpenApi\Model\Api;
  3. use Doctrine\Common\Annotations\AnnotationRegistry;
  4. use OpenApi\Events\ModelExtendDataEvent;
  5. use OpenApi\Events\ModelValidationEvent;
  6. use OpenApi\Exception\OpenApiException;
  7. use OpenApi\Normalizer\ModelApiNormalizer;
  8. use OpenApi\OpenApi;
  9. use Propel\Runtime\ActiveRecord\ActiveRecordInterface;
  10. use Propel\Runtime\Collection\Collection;
  11. use Symfony\Component\HttpFoundation\RequestStack;
  12. use Symfony\Component\Serializer\Serializer;
  13. use Symfony\Component\Validator\Validation;
  14. use Symfony\Component\Validator\Validator\ValidatorInterface;
  15. use Symfony\Contracts\EventDispatcher\EventDispatcherInterface;
  16. use Thelia\Core\HttpFoundation\Request;
  17. use Thelia\Core\Translation\Translator;
  18. use Thelia\Model\Country;
  19. use Thelia\Model\State;
  20. use Thelia\TaxEngine\TaxEngine;
  21. abstract class BaseApiModel implements \JsonSerializable
  22. {
  23.     /** @var ValidatorInterface */
  24.     protected $validator;
  25.     /** @var ModelFactory */
  26.     protected $modelFactory;
  27.     /** @var Request */
  28.     protected $request;
  29.     /** @var Country */
  30.     protected $country;
  31.     /** @var State|null */
  32.     protected $state;
  33.     /** @var EventDispatcherInterface */
  34.     protected $dispatcher;
  35.     protected $extendedData;
  36.     public function __construct(
  37.         ModelFactory $modelFactory,
  38.         RequestStack $requestStack,
  39.         TaxEngine $taxEngine,
  40.         EventDispatcherInterface $dispatcher,
  41.         ValidatorInterface $validator
  42.     ) {
  43.         $this->dispatcher $dispatcher;
  44.         $this->validator $validator;
  45.         $this->modelFactory $modelFactory;
  46.         $this->request $requestStack->getCurrentRequest();
  47.         $this->country $taxEngine->getDeliveryCountry();
  48.         $this->state $taxEngine->getDeliveryState();
  49.         if (method_exists($this'initI18n')) {
  50.             $this->initI18n($modelFactory);
  51.         }
  52.     }
  53.     public function getCurrentLocale()
  54.     {
  55.         return $this->request?->getSession()?->getLang(true)->getLocale();
  56.     }
  57.     /**
  58.      * @param $groups
  59.      *
  60.      * @return BaseApiModel
  61.      *
  62.      * @throws OpenApiException
  63.      */
  64.     public function validate($groups$recursively true)
  65.     {
  66.         $violations $this->getViolations($groups$recursively);
  67.         if (empty($violations)) {
  68.             return $this;
  69.         }
  70.         /** @var Error $error */
  71.         $error $this->modelFactory->buildModel(
  72.             'Error',
  73.             ['title' => Translator::getInstance()->trans('Invalid data', [], OpenApi::DOMAIN_NAME)]
  74.         );
  75.         $error->setSchemaViolations($violations);
  76.         throw new OpenApiException($error);
  77.     }
  78.     public function getViolations($groups$recursively true$propertyPatchPrefix '')
  79.     {
  80.         $modelFactory $this->modelFactory;
  81.         $violations array_reduce(
  82.             iterator_to_array($this->validator->validate($thisnull$groups)),
  83.             function ($carry$violation) use ($modelFactory$propertyPatchPrefix) {
  84.                 $carry[$propertyPatchPrefix.$violation->getPropertyPath()] = $modelFactory->buildModel('SchemaViolation', ['message' => $violation->getMessage()]);
  85.                 return $carry;
  86.             },
  87.             []
  88.         );
  89.         if ($recursively === true) {
  90.             foreach (get_object_vars($this) as $key => $value) {
  91.                 if ($value instanceof self) {
  92.                     $violations array_merge($violations$value->getViolations('read'true$propertyPatchPrefix.$key.'.'));
  93.                 }
  94.             }
  95.         }
  96.         $event = new ModelValidationEvent($this$modelFactory$groups$propertyPatchPrefix);
  97.         $this->dispatcher->dispatch($eventModelValidationEvent::MODEL_VALIDATION_EVENT_PREFIX.$this->snakeCaseName());
  98.         return array_merge($violations$event->getViolations());
  99.     }
  100.     public function jsonSerialize()
  101.     {
  102.         $normalizer = new ModelApiNormalizer();
  103.         $serializer = new Serializer([$normalizer]);
  104.         return $serializer->normalize($thisnull);
  105.     }
  106.     public function createOrUpdateFromData($data$locale null): void
  107.     {
  108.         if (null === $locale) {
  109.             $locale $this->getCurrentLocale();
  110.         }
  111.         if (\is_object($data)) {
  112.             $this->createFromTheliaModel($data$locale);
  113.         }
  114.         if (\is_string($data)) {
  115.             $data json_decode($datatrue);
  116.         }
  117.         if (\is_array($data) || $data instanceof \Traversable) {
  118.             foreach ($data as $key => $value) {
  119.                 $setMethodName 'set'.ucfirst($key);
  120.                 $getMethodName 'get'.ucfirst($key);
  121.                 if (method_exists($this$setMethodName)) {
  122.                     if (\is_array($value)) {
  123.                         if (method_exists($this$getMethodName) && $this->$getMethodName() instanceof self) {
  124.                             $this->$setMethodName($this->$getMethodName()->updateFromData($value));
  125.                             continue;
  126.                         }
  127.                         $openApiModel $this->modelFactory->buildModel(ucfirst($key), $value);
  128.                         $value null !== $openApiModel $openApiModel $value;
  129.                     }
  130.                     $this->$setMethodName($value);
  131.                 }
  132.             }
  133.         }
  134.         $modelExtendEvent = (new ModelExtendDataEvent())
  135.             ->setData($data)
  136.             ->setLocale($locale)
  137.             ->setModel($this);
  138.         $this->dispatcher->dispatch(
  139.             $modelExtendEvent,
  140.             ModelExtendDataEvent::ADD_EXTEND_DATA_PREFIX.$this->snakeCaseName()
  141.         );
  142.         $this->setExtendData($modelExtendEvent->getExtendData());
  143.     }
  144.     /**
  145.      * Override to return the propel model associated with the OpenApi model instead of null.
  146.      *
  147.      * @return mixed
  148.      */
  149.     protected function getTheliaModel($propelModelName null)
  150.     {
  151.         if (null === $propelModelName) {
  152.             $propelModelName "Thelia\Model\\".basename(str_replace('\\''/', static::class));
  153.         }
  154.         if (!class_exists($propelModelName)) {
  155.             return null;
  156.         }
  157.         if (method_exists($this'getId') && null !== $id $this->getId()) {
  158.             $theliaModelQueryName $propelModelName.'Query';
  159.             return $theliaModelQueryName::create()->filterById($id)->findOne();
  160.         }
  161.         /** @var ActiveRecordInterface $newTheliaModel */
  162.         $newTheliaModel = new $propelModelName();
  163.         $newTheliaModel->setNew(true);
  164.         return $newTheliaModel;
  165.     }
  166.     public function toTheliaModel($locale null)
  167.     {
  168.         if (null === $theliaModel $this->getTheliaModel()) {
  169.             throw new \Exception(Translator::getInstance()->trans('Propel model not found automatically for class %className%. Please override the getTheliaModel method to use the toTheliaModel method.', ['%className%' => basename(static::class)], OpenApi::DOMAIN_NAME));
  170.         }
  171.         // If model need locale, set it
  172.         if (method_exists($theliaModel'setLocale')) {
  173.             $theliaModel->setLocale($locale !== null $locale $this->getCurrentLocale());
  174.         }
  175.         // Look all method of Open API model
  176.         foreach (get_class_methods($this) as $methodName) {
  177.             $getter $methodName;
  178.             $setter null;
  179.             // If it's not a getter skip it
  180.             if (=== strncasecmp('get'$methodName3)) {
  181.                 // Build thelia setter name
  182.                 $setter 'set'.substr($getter3);
  183.             }
  184.             // For boolean method like "isVisible"
  185.             if ($setter === null && === strncasecmp('is'$methodName2)) {
  186.                 // Build thelia setter name
  187.                 $setter 'set'.substr($getter2);
  188.             }
  189.             // Check if setter exist in Thelia model
  190.             if (null === $setter || !method_exists($theliaModel$setter)) {
  191.                 continue;
  192.             }
  193.             $value $this->$getter();
  194.             // If Values are the same skip this property
  195.             if (method_exists($theliaModel$getter) && $theliaModel->$getter() === $value) {
  196.                 continue;
  197.             }
  198.             // if the property is another Api model
  199.             if ($value instanceof self) {
  200.                 // If it doesn't have a correspondant thelia model skip it
  201.                 if (null === $value->getTheliaModel()) {
  202.                     continue;
  203.                 }
  204.                 // Else try to set the model id
  205.                 $setModelIdMethod $setter.'Id';
  206.                 if (!method_exists($theliaModel$setModelIdMethod)) {
  207.                     continue;
  208.                 }
  209.                 $setter $setModelIdMethod;
  210.                 $value $value->getId();
  211.             }
  212.             // Todo transform array to collection
  213.             if (is_array($value)) {
  214.                 continue;
  215.             }
  216.             $theliaModel->$setter($value);
  217.         }
  218.         return $theliaModel;
  219.     }
  220.     public function createFromTheliaModel($theliaModel$locale null)
  221.     {
  222.         if (method_exists($theliaModel'setLocale')) {
  223.             $theliaModel->setLocale($locale !== null $locale $this->getCurrentLocale());
  224.         }
  225.         foreach (get_class_methods($this) as $modelMethod) {
  226.             if (=== strncasecmp('set'$modelMethod3)) {
  227.                 $property ucfirst(substr($modelMethod3));
  228.                 $lowercaseProperty ucfirst(strtolower($property));
  229.                 // List all possible getters for this property in propel
  230.                 $propelPossibleMethods = [                                                                                                       //  EXAMPLE :
  231.                     'get'.$property,                                                                                                           //  getProductSaleElements
  232.                     'get'.$property.'s',                                                                                                       //  getProductSaleElementss
  233.                     'get'.$lowercaseProperty,                                                                                                  //  getProductsaleelements
  234.                     'get'.$lowercaseProperty.'s',                                                                                              //  getProductsaleelementss
  235.                     'get'.$property.'Model',                                                                                                 //  getProductSaleElementsModel
  236.                     'get'.$lowercaseProperty.'Model',                                                                                        //  getProductsaleelementsModel
  237.                     'get'.substr(\get_class($theliaModel), strrpos(\get_class($theliaModel), '\\') + 1).$property,                //  getCartProductSaleElements
  238.                     'get'.substr(\get_class($theliaModel), strrpos(\get_class($theliaModel), '\\') + 1).$lowercaseProperty,        //  getCartProductsaleelements
  239.                 ];
  240.                 $availableMethods array_filter(array_intersect($propelPossibleMethodsget_class_methods($theliaModel)));
  241.                 if (empty($availableMethods)) {
  242.                     continue;
  243.                 }
  244.                 $theliaValue null;
  245.                 while (!empty($availableMethods) && ($theliaValue === null || empty($theliaValue))) {
  246.                     $theliaMethod array_pop($availableMethods);
  247.                     $theliaValue $theliaModel->$theliaMethod();
  248.                     if ($theliaValue instanceof Collection) {
  249.                         $theliaValue array_filter(array_map(fn ($value) => $this->modelFactory->buildModel($property$value), iterator_to_array($theliaValue)));
  250.                         continue;
  251.                     }
  252.                     if (\is_object($theliaValue) && $this->modelFactory->modelExists($property)) {
  253.                         $theliaValue $this->modelFactory->buildModel($property$theliaValue);
  254.                     }
  255.                 }
  256.                 $this->$modelMethod($theliaValue);
  257.             }
  258.         }
  259.         if (method_exists($this'sortImagesByPosition')) {
  260.             $this->sortImagesByPosition();
  261.         }
  262.         return $this;
  263.     }
  264.     public function setExtendData($extendedData)
  265.     {
  266.         $this->extendedData $extendedData;
  267.         return $this;
  268.     }
  269.     public function extendedDataValue()
  270.     {
  271.         return $this->extendedData;
  272.     }
  273.     protected function snakeCaseName()
  274.     {
  275.         $name basename(str_replace('\\''/', static::class));
  276.         preg_match_all('!([A-Z][A-Z0-9]*(?=$|[A-Z][a-z0-9])|[A-Za-z][a-z0-9]+)!'$name$matches);
  277.         $ret $matches[0];
  278.         foreach ($ret as &$match) {
  279.             $match $match == strtoupper($match) ? strtolower($match) : lcfirst($match);
  280.         }
  281.         return implode('_'$ret);
  282.     }
  283. }