feat: asm初步接口

This commit is contained in:
caoqianming 2025-12-19 14:02:16 +08:00
parent 4996b0e4a3
commit a81e851588
5 changed files with 161 additions and 52 deletions

View File

@ -0,0 +1,86 @@
# Generated by Django 3.2.12 on 2025-12-18 08:36
from django.conf import settings
from django.db import migrations, models
import django.db.models.deletion
import django.utils.timezone
class Migration(migrations.Migration):
initial = True
dependencies = [
('system', '0006_auto_20241213_1249'),
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
('pum', '0009_supplieraudit'),
('wf', '0006_auto_20251215_1645'),
]
operations = [
migrations.CreateModel(
name='AssetLog',
fields=[
('id', models.CharField(editable=False, help_text='主键ID', max_length=20, primary_key=True, serialize=False, verbose_name='主键ID')),
('create_time', models.DateTimeField(default=django.utils.timezone.now, help_text='创建时间', verbose_name='创建时间')),
('update_time', models.DateTimeField(auto_now=True, help_text='修改时间', verbose_name='修改时间')),
('is_deleted', models.BooleanField(default=False, help_text='删除标记', verbose_name='删除标记')),
('type', models.CharField(help_text='入库/出库', max_length=50, verbose_name='流水类型')),
('start_date', models.DateField(blank=True, null=True, verbose_name='启用日期')),
('items', models.JSONField(blank=True, default=list, null=True, verbose_name='资产明细')),
('create_by', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='assetlog_create_by', to=settings.AUTH_USER_MODEL, verbose_name='创建人')),
('keep_dept', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to='system.dept', verbose_name='保管部门')),
('ticket', models.OneToOneField(blank=True, null=True, on_delete=django.db.models.deletion.PROTECT, related_name='assetlog_ticket', to='wf.ticket', verbose_name='关联工单')),
('update_by', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='assetlog_update_by', to=settings.AUTH_USER_MODEL, verbose_name='最后编辑人')),
],
options={
'abstract': False,
},
),
migrations.CreateModel(
name='AssetCate',
fields=[
('id', models.CharField(editable=False, help_text='主键ID', max_length=20, primary_key=True, serialize=False, verbose_name='主键ID')),
('create_time', models.DateTimeField(default=django.utils.timezone.now, help_text='创建时间', verbose_name='创建时间')),
('update_time', models.DateTimeField(auto_now=True, help_text='修改时间', verbose_name='修改时间')),
('is_deleted', models.BooleanField(default=False, help_text='删除标记', verbose_name='删除标记')),
('name', models.CharField(max_length=50, unique=True, verbose_name='类别名称')),
('code', models.CharField(blank=True, max_length=50, null=True, unique=True, verbose_name='类别编码')),
('default_unit', models.CharField(default='', max_length=20, verbose_name='默认单位')),
('create_by', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='assetcate_create_by', to=settings.AUTH_USER_MODEL, verbose_name='创建人')),
('update_by', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='assetcate_update_by', to=settings.AUTH_USER_MODEL, verbose_name='最后编辑人')),
],
options={
'abstract': False,
},
),
migrations.CreateModel(
name='Asset',
fields=[
('id', models.CharField(editable=False, help_text='主键ID', max_length=20, primary_key=True, serialize=False, verbose_name='主键ID')),
('create_time', models.DateTimeField(default=django.utils.timezone.now, help_text='创建时间', verbose_name='创建时间')),
('update_time', models.DateTimeField(auto_now=True, help_text='修改时间', verbose_name='修改时间')),
('is_deleted', models.BooleanField(default=False, help_text='删除标记', verbose_name='删除标记')),
('card_number', models.CharField(blank=True, max_length=50, null=True, unique=True, verbose_name='卡片编号')),
('name', models.CharField(max_length=100, verbose_name='固定资产名称')),
('specification', models.CharField(blank=True, max_length=100, null=True, verbose_name='规格型号')),
('quantity', models.PositiveIntegerField(default=1, verbose_name='数量')),
('start_date', models.DateField(verbose_name='启用日期')),
('canuse_year', models.PositiveSmallIntegerField(blank=True, null=True, verbose_name='可用年限')),
('original_value', models.DecimalField(decimal_places=2, max_digits=15, verbose_name='资产原值')),
('storage_location', models.CharField(blank=True, max_length=100, null=True, verbose_name='存放地点')),
('state', models.CharField(help_text='在用/闲置', max_length=50, verbose_name='使用状态')),
('unit', models.CharField(default='', max_length=50, verbose_name='计量单位')),
('note', models.TextField(blank=True, null=True, verbose_name='备注')),
('cate', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, to='asm.assetcate', verbose_name='资产类别')),
('create_by', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='asset_create_by', to=settings.AUTH_USER_MODEL, verbose_name='创建人')),
('keep_dept', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to='system.dept', verbose_name='保管部门')),
('keeper', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL, verbose_name='保管人')),
('supplier', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to='pum.supplier', verbose_name='供应商')),
('update_by', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='asset_update_by', to=settings.AUTH_USER_MODEL, verbose_name='最后编辑人')),
],
options={
'abstract': False,
},
),
]

View File

@ -1,30 +1,38 @@
from apps.utils.models import CommonADModel, CommonBDModel from apps.utils.models import CommonADModel, CommonBDModel, CommonAModel
from django.db import models from django.db import models
class AssetAudit(CommonBDModel): class AssetCate(CommonAModel):
asset_data = models.JSONField(verbose_name='资产信息', blank=True, null=True) name = models.CharField('类别名称',max_length=50, unique=True)
ticket = models.OneToOneField('wf.ticket', verbose_name='关联工单', code = models.CharField('类别编码',max_length=50, unique=True, null=True, blank=True)
on_delete=models.SET_NULL, related_name='asset_ticket', null=True, blank=True, db_constraint=False) default_unit = models.CharField('默认单位', max_length=20, default='')
def __str__(self): class Asset(CommonAModel):
return f"{self.code} - {self.name}" """
TN:固定资产台账
"""
class Asset(CommonADModel): card_number = models.CharField('卡片编号',max_length=50, unique=True, blank=True, null=True)
code = models.CharField('固定资产编号',max_length=50, unique=True)
name = models.CharField('固定资产名称',max_length=100) name = models.CharField('固定资产名称',max_length=100)
model = models.CharField("资产型号", max_length=100, blank=True, null=True) specification = models.CharField("规格型号", max_length=100, blank=True, null=True)
quantity = models.IntegerField("数量", default=1) cate = models.ForeignKey(AssetCate, verbose_name='资产类别', on_delete=models.PROTECT)
price = models.DecimalField("资产原值/单价(元)", max_digits=12, decimal_places=2) quantity = models.PositiveIntegerField("数量", default=1)
net_value = models.DecimalField("净值", max_digits=12, decimal_places=2) start_date = models.DateField("启用日期")
asm_user = models.CharField("使用人", max_length=50, blank=True, null=True) canuse_year = models.PositiveSmallIntegerField("可用年限", null=True, blank=True)
keeper = models.CharField("使用保管人", max_length=100, blank=True, null=True) original_value = models.DecimalField('资产原值', max_digits=15, decimal_places=2)
useful_life = models.IntegerField("使用年限(年)", blank=True, null=True) storage_location = models.CharField("存放地点", max_length=100, blank=True, null=True)
location = models.CharField("存放地点", max_length=100, blank=True, null=True) keeper = models.ForeignKey('system.user', verbose_name='保管人', on_delete=models.SET_NULL, null=True, blank=True)
department = models.CharField("存放部门", max_length=100, blank=True, null=True) keep_dept = models.ForeignKey('system.dept', verbose_name='保管部门', on_delete=models.SET_NULL, null=True, blank=True)
vendor = models.CharField("客商", max_length=100, blank=True, null=True) state = models.CharField("使用状态", max_length=50, help_text="在用/闲置")
category = models.CharField("固定资产类别", max_length=100, blank=True, null=True) supplier = models.ForeignKey('pum.supplier', verbose_name='供应商', on_delete=models.SET_NULL, null=True, blank=True)
start_date = models.DateField("启用日期", blank=True, null=True) unit = models.CharField("计量单位", max_length=50, default='')
expire_date = models.DateField("到期日期", blank=True, null=True) note = models.TextField("备注", blank=True, null=True)
unit = models.CharField("计量单位", max_length=50, blank=True, null=True)
remark = models.TextField("备注", blank=True, null=True) class AssetLog(CommonADModel):
"""
TN:资产操作日志
"""
type = models.CharField("流水类型", max_length=50, help_text="入库/出库")
keep_dept = models.ForeignKey('system.dept', verbose_name='保管部门', on_delete=models.SET_NULL, null=True, blank=True)
start_date = models.DateField("启用日期", null=True, blank=True)
items = models.JSONField(verbose_name='资产明细', default=list, null=True, blank=True)
ticket = models.OneToOneField('wf.ticket', verbose_name='关联工单',
on_delete=models.PROTECT, related_name='assetlog_ticket', null=True, blank=True)

View File

@ -1,16 +1,23 @@
from apps.asm.models import Asset, AssetAudit from apps.asm.models import Asset, AssetLog, AssetCate
from apps.utils.serializers import CustomModelSerializer from apps.utils.serializers import CustomModelSerializer
from apps.utils.constants import EXCLUDE_FIELDS_DEPT from apps.utils.constants import EXCLUDE_FIELDS_DEPT, EXCLUDE_FIELDS
class AssetCateSerializer(CustomModelSerializer):
class Meta:
model = AssetCate
fields = '__all__'
read_only_fields = EXCLUDE_FIELDS
class AssetSerializer(CustomModelSerializer): class AssetSerializer(CustomModelSerializer):
class Meta: class Meta:
model = Asset model = Asset
fields = '__all__' fields = '__all__'
read_only_fields = EXCLUDE_FIELDS_DEPT read_only_fields = EXCLUDE_FIELDS
class AssetlogSerializer(CustomModelSerializer): class AssetLogSerializer(CustomModelSerializer):
class Meta: class Meta:
model = AssetAudit model = AssetLog
fields = '__all__' fields = '__all__'
read_only_fields = EXCLUDE_FIELDS_DEPT read_only_fields = EXCLUDE_FIELDS_DEPT

View File

@ -1,12 +1,13 @@
from django.urls import path, include from django.urls import path, include
from rest_framework.routers import DefaultRouter from rest_framework.routers import DefaultRouter
from apps.asm.views import AssetViewSet, AssetAuditViewSet from apps.asm.views import AssetViewSet, AssetLogViewSet, AssetCateViewSet
API_BASE_URL = 'api/asm/' API_BASE_URL = 'api/asm/'
router = DefaultRouter() router = DefaultRouter()
router.register('assetcate', AssetCateViewSet, basename='assetcate')
router.register('asset', AssetViewSet, basename='asset') router.register('asset', AssetViewSet, basename='asset')
router.register('assetlog', AssetAuditViewSet, basename='assetlog') router.register('assetlog', AssetLogViewSet, basename='assetlog')
urlpatterns = [ urlpatterns = [
path(API_BASE_URL, include(router.urls)), path(API_BASE_URL, include(router.urls)),
] ]

View File

@ -1,27 +1,20 @@
from apps.wf.mixins import TicketMixin from apps.wf.mixins import TicketMixin
from apps.utils.viewsets import CustomModelViewSet from apps.utils.viewsets import CustomModelViewSet
from apps.asm.models import Asset from apps.asm.models import Asset, AssetLog, AssetCate
from rest_framework.exceptions import ParseError from rest_framework.exceptions import ParseError
from apps.asm.serializers import AssetSerializer, AssetlogSerializer from apps.asm.serializers import AssetSerializer, AssetLogSerializer, AssetCateSerializer
from apps.wf.models import Ticket from apps.wf.models import Ticket
class AssetAuditViewSet(TicketMixin, CustomModelViewSet):
"""
list: 固定资产审核
固定资产审核 class AssetCateViewSet(CustomModelViewSet):
""" """
queryset = Asset.objects.all() list: 固定资产分类
serializer_class = AssetSerializer
search_fields = ['assetlog']
workflow_key = 'wf_asset'
@staticmethod 固定资产分类
def add_asset(ticket:Ticket, transition, new_ticket_data: dict): """
asset = Asset.objects.get(ticket=ticket) queryset = AssetCate.objects.all()
if AssetLog.objects.filter(code=asset.code).exists(): serializer_class = AssetCateSerializer
raise ParseError('资产名称已存在') search_fields = ['name', 'code']
AssetLog.objects.create(**new_ticket_data)
class AssetViewSet(CustomModelViewSet): class AssetViewSet(CustomModelViewSet):
@ -31,6 +24,20 @@ class AssetViewSet(CustomModelViewSet):
固定资产台账 固定资产台账
""" """
queryset = Asset.objects.all() queryset = Asset.objects.all()
serializer_class = AssetlogSerializer serializer_class = AssetSerializer
search_fields = ['name', 'code', 'category'] search_fields = ['name', 'card_number', 'specification']
filterset_fields = ['category'] filterset_fields = ['keep_dept', 'state']
class AssetLogViewSet(TicketMixin, CustomModelViewSet):
"""
list: 固定资产操作
固定资产操作
"""
queryset = AssetLog.objects.all()
serializer_class = AssetLogSerializer
search_fields = ['assetlog']
workflow_key = 'wf_assetlog'