Ответы пользователя по тегу Symfony
  • Как изменять сущности без форм?

    iNikNik
    @iNikNik Автор вопроса
    В итоге получилось вот что: по совету Сергей Протько я нарыл как сделать необходимый конструктор - ссылка. На самом деле нужный класс уже есть тут:
    vendor/jms/serializer/tests/JMS/Serializer/Tests/Fixtures/InitializedObjectConstructor.php


    И мы просто копируем его к себе в services (меняя при этом неймспейс).
    InitializedObjectConstructor.php
    <?php
    
    /*
     * Copyright 2013 Johannes M. Schmitt <schmittjoh@gmail.com>
     *
     * Licensed under the Apache License, Version 2.0 (the "License");
     * you may not use this file except in compliance with the License.
     * You may obtain a copy of the License at
     *
     *     http://www.apache.org/licenses/LICENSE-2.0
     *
     * Unless required by applicable law or agreed to in writing, software
     * distributed under the License is distributed on an "AS IS" BASIS,
     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     * See the License for the specific language governing permissions and
     * limitations under the License.
     */
    
    namespace FiveToFive\ergil\DomainBundle\Service;
    
    use JMS\Serializer\VisitorInterface;
    use JMS\Serializer\Metadata\ClassMetadata;
    use JMS\Serializer\DeserializationContext;
    use JMS\Serializer\Construction\ObjectConstructorInterface;
    
    /**
     * Object constructor that allows deserialization into already constructed
     * objects passed through the deserialization context
     */
    class InitializedObjectConstructor implements ObjectConstructorInterface
    {
        private $fallbackConstructor;
    
        /**
         * Constructor.
         *
         * @param ObjectConstructorInterface $fallbackConstructor Fallback object constructor
         */
        public function __construct(ObjectConstructorInterface $fallbackConstructor)
        {
            $this->fallbackConstructor = $fallbackConstructor;
        }
    
        /**
         * {@inheritdoc}
         */
        public function construct(VisitorInterface $visitor, ClassMetadata $metadata, $data, array $type, DeserializationContext $context)
        {
            if ($context->attributes->containsKey('target') && $context->getDepth() === 1) {
                return $context->attributes->get('target')->get();
            }
    
            return $this->fallbackConstructor->construct($visitor, $metadata, $data, $type, $context);
        }
    
    }


    Затем в конфиге прописываем:
    services:
        jms_serializer.object_constructor:
            alias: jms_serializer.initialized_object_constructor
            public: false
        jms_serializer.initialized_object_constructor:
             class:        VendorName\Bundle\ApiBundle\Services\InitializedObjectConstructor
             arguments:    ["@jms_serializer.doctrine_object_constructor"]


    Далее - контроллер:
    /**
         * @param Product $slug
         * @param Request $request
         * @return Product
         *
         * @Rest\Route(requirements={"slug" = "\d+"})
         * @Rest\View(serializerGroups={ "Default", "product_details", "product_categories_list", "image_details" })
         */
        public function putAction(Product $slug, Request $request)
        {
            $productService = $this->get("product.service");
            return $productService->update($slug, $request);
        }


    и сервис:
    public function update(Product $product, Request $request)
        {
            $deContext = new DeserializationContext();
            $deContext->attributes->set('target', $product);
            // save original id
            $targetId = $product->getId();
            // product and updatedProduct - the same entities with same data
            // I'm assign updatedProduct var ony for better understanding
            $updatedProduct = $this->serializer
                ->deserialize($request->getContent(), 'FiveToFive\ergil\DomainBundle\Entity\Product', 
                                      $request->getContentType(), $deContext);
            // ... some validation ...
            if(!$this->isValid($updatedProduct) || $targetId !== $updatedProduct->getId()) // compare id's
                throw new BadRequestHttpException("Wrong product data");
            $this->em->flush($updatedProduct);
            return $updatedProduct;
        }


    Основная фишка - установить target для десерриализации:
    $deContext = new DeserializationContext();
    $deContext->attributes->set('target', $product);


    В качестве бонуса: когда идет запрос на
    [PUT] http://my-api-url/api/products/1
    Т.е. запрос на изменение продукта с id = 1, а в теле запроса у нас:
    {
        "id": 124,
        "title": "ID dosen't changes!!!!"
    }

    id равен 124 (то есть он отличается от 1) - в базе все равно обновляется сущность с id === 1 (магия, да и только). (upd: обновил код сервиса, чтобы выбрасывать исключение в такой ситуации)

    PS В итоге буду переходить на symfony\serializer
    Ответ написан
  • Как подключить symfony2 в существующий проект?

    Symfony - это фреймворк. Одна из особенностей фреймоврка - он вызывает пользовательский код, а не наоборот.
    Так что если нужно сделать проект на symfony - Вам нужно полностью переделать его на симфони (сделать так, чтобы симфони вызывал ваш код).

    Или есть вариант использовать отдельные компоненты симфони, используя их как вам удобно.
    Ответ написан
  • Как автоматически загружать вложенные сущности (ParamConverter)?

    iNikNik
    @iNikNik Автор вопроса
    Время шло и я внезапно понял, что в добавок ко всему, param converter ВООБЩЕ не вызывает конструкторы сущностей. Немного погуглив нашел это (git issue) и это (stackoverflow): если добавить в конфиг сервисов:
    services:
        # ...
        jms_serializer.object_constructor:
            alias: jms_serializer.doctrine_object_constructor
            public: false

    То сущности будут инициализироваться как надо и в качестве бонуса param converter таки начнет загружать вложенные сущности. В документации я это конечно нашел, но только сейчас.
    Ответ написан
    Комментировать