Разработка кастомного плагина Sylius

Наша компания занимается разработкой, поддержкой и обслуживанием сайтов любой сложности. От простых одностраничных сайтов до масштабных кластерных систем построенных на микро сервисах. Опыт разработчиков подтвержден сертификатами от вендоров.

Разработка и обслуживание любых видов сайтов:

Информационные сайты или веб-приложения
Сайты визитки, landing page, корпоративные сайты, онлайн каталоги, квиз, промо-сайты, блоги, новостные ресурсы, информационные порталы, форумы, агрегаторы
Сайты или веб-приложения электронной коммерции
Интернет-магазины, B2B-порталы, маркетплейсы, онлайн-обменники, кэшбэк-сайты, биржи, дропшиппинг-платформы, парсеры товаров
Веб-приложения для управления бизнес-процессами
CRM-системы, ERP-системы, корпоративные порталы, системы управления производством, парсеры информации
Сайты или веб-приложения электронных услуг
Доски объявлений, онлайн-школы, онлайн-кинотеатры, конструкторы сайтов, порталы предоставления электронных услуг, видеохостинги, тематические порталы

Это лишь некоторые из технических типов сайтов, с которыми мы работаем, и каждый из них может иметь свои специфические особенности и функциональность, а также быть адаптированным под конкретные потребности и цели клиента

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Разработка кастомного плагина Sylius
Средняя
~3-5 рабочих дней
Часто задаваемые вопросы

Наши компетенции:

Этапы разработки

Последние работы

  • image_website-b2b-advance_0.png
    Разработка сайта компании B2B ADVANCE
    1262
  • image_web-applications_feedme_466_0.webp
    Разработка веб-приложения для компании FEEDME
    1171
  • image_websites_belfingroup_462_0.webp
    Разработка веб-сайта для компании БЕЛФИНГРУПП
    874
  • image_ecommerce_furnoro_435_0.webp
    Разработка интернет магазина для компании FURNORO
    1094
  • image_crm_enviok_479_0.webp
    Разработка веб-приложения для компании Enviok
    831
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Разработка веб-сайта для компании ФИКСПЕР
    851

Разработка кастомного плагина Sylius

Плагин Sylius — Symfony Bundle с дополнительной структурой, соответствующей соглашениям Sylius. Механизм расширения основан на Resource System: плагин регистрирует свои ресурсы, и Sylius автоматически создаёт для них CRUD, API endpoints, события. Для кастомизации существующих ресурсов используется Decorator/Override паттерн через sylius_*.yaml конфигурацию.

Создание структуры плагина

composer require --dev sylius-labs/plugin-skeleton
# или вручную:
mkdir -p src/SyliusLoyaltyPlugin/{DependencyInjection,Entity,Form,Menu,Repository,Resources/config}

Минимальная структура:

src/SyliusLoyaltyPlugin/
├── SyliusLoyaltyPlugin.php           # Главный класс Bundle
├── DependencyInjection/
│   ├── Configuration.php
│   └── SyliusLoyaltyExtension.php
├── Entity/
│   └── LoyaltyAccount.php
├── Repository/
│   └── LoyaltyAccountRepository.php
├── Form/
│   └── Type/
│       └── LoyaltyAccountType.php
├── EventListener/
│   └── OrderPlacedListener.php
└── Resources/
    ├── config/
    │   ├── services.xml
    │   └── doctrine/
    │       └── LoyaltyAccount.orm.xml
    ├── views/
    │   └── Admin/
    │       └── LoyaltyAccount/
    └── translations/
        └── messages.ru.yaml

Главный класс Bundle

// src/SyliusLoyaltyPlugin/SyliusLoyaltyPlugin.php
namespace Acme\SyliusLoyaltyPlugin;

use Sylius\Bundle\CoreBundle\Application\SyliusPluginTrait;
use Symfony\Component\HttpKernel\Bundle\Bundle;

final class SyliusLoyaltyPlugin extends Bundle
{
    use SyliusPluginTrait;
}

Трейт SyliusPluginTrait позволяет плагину регистрировать ресурсы через конфигурацию и появляться в Sylius Plugin Registry.

Doctrine Entity

// src/SyliusLoyaltyPlugin/Entity/LoyaltyAccount.php
namespace Acme\SyliusLoyaltyPlugin\Entity;

use Doctrine\ORM\Mapping as ORM;
use Sylius\Component\Customer\Model\CustomerInterface;

#[ORM\Entity(repositoryClass: LoyaltyAccountRepository::class)]
#[ORM\Table(name: 'acme_loyalty_account')]
class LoyaltyAccount
{
    #[ORM\Id]
    #[ORM\GeneratedValue]
    #[ORM\Column(type: 'integer')]
    private ?int $id = null;

    #[ORM\OneToOne(targetEntity: CustomerInterface::class)]
    #[ORM\JoinColumn(nullable: false, onDelete: 'CASCADE')]
    private CustomerInterface $customer;

    #[ORM\Column(type: 'integer', options: ['default' => 0])]
    private int $points = 0;

    #[ORM\Column(type: 'json')]
    private array $transactions = [];

    #[ORM\Column(type: 'datetime_immutable')]
    private \DateTimeImmutable $createdAt;

    public function __construct()
    {
        $this->createdAt = new \DateTimeImmutable();
    }

    public function addPoints(int $points, string $reason, ?string $orderId = null): void
    {
        $this->points += $points;
        $this->transactions[] = [
            'type' => 'earn',
            'points' => $points,
            'reason' => $reason,
            'order_id' => $orderId,
            'date' => (new \DateTimeImmutable())->format(\DateTimeInterface::ATOM),
        ];
    }

    public function spendPoints(int $points, string $reason): void
    {
        if ($this->points < $points) {
            throw new \DomainException('Недостаточно баллов');
        }
        $this->points -= $points;
        $this->transactions[] = [
            'type' => 'spend',
            'points' => $points,
            'reason' => $reason,
            'date' => (new \DateTimeImmutable())->format(\DateTimeInterface::ATOM),
        ];
    }

    public function getId(): ?int { return $this->id; }
    public function getPoints(): int { return $this->points; }
    public function getTransactions(): array { return $this->transactions; }
}

EventListener: начисление баллов при заказе

// src/SyliusLoyaltyPlugin/EventListener/OrderPlacedListener.php
namespace Acme\SyliusLoyaltyPlugin\EventListener;

use Acme\SyliusLoyaltyPlugin\Repository\LoyaltyAccountRepository;
use Doctrine\ORM\EntityManagerInterface;
use Sylius\Bundle\ResourceBundle\Event\ResourceControllerEvent;
use Sylius\Component\Core\Model\OrderInterface;

final class OrderPlacedListener
{
    public function __construct(
        private LoyaltyAccountRepository $accountRepository,
        private EntityManagerInterface $em,
    ) {}

    public function onOrderComplete(ResourceControllerEvent $event): void
    {
        /** @var OrderInterface $order */
        $order = $event->getSubject();
        $customer = $order->getCustomer();

        if (!$customer) {
            return; // гостевой заказ
        }

        $pointsToAward = (int) floor($order->getTotal() / 10000); // 1 балл = 100 руб

        $account = $this->accountRepository->findOneByCustomer($customer);
        if (!$account) {
            $account = new LoyaltyAccount();
            $account->setCustomer($customer);
        }

        $account->addPoints(
            $pointsToAward,
            sprintf('Заказ #%s', $order->getNumber()),
            $order->getId()
        );

        $this->em->persist($account);
        $this->em->flush();
    }
}
<!-- src/SyliusLoyaltyPlugin/Resources/config/services.xml -->
<service id="acme.loyalty.event_listener.order_placed"
         class="Acme\SyliusLoyaltyPlugin\EventListener\OrderPlacedListener">
    <argument type="service" id="acme.loyalty.repository.loyalty_account"/>
    <argument type="service" id="doctrine.orm.entity_manager"/>
    <tag name="kernel.event_listener"
         event="sylius.order.post_complete"
         method="onOrderComplete"/>
</service>

Расширение Admin Menu

// src/SyliusLoyaltyPlugin/Menu/AdminMenuListener.php
namespace Acme\SyliusLoyaltyPlugin\Menu;

use Knp\Menu\ItemInterface;
use Sylius\Bundle\UiBundle\Menu\Event\MenuBuilderEvent;

final class AdminMenuListener
{
    public function addAdminMenuItems(MenuBuilderEvent $event): void
    {
        $menu = $event->getMenu();

        $customers = $menu->getChild('customers');
        if (!$customers) {
            return;
        }

        $customers
            ->addChild('loyalty_accounts', [
                'route' => 'acme_loyalty_admin_loyalty_account_index',
            ])
            ->setLabel('Программа лояльности')
            ->setLabelAttribute('icon', 'star');
    }
}
<service id="acme.loyalty.menu.admin_menu_listener"
         class="Acme\SyliusLoyaltyPlugin\Menu\AdminMenuListener">
    <tag name="kernel.event_listener"
         event="sylius.menu.admin.main"
         method="addAdminMenuItems"/>
</service>

API Extension (API Platform)

// src/SyliusLoyaltyPlugin/Api/Resource/LoyaltyAccountResource.php
namespace Acme\SyliusLoyaltyPlugin\Api\Resource;

use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\Get;
use Acme\SyliusLoyaltyPlugin\Api\Provider\LoyaltyAccountProvider;

#[ApiResource(
    shortName: 'LoyaltyAccount',
    operations: [
        new Get(
            uriTemplate: '/shop/loyalty-account',
            provider: LoyaltyAccountProvider::class,
        ),
    ],
    normalizationContext: ['groups' => ['loyalty:read']],
)]
final class LoyaltyAccountResource
{
    public int $points = 0;
    public array $transactions = [];
}
// src/SyliusLoyaltyPlugin/Api/Provider/LoyaltyAccountProvider.php
final class LoyaltyAccountProvider implements ProviderInterface
{
    public function provide(Operation $operation, array $uriVariables = [], array $context = []): object|array|null
    {
        $customer = $this->tokenStorage->getToken()?->getUser();
        if (!$customer instanceof CustomerInterface) {
            throw new AccessDeniedException();
        }

        $account = $this->repository->findOneByCustomer($customer);
        if (!$account) {
            return new LoyaltyAccountResource(); // 0 баллов
        }

        $resource = new LoyaltyAccountResource();
        $resource->points = $account->getPoints();
        $resource->transactions = $account->getTransactions();

        return $resource;
    }
}

Регистрация плагина в приложении

// config/bundles.php
return [
    // ...
    Acme\SyliusLoyaltyPlugin\SyliusLoyaltyPlugin::class => ['all' => true],
];
bin/console doctrine:migrations:diff
bin/console doctrine:migrations:migrate

Тестирование плагина

Sylius предоставляет sylius/resource-bundle для PHPUnit-тестов и Behat для приёмочного:

// tests/Unit/EventListener/OrderPlacedListenerTest.php
class OrderPlacedListenerTest extends TestCase
{
    public function testAwardsPointsOnOrderComplete(): void
    {
        $order = $this->createMock(OrderInterface::class);
        $order->method('getTotal')->willReturn(50000); // 500 руб = 5 баллов
        $order->method('getCustomer')->willReturn($this->createMock(CustomerInterface::class));
        $order->method('getNumber')->willReturn('0000001');

        $account = new LoyaltyAccount();
        $this->repository->method('findOneByCustomer')->willReturn($account);

        $event = new ResourceControllerEvent($order);
        $this->listener->onOrderComplete($event);

        self::assertSame(5, $account->getPoints());
        self::assertCount(1, $account->getTransactions());
        self::assertSame('earn', $account->getTransactions()[0]['type']);
    }
}

Публикация плагина как Composer-пакета

{
  "name": "acme/sylius-loyalty-plugin",
  "type": "sylius-plugin",
  "require": {
    "php": "^8.1",
    "sylius/sylius": "^2.0"
  },
  "extra": {
    "sylius-plugin": {
      "title": "Sylius Loyalty Plugin",
      "description": "Программа лояльности для Sylius"
    }
  }
}

Тип sylius-plugin в composer.json позволяет плагину появляться в официальном каталоге Sylius Plugins.