Compare commits
40 Commits
3781ce29e5
...
behruz
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
80a1f5ff17 | ||
|
|
25e92623fd | ||
|
|
cb795fe3b8 | ||
|
|
9a730fa0a3 | ||
|
|
e665939b1b | ||
|
|
928561be51 | ||
|
|
9dcce296a6 | ||
|
|
4ac21100a3 | ||
|
|
b0d5a2f334 | ||
| ba3d6c4e47 | |||
| f3834fad70 | |||
|
|
269817f25e | ||
|
|
8158d7146e | ||
|
|
617bed99ae | ||
|
|
9b782fe7bd | ||
| 135f580db2 | |||
|
|
0c622759cc | ||
|
|
1d750b1c1c | ||
| 2b26c52a5c | |||
|
|
51b3535a80 | ||
|
|
9028e2f102 | ||
|
|
0c9c726756 | ||
|
|
c88ea1aa77 | ||
|
|
581021cbb7 | ||
| 62f65385e1 | |||
|
|
76d2fe5090 | ||
| 92d23901a1 | |||
|
|
42987e4154 | ||
| 84b14da3f4 | |||
|
|
1ff23af8bf | ||
|
|
feecb580c1 | ||
|
|
cb53924f9b | ||
|
|
f53125cfdc | ||
| 65ab51e652 | |||
| 2997810fae | |||
|
|
d014f5a2fb | ||
|
|
7d49929772 | ||
|
|
c29546a04b | ||
|
|
b39c080de3 | ||
|
|
7ad385af94 |
@@ -23,7 +23,7 @@ DB_ENGINE=django.db.backends.postgresql_psycopg2
|
|||||||
DB_NAME=django
|
DB_NAME=django
|
||||||
DB_USER=postgres
|
DB_USER=postgres
|
||||||
DB_PASSWORD=2309
|
DB_PASSWORD=2309
|
||||||
DB_HOST=db
|
DB_HOST=postgres
|
||||||
DB_PORT=5432
|
DB_PORT=5432
|
||||||
|
|
||||||
# Cache
|
# Cache
|
||||||
|
|||||||
7
.github/workflows/deploy.yaml
vendored
7
.github/workflows/deploy.yaml
vendored
@@ -151,6 +151,11 @@ jobs:
|
|||||||
git fetch origin main
|
git fetch origin main
|
||||||
git reset --hard origin/main
|
git reset --hard origin/main
|
||||||
|
|
||||||
cp .env.example .env
|
if [ ! -f .env ]; then
|
||||||
|
cp .env.example .env
|
||||||
|
echo ".env yaratildi, production qiymatlarini kiriting!"
|
||||||
|
fi
|
||||||
|
|
||||||
export PORT=8085
|
export PORT=8085
|
||||||
|
docker pull ${{ secrets.DOCKER_USERNAME }}/${{ env.PROJECT_NAME }}:${{ github.run_number }}
|
||||||
docker stack deploy -c stack.yaml ${{ env.PROJECT_NAME }} --with-registry-auth
|
docker stack deploy -c stack.yaml ${{ env.PROJECT_NAME }} --with-registry-auth
|
||||||
|
|||||||
@@ -14,6 +14,8 @@ APPS = [
|
|||||||
"rest_framework_simplejwt",
|
"rest_framework_simplejwt",
|
||||||
"django_core",
|
"django_core",
|
||||||
"core.apps.accounts.apps.AccountsConfig",
|
"core.apps.accounts.apps.AccountsConfig",
|
||||||
|
'core.apps.tasks.apps.TasksConfig',
|
||||||
|
'core.apps.documents.apps.DocumentsConfig',
|
||||||
]
|
]
|
||||||
|
|
||||||
if env.bool("SILK_ENABLED", False):
|
if env.bool("SILK_ENABLED", False):
|
||||||
|
|||||||
@@ -212,5 +212,31 @@ PAGES = [
|
|||||||
"link": reverse_lazy("admin:accounts_role_changelist"),
|
"link": reverse_lazy("admin:accounts_role_changelist"),
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Task Management"),
|
||||||
|
"separator": True,
|
||||||
|
"items": [
|
||||||
|
{
|
||||||
|
"title": _("Task"),
|
||||||
|
"icon": "task",
|
||||||
|
"link": reverse_lazy("admin:tasks_task_changelist"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Column"),
|
||||||
|
"icon": "tag",
|
||||||
|
"link": reverse_lazy("admin:tasks_column_changelist"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Comment"),
|
||||||
|
"icon": "message",
|
||||||
|
"link": reverse_lazy("admin:tasks_comment_changelist"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Label"),
|
||||||
|
"icon": "tag",
|
||||||
|
"link": reverse_lazy("admin:tasks_label_changelist"),
|
||||||
|
},
|
||||||
|
]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ from config.env import env
|
|||||||
|
|
||||||
|
|
||||||
def home(request):
|
def home(request):
|
||||||
return HttpResponse("OK: #f238c92a09f91761ca82674f506515b9108881c6")
|
return HttpResponse("OK: #ba3d6c4e475ba59a6c3dea6f204e435f08e6ab3c")
|
||||||
|
|
||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
@@ -23,6 +23,8 @@ urlpatterns = [
|
|||||||
path("api/v1/", include("core.apps.evaluation.urls")),
|
path("api/v1/", include("core.apps.evaluation.urls")),
|
||||||
path("api/v1/", include("core.apps.payment.urls")),
|
path("api/v1/", include("core.apps.payment.urls")),
|
||||||
path("api/v1/", include("core.apps.chat.urls")),
|
path("api/v1/", include("core.apps.chat.urls")),
|
||||||
|
path("api/v1/tasks/", include("core.apps.tasks.urls")),
|
||||||
|
path("api/v1/documents/", include("core.apps.documents.urls")),
|
||||||
]
|
]
|
||||||
urlpatterns += [
|
urlpatterns += [
|
||||||
path("admin/", admin.site.urls),
|
path("admin/", admin.site.urls),
|
||||||
|
|||||||
15
core/apps/accounts/permissions.py
Normal file
15
core/apps/accounts/permissions.py
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
from rest_framework.exceptions import PermissionDenied
|
||||||
|
from rest_framework.permissions import BasePermission
|
||||||
|
|
||||||
|
from core.apps.accounts.choices import RoleChoice
|
||||||
|
|
||||||
|
|
||||||
|
class IsAdminRole(BasePermission):
|
||||||
|
def has_permission(self, request, view):
|
||||||
|
if not request.user.is_authenticated:
|
||||||
|
return False
|
||||||
|
|
||||||
|
if request.user.role != RoleChoice.ADMIN:
|
||||||
|
raise PermissionDenied("Only admin can access this")
|
||||||
|
|
||||||
|
return True
|
||||||
@@ -54,4 +54,22 @@ class UserCreateSerializer(serializers.ModelSerializer):
|
|||||||
"first_name",
|
"first_name",
|
||||||
"last_name",
|
"last_name",
|
||||||
"password",
|
"password",
|
||||||
"role"]
|
"role"
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class ShortUserSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = get_user_model()
|
||||||
|
fields = [
|
||||||
|
'id',
|
||||||
|
'first_name',
|
||||||
|
'last_name',
|
||||||
|
'avatar',
|
||||||
|
]
|
||||||
|
|
||||||
|
def get_avatar(self, obj):
|
||||||
|
request = self.context.get('request')
|
||||||
|
if obj.avatar:
|
||||||
|
return request.build_absolute_uri(obj.avatar.url)
|
||||||
|
return None
|
||||||
@@ -27,7 +27,7 @@ urlpatterns = [
|
|||||||
path("", include(router.urls)),
|
path("", include(router.urls)),
|
||||||
path("auth/token/", jwt_views.TokenObtainPairView.as_view(), name="token_obtain_pair"),
|
path("auth/token/", jwt_views.TokenObtainPairView.as_view(), name="token_obtain_pair"),
|
||||||
path("auth/token/verify/", jwt_views.TokenVerifyView.as_view(), name="token_verify"),
|
path("auth/token/verify/", jwt_views.TokenVerifyView.as_view(), name="token_verify"),
|
||||||
path("auth/token/refresh/",jwt_views.TokenRefreshView.as_view()),
|
path("auth/token/refresh/", jwt_views.TokenRefreshView.as_view()),
|
||||||
path("user/list/", UserListApiView.as_view(), name="user-list"),
|
path("user/list/", UserListApiView.as_view(), name="user-list"),
|
||||||
path("admin-user/list/", AdminUserListApiView.as_view(), name="admin-user-list"),
|
path("admin-user/list/", AdminUserListApiView.as_view(), name="admin-user-list"),
|
||||||
path("admin/create/", AdminCreateAPIView.as_view(), name="user-create"),
|
path("admin/create/", AdminCreateAPIView.as_view(), name="user-create"),
|
||||||
|
|||||||
@@ -87,14 +87,8 @@ class DeleteAdminUserApiView(APIView):
|
|||||||
permission_classes = [IsAuthenticated]
|
permission_classes = [IsAuthenticated]
|
||||||
|
|
||||||
def delete(self, request, pk):
|
def delete(self, request, pk):
|
||||||
if request.user.role != RoleChoice.SUPERUSER:
|
|
||||||
return Response({'detail': 'Forbidden'}, status=403)
|
|
||||||
|
|
||||||
user = get_object_or_404(User, pk=pk)
|
user = get_object_or_404(User, pk=pk)
|
||||||
if user.role != RoleChoice.ADMIN:
|
|
||||||
return Response({'detail': 'This user is not an admin'}, status=400)
|
|
||||||
user.delete()
|
user.delete()
|
||||||
|
|
||||||
return Response(status=204)
|
return Response(status=204)
|
||||||
|
|
||||||
|
|
||||||
@@ -106,6 +100,7 @@ class UserDetailAPIView(generics.RetrieveAPIView):
|
|||||||
|
|
||||||
class AdminPermissionsAPIView(generics.GenericAPIView):
|
class AdminPermissionsAPIView(generics.GenericAPIView):
|
||||||
permission_classes = [IsAuthenticated]
|
permission_classes = [IsAuthenticated]
|
||||||
|
queryset = User.objects.all()
|
||||||
|
|
||||||
def get(self, request):
|
def get(self, request):
|
||||||
if request.user.role.name != RoleChoice.ADMIN:
|
if request.user.role.name != RoleChoice.ADMIN:
|
||||||
|
|||||||
0
core/apps/documents/__init__.py
Normal file
0
core/apps/documents/__init__.py
Normal file
5
core/apps/documents/apps.py
Normal file
5
core/apps/documents/apps.py
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
from django.apps import AppConfig
|
||||||
|
|
||||||
|
|
||||||
|
class DocumentsConfig(AppConfig):
|
||||||
|
name = "core.apps.documents"
|
||||||
0
core/apps/documents/migrations/__init__.py
Normal file
0
core/apps/documents/migrations/__init__.py
Normal file
0
core/apps/documents/serializers/__init__.py
Normal file
0
core/apps/documents/serializers/__init__.py
Normal file
13
core/apps/documents/serializers/contract.py
Normal file
13
core/apps/documents/serializers/contract.py
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
|
||||||
|
class InspectionSerializer(serializers.Serializer):
|
||||||
|
tires = serializers.CharField(required=False, allow_blank=True, default="Qoniqarli")
|
||||||
|
engine = serializers.CharField(required=False, allow_blank=True, default="Qoniqarli")
|
||||||
|
chassis = serializers.CharField(required=False, allow_blank=True, default="Qoniqarli")
|
||||||
|
transmission = serializers.CharField(required=False, allow_blank=True, default="Qoniqarli")
|
||||||
|
body = serializers.CharField(required=False, allow_blank=True, default="Qoniqarli")
|
||||||
|
|
||||||
|
|
||||||
|
class ContractPDFRequestSerializer(serializers.Serializer):
|
||||||
|
inspection = InspectionSerializer(required=False)
|
||||||
0
core/apps/documents/services/__init__.py
Normal file
0
core/apps/documents/services/__init__.py
Normal file
33
core/apps/documents/services/cbu_rates.py
Normal file
33
core/apps/documents/services/cbu_rates.py
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
from datetime import date
|
||||||
|
|
||||||
|
import requests
|
||||||
|
|
||||||
|
CBU_URL = "https://cbu.uz/oz/arkhiv-kursov-valyut/json/{code}/{date}/"
|
||||||
|
TIMEOUT_SECONDS = 5
|
||||||
|
CURRENCY_CODES = ("USD", "EUR", "RUB")
|
||||||
|
|
||||||
|
|
||||||
|
def fetch_rates(target_date):
|
||||||
|
"""CBU.uz dan berilgan sanaga oid USD, EUR, RUB kurslarini olish.
|
||||||
|
|
||||||
|
Tarmoq xatosi yoki notogri javob bolsa bosh dict qaytadi.
|
||||||
|
"""
|
||||||
|
if target_date is None:
|
||||||
|
target_date = date.today()
|
||||||
|
date_str = target_date.strftime("%Y-%m-%d")
|
||||||
|
rates = {}
|
||||||
|
for code in CURRENCY_CODES:
|
||||||
|
try:
|
||||||
|
resp = requests.get(
|
||||||
|
CBU_URL.format(code=code, date=date_str),
|
||||||
|
timeout=TIMEOUT_SECONDS,
|
||||||
|
)
|
||||||
|
resp.raise_for_status()
|
||||||
|
data = resp.json()
|
||||||
|
if isinstance(data, list) and data:
|
||||||
|
rate_value = data[0].get("Rate")
|
||||||
|
if rate_value:
|
||||||
|
rates[code] = rate_value
|
||||||
|
except (requests.RequestException, ValueError):
|
||||||
|
continue
|
||||||
|
return rates
|
||||||
7
core/apps/documents/urls.py
Normal file
7
core/apps/documents/urls.py
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
from django.urls import path
|
||||||
|
|
||||||
|
from core.apps.documents.views.contract import ValuationReportPDFView
|
||||||
|
|
||||||
|
urlpatterns = [
|
||||||
|
path('generate-contract-pdf/<int:pk>/', ValuationReportPDFView.as_view(), name='generate_contract_pdf'),
|
||||||
|
]
|
||||||
386
core/apps/documents/views/contract.py
Normal file
386
core/apps/documents/views/contract.py
Normal file
@@ -0,0 +1,386 @@
|
|||||||
|
from datetime import date
|
||||||
|
from decimal import Decimal
|
||||||
|
|
||||||
|
from django.shortcuts import get_object_or_404
|
||||||
|
from django.template.loader import render_to_string
|
||||||
|
from django.http import HttpResponse
|
||||||
|
from rest_framework.views import APIView
|
||||||
|
from rest_framework.response import Response
|
||||||
|
from rest_framework import status
|
||||||
|
from weasyprint import HTML
|
||||||
|
|
||||||
|
from core.apps.evaluation.models import AutoEvaluationModel
|
||||||
|
from core.apps.evaluation.choices.auto import ObjectOwnerType
|
||||||
|
from core.apps.documents.serializers.contract import ContractPDFRequestSerializer
|
||||||
|
from core.apps.documents.services.cbu_rates import fetch_rates
|
||||||
|
|
||||||
|
|
||||||
|
UZ_MONTHS = {
|
||||||
|
1: "yanvar", 2: "fevral", 3: "mart", 4: "aprel",
|
||||||
|
5: "may", 6: "iyun", 7: "iyul", 8: "avgust",
|
||||||
|
9: "sentabr", 10: "oktabr", 11: "noyabr", 12: "dekabr",
|
||||||
|
}
|
||||||
|
|
||||||
|
UZ_ONES = [
|
||||||
|
"", "bir", "ikki", "uch", "to'rt", "besh",
|
||||||
|
"olti", "yetti", "sakkiz", "to'qqiz",
|
||||||
|
]
|
||||||
|
UZ_TENS = [
|
||||||
|
"", "o'n", "yigirma", "o'ttiz", "qirq", "ellik",
|
||||||
|
"oltmish", "yetmish", "sakson", "to'qson",
|
||||||
|
]
|
||||||
|
|
||||||
|
DEFAULT_INSPECTION = {
|
||||||
|
"tires": "Qoniqarli",
|
||||||
|
"engine": "Qoniqarli",
|
||||||
|
"chassis": "Qoniqarli",
|
||||||
|
"transmission": "Qoniqarli",
|
||||||
|
"body": "Qoniqarli",
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def _format_currency(value):
|
||||||
|
if value is None:
|
||||||
|
return "0"
|
||||||
|
try:
|
||||||
|
int_val = int(Decimal(value))
|
||||||
|
except (ValueError, TypeError):
|
||||||
|
return "0"
|
||||||
|
return f"{int_val:,}".replace(",", " ")
|
||||||
|
|
||||||
|
|
||||||
|
def _format_date(value):
|
||||||
|
if not value:
|
||||||
|
return ""
|
||||||
|
return value.strftime("%d.%m.%Y")
|
||||||
|
|
||||||
|
|
||||||
|
def _three_digit_words(num):
|
||||||
|
if num == 0:
|
||||||
|
return ""
|
||||||
|
words = []
|
||||||
|
hundreds = num // 100
|
||||||
|
rest = num % 100
|
||||||
|
if hundreds:
|
||||||
|
if hundreds == 1:
|
||||||
|
words.append("bir yuz")
|
||||||
|
else:
|
||||||
|
words.append(f"{UZ_ONES[hundreds]} yuz")
|
||||||
|
tens = rest // 10
|
||||||
|
ones = rest % 10
|
||||||
|
if tens:
|
||||||
|
words.append(UZ_TENS[tens])
|
||||||
|
if ones:
|
||||||
|
words.append(UZ_ONES[ones])
|
||||||
|
return " ".join(words)
|
||||||
|
|
||||||
|
|
||||||
|
def _number_to_uzbek_words(value):
|
||||||
|
if value is None:
|
||||||
|
return ""
|
||||||
|
try:
|
||||||
|
num = int(Decimal(value))
|
||||||
|
except (ValueError, TypeError):
|
||||||
|
return ""
|
||||||
|
if num == 0:
|
||||||
|
return "nol"
|
||||||
|
|
||||||
|
parts = []
|
||||||
|
billions = num // 1_000_000_000
|
||||||
|
millions = (num % 1_000_000_000) // 1_000_000
|
||||||
|
thousands = (num % 1_000_000) // 1_000
|
||||||
|
rest = num % 1_000
|
||||||
|
|
||||||
|
if billions:
|
||||||
|
parts.append(f"{_three_digit_words(billions)} milliard")
|
||||||
|
if millions:
|
||||||
|
parts.append(f"{_three_digit_words(millions)} million")
|
||||||
|
if thousands:
|
||||||
|
parts.append(f"{_three_digit_words(thousands)} ming")
|
||||||
|
if rest:
|
||||||
|
parts.append(_three_digit_words(rest))
|
||||||
|
|
||||||
|
text = " ".join(parts).strip()
|
||||||
|
return text[0].upper() + text[1:] if text else ""
|
||||||
|
|
||||||
|
|
||||||
|
class ValuationReportPDFView(APIView):
|
||||||
|
"""
|
||||||
|
Baholash hisobotini PDF formatida yuklab olish uchun API.
|
||||||
|
|
||||||
|
GET /api/documents/generate-contract-pdf/<pk>/
|
||||||
|
POST /api/documents/generate-contract-pdf/<pk>/
|
||||||
|
|
||||||
|
POST so'rov tanasida inspection malumotlarini yuborish mumkin:
|
||||||
|
{
|
||||||
|
"inspection": {
|
||||||
|
"tires": "...",
|
||||||
|
"engine": "...",
|
||||||
|
"chassis": "...",
|
||||||
|
"transmission": "...",
|
||||||
|
"body": "..."
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
|
||||||
|
def get(self, request, pk, *args, **kwargs):
|
||||||
|
return self._generate_pdf(request, pk, payload={})
|
||||||
|
|
||||||
|
def post(self, request, pk, *args, **kwargs):
|
||||||
|
serializer = ContractPDFRequestSerializer(data=request.data or {})
|
||||||
|
serializer.is_valid(raise_exception=True)
|
||||||
|
return self._generate_pdf(request, pk, payload=serializer.validated_data)
|
||||||
|
|
||||||
|
def _generate_pdf(self, request, pk, payload):
|
||||||
|
auto_evaluation = get_object_or_404(
|
||||||
|
AutoEvaluationModel.objects.select_related(
|
||||||
|
"user",
|
||||||
|
"vehicle",
|
||||||
|
"vehicle__brand",
|
||||||
|
"vehicle__model",
|
||||||
|
"vehicle__color",
|
||||||
|
"vehicle__fuel_type",
|
||||||
|
"vehicle__body_type",
|
||||||
|
),
|
||||||
|
pk=pk,
|
||||||
|
)
|
||||||
|
|
||||||
|
context = self._build_context(auto_evaluation, payload)
|
||||||
|
|
||||||
|
html_string = render_to_string("documents/contract.html", context)
|
||||||
|
base_url = request.build_absolute_uri("/")
|
||||||
|
|
||||||
|
try:
|
||||||
|
pdf_file = HTML(string=html_string, base_url=base_url).write_pdf()
|
||||||
|
except Exception as e:
|
||||||
|
return Response(
|
||||||
|
{"error": f"PDF yaratishda xatolik: {str(e)}"},
|
||||||
|
status=status.HTTP_500_INTERNAL_SERVER_ERROR,
|
||||||
|
)
|
||||||
|
|
||||||
|
report_number = context["report"]["number"]
|
||||||
|
filename = f"baholash_hisoboti_{report_number}.pdf"
|
||||||
|
|
||||||
|
response = HttpResponse(pdf_file, content_type="application/pdf")
|
||||||
|
response["Content-Disposition"] = f'attachment; filename="{filename}"'
|
||||||
|
response["Content-Length"] = len(pdf_file)
|
||||||
|
return response
|
||||||
|
|
||||||
|
def _build_context(self, auto, payload):
|
||||||
|
vehicle = auto.vehicle
|
||||||
|
user = auto.user
|
||||||
|
|
||||||
|
report_date = auto.rate_report_date or auto.contract_date or date.today()
|
||||||
|
valuation_date = auto.rate_date or report_date
|
||||||
|
inspection_date = auto.object_inspection_date or report_date
|
||||||
|
|
||||||
|
report_number = auto.registration_number or f"{auto.pk}/{report_date.year}"
|
||||||
|
|
||||||
|
# Bozor qiymati hozircha hisoblanmagan — default 0.
|
||||||
|
final_value = None
|
||||||
|
market_value_formatted = (
|
||||||
|
f"{_format_currency(final_value)} so'm" if final_value is not None else "0 so'm"
|
||||||
|
)
|
||||||
|
market_value_words = (
|
||||||
|
f"{_number_to_uzbek_words(final_value)} so'm" if final_value is not None else ""
|
||||||
|
)
|
||||||
|
|
||||||
|
cost_final = final_value
|
||||||
|
comparative_final = final_value
|
||||||
|
|
||||||
|
vehicle_ctx = self._vehicle_context(auto, vehicle)
|
||||||
|
customer_ctx = self._customer_context(user)
|
||||||
|
owner_ctx = self._owner_context(auto)
|
||||||
|
contract_ctx = self._contract_context(auto, report_date)
|
||||||
|
inspection_ctx = self._inspection_context(payload)
|
||||||
|
rates_ctx = self._rates_context(valuation_date)
|
||||||
|
|
||||||
|
ctx = {
|
||||||
|
"logo_url": "",
|
||||||
|
"report": {
|
||||||
|
"number": report_number,
|
||||||
|
"date": _format_date(report_date),
|
||||||
|
"valuation_date": _format_date(valuation_date),
|
||||||
|
"inspection_date": _format_date(inspection_date),
|
||||||
|
"year": str(report_date.year or date.today().year),
|
||||||
|
"market_value_formatted": market_value_formatted,
|
||||||
|
"market_value_words": market_value_words,
|
||||||
|
},
|
||||||
|
"vehicle": vehicle_ctx,
|
||||||
|
"customer": customer_ctx,
|
||||||
|
"owner": owner_ctx,
|
||||||
|
"contract": contract_ctx,
|
||||||
|
"company": {
|
||||||
|
"director": "—",
|
||||||
|
},
|
||||||
|
"rates": rates_ctx,
|
||||||
|
"inspection": inspection_ctx,
|
||||||
|
"cost": {
|
||||||
|
"engine_volume": "",
|
||||||
|
"factory_price": _format_currency(cost_final),
|
||||||
|
"replacement_value": _format_currency(cost_final),
|
||||||
|
"wear_percent": "",
|
||||||
|
"final_value": _format_currency(cost_final),
|
||||||
|
"final_value_words": _number_to_uzbek_words(cost_final) + (" so'm" if cost_final else ""),
|
||||||
|
},
|
||||||
|
"comparative": {
|
||||||
|
"final_value": _format_currency(comparative_final),
|
||||||
|
"final_value_usd": "",
|
||||||
|
"final_value_words": _number_to_uzbek_words(comparative_final) + (" so'm" if comparative_final else ""),
|
||||||
|
},
|
||||||
|
"approach": {
|
||||||
|
"cost": {
|
||||||
|
"value": _format_currency(cost_final),
|
||||||
|
"weight": "30%",
|
||||||
|
"weighted": "",
|
||||||
|
},
|
||||||
|
"comparative": {
|
||||||
|
"value": _format_currency(comparative_final),
|
||||||
|
"weight": "70%",
|
||||||
|
"weighted": "",
|
||||||
|
},
|
||||||
|
"weighted_total": _format_currency(final_value),
|
||||||
|
},
|
||||||
|
"analog_1": self._empty_analog(),
|
||||||
|
"analog_2": self._empty_analog(),
|
||||||
|
"analog_3": self._empty_analog(),
|
||||||
|
}
|
||||||
|
return ctx
|
||||||
|
|
||||||
|
def _vehicle_context(self, auto, vehicle):
|
||||||
|
brand_name = ""
|
||||||
|
model_name = ""
|
||||||
|
if vehicle:
|
||||||
|
brand_name = vehicle.brand.name if vehicle.brand else ""
|
||||||
|
model_name = vehicle.model.name if vehicle.model else ""
|
||||||
|
if not brand_name:
|
||||||
|
brand_name = auto.car_brand or ""
|
||||||
|
if not model_name:
|
||||||
|
model_name = auto.car_model or ""
|
||||||
|
full_brand = f"{brand_name} {model_name}".strip()
|
||||||
|
|
||||||
|
plate_number = (vehicle.license_plate if vehicle else None) or auto.car_number or ""
|
||||||
|
manufacture_year = ""
|
||||||
|
if vehicle and vehicle.manufacture_year:
|
||||||
|
manufacture_year = str(vehicle.manufacture_year)
|
||||||
|
elif auto.manufacture_year:
|
||||||
|
manufacture_year = str(auto.manufacture_year)
|
||||||
|
|
||||||
|
production_date = f"{manufacture_year}-yil" if manufacture_year else ""
|
||||||
|
engine_number = (vehicle.engine_number if vehicle else None) or auto.car_dvigatel_number or ""
|
||||||
|
body_number = vehicle.vin_number if vehicle and vehicle.vin_number else ""
|
||||||
|
|
||||||
|
color_value = ""
|
||||||
|
if vehicle and vehicle.color:
|
||||||
|
color_value = vehicle.color.name
|
||||||
|
elif auto.car_color:
|
||||||
|
color_value = auto.car_color
|
||||||
|
|
||||||
|
fuel_type_value = ""
|
||||||
|
if vehicle and vehicle.fuel_type:
|
||||||
|
fuel_type_value = vehicle.fuel_type.name
|
||||||
|
|
||||||
|
tech_passport_value = ""
|
||||||
|
if vehicle and (vehicle.tech_passport_series or vehicle.tech_passport_number):
|
||||||
|
tech_passport_value = (
|
||||||
|
f"{vehicle.tech_passport_series or ''} № {vehicle.tech_passport_number or ''}"
|
||||||
|
).strip()
|
||||||
|
elif auto.tex_passport_serie_num:
|
||||||
|
tech_passport_value = auto.tex_passport_serie_num
|
||||||
|
|
||||||
|
return {
|
||||||
|
"brand": full_brand,
|
||||||
|
"plate_number": plate_number,
|
||||||
|
"production_date": production_date,
|
||||||
|
"production_year": manufacture_year,
|
||||||
|
"type": auto.get_object_type_display() if auto.object_type else "",
|
||||||
|
"engine_number": engine_number,
|
||||||
|
"body_number": body_number,
|
||||||
|
"chassis_number": body_number,
|
||||||
|
"color": color_value,
|
||||||
|
"tech_passport": tech_passport_value,
|
||||||
|
"fuel_type": fuel_type_value,
|
||||||
|
"engine_power": "",
|
||||||
|
"full_weight": "",
|
||||||
|
"empty_weight": "",
|
||||||
|
}
|
||||||
|
|
||||||
|
def _customer_context(self, user):
|
||||||
|
empty = {
|
||||||
|
"name": "",
|
||||||
|
"address": "",
|
||||||
|
"phone": "",
|
||||||
|
"tin": "",
|
||||||
|
"account": "",
|
||||||
|
"bank": "",
|
||||||
|
"mfo": "",
|
||||||
|
}
|
||||||
|
if not user:
|
||||||
|
return empty
|
||||||
|
full_name = " ".join(filter(None, [user.last_name, user.first_name])).strip()
|
||||||
|
if not full_name:
|
||||||
|
full_name = user.username or user.phone or ""
|
||||||
|
return {
|
||||||
|
"name": full_name,
|
||||||
|
"address": "",
|
||||||
|
"phone": user.phone or "",
|
||||||
|
"tin": "",
|
||||||
|
"account": "",
|
||||||
|
"bank": "",
|
||||||
|
"mfo": "",
|
||||||
|
}
|
||||||
|
|
||||||
|
def _owner_context(self, auto):
|
||||||
|
if auto.object_owner_type == ObjectOwnerType.LEGAL:
|
||||||
|
return {
|
||||||
|
"name": auto.object_owner_legal_entity or "",
|
||||||
|
"address": auto.object_owner_residence or "",
|
||||||
|
}
|
||||||
|
full_name = " ".join(
|
||||||
|
filter(None, [
|
||||||
|
auto.object_owner_individual_person_l_name,
|
||||||
|
auto.object_owner_individual_person_f_name,
|
||||||
|
auto.object_owner_individual_person_p_name,
|
||||||
|
])
|
||||||
|
).strip()
|
||||||
|
return {
|
||||||
|
"name": full_name,
|
||||||
|
"address": auto.object_owner_residence or "",
|
||||||
|
}
|
||||||
|
|
||||||
|
def _contract_context(self, auto, fallback_date):
|
||||||
|
contract_date = auto.contract_date or auto.rate_report_date or fallback_date
|
||||||
|
return {
|
||||||
|
"number": auto.registration_number or str(auto.pk),
|
||||||
|
"day": f"{contract_date.day:02d}",
|
||||||
|
"month": UZ_MONTHS.get(contract_date.month, ""),
|
||||||
|
"year": str(contract_date.year),
|
||||||
|
}
|
||||||
|
|
||||||
|
def _inspection_context(self, payload):
|
||||||
|
provided = (payload or {}).get("inspection") or {}
|
||||||
|
return {
|
||||||
|
key: provided.get(key) or default
|
||||||
|
for key, default in DEFAULT_INSPECTION.items()
|
||||||
|
}
|
||||||
|
|
||||||
|
def _rates_context(self, target_date):
|
||||||
|
rates = fetch_rates(target_date)
|
||||||
|
return {
|
||||||
|
"rur": rates.get("RUB", ""),
|
||||||
|
"usd": rates.get("USD", ""),
|
||||||
|
"eur": rates.get("EUR", ""),
|
||||||
|
}
|
||||||
|
|
||||||
|
def _empty_analog(self):
|
||||||
|
return {
|
||||||
|
"source": "",
|
||||||
|
"phone": "",
|
||||||
|
"description": "",
|
||||||
|
"year": "",
|
||||||
|
"mileage": "",
|
||||||
|
"price": "",
|
||||||
|
"adjusted_price_1": "",
|
||||||
|
"final_price": "",
|
||||||
|
"weight": "",
|
||||||
|
}
|
||||||
@@ -55,7 +55,7 @@ class AutoEvaluationAdmin(ModelAdmin):
|
|||||||
"fields": (
|
"fields": (
|
||||||
"tex_passport_serie_num",
|
"tex_passport_serie_num",
|
||||||
("tex_passport_gived_date", "tex_passport_gived_location"),
|
("tex_passport_gived_date", "tex_passport_gived_location"),
|
||||||
("car_type", "car_wheel"),
|
("car_wheel",),
|
||||||
("car_brand", "car_model"),
|
("car_brand", "car_model"),
|
||||||
("car_number", "manufacture_year"),
|
("car_number", "manufacture_year"),
|
||||||
("car_dvigatel_number", "car_color"),
|
("car_dvigatel_number", "car_color"),
|
||||||
|
|||||||
@@ -6,6 +6,8 @@ class AutoObjectType(models.TextChoices):
|
|||||||
LIGHTWEIGHT_AUTO = "lightweight_auto", _("Yengil automobil")
|
LIGHTWEIGHT_AUTO = "lightweight_auto", _("Yengil automobil")
|
||||||
TRUCK_CAR = "truck_car", _("Yuk automobil")
|
TRUCK_CAR = "truck_car", _("Yuk automobil")
|
||||||
SPECIAL_TECH = "special_tech", _("Maxsus texnika")
|
SPECIAL_TECH = "special_tech", _("Maxsus texnika")
|
||||||
|
BUS = "bus", _("Avtobus")
|
||||||
|
MOTO = "moto", _("Mototsikl")
|
||||||
|
|
||||||
|
|
||||||
class AutoEvaluationStatus(models.TextChoices):
|
class AutoEvaluationStatus(models.TextChoices):
|
||||||
|
|||||||
@@ -14,3 +14,4 @@ class ReferenceType(models.TextChoices):
|
|||||||
DETERMINED_VALUE = "determined_value", _("Determined value type")
|
DETERMINED_VALUE = "determined_value", _("Determined value type")
|
||||||
PROPERTY_RIGHTS = "property_rights", _("Property rights")
|
PROPERTY_RIGHTS = "property_rights", _("Property rights")
|
||||||
OWNERSHIP_FORM = "ownership_form", _("Ownership form")
|
OWNERSHIP_FORM = "ownership_form", _("Ownership form")
|
||||||
|
ASSESSMENT_TASK_TYPE = "assessment_task_type", _("Assessment task type")
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
from .auto import * # noqa
|
from .auto import * # noqa
|
||||||
|
from .mechanic_auto import * # noqa
|
||||||
from .customer import * # noqa
|
from .customer import * # noqa
|
||||||
from .document import * # noqa
|
from .document import * # noqa
|
||||||
from .documentcategory import * # noqa
|
from .documentcategory import * # noqa
|
||||||
|
|||||||
@@ -1,7 +1,11 @@
|
|||||||
from django_filters import rest_framework as filters
|
from django_filters import rest_framework as filters
|
||||||
|
|
||||||
from core.apps.evaluation.choices.history import EvaluationEventType
|
from core.apps.evaluation.choices.history import EvaluationEventType
|
||||||
from core.apps.evaluation.models import AutoevaluationhistoryModel, QuickevaluationhistoryModel
|
from core.apps.evaluation.models import (
|
||||||
|
AutoevaluationhistoryModel,
|
||||||
|
MechanicAutoevaluationhistoryModel,
|
||||||
|
QuickevaluationhistoryModel,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class AutoevaluationhistoryFilter(filters.FilterSet):
|
class AutoevaluationhistoryFilter(filters.FilterSet):
|
||||||
@@ -29,6 +33,31 @@ class AutoevaluationhistoryFilter(filters.FilterSet):
|
|||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class MechanicAutoevaluationhistoryFilter(filters.FilterSet):
|
||||||
|
mechanic_auto_evaluation = filters.NumberFilter(
|
||||||
|
field_name="mechanic_auto_evaluation", lookup_expr="exact"
|
||||||
|
)
|
||||||
|
event_type = filters.ChoiceFilter(
|
||||||
|
field_name="event_type",
|
||||||
|
choices=EvaluationEventType.choices,
|
||||||
|
)
|
||||||
|
created_from = filters.DateTimeFilter(
|
||||||
|
field_name="created_at", lookup_expr="gte"
|
||||||
|
)
|
||||||
|
created_to = filters.DateTimeFilter(
|
||||||
|
field_name="created_at", lookup_expr="lte"
|
||||||
|
)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = MechanicAutoevaluationhistoryModel
|
||||||
|
fields = [
|
||||||
|
"mechanic_auto_evaluation",
|
||||||
|
"event_type",
|
||||||
|
"created_from",
|
||||||
|
"created_to",
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
class QuickevaluationhistoryFilter(filters.FilterSet):
|
class QuickevaluationhistoryFilter(filters.FilterSet):
|
||||||
quick_evaluation = filters.NumberFilter(
|
quick_evaluation = filters.NumberFilter(
|
||||||
field_name="quick_evaluation", lookup_expr="exact"
|
field_name="quick_evaluation", lookup_expr="exact"
|
||||||
|
|||||||
37
core/apps/evaluation/filters/mechanic_auto.py
Normal file
37
core/apps/evaluation/filters/mechanic_auto.py
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
from django_filters import rest_framework as filters
|
||||||
|
|
||||||
|
from core.apps.evaluation.models import MechanicAutoEvaluationModel
|
||||||
|
|
||||||
|
|
||||||
|
class MechanicAutoevaluationFilter(filters.FilterSet):
|
||||||
|
status = filters.CharFilter(method="filter_status")
|
||||||
|
object_type = filters.CharFilter(field_name="object_type", lookup_expr="exact")
|
||||||
|
object_owner_type = filters.NumberFilter(field_name="object_owner_type", lookup_expr="exact")
|
||||||
|
rate_type = filters.NumberFilter(field_name="rate_type", lookup_expr="exact")
|
||||||
|
value_determined = filters.NumberFilter(field_name="value_determined", lookup_expr="exact")
|
||||||
|
client = filters.NumberFilter(field_name="valuation__customer", lookup_expr="exact")
|
||||||
|
created_from = filters.DateFilter(field_name="created_at", lookup_expr="gte")
|
||||||
|
created_to = filters.DateFilter(field_name="created_at", lookup_expr="lte")
|
||||||
|
rate_date_from = filters.DateFilter(field_name="rate_date", lookup_expr="gte")
|
||||||
|
rate_date_to = filters.DateFilter(field_name="rate_date", lookup_expr="lte")
|
||||||
|
|
||||||
|
def filter_status(self, queryset, name, value):
|
||||||
|
if value:
|
||||||
|
statuses = [s.strip() for s in value.split(",") if s.strip()]
|
||||||
|
return queryset.filter(status__in=statuses)
|
||||||
|
return queryset
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = MechanicAutoEvaluationModel
|
||||||
|
fields = [
|
||||||
|
"status",
|
||||||
|
"object_type",
|
||||||
|
"object_owner_type",
|
||||||
|
"rate_type",
|
||||||
|
"value_determined",
|
||||||
|
"client",
|
||||||
|
"created_from",
|
||||||
|
"created_to",
|
||||||
|
"rate_date_from",
|
||||||
|
"rate_date_to",
|
||||||
|
]
|
||||||
31
core/apps/evaluation/migrations/0039_bonus.py
Normal file
31
core/apps/evaluation/migrations/0039_bonus.py
Normal 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,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -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',
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -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',
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -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),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,268 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-05-05 11:52
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.conf import settings
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("evaluation", "0042_alter_bonuscategory_category"),
|
||||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name="MechanicAutoEvaluationModel",
|
||||||
|
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)),
|
||||||
|
(
|
||||||
|
"tex_passport_file",
|
||||||
|
models.FileField(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
upload_to="mechanic_evaluation/tech_passports/%Y/%m/",
|
||||||
|
verbose_name="tech passport file",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"registration_number",
|
||||||
|
models.CharField(blank=True, max_length=50, null=True, verbose_name="registration number"),
|
||||||
|
),
|
||||||
|
("contract_date", models.DateField(blank=True, null=True, verbose_name="contract date")),
|
||||||
|
(
|
||||||
|
"object_inspection_date",
|
||||||
|
models.DateField(blank=True, null=True, verbose_name="object inspection date"),
|
||||||
|
),
|
||||||
|
("rate_date", models.DateField(blank=True, null=True, verbose_name="rate date")),
|
||||||
|
("rate_report_date", models.DateField(blank=True, null=True, verbose_name="rate report date")),
|
||||||
|
(
|
||||||
|
"object_type",
|
||||||
|
models.CharField(
|
||||||
|
blank=True,
|
||||||
|
choices=[
|
||||||
|
("lightweight_auto", "Yengil automobil"),
|
||||||
|
("truck_car", "Yuk automobil"),
|
||||||
|
("special_tech", "Maxsus texnika"),
|
||||||
|
],
|
||||||
|
max_length=50,
|
||||||
|
null=True,
|
||||||
|
verbose_name="object type",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"object_owner_type",
|
||||||
|
models.IntegerField(
|
||||||
|
blank=True,
|
||||||
|
choices=[(1, "Jismoniy shaxs"), (2, "Yuridik shaxs")],
|
||||||
|
null=True,
|
||||||
|
verbose_name="object owner type",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"object_owner_individual_person_f_name",
|
||||||
|
models.CharField(blank=True, max_length=100, null=True, verbose_name="owner first name"),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"object_owner_individual_person_l_name",
|
||||||
|
models.CharField(blank=True, max_length=100, null=True, verbose_name="owner last name"),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"object_owner_individual_person_p_name",
|
||||||
|
models.CharField(blank=True, max_length=100, null=True, verbose_name="owner patronymic"),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"object_owner_individual_person_passport_num",
|
||||||
|
models.CharField(blank=True, max_length=20, null=True, verbose_name="owner passport number"),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"object_owner_legal_entity",
|
||||||
|
models.CharField(blank=True, max_length=255, null=True, verbose_name="legal entity name"),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"object_owner_legal_inn",
|
||||||
|
models.CharField(blank=True, max_length=20, null=True, verbose_name="legal entity INN"),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
models.CharField(
|
||||||
|
blank=True, max_length=20, null=True, verbose_name="tech passport series and number"
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"tex_passport_gived_date",
|
||||||
|
models.DateField(blank=True, null=True, verbose_name="tech passport given date"),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"tex_passport_gived_location",
|
||||||
|
models.CharField(
|
||||||
|
blank=True, max_length=255, null=True, verbose_name="tech passport given location"
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"car_type",
|
||||||
|
models.IntegerField(
|
||||||
|
blank=True, choices=[(1, "Xetchbek"), (2, "Universal")], null=True, verbose_name="car type"
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"car_wheel",
|
||||||
|
models.IntegerField(blank=True, choices=[(1, "4x4")], null=True, verbose_name="car wheel"),
|
||||||
|
),
|
||||||
|
("car_brand", models.CharField(blank=True, max_length=100, null=True, verbose_name="car brand")),
|
||||||
|
("car_model", models.CharField(blank=True, max_length=100, null=True, verbose_name="car model")),
|
||||||
|
("car_number", models.CharField(blank=True, max_length=20, null=True, verbose_name="car number")),
|
||||||
|
(
|
||||||
|
"manufacture_year",
|
||||||
|
models.CharField(blank=True, max_length=10, null=True, verbose_name="manufacture year"),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"car_dvigatel_number",
|
||||||
|
models.CharField(blank=True, max_length=50, null=True, verbose_name="engine number"),
|
||||||
|
),
|
||||||
|
("car_color", models.CharField(blank=True, max_length=50, null=True, verbose_name="car color")),
|
||||||
|
("rating_goal", models.CharField(blank=True, max_length=50, null=True, verbose_name="rating goal")),
|
||||||
|
(
|
||||||
|
"status",
|
||||||
|
models.CharField(
|
||||||
|
choices=[
|
||||||
|
("yaratildi", "Yaratildi"),
|
||||||
|
("baxolovchi_biriktirildi", "Baholovchi biriktirildi"),
|
||||||
|
("baxolandi", "Baholandi"),
|
||||||
|
("rad_etildi", "Rad etildi"),
|
||||||
|
("tasdiqlandi", "Tasdiqlandi"),
|
||||||
|
],
|
||||||
|
default="yaratildi",
|
||||||
|
max_length=50,
|
||||||
|
verbose_name="status",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
("is_archived", models.BooleanField(default=False, verbose_name="is archived")),
|
||||||
|
(
|
||||||
|
"appraisers",
|
||||||
|
models.ManyToManyField(
|
||||||
|
blank=True,
|
||||||
|
related_name="mechanic_auto_evaluation_appraisers",
|
||||||
|
to=settings.AUTH_USER_MODEL,
|
||||||
|
verbose_name="appraisers",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"evaluation_request",
|
||||||
|
models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="mechanic_auto_evaluations_request",
|
||||||
|
to="evaluation.evaluationrequestmodel",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"rate_type",
|
||||||
|
models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="evaluation_mechanic_auto_rate_type",
|
||||||
|
to="evaluation.referenceitemmodel",
|
||||||
|
verbose_name="rate type",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"user",
|
||||||
|
models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="mechanic_auto_evaluations_user",
|
||||||
|
to=settings.AUTH_USER_MODEL,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"valuation",
|
||||||
|
models.OneToOneField(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.CASCADE,
|
||||||
|
related_name="mechanic_auto_detail",
|
||||||
|
to="evaluation.valuationmodel",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"value_determined",
|
||||||
|
models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="evaluation_mechanic_auto_value_determined",
|
||||||
|
to="evaluation.referenceitemmodel",
|
||||||
|
verbose_name="value determined",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"vehicle",
|
||||||
|
models.OneToOneField(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.CASCADE,
|
||||||
|
related_name="mechanic_evaluation",
|
||||||
|
to="evaluation.vehiclemodel",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
"verbose_name": "Mechanic Auto Evaluation",
|
||||||
|
"verbose_name_plural": "Mechanic Auto Evaluations",
|
||||||
|
"db_table": "MechanicAutoEvaluation",
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name="MechanicAutoevaluationhistoryModel",
|
||||||
|
fields=[
|
||||||
|
("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")),
|
||||||
|
(
|
||||||
|
"event_type",
|
||||||
|
models.CharField(
|
||||||
|
choices=[
|
||||||
|
("order_created", "Buyurtma yaratildi"),
|
||||||
|
("status_changed", "Status o'zgartirildi"),
|
||||||
|
("evaluator_assigned", "Baholovchi biriktirildi"),
|
||||||
|
("document_uploaded", "Hujjat yuklandi"),
|
||||||
|
("payment_made", "To'lov qilindi"),
|
||||||
|
],
|
||||||
|
max_length=50,
|
||||||
|
verbose_name="event type",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
("actor_id", models.BigIntegerField(blank=True, null=True, verbose_name="actor id")),
|
||||||
|
("actor_full_name", models.CharField(default="Tizim", max_length=255, verbose_name="actor full name")),
|
||||||
|
("actor_role", models.CharField(default="system", max_length=50, verbose_name="actor role")),
|
||||||
|
("meta", models.JSONField(blank=True, default=dict, verbose_name="meta")),
|
||||||
|
("created_at", models.DateTimeField(auto_now_add=True, verbose_name="created at")),
|
||||||
|
(
|
||||||
|
"mechanic_auto_evaluation",
|
||||||
|
models.ForeignKey(
|
||||||
|
on_delete=django.db.models.deletion.CASCADE,
|
||||||
|
related_name="history",
|
||||||
|
to="evaluation.mechanicautoevaluationmodel",
|
||||||
|
verbose_name="mechanic auto evaluation",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
"verbose_name": "Mechanic Auto Evaluation History",
|
||||||
|
"verbose_name_plural": "Mechanic Auto Evaluation Histories",
|
||||||
|
"db_table": "MechanicAutoEvaluationHistory",
|
||||||
|
"ordering": ["created_at"],
|
||||||
|
"indexes": [
|
||||||
|
models.Index(
|
||||||
|
fields=["mechanic_auto_evaluation_id", "created_at"], name="mech_auto_hist_eval_date_idx"
|
||||||
|
),
|
||||||
|
models.Index(fields=["event_type"], name="mech_auto_hist_event_type_idx"),
|
||||||
|
],
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,167 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-05-05 12:06
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("evaluation", "0043_mechanicautoevaluationmodel_and_more"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name="autoevaluationmodel",
|
||||||
|
name="car_type",
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name="autoevaluationmodel",
|
||||||
|
name="tex_passport_file",
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="autoevaluationmodel",
|
||||||
|
name="assessment_task_type",
|
||||||
|
field=models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="evaluation_auto_assessment_task_type",
|
||||||
|
to="evaluation.referenceitemmodel",
|
||||||
|
verbose_name="assessment task type",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="autoevaluationmodel",
|
||||||
|
name="body_type",
|
||||||
|
field=models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="evaluation_auto_body_type",
|
||||||
|
to="evaluation.referenceitemmodel",
|
||||||
|
verbose_name="body type",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="autoevaluationmodel",
|
||||||
|
name="car_position",
|
||||||
|
field=models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="evaluation_auto_car_position",
|
||||||
|
to="evaluation.referenceitemmodel",
|
||||||
|
verbose_name="car position",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="autoevaluationmodel",
|
||||||
|
name="distance_covered",
|
||||||
|
field=models.PositiveIntegerField(blank=True, null=True, verbose_name="distance covered"),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="autoevaluationmodel",
|
||||||
|
name="fuel_type",
|
||||||
|
field=models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="evaluation_auto_fuel_type",
|
||||||
|
to="evaluation.referenceitemmodel",
|
||||||
|
verbose_name="fuel type",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="autoevaluationmodel",
|
||||||
|
name="object_owner_residence",
|
||||||
|
field=models.CharField(blank=True, max_length=255, null=True, verbose_name="object owner residence"),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="autoevaluationmodel",
|
||||||
|
name="state_car",
|
||||||
|
field=models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="evaluation_auto_state_car",
|
||||||
|
to="evaluation.referenceitemmodel",
|
||||||
|
verbose_name="state car",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="autoevaluationmodel",
|
||||||
|
name="object_type",
|
||||||
|
field=models.CharField(
|
||||||
|
blank=True,
|
||||||
|
choices=[
|
||||||
|
("lightweight_auto", "Yengil automobil"),
|
||||||
|
("truck_car", "Yuk automobil"),
|
||||||
|
("special_tech", "Maxsus texnika"),
|
||||||
|
("bus", "Avtobus"),
|
||||||
|
("moto", "Mototsikl"),
|
||||||
|
],
|
||||||
|
max_length=50,
|
||||||
|
null=True,
|
||||||
|
verbose_name="object type",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="bonuscategory",
|
||||||
|
name="category",
|
||||||
|
field=models.CharField(
|
||||||
|
choices=[
|
||||||
|
("lightweight_auto", "Yengil automobil"),
|
||||||
|
("truck_car", "Yuk automobil"),
|
||||||
|
("special_tech", "Maxsus texnika"),
|
||||||
|
("bus", "Avtobus"),
|
||||||
|
("moto", "Mototsikl"),
|
||||||
|
],
|
||||||
|
max_length=50,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="mechanicautoevaluationmodel",
|
||||||
|
name="object_type",
|
||||||
|
field=models.CharField(
|
||||||
|
blank=True,
|
||||||
|
choices=[
|
||||||
|
("lightweight_auto", "Yengil automobil"),
|
||||||
|
("truck_car", "Yuk automobil"),
|
||||||
|
("special_tech", "Maxsus texnika"),
|
||||||
|
("bus", "Avtobus"),
|
||||||
|
("moto", "Mototsikl"),
|
||||||
|
],
|
||||||
|
max_length=50,
|
||||||
|
null=True,
|
||||||
|
verbose_name="object type",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name="AutoEvaluationTexPassportFile",
|
||||||
|
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)),
|
||||||
|
(
|
||||||
|
"file",
|
||||||
|
models.FileField(
|
||||||
|
upload_to="auto_evaluation/tech_passports/%Y/%m/", verbose_name="tech passport file"
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"auto_evaluation",
|
||||||
|
models.ForeignKey(
|
||||||
|
on_delete=django.db.models.deletion.CASCADE,
|
||||||
|
related_name="tex_passport_files",
|
||||||
|
to="evaluation.autoevaluationmodel",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
"verbose_name": "Auto Evaluation Tex Passport File",
|
||||||
|
"verbose_name_plural": "Auto Evaluation Tex Passport Files",
|
||||||
|
"db_table": "AutoEvaluationTexPassportFile",
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0044_remove_autoevaluationmodel_car_type_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='referenceitemmodel',
|
||||||
|
name='type',
|
||||||
|
field=models.CharField(choices=[('brand', 'Brand'), ('marka', 'Marka'), ('color', 'Color'), ('fuel_type', 'Fuel type'), ('body_type', 'Body type'), ('car_position', 'Car position'), ('state_car', 'Car state'), ('evaluation_purpose', 'Evaluation purpose'), ('determined_value', 'Determined value type'), ('property_rights', 'Property rights'), ('ownership_form', 'Ownership form'), ('assessment_task_type', 'Assessment task type')], max_length=50, verbose_name='type'),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,118 @@
|
|||||||
|
import django.db.models.deletion
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("evaluation", "0045_alter_referenceitemmodel_type"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name="mechanicautoevaluationmodel",
|
||||||
|
name="car_type",
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name="mechanicautoevaluationmodel",
|
||||||
|
name="tex_passport_file",
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="mechanicautoevaluationmodel",
|
||||||
|
name="distance_covered",
|
||||||
|
field=models.PositiveIntegerField(blank=True, null=True, verbose_name="distance covered"),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="mechanicautoevaluationmodel",
|
||||||
|
name="object_owner_residence",
|
||||||
|
field=models.CharField(blank=True, max_length=255, null=True, verbose_name="object owner residence"),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="mechanicautoevaluationmodel",
|
||||||
|
name="car_position",
|
||||||
|
field=models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="evaluation_mechanic_auto_car_position",
|
||||||
|
to="evaluation.referenceitemmodel",
|
||||||
|
verbose_name="car position",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="mechanicautoevaluationmodel",
|
||||||
|
name="body_type",
|
||||||
|
field=models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="evaluation_mechanic_auto_body_type",
|
||||||
|
to="evaluation.referenceitemmodel",
|
||||||
|
verbose_name="body type",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="mechanicautoevaluationmodel",
|
||||||
|
name="fuel_type",
|
||||||
|
field=models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="evaluation_mechanic_auto_fuel_type",
|
||||||
|
to="evaluation.referenceitemmodel",
|
||||||
|
verbose_name="fuel type",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="mechanicautoevaluationmodel",
|
||||||
|
name="state_car",
|
||||||
|
field=models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="evaluation_mechanic_auto_state_car",
|
||||||
|
to="evaluation.referenceitemmodel",
|
||||||
|
verbose_name="state car",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="mechanicautoevaluationmodel",
|
||||||
|
name="assessment_task_type",
|
||||||
|
field=models.ForeignKey(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.SET_NULL,
|
||||||
|
related_name="evaluation_mechanic_auto_assessment_task_type",
|
||||||
|
to="evaluation.referenceitemmodel",
|
||||||
|
verbose_name="assessment task type",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name="MechanicAutoEvaluationTexPassportFile",
|
||||||
|
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)),
|
||||||
|
(
|
||||||
|
"file",
|
||||||
|
models.FileField(
|
||||||
|
upload_to="mechanic_evaluation/tech_passports/%Y/%m/",
|
||||||
|
verbose_name="tech passport file",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"mechanic_auto_evaluation",
|
||||||
|
models.ForeignKey(
|
||||||
|
on_delete=django.db.models.deletion.CASCADE,
|
||||||
|
related_name="tex_passport_files",
|
||||||
|
to="evaluation.mechanicautoevaluationmodel",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
"verbose_name": "Mechanic Auto Evaluation Tex Passport File",
|
||||||
|
"verbose_name_plural": "Mechanic Auto Evaluation Tex Passport Files",
|
||||||
|
"db_table": "MechanicAutoEvaluationTexPassportFile",
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
from .auto import * # noqa
|
from .auto import * # noqa
|
||||||
|
from .mechanic_auto import * # noqa
|
||||||
from .customer import * # noqa
|
from .customer import * # noqa
|
||||||
from .document import * # noqa
|
from .document import * # noqa
|
||||||
from .documentcategory import * # noqa
|
from .documentcategory import * # noqa
|
||||||
|
|||||||
@@ -4,19 +4,15 @@ from django_core.models import AbstractBaseModel
|
|||||||
from model_bakery import baker
|
from model_bakery import baker
|
||||||
|
|
||||||
from core.apps.evaluation.choices.auto import (
|
from core.apps.evaluation.choices.auto import (
|
||||||
AutoCarType,
|
|
||||||
AutoCarWheel,
|
AutoCarWheel,
|
||||||
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
|
||||||
|
|
||||||
@@ -57,12 +53,57 @@ class AutoEvaluationModel(AbstractBaseModel):
|
|||||||
null=True,
|
null=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
tex_passport_file = models.FileField(
|
distance_covered = models.PositiveIntegerField(
|
||||||
verbose_name=_("tech passport file"),
|
verbose_name=_("distance covered"),
|
||||||
upload_to="quick_evaluation/tech_passports/%Y/%m/",
|
|
||||||
blank=True,
|
blank=True,
|
||||||
null=True,
|
null=True,
|
||||||
)
|
)
|
||||||
|
object_owner_residence = models.CharField(
|
||||||
|
verbose_name=_("object owner residence"),
|
||||||
|
max_length=255,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
car_position = models.ForeignKey(
|
||||||
|
'evaluation.ReferenceitemModel',
|
||||||
|
verbose_name=_("car position"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name='evaluation_auto_car_position',
|
||||||
|
)
|
||||||
|
body_type = models.ForeignKey(
|
||||||
|
'evaluation.ReferenceitemModel',
|
||||||
|
verbose_name=_("body type"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name='evaluation_auto_body_type',
|
||||||
|
)
|
||||||
|
fuel_type = models.ForeignKey(
|
||||||
|
'evaluation.ReferenceitemModel',
|
||||||
|
verbose_name=_("fuel type"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name='evaluation_auto_fuel_type',
|
||||||
|
)
|
||||||
|
state_car = models.ForeignKey(
|
||||||
|
'evaluation.ReferenceitemModel',
|
||||||
|
verbose_name=_("state car"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name='evaluation_auto_state_car',
|
||||||
|
)
|
||||||
|
assessment_task_type = models.ForeignKey(
|
||||||
|
'evaluation.ReferenceitemModel',
|
||||||
|
verbose_name=_("assessment task type"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name='evaluation_auto_assessment_task_type',
|
||||||
|
)
|
||||||
|
|
||||||
# ── Step 1 — Umumiy ma'lumotlar ──────────────────────────────────
|
# ── Step 1 — Umumiy ma'lumotlar ──────────────────────────────────
|
||||||
registration_number = models.CharField(
|
registration_number = models.CharField(
|
||||||
@@ -177,12 +218,6 @@ class AutoEvaluationModel(AbstractBaseModel):
|
|||||||
blank=True,
|
blank=True,
|
||||||
null=True,
|
null=True,
|
||||||
)
|
)
|
||||||
car_type = models.IntegerField(
|
|
||||||
verbose_name=_("car type"),
|
|
||||||
choices=AutoCarType.choices,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
car_wheel = models.IntegerField(
|
car_wheel = models.IntegerField(
|
||||||
verbose_name=_("car wheel"),
|
verbose_name=_("car wheel"),
|
||||||
choices=AutoCarWheel.choices,
|
choices=AutoCarWheel.choices,
|
||||||
@@ -244,8 +279,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}"
|
||||||
|
|
||||||
@@ -257,3 +290,27 @@ class AutoEvaluationModel(AbstractBaseModel):
|
|||||||
db_table = "AutoEvaluation"
|
db_table = "AutoEvaluation"
|
||||||
verbose_name = _("Auto Evaluation")
|
verbose_name = _("Auto Evaluation")
|
||||||
verbose_name_plural = _("Auto Evaluations")
|
verbose_name_plural = _("Auto Evaluations")
|
||||||
|
|
||||||
|
|
||||||
|
class AutoEvaluationTexPassportFile(AbstractBaseModel):
|
||||||
|
auto_evaluation = models.ForeignKey(
|
||||||
|
AutoEvaluationModel,
|
||||||
|
on_delete=models.CASCADE,
|
||||||
|
related_name="tex_passport_files",
|
||||||
|
)
|
||||||
|
file = models.FileField(
|
||||||
|
verbose_name=_("tech passport file"),
|
||||||
|
upload_to="auto_evaluation/tech_passports/%Y/%m/",
|
||||||
|
)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"Tex passport file for AutoEvaluation #{self.auto_evaluation_id}"
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def _baker(cls):
|
||||||
|
return baker.make(cls)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
db_table = "AutoEvaluationTexPassportFile"
|
||||||
|
verbose_name = _("Auto Evaluation Tex Passport File")
|
||||||
|
verbose_name_plural = _("Auto Evaluation Tex Passport Files")
|
||||||
|
|||||||
33
core/apps/evaluation/models/bonus.py
Normal file
33
core/apps/evaluation/models/bonus.py
Normal 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")
|
||||||
@@ -69,6 +69,66 @@ class AutoevaluationhistoryModel(models.Model):
|
|||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class MechanicAutoevaluationhistoryModel(models.Model):
|
||||||
|
"""MechanicAutoEvaluation bo'yicha barcha harakatlar logi — faqat o'qiladi, signallar tomonidan yoziladi."""
|
||||||
|
|
||||||
|
mechanic_auto_evaluation = models.ForeignKey(
|
||||||
|
"evaluation.MechanicAutoEvaluationModel",
|
||||||
|
on_delete=models.CASCADE,
|
||||||
|
related_name="history",
|
||||||
|
verbose_name=_("mechanic auto evaluation"),
|
||||||
|
)
|
||||||
|
event_type = models.CharField(
|
||||||
|
verbose_name=_("event type"),
|
||||||
|
max_length=50,
|
||||||
|
choices=EvaluationEventType.choices,
|
||||||
|
)
|
||||||
|
|
||||||
|
actor_id = models.BigIntegerField(
|
||||||
|
verbose_name=_("actor id"),
|
||||||
|
null=True,
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
|
actor_full_name = models.CharField(
|
||||||
|
verbose_name=_("actor full name"),
|
||||||
|
max_length=255,
|
||||||
|
default="Tizim",
|
||||||
|
)
|
||||||
|
actor_role = models.CharField(
|
||||||
|
verbose_name=_("actor role"),
|
||||||
|
max_length=50,
|
||||||
|
default="system",
|
||||||
|
)
|
||||||
|
|
||||||
|
meta = models.JSONField(
|
||||||
|
verbose_name=_("meta"),
|
||||||
|
default=dict,
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
created_at = models.DateTimeField(
|
||||||
|
verbose_name=_("created at"),
|
||||||
|
auto_now_add=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"{self.get_event_type_display()} — MechanicAutoEval #{self.mechanic_auto_evaluation_id}"
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def _baker(cls):
|
||||||
|
return baker.make(cls)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
db_table = "MechanicAutoEvaluationHistory"
|
||||||
|
verbose_name = _("Mechanic Auto Evaluation History")
|
||||||
|
verbose_name_plural = _("Mechanic Auto Evaluation Histories")
|
||||||
|
ordering = ["created_at"]
|
||||||
|
indexes = [
|
||||||
|
models.Index(fields=["mechanic_auto_evaluation_id", "created_at"], name="mech_auto_hist_eval_date_idx"),
|
||||||
|
models.Index(fields=["event_type"], name="mech_auto_hist_event_type_idx"),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
class QuickevaluationhistoryModel(models.Model):
|
class QuickevaluationhistoryModel(models.Model):
|
||||||
"""QuickEvaluation bo'yicha barcha harakatlar logi — faqat o'qiladi, signallar tomonidan yoziladi."""
|
"""QuickEvaluation bo'yicha barcha harakatlar logi — faqat o'qiladi, signallar tomonidan yoziladi."""
|
||||||
|
|
||||||
|
|||||||
312
core/apps/evaluation/models/mechanic_auto.py
Normal file
312
core/apps/evaluation/models/mechanic_auto.py
Normal file
@@ -0,0 +1,312 @@
|
|||||||
|
from django.db import models
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
from django_core.models import AbstractBaseModel
|
||||||
|
from model_bakery import baker
|
||||||
|
|
||||||
|
from core.apps.evaluation.choices.auto import (
|
||||||
|
AutoCarWheel,
|
||||||
|
AutoEvaluationStatus,
|
||||||
|
AutoObjectType,
|
||||||
|
ObjectOwnerType,
|
||||||
|
)
|
||||||
|
from .valuation import ValuationModel
|
||||||
|
from .vehicle import VehicleModel
|
||||||
|
|
||||||
|
|
||||||
|
class MechanicAutoEvaluationModel(AbstractBaseModel):
|
||||||
|
user = models.ForeignKey(
|
||||||
|
"accounts.User",
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name="mechanic_auto_evaluations_user",
|
||||||
|
null=True,
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
|
evaluation_request = models.ForeignKey(
|
||||||
|
"evaluation.EvaluationRequestModel",
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name="mechanic_auto_evaluations_request",
|
||||||
|
null=True,
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
|
valuation = models.OneToOneField(
|
||||||
|
ValuationModel,
|
||||||
|
on_delete=models.CASCADE,
|
||||||
|
related_name="mechanic_auto_detail",
|
||||||
|
null=True,
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
|
vehicle = models.OneToOneField(
|
||||||
|
VehicleModel,
|
||||||
|
on_delete=models.CASCADE,
|
||||||
|
related_name="mechanic_evaluation",
|
||||||
|
null=True,
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
|
appraisers = models.ManyToManyField(
|
||||||
|
"accounts.User",
|
||||||
|
verbose_name=_("appraisers"),
|
||||||
|
related_name="mechanic_auto_evaluation_appraisers",
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
distance_covered = models.PositiveIntegerField(
|
||||||
|
verbose_name=_("distance covered"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
object_owner_residence = models.CharField(
|
||||||
|
verbose_name=_("object owner residence"),
|
||||||
|
max_length=255,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
car_position = models.ForeignKey(
|
||||||
|
'evaluation.ReferenceitemModel',
|
||||||
|
verbose_name=_("car position"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name='evaluation_mechanic_auto_car_position',
|
||||||
|
)
|
||||||
|
body_type = models.ForeignKey(
|
||||||
|
'evaluation.ReferenceitemModel',
|
||||||
|
verbose_name=_("body type"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name='evaluation_mechanic_auto_body_type',
|
||||||
|
)
|
||||||
|
fuel_type = models.ForeignKey(
|
||||||
|
'evaluation.ReferenceitemModel',
|
||||||
|
verbose_name=_("fuel type"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name='evaluation_mechanic_auto_fuel_type',
|
||||||
|
)
|
||||||
|
state_car = models.ForeignKey(
|
||||||
|
'evaluation.ReferenceitemModel',
|
||||||
|
verbose_name=_("state car"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name='evaluation_mechanic_auto_state_car',
|
||||||
|
)
|
||||||
|
assessment_task_type = models.ForeignKey(
|
||||||
|
'evaluation.ReferenceitemModel',
|
||||||
|
verbose_name=_("assessment task type"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name='evaluation_mechanic_auto_assessment_task_type',
|
||||||
|
)
|
||||||
|
|
||||||
|
# ── Step 1 — Umumiy ma'lumotlar ──────────────────────────────────
|
||||||
|
registration_number = models.CharField(
|
||||||
|
verbose_name=_("registration number"),
|
||||||
|
max_length=50,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
contract_date = models.DateField(
|
||||||
|
verbose_name=_("contract date"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
object_inspection_date = models.DateField(
|
||||||
|
verbose_name=_("object inspection date"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
rate_date = models.DateField(
|
||||||
|
verbose_name=_("rate date"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
rate_report_date = models.DateField(
|
||||||
|
verbose_name=_("rate report date"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
object_type = models.CharField(
|
||||||
|
verbose_name=_("object type"),
|
||||||
|
max_length=50,
|
||||||
|
choices=AutoObjectType.choices,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
# ── Step 2 — Shaxs ma'lumotlari ─────────────────────────────────
|
||||||
|
object_owner_type = models.IntegerField(
|
||||||
|
verbose_name=_("object owner type"),
|
||||||
|
choices=ObjectOwnerType.choices,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
object_owner_individual_person_f_name = models.CharField(
|
||||||
|
verbose_name=_("owner first name"),
|
||||||
|
max_length=100,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
object_owner_individual_person_l_name = models.CharField(
|
||||||
|
verbose_name=_("owner last name"),
|
||||||
|
max_length=100,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
object_owner_individual_person_p_name = models.CharField(
|
||||||
|
verbose_name=_("owner patronymic"),
|
||||||
|
max_length=100,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
object_owner_individual_person_passport_num = models.CharField(
|
||||||
|
verbose_name=_("owner passport number"),
|
||||||
|
max_length=20,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
object_owner_legal_entity = models.CharField(
|
||||||
|
verbose_name=_("legal entity name"),
|
||||||
|
max_length=255,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
object_owner_legal_inn = models.CharField(
|
||||||
|
verbose_name=_("legal entity INN"),
|
||||||
|
max_length=20,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
value_determined = models.ForeignKey(
|
||||||
|
'evaluation.ReferenceitemModel',
|
||||||
|
verbose_name=_("value determined"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name='evaluation_mechanic_auto_value_determined'
|
||||||
|
)
|
||||||
|
rate_type = models.ForeignKey(
|
||||||
|
'evaluation.ReferenceitemModel',
|
||||||
|
verbose_name=_("rate type"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name='evaluation_mechanic_auto_rate_type'
|
||||||
|
)
|
||||||
|
|
||||||
|
# ── Step 4 — Avtomobil ma'lumotlari ─────────────────────────────
|
||||||
|
tex_passport_serie_num = models.CharField(
|
||||||
|
verbose_name=_("tech passport series and number"),
|
||||||
|
max_length=20,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
tex_passport_gived_date = models.DateField(
|
||||||
|
verbose_name=_("tech passport given date"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
tex_passport_gived_location = models.CharField(
|
||||||
|
verbose_name=_("tech passport given location"),
|
||||||
|
max_length=255,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
car_wheel = models.IntegerField(
|
||||||
|
verbose_name=_("car wheel"),
|
||||||
|
choices=AutoCarWheel.choices,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
car_brand = models.CharField(
|
||||||
|
verbose_name=_("car brand"),
|
||||||
|
max_length=100,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
car_model = models.CharField(
|
||||||
|
verbose_name=_("car model"),
|
||||||
|
max_length=100,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
car_number = models.CharField(
|
||||||
|
verbose_name=_("car number"),
|
||||||
|
max_length=20,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
manufacture_year = models.CharField(
|
||||||
|
verbose_name=_("manufacture year"),
|
||||||
|
max_length=10,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
car_dvigatel_number = models.CharField(
|
||||||
|
verbose_name=_("engine number"),
|
||||||
|
max_length=50,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
car_color = models.CharField(
|
||||||
|
verbose_name=_("car color"),
|
||||||
|
max_length=50,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
# ── Natija ───────────────────────────────────────────────────────
|
||||||
|
rating_goal = models.CharField(
|
||||||
|
verbose_name=_("rating goal"),
|
||||||
|
max_length=50,
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
status = models.CharField(
|
||||||
|
verbose_name=_("status"),
|
||||||
|
max_length=50,
|
||||||
|
choices=AutoEvaluationStatus.choices,
|
||||||
|
default=AutoEvaluationStatus.CREATED,
|
||||||
|
)
|
||||||
|
is_archived = models.BooleanField(
|
||||||
|
verbose_name=_("is archived"),
|
||||||
|
default=False,
|
||||||
|
)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"Mechanic Auto Evaluation {self.registration_number or self.pk}"
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def _baker(cls):
|
||||||
|
return baker.make(cls)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
db_table = "MechanicAutoEvaluation"
|
||||||
|
verbose_name = _("Mechanic Auto Evaluation")
|
||||||
|
verbose_name_plural = _("Mechanic Auto Evaluations")
|
||||||
|
|
||||||
|
|
||||||
|
class MechanicAutoEvaluationTexPassportFile(AbstractBaseModel):
|
||||||
|
mechanic_auto_evaluation = models.ForeignKey(
|
||||||
|
MechanicAutoEvaluationModel,
|
||||||
|
on_delete=models.CASCADE,
|
||||||
|
related_name="tex_passport_files",
|
||||||
|
)
|
||||||
|
file = models.FileField(
|
||||||
|
verbose_name=_("tech passport file"),
|
||||||
|
upload_to="mechanic_evaluation/tech_passports/%Y/%m/",
|
||||||
|
)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"Tex passport file for MechanicAutoEvaluation #{self.mechanic_auto_evaluation_id}"
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def _baker(cls):
|
||||||
|
return baker.make(cls)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
db_table = "MechanicAutoEvaluationTexPassportFile"
|
||||||
|
verbose_name = _("Mechanic Auto Evaluation Tex Passport File")
|
||||||
|
verbose_name_plural = _("Mechanic Auto Evaluation Tex Passport Files")
|
||||||
@@ -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")
|
||||||
|
|
||||||
|
|||||||
@@ -3,13 +3,36 @@ import re
|
|||||||
from django.contrib.auth import get_user_model
|
from django.contrib.auth import get_user_model
|
||||||
from rest_framework import serializers
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from django.db import transaction
|
||||||
|
|
||||||
from core.apps.evaluation.choices.request import RequestStatus
|
from core.apps.evaluation.choices.request import RequestStatus
|
||||||
from core.apps.evaluation.models import AutoEvaluationModel, ReferenceitemModel, EvaluationrequestModel
|
from core.apps.evaluation.models import (
|
||||||
|
AutoEvaluationModel,
|
||||||
|
AutoEvaluationTexPassportFile,
|
||||||
|
ReferenceitemModel,
|
||||||
|
EvaluationrequestModel,
|
||||||
|
)
|
||||||
from core.apps.evaluation.serializers.reference import ListReferenceitemSerializer
|
from core.apps.evaluation.serializers.reference import ListReferenceitemSerializer
|
||||||
|
|
||||||
User = get_user_model()
|
User = get_user_model()
|
||||||
|
|
||||||
|
|
||||||
|
class AutoEvaluationTexPassportFileSerializer(serializers.ModelSerializer):
|
||||||
|
file = serializers.SerializerMethodField()
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = AutoEvaluationTexPassportFile
|
||||||
|
fields = ["id", "file"]
|
||||||
|
|
||||||
|
def get_file(self, obj):
|
||||||
|
request = self.context.get("request")
|
||||||
|
if not obj.file:
|
||||||
|
return None
|
||||||
|
if request:
|
||||||
|
return request.build_absolute_uri(obj.file.url)
|
||||||
|
return obj.file.url
|
||||||
|
|
||||||
|
|
||||||
class BaseAutoevaluationSerializer(serializers.ModelSerializer):
|
class BaseAutoevaluationSerializer(serializers.ModelSerializer):
|
||||||
status_display = serializers.CharField(source="get_status_display", read_only=True)
|
status_display = serializers.CharField(source="get_status_display", read_only=True)
|
||||||
object_type_display = serializers.CharField(source="get_object_type_display", read_only=True, default=None)
|
object_type_display = serializers.CharField(source="get_object_type_display", read_only=True, default=None)
|
||||||
@@ -17,6 +40,12 @@ class BaseAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
default=None)
|
default=None)
|
||||||
rate_type = ListReferenceitemSerializer(read_only=True)
|
rate_type = ListReferenceitemSerializer(read_only=True)
|
||||||
value_determined = ListReferenceitemSerializer(read_only=True)
|
value_determined = ListReferenceitemSerializer(read_only=True)
|
||||||
|
car_position = ListReferenceitemSerializer(read_only=True)
|
||||||
|
body_type = ListReferenceitemSerializer(read_only=True)
|
||||||
|
fuel_type = ListReferenceitemSerializer(read_only=True)
|
||||||
|
state_car = ListReferenceitemSerializer(read_only=True)
|
||||||
|
assessment_task_type = ListReferenceitemSerializer(read_only=True)
|
||||||
|
tex_passport_files = AutoEvaluationTexPassportFileSerializer(many=True, read_only=True)
|
||||||
user = serializers.SerializerMethodField(method_name="get_user", read_only=True)
|
user = serializers.SerializerMethodField(method_name="get_user", read_only=True)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
@@ -32,7 +61,9 @@ class BaseAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
"object_owner_individual_person_p_name",
|
"object_owner_individual_person_p_name",
|
||||||
"object_owner_legal_entity",
|
"object_owner_legal_entity",
|
||||||
"object_owner_legal_inn",
|
"object_owner_legal_inn",
|
||||||
|
"object_owner_residence",
|
||||||
"tex_passport_serie_num",
|
"tex_passport_serie_num",
|
||||||
|
"tex_passport_files",
|
||||||
"rating_goal",
|
"rating_goal",
|
||||||
"registration_number",
|
"registration_number",
|
||||||
"object_type",
|
"object_type",
|
||||||
@@ -42,6 +73,12 @@ class BaseAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
"car_number",
|
"car_number",
|
||||||
"manufacture_year",
|
"manufacture_year",
|
||||||
"car_color",
|
"car_color",
|
||||||
|
"distance_covered",
|
||||||
|
"car_position",
|
||||||
|
"body_type",
|
||||||
|
"fuel_type",
|
||||||
|
"state_car",
|
||||||
|
"assessment_task_type",
|
||||||
"status",
|
"status",
|
||||||
"status_display",
|
"status_display",
|
||||||
"created_at",
|
"created_at",
|
||||||
@@ -68,7 +105,6 @@ class ListAutoevaluationSerializer(BaseAutoevaluationSerializer):
|
|||||||
|
|
||||||
|
|
||||||
class RetrieveAutoevaluationSerializer(BaseAutoevaluationSerializer):
|
class RetrieveAutoevaluationSerializer(BaseAutoevaluationSerializer):
|
||||||
car_type_display = serializers.CharField(source="get_car_type_display", read_only=True, default=None)
|
|
||||||
car_wheel_display = serializers.CharField(source="get_car_wheel_display", read_only=True, default=None)
|
car_wheel_display = serializers.CharField(source="get_car_wheel_display", read_only=True, default=None)
|
||||||
|
|
||||||
class Meta(BaseAutoevaluationSerializer.Meta):
|
class Meta(BaseAutoevaluationSerializer.Meta):
|
||||||
@@ -89,11 +125,8 @@ class RetrieveAutoevaluationSerializer(BaseAutoevaluationSerializer):
|
|||||||
"object_owner_legal_inn",
|
"object_owner_legal_inn",
|
||||||
# Step 4
|
# Step 4
|
||||||
"tex_passport_serie_num",
|
"tex_passport_serie_num",
|
||||||
"tex_passport_file",
|
|
||||||
"tex_passport_gived_date",
|
"tex_passport_gived_date",
|
||||||
"tex_passport_gived_location",
|
"tex_passport_gived_location",
|
||||||
"car_type",
|
|
||||||
"car_type_display",
|
|
||||||
"car_wheel",
|
"car_wheel",
|
||||||
"car_wheel_display",
|
"car_wheel_display",
|
||||||
"car_dvigatel_number",
|
"car_dvigatel_number",
|
||||||
@@ -106,11 +139,6 @@ class RetrieveAutoevaluationSerializer(BaseAutoevaluationSerializer):
|
|||||||
|
|
||||||
|
|
||||||
class UpdateAutoevaluationSerializer(serializers.ModelSerializer):
|
class UpdateAutoevaluationSerializer(serializers.ModelSerializer):
|
||||||
value_determined = serializers.PrimaryKeyRelatedField(
|
|
||||||
queryset=ReferenceitemModel.objects.all(),
|
|
||||||
required=False,
|
|
||||||
allow_null=True,
|
|
||||||
)
|
|
||||||
value_determined = serializers.PrimaryKeyRelatedField(
|
value_determined = serializers.PrimaryKeyRelatedField(
|
||||||
queryset=ReferenceitemModel.objects.all(),
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
required=False,
|
required=False,
|
||||||
@@ -121,6 +149,36 @@ class UpdateAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
required=False,
|
required=False,
|
||||||
allow_null=True,
|
allow_null=True,
|
||||||
)
|
)
|
||||||
|
car_position = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
body_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
fuel_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
state_car = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
assessment_task_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
tex_passport_files = serializers.ListField(
|
||||||
|
child=serializers.FileField(),
|
||||||
|
required=False,
|
||||||
|
write_only=True,
|
||||||
|
)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = AutoEvaluationModel
|
model = AutoEvaluationModel
|
||||||
@@ -140,14 +198,15 @@ class UpdateAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
"object_owner_individual_person_passport_num",
|
"object_owner_individual_person_passport_num",
|
||||||
"object_owner_legal_entity",
|
"object_owner_legal_entity",
|
||||||
"object_owner_legal_inn",
|
"object_owner_legal_inn",
|
||||||
|
"object_owner_residence",
|
||||||
"value_determined",
|
"value_determined",
|
||||||
"rate_type",
|
"rate_type",
|
||||||
|
"assessment_task_type",
|
||||||
# Step 4
|
# Step 4
|
||||||
"tex_passport_file",
|
"tex_passport_files",
|
||||||
"tex_passport_serie_num",
|
"tex_passport_serie_num",
|
||||||
"tex_passport_gived_date",
|
"tex_passport_gived_date",
|
||||||
"tex_passport_gived_location",
|
"tex_passport_gived_location",
|
||||||
"car_type",
|
|
||||||
"car_wheel",
|
"car_wheel",
|
||||||
"car_brand",
|
"car_brand",
|
||||||
"car_model",
|
"car_model",
|
||||||
@@ -155,6 +214,11 @@ class UpdateAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
"manufacture_year",
|
"manufacture_year",
|
||||||
"car_dvigatel_number",
|
"car_dvigatel_number",
|
||||||
"car_color",
|
"car_color",
|
||||||
|
"distance_covered",
|
||||||
|
"car_position",
|
||||||
|
"body_type",
|
||||||
|
"fuel_type",
|
||||||
|
"state_car",
|
||||||
]
|
]
|
||||||
|
|
||||||
def validate_tex_passport_serie_num(self, value):
|
def validate_tex_passport_serie_num(self, value):
|
||||||
@@ -199,13 +263,23 @@ class UpdateAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
|
|
||||||
return attrs
|
return attrs
|
||||||
|
|
||||||
|
def update(self, instance, validated_data):
|
||||||
|
files = validated_data.pop("tex_passport_files", None)
|
||||||
|
with transaction.atomic():
|
||||||
|
for attr, value in validated_data.items():
|
||||||
|
setattr(instance, attr, value)
|
||||||
|
instance.save()
|
||||||
|
if files is not None:
|
||||||
|
AutoEvaluationTexPassportFile.objects.bulk_create([
|
||||||
|
AutoEvaluationTexPassportFile(auto_evaluation=instance, file=f) for f in files
|
||||||
|
])
|
||||||
|
return instance
|
||||||
|
|
||||||
|
def to_representation(self, instance):
|
||||||
|
return RetrieveAutoevaluationSerializer(instance, context=self.context).data
|
||||||
|
|
||||||
|
|
||||||
class CreateAutoevaluationSerializer(serializers.ModelSerializer):
|
class CreateAutoevaluationSerializer(serializers.ModelSerializer):
|
||||||
value_determined = serializers.PrimaryKeyRelatedField(
|
|
||||||
queryset=ReferenceitemModel.objects.all(),
|
|
||||||
required=False,
|
|
||||||
allow_null=True,
|
|
||||||
)
|
|
||||||
value_determined = serializers.PrimaryKeyRelatedField(
|
value_determined = serializers.PrimaryKeyRelatedField(
|
||||||
queryset=ReferenceitemModel.objects.all(),
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
required=False,
|
required=False,
|
||||||
@@ -216,11 +290,41 @@ class CreateAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
required=False,
|
required=False,
|
||||||
allow_null=True,
|
allow_null=True,
|
||||||
)
|
)
|
||||||
|
car_position = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
body_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
fuel_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
state_car = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
assessment_task_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
evaluation_request = serializers.PrimaryKeyRelatedField(
|
evaluation_request = serializers.PrimaryKeyRelatedField(
|
||||||
queryset=EvaluationrequestModel.objects.all(),
|
queryset=EvaluationrequestModel.objects.all(),
|
||||||
required=False,
|
required=False,
|
||||||
allow_null=True,
|
allow_null=True,
|
||||||
)
|
)
|
||||||
|
tex_passport_files = serializers.ListField(
|
||||||
|
child=serializers.FileField(),
|
||||||
|
required=False,
|
||||||
|
write_only=True,
|
||||||
|
)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = AutoEvaluationModel
|
model = AutoEvaluationModel
|
||||||
@@ -241,14 +345,15 @@ class CreateAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
"object_owner_individual_person_passport_num",
|
"object_owner_individual_person_passport_num",
|
||||||
"object_owner_legal_entity",
|
"object_owner_legal_entity",
|
||||||
"object_owner_legal_inn",
|
"object_owner_legal_inn",
|
||||||
|
"object_owner_residence",
|
||||||
"value_determined",
|
"value_determined",
|
||||||
"rate_type",
|
"rate_type",
|
||||||
|
"assessment_task_type",
|
||||||
# Step 4
|
# Step 4
|
||||||
"tex_passport_serie_num",
|
"tex_passport_serie_num",
|
||||||
"tex_passport_file",
|
"tex_passport_files",
|
||||||
"tex_passport_gived_date",
|
"tex_passport_gived_date",
|
||||||
"tex_passport_gived_location",
|
"tex_passport_gived_location",
|
||||||
"car_type",
|
|
||||||
"car_wheel",
|
"car_wheel",
|
||||||
"car_brand",
|
"car_brand",
|
||||||
"car_model",
|
"car_model",
|
||||||
@@ -256,6 +361,11 @@ class CreateAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
"manufacture_year",
|
"manufacture_year",
|
||||||
"car_dvigatel_number",
|
"car_dvigatel_number",
|
||||||
"car_color",
|
"car_color",
|
||||||
|
"distance_covered",
|
||||||
|
"car_position",
|
||||||
|
"body_type",
|
||||||
|
"fuel_type",
|
||||||
|
"state_car",
|
||||||
]
|
]
|
||||||
|
|
||||||
def validate_tex_passport_serie_num(self, value):
|
def validate_tex_passport_serie_num(self, value):
|
||||||
@@ -300,13 +410,23 @@ class CreateAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
return attrs
|
return attrs
|
||||||
|
|
||||||
def create(self, validated_data):
|
def create(self, validated_data):
|
||||||
|
files = validated_data.pop("tex_passport_files", [])
|
||||||
user = self.context.get('request').user
|
user = self.context.get('request').user
|
||||||
validated_data['user'] = user
|
validated_data['user'] = user
|
||||||
evaluation_req = validated_data.get("evaluation_request")
|
evaluation_req = validated_data.get("evaluation_request")
|
||||||
if evaluation_req:
|
if evaluation_req:
|
||||||
evaluation_req.status = RequestStatus.IN_PROGRESS
|
evaluation_req.status = RequestStatus.IN_PROGRESS
|
||||||
evaluation_req.save()
|
evaluation_req.save()
|
||||||
return super().create(validated_data)
|
with transaction.atomic():
|
||||||
|
instance = super().create(validated_data)
|
||||||
|
if files:
|
||||||
|
AutoEvaluationTexPassportFile.objects.bulk_create([
|
||||||
|
AutoEvaluationTexPassportFile(auto_evaluation=instance, file=f) for f in files
|
||||||
|
])
|
||||||
|
return instance
|
||||||
|
|
||||||
|
def to_representation(self, instance):
|
||||||
|
return RetrieveAutoevaluationSerializer(instance, context=self.context).data
|
||||||
|
|
||||||
|
|
||||||
class AutoEvaluationAppraisersSerializer(serializers.Serializer):
|
class AutoEvaluationAppraisersSerializer(serializers.Serializer):
|
||||||
@@ -321,6 +441,7 @@ class AutoEvaluationAppraisersSerializer(serializers.Serializer):
|
|||||||
data['users'] = users
|
data['users'] = users
|
||||||
return data
|
return data
|
||||||
|
|
||||||
|
|
||||||
class AutoEvaluationSerializer(serializers.Serializer):
|
class AutoEvaluationSerializer(serializers.Serializer):
|
||||||
brand = serializers.CharField()
|
brand = serializers.CharField()
|
||||||
brand_model = serializers.CharField()
|
brand_model = serializers.CharField()
|
||||||
@@ -331,7 +452,63 @@ class AutoEvaluationSerializer(serializers.Serializer):
|
|||||||
fuel_type = serializers.CharField()
|
fuel_type = serializers.CharField()
|
||||||
mileage = serializers.CharField()
|
mileage = serializers.CharField()
|
||||||
|
|
||||||
|
|
||||||
class AutoEvaluationModelSerializer(serializers.ModelSerializer):
|
class AutoEvaluationModelSerializer(serializers.ModelSerializer):
|
||||||
|
user = serializers.StringRelatedField(read_only=True)
|
||||||
|
appraisers = serializers.PrimaryKeyRelatedField(
|
||||||
|
many=True,
|
||||||
|
queryset=User.objects.all(),
|
||||||
|
required=False
|
||||||
|
)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = AutoEvaluationModel
|
model = AutoEvaluationModel
|
||||||
fields = "__all__"
|
fields = (
|
||||||
|
"registration_number",
|
||||||
|
"contract_date",
|
||||||
|
"object_inspection_date",
|
||||||
|
"rate_date",
|
||||||
|
"rate_report_date",
|
||||||
|
"object_type",
|
||||||
|
|
||||||
|
"object_owner_type",
|
||||||
|
"object_owner_individual_person_f_name",
|
||||||
|
"object_owner_individual_person_l_name",
|
||||||
|
"object_owner_individual_person_p_name",
|
||||||
|
"object_owner_individual_person_passport_num",
|
||||||
|
"object_owner_legal_entity",
|
||||||
|
"object_owner_legal_inn",
|
||||||
|
"object_owner_residence",
|
||||||
|
"value_determined",
|
||||||
|
"rate_type",
|
||||||
|
"assessment_task_type",
|
||||||
|
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"tex_passport_gived_date",
|
||||||
|
"tex_passport_gived_location",
|
||||||
|
"car_wheel",
|
||||||
|
"car_brand",
|
||||||
|
"car_model",
|
||||||
|
"car_number",
|
||||||
|
"manufacture_year",
|
||||||
|
"car_dvigatel_number",
|
||||||
|
"car_color",
|
||||||
|
"distance_covered",
|
||||||
|
"car_position",
|
||||||
|
"body_type",
|
||||||
|
"fuel_type",
|
||||||
|
"state_car",
|
||||||
|
|
||||||
|
"rating_goal",
|
||||||
|
"status",
|
||||||
|
"is_archived",
|
||||||
|
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
|
|
||||||
|
read_only_fields = (
|
||||||
|
"id",
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
|
|||||||
501
core/apps/evaluation/serializers/auto/MechanicAutoEvaluation.py
Normal file
501
core/apps/evaluation/serializers/auto/MechanicAutoEvaluation.py
Normal file
@@ -0,0 +1,501 @@
|
|||||||
|
import re
|
||||||
|
|
||||||
|
from django.contrib.auth import get_user_model
|
||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from django.db import transaction
|
||||||
|
|
||||||
|
from core.apps.evaluation.choices.request import RequestStatus
|
||||||
|
from core.apps.evaluation.models import (
|
||||||
|
MechanicAutoEvaluationModel,
|
||||||
|
MechanicAutoEvaluationTexPassportFile,
|
||||||
|
ReferenceitemModel,
|
||||||
|
EvaluationrequestModel,
|
||||||
|
)
|
||||||
|
from core.apps.evaluation.serializers.reference import ListReferenceitemSerializer
|
||||||
|
|
||||||
|
User = get_user_model()
|
||||||
|
|
||||||
|
|
||||||
|
class MechanicAutoEvaluationTexPassportFileSerializer(serializers.ModelSerializer):
|
||||||
|
file = serializers.SerializerMethodField()
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = MechanicAutoEvaluationTexPassportFile
|
||||||
|
fields = ["id", "file"]
|
||||||
|
|
||||||
|
def get_file(self, obj):
|
||||||
|
request = self.context.get("request")
|
||||||
|
if not obj.file:
|
||||||
|
return None
|
||||||
|
if request:
|
||||||
|
return request.build_absolute_uri(obj.file.url)
|
||||||
|
return obj.file.url
|
||||||
|
|
||||||
|
|
||||||
|
class BaseMechanicAutoevaluationSerializer(serializers.ModelSerializer):
|
||||||
|
status_display = serializers.CharField(source="get_status_display", read_only=True)
|
||||||
|
object_type_display = serializers.CharField(source="get_object_type_display", read_only=True, default=None)
|
||||||
|
object_owner_type_display = serializers.CharField(source="get_object_owner_type_display", read_only=True,
|
||||||
|
default=None)
|
||||||
|
rate_type = ListReferenceitemSerializer(read_only=True)
|
||||||
|
value_determined = ListReferenceitemSerializer(read_only=True)
|
||||||
|
car_position = ListReferenceitemSerializer(read_only=True)
|
||||||
|
body_type = ListReferenceitemSerializer(read_only=True)
|
||||||
|
fuel_type = ListReferenceitemSerializer(read_only=True)
|
||||||
|
state_car = ListReferenceitemSerializer(read_only=True)
|
||||||
|
assessment_task_type = ListReferenceitemSerializer(read_only=True)
|
||||||
|
tex_passport_files = MechanicAutoEvaluationTexPassportFileSerializer(many=True, read_only=True)
|
||||||
|
user = serializers.SerializerMethodField(method_name="get_user", read_only=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = MechanicAutoEvaluationModel
|
||||||
|
fields = [
|
||||||
|
"id",
|
||||||
|
"contract_date",
|
||||||
|
"object_inspection_date",
|
||||||
|
"object_owner_individual_person_passport_num",
|
||||||
|
"object_owner_type",
|
||||||
|
"object_owner_individual_person_f_name",
|
||||||
|
"object_owner_individual_person_l_name",
|
||||||
|
"object_owner_individual_person_p_name",
|
||||||
|
"object_owner_legal_entity",
|
||||||
|
"object_owner_legal_inn",
|
||||||
|
"object_owner_residence",
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"tex_passport_files",
|
||||||
|
"rating_goal",
|
||||||
|
"registration_number",
|
||||||
|
"object_type",
|
||||||
|
"object_type_display",
|
||||||
|
"car_brand",
|
||||||
|
"car_model",
|
||||||
|
"car_number",
|
||||||
|
"manufacture_year",
|
||||||
|
"car_color",
|
||||||
|
"distance_covered",
|
||||||
|
"car_position",
|
||||||
|
"body_type",
|
||||||
|
"fuel_type",
|
||||||
|
"state_car",
|
||||||
|
"assessment_task_type",
|
||||||
|
"status",
|
||||||
|
"status_display",
|
||||||
|
"created_at",
|
||||||
|
"value_determined",
|
||||||
|
"rate_type",
|
||||||
|
"user",
|
||||||
|
"evaluation_request",
|
||||||
|
]
|
||||||
|
|
||||||
|
def get_user(self, obj):
|
||||||
|
request = self.context.get('request')
|
||||||
|
return {
|
||||||
|
"id": obj.user.id,
|
||||||
|
"phone": obj.user.phone,
|
||||||
|
"first_name": obj.user.first_name,
|
||||||
|
"last_name": obj.user.last_name,
|
||||||
|
"avatar": request.build_absolute_uri(obj.user.avatar.url) if obj.user.avatar else None
|
||||||
|
} if obj.user else None
|
||||||
|
|
||||||
|
|
||||||
|
class ListMechanicAutoevaluationSerializer(BaseMechanicAutoevaluationSerializer):
|
||||||
|
class Meta(BaseMechanicAutoevaluationSerializer.Meta):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class RetrieveMechanicAutoevaluationSerializer(BaseMechanicAutoevaluationSerializer):
|
||||||
|
car_wheel_display = serializers.CharField(source="get_car_wheel_display", read_only=True, default=None)
|
||||||
|
|
||||||
|
class Meta(BaseMechanicAutoevaluationSerializer.Meta):
|
||||||
|
fields = BaseMechanicAutoevaluationSerializer.Meta.fields + [
|
||||||
|
"contract_date",
|
||||||
|
"object_inspection_date",
|
||||||
|
"rate_date",
|
||||||
|
"rate_report_date",
|
||||||
|
"object_owner_type",
|
||||||
|
"object_owner_type_display",
|
||||||
|
"object_owner_individual_person_f_name",
|
||||||
|
"object_owner_individual_person_l_name",
|
||||||
|
"object_owner_individual_person_p_name",
|
||||||
|
"object_owner_individual_person_passport_num",
|
||||||
|
"object_owner_legal_entity",
|
||||||
|
"object_owner_legal_inn",
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"tex_passport_gived_date",
|
||||||
|
"tex_passport_gived_location",
|
||||||
|
"car_wheel",
|
||||||
|
"car_wheel_display",
|
||||||
|
"car_dvigatel_number",
|
||||||
|
"valuation",
|
||||||
|
"vehicle",
|
||||||
|
"rating_goal",
|
||||||
|
"updated_at",
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class UpdateMechanicAutoevaluationSerializer(serializers.ModelSerializer):
|
||||||
|
value_determined = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
rate_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
car_position = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
body_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
fuel_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
state_car = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
assessment_task_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
user = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=User.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
tex_passport_files = serializers.ListField(
|
||||||
|
child=serializers.FileField(),
|
||||||
|
required=False,
|
||||||
|
write_only=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = MechanicAutoEvaluationModel
|
||||||
|
fields = [
|
||||||
|
"user",
|
||||||
|
"registration_number",
|
||||||
|
"contract_date",
|
||||||
|
"object_inspection_date",
|
||||||
|
"rate_date",
|
||||||
|
"rate_report_date",
|
||||||
|
"object_type",
|
||||||
|
"object_owner_type",
|
||||||
|
"object_owner_individual_person_f_name",
|
||||||
|
"object_owner_individual_person_l_name",
|
||||||
|
"object_owner_individual_person_p_name",
|
||||||
|
"object_owner_individual_person_passport_num",
|
||||||
|
"object_owner_legal_entity",
|
||||||
|
"object_owner_legal_inn",
|
||||||
|
"object_owner_residence",
|
||||||
|
"value_determined",
|
||||||
|
"rate_type",
|
||||||
|
"assessment_task_type",
|
||||||
|
"tex_passport_files",
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"tex_passport_gived_date",
|
||||||
|
"tex_passport_gived_location",
|
||||||
|
"car_wheel",
|
||||||
|
"car_brand",
|
||||||
|
"car_model",
|
||||||
|
"car_number",
|
||||||
|
"manufacture_year",
|
||||||
|
"car_dvigatel_number",
|
||||||
|
"car_color",
|
||||||
|
"distance_covered",
|
||||||
|
"car_position",
|
||||||
|
"body_type",
|
||||||
|
"fuel_type",
|
||||||
|
"state_car",
|
||||||
|
]
|
||||||
|
|
||||||
|
def validate_tex_passport_serie_num(self, value):
|
||||||
|
if value and not re.match(r"^[A-Z]{3}\s?\d{7}$", value):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
"Format: AAA 1234567 (3 harf + 7 raqam)"
|
||||||
|
)
|
||||||
|
return value
|
||||||
|
|
||||||
|
def validate_object_owner_individual_person_passport_num(self, value):
|
||||||
|
if value and not re.match(r"^[A-Z]{2}\s?\d{7}$", value):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
"Format: AA 1234567 (2 harf + 7 raqam)"
|
||||||
|
)
|
||||||
|
return value
|
||||||
|
|
||||||
|
def validate(self, attrs):
|
||||||
|
owner_type = attrs.get("object_owner_type")
|
||||||
|
|
||||||
|
if owner_type == 1:
|
||||||
|
required_fields = {
|
||||||
|
"object_owner_individual_person_f_name": "Ismi",
|
||||||
|
"object_owner_individual_person_l_name": "Familiyasi",
|
||||||
|
"object_owner_individual_person_p_name": "Sharifi",
|
||||||
|
"object_owner_individual_person_passport_num": "Passport raqami",
|
||||||
|
}
|
||||||
|
for field, label in required_fields.items():
|
||||||
|
if not attrs.get(field):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
{field: f"Jismoniy shaxs uchun {label} majburiy."}
|
||||||
|
)
|
||||||
|
|
||||||
|
elif owner_type == 2:
|
||||||
|
if not attrs.get("object_owner_legal_entity"):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
{"object_owner_legal_entity": "Yuridik shaxs nomi majburiy."}
|
||||||
|
)
|
||||||
|
if not attrs.get("object_owner_legal_inn"):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
{"object_owner_legal_inn": "INN raqami majburiy."}
|
||||||
|
)
|
||||||
|
|
||||||
|
return attrs
|
||||||
|
|
||||||
|
def update(self, instance, validated_data):
|
||||||
|
files = validated_data.pop("tex_passport_files", None)
|
||||||
|
with transaction.atomic():
|
||||||
|
for attr, value in validated_data.items():
|
||||||
|
setattr(instance, attr, value)
|
||||||
|
instance.save()
|
||||||
|
if files is not None:
|
||||||
|
MechanicAutoEvaluationTexPassportFile.objects.bulk_create([
|
||||||
|
MechanicAutoEvaluationTexPassportFile(mechanic_auto_evaluation=instance, file=f) for f in files
|
||||||
|
])
|
||||||
|
return instance
|
||||||
|
|
||||||
|
def to_representation(self, instance):
|
||||||
|
return RetrieveMechanicAutoevaluationSerializer(instance, context=self.context).data
|
||||||
|
|
||||||
|
|
||||||
|
class CreateMechanicAutoevaluationSerializer(serializers.ModelSerializer):
|
||||||
|
value_determined = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
rate_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
car_position = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
body_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
fuel_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
state_car = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
assessment_task_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
evaluation_request = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=EvaluationrequestModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
user = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=User.objects.all(),
|
||||||
|
required=True,
|
||||||
|
allow_null=False,
|
||||||
|
)
|
||||||
|
tex_passport_files = serializers.ListField(
|
||||||
|
child=serializers.FileField(),
|
||||||
|
required=False,
|
||||||
|
write_only=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = MechanicAutoEvaluationModel
|
||||||
|
fields = [
|
||||||
|
"user",
|
||||||
|
"registration_number",
|
||||||
|
"evaluation_request",
|
||||||
|
"contract_date",
|
||||||
|
"object_inspection_date",
|
||||||
|
"rate_date",
|
||||||
|
"rate_report_date",
|
||||||
|
"object_type",
|
||||||
|
"object_owner_type",
|
||||||
|
"object_owner_individual_person_f_name",
|
||||||
|
"object_owner_individual_person_l_name",
|
||||||
|
"object_owner_individual_person_p_name",
|
||||||
|
"object_owner_individual_person_passport_num",
|
||||||
|
"object_owner_legal_entity",
|
||||||
|
"object_owner_legal_inn",
|
||||||
|
"object_owner_residence",
|
||||||
|
"value_determined",
|
||||||
|
"rate_type",
|
||||||
|
"assessment_task_type",
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"tex_passport_files",
|
||||||
|
"tex_passport_gived_date",
|
||||||
|
"tex_passport_gived_location",
|
||||||
|
"car_wheel",
|
||||||
|
"car_brand",
|
||||||
|
"car_model",
|
||||||
|
"car_number",
|
||||||
|
"manufacture_year",
|
||||||
|
"car_dvigatel_number",
|
||||||
|
"car_color",
|
||||||
|
"distance_covered",
|
||||||
|
"car_position",
|
||||||
|
"body_type",
|
||||||
|
"fuel_type",
|
||||||
|
"state_car",
|
||||||
|
]
|
||||||
|
|
||||||
|
def validate_tex_passport_serie_num(self, value):
|
||||||
|
if value and not re.match(r"^[A-Z]{3}\s?\d{7}$", value):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
"Format: AAA 1234567 (3 harf + 7 raqam)"
|
||||||
|
)
|
||||||
|
return value
|
||||||
|
|
||||||
|
def validate_object_owner_individual_person_passport_num(self, value):
|
||||||
|
if value and not re.match(r"^[A-Z]{2}\s?\d{7}$", value):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
"Format: AA 1234567 (2 harf + 7 raqam)"
|
||||||
|
)
|
||||||
|
return value
|
||||||
|
|
||||||
|
def validate(self, attrs):
|
||||||
|
owner_type = attrs.get("object_owner_type")
|
||||||
|
if owner_type == 1:
|
||||||
|
required_fields = {
|
||||||
|
"object_owner_individual_person_f_name": "Ismi",
|
||||||
|
"object_owner_individual_person_l_name": "Familiyasi",
|
||||||
|
"object_owner_individual_person_p_name": "Sharifi",
|
||||||
|
"object_owner_individual_person_passport_num": "Passport raqami",
|
||||||
|
}
|
||||||
|
for field, label in required_fields.items():
|
||||||
|
if not attrs.get(field):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
{field: f"Jismoniy shaxs uchun {label} majburiy."}
|
||||||
|
)
|
||||||
|
|
||||||
|
elif owner_type == 2:
|
||||||
|
if not attrs.get("object_owner_legal_entity"):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
{"object_owner_legal_entity": "Yuridik shaxs nomi majburiy."}
|
||||||
|
)
|
||||||
|
if not attrs.get("object_owner_legal_inn"):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
{"object_owner_legal_inn": "INN raqami majburiy."}
|
||||||
|
)
|
||||||
|
|
||||||
|
return attrs
|
||||||
|
|
||||||
|
def create(self, validated_data):
|
||||||
|
files = validated_data.pop("tex_passport_files", [])
|
||||||
|
evaluation_req = validated_data.get("evaluation_request")
|
||||||
|
if evaluation_req:
|
||||||
|
evaluation_req.status = RequestStatus.IN_PROGRESS
|
||||||
|
evaluation_req.save()
|
||||||
|
with transaction.atomic():
|
||||||
|
instance = super().create(validated_data)
|
||||||
|
if files:
|
||||||
|
MechanicAutoEvaluationTexPassportFile.objects.bulk_create([
|
||||||
|
MechanicAutoEvaluationTexPassportFile(mechanic_auto_evaluation=instance, file=f) for f in files
|
||||||
|
])
|
||||||
|
return instance
|
||||||
|
|
||||||
|
def to_representation(self, instance):
|
||||||
|
return RetrieveMechanicAutoevaluationSerializer(instance, context=self.context).data
|
||||||
|
|
||||||
|
|
||||||
|
class MechanicAutoEvaluationAppraisersSerializer(serializers.Serializer):
|
||||||
|
ids = serializers.ListField(child=serializers.IntegerField())
|
||||||
|
|
||||||
|
def validate(self, data):
|
||||||
|
if not data.get("ids"):
|
||||||
|
raise serializers.ValidationError("Appraisers IDs are required.")
|
||||||
|
users = User.objects.filter(id__in=data["ids"])
|
||||||
|
if not users:
|
||||||
|
raise serializers.ValidationError("Invalid appraisers IDs.")
|
||||||
|
data['users'] = users
|
||||||
|
return data
|
||||||
|
|
||||||
|
|
||||||
|
class MechanicAutoEvaluationModelSerializer(serializers.ModelSerializer):
|
||||||
|
user = serializers.StringRelatedField(read_only=True)
|
||||||
|
appraisers = serializers.PrimaryKeyRelatedField(
|
||||||
|
many=True,
|
||||||
|
queryset=User.objects.all(),
|
||||||
|
required=False
|
||||||
|
)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = MechanicAutoEvaluationModel
|
||||||
|
fields = (
|
||||||
|
"registration_number",
|
||||||
|
"contract_date",
|
||||||
|
"object_inspection_date",
|
||||||
|
"rate_date",
|
||||||
|
"rate_report_date",
|
||||||
|
"object_type",
|
||||||
|
"object_owner_type",
|
||||||
|
"object_owner_individual_person_f_name",
|
||||||
|
"object_owner_individual_person_l_name",
|
||||||
|
"object_owner_individual_person_p_name",
|
||||||
|
"object_owner_individual_person_passport_num",
|
||||||
|
"object_owner_legal_entity",
|
||||||
|
"object_owner_legal_inn",
|
||||||
|
"object_owner_residence",
|
||||||
|
"value_determined",
|
||||||
|
"rate_type",
|
||||||
|
"assessment_task_type",
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"tex_passport_gived_date",
|
||||||
|
"tex_passport_gived_location",
|
||||||
|
"car_wheel",
|
||||||
|
"car_brand",
|
||||||
|
"car_model",
|
||||||
|
"car_number",
|
||||||
|
"manufacture_year",
|
||||||
|
"car_dvigatel_number",
|
||||||
|
"car_color",
|
||||||
|
"distance_covered",
|
||||||
|
"car_position",
|
||||||
|
"body_type",
|
||||||
|
"fuel_type",
|
||||||
|
"state_car",
|
||||||
|
"rating_goal",
|
||||||
|
"status",
|
||||||
|
"is_archived",
|
||||||
|
"user",
|
||||||
|
"appraisers",
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
|
|
||||||
|
read_only_fields = (
|
||||||
|
"id",
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
@@ -1 +1,2 @@
|
|||||||
from .AutoEvaluation import * # noqa
|
from .AutoEvaluation import * # noqa
|
||||||
|
from .MechanicAutoEvaluation import * # noqa
|
||||||
|
|||||||
56
core/apps/evaluation/serializers/bonus/Bonus.py
Normal file
56
core/apps/evaluation/serializers/bonus/Bonus.py
Normal 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
|
||||||
0
core/apps/evaluation/serializers/bonus/__init__.py
Normal file
0
core/apps/evaluation/serializers/bonus/__init__.py
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from core.apps.evaluation.models import MechanicAutoevaluationhistoryModel
|
||||||
|
|
||||||
|
|
||||||
|
class BaseMechanicAutoevaluationhistorySerializer(serializers.ModelSerializer):
|
||||||
|
event_type_display = serializers.CharField(source="get_event_type_display", read_only=True)
|
||||||
|
actor = serializers.SerializerMethodField()
|
||||||
|
|
||||||
|
def get_actor(self, obj):
|
||||||
|
return {
|
||||||
|
"id": obj.actor_id,
|
||||||
|
"full_name": obj.actor_full_name,
|
||||||
|
"role": obj.actor_role,
|
||||||
|
}
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = MechanicAutoevaluationhistoryModel
|
||||||
|
fields = [
|
||||||
|
"id",
|
||||||
|
"event_type",
|
||||||
|
"event_type_display",
|
||||||
|
"actor",
|
||||||
|
"meta",
|
||||||
|
"created_at",
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class ListMechanicAutoevaluationhistorySerializer(BaseMechanicAutoevaluationhistorySerializer):
|
||||||
|
class Meta(BaseMechanicAutoevaluationhistorySerializer.Meta): ...
|
||||||
|
|
||||||
|
|
||||||
|
class RetrieveMechanicAutoevaluationhistorySerializer(BaseMechanicAutoevaluationhistorySerializer):
|
||||||
|
class Meta(BaseMechanicAutoevaluationhistorySerializer.Meta):
|
||||||
|
fields = BaseMechanicAutoevaluationhistorySerializer.Meta.fields + ["mechanic_auto_evaluation"]
|
||||||
|
|
||||||
|
|
||||||
|
class CreateMechanicAutoevaluationhistorySerializer(BaseMechanicAutoevaluationhistorySerializer):
|
||||||
|
class Meta(BaseMechanicAutoevaluationhistorySerializer.Meta): ...
|
||||||
@@ -1,2 +1,3 @@
|
|||||||
from .AutoEvaluationHistory import * # noqa
|
from .AutoEvaluationHistory import * # noqa
|
||||||
|
from .MechanicAutoEvaluationHistory import * # noqa
|
||||||
from .QuickEvaluationHistory import * # noqa
|
from .QuickEvaluationHistory import * # noqa
|
||||||
|
|||||||
@@ -131,4 +131,39 @@ class CreateQuickevaluationSerializer(serializers.ModelSerializer):
|
|||||||
class QuickEvaluationModelSerializer(serializers.ModelSerializer):
|
class QuickEvaluationModelSerializer(serializers.ModelSerializer):
|
||||||
class Meta:
|
class Meta:
|
||||||
model = QuickEvaluationModel
|
model = QuickEvaluationModel
|
||||||
fields = '__all__'
|
fields = (
|
||||||
|
"id",
|
||||||
|
|
||||||
|
"created_by",
|
||||||
|
"brand",
|
||||||
|
"marka",
|
||||||
|
"car_position",
|
||||||
|
"body_type",
|
||||||
|
"color",
|
||||||
|
"fuel_type",
|
||||||
|
"state_car",
|
||||||
|
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"tech_passport_issued_date",
|
||||||
|
"tech_passport_issued_place",
|
||||||
|
|
||||||
|
"car_type",
|
||||||
|
"distance_covered",
|
||||||
|
"vin_number",
|
||||||
|
"car_number",
|
||||||
|
"car_manufactured_date",
|
||||||
|
"engine_number",
|
||||||
|
|
||||||
|
"estimated_price",
|
||||||
|
"status",
|
||||||
|
"is_archive",
|
||||||
|
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
|
|
||||||
|
read_only_fields = (
|
||||||
|
"id",
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
@@ -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)
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ router = DefaultRouter()
|
|||||||
router.register("document-category", views.DocumentCategoryView, basename="DocumentCategory")
|
router.register("document-category", views.DocumentCategoryView, basename="DocumentCategory")
|
||||||
router.register("document", views.DocumentView, basename="Document")
|
router.register("document", views.DocumentView, basename="Document")
|
||||||
router.register("auto-evaluation-history", views.AutoEvaluationHistoryView, basename="auto-evaluation-history")
|
router.register("auto-evaluation-history", views.AutoEvaluationHistoryView, basename="auto-evaluation-history")
|
||||||
|
router.register("mechanic-auto-evaluation-history", views.MechanicAutoEvaluationHistoryView, basename="mechanic-auto-evaluation-history")
|
||||||
router.register("quick-evaluation-history", views.QuickEvaluationHistoryView, basename="quick-evaluation-history")
|
router.register("quick-evaluation-history", views.QuickEvaluationHistoryView, basename="quick-evaluation-history")
|
||||||
router.register("determined-value", views.DeterminedValueView, basename="determined-value")
|
router.register("determined-value", views.DeterminedValueView, basename="determined-value")
|
||||||
router.register("evaluation-purpose", views.EvaluationPurposeView, basename="evaluation-purpose")
|
router.register("evaluation-purpose", views.EvaluationPurposeView, basename="evaluation-purpose")
|
||||||
@@ -22,11 +23,15 @@ router.register("quick-evaluation", views.QuickEvaluationView, basename="quick-e
|
|||||||
router.register("movable-property-evaluation", views.MovablePropertyEvaluationView, basename="movable-property-evaluation")
|
router.register("movable-property-evaluation", views.MovablePropertyEvaluationView, basename="movable-property-evaluation")
|
||||||
router.register("real-estate-evaluation", views.RealEstateEvaluationView, basename="real-estate-evaluation")
|
router.register("real-estate-evaluation", views.RealEstateEvaluationView, basename="real-estate-evaluation")
|
||||||
router.register("auto-evaluation", views.AutoEvaluationView, basename="auto-evaluation")
|
router.register("auto-evaluation", views.AutoEvaluationView, basename="auto-evaluation")
|
||||||
|
router.register("mechanic-auto-evaluation", views.MechanicAutoEvaluationView, basename="mechanic-auto-evaluation")
|
||||||
router.register("vehicle", views.VehicleView, basename="vehicle")
|
router.register("vehicle", views.VehicleView, basename="vehicle")
|
||||||
router.register("valuation", views.ValuationView, basename="valuation")
|
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)),
|
||||||
@@ -69,6 +74,26 @@ urlpatterns = [
|
|||||||
]
|
]
|
||||||
)),
|
)),
|
||||||
|
|
||||||
|
# Mechanic Auto Evaluation
|
||||||
|
path("mechanic-auto-evaluation/", include(
|
||||||
|
[
|
||||||
|
path("admin/", views.AdminMechanicEvaluationsAPIView.as_view(), name="admin-mechanic-evaluations"),
|
||||||
|
path('archive/', include(
|
||||||
|
[
|
||||||
|
path('<int:pk>/', views.MechanicAutoEvaluationArchiveAPIView.as_view()),
|
||||||
|
path('list/', views.MechanicAutoEvaluationArchivedListAPIView.as_view())
|
||||||
|
]
|
||||||
|
)),
|
||||||
|
path('appraisers/', include(
|
||||||
|
[
|
||||||
|
path("<int:id>/list/", views.MechanicAutoEvaluationListAppraisersView.as_view()),
|
||||||
|
path("<int:id>/set/", views.MechanicAutoEvaluationSetAppraisersView.as_view()),
|
||||||
|
path("<int:id>/remove/", views.MechanicAutoEvaluationRemoveAppraisersView.as_view()),
|
||||||
|
]
|
||||||
|
))
|
||||||
|
]
|
||||||
|
)),
|
||||||
|
|
||||||
# Evaluation Request
|
# Evaluation Request
|
||||||
path("evaluation-request/", include(
|
path("evaluation-request/", include(
|
||||||
[
|
[
|
||||||
@@ -85,4 +110,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()),
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
from .auto import * # noqa
|
from .auto import * # noqa
|
||||||
|
from .mechanic_auto import * # noqa
|
||||||
from .customer import * # noqa
|
from .customer import * # noqa
|
||||||
from .document import * # noqa
|
from .document import * # noqa
|
||||||
from .documentcategory import * # noqa
|
from .documentcategory import * # noqa
|
||||||
@@ -15,3 +16,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 *
|
||||||
@@ -11,7 +11,7 @@ from rest_framework.response import Response
|
|||||||
from rest_framework.views import APIView
|
from rest_framework.views import APIView
|
||||||
from rest_framework.viewsets import ModelViewSet
|
from rest_framework.viewsets import ModelViewSet
|
||||||
|
|
||||||
from core.apps.accounts.choices import RoleChoice
|
from core.apps.accounts.permissions import IsAdminRole
|
||||||
from core.apps.accounts.serializers.user import UserSerializer
|
from core.apps.accounts.serializers.user import UserSerializer
|
||||||
from core.apps.evaluation.filters.auto import AutoevaluationFilter
|
from core.apps.evaluation.filters.auto import AutoevaluationFilter
|
||||||
from core.apps.evaluation.models import AutoEvaluationModel
|
from core.apps.evaluation.models import AutoEvaluationModel
|
||||||
@@ -177,13 +177,14 @@ class AutoEvaluationArchiveAPIView(APIView):
|
|||||||
status=200
|
status=200
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@extend_schema(tags=["AutoEvaluation"])
|
@extend_schema(tags=["AutoEvaluation"])
|
||||||
class AdminEvaluationsAPIView(generics.GenericAPIView):
|
class AdminEvaluationsAPIView(generics.GenericAPIView):
|
||||||
permission_classes = [IsAuthenticated]
|
permission_classes = [IsAuthenticated, IsAdminRole]
|
||||||
|
queryset = AutoEvaluationModel.objects.all()
|
||||||
|
serializer_class = AutoEvaluationModel
|
||||||
|
|
||||||
def get(self, request):
|
def get(self, request):
|
||||||
if request.user.role != RoleChoice.ADMIN:
|
|
||||||
return Response({'detail': 'Forbidden'}, status=403)
|
|
||||||
auto_eval = AutoEvaluationModel.objects.filter(
|
auto_eval = AutoEvaluationModel.objects.filter(
|
||||||
created_by=self.request.user
|
created_by=self.request.user
|
||||||
).select_related('appraisers').distinct()
|
).select_related('appraisers').distinct()
|
||||||
|
|||||||
60
core/apps/evaluation/views/bonus.py
Normal file
60
core/apps/evaluation/views/bonus.py
Normal 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],
|
||||||
|
}
|
||||||
@@ -16,9 +16,14 @@ from rest_framework.viewsets import ReadOnlyModelViewSet
|
|||||||
# core apps
|
# core apps
|
||||||
from core.apps.evaluation.filters.history import (
|
from core.apps.evaluation.filters.history import (
|
||||||
AutoevaluationhistoryFilter,
|
AutoevaluationhistoryFilter,
|
||||||
|
MechanicAutoevaluationhistoryFilter,
|
||||||
QuickevaluationhistoryFilter,
|
QuickevaluationhistoryFilter,
|
||||||
)
|
)
|
||||||
from core.apps.evaluation.models import AutoevaluationhistoryModel, QuickevaluationhistoryModel
|
from core.apps.evaluation.models import (
|
||||||
|
AutoevaluationhistoryModel,
|
||||||
|
MechanicAutoevaluationhistoryModel,
|
||||||
|
QuickevaluationhistoryModel,
|
||||||
|
)
|
||||||
from core.apps.evaluation.serializers import history as serializers
|
from core.apps.evaluation.serializers import history as serializers
|
||||||
|
|
||||||
|
|
||||||
@@ -72,6 +77,56 @@ class AutoEvaluationHistoryView(BaseViewSetMixin, ReadOnlyModelViewSet):
|
|||||||
})
|
})
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(
|
||||||
|
tags=["MechanicAutoEvaluationHistory"],
|
||||||
|
parameters=[
|
||||||
|
OpenApiParameter("mechanic_auto_evaluation", int, description="MechanicAutoEvaluation ID bo'yicha filter"),
|
||||||
|
OpenApiParameter("event_type", str, description="Event turi bo'yicha filter"),
|
||||||
|
OpenApiParameter("created_from", str, description="Boshlanish sanasi (ISO 8601)"),
|
||||||
|
OpenApiParameter("created_to", str, description="Tugash sanasi (ISO 8601)"),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
class MechanicAutoEvaluationHistoryView(BaseViewSetMixin, ReadOnlyModelViewSet):
|
||||||
|
queryset = MechanicAutoevaluationhistoryModel.objects.only(
|
||||||
|
"id", "mechanic_auto_evaluation_id", "event_type",
|
||||||
|
"actor_id", "actor_full_name", "actor_role",
|
||||||
|
"meta", "created_at",
|
||||||
|
)
|
||||||
|
serializer_class = serializers.ListMechanicAutoevaluationhistorySerializer
|
||||||
|
permission_classes = [AllowAny]
|
||||||
|
pagination_class = None
|
||||||
|
|
||||||
|
filter_backends = [DjangoFilterBackend, OrderingFilter]
|
||||||
|
filterset_class = MechanicAutoevaluationhistoryFilter
|
||||||
|
ordering_fields = [
|
||||||
|
"id",
|
||||||
|
"event_type",
|
||||||
|
"event_type_display",
|
||||||
|
"actor",
|
||||||
|
"meta",
|
||||||
|
"created_at",
|
||||||
|
]
|
||||||
|
ordering = ["created_at"]
|
||||||
|
|
||||||
|
action_permission_classes = {}
|
||||||
|
action_serializer_class = {
|
||||||
|
"list": serializers.ListMechanicAutoevaluationhistorySerializer,
|
||||||
|
"retrieve": serializers.RetrieveMechanicAutoevaluationhistorySerializer,
|
||||||
|
"create": serializers.CreateMechanicAutoevaluationhistorySerializer,
|
||||||
|
}
|
||||||
|
|
||||||
|
def list(self, request, *args, **kwargs):
|
||||||
|
queryset = self.filter_queryset(self.get_queryset())
|
||||||
|
results = list(queryset)
|
||||||
|
serializer = self.get_serializer(results, many=True)
|
||||||
|
return Response({
|
||||||
|
"count": len(results),
|
||||||
|
"next": None,
|
||||||
|
"previous": None,
|
||||||
|
"results": serializer.data,
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
@extend_schema(
|
@extend_schema(
|
||||||
tags=["QuickEvaluationHistory"],
|
tags=["QuickEvaluationHistory"],
|
||||||
parameters=[
|
parameters=[
|
||||||
|
|||||||
198
core/apps/evaluation/views/mechanic_auto.py
Normal file
198
core/apps/evaluation/views/mechanic_auto.py
Normal file
@@ -0,0 +1,198 @@
|
|||||||
|
from django.db.models import Q
|
||||||
|
from django.shortcuts import get_object_or_404
|
||||||
|
from django_core.mixins import BaseViewSetMixin
|
||||||
|
from django_filters.rest_framework import DjangoFilterBackend
|
||||||
|
from drf_spectacular.utils import extend_schema, OpenApiParameter
|
||||||
|
from rest_framework import generics
|
||||||
|
from rest_framework.filters import OrderingFilter, SearchFilter
|
||||||
|
from rest_framework.generics import GenericAPIView, ListAPIView
|
||||||
|
from rest_framework.permissions import AllowAny, IsAuthenticated
|
||||||
|
from rest_framework.response import Response
|
||||||
|
from rest_framework.views import APIView
|
||||||
|
from rest_framework.viewsets import ModelViewSet
|
||||||
|
|
||||||
|
from core.apps.accounts.permissions import IsAdminRole
|
||||||
|
from core.apps.accounts.serializers.user import UserSerializer
|
||||||
|
from core.apps.evaluation.filters.mechanic_auto import MechanicAutoevaluationFilter
|
||||||
|
from core.apps.evaluation.models import MechanicAutoEvaluationModel
|
||||||
|
from core.apps.evaluation.serializers.auto.MechanicAutoEvaluation import (
|
||||||
|
ListMechanicAutoevaluationSerializer,
|
||||||
|
RetrieveMechanicAutoevaluationSerializer,
|
||||||
|
CreateMechanicAutoevaluationSerializer,
|
||||||
|
UpdateMechanicAutoevaluationSerializer,
|
||||||
|
MechanicAutoEvaluationAppraisersSerializer,
|
||||||
|
MechanicAutoEvaluationModelSerializer,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=["MechanicAutoEvaluation"])
|
||||||
|
class MechanicAutoEvaluationView(BaseViewSetMixin, ModelViewSet):
|
||||||
|
queryset = MechanicAutoEvaluationModel.objects.select_related(
|
||||||
|
"valuation",
|
||||||
|
"valuation__customer",
|
||||||
|
"vehicle",
|
||||||
|
).filter(is_archived=False)
|
||||||
|
serializer_class = ListMechanicAutoevaluationSerializer
|
||||||
|
permission_classes = [AllowAny]
|
||||||
|
|
||||||
|
filter_backends = [DjangoFilterBackend, SearchFilter, OrderingFilter]
|
||||||
|
filterset_class = MechanicAutoevaluationFilter
|
||||||
|
search_fields = [
|
||||||
|
"registration_number",
|
||||||
|
"car_model",
|
||||||
|
"car_brand",
|
||||||
|
"car_number",
|
||||||
|
]
|
||||||
|
ordering_fields = [
|
||||||
|
"id",
|
||||||
|
"contract_date",
|
||||||
|
"object_inspection_date",
|
||||||
|
"object_owner_individual_person_passport_num",
|
||||||
|
"object_owner_type",
|
||||||
|
"object_owner_individual_person_f_name",
|
||||||
|
"object_owner_individual_person_l_name",
|
||||||
|
"object_owner_individual_person_p_name",
|
||||||
|
"object_owner_legal_entity",
|
||||||
|
"object_owner_legal_inn",
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"rating_goal",
|
||||||
|
"registration_number",
|
||||||
|
"object_type",
|
||||||
|
"object_type_display",
|
||||||
|
"car_brand",
|
||||||
|
"car_model",
|
||||||
|
"car_number",
|
||||||
|
"manufacture_year",
|
||||||
|
"car_color",
|
||||||
|
"status",
|
||||||
|
"status_display",
|
||||||
|
"created_at",
|
||||||
|
"value_determined",
|
||||||
|
"rate_type",
|
||||||
|
]
|
||||||
|
ordering = ["-created_at"]
|
||||||
|
|
||||||
|
action_permission_classes = {}
|
||||||
|
action_serializer_class = {
|
||||||
|
"list": ListMechanicAutoevaluationSerializer,
|
||||||
|
"retrieve": RetrieveMechanicAutoevaluationSerializer,
|
||||||
|
"create": CreateMechanicAutoevaluationSerializer,
|
||||||
|
"update": UpdateMechanicAutoevaluationSerializer,
|
||||||
|
"partial_update": UpdateMechanicAutoevaluationSerializer,
|
||||||
|
}
|
||||||
|
|
||||||
|
def serializer_context(self):
|
||||||
|
return self.serializer_class(context={'request': self.request})
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=["MechanicAutoEvaluation"])
|
||||||
|
class MechanicAutoEvaluationSetAppraisersView(GenericAPIView):
|
||||||
|
permission_classes = [IsAuthenticated]
|
||||||
|
queryset = MechanicAutoEvaluationModel.objects.all()
|
||||||
|
serializer_class = MechanicAutoEvaluationAppraisersSerializer
|
||||||
|
|
||||||
|
def post(self, request, id):
|
||||||
|
try:
|
||||||
|
evaluation = get_object_or_404(MechanicAutoEvaluationModel, id=id)
|
||||||
|
serializer = self.serializer_class(data=request.data)
|
||||||
|
serializer.is_valid(raise_exception=True)
|
||||||
|
users = serializer.validated_data.get("users")
|
||||||
|
evaluation.appraisers.set(users)
|
||||||
|
evaluation.save()
|
||||||
|
return Response({"success": True, "data": "Foydalanuvchilar muvaffaqiyatli qo'shildi"})
|
||||||
|
except Exception as e:
|
||||||
|
return Response({"error": str(e)}, status=500)
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=["MechanicAutoEvaluation"])
|
||||||
|
class MechanicAutoEvaluationRemoveAppraisersView(GenericAPIView):
|
||||||
|
permission_classes = [IsAuthenticated]
|
||||||
|
queryset = MechanicAutoEvaluationModel.objects.all()
|
||||||
|
serializer_class = MechanicAutoEvaluationAppraisersSerializer
|
||||||
|
|
||||||
|
def post(self, request, id):
|
||||||
|
try:
|
||||||
|
evaluation = get_object_or_404(MechanicAutoEvaluationModel, id=id)
|
||||||
|
serializer = self.serializer_class(data=request.data)
|
||||||
|
serializer.is_valid(raise_exception=True)
|
||||||
|
users = serializer.validated_data.get("users")
|
||||||
|
evaluation.appraisers.remove(*users)
|
||||||
|
evaluation.save()
|
||||||
|
return Response({"success": True, "data": "Foydalanuvchilar muvaffaqiyatli o'chirildi"})
|
||||||
|
except Exception as e:
|
||||||
|
return Response({"error": str(e)}, status=500)
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=["MechanicAutoEvaluation"])
|
||||||
|
class MechanicAutoEvaluationListAppraisersView(GenericAPIView):
|
||||||
|
permission_classes = [IsAuthenticated]
|
||||||
|
queryset = MechanicAutoEvaluationModel.objects.all()
|
||||||
|
serializer_class = UserSerializer
|
||||||
|
|
||||||
|
@extend_schema(
|
||||||
|
parameters=[
|
||||||
|
OpenApiParameter(
|
||||||
|
name="search",
|
||||||
|
type=str,
|
||||||
|
description="Search query",
|
||||||
|
required=False,
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
def get(self, request, id):
|
||||||
|
try:
|
||||||
|
search_query = request.query_params.get("search", "")
|
||||||
|
evaluation = get_object_or_404(MechanicAutoEvaluationModel, id=id)
|
||||||
|
query = evaluation.appraisers.all()
|
||||||
|
if search_query:
|
||||||
|
query = query.filter(
|
||||||
|
Q(phone__icontains=search_query) |
|
||||||
|
Q(first_name__icontains=search_query) |
|
||||||
|
Q(last_name__icontains=search_query)
|
||||||
|
)
|
||||||
|
page = self.paginate_queryset(query)
|
||||||
|
serializer = self.serializer_class(page, many=True)
|
||||||
|
return self.get_paginated_response(serializer.data)
|
||||||
|
except Exception as e:
|
||||||
|
return Response({"error": str(e)}, status=500)
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=["MechanicAutoEvaluation"])
|
||||||
|
class MechanicAutoEvaluationArchivedListAPIView(ListAPIView):
|
||||||
|
permission_classes = [IsAuthenticated]
|
||||||
|
serializer_class = ListMechanicAutoevaluationSerializer
|
||||||
|
|
||||||
|
def get_queryset(self):
|
||||||
|
return MechanicAutoEvaluationModel.objects.filter(is_archived=True)
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=["MechanicAutoEvaluation"])
|
||||||
|
class MechanicAutoEvaluationArchiveAPIView(APIView):
|
||||||
|
permission_classes = [IsAuthenticated]
|
||||||
|
|
||||||
|
def post(self, request, pk):
|
||||||
|
evaluation = get_object_or_404(MechanicAutoEvaluationModel, pk=pk)
|
||||||
|
evaluation.is_archived = request.data["is_archived"]
|
||||||
|
evaluation.save()
|
||||||
|
return Response(
|
||||||
|
{
|
||||||
|
"success": True,
|
||||||
|
"status": evaluation.status,
|
||||||
|
"id": evaluation.pk
|
||||||
|
},
|
||||||
|
status=200
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=["MechanicAutoEvaluation"])
|
||||||
|
class AdminMechanicEvaluationsAPIView(generics.GenericAPIView):
|
||||||
|
permission_classes = [IsAuthenticated, IsAdminRole]
|
||||||
|
queryset = MechanicAutoEvaluationModel.objects.all()
|
||||||
|
serializer_class = MechanicAutoEvaluationModelSerializer
|
||||||
|
|
||||||
|
def get(self, request):
|
||||||
|
evaluations = MechanicAutoEvaluationModel.objects.filter(
|
||||||
|
created_by=self.request.user
|
||||||
|
).distinct()
|
||||||
|
serializer = MechanicAutoEvaluationModelSerializer(evaluations, many=True)
|
||||||
|
return Response(serializer.data)
|
||||||
@@ -16,7 +16,7 @@ from rest_framework.response import Response
|
|||||||
from rest_framework.views import APIView
|
from rest_framework.views import APIView
|
||||||
from rest_framework.viewsets import ModelViewSet
|
from rest_framework.viewsets import ModelViewSet
|
||||||
|
|
||||||
from core.apps.accounts.choices import RoleChoice
|
from core.apps.accounts.permissions import IsAdminRole
|
||||||
# core apps
|
# core apps
|
||||||
from core.apps.evaluation.filters.quick import QuickevaluationFilter
|
from core.apps.evaluation.filters.quick import QuickevaluationFilter
|
||||||
from core.apps.evaluation.models import QuickEvaluationModel
|
from core.apps.evaluation.models import QuickEvaluationModel
|
||||||
@@ -88,11 +88,11 @@ class QuickEvaluationArchivedListAPIView(ListAPIView):
|
|||||||
|
|
||||||
@extend_schema(tags=["QuickEvaluation"])
|
@extend_schema(tags=["QuickEvaluation"])
|
||||||
class AdminQuickEvalAPIView(generics.GenericAPIView):
|
class AdminQuickEvalAPIView(generics.GenericAPIView):
|
||||||
permission_classes = [IsAuthenticated]
|
permission_classes = [IsAuthenticated, IsAdminRole]
|
||||||
|
queryset = QuickEvaluationModel.objects.all()
|
||||||
|
serializer_class = QuickEvaluationModelSerializer
|
||||||
|
|
||||||
def get(self, request):
|
def get(self, request):
|
||||||
if request.user.role != RoleChoice.ADMIN:
|
|
||||||
return Response({'detail': 'Forbidden'}, status=403)
|
|
||||||
quick_eval = QuickEvaluationModel.objects.filter(
|
quick_eval = QuickEvaluationModel.objects.filter(
|
||||||
created_by=self.request.user
|
created_by=self.request.user
|
||||||
).select_related('created_by').distinct()
|
).select_related('created_by').distinct()
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
# rest framework
|
# rest framework
|
||||||
from rest_framework.response import Response
|
from rest_framework.response import Response
|
||||||
from rest_framework import status
|
from rest_framework import status
|
||||||
from rest_framework.permissions import IsAuthenticated
|
from rest_framework.permissions import BasePermission
|
||||||
from rest_framework.generics import GenericAPIView
|
from rest_framework.generics import GenericAPIView
|
||||||
|
|
||||||
# swagger
|
# swagger
|
||||||
@@ -10,11 +10,18 @@ from drf_spectacular.utils import extend_schema
|
|||||||
# core apps
|
# core apps
|
||||||
from core.services.tech_passport import TechPassportService
|
from core.services.tech_passport import TechPassportService
|
||||||
from core.apps.evaluation.serializers import TechPassportSerializer
|
from core.apps.evaluation.serializers import TechPassportSerializer
|
||||||
|
from core.apps.accounts.choices import RoleChoice
|
||||||
|
|
||||||
|
|
||||||
|
class IsNotUserRole(BasePermission):
|
||||||
|
def has_permission(self, request, view):
|
||||||
|
if not request.user or not request.user.is_authenticated:
|
||||||
|
return False
|
||||||
|
return request.user.role != RoleChoice.USER
|
||||||
|
|
||||||
|
|
||||||
class TechPassportAPIView(GenericAPIView):
|
class TechPassportAPIView(GenericAPIView):
|
||||||
authentication_classes = []
|
permission_classes = [IsNotUserRole]
|
||||||
permission_classes = [IsAuthenticated]
|
|
||||||
|
|
||||||
@extend_schema(
|
@extend_schema(
|
||||||
tags=["Tech Passport"],
|
tags=["Tech Passport"],
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
0
core/apps/tasks/__init__.py
Normal file
0
core/apps/tasks/__init__.py
Normal file
4
core/apps/tasks/admin/__init__.py
Normal file
4
core/apps/tasks/admin/__init__.py
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
from .column import *
|
||||||
|
from .comment import *
|
||||||
|
from .task import *
|
||||||
|
from .label import *
|
||||||
7
core/apps/tasks/admin/column.py
Normal file
7
core/apps/tasks/admin/column.py
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
from django.contrib import admin
|
||||||
|
|
||||||
|
from core.apps.tasks.models import Column
|
||||||
|
|
||||||
|
@admin.register(Column)
|
||||||
|
class ColumnAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ('name',)
|
||||||
7
core/apps/tasks/admin/comment.py
Normal file
7
core/apps/tasks/admin/comment.py
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
from django.contrib import admin
|
||||||
|
|
||||||
|
from core.apps.tasks.models import Comment
|
||||||
|
|
||||||
|
@admin.register(Comment)
|
||||||
|
class CommentAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ('created_by', 'type')
|
||||||
7
core/apps/tasks/admin/label.py
Normal file
7
core/apps/tasks/admin/label.py
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
from django.contrib import admin
|
||||||
|
|
||||||
|
from core.apps.tasks.models import Label
|
||||||
|
|
||||||
|
@admin.register(Label)
|
||||||
|
class LabelAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ('name',)
|
||||||
7
core/apps/tasks/admin/task.py
Normal file
7
core/apps/tasks/admin/task.py
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
from django.contrib import admin
|
||||||
|
|
||||||
|
from core.apps.tasks.models import Task
|
||||||
|
|
||||||
|
@admin.register(Task)
|
||||||
|
class TaskAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ('name', 'created_by', 'priority')
|
||||||
8
core/apps/tasks/apps.py
Normal file
8
core/apps/tasks/apps.py
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
from django.apps import AppConfig
|
||||||
|
|
||||||
|
|
||||||
|
class TasksConfig(AppConfig):
|
||||||
|
name = "core.apps.tasks"
|
||||||
|
|
||||||
|
def ready(self):
|
||||||
|
from core.apps.tasks import admin
|
||||||
6
core/apps/tasks/choices/comment.py
Normal file
6
core/apps/tasks/choices/comment.py
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
from django.db import models
|
||||||
|
|
||||||
|
|
||||||
|
class MessageChoice(models.TextChoices):
|
||||||
|
FILE = "file", "File"
|
||||||
|
TEXT = "text", "Text"
|
||||||
7
core/apps/tasks/choices/task.py
Normal file
7
core/apps/tasks/choices/task.py
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
from django.db import models
|
||||||
|
|
||||||
|
|
||||||
|
class PriorityChoice(models.TextChoices):
|
||||||
|
LOW = "low", "Low"
|
||||||
|
MEDIUM = "medium", "Medium"
|
||||||
|
HIGH = "high", "High"
|
||||||
77
core/apps/tasks/migrations/0001_initial.py
Normal file
77
core/apps/tasks/migrations/0001_initial.py
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-29 13:18
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.conf import settings
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
initial = True
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Column',
|
||||||
|
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)),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'abstract': False,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Label',
|
||||||
|
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)),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'abstract': False,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Task',
|
||||||
|
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)),
|
||||||
|
('description', models.TextField(blank=True)),
|
||||||
|
('priority', models.CharField(choices=[('low', 'Low'), ('medium', 'Medium'), ('high', 'High')], max_length=255)),
|
||||||
|
('from_date', models.DateField(blank=True, null=True)),
|
||||||
|
('to_date', models.DateField(blank=True, null=True)),
|
||||||
|
('assignees', models.ManyToManyField(related_name='assigned_tasks', to=settings.AUTH_USER_MODEL)),
|
||||||
|
('column', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='tasks', to='tasks.column')),
|
||||||
|
('created_by', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='created_tasks', to=settings.AUTH_USER_MODEL)),
|
||||||
|
('labels', models.ManyToManyField(related_name='tasks', to='tasks.label')),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'abstract': False,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Comment',
|
||||||
|
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)),
|
||||||
|
('message', models.TextField()),
|
||||||
|
('file', models.FileField(blank=True, null=True, upload_to='comments/')),
|
||||||
|
('type', models.CharField(choices=[('file', 'File'), ('text', 'Text')], max_length=255)),
|
||||||
|
('created_by', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='created_comments', to=settings.AUTH_USER_MODEL)),
|
||||||
|
('task', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='comments', to='tasks.task')),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'abstract': False,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
||||||
21
core/apps/tasks/migrations/0002_alter_comment_created_by.py
Normal file
21
core/apps/tasks/migrations/0002_alter_comment_created_by.py
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-29 13:20
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.conf import settings
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('tasks', '0001_initial'),
|
||||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='comment',
|
||||||
|
name='created_by',
|
||||||
|
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='comments', to=settings.AUTH_USER_MODEL),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,29 @@
|
|||||||
|
import django.db.models.deletion
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('tasks', '0002_alter_comment_created_by'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='comment',
|
||||||
|
name='file',
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='CommentFile',
|
||||||
|
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)),
|
||||||
|
('file', models.FileField(upload_to='comments/')),
|
||||||
|
('comment', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='files', to='tasks.comment')),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'abstract': False,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
||||||
0
core/apps/tasks/migrations/__init__.py
Normal file
0
core/apps/tasks/migrations/__init__.py
Normal file
4
core/apps/tasks/models/__init__.py
Normal file
4
core/apps/tasks/models/__init__.py
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
from .column import *
|
||||||
|
from .comment import *
|
||||||
|
from .task import *
|
||||||
|
from .label import *
|
||||||
10
core/apps/tasks/models/column.py
Normal file
10
core/apps/tasks/models/column.py
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
from django.db import models
|
||||||
|
|
||||||
|
from django_core.models import AbstractBaseModel
|
||||||
|
|
||||||
|
|
||||||
|
class Column(AbstractBaseModel):
|
||||||
|
name = models.CharField(max_length=255)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.name
|
||||||
25
core/apps/tasks/models/comment.py
Normal file
25
core/apps/tasks/models/comment.py
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
from django.db import models
|
||||||
|
|
||||||
|
from django_core.models import AbstractBaseModel
|
||||||
|
|
||||||
|
from core.apps.tasks.choices.comment import MessageChoice
|
||||||
|
|
||||||
|
|
||||||
|
class Comment(AbstractBaseModel):
|
||||||
|
task = models.ForeignKey('tasks.Task', on_delete=models.CASCADE, related_name='comments')
|
||||||
|
message = models.TextField()
|
||||||
|
type = models.CharField(max_length=255, choices=MessageChoice.choices)
|
||||||
|
created_by = models.ForeignKey('accounts.User', on_delete=models.CASCADE, related_name='comments')
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"{self.message} created by {self.created_by}"
|
||||||
|
|
||||||
|
|
||||||
|
class CommentFile(AbstractBaseModel):
|
||||||
|
comment = models.ForeignKey(
|
||||||
|
'tasks.Comment', on_delete=models.CASCADE, related_name='files'
|
||||||
|
)
|
||||||
|
file = models.FileField(upload_to='comments/')
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"File for comment {self.comment_id}"
|
||||||
10
core/apps/tasks/models/label.py
Normal file
10
core/apps/tasks/models/label.py
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
from django.db import models
|
||||||
|
|
||||||
|
from django_core.models import AbstractBaseModel
|
||||||
|
|
||||||
|
|
||||||
|
class Label(AbstractBaseModel):
|
||||||
|
name = models.CharField(max_length=255)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.name
|
||||||
20
core/apps/tasks/models/task.py
Normal file
20
core/apps/tasks/models/task.py
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
from django.db import models
|
||||||
|
|
||||||
|
from django_core.models import AbstractBaseModel
|
||||||
|
|
||||||
|
from core.apps.tasks.choices.task import PriorityChoice
|
||||||
|
|
||||||
|
|
||||||
|
class Task(AbstractBaseModel):
|
||||||
|
column = models.ForeignKey('tasks.Column', on_delete=models.CASCADE, related_name='tasks')
|
||||||
|
name = models.CharField(max_length=255)
|
||||||
|
description = models.TextField(blank=True)
|
||||||
|
priority = models.CharField(max_length=255, choices=PriorityChoice.choices)
|
||||||
|
from_date = models.DateField(null=True, blank=True)
|
||||||
|
to_date = models.DateField(null=True, blank=True)
|
||||||
|
labels = models.ManyToManyField('tasks.Label', related_name='tasks')
|
||||||
|
assignees = models.ManyToManyField('accounts.User', related_name='assigned_tasks')
|
||||||
|
created_by = models.ForeignKey('accounts.User', on_delete=models.CASCADE, related_name='created_tasks')
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"{self.name} created by {self.created_by}"
|
||||||
22
core/apps/tasks/serializers/board.py
Normal file
22
core/apps/tasks/serializers/board.py
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from core.apps.tasks.serializers.comment import CommentSerializer
|
||||||
|
from core.apps.tasks.serializers.task import TaskSerializer
|
||||||
|
from core.apps.tasks.models import Column
|
||||||
|
|
||||||
|
|
||||||
|
class BoardTaskSerializer(TaskSerializer):
|
||||||
|
comments = CommentSerializer(many=True, read_only=True)
|
||||||
|
|
||||||
|
class Meta(TaskSerializer.Meta):
|
||||||
|
TaskSerializer.Meta.fields += ['comments']
|
||||||
|
|
||||||
|
|
||||||
|
class BoardSerializer(serializers.ModelSerializer):
|
||||||
|
tasks = BoardTaskSerializer(many=True, read_only=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = Column
|
||||||
|
fields = [
|
||||||
|
'id', 'name', 'tasks',
|
||||||
|
]
|
||||||
11
core/apps/tasks/serializers/column.py
Normal file
11
core/apps/tasks/serializers/column.py
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from core.apps.tasks.models.column import Column
|
||||||
|
|
||||||
|
|
||||||
|
class ColumnSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = Column
|
||||||
|
fields = [
|
||||||
|
'id', 'name'
|
||||||
|
]
|
||||||
89
core/apps/tasks/serializers/comment.py
Normal file
89
core/apps/tasks/serializers/comment.py
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
from django.db import transaction
|
||||||
|
|
||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from core.apps.tasks.models.comment import Comment, CommentFile
|
||||||
|
|
||||||
|
|
||||||
|
class CommentCreatedBySerializer(serializers.Serializer):
|
||||||
|
id = serializers.IntegerField()
|
||||||
|
first_name = serializers.CharField()
|
||||||
|
last_name = serializers.CharField()
|
||||||
|
avatar = serializers.SerializerMethodField()
|
||||||
|
|
||||||
|
def get_avatar(self, obj):
|
||||||
|
request = self.context.get('request')
|
||||||
|
if obj.avatar and request:
|
||||||
|
return request.build_absolute_uri(obj.avatar.url)
|
||||||
|
if obj.avatar:
|
||||||
|
return obj.avatar.url
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
class CommentFileSerializer(serializers.ModelSerializer):
|
||||||
|
file = serializers.SerializerMethodField()
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = CommentFile
|
||||||
|
fields = ['id', 'file']
|
||||||
|
|
||||||
|
def get_file(self, obj):
|
||||||
|
request = self.context.get('request')
|
||||||
|
if not obj.file:
|
||||||
|
return None
|
||||||
|
if request:
|
||||||
|
return request.build_absolute_uri(obj.file.url)
|
||||||
|
return obj.file.url
|
||||||
|
|
||||||
|
|
||||||
|
class CommentSerializer(serializers.ModelSerializer):
|
||||||
|
created_by = CommentCreatedBySerializer(read_only=True)
|
||||||
|
files = CommentFileSerializer(many=True, read_only=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = Comment
|
||||||
|
fields = [
|
||||||
|
'id', 'message', 'type', 'created_by', 'created_at', 'files'
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class CommentCreateSerializer(serializers.ModelSerializer):
|
||||||
|
files = serializers.ListField(
|
||||||
|
child=serializers.FileField(),
|
||||||
|
required=False,
|
||||||
|
write_only=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = Comment
|
||||||
|
fields = [
|
||||||
|
'id', 'message', 'type', 'task', 'files', 'created_at'
|
||||||
|
]
|
||||||
|
read_only_fields = ['id', 'created_at']
|
||||||
|
|
||||||
|
def create(self, validated_data):
|
||||||
|
files = validated_data.pop('files', [])
|
||||||
|
with transaction.atomic():
|
||||||
|
comment = Comment.objects.create(
|
||||||
|
created_by=self.context['request'].user,
|
||||||
|
**validated_data
|
||||||
|
)
|
||||||
|
CommentFile.objects.bulk_create([
|
||||||
|
CommentFile(comment=comment, file=f) for f in files
|
||||||
|
])
|
||||||
|
return comment
|
||||||
|
|
||||||
|
def update(self, instance, validated_data):
|
||||||
|
files = validated_data.pop('files', None)
|
||||||
|
with transaction.atomic():
|
||||||
|
for attr, value in validated_data.items():
|
||||||
|
setattr(instance, attr, value)
|
||||||
|
instance.save()
|
||||||
|
if files is not None:
|
||||||
|
CommentFile.objects.bulk_create([
|
||||||
|
CommentFile(comment=instance, file=f) for f in files
|
||||||
|
])
|
||||||
|
return instance
|
||||||
|
|
||||||
|
def to_representation(self, instance):
|
||||||
|
return CommentSerializer(instance, context=self.context).data
|
||||||
11
core/apps/tasks/serializers/label.py
Normal file
11
core/apps/tasks/serializers/label.py
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from core.apps.tasks.models.label import Label
|
||||||
|
|
||||||
|
|
||||||
|
class LabelSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = Label
|
||||||
|
fields = [
|
||||||
|
'id', 'name'
|
||||||
|
]
|
||||||
52
core/apps/tasks/serializers/task.py
Normal file
52
core/apps/tasks/serializers/task.py
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from core.apps.tasks.models.task import Task
|
||||||
|
from core.apps.accounts.serializers.user import ShortUserSerializer
|
||||||
|
from core.apps.tasks.serializers.label import LabelSerializer
|
||||||
|
|
||||||
|
|
||||||
|
class TaskSerializer(serializers.ModelSerializer):
|
||||||
|
labels = LabelSerializer(many=True)
|
||||||
|
assignees = serializers.SerializerMethodField(method_name='get_assignees')
|
||||||
|
created_by = serializers.SerializerMethodField(method_name='get_created_by')
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = Task
|
||||||
|
fields = [
|
||||||
|
'id',
|
||||||
|
'column',
|
||||||
|
'name',
|
||||||
|
'description',
|
||||||
|
'priority',
|
||||||
|
'from_date',
|
||||||
|
'to_date',
|
||||||
|
'labels',
|
||||||
|
'assignees',
|
||||||
|
'created_by'
|
||||||
|
]
|
||||||
|
|
||||||
|
def get_assignees(self, obj):
|
||||||
|
return ShortUserSerializer(obj.assignees.all(), many=True, context={"request": self.context['request']}).data
|
||||||
|
|
||||||
|
def get_created_by(self, obj):
|
||||||
|
return ShortUserSerializer(obj.created_by, context={"request": self.context['request']}).data
|
||||||
|
|
||||||
|
|
||||||
|
class TaskCreateSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = Task
|
||||||
|
fields = [
|
||||||
|
'id',
|
||||||
|
'column',
|
||||||
|
'name',
|
||||||
|
'description',
|
||||||
|
'priority',
|
||||||
|
'from_date',
|
||||||
|
'to_date',
|
||||||
|
'labels',
|
||||||
|
'assignees',
|
||||||
|
]
|
||||||
|
|
||||||
|
def create(self, validated_data):
|
||||||
|
validated_data['created_by'] = self.context['request'].user
|
||||||
|
return super().create(validated_data)
|
||||||
34
core/apps/tasks/urls.py
Normal file
34
core/apps/tasks/urls.py
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
from django.urls import path, include
|
||||||
|
|
||||||
|
from core.apps.tasks.views import task, column, comment, label, board
|
||||||
|
|
||||||
|
urlpatterns = [
|
||||||
|
path('column/', include(
|
||||||
|
[
|
||||||
|
path('list/', column.ColumnListApiView.as_view()),
|
||||||
|
path('create/', column.ColumnCreateApiView.as_view()),
|
||||||
|
path('<int:id>/update/', column.ColumnUpdateApiView.as_view()),
|
||||||
|
path('<int:id>/delete/', column.ColumnDeleteApiView.as_view())
|
||||||
|
]
|
||||||
|
)),
|
||||||
|
path('label/', include(
|
||||||
|
[
|
||||||
|
path('', label.LabelListCreateApiView.as_view()),
|
||||||
|
path('<int:id>/', label.LabelRetrieveUpdateDestroyApiView.as_view()),
|
||||||
|
]
|
||||||
|
)),
|
||||||
|
path('task/', include(
|
||||||
|
[
|
||||||
|
path('list/', task.TaskListView.as_view()),
|
||||||
|
path('<int:pk>/', task.TaskDetailView.as_view()),
|
||||||
|
path('create/', task.TaskCreateView.as_view()),
|
||||||
|
]
|
||||||
|
)),
|
||||||
|
path('comment/', include(
|
||||||
|
[
|
||||||
|
path('', comment.CommentListCreateAPIView.as_view()),
|
||||||
|
path('<int:pk>/', comment.CommentDetailAPIView.as_view()),
|
||||||
|
]
|
||||||
|
)),
|
||||||
|
path('board/', board.BoardListView.as_view()),
|
||||||
|
]
|
||||||
15
core/apps/tasks/views/board.py
Normal file
15
core/apps/tasks/views/board.py
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
from rest_framework import generics, permissions
|
||||||
|
from rest_framework.response import Response
|
||||||
|
|
||||||
|
from core.apps.tasks.serializers.board import BoardSerializer
|
||||||
|
from core.apps.tasks.models import Column
|
||||||
|
|
||||||
|
|
||||||
|
#test commit
|
||||||
|
|
||||||
|
class BoardListView(generics.ListAPIView):
|
||||||
|
queryset = Column.objects.order_by('id')
|
||||||
|
serializer_class = BoardSerializer
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
|
||||||
|
|
||||||
47
core/apps/tasks/views/column.py
Normal file
47
core/apps/tasks/views/column.py
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
from django.db import transaction
|
||||||
|
|
||||||
|
from rest_framework import generics, permissions
|
||||||
|
from rest_framework.response import Response
|
||||||
|
|
||||||
|
from drf_spectacular.utils import extend_schema
|
||||||
|
|
||||||
|
from core.apps.tasks.serializers.column import ColumnSerializer
|
||||||
|
from core.apps.tasks.models.column import Column
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=['Tasks'])
|
||||||
|
class ColumnCreateApiView(generics.GenericAPIView):
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
serializer_class = ColumnSerializer
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
|
def post(self, request):
|
||||||
|
serializer = self.get_serializer(data=request.data)
|
||||||
|
serializer.is_valid(raise_exception=True)
|
||||||
|
column = serializer.save()
|
||||||
|
return Response(serializer.data)
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=['Tasks'])
|
||||||
|
class ColumnListApiView(generics.ListAPIView):
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
serializer_class = ColumnSerializer
|
||||||
|
|
||||||
|
def get_queryset(self):
|
||||||
|
return Column.objects.order_by('id')
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=['Tasks'])
|
||||||
|
class ColumnUpdateApiView(generics.UpdateAPIView):
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
serializer_class = ColumnSerializer
|
||||||
|
lookup_field = 'id'
|
||||||
|
queryset = Column.objects.all()
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=['Tasks'])
|
||||||
|
class ColumnDeleteApiView(generics.DestroyAPIView):
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
serializer_class = ColumnSerializer
|
||||||
|
lookup_field = 'id'
|
||||||
|
queryset = Column.objects.all()
|
||||||
51
core/apps/tasks/views/comment.py
Normal file
51
core/apps/tasks/views/comment.py
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
from rest_framework import generics, permissions
|
||||||
|
from rest_framework.exceptions import PermissionDenied
|
||||||
|
|
||||||
|
from core.apps.tasks.models.comment import Comment
|
||||||
|
from core.apps.tasks.serializers.comment import CommentSerializer, CommentCreateSerializer
|
||||||
|
|
||||||
|
|
||||||
|
class CommentListCreateAPIView(generics.ListCreateAPIView):
|
||||||
|
queryset = Comment.objects.all()
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
|
||||||
|
def get_serializer_class(self):
|
||||||
|
if self.request.method == 'POST':
|
||||||
|
return CommentCreateSerializer
|
||||||
|
return CommentSerializer
|
||||||
|
|
||||||
|
def get_queryset(self):
|
||||||
|
task_id = self.request.query_params.get('task_id')
|
||||||
|
queryset = self.queryset
|
||||||
|
|
||||||
|
if task_id:
|
||||||
|
queryset = queryset.filter(task_id=task_id)
|
||||||
|
|
||||||
|
return queryset.order_by('-id')
|
||||||
|
|
||||||
|
def get_serializer_context(self):
|
||||||
|
return {"request": self.request}
|
||||||
|
|
||||||
|
|
||||||
|
class CommentDetailAPIView(generics.RetrieveUpdateDestroyAPIView):
|
||||||
|
queryset = Comment.objects.all()
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
|
||||||
|
def get_serializer_class(self):
|
||||||
|
if self.request.method in ['PUT', 'PATCH']:
|
||||||
|
return CommentCreateSerializer
|
||||||
|
return CommentSerializer
|
||||||
|
|
||||||
|
def get_serializer_context(self):
|
||||||
|
return {"request": self.request}
|
||||||
|
|
||||||
|
def perform_update(self, serializer):
|
||||||
|
comment = self.get_object()
|
||||||
|
if comment.created_by != self.request.user:
|
||||||
|
raise PermissionDenied("You cannot edit this comment")
|
||||||
|
serializer.save()
|
||||||
|
|
||||||
|
def perform_destroy(self, instance):
|
||||||
|
if instance.created_by != self.request.user:
|
||||||
|
raise PermissionDenied("You cannot delete this comment")
|
||||||
|
instance.delete()
|
||||||
22
core/apps/tasks/views/label.py
Normal file
22
core/apps/tasks/views/label.py
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
from rest_framework import generics, permissions
|
||||||
|
from rest_framework.response import Response
|
||||||
|
|
||||||
|
from drf_spectacular.utils import extend_schema
|
||||||
|
|
||||||
|
from core.apps.tasks.serializers.label import LabelSerializer
|
||||||
|
from core.apps.tasks.models.label import Label
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=['Tasks'])
|
||||||
|
class LabelListCreateApiView(generics.ListCreateAPIView):
|
||||||
|
queryset = Label.objects.order_by('id')
|
||||||
|
serializer_class = LabelSerializer
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=['Tasks'])
|
||||||
|
class LabelRetrieveUpdateDestroyApiView(generics.RetrieveUpdateDestroyAPIView):
|
||||||
|
queryset = Label.objects.order_by('id')
|
||||||
|
serializer_class = LabelSerializer
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
lookup_field = 'id'
|
||||||
39
core/apps/tasks/views/task.py
Normal file
39
core/apps/tasks/views/task.py
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
from django.db import transaction
|
||||||
|
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.serializers.task import TaskSerializer, TaskCreateSerializer
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=['Tasks'])
|
||||||
|
class TaskCreateView(generics.GenericAPIView):
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
serializer_class = TaskCreateSerializer
|
||||||
|
queryset = Task.objects.order_by('id')
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
|
def post(self, request):
|
||||||
|
serializer = self.get_serializer(data=request.data)
|
||||||
|
if not serializer.is_valid(raise_exception=True):
|
||||||
|
return Response(serializer.validated_data, status=status.HTTP_400_BAD_REQUEST)
|
||||||
|
serializer.save()
|
||||||
|
return Response(serializer.data)
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=['Tasks'])
|
||||||
|
class TaskListView(generics.ListAPIView):
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
serializer_class = TaskSerializer
|
||||||
|
queryset = Task.objects.order_by('id')
|
||||||
|
|
||||||
|
def serializer_context(self):
|
||||||
|
return self.serializer_class(context={"request": self.request})
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=['Tasks'])
|
||||||
|
class TaskDetailView(generics.RetrieveUpdateDestroyAPIView):
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
serializer_class = TaskSerializer
|
||||||
|
queryset = Task.objects.order_by('id')
|
||||||
138
core/utils/generation_pdf.py
Normal file
138
core/utils/generation_pdf.py
Normal 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
|
||||||
@@ -5,6 +5,17 @@ ENV SCRIPT=$SCRIPT
|
|||||||
|
|
||||||
WORKDIR /code
|
WORKDIR /code
|
||||||
|
|
||||||
|
RUN apk add --no-cache \
|
||||||
|
glib \
|
||||||
|
gdk-pixbuf \
|
||||||
|
pango \
|
||||||
|
cairo \
|
||||||
|
libffi \
|
||||||
|
shared-mime-info \
|
||||||
|
fontconfig \
|
||||||
|
ttf-dejavu \
|
||||||
|
ttf-liberation
|
||||||
|
|
||||||
COPY requirements.txt /code/requirements.txt
|
COPY requirements.txt /code/requirements.txt
|
||||||
|
|
||||||
RUN uv pip install -r requirements.txt
|
RUN uv pip install -r requirements.txt
|
||||||
@@ -16,5 +27,3 @@ COPY ./resources/scripts/$SCRIPT /code/$SCRIPT
|
|||||||
RUN chmod +x /code/resources/scripts/$SCRIPT
|
RUN chmod +x /code/resources/scripts/$SCRIPT
|
||||||
|
|
||||||
CMD sh /code/resources/scripts/$SCRIPT
|
CMD sh /code/resources/scripts/$SCRIPT
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
184
requirements.txt
184
requirements.txt
@@ -1,46 +1,146 @@
|
|||||||
backports.tarfile==1.2.0
|
aiohappyeyeballs
|
||||||
celery==5.4.0
|
aiohttp
|
||||||
django-cors-headers==4.6.0
|
aiosignal
|
||||||
django-environ==0.11.2
|
amqp
|
||||||
django-extensions==3.2.3
|
annotated-doc
|
||||||
django-filter==24.3
|
annotated-types
|
||||||
django-redis==5.4.0
|
arrow
|
||||||
django-unfold==0.65.0
|
asgiref
|
||||||
djangorestframework-simplejwt==5.3.1
|
astor
|
||||||
drf-spectacular==0.28.0
|
attrs
|
||||||
importlib-metadata==8.5.0
|
backports.tarfile
|
||||||
importlib-resources==6.4.5
|
|
||||||
inflect==7.3.1
|
|
||||||
jaraco.collections==5.1.0
|
|
||||||
packaging==24.2
|
|
||||||
pip-chill==1.0.3
|
|
||||||
platformdirs==4.3.6
|
|
||||||
psycopg2-binary==2.9.10
|
|
||||||
tomli==2.2.1
|
|
||||||
uvicorn==0.32.1
|
|
||||||
jst-django-core~=1.2.2
|
|
||||||
rich
|
|
||||||
pydantic
|
|
||||||
bcrypt
|
bcrypt
|
||||||
pytest-django
|
billiard
|
||||||
requests
|
binaryornot
|
||||||
model_bakery
|
black
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
django-modeltranslation~=0.19.11
|
|
||||||
django-ckeditor-5==0.2.15
|
|
||||||
|
|
||||||
django-rosetta==0.10.1
|
|
||||||
django-cacheops~=7.1
|
|
||||||
|
|
||||||
|
|
||||||
# !NOTE: on-server
|
|
||||||
# gunicorn
|
|
||||||
|
|
||||||
|
|
||||||
django-storages
|
|
||||||
boto3
|
boto3
|
||||||
|
botocore
|
||||||
|
brotli
|
||||||
|
celery
|
||||||
|
certifi
|
||||||
|
cffi
|
||||||
|
chardet
|
||||||
|
charset-normalizer
|
||||||
|
click
|
||||||
|
click-didyoumean
|
||||||
|
click-plugins
|
||||||
|
click-repl
|
||||||
|
colorlog
|
||||||
|
cookiecutter
|
||||||
|
cssselect2
|
||||||
|
Django
|
||||||
|
django-cacheops
|
||||||
|
django-ckeditor-5
|
||||||
|
django-cors-headers
|
||||||
|
django-environ
|
||||||
|
django-extensions
|
||||||
|
django-filter
|
||||||
|
django-modeltranslation
|
||||||
|
django-redis
|
||||||
|
django-rosetta
|
||||||
|
django-storages
|
||||||
|
django-unfold
|
||||||
|
djangorestframework
|
||||||
|
djangorestframework-simplejwt
|
||||||
|
drf-spectacular
|
||||||
|
flake8
|
||||||
|
fonttools
|
||||||
|
frozenlist
|
||||||
|
funcy
|
||||||
|
g4f
|
||||||
|
grpcio
|
||||||
|
grpcio-tools
|
||||||
|
h11
|
||||||
|
idna
|
||||||
|
importlib_metadata
|
||||||
|
importlib_resources
|
||||||
|
inflect
|
||||||
|
inflection
|
||||||
|
iniconfig
|
||||||
|
isort
|
||||||
|
jaraco.collections
|
||||||
|
jaraco.context
|
||||||
|
jaraco.functools
|
||||||
|
jaraco.text
|
||||||
|
Jinja2
|
||||||
|
jmespath
|
||||||
|
jsonschema
|
||||||
|
jsonschema-specifications
|
||||||
|
jst-aicommit
|
||||||
|
jst-django
|
||||||
|
jst-django-core
|
||||||
|
kombu
|
||||||
|
markdown-it-py
|
||||||
|
MarkupSafe
|
||||||
|
mccabe
|
||||||
|
mdurl
|
||||||
|
model-bakery
|
||||||
|
more-itertools
|
||||||
|
multidict
|
||||||
|
mypy-extensions
|
||||||
|
nest-asyncio
|
||||||
|
packaging
|
||||||
|
pathspec
|
||||||
|
pillow
|
||||||
|
pip-chill
|
||||||
|
platformdirs
|
||||||
|
pluggy
|
||||||
|
polib
|
||||||
|
prompt-toolkit
|
||||||
|
propcache
|
||||||
|
protobuf
|
||||||
|
psycopg2-binary
|
||||||
|
pycodestyle
|
||||||
|
pycparser
|
||||||
|
pycryptodome
|
||||||
|
pydantic
|
||||||
|
pydantic_core
|
||||||
|
pydyf
|
||||||
|
pyflakes
|
||||||
|
Pygments
|
||||||
|
PyJWT
|
||||||
|
pyphen
|
||||||
|
pytest
|
||||||
|
pytest-django
|
||||||
|
python-dateutil
|
||||||
|
python-slugify
|
||||||
|
PyYAML
|
||||||
|
questionary
|
||||||
|
redis
|
||||||
|
referencing
|
||||||
|
reportlab
|
||||||
|
requests
|
||||||
|
rich
|
||||||
|
rpds-py
|
||||||
|
s3transfer
|
||||||
|
setuptools
|
||||||
|
shellingham
|
||||||
|
six
|
||||||
|
sqlparse
|
||||||
|
tenacity
|
||||||
|
text-unidecode
|
||||||
|
tinycss2
|
||||||
|
tinyhtml5
|
||||||
|
tomli
|
||||||
|
tqdm
|
||||||
|
typeguard
|
||||||
|
typer
|
||||||
|
typer-slim
|
||||||
|
types-python-dateutil
|
||||||
|
typing-inspection
|
||||||
|
typing_extensions
|
||||||
|
tzdata
|
||||||
|
uritemplate
|
||||||
|
urllib3
|
||||||
|
uvicorn
|
||||||
|
vine
|
||||||
|
wcwidth
|
||||||
|
weasyprint
|
||||||
|
webencodings
|
||||||
|
yarl
|
||||||
|
zipfile36
|
||||||
|
zipp
|
||||||
|
zopfli
|
||||||
|
|
||||||
|
|
||||||
# !NOTE: on-websocket
|
# !NOTE: on-websocket
|
||||||
@@ -50,3 +150,5 @@ 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
|
||||||
|
|||||||
2417
resources/templates/documents/contract.html
Normal file
2417
resources/templates/documents/contract.html
Normal file
File diff suppressed because it is too large
Load Diff
@@ -84,7 +84,7 @@ services:
|
|||||||
max-file: "5"
|
max-file: "5"
|
||||||
|
|
||||||
web:
|
web:
|
||||||
image: husanjon/sifatbaho:144
|
image: husanjon/sifatbaho:154
|
||||||
env_file:
|
env_file:
|
||||||
- .env
|
- .env
|
||||||
environment:
|
environment:
|
||||||
@@ -129,7 +129,7 @@ services:
|
|||||||
max-file: "5"
|
max-file: "5"
|
||||||
|
|
||||||
celery:
|
celery:
|
||||||
image: husanjon/sifatbaho:144
|
image: husanjon/sifatbaho:154
|
||||||
env_file:
|
env_file:
|
||||||
- .env
|
- .env
|
||||||
environment:
|
environment:
|
||||||
|
|||||||
41
task.txt
Normal file
41
task.txt
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
object_type => Bu hozirda select. O’shanga value qo’shish kerak: bus, moto
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
car_position => Qo’shish kerak. Bu hozirda select. /api/v1/reference-item/ api’dan value yuboraman
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
distance_covered => Qo’shish kerak. Number. Bosib o’tilgan masofasi
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
body_type => Qo’shish kerak. Bu hozirda select. /api/v1/reference-item/ api’dan value yuboraman
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
fuel_type => Qo’shish kerak. Bu hozirda select. /api/v1/reference-item/ api’dan value yuboraman
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
state_car => Qo’shish kerak. Bu hozirda select. /api/v1/reference-item/ api’dan value yuboraman
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
car_type => manashu field ni olib tashlash kerak
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
tex_passport_file => multiple qilish kerak
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
assessment_task_type => Baholash vazifasi. Qo’shish kerak. Select bo’ladi. /api/v1/reference-item/ api’dan value yuboraman
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
object_owner_residence => Obyekt egasi yashash joyi. Qo’shish kerak. string bo’ladi.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
manashu fieldlarni detail apidagi serializerga qoshish kerak auto-evaluationda ham va mechnic-auto-evaluationda ham
|
||||||
Reference in New Issue
Block a user