What is the right(official) way to programmatically add product attribute option in M2?
E.g. for manufacturer product attribute. Obviously existing option would be matched by "Admin" title value.
7 Answers 7
Here's the approach I've come up with for handling attribute options. Helper class:
<?php
namespace My\Module\Helper;
class Data extends \Magento\Framework\App\Helper\AbstractHelper
{
 /**
 * @var \Magento\Catalog\Api\ProductAttributeRepositoryInterface
 */
 protected $attributeRepository;
 /**
 * @var array
 */
 protected $attributeValues;
 /**
 * @var \Magento\Eav\Model\Entity\Attribute\Source\TableFactory
 */
 protected $tableFactory;
 /**
 * @var \Magento\Eav\Api\AttributeOptionManagementInterface
 */
 protected $attributeOptionManagement;
 /**
 * @var \Magento\Eav\Api\Data\AttributeOptionLabelInterfaceFactory
 */
 protected $optionLabelFactory;
 /**
 * @var \Magento\Eav\Api\Data\AttributeOptionInterfaceFactory
 */
 protected $optionFactory;
 /**
 * Data constructor.
 *
 * @param \Magento\Framework\App\Helper\Context $context
 * @param \Magento\Catalog\Api\ProductAttributeRepositoryInterface $attributeRepository
 * @param \Magento\Eav\Model\Entity\Attribute\Source\TableFactory $tableFactory
 * @param \Magento\Eav\Api\AttributeOptionManagementInterface $attributeOptionManagement
 * @param \Magento\Eav\Api\Data\AttributeOptionLabelInterfaceFactory $optionLabelFactory
 * @param \Magento\Eav\Api\Data\AttributeOptionInterfaceFactory $optionFactory
 */
 public function __construct(
 \Magento\Framework\App\Helper\Context $context,
 \Magento\Catalog\Api\ProductAttributeRepositoryInterface $attributeRepository,
 \Magento\Eav\Model\Entity\Attribute\Source\TableFactory $tableFactory,
 \Magento\Eav\Api\AttributeOptionManagementInterface $attributeOptionManagement,
 \Magento\Eav\Api\Data\AttributeOptionLabelInterfaceFactory $optionLabelFactory,
 \Magento\Eav\Api\Data\AttributeOptionInterfaceFactory $optionFactory
 ) {
 parent::__construct($context);
 $this->attributeRepository = $attributeRepository;
 $this->tableFactory = $tableFactory;
 $this->attributeOptionManagement = $attributeOptionManagement;
 $this->optionLabelFactory = $optionLabelFactory;
 $this->optionFactory = $optionFactory;
 }
 /**
 * Get attribute by code.
 *
 * @param string $attributeCode
 * @return \Magento\Catalog\Api\Data\ProductAttributeInterface
 */
 public function getAttribute($attributeCode)
 {
 return $this->attributeRepository->get($attributeCode);
 }
 /**
 * Find or create a matching attribute option
 *
 * @param string $attributeCode Attribute the option should exist in
 * @param string $label Label to find or add
 * @return int
 * @throws \Magento\Framework\Exception\LocalizedException
 */
 public function createOrGetId($attributeCode, $label)
 {
 if (strlen($label) < 1) {
 throw new \Magento\Framework\Exception\LocalizedException(
 __('Label for %1 must not be empty.', $attributeCode)
 );
 }
 // Does it already exist?
 $optionId = $this->getOptionId($attributeCode, $label);
 if (!$optionId) {
 // If no, add it.
 /** @var \Magento\Eav\Model\Entity\Attribute\OptionLabel $optionLabel */
 $optionLabel = $this->optionLabelFactory->create();
 $optionLabel->setStoreId(0);
 $optionLabel->setLabel($label);
 $option = $this->optionFactory->create();
 $option->setLabel($optionLabel->getLabel());
 $option->setStoreLabels([$optionLabel]);
 $option->setSortOrder(0);
 $option->setIsDefault(false);
 $this->attributeOptionManagement->add(
 \Magento\Catalog\Model\Product::ENTITY,
 $this->getAttribute($attributeCode)->getAttributeId(),
 $option
 );
 // Get the inserted ID. Should be returned from the installer, but it isn't.
 $optionId = $this->getOptionId($attributeCode, $label, true);
 }
 return $optionId;
 }
 /**
 * Find the ID of an option matching $label, if any.
 *
 * @param string $attributeCode Attribute code
 * @param string $label Label to find
 * @param bool $force If true, will fetch the options even if they're already cached.
 * @return int|false
 */
 public function getOptionId($attributeCode, $label, $force = false)
 {
 /** @var \Magento\Catalog\Model\ResourceModel\Eav\Attribute $attribute */
 $attribute = $this->getAttribute($attributeCode);
 // Build option array if necessary
 if ($force === true || !isset($this->attributeValues[ $attribute->getAttributeId() ])) {
 $this->attributeValues[ $attribute->getAttributeId() ] = [];
 // We have to generate a new sourceModel instance each time through to prevent it from
 // referencing its _options cache. No other way to get it to pick up newly-added values.
 /** @var \Magento\Eav\Model\Entity\Attribute\Source\Table $sourceModel */
 $sourceModel = $this->tableFactory->create();
 $sourceModel->setAttribute($attribute);
 foreach ($sourceModel->getAllOptions() as $option) {
 $this->attributeValues[ $attribute->getAttributeId() ][ $option['label'] ] = $option['value'];
 }
 }
 // Return option ID if exists
 if (isset($this->attributeValues[ $attribute->getAttributeId() ][ $label ])) {
 return $this->attributeValues[ $attribute->getAttributeId() ][ $label ];
 }
 // Return false if does not exist
 return false;
 }
}
Then, either in the same class or including it via dependency injection, you can add or get your option ID by calling createOrGetId($attributeCode, $label).
For example, if you inject My\Module\Helper\Data as $this->moduleHelper, then you can call:
$manufacturerId = $this->moduleHelper->createOrGetId('manufacturer', 'ABC Corp');
If 'ABC Corp' is an existing manufacturer, it will pull the ID. If not, it will add it.
UPDATED 2016年09月09日: Per Ruud N., the original solution used CatalogSetup, which resulted in a bug starting in Magento 2.1. This revised solution bypasses that model, creating the option and label explicitly. It should work on 2.0+.
- 
 3It's as official as you're going to get. All of the lookups and option adding go through core Magento. My class is just a wrapper for those core methods that makes them easier to use.Ryan Hoerr– Ryan Hoerr2016年02月29日 15:35:34 +00:00Commented Feb 29, 2016 at 15:35
- 
 1Hi Ryan, you shouldn't set the value on the option, this is the internal id magento uses and I found out the hard way that if you set the value to a string value with a leading number like '123 abc corp' it causes some serious problems due to the implementation ofMagento\Eav\Model\ResourceModel\Entity\Attribute::_processAttributeOptions. See for yourself, if you remove the$option->setValue($label);statement from your code, it will save the option, then when you fetch it Magento will return the value from an auto-increment on theeav_attribute_optiontable.quickshiftin– quickshiftin2016年09月12日 19:37:12 +00:00Commented Sep 12, 2016 at 19:37
- 
 2if I add this in a foreach function, in the second iteration I will get the error "Magento\Eav\Model\Entity\Attribute\OptionManagement::setOptionValue() must be of the type string, object given"JJ15– JJ152018年12月12日 14:57:48 +00:00Commented Dec 12, 2018 at 14:57
- 
 1Yes this code not workingSourav– Sourav2019年03月23日 11:45:21 +00:00Commented Mar 23, 2019 at 11:45
- 
 2@JELLEJ If you are getting issue Uncaught TypeError: Argument 3 passed to Magento\Eav\Model\Entity\Attribute\OptionManagement::setOptionValue() must be of the type string, object given in foreach function then change $option->setLabel($optionLabel); to $option->setLabel($label); at line 102Nadeem0035– Nadeem00352019年04月17日 07:21:45 +00:00Commented Apr 17, 2019 at 7:21
tested on Magento 2.1.3.
I didn't find any workable way how to create attribute with options at once. So initially we need to create an attribute and then add options for it.
Inject following class \Magento\Eav\Setup\EavSetupFactory
 $setup->startSetup();
 /** @var \Magento\Eav\Setup\EavSetup $eavSetup */
 $eavSetup = $this->eavSetupFactory->create(['setup' => $setup]);
Create new attribute:
$eavSetup->addAttribute(
 'catalog_product',
 $attributeCode,
 [
 'type' => 'varchar',
 'input' => 'select',
 'required' => false,
 ...
 ],
);
Add custom options.
Function addAttribute doesn't return anything useful which can be used in future. So after attribute creation we need to retrieve attribute object by ourself. !!!Important We need it because function expects only attribute_id, but don't want to work with attribute_code.
In that case we need to get attribute_id and pass it to attribute creation function.
$attributeId = $eavSetup->getAttributeId('catalog_product', 'attribute_code');
Then we need to generate options array in the way magento expects:
$options = [
 'values' => [
 'sort_order1' => 'title1',
 'sort_order2' => 'title2',
 'sort_order3' => 'title3',
 ],
 'attribute_id' => 'some_id',
];
As example:
$options = [
 'values' => [
 '1' => 'Red',
 '2' => 'Yellow',
 '3' => 'Green',
 ],
 'attribute_id' => '32',
];
And pass it to function:
$eavSetup->addAttributeOption($options);
- 
 3rd param of addAttribute can take the array parameter ['option']DWils– DWils2019年05月07日 19:38:55 +00:00Commented May 7, 2019 at 19:38
Using the Magento\Eav\Setup\EavSetupFactory or even the \Magento\Catalog\Setup\CategorySetupFactory class may lead to the following problem: https://github.com/magento/magento2/issues/4896.
The classes you should use:
protected $_logger;
protected $_attributeRepository;
protected $_attributeOptionManagement;
protected $_option;
protected $_attributeOptionLabel;
 public function __construct(
 \Psr\Log\LoggerInterface $logger,
 \Magento\Eav\Model\AttributeRepository $attributeRepository,
 \Magento\Eav\Api\AttributeOptionManagementInterface $attributeOptionManagement,
 \Magento\Eav\Api\Data\AttributeOptionLabelInterface $attributeOptionLabel,
 \Magento\Eav\Model\Entity\Attribute\Option $option
 ){
 $this->_logger = $logger;
 $this->_attributeRepository = $attributeRepository;
 $this->_attributeOptionManagement = $attributeOptionManagement;
 $this->_option = $option;
 $this->_attributeOptionLabel = $attributeOptionLabel;
 }
Then in your function do something like this:
 $attribute_id = $this->_attributeRepository->get('catalog_product', 'your_attribute')->getAttributeId();
$options = $this->_attributeOptionManagement->getItems('catalog_product', $attribute_id);
/* if attribute option already exists, remove it */
foreach($options as $option) {
 if ($option->getLabel() == $oldname) {
 $this->_attributeOptionManagement->delete('catalog_product', $attribute_id, $option->getValue());
 }
}
/* new attribute option */
 $this->_option->setValue($name);
 $this->_attributeOptionLabel->setStoreId(0);
 $this->_attributeOptionLabel->setLabel($name);
 $this->_option->setLabel($this->_attributeOptionLabel);
 $this->_option->setStoreLabels([$this->_attributeOptionLabel]);
 $this->_option->setSortOrder(0);
 $this->_option->setIsDefault(false);
 $this->_attributeOptionManagement->add('catalog_product', $attribute_id, $this->_option);
- 
 1Thanks, you are correct. I've updated my answer accordingly. Note that$attributeOptionLabeland$optionare ORM classes; you should not inject them directly. The proper approach is to inject their factory class, then create an instance as needed. Also note you aren't using the API data interfaces consistently.Ryan Hoerr– Ryan Hoerr2016年09月09日 14:55:53 +00:00Commented Sep 9, 2016 at 14:55
- 
 3Hi @Rudd, see my comment on Ryan's answer. You don't want to call$option->setValue()as that is for an internal magentooption_idfield on theeav_attribute_optiontable.quickshiftin– quickshiftin2016年09月12日 19:42:42 +00:00Commented Sep 12, 2016 at 19:42
- 
 Thank you. That's what I found out too. Will edit my answer accordingly.Ruud N.– Ruud N.2016年09月14日 05:49:45 +00:00Commented Sep 14, 2016 at 5:49
- 
 1@RuudN.,not working for me, \OptionManagement::setOptionValue() must be of the type string, object given, called inJafar Pinjar– Jafar Pinjar2020年03月23日 07:39:36 +00:00Commented Mar 23, 2020 at 7:39
Here's updated code based on Ryan Hoerr. Since magento 2.3.x it's a little bit different.
Update createOrGetId function.
public function createOrGetId($attributeCode, $label) {
 if (strlen($label) < 1) {
 throw new \Magento\Framework\Exception\LocalizedException(
 __('Label for %1 must not be empty.', $attributeCode)
 );
 }
 try {
 $optionId = $this->getOptionId($attributeCode, $label);
 if (!$optionId) {
 // If no, add it.
 /** @var \Magento\Eav\Model\Entity\Attribute\OptionLabel $optionLabel */
 $optionLabel = $this->optionLabelFactory->create();
 $optionLabel->setStoreId(0);
 $optionLabel->setLabel($label);
 $option = $this->optionFactory->create();
 $option->setLabel($optionLabel->getLabel());
 // $option->setLabel($label);
 $option->setStoreLabels([$optionLabel]);
 $option->setSortOrder(0);
 $option->setIsDefault(false);
 $this->attributeOptionManagement->add(
 \Magento\Catalog\Model\Product::ENTITY,
 $this->getAttribute($attributeCode)->getAttributeId(),
 $option
 );
 // Get the inserted ID. Should be returned from the installer, but it isn't.
 $optionId = $this->getOptionId($attributeCode, $label, true);
 }
 } catch (\Exception $e) {
 throw new \Exception($e->getMessage());
 }
 // Does it already exist?
 return $optionId;
 }
For Magento 2.3.3 I found that you can take Magento DevTeam approach.
- Add Patch
bin/magento setup:db-declaration:generate-patch Vendor_Module PatchName
- Add CategorySetupFactory to constructor
public function __construct( ModuleDataSetupInterface $moduleDataSetup, Factory $configFactory CategorySetupFactory $categorySetupFactory ) { $this->moduleDataSetup = $moduleDataSetup; $this->configFactory = $configFactory; $this->categorySetupFactory = $categorySetupFactory; }
- Add attribute in apply() function - public function apply() { $categorySetup = $this->categorySetupFactory->create(['setup' => $this->moduleDataSetup]); $categorySetup->addAttribute( \Magento\Catalog\Model\Product::ENTITY, 'custom_layout', [ 'type' => 'varchar', 'label' => 'New Layout', 'input' => 'select', 'source' => \Magento\Catalog\Model\Product\Attribute\Source\Layout::class, 'required' => false, 'sort_order' => 50, 'global' => \Magento\Eav\Model\Entity\Attribute\ScopedAttributeInterface::SCOPE_STORE, 'group' => 'Schedule Design Update', 'is_used_in_grid' => true, 'is_visible_in_grid' => false, 'is_filterable_in_grid' => false ] ); }
- 
 uhmm i just find out that i wanted to add this answer to diffrent question. I will just live it here and add reference to this answer there. I hope that it is ok. This is partiaclly answer for this question as well :)embed0– embed02019年11月04日 13:59:31 +00:00Commented Nov 4, 2019 at 13:59
- 
 Can you please add the complete solution if possible ? Thanks in advance :)shankar boss– shankar boss2020年02月11日 07:16:38 +00:00Commented Feb 11, 2020 at 7:16
For those who are receiving error after implementing accepted answer of @reyan-hoeer
PHP Fatal error: Uncaught TypeError: Argument 3 passed to Magento\\Eav\\Model\\Entity\\Attribute\\OptionManagement::setOptionValue() must be of the type string, object given, called in ....
To fix this issue, update the function createOrGetId with following
 /**
 * Find or create a matching attribute option
 *
 * @param string $attributeCode Attribute the option should exist in
 * @param string $label Label to find or add
 * @return int
 * @throws \Magento\Framework\Exception\LocalizedException
 */
 public function createOrGetId($attributeCode, $label)
 {
 if (strlen($label) < 1) {
 throw new \Magento\Framework\Exception\LocalizedException(
 __('Label for %1 must not be empty.', $attributeCode)
 );
 }
 // Does it already exist?
 $optionId = $this->getOptionId($attributeCode, $label);
 if (!$optionId) {
 // If no, add it.
 try{
 /** @var \Magento\Eav\Model\Entity\Attribute\OptionLabel $optionLabel */
 $optionLabel = $this->optionLabelFactory->create();
 $optionLabel->setStoreId(0);
 $optionLabel->setLabel($label);
 $option = $this->optionFactory->create();
 $option->setLabel($label); // this line is changed to fix the error.
 $option->setStoreLabels([$optionLabel]);
 $option->setSortOrder(0);
 $option->setIsDefault(false);
 $this->attributeOptionManagement->add(
 \Magento\Catalog\Model\Product::ENTITY,
 $this->getAttribute($attributeCode)->getAttributeId(),
 $option
 );
 }catch(\Exception $e){
 $message = $e->getMessage();
 echo $message;
 }
 // Get the inserted ID. Should be returned from the installer, but it isn't.
 $optionId = $this->getOptionId($attributeCode, $label, true);
 }
 return $optionId;
 }
My two cents here (similar to some of the answers provided):
<?php
declare(strict_types=1);
namespace Acme\Setup\Setup\Patch\Data;
use Magento\Customer\Model\Indexer\Address\AttributeProvider;
use Magento\Customer\Setup\CustomerSetupFactory;
use Magento\Framework\Exception\LocalizedException;
use Magento\Framework\Setup\ModuleDataSetupInterface;
use Magento\Framework\Setup\Patch\DataPatchInterface;
class CustomerAddressTitleAttributeExtraOptions implements DataPatchInterface
{
 private ModuleDataSetupInterface $moduleDataSetup;
 private CustomerSetupFactory $customerSetupFactory;
 /**
 * @param ModuleDataSetupInterface $moduleDataSetup
 * @param CustomerSetupFactory $customerSetupFactory
 */
 public function __construct(
 ModuleDataSetupInterface $moduleDataSetup,
 CustomerSetupFactory $customerSetupFactory,
 ) {
 $this->moduleDataSetup = $moduleDataSetup;
 $this->customerSetupFactory = $customerSetupFactory;
 }
 /**
 * @return void
 * @throws LocalizedException
 */
 public function apply(): void
 {
 $customerSetup = $this->customerSetupFactory->create(['setup' => $this->moduleDataSetup]);
 $attributeCode = 'title';
 $attributeId = $customerSetup->getAttributeId(AttributeProvider::ENTITY, $attributeCode);
 $options = [
 "Baron",
 "Baroness",
 "Countess",
 "Dame",
 "Dr",
 "Earl",
 "Lady",
 "Lord",
 "Miss",
 "Mr",
 "Mrs",
 "Ms",
 "Prince",
 "Princess",
 "Professor",
 "Sir",
 "The Duchess of",
 "The Duchess of",
 "The Duke of",
 "The Earl of",
 "The Marchioness of",
 "The Marquess",
 "The Revd",
 "Viscount",
 "Viscountess",
 "The Hon",
 "The Hon Mrs",
 "None",
 ];
 $customerSetup->addAttributeOption([
 'values' => $options,
 'attribute_id' => $attributeId
 ]);
 }
 /**
 * {@inheritdoc}
 */
 public static function getDependencies(): array
 {
 return [];
 }
 /**
 * {@inheritdoc}
 */
 public function getAliases(): array
 {
 return [];
 }
}