fix bug of inventory entry - fix weight of quotas

This commit is contained in:
2025-08-09 16:15:37 +03:30
parent 365362b739
commit 92a5d3a2eb
10 changed files with 181 additions and 32 deletions

View File

@@ -0,0 +1,18 @@
# Generated by Django 5.0 on 2025-08-09 08:09
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('authentication', '0030_alter_bankaccountinformation_user'),
]
operations = [
migrations.AddField(
model_name='organization',
name='separate_warehouse',
field=models.BooleanField(default=False),
),
]

View File

@@ -0,0 +1,18 @@
# Generated by Django 5.0 on 2025-08-09 12:44
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('authentication', '0031_organization_separate_warehouse'),
]
operations = [
migrations.AddField(
model_name='organization',
name='has_pos',
field=models.BooleanField(default=False),
),
]

View File

@@ -126,6 +126,7 @@ class Organization(BaseModel):
null=True
)
separate_warehouse = models.BooleanField(default=False)
has_pos = models.BooleanField(default=False)
additional_data = models.JSONField(default=dict)
def __str__(self):

View File

@@ -0,0 +1,42 @@
# Generated by Django 5.0 on 2025-08-09 08:09
import datetime
import django.db.models.deletion
from django.conf import settings
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('authentication', '0031_organization_separate_warehouse'),
('pos_device', '0045_alter_device_acceptor_alter_device_latitude_and_more'),
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
]
operations = [
migrations.AlterField(
model_name='deviceactivationcode',
name='expires_at',
field=models.DateTimeField(default=datetime.datetime(2025, 8, 9, 11, 39, 37, 865832)),
),
migrations.CreateModel(
name='StakeHolders',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('create_date', models.DateTimeField(auto_now_add=True)),
('modify_date', models.DateTimeField(auto_now=True)),
('creator_info', models.CharField(max_length=100, null=True)),
('modifier_info', models.CharField(max_length=100, null=True)),
('trash', models.BooleanField(default=False)),
('share_percent', models.FloatField(default=0)),
('assignment', models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, related_name='stake_holders', to='pos_device.deviceassignment')),
('created_by', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='%(class)s_createddby', to=settings.AUTH_USER_MODEL)),
('modified_by', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='%(class)s_modifiedby', to=settings.AUTH_USER_MODEL)),
('organization', models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, related_name='stake_holders', to='authentication.organization')),
],
options={
'abstract': False,
},
),
]

View File

@@ -0,0 +1,36 @@
# Generated by Django 5.0 on 2025-08-09 12:44
import datetime
import django.db.models.deletion
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('authentication', '0032_organization_has_pos'),
('authorization', '0019_page_is_active_permissions_is_active'),
('pos_device', '0046_alter_deviceactivationcode_expires_at_stakeholders'),
]
operations = [
migrations.AddField(
model_name='posclientattribute',
name='custom_field',
field=models.BooleanField(default=False),
),
migrations.AlterField(
model_name='deviceactivationcode',
name='expires_at',
field=models.DateTimeField(default=datetime.datetime(2025, 8, 9, 16, 14, 34, 958731)),
),
migrations.AlterField(
model_name='stakeholders',
name='organization',
field=models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, related_name='pos_stake_holders', to='authentication.organization'),
),
migrations.AlterUniqueTogether(
name='providercompany',
unique_together={('user_relation', 'name_fa')},
),
]

View File

@@ -14,12 +14,15 @@ class ProviderCompany(BaseModel):
UserRelations,
related_name='pos_provider',
on_delete=models.CASCADE,
null=True
null=True,
)
name_fa = models.CharField(max_length=250, null=True)
name_en = models.CharField(max_length=250, null=True)
activation = models.BooleanField(default=False)
class Meta:
unique_together = ('user_relation', 'name_fa')
def __str__(self):
return f'Payment Company: {self.name_fa}-{self.id}'
@@ -175,6 +178,7 @@ class POSClientAttribute(BaseModel):
], null=True)
required = models.BooleanField(default=False)
choices = ArrayField(base_field=models.CharField(max_length=150), null=True, blank=True)
custom_field = models.BooleanField(default=False)
def __str__(self):
return f'attribute: {self.key}-{self.field_type}'
@@ -242,7 +246,7 @@ class StakeHolders(BaseModel):
organization = models.ForeignKey(
Organization,
on_delete=models.CASCADE,
related_name='stake_holders',
related_name='pos_stake_holders',
null=True
)
share_percent = models.FloatField(default=0)
@@ -251,4 +255,4 @@ class StakeHolders(BaseModel):
return f'Device: {self.assignment.device.serial}-organization: {self.organization.name}'
def save(self, *args, **kwargs):
return super(StakeHolders, self).save(*args, **kwargs)
return super(StakeHolders, self).save(*args, **kwargs)

View File

@@ -38,3 +38,9 @@ class DeviceAssignmentSerializer(ModelSerializer):
class Meta:
model = pos_models.DeviceAssignment
fields = '__all__'
class StakeHoldersSerializer(ModelSerializer):
class Meta:
model = pos_models.StakeHolders
fields = '__all__'

View File

@@ -1,6 +1,7 @@
from datetime import timedelta
from apps.pos_device.web.api.v1.serilaizers import device as device_serializer
from apps.authorization.api.v1.serializers import UserRelationSerializer
from apps.authentication.api.v1.api import UserViewSet
from apps.authorization.models import UserRelations
from rest_framework.exceptions import APIException
@@ -19,30 +20,34 @@ class ProviderCompanyViewSet(viewsets.ModelViewSet): # noqa
queryset = pos_models.ProviderCompany.objects.all()
serializer_class = device_serializer.ProviderCompanySerializer
def create(self, request, *args, **kwargs):
""" custom create of provider client """
def list(self, request, *args, **kwargs):
""" list of provider companies """
try:
# creating user & relations
client = UserViewSet().create(request=request)
# paginate devices
page = self.paginate_queryset(self.queryset.order_by('-create_date'))
if page is not None:
serializer = self.get_serializer(page, many=True)
return self.get_paginated_response(serializer.data)
if client.status_code == 201:
@action(
methods=['get'],
detail=False,
url_path='psp_users',
url_name='psp_users',
name='psp_users'
)
def users_with_psp_org(self):
""" list of psp users """
# create provider
serializer = self.serializer_class(data=request.data['provider'])
if serializer.is_valid():
provider = serializer.save()
provider.user_relation = UserRelations.objects.get(
user_id=client.data['id']
)
provider.save()
users = UserRelations.objects.filter(
organization__type__key='PSP'
)
return Response(serializer.data, status=status.HTTP_201_CREATED)
return Response(serializer.errors, status=status.HTTP_403_FORBIDDEN)
else:
return client
except Exception as e:
raise e
# paginate devices
page = self.paginate_queryset(users)
if page is not None:
serializer = UserRelationSerializer(page, many=True)
return self.get_paginated_response(serializer.data)
class DeviceViewSet(viewsets.ModelViewSet):
@@ -127,16 +132,21 @@ class DeviceAssignmentViewSet(viewsets.ModelViewSet):
def device_assignment(self, request):
""" assign pos device to client by company """
company = pos_models.ProviderCompany.objects.get(
user_relation__user=request.user
)
request.data.update({'company': company.id})
if 'company' not in request.data.keys():
company = pos_models.ProviderCompany.objects.get(
user_relation__user=request.user
)
request.data.update({'company': company.id})
# create assignment
serializer = self.serializer_class(data=request.data)
if serializer.is_valid():
assignment = serializer.save()
# set organization having pos status
assignment.company.user_relation.organization.has_pos = True
assignment.company.user_relation.organization.save()
return Response(serializer.data, status=status.HTTP_201_CREATED)
return Response(serializer.errors, status=status.HTTP_403_FORBIDDEN)
@@ -162,3 +172,8 @@ class DeviceAssignmentViewSet(viewsets.ModelViewSet):
return Response(serializer.data, status=status.HTTP_200_OK)
except Exception as e:
raise APIException('Non Object Error', code=403)
class StakeHoldersViewSet(viewsets.ModelViewSet):
queryset = pos_models.StakeHolders.objects.all()
serializer_class = device_serializer.StakeHoldersSerializer

View File

@@ -20,7 +20,9 @@ from crum import get_current_user
def recalculate_remaining_amount(quota):
""" calculate remaining weight from distribution """
total_distributed = quota.distributions_assigned.aggregate(
total_distributed = quota.distributions_assigned.filter(
parent_distribution__isnull=True,
).aggregate(
total=Sum('weight')
)['total'] or 0

View File

@@ -42,12 +42,19 @@ class InventoryEntrySerializer(serializers.ModelSerializer):
total=models.Sum('weight')
)['total'] or 0
if self.instance.weight is 0:
# if instance exists, for update check weight with distribution weight
if self.instance:
if self.instance.weight is 0:
if total_entered + attrs['weight'] > distribution.weight:
raise InventoryEntryWeightException()
elif self.instance.weight is not 0:
if total_entered - self.instance.weight + attrs['weight'] > distribution.weight:
raise InventoryEntryWeightException()
# if instance is not exists for create, check entry weight with distribution
elif not self.instance:
if total_entered + attrs['weight'] > distribution.weight:
raise InventoryEntryWeightException()
elif self.instance.weight is not 0:
if total_entered - self.instance.weight + attrs['weight'] > distribution.weight:
raise InventoryEntryWeightException()
return attrs