Files
ibapp-backend/core/apps/finance/views/income.py
behruz-dev 3d9bf5bea6 fix
2025-11-05 16:17:41 +05:00

198 lines
8.2 KiB
Python

from django.shortcuts import get_object_or_404
from django.db import transaction
from rest_framework import generics, views, parsers, filters
from rest_framework.response import Response
from django_filters.rest_framework.backends import DjangoFilterBackend
from core.apps.finance.models import Income, DeletedIncome
from core.apps.finance.serializers import income as serializers
from core.apps.accounts.permissions.permissions import HasRolePermission
from core.apps.finance.filters.income import IncomeFilter
from core.apps.counterparty.models import Counterparty
class IncomeListApiView(generics.GenericAPIView):
serializer_class = serializers.IncomeListSerializer
queryset = Income.objects.select_related(
'cash_transaction', 'payment_type', 'project_folder', 'project', 'counterparty', 'type_income',
'user'
).exclude(is_deleted=True)
permission_classes = [HasRolePermission]
filter_backends = [DjangoFilterBackend, filters.SearchFilter]
filterset_class = IncomeFilter
search_fields = [
'cash_transaction__name', 'payment_type__name', 'project_folder__name', 'project__name', 'counterparty__name', 'type_income__name', 'user__full_name'
]
def get(self, request):
cash_transaction_ids = request.query_params.getlist('cash_transaction')
if cash_transaction_ids:
self.queryset = self.queryset.filter(cash_transaction__in=cash_transaction_ids)
page = self.paginate_queryset(self.filter_queryset(self.queryset))
if page is not None:
serializer = self.serializer_class(page, many=True)
return self.get_paginated_response(serializer.data)
class IncomeCreateApiView(generics.GenericAPIView):
serializer_class = serializers.IncomeCreateSerializer
queryset = Income.objects.all()
permission_classes = [HasRolePermission]
parser_classes = [parsers.FormParser, parsers.MultiPartParser]
def post(self, request):
ser = self.serializer_class(data=request.data, context={'user': request.user})
if ser.is_valid(raise_exception=True):
ser.save()
return Response(
{
'success': True,
'message': 'income created'
},
status=201
)
return Response(
{
'success': False,
'message': 'income create failed',
'error': ser.errors,
},
status=400
)
class CounterpartyIncomeListApiView(generics.GenericAPIView):
permission_classes = [HasRolePermission]
queryset = Income.objects.select_related(
'cash_transaction', 'payment_type', 'project_folder', 'project', 'counterparty', 'type_income',
'user'
).exclude(counterparty__isnull=True).distinct()
serializer_class = serializers.IncomeListSerializer
def get(self, request, counterparty_id):
counterparty = get_object_or_404(Counterparty, id=counterparty_id)
page = self.paginate_queryset(self.queryset.filter(counterparty=counterparty))
if page is not None:
ser = self.serializer_class(page, many=True)
return self.get_paginated_response(ser.data)
class IncomeDeleteApiView(generics.GenericAPIView):
serializer_class = serializers.IncomeDeleteSerializer
queryset = Income.objects.all()
permission_classes = [HasRolePermission]
def post(self, request, id):
income = get_object_or_404(Income, 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 = income.currency.lower()
# Deleted record yaratish
DeletedIncome.objects.create(
income=income,
comment=comment
)
cash_transaction = income.cash_transaction
payment_type = income.payment_type
counterparty = income.counterparty
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
)
class IncomeUpdateApiView(generics.GenericAPIView):
serializer_class = serializers.IncomeUpdateSerializer
queryset = Income.objects.all()
permission_classes = [HasRolePermission]
def patch(self, request, id):
income = get_object_or_404(Income, id=id)
serializer = self.serializer_class(data=request.data, instance=income)
if serializer.is_valid(raise_exception=True):
serializer.save()
return Response(
{
'success': True,
'message': 'Income updated'
}, status=200
)
return Response(
{
'success': False,
'message': serializer.errors,
}, status=400
)