Merge pull request 'shaxob' (#132) from shaxob into main
All checks were successful
Deploy to Production / build-and-deploy (push) Successful in 2m23s

Reviewed-on: #132
This commit is contained in:
2026-05-04 12:49:58 +00:00
19 changed files with 479 additions and 28 deletions

View File

@@ -0,0 +1,31 @@
# Generated by Django 5.2.7 on 2026-05-01 06:45
import django.db.models.deletion
from django.conf import settings
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('evaluation', '0038_evaluationrequestmodel_distance_covered_and_more'),
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
]
operations = [
migrations.CreateModel(
name='Bonus',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('updated_at', models.DateTimeField(auto_now=True)),
('bonus_type', models.CharField(choices=[('lightweight_auto', 'Yengil automobil'), ('truck_car', 'Yuk automobil'), ('special_tech', 'Maxsus texnika')], max_length=50)),
('percentage', models.FloatField()),
('created_at', models.DateTimeField(auto_now_add=True)),
('price', models.FloatField()),
('user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='bonuses', to=settings.AUTH_USER_MODEL)),
],
options={
'abstract': False,
},
),
]

View File

@@ -0,0 +1,59 @@
# Generated by Django 5.2.7 on 2026-05-01 11:43
import django.db.models.deletion
from django.conf import settings
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('evaluation', '0039_bonus'),
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
]
operations = [
migrations.CreateModel(
name='BaseValueBonus',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('created_at', models.DateTimeField(auto_now_add=True)),
('updated_at', models.DateTimeField(auto_now=True)),
('base_price', models.DecimalField(decimal_places=2, max_digits=12)),
],
options={
'abstract': False,
},
),
migrations.CreateModel(
name='BonusType',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('created_at', models.DateTimeField(auto_now_add=True)),
('updated_at', models.DateTimeField(auto_now=True)),
('name', models.CharField(max_length=255)),
('category', models.CharField(choices=[('auto_transport', 'Avtotransport'), ('real estate', "ko'chmas mulk"), ('equipment', 'uskuna va jihozlar')], max_length=50)),
('percentage', models.PositiveIntegerField()),
],
options={
'abstract': False,
},
),
migrations.CreateModel(
name='EmployeeBonus',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('created_at', models.DateTimeField(auto_now_add=True)),
('updated_at', models.DateTimeField(auto_now=True)),
('percentage', models.PositiveIntegerField()),
('bonus_type', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='evaluation.bonustype')),
('user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='bonuses', to=settings.AUTH_USER_MODEL)),
],
options={
'unique_together': {('user', 'bonus_type')},
},
),
migrations.DeleteModel(
name='Bonus',
),
]

View File

@@ -0,0 +1,17 @@
# Generated by Django 5.2.7 on 2026-05-01 12:06
from django.db import migrations
class Migration(migrations.Migration):
dependencies = [
('evaluation', '0040_basevaluebonus_bonustype_employeebonus_delete_bonus'),
]
operations = [
migrations.RenameModel(
old_name='BonusType',
new_name='BonusCategory',
),
]

View File

@@ -0,0 +1,18 @@
# Generated by Django 5.2.7 on 2026-05-04 12:40
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('evaluation', '0041_rename_bonustype_bonuscategory'),
]
operations = [
migrations.AlterField(
model_name='bonuscategory',
name='category',
field=models.CharField(choices=[('lightweight_auto', 'Yengil automobil'), ('truck_car', 'Yuk automobil'), ('special_tech', 'Maxsus texnika')], max_length=50),
),
]

View File

@@ -9,14 +9,11 @@ from core.apps.evaluation.choices.auto import (
AutoEvaluationStatus, AutoEvaluationStatus,
AutoObjectType, AutoObjectType,
# FormOwnership, # FormOwnership,
LocationConvenience,
LocationHighways,
ObjectOwnerType, ObjectOwnerType,
# PropertyRights, # PropertyRights,
# RateType, # RateType,
# ValueDetermined, # ValueDetermined,
) )
from .valuation import ValuationModel from .valuation import ValuationModel
from .vehicle import VehicleModel from .vehicle import VehicleModel
@@ -244,8 +241,6 @@ class AutoEvaluationModel(AbstractBaseModel):
default=False, default=False,
) )
def __str__(self): def __str__(self):
return f"Auto Evaluation {self.registration_number or self.pk}" return f"Auto Evaluation {self.registration_number or self.pk}"

View File

@@ -0,0 +1,33 @@
from django.db import models
from django.db.models.fields import PositiveIntegerField
from django_core.models import AbstractBaseModel
from core.apps.evaluation.choices.auto import AutoObjectType
class BaseValueBonus(AbstractBaseModel):
base_price = models.DecimalField(max_digits=12, decimal_places=2)
def __str__(self):
return f"Base: {self.base_price}"
class BonusCategory(AbstractBaseModel):
name = models.CharField(max_length=255)
category = models.CharField(
max_length=50,
choices=AutoObjectType.choices
)
percentage = PositiveIntegerField()
def __str__(self):
return self.name
class EmployeeBonus(AbstractBaseModel):
user = models.ForeignKey("accounts.User", on_delete=models.CASCADE, related_name="bonuses", )
bonus_type = models.ForeignKey(BonusCategory, on_delete=models.CASCADE)
percentage = models.PositiveIntegerField()
class Meta:
unique_together = ("user", "bonus_type")

View File

@@ -3,12 +3,11 @@ from django.utils.translation import gettext_lazy as _
from django_core.models import AbstractBaseModel from django_core.models import AbstractBaseModel
from model_bakery import baker from model_bakery import baker
from .valuation import ValuationModel
from core.apps.evaluation.choices.movable import ( from core.apps.evaluation.choices.movable import (
MovablePropertyCategory, MovablePropertyCategory,
MovablePropertyCondition, MovablePropertyCondition,
) )
from .valuation import ValuationModel
class MovablePropertyEvaluationModel(AbstractBaseModel): class MovablePropertyEvaluationModel(AbstractBaseModel):
@@ -51,4 +50,3 @@ class MovablePropertyEvaluationModel(AbstractBaseModel):
db_table = "MovablePropertyEvaluation" db_table = "MovablePropertyEvaluation"
verbose_name = _("Movable Property Evaluation") verbose_name = _("Movable Property Evaluation")
verbose_name_plural = _("Movable Property Evaluations") verbose_name_plural = _("Movable Property Evaluations")

View File

@@ -0,0 +1,56 @@
from rest_framework import serializers
from core.apps.evaluation.models.bonus import BonusCategory, EmployeeBonus, BaseValueBonus
class BaseBonusSerializer(serializers.ModelSerializer):
class Meta:
model = BaseValueBonus
fields = ['id', 'base_price']
def create(self, validated_data):
if BaseValueBonus.objects.exists():
raise serializers.ValidationError("Base bonus already exists")
return super().create(validated_data)
class BonusCategorySerializer(serializers.ModelSerializer):
class Meta:
model = BonusCategory
fields = ['name', 'category', 'percentage']
class BonusCategoryListSerializer(serializers.ModelSerializer):
price = serializers.DecimalField(max_digits=12, decimal_places=2)
class Meta:
model = BonusCategory
fields = ['id', 'name', 'category', 'percentage' , 'price']
def get_price(self, obj):
base_obj = BaseValueBonus.objects.first()
if not base_obj:
return 0
return (base_obj.base_price * obj.percentage) / 100
class BonusEmployeeBonusSerializer(serializers.ModelSerializer):
class Meta:
model = EmployeeBonus
fields = ['user', 'bonus_type', 'percentage']
class EmployeeBonusListSerializer(serializers.ModelSerializer):
price = serializers.DecimalField(max_digits=12, decimal_places=2)
class Meta:
model = EmployeeBonus
fields = ['id', 'user', 'bonus_type', 'percentage' , 'price']
def get_price(self, obj):
base_obj = BaseValueBonus.objects.first()
if not base_obj:
return 0
return (base_obj.base_price * obj.percentage) / 100

View File

@@ -87,3 +87,29 @@ class CreateVehicleSerializer(serializers.ModelSerializer):
"condition", "condition",
"position", "position",
] ]
class VehicleApplicationSerializer(serializers.Serializer):
person_name = serializers.CharField()
property_owner = serializers.CharField(max_length=100)
address = serializers.CharField(max_length=255)
marka = serializers.CharField(max_length=100)
model = serializers.CharField(max_length=100)
configuration = serializers.CharField(max_length=100)
auto_number = serializers.CharField(max_length=100)
date_created = serializers.DateTimeField()
mileage = serializers.IntegerField()
vehicle_identification = serializers.CharField(max_length=100)
engine_number = serializers.CharField(max_length=100)
colour = serializers.CharField(max_length=100)
registration_certificate_series = serializers.CharField(max_length=100)
tec_passport_number = serializers.CharField(max_length=100)
tec_passport_date = serializers.DateTimeField()
tec_passport_place = serializers.CharField(max_length=255)
body_type = serializers.CharField(max_length=100)
chassis = serializers.CharField(max_length=100)
plate = serializers.CharField(max_length=100)
value_type = serializers.CharField(max_length=100)
evaluation_purpose = serializers.CharField(max_length=100)
personal_id_number = serializers.IntegerField()
id_number = serializers.CharField(max_length=9)

View File

@@ -27,6 +27,9 @@ router.register("valuation", views.ValuationView, basename="valuation")
router.register("property-owner", views.PropertyOwnerView, basename="property-owner") router.register("property-owner", views.PropertyOwnerView, basename="property-owner")
router.register("customer", views.CustomerView, basename="customer") router.register("customer", views.CustomerView, basename="customer")
router.register("certificate", views.CertificateView, basename="certificate") router.register("certificate", views.CertificateView, basename="certificate")
router.register("bonus-type", views.BonusTypeView, basename="bonus-type")
router.register("bonus-employee", views.BonusEmployeeViewSet, basename="bonus-employee")
router.register("bonus-base", views.BaseBonusViewSet, basename="bonus-base")
urlpatterns = [ urlpatterns = [
path("", include(router.urls)), path("", include(router.urls)),
@@ -85,4 +88,5 @@ urlpatterns = [
)), )),
path("calculate_avg_cost/", views.AvgCostAPIView.as_view()), path("calculate_avg_cost/", views.AvgCostAPIView.as_view()),
path("vehicle_document/", views.GeneratePDFView.as_view()),
] ]

View File

@@ -15,3 +15,4 @@ from .didox import * # noqa
from .tech_passport import * # noqa from .tech_passport import * # noqa
from .certificate import * # noqa from .certificate import * # noqa
from .avg_cost import * from .avg_cost import *
from .bonus import *

View File

@@ -0,0 +1,60 @@
from django_core.mixins import BaseViewSetMixin
from drf_spectacular.utils import extend_schema
from rest_framework import viewsets
from rest_framework.permissions import IsAdminUser
from rest_framework.viewsets import ModelViewSet
# core
from core.apps.evaluation.models.bonus import BonusCategory, EmployeeBonus, BaseValueBonus
from core.apps.evaluation.serializers.bonus.Bonus import BonusCategorySerializer, \
BonusCategoryListSerializer, EmployeeBonusListSerializer, BonusEmployeeBonusSerializer, BaseBonusSerializer
@extend_schema(tags=["BaseBonus"])
class BaseBonusViewSet(BaseViewSetMixin, viewsets.ModelViewSet):
queryset = BaseValueBonus.objects.all()
serializer_class = BaseBonusSerializer
@extend_schema(tags=["Bonus-Category"])
class BonusTypeView(BaseViewSetMixin, ModelViewSet):
queryset = BonusCategory.objects.all()
serializer_class = BonusCategorySerializer
action_serializer_class = {
'create': BonusCategorySerializer,
'update': BonusCategorySerializer,
'partial_update': BonusCategorySerializer,
'list': BonusCategoryListSerializer,
'retrieve': BonusCategoryListSerializer,
}
action_permission_classes = {
'create': [IsAdminUser],
'update': [IsAdminUser],
'partial_update': [IsAdminUser],
'destroy': [IsAdminUser],
'list': [IsAdminUser],
}
class BonusEmployeeViewSet(BaseViewSetMixin, ModelViewSet):
queryset = EmployeeBonus.objects.all()
serializer_class = BonusEmployeeBonusSerializer
action_serializer_class = {
'create': BonusEmployeeBonusSerializer,
'update': BonusEmployeeBonusSerializer,
'partial_update': BonusEmployeeBonusSerializer,
'list': EmployeeBonusListSerializer,
'retrieve': EmployeeBonusListSerializer,
}
action_permission_classes = {
'create': [IsAdminUser],
'update': [IsAdminUser],
'partial_update': [IsAdminUser],
'destroy': [IsAdminUser],
'list': [IsAdminUser],
}

View File

@@ -1,16 +1,19 @@
# django core # django core
from django.http import HttpResponse
from django_core.mixins import BaseViewSetMixin from django_core.mixins import BaseViewSetMixin
# swagger # swagger
from drf_spectacular.utils import extend_schema from drf_spectacular.utils import extend_schema
# rest framework # rest framework
from rest_framework.permissions import AllowAny from rest_framework.permissions import AllowAny
from rest_framework.permissions import IsAuthenticated
from rest_framework.response import Response
from rest_framework.views import APIView
from rest_framework.viewsets import ReadOnlyModelViewSet from rest_framework.viewsets import ReadOnlyModelViewSet
# core apps # core apps
from core.apps.evaluation.models import VehicleModel from core.apps.evaluation.models import VehicleModel
from core.apps.evaluation.serializers import vehicle as serialziers from core.apps.evaluation.serializers import vehicle as serialziers, VehicleApplicationSerializer
from core.utils.generation_pdf import PDFService
@extend_schema(tags=["Vehicle"]) @extend_schema(tags=["Vehicle"])
@@ -27,3 +30,19 @@ class VehicleView(BaseViewSetMixin, ReadOnlyModelViewSet):
"retrieve": serialziers.RetrieveVehicleSerializer, "retrieve": serialziers.RetrieveVehicleSerializer,
"create": serialziers.CreateVehicleSerializer, "create": serialziers.CreateVehicleSerializer,
} }
@extend_schema(tags=['GenerationDocument'], request=VehicleApplicationSerializer)
class GeneratePDFView(APIView):
permission_classes = [IsAuthenticated]
def post(self, request):
serializer = VehicleApplicationSerializer(data=request.data)
if not serializer.is_valid():
return Response(serializer.errors, status=400)
pdf_buffer = PDFService.generate_vehicle_pdf(serializer.validated_data)
response = HttpResponse(pdf_buffer, content_type='application/pdf')
response['Content-Disposition'] = 'attachment; filename="ariza.pdf"'
return response

View File

@@ -30,15 +30,10 @@ class CommentCreateSerializer(serializers.ModelSerializer):
'id', 'message', 'file', 'type', 'task' 'id', 'message', 'file', 'type', 'task'
] ]
def validate(self, data):
task = Task.objects.filter(id=data['task']).first()
if not task:
raise serializers.ValidationError("Task not found")
data['task'] = task
return data
def create(self, validated_data): def create(self, validated_data):
with transaction.atomic(): with transaction.atomic():
task = validated_data.pop('task') comment = Comment.objects.create(
comment = Comment.objects.create(task=task, created_by=self.context['request'].user, **validated_data) created_by=self.context['request'].user,
return comment **validated_data
)
return comment

View File

@@ -20,7 +20,7 @@ urlpatterns = [
path('task/', include( path('task/', include(
[ [
path('list/', task.TaskListView.as_view()), path('list/', task.TaskListView.as_view()),
path('<int:id>/', task.TaskDetailView.as_view()), path('<int:pk>/', task.TaskDetailView.as_view()),
path('create/', task.TaskCreateView.as_view()), path('create/', task.TaskCreateView.as_view()),
] ]
)), )),

View File

@@ -1,9 +1,7 @@
from django.db import transaction from django.db import transaction
from rest_framework import permissions, generics
from rest_framework.response import Response
from drf_spectacular.utils import extend_schema from drf_spectacular.utils import extend_schema
from rest_framework import permissions, generics, status
from rest_framework.response import Response
from core.apps.tasks.models.task import Task from core.apps.tasks.models.task import Task
from core.apps.tasks.serializers.task import TaskSerializer, TaskCreateSerializer from core.apps.tasks.serializers.task import TaskSerializer, TaskCreateSerializer
@@ -18,7 +16,8 @@ class TaskCreateView(generics.GenericAPIView):
@transaction.atomic @transaction.atomic
def post(self, request): def post(self, request):
serializer = self.get_serializer(data=request.data) serializer = self.get_serializer(data=request.data)
serializer.is_valid(raise_exception=True) if not serializer.is_valid(raise_exception=True):
return Response(serializer.validated_data, status=status.HTTP_400_BAD_REQUEST)
serializer.save() serializer.save()
return Response(serializer.data) return Response(serializer.data)

View File

@@ -0,0 +1,138 @@
# services.py
from io import BytesIO
from reportlab.lib import colors
from reportlab.lib.enums import TA_RIGHT, TA_CENTER
from reportlab.lib.pagesizes import A4
from reportlab.lib.styles import getSampleStyleSheet, ParagraphStyle
from reportlab.platypus import SimpleDocTemplate, Table, TableStyle, Paragraph, Spacer
class PDFService:
@staticmethod
def generate_vehicle_pdf(data):
buffer = BytesIO()
doc = SimpleDocTemplate(
buffer, pagesize=A4,
rightMargin=40, leftMargin=40,
topMargin=40, bottomMargin=40
)
styles = getSampleStyleSheet()
cell_style = ParagraphStyle(
'CellStyle', parent=styles['Normal'],
fontSize=9, leading=11
)
cell_style_bold = ParagraphStyle(
'CellStyleBold', parent=cell_style,
fontName='Helvetica-Bold',
textColor=colors.red
)
header_style = ParagraphStyle(
'HeaderStyle', parent=styles['Normal'],
alignment=TA_RIGHT, fontSize=10, leading=12
)
title_style = ParagraphStyle(
'TitleStyle', parent=styles['Normal'],
alignment=TA_CENTER, fontSize=14, leading=16
)
elements = []
header_text = (
f'<b>"Sifat baholash" MChJ direktori<br/>'
f"T.R.To'rayevga</b><br/><br/>"
f"{data.get('address', '')}<br/>"
f"ro'yxatda turuvchi fuqaro<br/>"
f"<u>{data.get('person_name', '')}</u> tomonidan<br/>"
f"<u>Avtotransport vositasini baholash uchun</u>"
)
elements.append(Paragraph(header_text, header_style))
elements.append(Spacer(1, 25))
elements.append(Paragraph("<b>A R I Z A</b>", title_style))
elements.append(Spacer(1, 10))
elements.append(Paragraph(
"Ushbu orqali quyidagi avtotransport vositasini baholab berishingizni so'rayman:",
cell_style
))
elements.append(Spacer(1, 10))
date_created = data.get('date_created')
year_str = str(date_created.year) + " yil" if date_created else ""
tec_date = data.get('tec_passport_date')
tec_date_str = tec_date.strftime('%d.%m.%Y yil') if tec_date else ""
raw_data = [
["Mulk egasi", data.get('property_owner', '')],
["Manzil", data.get('address', '')],
["Marka", data.get('marka', '')],
["Model", data.get('model', '')],
["Komplektatsiya", data.get('configuration', '')],
["Davlat raqami", data.get('auto_number', '')],
["Ishlab chiqarilgan yili", year_str],
["Bosib o'tgan masofasi", f"{data.get('mileage', 0):,}".replace(',', ' ')],
["№ kuzov (VIN)", data.get('vehicle_identification', '')],
["№ dvigatel", data.get('engine_number', '')],
["Rang", data.get('colour', '')],
["Texnik passport seriyasi", data.get('registration_certificate_series', '')],
["Texnik passport raqami", data.get('tec_passport_number', '')],
["Texnik passport berilgan sanasi", tec_date_str],
["Texnik passport berilgan joyi", data.get('tec_passport_place', '')],
["Kuzov turi", data.get('body_type', '')],
["Shassi", data.get('chassis', '')],
["Davlat belgisi (plate)", data.get('plate', '')], # ← was missing
]
table_data = [
[Paragraph(row[0], cell_style), Paragraph(str(row[1]), cell_style_bold)]
for row in raw_data
]
table = Table(table_data, colWidths=[170, 345])
table.setStyle(TableStyle([
('GRID', (0, 0), (-1, -1), 0.5, colors.black),
('VALIGN', (0, 0), (-1, -1), 'TOP'),
('LEFTPADDING', (0, 0), (-1, -1), 6),
('TOPPADDING', (0, 0), (-1, -1), 4),
('BOTTOMPADDING', (0, 0), (-1, -1), 4),
]))
elements.append(table)
elements.append(Spacer(1, 15))
elements.append(Paragraph("<b>Baholash maqsadi:</b>", cell_style))
purpose_raw = [
["Aniqlanayotgan qiymat turi", data.get('value_type', '')],
["Baholash maqsadi", data.get('evaluation_purpose', '')],
]
purpose_data = [
[Paragraph(r[0], cell_style), Paragraph(str(r[1]), cell_style_bold)]
for r in purpose_raw
]
pt = Table(purpose_data, colWidths=[170, 345])
pt.setStyle(TableStyle([
('GRID', (0, 0), (-1, -1), 0.5, colors.black),
('VALIGN', (0, 0), (-1, -1), 'TOP'),
]))
elements.append(pt)
elements.append(Spacer(1, 15))
elements.append(Paragraph("<b>Buyurtmachi rekvizitlari:</b>", cell_style))
footer_raw = [
["Shaxsiy raqam (PINFL)", str(data.get('personal_id_number', ''))],
["ID karta raqami", data.get('id_number', '')],
]
footer_data = [
[Paragraph(f[0], cell_style), Paragraph(str(f[1]), cell_style_bold)]
for f in footer_raw
]
ft = Table(footer_data, colWidths=[170, 345])
ft.setStyle(TableStyle([
('GRID', (0, 0), (-1, -1), 0.5, colors.black),
('VALIGN', (0, 0), (-1, -1), 'TOP'),
]))
elements.append(ft)
doc.build(elements)
buffer.seek(0)
return buffer

View File

@@ -49,4 +49,6 @@ boto3
grpcio>=1.62.0 grpcio>=1.62.0
grpcio-tools>=1.62.0 grpcio-tools>=1.62.0
protobuf>=4.25.0 protobuf>=4.25.0
reportlab