在 django 信号中,如何将发送方的类变量模型字段传递给接收方management.call_command 函数并使用它

问题描述

我正在使用第三方包 (django-tenants) 构建一个多租户项目,这使我能够从它自己的类中派生租户和域模型:

from django_tenants.models import TenantMixin,DomainMixin

class Tenant(TenantMixin):
    name = models.CharField(max_length=100)
    email = models.CharField(max_length=100)
    created_on = models.DateField(auto_Now_add=True)
    auto_create_schema = True
    auto_drop_shema = True

class Domain(DomainMixin):
    pass;

长话短说,我想创建一个 API 视图来注册租户,但我无法理解包代码并且在尝试创建 TenantManager 时失败;但我确实弄清楚了如何使用 django.core.management.call_commands 从代码内部调用 django-tenants (python manage.py create_tenant/create_tenant_superuser) 提供的 CLI 管理命令:

management.call_command(
    'create_tenant',domain_domain = 'tenant1.mysite.com',schema_name = 'tenant1Schema',name ='tenant1Name',email = 't1@t1.com',domain_is_primary = True,)

所以我想出了一个临时简化的解决方案 - 另一个模型(创始人)用常规模型制作。模型和它自己的 genericAPIview,我知道如何创建。发布此模型后,我想对我的自定义 management.call_command 执行信号。

据我了解,django 信号分为三个部分:

  • sender - 这应该是我的创始人模型中的“notify_about_sender_creation”方法
  • receiver - 这应该是一个调用的对象,在我的例子中是这个自定义 management.call_command
  • 信号 - 我不知道这应该是 post_save 信号还是自定义信号?我可以只向自定义变量提供变量吗?

基于这种理解,我得到了:

class Founder(models.Model):
    first_name = models.CharField(max_length=200)
    last_name = models.CharField(max_length=200)
    email = models.CharField(max_length=200,unique=True)
    tenant_name = models.CharField(max_length=200)
    domain_name = models.CharField(max_length=200)

    def generate(self):
        data = (Founder.first_name,Founder.last_name,Founder.email,Founder.tenant_name,Founder.domain_name)
        save_founder(sender=self,data=data)

save_founder = Signal(providing_args=['first_name','last_name','email','tenant_name','domain_name'])

@receiver(save_founder) def generate_tenant(sender,first_name,last_name,email,tenant_name,domain_name,**kwargs):
    management.call_command(
        'create_tenant',domain_domain = sender.domain_name,schema_name = sender.tenant_name,name = sender.tenant_name,email = sender.email,) save_founder.connect(generate_tenant)

但在我的控制台中我得到:

python manage.py shell
from tenant.models import Founder
f = Founder()
f.generate()
Traceback (most recent call last):
  File "<console>",line 1,in <module>
  File "/app/tenant/models.py",line 26,in generate
TypeError: 'Signal' object is not callable

解决方法

需要的是将实例传递给接收器函数。信号不应该是自定义的。我不知道我是否必须让我的接收器函数成为 Founder 类的方法,但这并没有什么坏处。 这是有效的:

class Founder(models.Model):
    tenant_name = models.CharField(max_length=200)
    domain_name = models.CharField(max_length=200)
    ...
    def generate(sender,instance,**kwargs):
        founder=instance
        management.call_command(
            'create_tenant',domain_domain = instance.domain_name + ".mysite.com",name = instance.tenant_name,...
            )
post_save.connect(Founder.generate,sender=Founder) 

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...