data-migration-tool icon indicating copy to clipboard operation
data-migration-tool copied to clipboard

Undefined offset in Migration/Handler/VisualMerchandiser/SmartAttribute.php on line 30

Open dfelton opened this issue 9 years ago • 5 comments

Environment Information:

Data Migration Tool Version: 2.1.3 Mode: data Step: VisualMerchandiser Step Sub-step: data Source Database Version: Magento EE 1.14.2.0 Destination Database Version: Magento EE 2.1.3

Summary:

If a category's visual merchandiser has been configured to use the same attribute code more than once for that category, the data migration tool will pass the integrity check but fail in data migration due to undefined index.

Additional Information:

Consider the following setup on a category: foo_bar

The values for this configuration in the merchandiser_category_values table of the MapInterface::TYPE_SOURCE database will be as follows:

attribute_codes: name
smart_attributes: a:2:{s:18:"_1484069909215_215";a:3:{s:9:"attribute";s:2:"65";s:5:"value";s:3:"Foo";s:4:"link";s:2:"OR";}s:18:"_1484069916735_735";a:3:{s:9:"attribute";s:2:"65";s:5:"value";s:3:"Bar";s:4:"link";s:2:"OR";}}

Come migration time, the data migration tool expects the number of comma separated attribute_codes and the size of the serialized array to match. However because the attribute_codes are only stored uniquely in the source database, they will not match if used more than once.

Migration\Handler\VisualMerchandiser\SmartAttribute::handle()

The size of $attributeCodeArr and $attributeValues will not match, causing an error to be thrown on line 30.

[Exception]
Notice: Undefined offset: 1 in /path/to/install/vendor/magento/data-migration-tool/src/Migration/Handler/VisualMerchandiser/SmartAttribute.php on line 30

dfelton avatar Jan 10 '17 18:01 dfelton

Thank you for reporting the issue! Internal ticket to fix it MAGETWO-63104

victor-v-rad avatar Feb 03 '17 07:02 victor-v-rad

I am also facing the same issue when migrating Magento EE 14.1.0 to EE 2.1.8. Please let us know if this issue is resolved or not?

shivendraranosys avatar Oct 15 '17 06:10 shivendraranosys

not solved yet

victor-v-rad avatar Nov 23 '17 09:11 victor-v-rad

Is there an update to this issue? I hit this overnight trying to do data migration.

npropst78 avatar Jun 21 '18 14:06 npropst78

Hello,

We also faced this issue. The problem is the way handler is trying to get the attribute codes, I wrote this workaround to get the attribute code using the attribute_id that is the attribute value inside the source data.

  1. Create custom class to extend \Migration\Handler\VisualMerchandiser\SmartAttribute

  2. Inject the adapter factory

   protected $attributesCached = [];

    public function __construct(\Migration\ResourceModel\AdapterFactory $adapterFactory)
    {
        $this->adapter = $adapterFactory->create(['resourceType' => 'source']);
    }
  1. Define method. You may need to update the table name of the eav attribute and the parameter 500 is the size of the attribute page, in our case we have around 350 attributes, you can change that :
    protected function getAttributeCode($attributeId) {
        $attributeTable = 'mg_eav_attribute';
        $attributes = $this->adapter->loadPage($attributeTable, 1, 500, 'attribute_id', 0);
        if (isset($this->attributesCached[$attributeId])) {
            return $this->attributesCached[$attributeId];
        }
        foreach ($attributes as $key => $data) {
            if (isset($data['attribute_id']) && $data['attribute_id'] == $attributeId && isset($data['attribute_code'])) {
                $this->attributesCached[$attributeId] = $data['attribute_code'];
                return $data['attribute_code'];
            }
        }
    }
  1. Update handle(Record $recordToHandle, Record $oppositeRecord) method, replace $attribute['attribute'] = $attributeCodeArr[$count]; with:
$attribute['attribute'] = $this->getAttributeCode($attributeValue['attribute']);
public function handle(Record $recordToHandle, Record $oppositeRecord)
    {
        $count = 0;
        $attributes = [];
        $this->validate($recordToHandle);
        $attributeCode = $recordToHandle->getValue(self::ATTRIBUTE_CODE_NAME);
        $attributeCodeArr = explode(',', $attributeCode);
        $attributeValues = unserialize($recordToHandle->getValue(self::ATTRIBUTE_VALUE_NAME));
        if (is_array($attributeValues)) {
            foreach ($attributeValues as $attributeValue) {
                $attribute = $this->parseOperator($attributeValue['value']);
                $attribute['attribute'] = $attributeCodeArr[$count];
                $attribute['logic'] = $attributeValue['link'];
                $count++;
                $attributes[] = $attribute;
            }
            $attributeString = \Zend_Json::encode($attributes);

            $recordToHandle->setValue($this->field, $attributeString);
        }
    }
  1. Update your visual_merchandiser_map.xml to use the new handler
           <transform>
                <field>merchandiser_category_values.smart_attributes</field>
                <handler class="< path to custom handler >\SmartAttribute"/>
            </transform>

I hope this help someone else

Regards

exequielserfe avatar Nov 20 '18 16:11 exequielserfe