Розробка кастомного плагіна 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.ua.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.