MultiDimensionProvider.php 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. <?php
  2. /**
  3. * Copyright © Magento, Inc. All rights reserved.
  4. * See COPYING.txt for license details.
  5. */
  6. declare(strict_types=1);
  7. namespace Magento\Framework\Indexer;
  8. /**
  9. * Multiply dimensions from provided DimensionProviderInterface
  10. */
  11. class MultiDimensionProvider implements \IteratorAggregate
  12. {
  13. /**
  14. * @var array
  15. */
  16. private $dimensionsIterators = [];
  17. /**
  18. * @var array
  19. */
  20. private $dimensionsDataProviders = [];
  21. /**
  22. * @var int
  23. */
  24. private $dimensionsProvidersCount = 0;
  25. /**
  26. * @param DimensionProviderInterface[] $dimensionProviders
  27. */
  28. public function __construct(array $dimensionProviders = [])
  29. {
  30. foreach ($dimensionProviders as $dimensionDataProvider) {
  31. $this->addDimensionDataProvider($dimensionDataProvider);
  32. }
  33. }
  34. /**
  35. * Returns generator that will return multiplied dimensions on each iteration
  36. *
  37. * @return \Traversable|Dimension[][]
  38. * @throws \LogicException
  39. */
  40. public function getIterator(): \Traversable
  41. {
  42. // just return empty array if we have no dimension providers to iterate over
  43. if ($this->dimensionsProvidersCount === 0) {
  44. yield [];
  45. return;
  46. }
  47. // this recreates iterators for dimension so we can iterate over them
  48. $this->rewind();
  49. // if at leas one dimension provider has no dimensions to return we can't multiple dimension at all
  50. if (!$this->hasCurrentDimension()) {
  51. throw new \LogicException('Can`t multiple dimensions because some of them are empty.');
  52. }
  53. // return dimensions until all iterators become invalid
  54. while ($this->hasCurrentDimension()) {
  55. yield $this->getCurrentDimension();
  56. $this->setNextDimension();
  57. }
  58. }
  59. /**
  60. * Return all dimensions for current state of each dimension provider
  61. *
  62. * @return array
  63. */
  64. private function getCurrentDimension(): array
  65. {
  66. $dimensions = [];
  67. foreach ($this->dimensionsIterators as $dimensionIterator) {
  68. /** @var Dimension $dimension */
  69. $dimension = $dimensionIterator->current();
  70. $dimensions[$dimension->getName()] = $dimension;
  71. }
  72. return $dimensions;
  73. }
  74. /**
  75. * Iterates over dimension iterators one by one starting from right to left
  76. * This approach emulates iterations over X nested foreach loops e.g.:
  77. *
  78. * @return void
  79. */
  80. private function setNextDimension()
  81. {
  82. $this->dimensionsIterators[$this->dimensionsProvidersCount - 1]->next();
  83. for ($i = ($this->dimensionsProvidersCount - 1); $i > 0; $i--) {
  84. if (!$this->dimensionsIterators[$i]->valid()) {
  85. $this->dimensionsIterators[$i] = $this->dimensionsDataProviders[$i]->getIterator();
  86. $this->dimensionsIterators[$i-1]->next();
  87. }
  88. }
  89. }
  90. /**
  91. * Recreates iterators so all MultiDimensionProvider can be iterated again
  92. *
  93. * @return void
  94. */
  95. private function rewind()
  96. {
  97. $this->dimensionsIterators = [];
  98. foreach ($this->dimensionsDataProviders as $dimensionDataProvider) {
  99. $this->dimensionsIterators[] = $dimensionDataProvider->getIterator();
  100. }
  101. }
  102. /**
  103. * Check if all dimension iterators are in valid state
  104. *
  105. * If at least one of dimension iterators is invalid before very first iteration - we assume
  106. * that dimension provider has no dimensions at all, which means we can't multiple all dimensions
  107. *
  108. * If all dimension iterators became invalid - we assume that multiplication is already done
  109. *
  110. * @return bool
  111. */
  112. private function hasCurrentDimension(): bool
  113. {
  114. $valid = true;
  115. foreach ($this->dimensionsIterators as $dimensionsIterator) {
  116. // if at least one data provider is invalid at this stage - all generator is invalid
  117. if (!$dimensionsIterator->valid()) {
  118. return false;
  119. }
  120. }
  121. // generator is valid only when all data providers are valid
  122. return $valid;
  123. }
  124. /**
  125. * Collects dimension data providers
  126. * This was done via separate method to ensure that each provider has required interface
  127. *
  128. * @param DimensionProviderInterface $dimensionDataProvider
  129. * @return void
  130. */
  131. private function addDimensionDataProvider(DimensionProviderInterface $dimensionDataProvider)
  132. {
  133. $this->dimensionsDataProviders[] = $dimensionDataProvider;
  134. $this->dimensionsProvidersCount++;
  135. }
  136. }