internal fixes
This commit is contained in:
@@ -271,17 +271,48 @@ class PlanChangeSerializer(serializers.Serializer):
|
|||||||
class LeadSerializer(serializers.ModelSerializer):
|
class LeadSerializer(serializers.ModelSerializer):
|
||||||
route = serializers.PrimaryKeyRelatedField(queryset=Route.objects.all())
|
route = serializers.PrimaryKeyRelatedField(queryset=Route.objects.all())
|
||||||
moving_user = serializers.PrimaryKeyRelatedField(queryset=User.objects.all())
|
moving_user = serializers.PrimaryKeyRelatedField(queryset=User.objects.all())
|
||||||
|
moving_price = serializers.DecimalField(max_digits=10, decimal_places=2)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = Leads
|
model = Leads
|
||||||
fields = ['route', 'moving_user', 'moving_price', 'moving_date', 'comment']
|
fields = ['route', 'moving_user', 'moving_price', 'moving_date', 'comment']
|
||||||
|
|
||||||
def validate_moving_date(self, value):
|
def validate_moving_date(self, value):
|
||||||
if value < timezone.now().date():
|
try:
|
||||||
raise serializers.ValidationError("Дата перевозки не может быть в прошлом")
|
if value < timezone.now().date():
|
||||||
return value
|
raise serializers.ValidationError("Дата перевозки не может быть в прошлом")
|
||||||
|
return value
|
||||||
|
except Exception as e:
|
||||||
|
raise
|
||||||
|
|
||||||
def validate_moving_price(self, value):
|
def validate_moving_price(self, value):
|
||||||
if value <= 0:
|
try:
|
||||||
raise serializers.ValidationError("Цена должна быть больше нуля")
|
if isinstance(value, str):
|
||||||
return value
|
value = value.replace(',', '.')
|
||||||
|
price = float(value)
|
||||||
|
if price <= 0:
|
||||||
|
raise serializers.ValidationError("Цена должна быть больше нуля")
|
||||||
|
return price
|
||||||
|
except (TypeError, ValueError) as e:
|
||||||
|
raise serializers.ValidationError("Некорректный формат цены")
|
||||||
|
|
||||||
|
def validate(self, data):
|
||||||
|
try:
|
||||||
|
# проверяем что все обязательные поля присутствуют
|
||||||
|
required_fields = ['route', 'moving_user', 'moving_price', 'moving_date']
|
||||||
|
missing_fields = [field for field in required_fields if field not in data]
|
||||||
|
if missing_fields:
|
||||||
|
raise serializers.ValidationError(f"Отсутствуют обязательные поля: {', '.join(missing_fields)}")
|
||||||
|
|
||||||
|
# проверяем что пользователь не пытается откликнуться на свою заявку
|
||||||
|
if data['route'].owner.id == data['moving_user'].id:
|
||||||
|
raise serializers.ValidationError("Вы не можете откликнуться на собственную заявку")
|
||||||
|
return data
|
||||||
|
except Exception as e:
|
||||||
|
raise
|
||||||
|
|
||||||
|
def to_internal_value(self, data):
|
||||||
|
try:
|
||||||
|
return super().to_internal_value(data)
|
||||||
|
except Exception as e:
|
||||||
|
raise
|
||||||
@@ -198,10 +198,11 @@ class LeadViewSet(ViewSet):
|
|||||||
|
|
||||||
@action(detail=False, methods=['post'])
|
@action(detail=False, methods=['post'])
|
||||||
@handle_exceptions
|
@handle_exceptions
|
||||||
def send_lead(self, request, id):
|
def send_lead(self, request):
|
||||||
"""
|
"""
|
||||||
Создание новой заявки на перевозку
|
Создание новой заявки на перевозку
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# добавляем текущего пользователя в данные
|
# добавляем текущего пользователя в данные
|
||||||
data = request.data.copy()
|
data = request.data.copy()
|
||||||
data['moving_user'] = request.user.id
|
data['moving_user'] = request.user.id
|
||||||
@@ -209,40 +210,54 @@ class LeadViewSet(ViewSet):
|
|||||||
# проверяем существование и доступность маршрута
|
# проверяем существование и доступность маршрута
|
||||||
try:
|
try:
|
||||||
route = Route.objects.get(id=data.get('route'))
|
route = Route.objects.get(id=data.get('route'))
|
||||||
if route.owner == request.user:
|
if route.owner.id == request.user.id:
|
||||||
|
print("Error: User trying to respond to their own route")
|
||||||
return Response(
|
return Response(
|
||||||
{"error": "Вы не можете откликнуться на собственную заявку"},
|
{"error": "Вы не можете откликнуться на собственную заявку"},
|
||||||
status=status.HTTP_400_BAD_REQUEST
|
status=status.HTTP_400_BAD_REQUEST
|
||||||
)
|
)
|
||||||
except Route.DoesNotExist:
|
except Route.DoesNotExist:
|
||||||
|
print("Route not found for ID:", data.get('route'))
|
||||||
return Response(
|
return Response(
|
||||||
{"error": "Указанный маршрут не найден"},
|
{"error": "Указанный маршрут не найден"},
|
||||||
status=status.HTTP_404_NOT_FOUND
|
status=status.HTTP_404_NOT_FOUND
|
||||||
)
|
)
|
||||||
|
|
||||||
serializer = LeadSerializer(data=data)
|
serializer = LeadSerializer(data=data)
|
||||||
if serializer.is_valid():
|
is_valid = serializer.is_valid()
|
||||||
lead = serializer.save()
|
|
||||||
|
|
||||||
# собираем ответ с данными о заявке для фронта
|
if is_valid:
|
||||||
response_data = {
|
try:
|
||||||
"status": "success",
|
lead = serializer.save()
|
||||||
"message": "Заявка успешно создана",
|
|
||||||
"data": {
|
# собираем ответ с данными о заявке для фронта
|
||||||
"id": lead.id,
|
response_data = {
|
||||||
"route_id": lead.route.id,
|
"status": "success",
|
||||||
"moving_price": lead.moving_price,
|
"message": "Заявка успешно создана",
|
||||||
"moving_date": lead.moving_date,
|
"data": {
|
||||||
|
"id": lead.id,
|
||||||
|
"route_id": lead.route.id,
|
||||||
|
"moving_price": lead.moving_price,
|
||||||
|
"moving_date": lead.moving_date,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
return Response(response_data, status=status.HTTP_201_CREATED)
|
||||||
return Response(response_data, status=status.HTTP_201_CREATED)
|
except Exception as e:
|
||||||
|
return Response(
|
||||||
return Response(
|
{
|
||||||
{
|
"status": "error",
|
||||||
"status": "error",
|
"message": "Ошибка при сохранении заявки",
|
||||||
"message": "Ошибка валидации данных",
|
"error": str(e)
|
||||||
"errors": serializer.errors
|
},
|
||||||
},
|
status=status.HTTP_500_INTERNAL_SERVER_ERROR
|
||||||
status=status.HTTP_400_BAD_REQUEST
|
)
|
||||||
)
|
else:
|
||||||
|
return Response(
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"message": "Ошибка валидации данных",
|
||||||
|
"errors": serializer.errors
|
||||||
|
},
|
||||||
|
status=status.HTTP_400_BAD_REQUEST
|
||||||
|
)
|
||||||
|
|
||||||
@@ -4,3 +4,4 @@ from .models import *
|
|||||||
admin.site.register(Country)
|
admin.site.register(Country)
|
||||||
admin.site.register(City)
|
admin.site.register(City)
|
||||||
admin.site.register(Route)
|
admin.site.register(Route)
|
||||||
|
admin.site.register(Leads)
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
# Generated by Django 5.2.1 on 2025-05-24 11:12
|
# Generated by Django 5.2.1 on 2025-05-24 12:05
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.conf import settings
|
||||||
from django.db import migrations, models
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
@@ -7,6 +9,7 @@ class Migration(migrations.Migration):
|
|||||||
|
|
||||||
dependencies = [
|
dependencies = [
|
||||||
('routes', '0003_alter_route_owner_type'),
|
('routes', '0003_alter_route_owner_type'),
|
||||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||||
]
|
]
|
||||||
|
|
||||||
operations = [
|
operations = [
|
||||||
@@ -14,9 +17,12 @@ class Migration(migrations.Migration):
|
|||||||
name='Leads',
|
name='Leads',
|
||||||
fields=[
|
fields=[
|
||||||
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('moving_price', models.DecimalField(decimal_places=2, default=0, max_digits=10)),
|
||||||
('moving_date', models.DateField()),
|
('moving_date', models.DateField()),
|
||||||
('comment', models.CharField(blank=True, max_length=500, null=True)),
|
('comment', models.CharField(blank=True, max_length=500, null=True)),
|
||||||
('created_at', models.DateTimeField(auto_now_add=True)),
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('moving_user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL, verbose_name='Перевозчик')),
|
||||||
|
('route', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='routes.route', verbose_name='Маршрут')),
|
||||||
],
|
],
|
||||||
options={
|
options={
|
||||||
'verbose_name': 'Заявка',
|
'verbose_name': 'Заявка',
|
||||||
|
|||||||
18
backend/routes/migrations/0005_alter_leads_moving_price.py
Normal file
18
backend/routes/migrations/0005_alter_leads_moving_price.py
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
# Generated by Django 5.2.1 on 2025-05-24 12:09
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('routes', '0004_leads'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='leads',
|
||||||
|
name='moving_price',
|
||||||
|
field=models.DecimalField(decimal_places=2, max_digits=10),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -103,10 +103,9 @@ class Route(models.Model):
|
|||||||
)
|
)
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
if self.owner:
|
from_city_name = self.from_city.name if self.from_city else 'Не указан'
|
||||||
return f'{self.owner}'
|
to_city_name = self.to_city.name if self.to_city else 'Не указан'
|
||||||
else:
|
return f"Маршрут #{self.id}: {from_city_name} → {to_city_name}"
|
||||||
return str(self.owner_type)
|
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
verbose_name = (u'Маршрут')
|
verbose_name = (u'Маршрут')
|
||||||
@@ -114,9 +113,9 @@ class Route(models.Model):
|
|||||||
ordering = ('id',)
|
ordering = ('id',)
|
||||||
|
|
||||||
class Leads(models.Model):
|
class Leads(models.Model):
|
||||||
route = models.ForeignKey(Route, verbose_name="Маршрут", on_delete=models.CASCADE),
|
route = models.ForeignKey(Route, verbose_name="Маршрут", on_delete=models.CASCADE)
|
||||||
moving_user = models.ForeignKey(User, verbose_name="Перевозчик", on_delete=models.CASCADE),
|
moving_user = models.ForeignKey(User, verbose_name="Перевозчик", on_delete=models.CASCADE)
|
||||||
moving_price = models.DecimalField(max_digits=10, decimal_places=2),
|
moving_price = models.DecimalField(max_digits=10, decimal_places=2)
|
||||||
moving_date = models.DateField()
|
moving_date = models.DateField()
|
||||||
comment = models.CharField(max_length=500, null=True, blank=True)
|
comment = models.CharField(max_length=500, null=True, blank=True)
|
||||||
created_at = models.DateTimeField(auto_now_add=True)
|
created_at = models.DateTimeField(auto_now_add=True)
|
||||||
@@ -127,4 +126,4 @@ class Leads(models.Model):
|
|||||||
ordering = ['-created_at']
|
ordering = ['-created_at']
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return f"Заявка по маршруту ${self.route}"
|
return f"Заявка по маршруту {self.route}"
|
||||||
@@ -244,4 +244,5 @@ export interface Lead {
|
|||||||
moving_price: string
|
moving_price: string
|
||||||
moving_date: string
|
moving_date: string
|
||||||
comment: string
|
comment: string
|
||||||
|
route?: number
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,11 +1,18 @@
|
|||||||
import { Lead } from '@/app/types'
|
import { Lead } from '@/app/types'
|
||||||
|
import { getSession } from 'next-auth/react'
|
||||||
|
|
||||||
export const sendLead = async (data: Lead) => {
|
export const sendLead = async (data: Lead & { id?: number }) => {
|
||||||
const API_URL = process.env.NEXT_PUBLIC_API_URL
|
const API_URL = process.env.NEXT_PUBLIC_API_URL
|
||||||
|
const session = await getSession()
|
||||||
|
|
||||||
|
if (!session?.accessToken) {
|
||||||
|
throw new Error('No access token found')
|
||||||
|
}
|
||||||
|
|
||||||
const headers: Record<string, string> = {
|
const headers: Record<string, string> = {
|
||||||
'Content-Type': 'application/json',
|
'Content-Type': 'application/json',
|
||||||
Accept: 'application/json',
|
Accept: 'application/json',
|
||||||
|
Authorization: `Bearer ${session.accessToken}`,
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -16,6 +23,7 @@ export const sendLead = async (data: Lead) => {
|
|||||||
moving_price: data.moving_price,
|
moving_price: data.moving_price,
|
||||||
moving_date: data.moving_date,
|
moving_date: data.moving_date,
|
||||||
comment: data.comment,
|
comment: data.comment,
|
||||||
|
route: data.id,
|
||||||
})
|
})
|
||||||
|
|
||||||
const response = await fetch(`${API_URL}/account/send_lead/`, {
|
const response = await fetch(`${API_URL}/account/send_lead/`, {
|
||||||
|
|||||||
Reference in New Issue
Block a user