/**
  * Normalize the field name from attribute and catalog context
  *
  * @param AttributeInterface $attribute
  * @param string             $locale
  * @param string             $scope
  *
  * @return string
  */
 public static function getNormalizedValueFieldFromAttribute(AttributeInterface $attribute, $locale = null, $scope = null)
 {
     if ($attribute->isLocalizable() && null === $locale) {
         throw new \LogicException('Locale is not configured');
     }
     if ($attribute->isScopable() && null === $scope) {
         throw new \LogicException('Scope is not configured');
     }
     return self::getNormalizedValueField($attribute->getCode(), $attribute->isLocalizable(), $attribute->isScopable(), $attribute->isLocalizable() ? $locale : null, $attribute->isScopable() ? $scope : null);
 }
 function it_throws_exception_when_scope_is_not_provided_but_expected(ProductInterface $product, AttributeInterface $price)
 {
     $price->getCode()->willReturn('price');
     $price->isLocalizable()->willReturn(false);
     $price->isScopable()->willReturn(true);
     $this->shouldThrow(new \InvalidArgumentException('A scope must be provided to create a value for the scopable attribute price'))->duringAddProductValue($product, $price);
 }
 function it_resolves_eligible_values_for_a_set_of_attributes($localeRepository, $channelRepository, AttributeInterface $sku, AttributeInterface $name, AttributeInterface $desc, AttributeInterface $tax, LocaleInterface $fr, LocaleInterface $en, ChannelInterface $ecom, ChannelInterface $print)
 {
     $sku->getCode()->willReturn('sku');
     $sku->getAttributeType()->willReturn('pim_catalog_identifier');
     $sku->isLocalizable()->willReturn(false);
     $sku->isScopable()->willReturn(false);
     $sku->isLocaleSpecific()->willReturn(false);
     $name->getCode()->willReturn('name');
     $name->getAttributeType()->willReturn('pim_catalog_text');
     $name->isLocalizable()->willReturn(true);
     $name->isScopable()->willReturn(false);
     $name->isLocaleSpecific()->willReturn(false);
     $desc->getCode()->willReturn('description');
     $desc->getAttributeType()->willReturn('pim_catalog_text');
     $desc->isLocalizable()->willReturn(true);
     $desc->isScopable()->willReturn(true);
     $desc->isLocaleSpecific()->willReturn(false);
     $tax->getCode()->willReturn('tax');
     $tax->getAttributeType()->willReturn('pim_catalog_text');
     $tax->isLocalizable()->willReturn(true);
     $tax->isScopable()->willReturn(false);
     $tax->isLocaleSpecific()->willReturn(true);
     $tax->getLocaleSpecificCodes()->willReturn(['fr_FR']);
     $fr->getCode()->willReturn('fr_FR');
     $en->getCode()->willReturn('en_US');
     $localeRepository->getActivatedLocales()->willReturn([$fr, $en]);
     $ecom->getCode()->willReturn('ecommerce');
     $ecom->getLocales()->willReturn([$en, $fr]);
     $print->getCode()->willReturn('print');
     $print->getLocales()->willReturn([$en, $fr]);
     $channelRepository->findAll()->willReturn([$ecom, $print]);
     $this->resolveEligibleValues([$sku, $name, $desc, $tax])->shouldReturn([['attribute' => 'sku', 'type' => 'pim_catalog_identifier', 'locale' => null, 'scope' => null], ['attribute' => 'name', 'type' => 'pim_catalog_text', 'locale' => 'fr_FR', 'scope' => null], ['attribute' => 'name', 'type' => 'pim_catalog_text', 'locale' => 'en_US', 'scope' => null], ['attribute' => 'description', 'type' => 'pim_catalog_text', 'locale' => 'en_US', 'scope' => 'ecommerce'], ['attribute' => 'description', 'type' => 'pim_catalog_text', 'locale' => 'fr_FR', 'scope' => 'ecommerce'], ['attribute' => 'description', 'type' => 'pim_catalog_text', 'locale' => 'en_US', 'scope' => 'print'], ['attribute' => 'description', 'type' => 'pim_catalog_text', 'locale' => 'fr_FR', 'scope' => 'print'], ['attribute' => 'tax', 'type' => 'pim_catalog_text', 'locale' => 'fr_FR', 'scope' => null]]);
 }
 function it_throws_an_exception_when_scope_is_not_activated($scopeRepository, AttributeInterface $description)
 {
     $scopeRepository->getChannelCodes()->willReturn([]);
     $description->isScopable()->willReturn(true);
     $description->getCode()->willReturn('description');
     $this->shouldThrow(new \LogicException('Attribute "description" expects an existing scope, "print" given.'))->during('validateScope', [$description, 'print']);
 }
 function let(SerializerInterface $serializer, AttributeInterface $simpleAttribute)
 {
     $serializer->implement('Symfony\\Component\\Serializer\\Normalizer\\NormalizerInterface');
     $this->setSerializer($serializer);
     $simpleAttribute->isLocalizable()->willReturn(false);
     $simpleAttribute->isScopable()->willReturn(false);
     $simpleAttribute->getCode()->willReturn('simple');
 }
 function it_throws_an_exception_when_the_scope_is_not_provided(AttributeInterface $price)
 {
     $price->getId()->willReturn(42);
     $price->isLocalizable()->willReturn(false);
     $price->isScopable()->willReturn(true);
     $price->getCode()->willReturn('price');
     $this->shouldThrow('\\InvalidArgumentException')->duringPrepareCondition($price, 'alias', 'en_US', null);
 }
 function let(Builder $qb, AttributeInterface $image, AttributeValidatorHelper $attrValidatorHelper)
 {
     $this->beConstructedWith($attrValidatorHelper, ['pim_catalog_image', 'pim_catalog_file'], ['STARTS WITH', 'ENDS WITH', 'CONTAINS', 'DOES NOT CONTAIN', '=', 'EMPTY']);
     $this->setQueryBuilder($qb);
     $image->getCode()->willReturn('picture');
     $image->isLocalizable()->willReturn(false);
     $image->isScopable()->willReturn(false);
 }
 /**
  * Prepare join to attribute condition with current locale and scope criterias
  *
  * @param AttributeInterface $attribute the attribute
  * @param string             $joinAlias the value join alias
  * @param string             $locale    the locale
  * @param string             $scope     the scope
  *
  * @return string
  */
 public function prepareCondition(AttributeInterface $attribute, $joinAlias, $locale = null, $scope = null)
 {
     $condition = $joinAlias . '.attribute = ' . $attribute->getId();
     if ($attribute->isLocalizable() && null === $locale) {
         throw new \InvalidArgumentException(sprintf('Cannot prepare condition on localizable attribute "%s" without locale', $attribute->getCode()));
     }
     if ($attribute->isLocalizable()) {
         $condition .= ' AND ' . $joinAlias . '.locale = ' . $this->qb->expr()->literal($locale);
     }
     if ($attribute->isScopable() && null === $scope) {
         throw new \InvalidArgumentException(sprintf('Cannot prepare condition on scopable attribute "%s" without scope', $attribute->getCode()));
     }
     if ($attribute->isScopable()) {
         $condition .= ' AND ' . $joinAlias . '.scope = ' . $this->qb->expr()->literal($scope);
     }
     return $condition;
 }
 function it_adds_a_order_by_on_an_attribute_value_in_the_query(Builder $queryBuilder, AttributeInterface $sku)
 {
     $sku->getCode()->willReturn('sku');
     $sku->isLocalizable()->willReturn(false);
     $sku->isScopable()->willReturn(false);
     $queryBuilder->sort('normalizedData.sku', 'desc')->willReturn($queryBuilder);
     $queryBuilder->sort('_id')->shouldBeCalled();
     $this->addAttributeSorter($sku, 'desc');
 }
 /**
  * Check if scope data is consistent with the attribute scopable property
  *
  * @param AttributeInterface $attribute
  * @param string             $scope
  *
  * @throws \LogicException
  */
 public function validateScope(AttributeInterface $attribute, $scope)
 {
     if (!$attribute->isScopable() && null === $scope) {
         return;
     }
     if ($attribute->isScopable() && null === $scope) {
         throw new \LogicException(sprintf('Attribute "%s" expects a scope, none given.', $attribute->getCode()));
     }
     if (!$attribute->isScopable() && null !== $scope) {
         throw new \LogicException(sprintf('Attribute "%s" does not expect a scope, "%s" given.', $attribute->getCode(), $scope));
     }
     if (null === static::$scopeCodes) {
         static::$scopeCodes = $this->getScopeCodes();
     }
     if (!in_array($scope, static::$scopeCodes)) {
         throw new \LogicException(sprintf('Attribute "%s" expects an existing scope, "%s" given.', $attribute->getCode(), $scope));
     }
 }
 function it_normalizes_value_with_decimal_support_backend(ProductValueInterface $value, AttributeInterface $attribute)
 {
     $attribute->getCode()->willReturn('code');
     $attribute->getBackendType()->willReturn('decimal');
     $attribute->isLocalizable()->willReturn(false);
     $attribute->isScopable()->willReturn(false);
     $value->getData()->willReturn('42.42');
     $value->getAttribute()->willReturn($attribute);
     $this->normalize($value, 'mongodb_json', [])->shouldReturn(['code' => 42.42]);
 }
 function it_throws_an_exception_when_scope_is_expected_but_not_existing($attrValidatorHelper, AttributeInterface $attribute)
 {
     $e = new \LogicException('Attribute "attributeCode" expects an existing scope, "ecommerce" given.');
     $attribute->getCode()->willReturn('attributeCode');
     $attribute->isLocalizable()->willReturn(false);
     $attribute->isScopable()->willReturn(true);
     $attrValidatorHelper->validateLocale($attribute, null)->shouldBeCalled();
     $attrValidatorHelper->validateScope($attribute, 'ecommerce')->willThrow($e);
     $this->shouldThrow(InvalidArgumentException::expectedFromPreviousException($e, 'attributeCode', 'setter', 'concrete'))->during('testLocaleAndScope', [$attribute, null, 'ecommerce']);
 }
 function it_denormalizes_variant_group_values($fieldNameBuilder, $valueDenormalizer, AttributeInterface $description)
 {
     $fieldNameBuilder->extractAttributeFieldNameInfos('description-ecommerce-en_US')->willReturn(['attribute' => $description, 'locale_code' => 'en_US', 'scope_code' => 'ecommerce'])->shouldBeCalled();
     $fieldNameBuilder->extractAttributeFieldNameInfos('description-ecommerce-fr_FR')->willReturn(['attribute' => $description, 'locale_code' => 'fr_FR', 'scope_code' => 'ecommerce'])->shouldBeCalled();
     $description->getCode()->willReturn('description');
     $description->isLocalizable()->willReturn(true);
     $description->isScopable()->willReturn(true);
     $valueDenormalizer->denormalize('My en_US desc', self::VALUE_CLASS, self::FORMAT_CSV, Argument::any())->shouldBeCalled();
     $valueDenormalizer->denormalize('My fr_FR desc', self::VALUE_CLASS, self::FORMAT_CSV, Argument::any())->shouldBeCalled();
     $csvData = ['description-ecommerce-en_US' => 'My en_US desc', 'description-ecommerce-fr_FR' => 'My fr_FR desc'];
     $this->denormalize($csvData, 'ProductValue[]', 'csv');
 }
 function let(QueryBuilder $qb, AttributeValidatorHelper $attrValidatorHelper, Expr $expr, AttributeInterface $image)
 {
     $this->beConstructedWith($attrValidatorHelper, ['pim_catalog_image', 'pim_catalog_file'], ['STARTS WITH', 'ENDS WITH', 'CONTAINS', 'DOES NOT CONTAIN', '=', 'EMPTY']);
     $this->setQueryBuilder($qb);
     $qb->getRootAlias()->willReturn('p');
     $qb->expr()->willReturn($expr);
     $image->getId()->willReturn(1);
     $image->getCode()->willReturn('picture');
     $image->isLocalizable()->willReturn(false);
     $image->isScopable()->willReturn(false);
     $image->getBackendType()->willReturn('media');
 }
 function it_sets_the_locale_and_scope_when_denormalizing_values($serializer, AttributeInterface $attribute)
 {
     $attribute->getAttributeType()->willReturn('pim_catalog_number');
     $attribute->getBackendType()->willReturn('decimal');
     $attribute->isLocalizable()->willReturn(true);
     $attribute->isScopable()->willReturn(true);
     $serializer->denormalize(1, 'pim_catalog_number', 'json', Argument::type('array'))->shouldBeCalled()->willReturn(1);
     $value = $this->denormalize(['value' => 1, 'locale' => 'en_US', 'scope' => 'ecommerce'], 'Pim\\Bundle\\CatalogBundle\\Model\\ProductValue', 'json', ['attribute' => $attribute]);
     $value->shouldBeAnInstanceOf('Pim\\Bundle\\CatalogBundle\\Model\\ProductValue');
     $value->getData()->shouldReturn(1);
     $value->getLocale()->shouldReturn('en_US');
     $value->getScope()->shouldReturn('ecommerce');
 }
 function it_adds_attributes_to_a_product_template($denormalizer, $normalizer, $productBuilder, ProductTemplateInterface $template, ProductValueInterface $colorValue, AttributeInterface $name, AttributeInterface $color)
 {
     $color->getCode()->willReturn('color');
     $color->isLocalizable()->willReturn(false);
     $color->isScopable()->willReturn(false);
     $colorValue->getAttribute()->willReturn($color);
     $colorValue->setEntity(Argument::type('Pim\\Bundle\\CatalogBundle\\Model\\Product'))->willReturn($colorValue);
     $template->getValuesData()->willReturn(['color' => 'bar']);
     $denormalizer->denormalize(['color' => 'bar'], 'ProductValue[]', 'json')->shouldBeCalled()->willReturn([$colorValue]);
     $productBuilder->addAttributeToProduct(Argument::type('Pim\\Bundle\\CatalogBundle\\Model\\Product'), $name)->shouldBeCalled();
     $normalizer->normalize(Argument::type('Doctrine\\Common\\Collections\\ArrayCollection'), 'json', ['entity' => 'product'])->shouldBeCalled()->willReturn(['name' => 'foo', 'color' => 'bar']);
     $template->setValuesData(['name' => 'foo', 'color' => 'bar'])->shouldBeCalled();
     $this->addAttributes($template, [$name]);
 }
 function it_adds_a_sorter_to_the_query($qb, AttributeInterface $metric)
 {
     $metric->getId()->willReturn(42);
     $metric->getCode()->willReturn('metric_code');
     $metric->getBackendType()->willReturn('metric');
     $metric->isLocalizable()->willReturn(false);
     $metric->isScopable()->willReturn(false);
     $condition = "sorterVmetric_code.attribute = 42";
     $qb->getRootAlias()->willReturn('r');
     $qb->leftJoin('r.values', 'sorterVmetric_code', 'WITH', $condition)->shouldBeCalled();
     $qb->leftJoin('sorterVmetric_code.metric', 'sorterMmetric_code')->shouldBeCalled();
     $qb->addOrderBy('sorterMmetric_code.baseData', 'DESC')->shouldBeCalled();
     $qb->addOrderBy('r.id')->shouldBeCalled();
     $this->addAttributeSorter($metric, 'DESC');
 }
 function it_retrieves_field_list_with_product_with_attributes($productRepository, $attributeRepository, $assocTypeManager, AttributeInterface $bar, AttributeInterface $baz, AssociationTypeInterface $association)
 {
     $bar->getCode()->willReturn('bar-code');
     $bar->isLocalizable()->willReturn(false);
     $bar->isScopable()->willReturn(false);
     $bar->getAttributeType()->willReturn(null);
     $baz->getCode()->willReturn('baz-code');
     $baz->isLocalizable()->willReturn(false);
     $baz->isScopable()->willReturn(false);
     $baz->getAttributeType()->willReturn(null);
     $association->getCode()->willReturn('association-type-code');
     $assocTypeManager->getAssociationTypes()->willReturn([$association]);
     $productRepository->getAvailableAttributeIdsToExport(['foo'])->willReturn(['bar', 'baz']);
     $attributeRepository->findBy(['id' => ['bar', 'baz']])->willReturn([$bar, $baz]);
     $this->getFieldsList(['foo'])->shouldReturn(["bar-code", "baz-code", "family", "categories", "groups", "association-type-code-groups", "association-type-code-products"]);
 }
 function it_adds_an_equal_filter_on_an_attribute_in_the_query($qb, $attrValidatorHelper, AttributeInterface $attribute, Expr $expr)
 {
     $attribute->getBackendType()->willReturn('backend_type');
     $attribute->getCode()->willReturn('code');
     $attribute->getId()->willReturn(42);
     $attribute->isLocalizable()->willReturn(false);
     $attribute->isScopable()->willReturn(false);
     $attrValidatorHelper->validateLocale($attribute, Argument::any())->shouldBeCalled();
     $attrValidatorHelper->validateScope($attribute, Argument::any())->shouldBeCalled();
     $qb->expr()->willReturn($expr);
     $qb->getRootAlias()->willReturn('p');
     $expr->eq(Argument::any(), true)->willReturn('filtercode.backend_type = true');
     $expr->literal(true)->willReturn(true);
     $qb->innerJoin('p.values', Argument::any(), 'WITH', Argument::any())->shouldBeCalled();
     $this->addAttributeFilter($attribute, '=', true);
 }
 function let(TranslationNormalizer $transnormalizer, AttributeInterface $attribute, AttributeGroupInterface $attributeGroup)
 {
     $this->beConstructedWith($transnormalizer);
     $transnormalizer->normalize(Argument::cetera())->willReturn([]);
     $attribute->getAttributeType()->willReturn('Yes/No');
     $attribute->getCode()->willReturn('attribute_size');
     $attribute->getGroup()->willReturn($attributeGroup);
     $attributeGroup->getCode()->willReturn('size');
     $attribute->isUnique()->willReturn(true);
     $attribute->isUseableAsGridFilter()->willReturn(false);
     $attribute->getAllowedExtensions()->willReturn(['csv', 'xml', 'json']);
     $attribute->getMetricFamily()->willReturn('Length');
     $attribute->getDefaultMetricUnit()->willReturn('Centimenter');
     $attribute->getReferenceDataName()->willReturn('color');
     $attribute->isLocalizable()->willReturn(true);
     $attribute->isScopable()->willReturn(false);
 }
 function it_adds_a_sorter_to_the_query($qb, AttributeInterface $attribute, Expr $expr)
 {
     $attribute->getId()->willReturn('42');
     $attribute->getCode()->willReturn('entity_code');
     $attribute->isLocalizable()->willReturn(false);
     $attribute->isScopable()->willReturn(false);
     $attribute->getBackendType()->willReturn('entity');
     $qb->getRootAlias()->willReturn('r');
     $qb->expr()->willReturn($expr);
     $qb->leftJoin('r.values', 'sorterVentity_code', 'WITH', 'sorterVentity_code.attribute = 42')->shouldBeCalled();
     $qb->leftJoin('sorterVentity_code.entity', 'sorterOentity_code', 'WITH', 'sorterOentity_code.attribute = 42')->shouldBeCalled();
     $expr->literal('en_US')->shouldBeCalled()->willReturn('en_US');
     $qb->leftJoin('sorterOentity_code.optionValues', 'sorterOVentity_code', 'WITH', 'sorterOVentity_code.locale = en_US')->shouldBeCalled();
     $qb->addOrderBy('sorterOentity_code.code', 'DESC')->shouldBeCalled();
     $qb->addOrderBy('sorterOVentity_code.value', 'DESC')->shouldBeCalled();
     $qb->addOrderBy('r.id')->shouldBeCalled();
     $this->addAttributeSorter($attribute, 'DESC', 'en_US');
 }
 function it_adds_an_attribute_sorter_to_the_query($qb, AttributeInterface $sku)
 {
     $sku->getId()->willReturn(42);
     $sku->getCode()->willReturn('sku');
     $sku->getBackendType()->willReturn('varchar');
     $sku->isLocalizable()->willReturn(false);
     $sku->isScopable()->willReturn(false);
     $qb->expr()->willReturn(new Expr());
     $qb->getRootAlias()->willReturn('p');
     $qb->getDQLPart('join')->willReturn([]);
     $qb->resetDQLPart('join')->shouldBeCalled();
     $condition = "sorterVsku.attribute = 42";
     $qb->leftJoin('p.values', 'sorterVsku', 'WITH', $condition)->shouldBeCalled();
     $qb->addOrderBy('sorterVsku.varchar', 'DESC')->shouldBeCalled();
     $qb->getRootAlias()->willReturn('p');
     $qb->addOrderBy("p.id")->shouldBeCalled();
     $this->addAttributeSorter($sku, 'DESC');
 }
 /**
  * Get all channel codes
  * @param AttributeInterface $attribute
  *
  * @return array
  */
 public function getChannelCodes(AttributeInterface $attribute = null)
 {
     $channelCodes = [];
     if (null === $attribute || $attribute->isScopable()) {
         foreach ($this->getChannels() as $channel) {
             $channelCodes[] = $channel->getCode();
         }
     }
     return $channelCodes;
 }
 /**
  * Prepare attribute view
  *
  * @param AttributeInterface    $attribute
  * @param ProductValueInterface $value
  * @param FormView              $view
  *
  * @return array
  */
 protected function prepareAttributeView(AttributeInterface $attribute, ProductValueInterface $value, FormView $view)
 {
     $attributeView = ['id' => $attribute->getId(), 'isRemovable' => $value->isRemovable(), 'code' => $attribute->getCode(), 'label' => $attribute->getLabel(), 'sortOrder' => $attribute->getSortOrder(), 'allowValueCreation' => in_array($attribute->getAttributeType(), $this->choiceAttributeTypes), 'locale' => $value->getLocale()];
     if ($attribute->isScopable()) {
         $attributeView['values'] = array_merge($this->getAttributeValues($attribute, $value->getLocale()), [$value->getScope() => $view]);
         ksort($attributeView['values']);
     } else {
         $attributeView['value'] = $view;
     }
     $classes = $this->getAttributeClasses($attribute);
     if (!empty($classes)) {
         $attributeView['classes'] = $classes;
     }
     return $attributeView;
 }
 /**
  * @param AttributeInterface $attribute
  * @param string             $locale
  * @param string             $scope
  *
  * @return string
  */
 protected function getValueCode(AttributeInterface $attribute, $locale, $scope)
 {
     $valueCode = $attribute->getCode();
     if ($attribute->isLocalizable()) {
         $valueCode .= '-' . $locale;
     }
     if ($attribute->isScopable()) {
         $valueCode .= '-' . $scope;
     }
     return $valueCode;
 }
 /**
  * Add a filter condition on an attribute
  *
  * @param AttributeFilterInterface $filter    the filter
  * @param AttributeInterface       $attribute the attribute
  * @param string                   $operator  the operator
  * @param mixed                    $value     the value to filter
  * @param array                    $context   the filter context
  *
  * @return ProductQueryBuilderInterface
  */
 protected function addAttributeFilter(AttributeFilterInterface $filter, AttributeInterface $attribute, $operator, $value, array $context)
 {
     $locale = $attribute->isLocalizable() ? $context['locale'] : null;
     $scope = $attribute->isScopable() ? $context['scope'] : null;
     $filter->setQueryBuilder($this->getQueryBuilder());
     $filter->addAttributeFilter($attribute, $operator, $value, $locale, $scope, $context);
     return $this;
 }
 function it_initializes_itself($productBuilder, $catalogContext, $attributeRepository, $massActionManager, LocaleInterface $deLocale, AttributeInterface $attr1, AttributeInterface $attr2, ProductValueInterface $prodVal1, ProductValueInterface $prodVal2, ChannelInterface $channel, AttributeGroupInterface $attrGroup)
 {
     $deLocale->getCode()->willReturn('de_DE');
     $this->setLocale($deLocale);
     $catalogContext->setLocaleCode('de_DE')->shouldBeCalled();
     $attributeRepository->findWithGroups([], ['conditions' => ['unique' => 0]])->shouldBeCalled()->willReturn([$attr1, $attr2]);
     $attr1->setLocale('de_DE')->shouldBeCalled();
     $attr2->setLocale('de_DE')->shouldBeCalled();
     $attr1->getGroup()->willReturn($attrGroup);
     $attr2->getGroup()->willReturn($attrGroup);
     $attrGroup->setLocale('de_DE')->shouldBeCalledTimes(2);
     $massActionManager->filterLocaleSpecificAttributes([$attr1, $attr2], 'de_DE')->willReturn([$attr1, $attr2]);
     // First attribute
     $deLocale->getChannels()->willReturn([$channel]);
     $attr1->isScopable()->willReturn(true);
     $attr1->getCode()->willReturn('color');
     $channel->getCode()->willReturn('mobile');
     $productBuilder->createProductValue($attr1, 'de_DE', 'mobile')->shouldBeCalled()->willReturn($prodVal1);
     $productBuilder->addMissingPrices($prodVal1)->shouldBeCalled();
     // Second attribute
     $attr2->isScopable()->willReturn(false);
     $attr2->getCode()->willReturn('price');
     $productBuilder->createProductValue($attr2, 'de_DE')->shouldBeCalled()->willReturn($prodVal2);
     $productBuilder->addMissingPrices($prodVal2)->shouldBeCalled();
     $this->initialize();
     $this->getValues()->shouldHaveCount(2);
 }
 /**
  * Get the name of a normalized data field
  *
  * @param AttributeInterface $attribute
  * @param ChannelInterface   $channel
  * @param LocaleInterface    $locale
  *
  * @return string
  */
 protected function getNormalizedFieldName(AttributeInterface $attribute, ChannelInterface $channel, LocaleInterface $locale)
 {
     $suffix = '';
     if ($attribute->isLocalizable()) {
         $suffix = sprintf('-%s', $locale->getCode());
     }
     if ($attribute->isScopable()) {
         $suffix .= sprintf('-%s', $channel->getCode());
     }
     return $attribute->getCode() . $suffix;
 }
 function it_normalizes_product_with_price($filter, ProductInterface $product, AttributeInterface $priceAttribute, ProductValueInterface $price, Collection $prices, Collection $values, ProductPriceInterface $productPrice, FamilyInterface $family, SerializerInterface $serializer)
 {
     $family->getCode()->willReturn('shoes');
     $priceAttribute->getCode()->willReturn('price');
     $priceAttribute->getAttributeType()->willReturn('pim_catalog_price_collection');
     $priceAttribute->isLocalizable()->willReturn(false);
     $priceAttribute->isScopable()->willReturn(false);
     $price->getAttribute()->willReturn($priceAttribute);
     $price->getData()->willReturn(null);
     $productPrice->getData()->willReturn("356.00");
     $productPrice->getCurrency()->willReturn("EUR");
     $prices->add($productPrice);
     $price->getPrices()->willReturn($prices);
     $product->getIdentifier()->willReturn($price);
     $product->getFamily()->willReturn($family);
     $product->isEnabled()->willReturn(true);
     $product->getGroupCodes()->willReturn('group1, group2, variant_group_1');
     $product->getCategoryCodes()->willReturn('nice shoes, converse');
     $product->getAssociations()->willReturn([]);
     $values->add($price);
     $product->getValues()->willReturn($values);
     $filter->filterCollection($values, 'pim.transform.product_value.flat', Argument::cetera())->willReturn([$price]);
     $serializer->normalize($price, 'flat', Argument::any())->willReturn(['price-EUR' => '356.00']);
     $this->normalize($product, 'flat', ['price-EUR' => ''])->shouldReturn(['price-EUR' => '356.00', 'family' => 'shoes', 'groups' => 'group1, group2, variant_group_1', 'categories' => 'nice shoes, converse', 'enabled' => 1]);
 }
 /**
  * Add all the values required by the given attribute
  *
  * @param AttributeInterface $attribute
  * @param LocaleInterface    $locale
  */
 protected function addValues(AttributeInterface $attribute, $locale)
 {
     if ($attribute->isScopable()) {
         foreach ($locale->getChannels() as $channel) {
             $key = $attribute->getCode() . '_' . $channel->getCode();
             $value = $this->productBuilder->createProductValue($attribute, $locale->getCode(), $channel->getCode());
             $this->productBuilder->addMissingPrices($value);
             $this->values[$key] = $value;
         }
     } else {
         $value = $this->productBuilder->createProductValue($attribute, $locale->getCode());
         $this->productBuilder->addMissingPrices($value);
         $this->values[$attribute->getCode()] = $value;
     }
 }