Skip to content

Commit

Permalink
Import category position from Akeneo
Browse files Browse the repository at this point in the history
  • Loading branch information
TheGrimmChester committed Nov 10, 2023
1 parent 39e71b6 commit 6744eee
Show file tree
Hide file tree
Showing 24 changed files with 518 additions and 243 deletions.
3 changes: 3 additions & 0 deletions docs/CONFIGURE_DETAIL.md
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,8 @@ The category import configuration contains two configurations.

`excluded_category_codes` allows you to choose the categories that you want to exclude from the import.

`use_akeneo_positions` import category position from Akeneo, this will bypass the default sortable event.

**Selecting a parent will exclude the parent and its children**.

```yaml
Expand All @@ -66,6 +68,7 @@ synolia_sylius_akeneo:
- led_tvs
- audio_video
- mp3_players
use_akeneo_positions: true
```


Expand Down
1 change: 1 addition & 0 deletions install/Application/config/services_test.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -10,4 +10,5 @@ services:
arguments:
$categoryCodesToImport: []
$categoryCodesToExclude: []
$useAkeneoPositions: false
public: true
2 changes: 1 addition & 1 deletion src/DependencyInjection/SynoliaSyliusAkeneoExtension.php
Original file line number Diff line number Diff line change
Expand Up @@ -126,7 +126,7 @@ private function processCategoryConfiguration(ContainerBuilder $container, array
$categoryConfigurationProviderDefinition
->setArgument('$categoryCodesToImport', $config['category_configuration']['root_category_codes'])
->setArgument('$categoryCodesToExclude', $config['category_configuration']['excluded_category_codes'])
->setArgument('$categoryCodesToExclude', $config['category_configuration']['use_akeneo_positions'])
->setArgument('$useAkeneoPositions', $config['category_configuration']['use_akeneo_positions'])
;

$container->setAlias(CategoryConfigurationProviderInterface::class, CategoryConfigurationProvider::class);
Expand Down
16 changes: 16 additions & 0 deletions src/Entity/CategoryConfiguration.php
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,10 @@ class CategoryConfiguration implements ResourceInterface
#[ORM\Column(type: Types::ARRAY)]
private array $rootCategories = [];

/** @ORM\Column(type="boolean") */
#[ORM\Column(type: Types::BOOLEAN)]
private bool $useAkeneoPositions = false;

public function getId(): int
{
return $this->id;
Expand Down Expand Up @@ -88,4 +92,16 @@ public function setRootCategories(array $rootCategories): self

return $this;
}

public function useAkeneoPositions(): bool
{
return $this->useAkeneoPositions;
}

public function setUseAkeneoPositions(bool $useAkeneoPositions): self
{
$this->useAkeneoPositions = $useAkeneoPositions;

return $this;
}
}
4 changes: 4 additions & 0 deletions src/Fixture/CategoryConfigurationFixture.php
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ public function load(array $options): void
$categoryConfiguration = $this->categoriesConfigurationFactory->createNew();
$categoryConfiguration->setRootCategories($options['root_categories_to_import']);
$categoryConfiguration->setNotImportCategories($options['categories_to_exclude']);
$categoryConfiguration->setUseAkeneoPositions($options['use_akeneo_positions']);

$this->entityManager->persist($categoryConfiguration);
$this->entityManager->flush();
Expand All @@ -46,6 +47,9 @@ protected function configureOptionsNode(ArrayNodeDefinition $optionsNode): void
->arrayNode('categories_to_exclude')
->scalarPrototype()->defaultValue([])->end()
->end()
->arrayNode('use_akeneo_positions')
->scalarPrototype()->defaultFalse()->end()
->end()
->end()
;
}
Expand Down
5 changes: 5 additions & 0 deletions src/Form/Type/CategoriesConfigurationType.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
namespace Synolia\SyliusAkeneoPlugin\Form\Type;

use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\Extension\Core\Type\CheckboxType;
use Symfony\Component\Form\Extension\Core\Type\SubmitType;
use Symfony\Component\Form\FormBuilderInterface;

Expand All @@ -26,6 +27,10 @@ public function buildForm(FormBuilderInterface $builder, array $options): void
'required' => false,
'multiple' => true,
])
->add('use_akeneo_positions', CheckboxType::class, [
'label' => 'sylius.ui.admin.akeneo.categories.use_akeneo_positions',
'required' => false,
])
->add('submit', SubmitType::class, [
'attr' => ['class' => 'ui icon primary button'],
'label' => 'sylius.ui.admin.akeneo.save',
Expand Down
40 changes: 40 additions & 0 deletions src/Manager/Doctrine/DoctrineSortableManager.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
<?php

declare(strict_types=1);

namespace Synolia\SyliusAkeneoPlugin\Manager\Doctrine;

use Doctrine\ORM\EntityManagerInterface;
use Gedmo\Sortable\SortableListener;

class DoctrineSortableManager
{
private array $originalEventListeners = [];

public function __construct(private EntityManagerInterface $entityManager)
{
}

public function disableSortableEventListener(): void
{
foreach ($this->entityManager->getEventManager()->getListeners() as $eventName => $listeners) {
foreach ($listeners as $listener) {
if ($listener instanceof SortableListener) {
$this->originalEventListeners[$eventName] = $listener;
$this->entityManager->getEventManager()->removeEventListener($eventName, $listener);
}
}
}
}

public function enableSortableEventListener(): void
{
if ($this->originalEventListeners === []) {
return;
}

foreach ($this->originalEventListeners as $eventName => $listener) {
$this->entityManager->getEventManager()->addEventListener($eventName, $listener);
}
}
}
29 changes: 29 additions & 0 deletions src/Migrations/Version20231106080715.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
<?php

declare(strict_types=1);

namespace Synolia\SyliusAkeneoPlugin\Migrations;

use Doctrine\DBAL\Schema\Schema;
use Doctrine\Migrations\AbstractMigration;

/**
* Auto-generated Migration: Please modify to your needs!
*/
final class Version20231106080715 extends AbstractMigration
{
public function getDescription(): string
{
return 'Added useAkeneoPositions column';
}

public function up(Schema $schema): void
{
$this->addSql('ALTER TABLE akeneo_api_configuration_categories ADD useAkeneoPositions TINYINT(1) NOT NULL');
}

public function down(Schema $schema): void
{
$this->addSql('ALTER TABLE akeneo_api_configuration_categories DROP useAkeneoPositions');
}
}
2 changes: 1 addition & 1 deletion src/Model/Configuration/CategoryConfiguration.php
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ public function setCategoryCodesToExclude(array $categoryCodesToExclude): self
return $this;
}

public function isUseAkeneoPositions(): bool
public function useAkeneoPositions(): bool
{
return $this->useAkeneoPositions;
}
Expand Down
2 changes: 1 addition & 1 deletion src/Model/Configuration/CategoryConfigurationInterface.php
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ public function setCategoryCodesToImport(array $categoryCodesToImport): self;

public function setCategoryCodesToExclude(array $categoryCodesToExclude): self;

public function isUseAkeneoPositions(): bool;
public function useAkeneoPositions(): bool;

public function setUseAkeneoPositions(bool $useAkeneoPositions): self;
}
150 changes: 150 additions & 0 deletions src/Processor/Category/AttributeProcessor.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,150 @@
<?php

declare(strict_types=1);

namespace Synolia\SyliusAkeneoPlugin\Processor\Category;

use Doctrine\ORM\EntityManagerInterface;
use Psr\Log\LoggerInterface;
use Sylius\Component\Core\Model\TaxonInterface;
use Sylius\Component\Resource\Factory\FactoryInterface;
use Sylius\Component\Resource\Repository\RepositoryInterface;
use Synolia\SyliusAkeneoPlugin\Builder\TaxonAttribute\TaxonAttributeValueBuilder;
use Synolia\SyliusAkeneoPlugin\Component\TaxonAttribute\Model\TaxonAttributeSubjectInterface;
use Synolia\SyliusAkeneoPlugin\Entity\TaxonAttribute;
use Synolia\SyliusAkeneoPlugin\Entity\TaxonAttributeInterface;
use Synolia\SyliusAkeneoPlugin\Entity\TaxonAttributeValueInterface;
use Synolia\SyliusAkeneoPlugin\Exceptions\UnsupportedAttributeTypeException;
use Synolia\SyliusAkeneoPlugin\TypeMatcher\TaxonAttribute\TaxonAttributeTypeMatcher;
use Webmozart\Assert\Assert;

class AttributeProcessor implements CategoryProcessorInterface
{
private array $taxonAttributes = [];

private array $taxonAttributeValues = [];

public static function getDefaultPriority(): int
{
return 700;
}

public function __construct(
private LoggerInterface $logger,
private EntityManagerInterface $entityManager,
private RepositoryInterface $taxonAttributeRepository,
private RepositoryInterface $taxonAttributeValueRepository,
private FactoryInterface $taxonAttributeFactory,
private FactoryInterface $taxonAttributeValueFactory,
private TaxonAttributeTypeMatcher $taxonAttributeTypeMatcher,
private TaxonAttributeValueBuilder $taxonAttributeValueBuilder,
) {
}

public function process(TaxonInterface $taxon, array $resource): void
{
foreach ($resource['values'] as $attributeValue) {
try {
$taxonAttribute = $this->getTaxonAttributes(
$attributeValue['attribute_code'],
$attributeValue['type'],
);

$taxonAttributeValue = $this->getTaxonAttributeValues(
$taxon,
$taxonAttribute,
$attributeValue['locale'],
);

$value = $this->taxonAttributeValueBuilder->build(
$attributeValue['attribute_code'],
$attributeValue['type'],
$attributeValue['locale'],
$attributeValue['channel'],
$attributeValue['data'],
);

$taxonAttributeValue->setValue($value);
} catch (UnsupportedAttributeTypeException $e) {
$this->logger->warning($e->getMessage(), [
'trace' => $e->getTrace(),
'exception' => $e,
]);
}
}
}

public function support(TaxonInterface $taxon, array $resource): bool
{
return $taxon instanceof TaxonAttributeSubjectInterface && array_key_exists('values', $resource);
}

private function getTaxonAttributes(string $attributeCode, string $type): TaxonAttributeInterface
{
if (array_key_exists($attributeCode, $this->taxonAttributes)) {
return $this->taxonAttributes[$attributeCode];
}

$taxonAttribute = $this->taxonAttributeRepository->findOneBy(['code' => $attributeCode]);

if ($taxonAttribute instanceof TaxonAttribute) {
$this->taxonAttributes[$attributeCode] = $taxonAttribute;

return $taxonAttribute;
}

$matcher = $this->taxonAttributeTypeMatcher->match($type);

/** @var TaxonAttributeInterface $taxonAttribute */
$taxonAttribute = $this->taxonAttributeFactory->createNew();
$taxonAttribute->setCode($attributeCode);
$taxonAttribute->setType($type);
$taxonAttribute->setStorageType($matcher->getAttributeType()->getStorageType());
$taxonAttribute->setTranslatable(false);

$this->entityManager->persist($taxonAttribute);
$this->taxonAttributes[$attributeCode] = $taxonAttribute;

return $taxonAttribute;
}

private function getTaxonAttributeValues(
TaxonInterface $taxon,
TaxonAttributeInterface $taxonAttribute,
?string $locale,
): TaxonAttributeValueInterface {
Assert::string($taxon->getCode());
Assert::string($taxonAttribute->getCode());

if (
array_key_exists($taxon->getCode(), $this->taxonAttributeValues) &&
array_key_exists($taxonAttribute->getCode(), $this->taxonAttributeValues[$taxon->getCode()]) &&
array_key_exists($locale ?? 'unknown', $this->taxonAttributeValues[$taxon->getCode()][$taxonAttribute->getCode()])
) {
return $this->taxonAttributeValues[$taxon->getCode()][$taxonAttribute->getCode()][$locale ?? 'unknown'];
}

$taxonAttributeValue = $this->taxonAttributeValueRepository->findOneBy([
'subject' => $taxon,
'attribute' => $taxonAttribute,
'localeCode' => $locale,
]);

if ($taxonAttributeValue instanceof TaxonAttributeValueInterface) {
$this->taxonAttributeValues[$taxon->getCode()][$taxonAttribute->getCode()][$locale ?? 'unknown'] = $taxonAttributeValue;

return $taxonAttributeValue;
}

/** @var TaxonAttributeValueInterface $taxonAttributeValue */
$taxonAttributeValue = $this->taxonAttributeValueFactory->createNew();
$taxonAttributeValue->setAttribute($taxonAttribute);
$taxonAttributeValue->setTaxon($taxon);
$taxonAttributeValue->setLocaleCode($locale);
$this->entityManager->persist($taxonAttributeValue);

$this->taxonAttributeValues[$taxon->getCode()][$taxonAttribute->getCode()][$locale ?? 'unknown'] = $taxonAttributeValue;

return $taxonAttributeValue;
}
}
37 changes: 37 additions & 0 deletions src/Processor/Category/CategoryProcessorChain.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
<?php

declare(strict_types=1);

namespace Synolia\SyliusAkeneoPlugin\Processor\Category;

use Psr\Log\LoggerInterface;
use Sylius\Component\Core\Model\TaxonInterface;
use Traversable;

final class CategoryProcessorChain implements CategoryProcessorChainInterface
{
/** @var array<CategoryProcessorInterface> */
private array $categoryProcessors;

public function __construct(Traversable $handlers, private LoggerInterface $logger)
{
$this->categoryProcessors = iterator_to_array($handlers);
}

public function chain(TaxonInterface $taxon, array $resource): void
{
foreach ($this->categoryProcessors as $processor) {
if ($processor->support($taxon, $resource)) {
$this->logger->debug(sprintf('Begin %s', $processor::class), [
'taxon_code' => $taxon->getCode(),
]);

$processor->process($taxon, $resource);

$this->logger->debug(sprintf('End %s', $processor::class), [
'taxon_code' => $taxon->getCode(),
]);
}
}
}
}
12 changes: 12 additions & 0 deletions src/Processor/Category/CategoryProcessorChainInterface.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
<?php

declare(strict_types=1);

namespace Synolia\SyliusAkeneoPlugin\Processor\Category;

use Sylius\Component\Core\Model\TaxonInterface;

interface CategoryProcessorChainInterface
{
public function chain(TaxonInterface $taxon, array $resource): void;
}
Loading

0 comments on commit 6744eee

Please sign in to comment.