1

是的,这是一项任务,是的,我已经花了一些时间,现在我需要帮助。

我的任务有两个模型,Server它们Client是 1-N 关系。如下所述

# models.py
class Server(models.Model):
    name = models.CharField(max_length=255, unique=True, null=False, blank=False)
    maximum_clients = models.IntegerField(default=1,null=False, blank=False)

class Client(models.Model):
    name = models.CharField(max_length=255, unique=True, null=False, blank=False)
    active = models.BooleanField(default=True)
    server = models.ForeignKey(Server)

我用 ModelForm 创建了一个表单,它允许我在给定的服务器上创建一个新客户端,但任务的先决条件是只提供具有可用容量的服务器(它们maximum_clients小于实际客户端)所以这就是我所做的

#forms.py
from django.db.models import Count

qs = Server.objects.annotate(Count('client'))
server_choices = []
for server in qs:
    if server.client__count < server.maximum_clients:
        server_choices.append((server,server))

class ClientForm(forms.ModelForm):
    name = forms.CharField(label='Client name')
    server = forms.ChoiceField(choices=server_choices)
    class Meta:
        model = Client
        fields = ('name','server',)

select这种方法根据我提到的先决条件填充正确的服务器。但是,保存此表单会产生错误Cannot assign "u'fifty'": "Client.server" must be a "Server" instance. Fifty,例如服务器名称maximum_clients = 50

屏幕上有一个类似的表单admin,我也对其进行了修改以仅显示可用的服务器并在那里保存会产生相同的错误。

4

1 回答 1

2

这不是正确的做法。除了您看到的错误之外,您还会发现 server_choices 仅在您重新启动网络服务器时更新,而不是在服务器对象本身更改时更新。

您有一个外键,需要从相关对象的子集中进行选择。正确的字段是 ModelChoiceField;这需要一个查询集,您可以在定义中对其进行过滤。由于您的过滤器依赖于同一模型中的字段,因此您需要使用一个F对象。

class ClientForm(forms.ModelForm):
    name = forms.CharField(label='Client name')
    server = forms.ModelChoiceField(
        queryset=Server.objects.annotate(client_count=Count('client')).filter(client_count__lt=F('maximum_clients')
    )
于 2015-07-26T13:42:58.033 回答