Добавлена автогенерация артикулов вариантов для ConfigurableProduct
Добавлено поле variant_sku в модель ConfigurableProductOption. Артикул варианта генерируется автоматически в формате VAR-XXXXXX-V1, VAR-XXXXXX-V2 и т.д. Счетчик не переиспользуется при удалении вариантов для защиты интеграций. Переименован property variant_sku в variant_base_sku для основного SKU. Обновлен шаблон с колонкой артикула варианта. Создана миграция для добавления поля и data migration для существующих записей. Назначение: дополнительный артикул для интеграций с внешними площадками.
This commit is contained in:
@@ -597,6 +597,12 @@ class ConfigurableProductOption(models.Model):
|
||||
default=False,
|
||||
verbose_name="Вариант по умолчанию"
|
||||
)
|
||||
variant_sku = models.CharField(
|
||||
max_length=50,
|
||||
blank=True,
|
||||
verbose_name="Артикул варианта",
|
||||
help_text="Дополнительный артикул для внешних площадок. Генерируется автоматически."
|
||||
)
|
||||
|
||||
class Meta:
|
||||
verbose_name = "Вариант товара"
|
||||
@@ -606,6 +612,7 @@ class ConfigurableProductOption(models.Model):
|
||||
models.Index(fields=['kit']),
|
||||
models.Index(fields=['product']),
|
||||
models.Index(fields=['parent', 'is_default']),
|
||||
models.Index(fields=['variant_sku']),
|
||||
]
|
||||
constraints = [
|
||||
# kit XOR product — один из двух должен быть заполнен
|
||||
@@ -622,6 +629,42 @@ class ConfigurableProductOption(models.Model):
|
||||
variant_name = self.kit.name if self.kit else (self.product.name if self.product else "N/A")
|
||||
return f"{self.parent.name} → {variant_name}"
|
||||
|
||||
def save(self, *args, **kwargs):
|
||||
"""При создании - генерируем variant_sku если не задан"""
|
||||
if not self.variant_sku and self.parent_id:
|
||||
# Генерируем артикул варианта
|
||||
self.variant_sku = self._generate_variant_sku()
|
||||
|
||||
super().save(*args, **kwargs)
|
||||
|
||||
def _generate_variant_sku(self):
|
||||
"""
|
||||
Генерирует артикул варианта в формате {parent.sku}-V{counter}.
|
||||
Счетчик не переиспользуется при удалении вариантов (защита интеграций).
|
||||
"""
|
||||
import re
|
||||
|
||||
# Получаем все варианты родителя с заполненным variant_sku
|
||||
existing_variants = ConfigurableProductOption.objects.filter(
|
||||
parent=self.parent,
|
||||
variant_sku__isnull=False
|
||||
).exclude(pk=self.pk).values_list('variant_sku', flat=True)
|
||||
|
||||
# Извлекаем номера из существующих variant_sku
|
||||
max_number = 0
|
||||
for sku in existing_variants:
|
||||
# Ищем паттерн -V\d+ в конце строки
|
||||
match = re.search(r'-V(\d+)$', sku)
|
||||
if match:
|
||||
number = int(match.group(1))
|
||||
max_number = max(max_number, number)
|
||||
|
||||
# Следующий номер
|
||||
next_number = max_number + 1
|
||||
|
||||
# Формируем артикул
|
||||
return f"{self.parent.sku}-V{next_number}"
|
||||
|
||||
@property
|
||||
def variant(self):
|
||||
"""Возвращает связанный вариант (kit или product)"""
|
||||
@@ -638,8 +681,8 @@ class ConfigurableProductOption(models.Model):
|
||||
return self.variant.name if self.variant else None
|
||||
|
||||
@property
|
||||
def variant_sku(self):
|
||||
"""SKU варианта"""
|
||||
def variant_base_sku(self):
|
||||
"""Основной SKU варианта (Product/ProductKit)"""
|
||||
return self.variant.sku if self.variant else None
|
||||
|
||||
@property
|
||||
|
||||
Reference in New Issue
Block a user