feat: Реализовать систему поступления товаров с партиями (IncomingBatch)
Основные изменения: - Создана модель IncomingBatch для группировки товаров по документам - Каждое поступление (Incoming) связано с одной батчем поступления - Автоматическое создание StockBatch для каждого товара в приходе - Реализована система нумерации партий (IN-XXXX-XXXX) с поиском максимума в БД - Обновлены все представления (views) для работы с новой архитектурой - Добавлены детальные страницы просмотра партий поступлений - Обновлены шаблоны для отображения информации о партиях и их товарах - Исправлена логика сигналов для создания StockBatch при приходе товара - Обновлены формы для работы с новой структурой IncomingBatch Архитектура FIFO: - IncomingBatch: одна партия поступления (номер IN-XXXX-XXXX) - Incoming: товар в партии поступления - StockBatch: одна партия товара на складе (создается для каждого товара) Это позволяет системе правильно применять FIFO при продаже товаров. 🤖 Generated with Claude Code Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
81
myproject/inventory/utils.py
Normal file
81
myproject/inventory/utils.py
Normal file
@@ -0,0 +1,81 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
Утилиты для модуля inventory (склад, приходы, продажи, партии).
|
||||
"""
|
||||
import re
|
||||
import os
|
||||
import logging
|
||||
from datetime import datetime
|
||||
|
||||
|
||||
# Настройка логирования в файл
|
||||
LOG_FILE = os.path.join(os.path.dirname(__file__), 'logs', 'incoming_sequence.log')
|
||||
os.makedirs(os.path.dirname(LOG_FILE), exist_ok=True)
|
||||
|
||||
# Создаём файловый логгер
|
||||
file_logger = logging.getLogger('incoming_sequence_file')
|
||||
if not file_logger.handlers:
|
||||
handler = logging.FileHandler(LOG_FILE, encoding='utf-8')
|
||||
formatter = logging.Formatter(
|
||||
'%(asctime)s | %(levelname)s | %(message)s',
|
||||
datefmt='%Y-%m-%d %H:%M:%S'
|
||||
)
|
||||
handler.setFormatter(formatter)
|
||||
file_logger.addHandler(handler)
|
||||
file_logger.setLevel(logging.DEBUG)
|
||||
|
||||
|
||||
def generate_incoming_document_number():
|
||||
"""
|
||||
Генерирует номер документа поступления вида 'IN-XXXX-XXXX'.
|
||||
|
||||
Алгоритм:
|
||||
1. Ищет максимальный номер в БД с префиксом 'IN-'
|
||||
2. Извлекает числовое значение из последней части (IN-XXXX-XXXX)
|
||||
3. Увеличивает на 1 и форматирует в 'IN-XXXX-XXXX'
|
||||
|
||||
Преимущества:
|
||||
- Работает без SEQUENCE (не требует миграций)
|
||||
- Гарантирует уникальность через unique constraint в модели
|
||||
- Простая логика, легко отладить
|
||||
- Работает с любым тенантом (django-tenants совместимо)
|
||||
|
||||
Возвращает:
|
||||
str: Номер вида 'IN-0000-0001', 'IN-0000-0002', итд
|
||||
"""
|
||||
from inventory.models import IncomingBatch
|
||||
import logging
|
||||
|
||||
logger = logging.getLogger('inventory.incoming')
|
||||
|
||||
try:
|
||||
# Найти все номера с префиксом IN-
|
||||
existing_batches = IncomingBatch.objects.filter(
|
||||
document_number__startswith='IN-'
|
||||
).values_list('document_number', flat=True).order_by('document_number')
|
||||
|
||||
if not existing_batches:
|
||||
# Если нет номеров - начинаем с 1
|
||||
next_num = 1
|
||||
file_logger.info(f"✓ No existing batches found, starting from 1")
|
||||
else:
|
||||
# Берем последний номер, извлекаем цифру и увеличиваем
|
||||
last_number = existing_batches.last() # 'IN-0000-0005'
|
||||
# Извлекаем последние 4 цифры
|
||||
last_digits = int(last_number.split('-')[-1]) # 5
|
||||
next_num = last_digits + 1
|
||||
file_logger.info(f"✓ Last number was {last_number}, next: {next_num}")
|
||||
|
||||
# Форматируем в IN-XXXX-XXXX
|
||||
combined_str = f"{next_num:08d}" # Гарантируем 8 цифр
|
||||
first_part = combined_str[:4] # '0000' или '0001'
|
||||
second_part = combined_str[4:] # '0001' или '0002'
|
||||
|
||||
result = f"IN-{first_part}-{second_part}"
|
||||
file_logger.info(f"✓ Generated: {result}")
|
||||
|
||||
return result
|
||||
|
||||
except Exception as e:
|
||||
file_logger.error(f"✗ Error generating number: {str(e)}")
|
||||
raise
|
||||
Reference in New Issue
Block a user