finance: fucking income expence

This commit is contained in:
behruz-dev
2025-10-31 22:49:14 +05:00
parent 700a92710f
commit 9d83ccd7ed
5 changed files with 589 additions and 241 deletions

View File

@@ -14,8 +14,29 @@ class ExpenceCreateSerializer(serializers.ModelSerializer):
'counterparty', 'price', 'exchange_rate', 'currency', 'date', 'comment', 'audit', 'file' 'counterparty', 'price', 'exchange_rate', 'currency', 'date', 'comment', 'audit', 'file'
] ]
def validate(self, data):
"""Balanslari tekshirish"""
cash_transaction = data.get('cash_transaction')
payment_type = data.get('payment_type')
price = data.get('price')
exchange_rate = data.get('exchange_rate', 1)
currency = data.get('currency', 'uzs')
final_price = price * exchange_rate if exchange_rate else price
# Payment type balansi tekshiruvi
if currency == 'uzs' and payment_type.total_uzs < final_price:
raise serializers.ValidationError(f"Yetarli UZS balansi yo'q. Mavjud: {payment_type.total_uzs}")
elif currency == 'usd' and payment_type.total_usd < final_price:
raise serializers.ValidationError(f"Yetarli USD balansi yo'q. Mavjud: {payment_type.total_usd}")
return data
def create(self, validated_data): def create(self, validated_data):
with transaction.atomic(): with transaction.atomic():
exchange_rate = validated_data.get('exchange_rate') or 1
final_price = validated_data.get('price') * exchange_rate
expence = Expence.objects.create( expence = Expence.objects.create(
user=self.context.get('user'), user=self.context.get('user'),
cash_transaction=validated_data.get('cash_transaction'), cash_transaction=validated_data.get('cash_transaction'),
@@ -24,61 +45,83 @@ class ExpenceCreateSerializer(serializers.ModelSerializer):
project=validated_data.get('project'), project=validated_data.get('project'),
expence_type=validated_data.get('expence_type'), expence_type=validated_data.get('expence_type'),
counterparty=validated_data.get('counterparty'), counterparty=validated_data.get('counterparty'),
price=validated_data.get('price') * validated_data.get('exchange_rate') if validated_data.get('exchange_rate') else validated_data.get('price'), price=final_price,
exchange_rate=validated_data.get('exchange_rate'), exchange_rate=exchange_rate,
currency=validated_data.get('currency'), currency=validated_data.get('currency'),
date=validated_data.get('date'), date=validated_data.get('date'),
comment=validated_data.get('comment'), comment=validated_data.get('comment'),
audit=validated_data.get('audit'), audit=validated_data.get('audit'),
file=validated_data.get('file'), file=validated_data.get('file'),
) )
cash_transaction = expence.cash_transaction cash_transaction = expence.cash_transaction
payment_type = expence.payment_type payment_type = expence.payment_type
currency = validated_data.get('currency', 'uzs').lower()
user = self.context.get('user') user = self.context.get('user')
if validated_data.get('currency') == 'uzs':
if currency == 'uzs':
# Cash transaction balansini yangilash
cash_transaction.expence_balance_uzs += expence.price cash_transaction.expence_balance_uzs += expence.price
cash_transaction.total_balance_uzs = cash_transaction.income_balance_uzs - cash_transaction.expence_balance_uzs cash_transaction.total_balance_uzs = cash_transaction.income_balance_uzs - cash_transaction.expence_balance_uzs
if payment_type.total_uzs > expence.price:
payment_type.total_uzs -= expence.price # Payment type balansini kamayitish
payment_type.total_uzs -= expence.price
# Kontrapartiya hisobini yangilash
if expence.counterparty: if expence.counterparty:
if expence.counterparty.kredit_uzs != 0: if expence.counterparty.kredit_uzs > 0:
expence.counterparty.kredit_uzs -= expence.price # Kredit bo'lsa, uni qisqartirish
expence.counterparty.kredit_uzs -= expence.price
expence.counterparty.total_kredit -= expence.price expence.counterparty.total_kredit -= expence.price
expence.counterparty.debit_uzs += expence.counterparty.kredit_uzs - expence.price # Debit oshirish
expence.counterparty.total_debit += expence.price
else:
expence.counterparty.debit_uzs += expence.price expence.counterparty.debit_uzs += expence.price
expence.counterparty.total_debit += expence.price expence.counterparty.total_debit += expence.price
else:
# Kredit yo'q bo'lsa, to'g'ridan-to'g'ri debit oshirish
expence.counterparty.debit_uzs += expence.price
expence.counterparty.total_debit += expence.price
expence.counterparty.save() expence.counterparty.save()
elif validated_data.get('currency') == 'usd': elif currency == 'usd':
# Cash transaction balansini yangilash
cash_transaction.expence_balance_usd += expence.price cash_transaction.expence_balance_usd += expence.price
cash_transaction.total_balance_usd = cash_transaction.income_balance_usd - cash_transaction.expence_balance_usd cash_transaction.total_balance_usd = cash_transaction.income_balance_usd - cash_transaction.expence_balance_usd
if payment_type.total_usd > expence.price:
payment_type.total_usd -= expence.price # Payment type balansini kamayitish
payment_type.total_usd -= expence.price
# Kontrapartiya hisobini yangilash
if expence.counterparty: if expence.counterparty:
if expence.counterparty.kredit_usd != 0: if expence.counterparty.kredit_usd > 0:
expence.counterparty.kredit_usd -= validated_data.get('price') # Kredit bo'lsa, uni qisqartirish
expence.counterparty.kredit_usd -= expence.price # ✅ TUZATILDI
expence.counterparty.total_kredit -= expence.price expence.counterparty.total_kredit -= expence.price
expence.counterparty.debit_usd += expence.counterparty.kredit_usd - validated_data.get('price') # Debit oshirish
expence.counterparty.debit_usd += expence.price # ✅ TUZATILDI
expence.counterparty.total_debit += expence.price expence.counterparty.total_debit += expence.price
else: else:
expence.counterparty.debit_usd += validated_data.get('price') # Kredit yo'q bo'lsa, to'g'ridan-to'g'ri debit oshirish
expence.counterparty.debit_usd += expence.price # ✅ TUZATILDI
expence.counterparty.total_debit += expence.price expence.counterparty.total_debit += expence.price
expence.counterparty.save() expence.counterparty.save()
# Bildirishnoma yuborish
body = f"""{user.full_name} {expence.price} {expence.currency.upper()}... \n screen: /monitoring""" body = f"""{user.full_name} {expence.price} {expence.currency.upper()}... \n screen: /monitoring"""
data = { data = {
"screen": "/monitoring", "screen": "/monitoring",
"type": "expence", "type": "expence",
} }
notify_user(user=user, title="Tasdiqlang yoki rad eting", body=body, data=data) notify_user(user=user, title="Tasdiqlang yoki rad eting", body=body, data=data)
cash_transaction.save() cash_transaction.save()
payment_type.save() payment_type.save()
return expence return expence
class ExpenceListSerializer(serializers.ModelSerializer): class ExpenceListSerializer(serializers.ModelSerializer):
@@ -156,40 +199,103 @@ class ExpenceUpdateSerializer(serializers.ModelSerializer):
'price': {'required': False}, 'price': {'required': False},
} }
def validate_price(self, value):
"""Narxi manfiy bo'lmasligi tekshiruvi"""
if value and value < 0:
raise serializers.ValidationError("Narxi manfiy bo'lishi mumkin emas")
return value
def update(self, instance, validated_data): def update(self, instance, validated_data):
old_price = instance.price with transaction.atomic():
instance.project_folder = validated_data.get('project_folder', instance.project_folder) old_price = instance.price
instance.project = validated_data.get('project', instance.project) old_counterparty = instance.counterparty
instance.price = validated_data.get('price', instance.price) new_price = validated_data.get('price', instance.price)
instance.expence_type = validated_data.get('expence_type', instance.expence_type) new_counterparty = validated_data.get('counterparty', instance.counterparty)
instance.counterparty = validated_data.get('counterparty', instance.counterparty) currency = instance.currency.lower()
instance.date = validated_data.get('date', instance.date)
instance.comment = validated_data.get('comment', instance.comment) # Asosiy ma'lumotlarni yangilash
instance.audit = validated_data.get('audit', instance.audit) instance.project_folder = validated_data.get('project_folder', instance.project_folder)
instance.file = validated_data.get('file', instance.file) instance.project = validated_data.get('project', instance.project)
if validated_data.get('price'): instance.price = new_price
cash_transaction = instance.cash_transaction instance.expence_type = validated_data.get('expence_type', instance.expence_type)
payment_type = instance.payment_type instance.counterparty = new_counterparty
if old_price > validated_data.get('price'): instance.date = validated_data.get('date', instance.date)
if instance.currency == 'uzs': instance.comment = validated_data.get('comment', instance.comment)
cash_transaction.expence_balance_uzs -= (old_price - validated_data.get('price')) instance.audit = validated_data.get('audit', instance.audit)
cash_transaction.total_balance_uzs -= (old_price - validated_data.get('price')) instance.file = validated_data.get('file', instance.file)
payment_type.total_uzs -= (old_price - validated_data.get('price')) if payment_type.total_uzs > (old_price - validated_data.get('price')) else 0
else: # Agar narxi o'zgarsa, balanslarni yangilash
cash_transaction.expence_balance_usd -= (old_price - validated_data.get('price')) if validated_data.get('price') and old_price != new_price:
cash_transaction.total_balance_usd -= (old_price - validated_data.get('price')) price_difference = new_price - old_price # ✅ To'g'ri farq
payment_type.total_usd -= (old_price - validated_data.get('price')) if payment_type.total_usd > (old_price - validated_data.get('price')) else 0 cash_transaction = instance.cash_transaction
payment_type = instance.payment_type
elif old_price < validated_data.get('price'):
if instance.currency == 'uzs': if currency == 'uzs':
cash_transaction.expence_balance_uzs += (old_price - validated_data.get('price')) # Cash transaction balansini yangilash
cash_transaction.total_balance_uzs += (old_price - validated_data.get('price')) cash_transaction.expence_balance_uzs += price_difference
payment_type.total_uzs += (old_price - validated_data.get('price')) cash_transaction.total_balance_uzs = (
else: cash_transaction.income_balance_uzs - cash_transaction.expence_balance_uzs
cash_transaction.expence_balance_usd += (old_price - validated_data.get('price')) )
cash_transaction.total_balance_usd += (old_price - validated_data.get('price'))
payment_type.total_usd += (old_price - validated_data.get('price')) # Payment type balansini yangilash
cash_transaction.save() payment_type.total_uzs -= price_difference # ✅ Narx oshsa, balans kamayadi
payment_type.save()
instance.save() elif currency == 'usd':
return instance # Cash transaction balansini yangilash
cash_transaction.expence_balance_usd += price_difference
cash_transaction.total_balance_usd = (
cash_transaction.income_balance_usd - cash_transaction.expence_balance_usd
)
# Payment type balansini yangilash
payment_type.total_usd -= price_difference # ✅ Narx oshsa, balans kamayadi
cash_transaction.save()
payment_type.save()
# Agar kontrapartiya o'zgarsa, uning hisobini yangilash
if new_counterparty != old_counterparty:
# Eski kontrapartyani qaytarish
if old_counterparty:
if currency == 'uzs':
if old_counterparty.debit_uzs > 0:
old_counterparty.debit_uzs -= old_price
old_counterparty.total_debit -= old_price
else:
old_counterparty.kredit_uzs += old_price
old_counterparty.total_kredit += old_price
else:
if old_counterparty.debit_usd > 0:
old_counterparty.debit_usd -= old_price
old_counterparty.total_debit -= old_price
else:
old_counterparty.kredit_usd += old_price
old_counterparty.total_kredit += old_price
old_counterparty.save()
# Yangi kontrapartyani yangilash
if new_counterparty:
if currency == 'uzs':
if new_counterparty.kredit_uzs > 0:
new_counterparty.kredit_uzs -= new_price
new_counterparty.total_kredit -= new_price
new_counterparty.debit_uzs += new_price
new_counterparty.total_debit += new_price
else:
new_counterparty.debit_uzs += new_price
new_counterparty.total_debit += new_price
else:
if new_counterparty.kredit_usd > 0:
new_counterparty.kredit_usd -= new_price
new_counterparty.total_kredit -= new_price
new_counterparty.debit_usd += new_price
new_counterparty.total_debit += new_price
else:
new_counterparty.debit_usd += new_price
new_counterparty.total_debit += new_price
new_counterparty.save()
instance.save()
return instance

View File

@@ -97,8 +97,25 @@ class IncomeCreateSerializer(serializers.ModelSerializer):
"audit", "audit",
] ]
def validate(self, data):
"""Validasiya qilish"""
price = data.get('price')
exchange_rate = data.get('exchange_rate')
if price and price < 0:
raise serializers.ValidationError("Narxi manfiy bo'lishi mumkin emas")
if exchange_rate and exchange_rate < 0:
raise serializers.ValidationError("Kurs manfiy bo'lishi mumkin emas")
return data
def create(self, validated_data): def create(self, validated_data):
with transaction.atomic(): with transaction.atomic():
# Exchange rate va final price hisoblash
exchange_rate = validated_data.get("exchange_rate") or 1
final_price = validated_data.get("price") * exchange_rate
income = Income.objects.create( income = Income.objects.create(
user=self.context.get("user"), user=self.context.get("user"),
cash_transaction=validated_data["cash_transaction"], cash_transaction=validated_data["cash_transaction"],
@@ -108,61 +125,72 @@ class IncomeCreateSerializer(serializers.ModelSerializer):
counterparty=validated_data.get("counterparty"), counterparty=validated_data.get("counterparty"),
type_income=validated_data.get("type_income"), type_income=validated_data.get("type_income"),
currency=validated_data.get("currency"), currency=validated_data.get("currency"),
price=( price=final_price,
validated_data.get("price") * validated_data.get("exchange_rate") exchange_rate=exchange_rate,
if validated_data.get("exchange_rate")
else validated_data.get("price")
),
exchange_rate=validated_data.get("exchange_rate"),
date=validated_data.get("date"), date=validated_data.get("date"),
comment=validated_data.get("comment"), comment=validated_data.get("comment"),
file=validated_data.get("file"), file=validated_data.get("file"),
audit=validated_data.get("audit"), audit=validated_data.get("audit"),
) )
cash_transaction = income.cash_transaction cash_transaction = income.cash_transaction
payment_type = income.payment_type payment_type = income.payment_type
currency = validated_data.get("currency", "uzs").lower()
if validated_data.get("currency") == "uzs": if currency == "uzs":
# Cash transaction balansini yangilash
cash_transaction.income_balance_uzs += income.price cash_transaction.income_balance_uzs += income.price
cash_transaction.total_balance_uzs = ( cash_transaction.total_balance_uzs = (
cash_transaction.income_balance_uzs cash_transaction.income_balance_uzs
- cash_transaction.expence_balance_uzs - cash_transaction.expence_balance_uzs
) )
# Payment type balansini yangilash
payment_type.total_uzs += income.price payment_type.total_uzs += income.price
# Kontrapartiya hisobini yangilash
if income.counterparty: if income.counterparty:
if income.counterparty.debit_uzs != 0: if income.counterparty.debit_uzs > 0:
# Debit bo'lsa, uni qisqartirish
income.counterparty.debit_uzs -= income.price income.counterparty.debit_uzs -= income.price
income.counterparty.total_debit -= income.price income.counterparty.total_debit -= income.price
income.counterparty.kredit_uzs += ( # Kredit oshirish
income.counterparty.debit_uzs - income.price income.counterparty.kredit_uzs += income.price # ✅ TUZATILDI
)
income.counterparty.total_kredit += income.price income.counterparty.total_kredit += income.price
else: else:
# Debit yo'q bo'lsa, to'g'ridan-to'g'ri kredit oshirish
income.counterparty.kredit_uzs += income.price income.counterparty.kredit_uzs += income.price
income.counterparty.total_kredit += income.price income.counterparty.total_kredit += income.price
income.counterparty.save() income.counterparty.save()
elif validated_data.get("currency") == "usd": elif currency == "usd":
# Cash transaction balansini yangilash
cash_transaction.income_balance_usd += income.price cash_transaction.income_balance_usd += income.price
cash_transaction.total_balance_usd = ( cash_transaction.total_balance_usd = (
cash_transaction.income_balance_usd cash_transaction.income_balance_usd
- cash_transaction.expence_balance_usd - cash_transaction.expence_balance_usd
) )
# Payment type balansini yangilash
payment_type.total_usd += income.price payment_type.total_usd += income.price
# Kontrapartiya hisobini yangilash
if income.counterparty: if income.counterparty:
if income.counterparty.debit_usd != 0: if income.counterparty.debit_usd > 0:
income.counterparty.debit_usd -= validated_data.get("price") # Debit bo'lsa, uni qisqartirish
income.counterparty.debit_usd -= income.price # ✅ TUZATILDI
income.counterparty.total_debit -= income.price income.counterparty.total_debit -= income.price
income.counterparty.kredit_usd += ( # Kredit oshirish
income.counterparty.debit_usd - validated_data.get("price") income.counterparty.kredit_usd += income.price # ✅ TUZATILDI
)
income.counterparty.total_kredit += income.price income.counterparty.total_kredit += income.price
else: else:
income.counterparty.kredit_usd += validated_data.get("price") # Debit yo'q bo'lsa, to'g'ridan-to'g'ri kredit oshirish
income.counterparty.kredit_usd += income.price # ✅ TUZATILDI
income.counterparty.total_kredit += income.price income.counterparty.total_kredit += income.price
income.counterparty.save() income.counterparty.save()
cash_transaction.save() cash_transaction.save()
@@ -192,70 +220,107 @@ class IncomeUpdateSerializer(serializers.ModelSerializer):
"price": {"required": False}, "price": {"required": False},
} }
def update(self, instance, validated_data): def validate_price(self, value):
old_price = instance.price """Narxi manfiy bo'lmasligi tekshiruvi"""
instance.project_folder = validated_data.get( if value and value < 0:
"project_folder", instance.project_folder raise serializers.ValidationError("Narxi manfiy bo'lishi mumkin emas")
) return value
instance.project = validated_data.get("project", instance.project)
instance.price = validated_data.get("price", instance.price)
instance.type_income = validated_data.get("type_income", instance.type_income)
instance.counterparty = validated_data.get(
"counterparty", instance.counterparty
)
instance.date = validated_data.get("date", instance.date)
instance.comment = validated_data.get("comment", instance.comment)
instance.audit = validated_data.get("audit", instance.audit)
instance.file = validated_data.get("file", instance.file)
if validated_data.get("price"):
cash_transaction = instance.cash_transaction
payment_type = instance.payment_type
if old_price > validated_data.get("price"):
if instance.currency == "uzs":
cash_transaction.income_balance_uzs -= (
old_price - validated_data.get("price")
)
cash_transaction.total_balance_uzs -= (
old_price - validated_data.get("price")
)
payment_type.total_uzs -= (
(old_price - validated_data.get("price"))
if payment_type.total_uzs
> (old_price - validated_data.get("price"))
else 0
)
else:
cash_transaction.income_balance_usd -= (
old_price - validated_data.get("price")
)
cash_transaction.total_balance_usd -= (
old_price - validated_data.get("price")
)
payment_type.total_usd -= (
(old_price - validated_data.get("price"))
if payment_type.total_usd
> (old_price - validated_data.get("price"))
else 0
)
elif old_price < validated_data.get("price"): def update(self, instance, validated_data):
if instance.currency == "uzs": with transaction.atomic():
cash_transaction.income_balance_uzs += ( old_price = instance.price
old_price - validated_data.get("price") old_counterparty = instance.counterparty
new_price = validated_data.get("price", instance.price)
new_counterparty = validated_data.get("counterparty", instance.counterparty)
currency = instance.currency.lower()
# Asosiy ma'lumotlarni yangilash
instance.project_folder = validated_data.get(
"project_folder", instance.project_folder
)
instance.project = validated_data.get("project", instance.project)
instance.price = new_price
instance.type_income = validated_data.get("type_income", instance.type_income)
instance.counterparty = new_counterparty
instance.date = validated_data.get("date", instance.date)
instance.comment = validated_data.get("comment", instance.comment)
instance.audit = validated_data.get("audit", instance.audit)
instance.file = validated_data.get("file", instance.file)
# Agar narxi o'zgarsa, balanslarni yangilash
if validated_data.get("price") and old_price != new_price:
price_difference = new_price - old_price # ✅ To'g'ri farq
cash_transaction = instance.cash_transaction
payment_type = instance.payment_type
if currency == "uzs":
# Cash transaction balansini yangilash
cash_transaction.income_balance_uzs += price_difference
cash_transaction.total_balance_uzs = (
cash_transaction.income_balance_uzs
- cash_transaction.expence_balance_uzs
) )
cash_transaction.total_balance_uzs += (
old_price - validated_data.get("price") # Payment type balansini yangilash
payment_type.total_uzs += price_difference # ✅ Narx oshsa, balans oshadi
elif currency == "usd":
# Cash transaction balansini yangilash
cash_transaction.income_balance_usd += price_difference
cash_transaction.total_balance_usd = (
cash_transaction.income_balance_usd
- cash_transaction.expence_balance_usd
) )
payment_type.total_uzs += old_price - validated_data.get("price")
else: # Payment type balansini yangilash
cash_transaction.income_balance_usd += ( payment_type.total_usd += price_difference # ✅ Narx oshsa, balans oshadi
old_price - validated_data.get("price")
) cash_transaction.save()
cash_transaction.total_balance_usd += ( payment_type.save()
old_price - validated_data.get("price")
) # Agar kontrapartiya o'zgarsa, uning hisobini yangilash
payment_type.total_usd += old_price - validated_data.get("price") if new_counterparty != old_counterparty:
cash_transaction.save() # Eski kontrapartyani qaytarish
payment_type.save() if old_counterparty:
instance.save() if currency == "uzs":
return instance if old_counterparty.kredit_uzs > 0:
old_counterparty.kredit_uzs -= old_price
old_counterparty.total_kredit -= old_price
else:
old_counterparty.debit_uzs += old_price
old_counterparty.total_debit += old_price
else:
if old_counterparty.kredit_usd > 0:
old_counterparty.kredit_usd -= old_price
old_counterparty.total_kredit -= old_price
else:
old_counterparty.debit_usd += old_price
old_counterparty.total_debit += old_price
old_counterparty.save()
# Yangi kontrapartyani yangilash
if new_counterparty:
if currency == "uzs":
if new_counterparty.debit_uzs > 0:
new_counterparty.debit_uzs -= new_price
new_counterparty.total_debit -= new_price
new_counterparty.kredit_uzs += new_price
new_counterparty.total_kredit += new_price
else:
new_counterparty.kredit_uzs += new_price
new_counterparty.total_kredit += new_price
else:
if new_counterparty.debit_usd > 0:
new_counterparty.debit_usd -= new_price
new_counterparty.total_debit -= new_price
new_counterparty.kredit_usd += new_price
new_counterparty.total_kredit += new_price
else:
new_counterparty.kredit_usd += new_price
new_counterparty.total_kredit += new_price
new_counterparty.save()
instance.save()
return instance

View File

@@ -1,4 +1,5 @@
from django.shortcuts import get_object_or_404 from django.shortcuts import get_object_or_404
from django.db import transaction
from rest_framework import generics, views, parsers, filters from rest_framework import generics, views, parsers, filters
from rest_framework.response import Response from rest_framework.response import Response
@@ -116,6 +117,93 @@ class ExpenceDeleteApiView(generics.GenericAPIView):
permission_classes = [HasRolePermission] permission_classes = [HasRolePermission]
def post(self, request, id): def post(self, request, id):
expence = get_object_or_404(Expence, id=id, is_deleted=False)
serializer = self.serializer_class(data=request.data)
if serializer.is_valid(raise_exception=True):
with transaction.atomic():
comment = serializer.validated_data.get('comment')
currency = expence.currency.lower()
# Deleted record yaratish
DeletedExpence.objects.create(
expence=expence,
comment=comment
)
cash_transaction = expence.cash_transaction
payment_type = expence.payment_type
counterparty = expence.counterparty
if currency == 'uzs':
# Balanslarni qaytarish (o'chirilgani teskari qilish)
cash_transaction.expence_balance_uzs -= expence.price # ✅ KAMAYISHI KERAK
cash_transaction.total_balance_uzs = (
cash_transaction.income_balance_uzs - cash_transaction.expence_balance_uzs
)
payment_type.total_uzs += expence.price # ✅ QAYTARISH
# Kontrapartiya hisobini qaytarish
if counterparty:
if counterparty.debit_uzs > 0:
# Agar debit bo'lsa, uni oshirish (teskari)
counterparty.debit_uzs += expence.price
counterparty.total_debit += expence.price
counterparty.kredit_uzs -= expence.price
counterparty.total_kredit -= expence.price
else:
# Agar kredit bo'lsa, uni kamayitish (teskari)
counterparty.kredit_uzs -= expence.price
counterparty.total_kredit -= expence.price
counterparty.save()
elif currency == 'usd':
# Balanslarni qaytarish (o'chirilgani teskari qilish)
cash_transaction.expence_balance_usd -= expence.price # ✅ KAMAYISHI KERAK
cash_transaction.total_balance_usd = (
cash_transaction.income_balance_usd - cash_transaction.expence_balance_usd
)
payment_type.total_usd += expence.price # ✅ QAYTARISH
# Kontrapartiya hisobini qaytarish
if counterparty:
if counterparty.debit_usd > 0:
# Agar debit bo'lsa, uni oshirish (teskari)
counterparty.debit_usd += expence.price
counterparty.total_debit += expence.price
counterparty.kredit_usd -= expence.price
counterparty.total_kredit -= expence.price
else:
# Agar kredit bo'lsa, uni kamayitish (teskari)
counterparty.kredit_usd -= expence.price
counterparty.total_kredit -= expence.price
counterparty.save()
# is_deleted = True qilish (hard delete emas, soft delete)
expence.is_deleted = True
# Barcha o'zgarishlari saqlash
cash_transaction.save()
payment_type.save()
expence.save()
return Response(
{
'success': True,
'message': 'Expence o\'chirildi',
'data': {
'expence_id': expence.id,
'price': expence.price,
'currency': expence.currency
}
},
status=200
)
expence = get_object_or_404(Expence, id=id) expence = get_object_or_404(Expence, id=id)
serializer = self.serializer_class(data=request.data) serializer = self.serializer_class(data=request.data)
if serializer.is_valid(raise_exception=True): if serializer.is_valid(raise_exception=True):

View File

@@ -1,4 +1,5 @@
from django.shortcuts import get_object_or_404 from django.shortcuts import get_object_or_404
from django.db import transaction
from rest_framework import generics, views, parsers, filters from rest_framework import generics, views, parsers, filters
from rest_framework.response import Response from rest_framework.response import Response
@@ -84,33 +85,93 @@ class IncomeDeleteApiView(generics.GenericAPIView):
permission_classes = [HasRolePermission] permission_classes = [HasRolePermission]
def post(self, request, id): def post(self, request, id):
income = get_object_or_404(Income, id=id) income = get_object_or_404(Income, id=id, is_deleted=False)
serializer = self.serializer_class(data=request.data) serializer = self.serializer_class(data=request.data)
if serializer.is_valid(raise_exception=True): if serializer.is_valid(raise_exception=True):
comment = serializer.validated_data.get('comment') with transaction.atomic():
DeletedIncome.objects.create( comment = serializer.validated_data.get('comment')
income=income, currency = income.currency.lower()
comment=comment
) # Deleted record yaratish
income.is_deleted = True DeletedIncome.objects.create(
if income.currency == 'uzs': income=income,
income.cash_transaction.expence_balance_uzs -= income.price comment=comment
income.cash_transaction.total_balance_uzs -= income.price )
income.payment_type.total_uzs -= income.price if income.payment_type.total_uzs > income.price else 0
else: cash_transaction = income.cash_transaction
income.cash_transaction.expence_balance_usd -= income.price payment_type = income.payment_type
income.cash_transaction.total_balance_usd -= income.price counterparty = income.counterparty
income.payment_type.total_usd -= income.price if income.payment_type.total_usd > income.price else 0
if currency == 'uzs':
# Balanslarni qaytarish (o'chirilgani teskari qilish)
cash_transaction.income_balance_uzs -= income.price # ✅ INCOME BALANCE KAMAYISHI KERAK
cash_transaction.total_balance_uzs = (
cash_transaction.income_balance_uzs - cash_transaction.expence_balance_uzs
)
payment_type.total_uzs -= income.price # ✅ QAYTARISH
# Kontrapartiya hisobini qaytarish
if counterparty:
if counterparty.kredit_uzs > 0:
# Agar kredit bo'lsa, uni kamayitish (teskari)
counterparty.kredit_uzs -= income.price
counterparty.total_kredit -= income.price
counterparty.debit_uzs += income.price
counterparty.total_debit += income.price
else:
# Agar debit bo'lsa, uni oshirish (teskari)
counterparty.debit_uzs += income.price
counterparty.total_debit += income.price
counterparty.save()
elif currency == 'usd':
# Balanslarni qaytarish (o'chirilgani teskari qilish)
cash_transaction.income_balance_usd -= income.price # ✅ INCOME BALANCE KAMAYISHI KERAK
cash_transaction.total_balance_usd = (
cash_transaction.income_balance_usd - cash_transaction.expence_balance_usd
)
payment_type.total_usd -= income.price # ✅ QAYTARISH
# Kontrapartiya hisobini qaytarish
if counterparty:
if counterparty.kredit_usd > 0:
# Agar kredit bo'lsa, uni kamayitish (teskari)
counterparty.kredit_usd -= income.price
counterparty.total_kredit -= income.price
counterparty.debit_usd += income.price
counterparty.total_debit += income.price
else:
# Agar debit bo'lsa, uni oshirish (teskari)
counterparty.debit_usd += income.price
counterparty.total_debit += income.price
counterparty.save()
# is_deleted = True qilish (hard delete emas, soft delete)
income.is_deleted = True
# Barcha o'zgarishlari saqlash
cash_transaction.save()
payment_type.save()
income.save()
return Response(
{
'success': True,
'message': 'Income o\'chirildi',
'data': {
'income_id': income.id,
'price': income.price,
'currency': income.currency
}
},
status=200
)
income.cash_transaction.save()
income.payment_type.save()
income.save()
return Response(
{
'success': True,
'message': 'Income deleted',
}, status=200
)
class IncomeUpdateApiView(generics.GenericAPIView): class IncomeUpdateApiView(generics.GenericAPIView):

View File

@@ -381,126 +381,154 @@ class PartyExpenceCreateSerializer(serializers.ModelSerializer):
"party", "party",
] ]
def validate(self, data):
"""Validasiya qilish"""
price = data.get('price')
exchange_rate = data.get('exchange_rate')
party = data.get('party')
currency = data.get('currency', 'uzs').lower()
if price and price < 0:
raise serializers.ValidationError("Narxi manfiy bo'lishi mumkin emas")
if exchange_rate and exchange_rate < 0:
raise serializers.ValidationError("Kurs manfiy bo'lishi mumkin emas")
if not party:
raise serializers.ValidationError("Party talab qilinadi")
# USD kursi tekshiruvi
try:
usd_course = UsdCourse.objects.first()
if not usd_course or not usd_course.value:
raise serializers.ValidationError("USD kursi topilmadi")
except UsdCourse.DoesNotExist:
raise serializers.ValidationError("USD kursi topilmadi")
return data
def create(self, validated_data): def create(self, validated_data):
with transaction.atomic(): with transaction.atomic():
usd_value = UsdCourse.objects.first().value # USD kursi olish
usd_course = UsdCourse.objects.first()
if not usd_course:
raise serializers.ValidationError("USD kursi topilmadi")
usd_value = usd_course.value
# Exchange rate va final price hisoblash
exchange_rate = validated_data.get("exchange_rate") or 1
final_price = validated_data.get("price") * exchange_rate
currency = validated_data.get("currency", "uzs").lower()
expence = Expence.objects.create( expence = Expence.objects.create(
cash_transaction=validated_data.get("cash_transaction"), cash_transaction=validated_data.get("cash_transaction"),
payment_type=validated_data.get("payment_type"), payment_type=validated_data.get("payment_type"),
project_folder=validated_data.get("project_folder"), project_folder=validated_data.get("project_folder"),
project=validated_data.get("project"), project=validated_data.get("project"),
counterparty=validated_data.get("counterparty"), counterparty=validated_data.get("counterparty"),
price=( price=final_price,
validated_data.get("price") * validated_data.get("exchange_rate") exchange_rate=exchange_rate,
if validated_data.get("exchange_rate") currency=currency,
else validated_data.get("price")
),
exchange_rate=validated_data.get("exchange_rate"),
currency=validated_data.get("currency"),
date=validated_data.get("date"), date=validated_data.get("date"),
comment=validated_data.get("comment"), comment=validated_data.get("comment"),
party=validated_data.get("party"), party=validated_data.get("party"),
user=self.context.get('user'), user=self.context.get('user'),
) )
cash_transaction = expence.cash_transaction cash_transaction = expence.cash_transaction
payment_type = expence.payment_type payment_type = expence.payment_type
party = expence.party
if validated_data.get("currency") == "uzs":
if currency == "uzs":
# Cash transaction balansini yangilash
cash_transaction.expence_balance_uzs += expence.price cash_transaction.expence_balance_uzs += expence.price
cash_transaction.total_balance_uzs = ( cash_transaction.total_balance_uzs = (
cash_transaction.income_balance_uzs cash_transaction.income_balance_uzs
- cash_transaction.expence_balance_uzs - cash_transaction.expence_balance_uzs
) )
# Payment type balansini yangilash
if payment_type.total_uzs > expence.price: if payment_type.total_uzs > expence.price:
payment_type.total_uzs -= expence.price payment_type.total_uzs -= expence.price
# Kontrapartiya hisobini yangilash
if expence.counterparty: if expence.counterparty:
if expence.counterparty.kredit_uzs != 0: if expence.counterparty.kredit_uzs > 0:
expence.counterparty.kredit_uzs -= expence.price expence.counterparty.kredit_uzs -= expence.price
expence.counterparty.total_kredit -= expence.price expence.counterparty.total_kredit -= expence.price
expence.counterparty.debit_uzs += ( # ✅ TUZATILDI: Debit to'g'ridan-to'g'ri oshadi
expence.counterparty.kredit_uzs - expence.price expence.counterparty.debit_uzs += expence.price
)
expence.counterparty.total_debit += expence.price expence.counterparty.total_debit += expence.price
else: else:
expence.counterparty.debit_uzs += expence.price expence.counterparty.debit_uzs += expence.price
expence.counterparty.total_debit += expence.price expence.counterparty.total_debit += expence.price
expence.counterparty.save() expence.counterparty.save()
if expence.party.currency == "uzs":
expence.party.party_amount.payment_amount -= expence.price # Party hisobini yangilash
expence.party.party_amount.paid_amount += expence.price if party:
expence.party.party_amount.save() if party.currency == "uzs":
expence.party.save() # ✅ Birjani to'g'ri hisoblash
elif expence.party.currency == 'usd': party.party_amount.payment_amount -= expence.price
expence.party.party_amount.payment_amount -= round(expence.price / usd_value) party.party_amount.paid_amount += expence.price
expence.party.party_amount.paid_amount += round(expence.price / usd_value) party.party_amount.save()
expence.party.save() elif party.currency == 'usd':
elif validated_data.get("currency") == "usd": # ✅ USD kursi bilan to'g'ri konvertatsiya
converted_price = round(expence.price / usd_value)
party.party_amount.payment_amount -= converted_price
party.party_amount.paid_amount += converted_price
party.party_amount.save()
party.save()
elif currency == "usd":
# Cash transaction balansini yangilash
cash_transaction.expence_balance_usd += expence.price cash_transaction.expence_balance_usd += expence.price
cash_transaction.total_balance_usd = ( cash_transaction.total_balance_usd = (
cash_transaction.income_balance_usd cash_transaction.income_balance_usd
- cash_transaction.expence_balance_usd - cash_transaction.expence_balance_usd
) )
# Payment type balansini yangilash
if payment_type.total_usd > expence.price: if payment_type.total_usd > expence.price:
payment_type.total_usd -= expence.price payment_type.total_usd -= expence.price
# Kontrapartiya hisobini yangilash
if expence.counterparty: if expence.counterparty:
if expence.counterparty.kredit_usd != 0: if expence.counterparty.kredit_usd > 0:
expence.counterparty.kredit_usd -= validated_data.get("price") # ✅ TUZATILDI: expence.price ishlatish
expence.counterparty.kredit_usd -= expence.price
expence.counterparty.total_kredit -= expence.price expence.counterparty.total_kredit -= expence.price
expence.counterparty.debit_usd += ( # ✅ TUZATILDI: Debit to'g'ridan-to'g'ri oshadi
expence.counterparty.kredit_usd expence.counterparty.debit_usd += expence.price
- validated_data.get("price")
)
expence.counterparty.total_debit += expence.price expence.counterparty.total_debit += expence.price
else: else:
expence.counterparty.debit_usd += validated_data.get("price") expence.counterparty.debit_usd += expence.price
expence.counterparty.total_debit += expence.price expence.counterparty.total_debit += expence.price
expence.counterparty.save() expence.counterparty.save()
if expence.party.currency == "usd": # Party hisobini yangilash
expence.party.party_amount.payment_amount -= validated_data.get( if party:
"price" if party.currency == "usd":
) # ✅ TUZATILDI: expence.price ishlatish
expence.party.party_amount.paid_amount += validated_data.get( party.party_amount.payment_amount -= expence.price
"price" party.party_amount.paid_amount += expence.price
) party.party_amount.save()
expence.party.save() elif party.currency == "uzs":
elif expence.party.currency == "uzs": # ✅ TUZATILDI: expence.price bilan konvertatsiya
expence.party.party_amount.payment_amount -= validated_data.get( converted_price = expence.price * usd_value
"price" party.party_amount.payment_amount -= converted_price
) * usd_value party.party_amount.paid_amount += converted_price
expence.party.party_amount.paid_amount += validated_data.get( party.party_amount.save()
"price"
) * usd_value party.save()
expence.party.save()
expence.party.party_amount.save() # Barcha o'zgarishlari saqlash
cash_transaction.save() cash_transaction.save()
payment_type.save() payment_type.save()
return expence return expence
class PartyPaymentHistorySerializer(serializers.ModelSerializer):
counterparty = serializers.SerializerMethodField(method_name='get_counterparty')
user = serializers.SerializerMethodField(method_name='get_user')
class Meta:
model = Expence
fields = [
'id', 'counterparty', 'price', 'created_at', 'date', 'user'
]
def get_counterparty(self, obj):
return {
'id': obj.counterparty.id,
'name': obj.counterparty.name,
} if obj.counterparty else None
def get_user(self, obj):
return {
'id': obj.user.id,
'full_name': obj.user.full_name,
'profile_image': obj.user.profile_image.url if obj.user.profile_image else None
} if obj.user else None