1: <?php
2: 3: 4: 5: 6: 7: 8: 9: 10: 11: 12: 13: 14: 15: 16: 17: 18: 19: 20: 21: 22: 23: 24: 25:
26:
27:
28: 29: 30: 31: 32: 33: 34:
35: class Mage_Eav_Model_Entity_Setup extends Mage_Core_Model_Resource_Setup
36: {
37: 38: 39: 40: 41:
42: protected $_generalGroupName = 'General';
43:
44: 45: 46: 47: 48:
49: public $defaultGroupIdAssociations = array(
50: 'General' => 1
51: );
52:
53: 54: 55: 56: 57:
58: protected $_defaultGroupName = 'Default';
59:
60: 61: 62: 63: 64:
65: protected $_defaultAttributeSetName = 'Default';
66:
67: 68: 69: 70: 71:
72: public function cleanCache()
73: {
74: Mage::app()->cleanCache(array('eav'));
75: return $this;
76: }
77:
78: 79: 80: 81: 82:
83: public function installDefaultGroupIds()
84: {
85: $setIds = $this->getAllAttributeSetIds();
86: foreach ($this->defaultGroupIdAssociations as $defaultGroupName => $defaultGroupId) {
87: foreach ($setIds as $set) {
88: $groupId = $this->getTableRow('eav/attribute_group',
89: 'attribute_group_name', $defaultGroupName, 'attribute_group_id', 'attribute_set_id', $set
90: );
91: if (!$groupId) {
92: $groupId = $this->getTableRow('eav/attribute_group',
93: 'attribute_set_id', $set, 'attribute_group_id'
94: );
95: }
96: $this->updateTableRow('eav/attribute_group',
97: 'attribute_group_id', $groupId, 'default_id', $defaultGroupId
98: );
99: }
100: }
101:
102: return $this;
103: }
104:
105:
106:
107:
108: 109: 110: 111: 112: 113: 114: 115: 116:
117: public function addEntityType($code, array $params)
118: {
119: $data = array(
120: 'entity_type_code' => $code,
121: 'entity_model' => $params['entity_model'],
122: 'attribute_model' => $this->_getValue($params, 'attribute_model'),
123: 'entity_table' => $this->_getValue($params, 'table', 'eav/entity'),
124: 'value_table_prefix' => $this->_getValue($params, 'table_prefix'),
125: 'entity_id_field' => $this->_getValue($params, 'id_field'),
126: 'increment_model' => $this->_getValue($params, 'increment_model'),
127: 'increment_per_store' => $this->_getValue($params, 'increment_per_store', 0),
128: 'increment_pad_length' => $this->_getValue($params, 'increment_pad_length', 8),
129: 'increment_pad_char' => $this->_getValue($params, 'increment_pad_char', 0),
130: 'additional_attribute_table' => $this->_getValue($params, 'additional_attribute_table'),
131: 'entity_attribute_collection' => $this->_getValue($params, 'entity_attribute_collection'),
132: );
133:
134: if ($this->getEntityType($code, 'entity_type_id')) {
135: $this->updateEntityType($code, $data);
136: } else {
137: $this->_conn->insert($this->getTable('eav/entity_type'), $data);
138: }
139:
140: if (!empty($params['default_group'])) {
141: $defaultGroup = $params['default_group'];
142: } else {
143: $defaultGroup = $this->_defaultGroupName;
144: }
145:
146: $this->addAttributeSet($code, $this->_defaultAttributeSetName);
147: $this->addAttributeGroup($code, $this->_defaultGroupName, $this->_generalGroupName);
148:
149: return $this;
150: }
151:
152: 153: 154: 155: 156: 157: 158: 159:
160: public function updateEntityType($code, $field, $value = null)
161: {
162: $this->updateTableRow('eav/entity_type',
163: 'entity_type_id', $this->getEntityTypeId($code), $field, $value
164: );
165: return $this;
166: }
167:
168: 169: 170: 171: 172: 173: 174:
175: public function getEntityType($id, $field = null)
176: {
177: return $this->getTableRow('eav/entity_type',
178: is_numeric($id) ? 'entity_type_id' : 'entity_type_code', $id, $field
179: );
180: }
181:
182: 183: 184: 185: 186: 187:
188: public function getEntityTypeId($entityTypeId)
189: {
190: if (!is_numeric($entityTypeId)) {
191: $entityTypeId = $this->getEntityType($entityTypeId, 'entity_type_id');
192: }
193: if (!is_numeric($entityTypeId)) {
194: throw Mage::exception('Mage_Eav', Mage::helper('eav')->__('Wrong entity ID'));
195: }
196:
197: return $entityTypeId;
198: }
199:
200: 201: 202: 203: 204: 205:
206: public function removeEntityType($id)
207: {
208: if (is_numeric($id)) {
209: $this->deleteTableRow('eav/entity_type', 'entity_type_id', $id);
210: } else {
211: $this->deleteTableRow('eav/entity_type', 'entity_type_code', (string)$id);
212: }
213:
214: return $this;
215: }
216:
217:
218:
219: 220: 221: 222: 223: 224: 225:
226: public function getAttributeSetSortOrder($entityTypeId, $sortOrder = null)
227: {
228: if (!is_numeric($sortOrder)) {
229: $bind = array('entity_type_id' => $this->getEntityTypeId($entityTypeId));
230: $select = $this->_conn->select()
231: ->from($this->getTable('eav/attribute_set'), 'MAX(sort_order)')
232: ->where('entity_type_id = :entity_type_id');
233:
234: $sortOrder = $this->_conn->fetchOne($select, $bind) + 1;
235: }
236:
237: return $sortOrder;
238: }
239:
240: 241: 242: 243: 244: 245: 246: 247:
248: public function addAttributeSet($entityTypeId, $name, $sortOrder = null)
249: {
250: $data = array(
251: 'entity_type_id' => $this->getEntityTypeId($entityTypeId),
252: 'attribute_set_name' => $name,
253: 'sort_order' => $this->getAttributeSetSortOrder($entityTypeId, $sortOrder),
254: );
255:
256: $setId = $this->getAttributeSet($entityTypeId, $name, 'attribute_set_id');
257: if ($setId) {
258: $this->updateAttributeSet($entityTypeId, $setId, $data);
259: } else {
260: $this->_conn->insert($this->getTable('eav/attribute_set'), $data);
261:
262: $this->addAttributeGroup($entityTypeId, $name, $this->_generalGroupName);
263: }
264:
265: return $this;
266: }
267:
268: 269: 270: 271: 272: 273: 274: 275: 276:
277: public function updateAttributeSet($entityTypeId, $id, $field, $value = null)
278: {
279: $this->updateTableRow('eav/attribute_set',
280: 'attribute_set_id', $this->getAttributeSetId($entityTypeId, $id),
281: $field, $value,
282: 'entity_type_id', $this->getEntityTypeId($entityTypeId)
283: );
284: return $this;
285: }
286:
287: 288: 289: 290: 291: 292: 293: 294:
295: public function getAttributeSet($entityTypeId, $id, $field = null)
296: {
297: return $this->getTableRow('eav/attribute_set',
298: is_numeric($id) ? 'attribute_set_id' : 'attribute_set_name', $id,
299: $field,
300: 'entity_type_id', $this->getEntityTypeId($entityTypeId)
301: );
302: }
303:
304: 305: 306: 307: 308: 309: 310: 311:
312: public function getAttributeSetId($entityTypeId, $setId)
313: {
314: if (!is_numeric($setId)) {
315: $setId = $this->getAttributeSet($entityTypeId, $setId, 'attribute_set_id');
316: }
317: if (!is_numeric($setId)) {
318: throw Mage::exception('Mage_Eav', Mage::helper('eav')->__('Wrong attribute set ID'));
319: }
320:
321: return $setId;
322: }
323:
324: 325: 326: 327: 328: 329: 330:
331: public function removeAttributeSet($entityTypeId, $id)
332: {
333: $this->deleteTableRow('eav/attribute_set', 'attribute_set_id', $this->getAttributeSetId($entityTypeId, $id));
334: return $this;
335: }
336:
337: 338: 339: 340: 341: 342:
343: public function setDefaultSetToEntityType($entityType, $attributeSet = 'Default')
344: {
345: $entityTypeId = $this->getEntityTypeId($entityType);
346: $setId = $this->getAttributeSetId($entityTypeId, $attributeSet);
347: $this->updateEntityType($entityTypeId, 'default_attribute_set_id', $setId);
348: return $this;
349: }
350:
351: 352: 353: 354: 355:
356: public function getAllAttributeSetIds($entityTypeId = null)
357: {
358: $select = $this->_conn->select()
359: ->from($this->getTable('eav/attribute_set'), 'attribute_set_id');
360:
361: $bind = array();
362: if ($entityTypeId !== null) {
363: $bind['entity_type_id'] = $this->getEntityTypeId($entityTypeId);
364: $select->where('entity_type_id = :entity_type_id');
365: }
366:
367: return $this->_conn->fetchCol($select, $bind);
368: }
369:
370: 371: 372: 373: 374: 375:
376: public function getDefaultAttributeSetId($entityType)
377: {
378: $bind = array('entity_type' => $entityType);
379: if (is_numeric($entityType)) {
380: $where = 'entity_type_id = :entity_type';
381: } else {
382: $where = 'entity_type_code = :entity_type';
383: }
384: $select = $this->getConnection()->select()
385: ->from($this->getTable('eav/entity_type'), 'default_attribute_set_id')
386: ->where($where);
387:
388: return $this->getConnection()->fetchOne($select, $bind);
389: }
390:
391:
392:
393: 394: 395: 396: 397: 398: 399: 400:
401: public function getAttributeGroupSortOrder($entityTypeId, $setId, $sortOrder = null)
402: {
403: if (!is_numeric($sortOrder)) {
404: $bind = array('attribute_set_id' => $this->getAttributeSetId($entityTypeId, $setId));
405: $select = $this->_conn->select()
406: ->from($this->getTable('eav/attribute_group'), 'MAX(sort_order)')
407: ->where('attribute_set_id = :attribute_set_id');
408:
409: $sortOrder = $this->_conn->fetchOne($select, $bind) + 1;
410: }
411:
412: return $sortOrder;
413: }
414:
415: 416: 417: 418: 419: 420: 421: 422: 423:
424: public function addAttributeGroup($entityTypeId, $setId, $name, $sortOrder = null)
425: {
426: $setId = $this->getAttributeSetId($entityTypeId, $setId);
427: $data = array(
428: 'attribute_set_id' => $setId,
429: 'attribute_group_name' => $name,
430: );
431:
432: if (isset($this->defaultGroupIdAssociations[$name])) {
433: $data['default_id'] = $this->defaultGroupIdAssociations[$name];
434: }
435:
436: if ($sortOrder !== null) {
437: $data['sort_order'] = $sortOrder;
438: }
439:
440: $groupId = $this->getAttributeGroup($entityTypeId, $setId, $name, 'attribute_group_id');
441: if ($groupId) {
442: $this->updateAttributeGroup($entityTypeId, $setId, $groupId, $data);
443: } else {
444: if ($sortOrder === null) {
445: $data['sort_order'] = $this->getAttributeGroupSortOrder($entityTypeId, $setId, $sortOrder);
446: }
447: $this->_conn->insert($this->getTable('eav/attribute_group'), $data);
448: }
449:
450: return $this;
451: }
452:
453: 454: 455: 456: 457: 458: 459: 460: 461: 462:
463: public function updateAttributeGroup($entityTypeId, $setId, $id, $field, $value = null)
464: {
465: $this->updateTableRow('eav/attribute_group',
466: 'attribute_group_id', $this->getAttributeGroupId($entityTypeId, $setId, $id),
467: $field, $value,
468: 'attribute_set_id', $this->getAttributeSetId($entityTypeId, $setId)
469: );
470:
471: return $this;
472: }
473:
474: 475: 476: 477: 478: 479: 480: 481: 482:
483: public function getAttributeGroup($entityTypeId, $setId, $id, $field = null)
484: {
485: $searchId = $id;
486: if (is_numeric($id)) {
487: $searchField = 'attribute_group_id';
488: } else {
489: if (isset($this->defaultGroupIdAssociations[$id])) {
490: $searchField = 'default_id';
491: $searchId = $this->defaultGroupIdAssociations[$id];
492: } else {
493: $searchField = 'attribute_group_name';
494: }
495: }
496:
497: return $this->getTableRow('eav/attribute_group',
498: $searchField, $searchId, $field,
499: 'attribute_set_id', $this->getAttributeSetId($entityTypeId, $setId)
500: );
501: }
502:
503: 504: 505: 506: 507: 508: 509: 510:
511: public function getAttributeGroupId($entityTypeId, $setId, $groupId)
512: {
513: if (!is_numeric($groupId)) {
514: $groupId = $this->getAttributeGroup($entityTypeId, $setId, $groupId, 'attribute_group_id');
515: }
516:
517: if (!is_numeric($groupId)) {
518: $groupId = $this->getDefaultAttributeGroupId($entityTypeId, $setId);
519: }
520:
521: if (!is_numeric($groupId)) {
522: throw Mage::exception('Mage_Eav', Mage::helper('eav')->__('Wrong attribute group ID'));
523: }
524: return $groupId;
525: }
526:
527: 528: 529: 530: 531: 532: 533: 534:
535: public function removeAttributeGroup($entityTypeId, $setId, $id)
536: {
537: $this->deleteTableRow(
538: 'eav/attribute_group',
539: 'attribute_group_id',
540: $this->getAttributeGroupId($entityTypeId, $setId, $id)
541: );
542: return $this;
543: }
544:
545: 546: 547: 548: 549: 550: 551:
552: public function getDefaultAttributeGroupId($entityType, $attributeSetId = null)
553: {
554: $entityType = $this->getEntityTypeId($entityType);
555: if (!is_numeric($attributeSetId)) {
556: $attributeSetId = $this->getDefaultAttributeSetId($entityType);
557: }
558: $bind = array('attribute_set_id' => $attributeSetId);
559: $select = $this->getConnection()->select()
560: ->from($this->getTable('eav/attribute_group'), 'attribute_group_id')
561: ->where('attribute_set_id = :attribute_set_id')
562: ->order(array('default_id ' . Varien_Db_Select::SQL_DESC, 'sort_order'))
563: ->limit(1);
564:
565: return $this->getConnection()->fetchOne($select, $bind);
566: }
567:
568:
569:
570: 571: 572: 573: 574: 575: 576: 577:
578: protected function _getValue($array, $key, $default = null)
579: {
580: if (isset($array[$key]) && is_bool($array[$key])) {
581: $array[$key] = (int) $array[$key];
582: }
583: return isset($array[$key]) ? $array[$key] : $default;
584: }
585:
586: 587: 588: 589: 590: 591:
592: protected function _prepareValues($attr)
593: {
594: $data = array(
595: 'backend_model' => $this->_getValue($attr, 'backend'),
596: 'backend_type' => $this->_getValue($attr, 'type', 'varchar'),
597: 'backend_table' => $this->_getValue($attr, 'table'),
598: 'frontend_model' => $this->_getValue($attr, 'frontend'),
599: 'frontend_input' => $this->_getValue($attr, 'input', 'text'),
600: 'frontend_label' => $this->_getValue($attr, 'label'),
601: 'frontend_class' => $this->_getValue($attr, 'frontend_class'),
602: 'source_model' => $this->_getValue($attr, 'source'),
603: 'is_required' => $this->_getValue($attr, 'required', 1),
604: 'is_user_defined' => $this->_getValue($attr, 'user_defined', 0),
605: 'default_value' => $this->_getValue($attr, 'default'),
606: 'is_unique' => $this->_getValue($attr, 'unique', 0),
607: 'note' => $this->_getValue($attr, 'note'),
608: 'is_global' => $this->_getValue($attr, 'global',
609: Mage_Catalog_Model_Resource_Eav_Attribute::SCOPE_GLOBAL
610: ),
611: );
612:
613: return $data;
614: }
615:
616: 617: 618: 619: 620: 621: 622:
623: protected function _validateAttributeData($data)
624: {
625: $attributeCodeMaxLength = Mage_Eav_Model_Entity_Attribute::ATTRIBUTE_CODE_MAX_LENGTH;
626:
627: if (isset($data['attribute_code']) &&
628: !Zend_Validate::is($data['attribute_code'], 'StringLength', array('max' => $attributeCodeMaxLength)))
629: {
630: throw Mage::exception('Mage_Eav',
631: Mage::helper('eav')->__('Maximum length of attribute code must be less then %s symbols', $attributeCodeMaxLength)
632: );
633: }
634:
635: return true;
636: }
637:
638: 639: 640: 641: 642: 643: 644: 645: 646: 647:
648: public function addAttribute($entityTypeId, $code, array $attr)
649: {
650: $entityTypeId = $this->getEntityTypeId($entityTypeId);
651: $data = array_merge(
652: array(
653: 'entity_type_id' => $entityTypeId,
654: 'attribute_code' => $code
655: ),
656: $this->_prepareValues($attr)
657: );
658:
659: $this->_validateAttributeData($data);
660:
661: $sortOrder = isset($attr['sort_order']) ? $attr['sort_order'] : null;
662: $attributeId = $this->getAttribute($entityTypeId, $code, 'attribute_id');
663: if ($attributeId) {
664: $this->updateAttribute($entityTypeId, $attributeId, $data, null, $sortOrder);
665: } else {
666: $this->_insertAttribute($data);
667: }
668:
669: if (!empty($attr['group']) || empty($attr['user_defined'])) {
670: $select = $this->_conn->select()
671: ->from($this->getTable('eav/attribute_set'))
672: ->where('entity_type_id = :entity_type_id');
673: $sets = $this->_conn->fetchAll($select, array('entity_type_id' => $entityTypeId));
674: foreach ($sets as $set) {
675: if (!empty($attr['group'])) {
676: $this->addAttributeGroup($entityTypeId, $set['attribute_set_id'],
677: $attr['group']);
678: $this->addAttributeToSet($entityTypeId, $set['attribute_set_id'],
679: $attr['group'], $code, $sortOrder);
680: } else {
681: $this->addAttributeToSet($entityTypeId, $set['attribute_set_id'],
682: $this->_generalGroupName, $code, $sortOrder);
683: }
684: }
685: }
686:
687: if (isset($attr['option']) && is_array($attr['option'])) {
688: $option = $attr['option'];
689: $option['attribute_id'] = $this->getAttributeId($entityTypeId, $code);
690: $this->addAttributeOption($option);
691: }
692:
693: return $this;
694: }
695:
696: 697: 698: 699: 700:
701: public function addAttributeOption($option)
702: {
703: $optionTable = $this->getTable('eav/attribute_option');
704: $optionValueTable = $this->getTable('eav/attribute_option_value');
705:
706: if (isset($option['value'])) {
707: foreach ($option['value'] as $optionId => $values) {
708: $intOptionId = (int) $optionId;
709: if (!empty($option['delete'][$optionId])) {
710: if ($intOptionId) {
711: $condition = array('option_id =?' => $intOptionId);
712: $this->_conn->delete($optionTable, $condition);
713: }
714: continue;
715: }
716:
717: if (!$intOptionId) {
718: $data = array(
719: 'attribute_id' => $option['attribute_id'],
720: 'sort_order' => isset($option['order'][$optionId]) ? $option['order'][$optionId] : 0,
721: );
722: $this->_conn->insert($optionTable, $data);
723: $intOptionId = $this->_conn->lastInsertId($optionTable);
724: } else {
725: $data = array(
726: 'sort_order' => isset($option['order'][$optionId]) ? $option['order'][$optionId] : 0,
727: );
728: $this->_conn->update($optionTable, $data, array('option_id=?' => $intOptionId));
729: }
730:
731:
732: if (!isset($values[0])) {
733: Mage::throwException(Mage::helper('eav')->__('Default option value is not defined'));
734: }
735: $condition = array('option_id =?' => $intOptionId);
736: $this->_conn->delete($optionValueTable, $condition);
737: foreach ($values as $storeId => $value) {
738: $data = array(
739: 'option_id' => $intOptionId,
740: 'store_id' => $storeId,
741: 'value' => $value,
742: );
743: $this->_conn->insert($optionValueTable, $data);
744: }
745: }
746: } else if (isset($option['values'])) {
747: foreach ($option['values'] as $sortOrder => $label) {
748:
749: $data = array(
750: 'attribute_id' => $option['attribute_id'],
751: 'sort_order' => $sortOrder,
752: );
753: $this->_conn->insert($optionTable, $data);
754: $intOptionId = $this->_conn->lastInsertId($optionTable);
755:
756: $data = array(
757: 'option_id' => $intOptionId,
758: 'store_id' => 0,
759: 'value' => $label,
760: );
761: $this->_conn->insert($optionValueTable, $data);
762: }
763: }
764: }
765:
766: 767: 768: 769: 770: 771: 772: 773: 774: 775:
776: public function updateAttribute($entityTypeId, $id, $field, $value = null, $sortOrder = null)
777: {
778: $this->_updateAttribute($entityTypeId, $id, $field, $value, $sortOrder);
779: $this->_updateAttributeAdditionalData($entityTypeId, $id, $field, $value);
780: return $this;
781: }
782:
783: 784: 785: 786: 787: 788: 789: 790: 791: 792:
793: protected function _updateAttribute($entityTypeId, $id, $field, $value = null, $sortOrder = null)
794: {
795: if ($sortOrder !== null) {
796: $this->updateTableRow('eav/entity_attribute',
797: 'attribute_id', $this->getAttributeId($entityTypeId, $id),
798: 'sort_order', $sortOrder
799: );
800: }
801:
802: $attributeFields = $this->_getAttributeTableFields();
803: if (is_array($field)) {
804: $bind = array();
805: foreach ($field as $k => $v) {
806: if (isset($attributeFields[$k])) {
807: $bind[$k] = $this->getConnection()->prepareColumnValue($attributeFields[$k], $v);
808: }
809: }
810: if (!$bind) {
811: return $this;
812: }
813: $field = $bind;
814: } else {
815: if (!isset($attributeFields[$field])) {
816: return $this;
817: }
818: }
819:
820: $this->updateTableRow('eav/attribute',
821: 'attribute_id', $this->getAttributeId($entityTypeId, $id),
822: $field, $value,
823: 'entity_type_id', $this->getEntityTypeId($entityTypeId)
824: );
825:
826: return $this;
827: }
828:
829: 830: 831: 832: 833: 834: 835: 836: 837:
838: protected function _updateAttributeAdditionalData($entityTypeId, $id, $field, $value = null)
839: {
840: $additionalTable = $this->getEntityType($entityTypeId, 'additional_attribute_table');
841: if (!$additionalTable) {
842: return $this;
843: }
844: $additionalTableExists = $this->getConnection()->isTableExists($this->getTable($additionalTable));
845: if ($additionalTable && $additionalTableExists) {
846: $attributeFields = $this->getConnection()->describeTable($this->getTable($additionalTable));
847: if (is_array($field)) {
848: $bind = array();
849: foreach ($field as $k => $v) {
850: if (isset($attributeFields[$k])) {
851: $bind[$k] = $this->getConnection()->prepareColumnValue($attributeFields[$k], $v);
852: }
853: }
854: if (!$bind) {
855: return $this;
856: }
857: $field = $bind;
858: } else {
859: if (!isset($attributeFields[$field])) {
860: return $this;
861: }
862: }
863: $this->updateTableRow($this->getTable($additionalTable),
864: 'attribute_id', $this->getAttributeId($entityTypeId, $id),
865: $field, $value
866: );
867: }
868:
869: return $this;
870: }
871:
872: 873: 874: 875: 876: 877: 878: 879:
880: public function getAttribute($entityTypeId, $id, $field = null)
881: {
882: $additionalTable = $this->getEntityType($entityTypeId, 'additional_attribute_table');
883: $entityTypeId = $this->getEntityTypeId($entityTypeId);
884: $idField = is_numeric($id) ? 'attribute_id' : 'attribute_code';
885: if (!$additionalTable) {
886: return $this->getTableRow('eav/attribute', $idField, $id, $field, 'entity_type_id', $entityTypeId);
887: }
888:
889: $mainTable = $this->getTable('eav/attribute');
890: if (empty($this->_setupCache[$mainTable][$entityTypeId][$id])) {
891: $additionalTable = $this->getTable($additionalTable);
892: $bind = array(
893: 'id' => $id,
894: 'entity_type_id' => $entityTypeId
895: );
896: $select = $this->_conn->select()
897: ->from(array('main' => $mainTable))
898: ->join(
899: array('additional' => $additionalTable),
900: 'main.attribute_id = additional.attribute_id')
901: ->where("main.{$idField} = :id")
902: ->where('main.entity_type_id = :entity_type_id');
903:
904: $row = $this->_conn->fetchRow($select, $bind);
905: if (!$row) {
906: $this->_setupCache[$mainTable][$entityTypeId][$id] = false;
907: } else {
908: $this->_setupCache[$mainTable][$entityTypeId][$row['attribute_id']] = $row;
909: $this->_setupCache[$mainTable][$entityTypeId][$row['attribute_code']] = $row;
910: }
911: }
912:
913: $row = $this->_setupCache[$mainTable][$entityTypeId][$id];
914: if ($field !== null) {
915: return isset($row[$field]) ? $row[$field] : false;
916: }
917:
918: return $row;
919: }
920:
921: 922: 923: 924: 925: 926: 927:
928: public function getAttributeId($entityTypeId, $id)
929: {
930: if (!is_numeric($id)) {
931: $id = $this->getAttribute($entityTypeId, $id, 'attribute_id');
932: }
933: if (!is_numeric($id)) {
934: return false;
935: }
936: return $id;
937: }
938:
939: 940: 941: 942: 943: 944: 945:
946: public function getAttributeTable($entityTypeId, $id)
947: {
948: $entityKeyName = is_numeric($entityTypeId) ? 'entity_type_id' : 'entity_type_code';
949: $attributeKeyName = is_numeric($id) ? 'attribute_id' : 'attribute_code';
950:
951: $bind = array(
952: 'id' => $id,
953: 'entity_type_id' => $entityTypeId
954: );
955: $select = $this->getConnection()->select()
956: ->from(
957: array('entity_type' => $this->getTable('eav/entity_type')),
958: array('entity_table'))
959: ->join(
960: array('attribute' => $this->getTable('eav/attribute')),
961: 'attribute.entity_type_id = entity_type.entity_type_id',
962: array('backend_type'))
963: ->where("entity_type.{$entityKeyName} = :entity_type_id")
964: ->where("attribute.{$attributeKeyName} = :id")
965: ->limit(1);
966:
967: $result = $this->getConnection()->fetchRow($select, $bind);
968: if ($result) {
969: $table = $this->getTable($result['entity_table']);
970: if ($result['backend_type'] != 'static') {
971: $table .= '_' . $result['backend_type'];
972: }
973: return $table;
974: }
975:
976: return false;
977: }
978:
979: 980: 981: 982: 983: 984: 985:
986: public function removeAttribute($entityTypeId, $code)
987: {
988: $mainTable = $this->getTable('eav/attribute');
989: $attribute = $this->getAttribute($entityTypeId, $code);
990: if ($attribute) {
991: $this->deleteTableRow('eav/attribute', 'attribute_id', $attribute['attribute_id']);
992: if (isset($this->_setupCache[$mainTable][$attribute['entity_type_id']][$attribute['attribute_code']])) {
993: unset($this->_setupCache[$mainTable][$attribute['entity_type_id']][$attribute['attribute_code']]);
994: }
995: }
996: return $this;
997: }
998:
999: 1000: 1001: 1002: 1003: 1004: 1005: 1006: 1007:
1008: public function getAttributeSortOrder($entityTypeId, $setId, $groupId, $sortOrder = null)
1009: {
1010: if (!is_numeric($sortOrder)) {
1011: $bind = array('attribute_group_id' => $this->getAttributeGroupId($entityTypeId, $setId, $groupId));
1012: $select = $this->_conn->select()
1013: ->from($this->getTable('eav/entity_attribute'), 'MAX(sort_order)')
1014: ->where('attribute_group_id = :attribute_group_id');
1015:
1016: $sortOrder = $this->_conn->fetchOne($select, $bind) + 1;
1017: }
1018:
1019: return $sortOrder;
1020: }
1021:
1022: 1023: 1024: 1025: 1026: 1027: 1028: 1029: 1030: 1031:
1032: public function addAttributeToSet($entityTypeId, $setId, $groupId, $attributeId, $sortOrder=null)
1033: {
1034: $entityTypeId = $this->getEntityTypeId($entityTypeId);
1035: $setId = $this->getAttributeSetId($entityTypeId, $setId);
1036: $groupId = $this->getAttributeGroupId($entityTypeId, $setId, $groupId);
1037: $attributeId = $this->getAttributeId($entityTypeId, $attributeId);
1038: $table = $this->getTable('eav/entity_attribute');
1039:
1040: $bind = array(
1041: 'attribute_set_id' => $setId,
1042: 'attribute_id' => $attributeId
1043: );
1044: $select = $this->_conn->select()
1045: ->from($table)
1046: ->where('attribute_set_id = :attribute_set_id')
1047: ->where('attribute_id = :attribute_id');
1048: $result = $this->_conn->fetchRow($select, $bind);
1049:
1050: if ($result) {
1051: if ($result['attribute_group_id'] != $groupId) {
1052: $where = array('entity_attribute_id =?' => $result['entity_attribute_id']);
1053: $data = array('attribute_group_id' => $groupId);
1054: $this->_conn->update($table, $data, $where);
1055: }
1056: } else {
1057: $data = array(
1058: 'entity_type_id' => $entityTypeId,
1059: 'attribute_set_id' => $setId,
1060: 'attribute_group_id' => $groupId,
1061: 'attribute_id' => $attributeId,
1062: 'sort_order' => $this->getAttributeSortOrder($entityTypeId, $setId, $groupId, $sortOrder),
1063: );
1064:
1065: $this->_conn->insert($table, $data);
1066: }
1067:
1068: return $this;
1069: }
1070:
1071: 1072: 1073: 1074: 1075: 1076: 1077: 1078: 1079: 1080:
1081: public function addAttributeToGroup($entityType, $setId, $groupId, $attributeId, $sortOrder = null)
1082: {
1083: $entityType = $this->getEntityTypeId($entityType);
1084: $setId = $this->getAttributeSetId($entityType, $setId);
1085: $groupId = $this->getAttributeGroupId($entityType, $setId, $groupId);
1086: $attributeId = $this->getAttributeId($entityType, $attributeId);
1087:
1088: $data = array(
1089: 'entity_type_id' => $entityType,
1090: 'attribute_set_id' => $setId,
1091: 'attribute_group_id' => $groupId,
1092: 'attribute_id' => $attributeId,
1093: );
1094:
1095: $bind = array(
1096: 'entity_type_id' => $entityType,
1097: 'attribute_set_id' => $setId,
1098: 'attribute_id' => $attributeId
1099: );
1100: $select = $this->getConnection()->select()
1101: ->from($this->getTable('eav/entity_attribute'))
1102: ->where('entity_type_id = :entity_type_id')
1103: ->where('attribute_set_id = :attribute_set_id')
1104: ->where('attribute_id = :attribute_id');
1105: $row = $this->getConnection()->fetchRow($select, $bind);
1106: if ($row) {
1107:
1108: if ($sortOrder !== null) {
1109: $data['sort_order'] = $sortOrder;
1110: }
1111:
1112: $this->getConnection()->update(
1113: $this->getTable('eav/entity_attribute'),
1114: $data,
1115: $this->getConnection()->quoteInto('entity_attribute_id=?', $row['entity_attribute_id'])
1116: );
1117: } else {
1118: if ($sortOrder === null) {
1119: $select = $this->getConnection()->select()
1120: ->from($this->getTable('eav/entity_attribute'), 'MAX(sort_order)')
1121: ->where('entity_type_id = :entity_type_id')
1122: ->where('attribute_set_id = :attribute_set_id')
1123: ->where('attribute_id = :attribute_id');
1124:
1125: $sortOrder = $this->getConnection()->fetchOne($select, $bind) + 10;
1126: }
1127: $sortOrder = is_numeric($sortOrder) ? $sortOrder : 1;
1128: $data['sort_order'] = $sortOrder;
1129: $this->getConnection()->insert($this->getTable('eav/entity_attribute'), $data);
1130: }
1131:
1132: return $this;
1133: }
1134:
1135:
1136:
1137: 1138: 1139: 1140: 1141: 1142:
1143: public function installEntities($entities = null)
1144: {
1145: $this->cleanCache();
1146:
1147: if ($entities === null) {
1148: $entities = $this->getDefaultEntities();
1149: }
1150:
1151: foreach ($entities as $entityName=>$entity) {
1152: $this->addEntityType($entityName, $entity);
1153:
1154: $frontendPrefix = isset($entity['frontend_prefix']) ? $entity['frontend_prefix'] : '';
1155: $backendPrefix = isset($entity['backend_prefix']) ? $entity['backend_prefix'] : '';
1156: $sourcePrefix = isset($entity['source_prefix']) ? $entity['source_prefix'] : '';
1157:
1158: foreach ($entity['attributes'] as $attrCode => $attr) {
1159: if (!empty($attr['backend'])) {
1160: if ('_' === $attr['backend']) {
1161: $attr['backend'] = $backendPrefix;
1162: } elseif ('_' === $attr['backend']{0}) {
1163: $attr['backend'] = $backendPrefix.$attr['backend'];
1164: } else {
1165: $attr['backend'] = $attr['backend'];
1166: }
1167: }
1168: if (!empty($attr['frontend'])) {
1169: if ('_' === $attr['frontend']) {
1170: $attr['frontend'] = $frontendPrefix;
1171: } elseif ('_' === $attr['frontend']{0}) {
1172: $attr['frontend'] = $frontendPrefix.$attr['frontend'];
1173: } else {
1174: $attr['frontend'] = $attr['frontend'];
1175: }
1176: }
1177: if (!empty($attr['source'])) {
1178: if ('_' === $attr['source']) {
1179: $attr['source'] = $sourcePrefix;
1180: } elseif ('_' === $attr['source']{0}) {
1181: $attr['source'] = $sourcePrefix . $attr['source'];
1182: } else {
1183: $attr['source'] = $attr['source'];
1184: }
1185: }
1186:
1187: $this->addAttribute($entityName, $attrCode, $attr);
1188: }
1189: $this->setDefaultSetToEntityType($entityName);
1190: }
1191:
1192: return $this;
1193: }
1194:
1195:
1196:
1197:
1198: 1199: 1200: 1201: 1202: 1203: 1204: 1205: 1206: 1207:
1208: public function createEntityTables($baseTableName, array $options = array())
1209: {
1210: $isNoCreateMainTable = $this->_getValue($options, 'no-main', false);
1211: $isNoDefaultTypes = $this->_getValue($options, 'no-default-types', false);
1212: $customTypes = $this->_getValue($options, 'types', array());
1213: $tables = array();
1214:
1215: if (!$isNoCreateMainTable) {
1216: 1217: 1218:
1219: $connection = $this->getConnection();
1220: $mainTable = $connection
1221: ->newTable($this->getTable($baseTableName))
1222: ->addColumn('entity_id', Varien_Db_Ddl_Table::TYPE_INTEGER, null, array(
1223: 'identity' => true,
1224: 'nullable' => false,
1225: 'primary' => true,
1226: ), 'Entity Id')
1227: ->addColumn('entity_type_id', Varien_Db_Ddl_Table::TYPE_SMALLINT, null, array(
1228: 'unsigned' => true,
1229: 'nullable' => false,
1230: 'default' => '0',
1231: ), 'Entity Type Id')
1232: ->addColumn('attribute_set_id', Varien_Db_Ddl_Table::TYPE_SMALLINT, null, array(
1233: 'unsigned' => true,
1234: 'nullable' => false,
1235: 'default' => '0',
1236: ), 'Attribute Set Id')
1237: ->addColumn('increment_id', Varien_Db_Ddl_Table::TYPE_TEXT, 50, array(
1238: 'nullable' => false,
1239: 'default' => '',
1240: ), 'Increment Id')
1241: ->addColumn('store_id', Varien_Db_Ddl_Table::TYPE_SMALLINT, null, array(
1242: 'unsigned' => true,
1243: 'nullable' => false,
1244: 'default' => '0',
1245: ), 'Store Id')
1246: ->addColumn('created_at', Varien_Db_Ddl_Table::TYPE_TIMESTAMP, null, array(
1247: 'nullable' => false,
1248: ), 'Created At')
1249: ->addColumn('updated_at', Varien_Db_Ddl_Table::TYPE_TIMESTAMP, null, array(
1250: 'nullable' => false,
1251: ), 'Updated At')
1252: ->addColumn('is_active', Varien_Db_Ddl_Table::TYPE_SMALLINT, null, array(
1253: 'unsigned' => true,
1254: 'nullable' => false,
1255: 'default' => '1',
1256: ), 'Defines Is Entity Active')
1257: ->addIndex($this->getIdxName($baseTableName, array('entity_type_id')),
1258: array('entity_type_id'))
1259: ->addIndex($this->getIdxName($baseTableName, array('store_id')),
1260: array('store_id'))
1261: ->addForeignKey($this->getFkName($baseTableName, 'entity_type_id', 'eav/entity_type', 'entity_type_id'),
1262: 'entity_type_id', $this->getTable('eav/entity_type'), 'entity_type_id',
1263: Varien_Db_Ddl_Table::ACTION_CASCADE, Varien_Db_Ddl_Table::ACTION_CASCADE)
1264: ->addForeignKey($this->getFkName($baseTableName, 'store_id', 'core/store', 'store_id'),
1265: 'store_id', $this->getTable('core/store'), 'store_id',
1266: Varien_Db_Ddl_Table::ACTION_CASCADE, Varien_Db_Ddl_Table::ACTION_CASCADE)
1267: ->setComment('Eav Entity Main Table');
1268:
1269: $tables[$this->getTable($baseTableName)] = $mainTable;
1270: }
1271:
1272: $types = array();
1273: if (!$isNoDefaultTypes) {
1274: $types = array(
1275: 'datetime' => array(Varien_Db_Ddl_Table::TYPE_DATETIME, null),
1276: 'decimal' => array(Varien_Db_Ddl_Table::TYPE_DECIMAL, '12,4'),
1277: 'int' => array(Varien_Db_Ddl_Table::TYPE_INTEGER, null),
1278: 'text' => array(Varien_Db_Ddl_Table::TYPE_TEXT, '64k'),
1279: 'varchar' => array(Varien_Db_Ddl_Table::TYPE_TEXT, '255'),
1280: 'char' => array(Varien_Db_Ddl_Table::TYPE_TEXT, '255')
1281: );
1282: }
1283:
1284: if (!empty($customTypes)) {
1285: foreach ($customTypes as $type => $fieldType) {
1286: if (count($fieldType) != 2) {
1287: throw Mage::exception('Mage_Eav', Mage::helper('eav')->__('Wrong type definition for %s', $type));
1288: }
1289: $types[$type] = $fieldType;
1290: }
1291: }
1292:
1293: 1294: 1295:
1296: foreach ($types as $type => $fieldType) {
1297: $eavTableName = array($baseTableName, $type);
1298:
1299: $eavTable = $connection->newTable($this->getTable($eavTableName));
1300: $eavTable
1301: ->addColumn('value_id', Varien_Db_Ddl_Table::TYPE_INTEGER, null, array(
1302: 'identity' => true,
1303: 'nullable' => false,
1304: 'primary' => true,
1305: ), 'Value Id')
1306: ->addColumn('entity_type_id', Varien_Db_Ddl_Table::TYPE_SMALLINT, null, array(
1307: 'unsigned' => true,
1308: 'nullable' => false,
1309: 'default' => '0',
1310: ), 'Entity Type Id')
1311: ->addColumn('attribute_id', Varien_Db_Ddl_Table::TYPE_SMALLINT, null, array(
1312: 'unsigned' => true,
1313: 'nullable' => false,
1314: 'default' => '0',
1315: ), 'Attribute Id')
1316: ->addColumn('store_id', Varien_Db_Ddl_Table::TYPE_SMALLINT, null, array(
1317: 'unsigned' => true,
1318: 'nullable' => false,
1319: 'default' => '0',
1320: ), 'Store Id')
1321: ->addColumn('entity_id', Varien_Db_Ddl_Table::TYPE_INTEGER, null, array(
1322: 'unsigned' => true,
1323: 'nullable' => false,
1324: 'default' => '0',
1325: ), 'Entity Id')
1326: ->addColumn('value', $fieldType[0], $fieldType[1], array(
1327: 'nullable' => false,
1328: ), 'Attribute Value')
1329: ->addIndex($this->getIdxName($eavTableName, array('entity_type_id')),
1330: array('entity_type_id'))
1331: ->addIndex($this->getIdxName($eavTableName, array('attribute_id')),
1332: array('attribute_id'))
1333: ->addIndex($this->getIdxName($eavTableName, array('store_id')),
1334: array('store_id'))
1335: ->addIndex($this->getIdxName($eavTableName, array('entity_id')),
1336: array('entity_id'));
1337: if ($type !== 'text') {
1338: $eavTable->addIndex($this->getIdxName($eavTableName, array('attribute_id', 'value')),
1339: array('attribute_id', 'value'));
1340: $eavTable->addIndex($this->getIdxName($eavTableName, array('entity_type_id', 'value')),
1341: array('entity_type_id', 'value'));
1342: }
1343:
1344: $eavTable
1345: ->addForeignKey($this->getFkName($eavTableName, 'entity_id', $baseTableName, 'entity_id'),
1346: 'entity_id', $this->getTable($baseTableName), 'entity_id',
1347: Varien_Db_Ddl_Table::ACTION_CASCADE, Varien_Db_Ddl_Table::ACTION_CASCADE)
1348: ->addForeignKey($this->getFkName($eavTableName, 'entity_type_id', 'eav/entity_type', 'entity_type_id'),
1349: 'entity_type_id', $this->getTable('eav/entity_type'), 'entity_type_id',
1350: Varien_Db_Ddl_Table::ACTION_CASCADE, Varien_Db_Ddl_Table::ACTION_CASCADE)
1351: ->addForeignKey($this->getFkName($eavTableName, 'store_id', 'core/store', 'store_id'),
1352: 'store_id', $this->getTable('core/store'), 'store_id',
1353: Varien_Db_Ddl_Table::ACTION_CASCADE, Varien_Db_Ddl_Table::ACTION_CASCADE)
1354: ->setComment('Eav Entity Value Table');
1355:
1356: $tables[$this->getTable($eavTableName)] = $eavTable;
1357: }
1358:
1359: $connection->beginTransaction();
1360: try {
1361: foreach ($tables as $tableName => $table) {
1362: $connection->createTable($table);
1363: }
1364: $connection->commit();
1365: } catch (Exception $e) {
1366: $connection->rollBack();
1367: throw Mage::exception('Mage_Eav', Mage::helper('eav')->__('Can\'t create table: %s', $tableName));
1368: }
1369:
1370: return $this;
1371: }
1372:
1373: 1374: 1375: 1376: 1377:
1378: protected function _getAttributeTableFields()
1379: {
1380: return $this->getConnection()->describeTable($this->getTable('eav/attribute'));
1381: }
1382:
1383: 1384: 1385: 1386: 1387: 1388:
1389: protected function _insertAttribute(array $data)
1390: {
1391: $bind = array();
1392:
1393: $fields = $this->_getAttributeTableFields();
1394:
1395: foreach ($data as $k => $v) {
1396: if (isset($fields[$k])) {
1397: $bind[$k] = $this->getConnection()->prepareColumnValue($fields[$k], $v);
1398: }
1399: }
1400: if (!$bind) {
1401: return $this;
1402: }
1403:
1404: $this->getConnection()->insert($this->getTable('eav/attribute'), $bind);
1405: $attributeId = $this->getConnection()->lastInsertId($this->getTable('eav/attribute'));
1406: $this->_insertAttributeAdditionalData(
1407: $data['entity_type_id'],
1408: array_merge(array('attribute_id' => $attributeId), $data)
1409: );
1410:
1411: return $this;
1412: }
1413:
1414: 1415: 1416: 1417: 1418: 1419: 1420:
1421: protected function _insertAttributeAdditionalData($entityTypeId, array $data)
1422: {
1423: $additionalTable = $this->getEntityType($entityTypeId, 'additional_attribute_table');
1424: if (!$additionalTable) {
1425: return $this;
1426: }
1427: $additionalTableExists = $this->getConnection()->isTableExists($this->getTable($additionalTable));
1428: if ($additionalTable && $additionalTableExists) {
1429: $bind = array();
1430: $fields = $this->getConnection()->describeTable($this->getTable($additionalTable));
1431: foreach ($data as $k => $v) {
1432: if (isset($fields[$k])) {
1433: $bind[$k] = $this->getConnection()->prepareColumnValue($fields[$k], $v);
1434: }
1435: }
1436: if (!$bind) {
1437: return $this;
1438: }
1439: $this->getConnection()->insert($this->getTable($additionalTable), $bind);
1440: }
1441:
1442: return $this;
1443: }
1444: }
1445: