Добавлен асинхронный импорт товаров с параллельной загрузкой фото + исправлен баг со счётчиком SKU
- Реализован импорт Product из CSV/XLSX через Celery с прогресс-баром - Параллельная загрузка фото товаров с внешних URL (масштабируемость до 500+ товаров) - Добавлена модель ProductImportJob для отслеживания статуса импорта - Создан таск download_product_photo_async для загрузки фото в фоне - Интеграция с существующим ImageProcessor (синхронная обработка через use_async=False) - Добавлены view и template для импорта с real-time обновлением через AJAX FIX: Исправлен баг со счётчиком SKU - инкремент только после успешного сохранения - Добавлен SKUCounter.peek_next_value() - возвращает следующий номер БЕЗ инкремента - Добавлен SKUCounter.increment_counter() - инкрементирует счётчик - generate_product_sku() использует peek_next_value() вместо get_next_value() - Добавлен post_save сигнал increment_sku_counter_after_save() для инкремента после создания - Предотвращает пропуски номеров при ошибках валидации (например cost_price NULL) FIX: Исправлена ошибка с is_main в ProductPhoto - ProductPhoto не имеет поля is_main, используется только order - Первое фото (order=0) автоматически считается главным - Удалён параметр is_main из download_product_photo_async и _collect_photo_tasks Изменены файлы: - products/models/base.py - методы для управления счётчиком SKU - products/models/import_job.py - модель для отслеживания импорта - products/services/import_export.py - сервис импорта с поддержкой Celery - products/tasks.py - таски для асинхронного импорта и загрузки фото - products/signals.py - сигнал для инкремента счётчика после сохранения - products/utils/sku_generator.py - использование peek_next_value() - products/views/product_import_views.py - view для импорта - products/templates/products/product_import*.html - UI для импорта - docker/entrypoint.sh - настройка Celery worker (concurrency=4) - requirements.txt - добавлен requests для загрузки фото
This commit is contained in:
@@ -22,6 +22,12 @@ urlpatterns = [
|
||||
path('product/<int:pk>/update/', views.ProductUpdateView.as_view(), name='product-update'),
|
||||
path('product/<int:pk>/delete/', views.ProductDeleteView.as_view(), name='product-delete'),
|
||||
|
||||
# Import/Export
|
||||
path('import/', views.product_import_view, name='product-import'),
|
||||
path('import/status/<int:job_id>/', views.product_import_status_view, name='product-import-status'),
|
||||
path('import/status/<int:job_id>/api/', views.product_import_status_api, name='product-import-status-api'),
|
||||
path('import/errors/download/', views.download_import_errors, name='product-import-errors-download'),
|
||||
|
||||
# Photo management for Product
|
||||
path('product/photo/<int:pk>/delete/', views.product_photo_delete, name='product-photo-delete'),
|
||||
path('product/photo/<int:pk>/set-main/', views.product_photo_set_main, name='product-photo-set-main'),
|
||||
|
||||
Reference in New Issue
Block a user