django modelform 问题请教

2020-04-30 23:22:39 +08:00
 fanne
models.py
class VirtualNode(models.Model):
    VIRTUAL_STATUS = (
        (0, '已停止'),
        (1, '已启动'),
        (2, '正在创建'),
        (3, '创建失败'),
    )
    virtual_id = models.AutoField(verbose_name="虚拟机 ID",primary_key=True)
    virtual_name = models.CharField(max_length=30,verbose_name="虚拟机名")
    virtual_ip = models.GenericIPAddressField(verbose_name="虚拟 IP")
    virtual_cpu = models.IntegerField(verbose_name="虚拟机 cpu(核数)")
    virtual_mem = models.IntegerField(verbose_name="虚拟机内存(GB)")
    virtual_disk = models.IntegerField(verbose_name="虚拟机硬盘(GB)")
    virtual_vmnode = models.ForeignKey(VmNode, verbose_name="所属宿主机")
    virtual_vlan = models.ForeignKey(VlanInfo,verbose_name="所属 Vlan")
    project = models.CharField(max_length=30,verbose_name="所属项目",null=True,blank=True)
    virtual_staus = models.IntegerField(verbose_name='运行状态',choices=VIRTUAL_STATUS,default=0)

    class Meta:
        verbose_name = "虚拟机节点"
        verbose_name_plural = verbose_name
forms.py
class VirtualNodeForm(forms.ModelForm):
    class Meta:
        model = VirtualNode
        fields = '__all__'
vievws.py
class VirtualNodeAddView(LoginRequireMixin, View):
    def get(self, request):
        pass
    def post(self, request):
        virtual_form = VirtualNodeForm(request.POST)
        if virtual_form.is_valid():

            virtual_form.save()
            return JsonResponse({'status':'success'})
        else:
            msg = virtual_form.errors.values()
            
            return JsonResponse({'status':'fail','msg':'虚拟机添加失败:{}'.format(msg)})  

前端的表单只会填 6 个字段

其中 virtual_name 、virtual_ip 会根据某些条件算出后自动保存。

那么问题来了

views.py里面怎么将virtual_name virtual_ip,回填到virtual_form里,否在在下面的校验无法通过。

2748 次点击
所在节点    Django
7 条回复
Hstar
2020-04-30 23:24:41 +08:00
不应该在 view 里处理,看下 form 的文档,定义两个函数 valid_virtual_name 和 valid_visual_ip 来生成数据
Hstar
2020-04-30 23:28:08 +08:00
记错了,应该是两个函数 clean_virtual_name 和 clean_virtual_name , 太久没用 form 这玩意。
看下 https://docs.djangoproject.com/en/3.0/ref/forms/validation/这一章
izzy27
2020-05-01 08:09:13 +08:00
Form 不要选__all__,把 virtual_name 和 virtual_ip 这两个字段排除掉,这样就不会校验这两个字段了
在你对应的 view 视图里面计算出正确的 virtual_name 和 virtual_ip 就行了
fanne
2020-05-01 08:35:49 +08:00
@izzy27 #3 这个方法试过,但在最后的 virtual_form.save()这里,还是要把计算出的 virtual_name,virtual_ip 两个字塞回
virtual_form,否则数据也无法保存起来。现在不知道怎么塞进去。
fanne
2020-05-01 08:46:10 +08:00
@Hstar #2
```
class UserAsk(models.Model):
name = models.CharField(max_length=20,verbose_name=u"用户名")
mobile = models.CharField(max_length=11,verbose_name=u"手机号码")
course_name = models.CharField(max_length=50,verbose_name=u"课程名")
add_time = models.DateTimeField(default=datetime.now)

```


```
import re
from django import forms
from operation.models import UserAsk
class UserAskForm(forms.ModelForm):
class Meta:
model = UserAsk
exclude = ['add_time']

def clean_mobile(self):
mobile = self.cleaned_data.get('mobile')
if re.match(r'^(13\d|14[5|7]|15\d|166|17[3|6|7]|18\d)\d{8}$',mobile):
return mobile
else:
raise forms.ValidationError(u"手机号码非法",code='mobile invalid')
```

是不是类似这种的,可以对某些字段重新赋值,类似 return mobile,就可以把 mobile 字段重新赋值一个,然后返回所需要的 form,传递给 views
izzy27
2020-05-01 09:28:18 +08:00
那你试一下在你的 form 里面重写 clean 函数,先获取父类的 clean 返回结果 cleaned_data,然后计算出 virtual_name 和 virtual_ip,然后 update
类似于这样
def clean(self):
cleaned_data = super(VirtualNodeForm, self).clean()
# 计算 virtual_name 和 virtual_ip
cleaned_data.update(...)
return cleaned_data
fanne
2020-05-06 12:02:11 +08:00
@izzy27 #6 这个 update 里面是更新啥?

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/667782

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX