Compare commits
22 Commits
d014f5a2fb
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
9b782fe7bd | ||
| 135f580db2 | |||
|
|
0c622759cc | ||
|
|
1d750b1c1c | ||
| 2b26c52a5c | |||
|
|
51b3535a80 | ||
|
|
9028e2f102 | ||
|
|
0c9c726756 | ||
|
|
c88ea1aa77 | ||
|
|
581021cbb7 | ||
| 62f65385e1 | |||
|
|
76d2fe5090 | ||
| 92d23901a1 | |||
|
|
42987e4154 | ||
| 84b14da3f4 | |||
|
|
1ff23af8bf | ||
|
|
feecb580c1 | ||
|
|
cb53924f9b | ||
|
|
f53125cfdc | ||
| 65ab51e652 | |||
| 2997810fae | |||
|
|
c29546a04b |
@@ -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
|
||||||
|
|||||||
5
.github/workflows/deploy.yaml
vendored
5
.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
|
||||||
|
|
||||||
|
if [ ! -f .env ]; then
|
||||||
cp .env.example .env
|
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
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ from config.env import env
|
|||||||
|
|
||||||
|
|
||||||
def home(request):
|
def home(request):
|
||||||
return HttpResponse("OK: #3781ce29e5447f1473964c4c47fbdef2a38c6751")
|
return HttpResponse("OK: #135f580db2234f2af65e32ac4b2525506a7a033a")
|
||||||
|
|
||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
|
|||||||
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
|
||||||
@@ -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:
|
||||||
|
|||||||
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),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -9,14 +9,11 @@ from core.apps.evaluation.choices.auto import (
|
|||||||
AutoEvaluationStatus,
|
AutoEvaluationStatus,
|
||||||
AutoObjectType,
|
AutoObjectType,
|
||||||
# FormOwnership,
|
# FormOwnership,
|
||||||
LocationConvenience,
|
|
||||||
LocationHighways,
|
|
||||||
ObjectOwnerType,
|
ObjectOwnerType,
|
||||||
# PropertyRights,
|
# PropertyRights,
|
||||||
# RateType,
|
# RateType,
|
||||||
# ValueDetermined,
|
# ValueDetermined,
|
||||||
)
|
)
|
||||||
|
|
||||||
from .valuation import ValuationModel
|
from .valuation import ValuationModel
|
||||||
from .vehicle import VehicleModel
|
from .vehicle import VehicleModel
|
||||||
|
|
||||||
@@ -244,8 +241,6 @@ class AutoEvaluationModel(AbstractBaseModel):
|
|||||||
default=False,
|
default=False,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return f"Auto Evaluation {self.registration_number or self.pk}"
|
return f"Auto Evaluation {self.registration_number or self.pk}"
|
||||||
|
|
||||||
|
|||||||
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")
|
||||||
@@ -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")
|
||||||
|
|
||||||
|
|||||||
@@ -321,6 +321,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 +332,58 @@ 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 = ("tex_passport_file",
|
||||||
|
|
||||||
|
"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",
|
||||||
|
"value_determined",
|
||||||
|
"rate_type",
|
||||||
|
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"tex_passport_gived_date",
|
||||||
|
"tex_passport_gived_location",
|
||||||
|
"car_type",
|
||||||
|
"car_wheel",
|
||||||
|
"car_brand",
|
||||||
|
"car_model",
|
||||||
|
"car_number",
|
||||||
|
"manufacture_year",
|
||||||
|
"car_dvigatel_number",
|
||||||
|
"car_color",
|
||||||
|
|
||||||
|
"rating_goal",
|
||||||
|
"status",
|
||||||
|
"is_archived",
|
||||||
|
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
|
|
||||||
|
read_only_fields = (
|
||||||
|
"id",
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
|
|||||||
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
@@ -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)
|
||||||
|
|||||||
@@ -27,6 +27,9 @@ router.register("valuation", views.ValuationView, basename="valuation")
|
|||||||
router.register("property-owner", views.PropertyOwnerView, basename="property-owner")
|
router.register("property-owner", views.PropertyOwnerView, basename="property-owner")
|
||||||
router.register("customer", views.CustomerView, basename="customer")
|
router.register("customer", views.CustomerView, basename="customer")
|
||||||
router.register("certificate", views.CertificateView, basename="certificate")
|
router.register("certificate", views.CertificateView, basename="certificate")
|
||||||
|
router.register("bonus-type", views.BonusTypeView, basename="bonus-type")
|
||||||
|
router.register("bonus-employee", views.BonusEmployeeViewSet, basename="bonus-employee")
|
||||||
|
router.register("bonus-base", views.BaseBonusViewSet, basename="bonus-base")
|
||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
path("", include(router.urls)),
|
path("", include(router.urls)),
|
||||||
@@ -85,4 +88,5 @@ urlpatterns = [
|
|||||||
)),
|
)),
|
||||||
|
|
||||||
path("calculate_avg_cost/", views.AvgCostAPIView.as_view()),
|
path("calculate_avg_cost/", views.AvgCostAPIView.as_view()),
|
||||||
|
path("vehicle_document/", views.GeneratePDFView.as_view()),
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -15,3 +15,4 @@ from .didox import * # noqa
|
|||||||
from .tech_passport import * # noqa
|
from .tech_passport import * # noqa
|
||||||
from .certificate import * # noqa
|
from .certificate import * # noqa
|
||||||
from .avg_cost import *
|
from .avg_cost import *
|
||||||
|
from .bonus import *
|
||||||
@@ -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,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,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
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ from rest_framework import serializers
|
|||||||
|
|
||||||
from core.apps.tasks.serializers.comment import CommentSerializer
|
from core.apps.tasks.serializers.comment import CommentSerializer
|
||||||
from core.apps.tasks.serializers.task import TaskSerializer
|
from core.apps.tasks.serializers.task import TaskSerializer
|
||||||
from core.apps.tasks.models import Column, Task
|
from core.apps.tasks.models import Column
|
||||||
|
|
||||||
|
|
||||||
class BoardTaskSerializer(TaskSerializer):
|
class BoardTaskSerializer(TaskSerializer):
|
||||||
|
|||||||
@@ -30,15 +30,10 @@ class CommentCreateSerializer(serializers.ModelSerializer):
|
|||||||
'id', 'message', 'file', 'type', 'task'
|
'id', 'message', 'file', 'type', 'task'
|
||||||
]
|
]
|
||||||
|
|
||||||
def validate(self, data):
|
|
||||||
task = Task.objects.filter(id=data['task']).first()
|
|
||||||
if not task:
|
|
||||||
raise serializers.ValidationError("Task not found")
|
|
||||||
data['task'] = task
|
|
||||||
return data
|
|
||||||
|
|
||||||
def create(self, validated_data):
|
def create(self, validated_data):
|
||||||
with transaction.atomic():
|
with transaction.atomic():
|
||||||
task = validated_data.pop('task')
|
comment = Comment.objects.create(
|
||||||
comment = Comment.objects.create(task=task, created_by=self.context['request'].user, **validated_data)
|
created_by=self.context['request'].user,
|
||||||
|
**validated_data
|
||||||
|
)
|
||||||
return comment
|
return comment
|
||||||
@@ -20,7 +20,7 @@ urlpatterns = [
|
|||||||
path('task/', include(
|
path('task/', include(
|
||||||
[
|
[
|
||||||
path('list/', task.TaskListView.as_view()),
|
path('list/', task.TaskListView.as_view()),
|
||||||
path('<int:id>/', task.TaskDetailView.as_view()),
|
path('<int:pk>/', task.TaskDetailView.as_view()),
|
||||||
path('create/', task.TaskCreateView.as_view()),
|
path('create/', task.TaskCreateView.as_view()),
|
||||||
]
|
]
|
||||||
)),
|
)),
|
||||||
|
|||||||
@@ -5,6 +5,8 @@ from core.apps.tasks.serializers.board import BoardSerializer
|
|||||||
from core.apps.tasks.models import Column
|
from core.apps.tasks.models import Column
|
||||||
|
|
||||||
|
|
||||||
|
#test commit
|
||||||
|
|
||||||
class BoardListView(generics.ListAPIView):
|
class BoardListView(generics.ListAPIView):
|
||||||
queryset = Column.objects.order_by('id')
|
queryset = Column.objects.order_by('id')
|
||||||
serializer_class = BoardSerializer
|
serializer_class = BoardSerializer
|
||||||
|
|||||||
@@ -1,9 +1,7 @@
|
|||||||
from django.db import transaction
|
from django.db import transaction
|
||||||
|
|
||||||
from rest_framework import permissions, generics
|
|
||||||
from rest_framework.response import Response
|
|
||||||
|
|
||||||
from drf_spectacular.utils import extend_schema
|
from drf_spectacular.utils import extend_schema
|
||||||
|
from rest_framework import permissions, generics, status
|
||||||
|
from rest_framework.response import Response
|
||||||
|
|
||||||
from core.apps.tasks.models.task import Task
|
from core.apps.tasks.models.task import Task
|
||||||
from core.apps.tasks.serializers.task import TaskSerializer, TaskCreateSerializer
|
from core.apps.tasks.serializers.task import TaskSerializer, TaskCreateSerializer
|
||||||
@@ -18,7 +16,8 @@ class TaskCreateView(generics.GenericAPIView):
|
|||||||
@transaction.atomic
|
@transaction.atomic
|
||||||
def post(self, request):
|
def post(self, request):
|
||||||
serializer = self.get_serializer(data=request.data)
|
serializer = self.get_serializer(data=request.data)
|
||||||
serializer.is_valid(raise_exception=True)
|
if not serializer.is_valid(raise_exception=True):
|
||||||
|
return Response(serializer.validated_data, status=status.HTTP_400_BAD_REQUEST)
|
||||||
serializer.save()
|
serializer.save()
|
||||||
return Response(serializer.data)
|
return Response(serializer.data)
|
||||||
|
|
||||||
|
|||||||
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
|
||||||
@@ -50,3 +50,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
|
||||||
@@ -84,7 +84,7 @@ services:
|
|||||||
max-file: "5"
|
max-file: "5"
|
||||||
|
|
||||||
web:
|
web:
|
||||||
image: husanjon/sifatbaho:145
|
image: husanjon/sifatbaho:152
|
||||||
env_file:
|
env_file:
|
||||||
- .env
|
- .env
|
||||||
environment:
|
environment:
|
||||||
@@ -129,7 +129,7 @@ services:
|
|||||||
max-file: "5"
|
max-file: "5"
|
||||||
|
|
||||||
celery:
|
celery:
|
||||||
image: husanjon/sifatbaho:145
|
image: husanjon/sifatbaho:152
|
||||||
env_file:
|
env_file:
|
||||||
- .env
|
- .env
|
||||||
environment:
|
environment:
|
||||||
|
|||||||
Reference in New Issue
Block a user