Django 프로젝트에서 내 signal.py 파일을 보관할 올바른 위치


88

내가 읽은 Django의 문서 signals.py에 따르면 앱 폴더에서 시작하기에 좋은 곳인 것 같지만 내가 직면 한 문제는 신호를 생성 pre_save하고 모델에서 클래스를 가져 오려고 할 때 import내 모델에서.

# models.py

from django.contrib.auth.models import User
from django.db import models
from django.utils.translation import gettext as _
from signals import *

class Comm_Queue(CommunicatorAbstract):
    queue_statuses = (
        ('P', _('Pending')),
        ('S', _('Sent')),
        ('E', _('Error')),
        ('R', _('Rejected')),
    )
    status          = models.CharField(max_length=10, db_index=True, default='P')
    is_html         = models.BooleanField(default=False)
    language        = models.CharField(max_length=6, choices=settings.LANGUAGES)
    sender_email    = models.EmailField()
    recipient_email = models.EmailField()
    subject         = models.CharField(max_length=100)
    content         = models.TextField()

# signals.py

from django.conf import settings
from django.db.models.signals import pre_save
from django.dispatch import receiver
from models import Comm_Queue

@receiver(pre_save, sender=Comm_Queue)
def get_sender_email_from_settings(sender, **kwargs):
    obj=kwargs['instance']
    if not obj.sender_email:
        obj.sender_email='%s' % settings.ADMINS[0][1]

이 코드는 Comm_Queue내부를 가져오고 내부 signals.py신호도 가져 오기 때문에 실행되지 않습니다 models.py.

이 문제를 어떻게 극복 할 수 있는지에 대한 조언을받을 수 있습니까?

문안 인사


답변:


65

Django <1.7에 대한 원래 답변 :

signals.py앱의 __init__.py파일 에서 가져 와서 신호를 등록 할 수 있습니다 .

# __init__.py
import signals

이렇게하면 순환 가져 오기 오류없이 models.py에서 signals.py가져올 수 있습니다.

이 접근 방식의 한 가지 문제점은 coverage.py를 사용하는 경우 적용 범위 결과가 엉망이라는 것입니다.

관련 토론

편집 : Django> = 1.7 :

AppConfig가 도입 된 이후로 권장되는 신호 가져 오기 방법은 init()기능입니다. 자세한 내용은 Eric Marcos의 답변 을 참조하십시오.


6
Django 1.9에서 신호를 사용하여 아래 방법을 사용하십시오 (django에서 권장). 이 방법은 포기 작동하지 않습니다AppRegistryNotReady("Apps aren't loaded yet.")
s0nskar

1
: 에릭 마르코스는 그의 대답은 허용 대답해야 stackoverflow.com/a/21612050/3202958 응용 프로그램의 설정 사용, 장고> = 1.7 이후
Nrzonline

1
동의합니다. Django 1.7+에 대한 Eric Marcos의 답변을 가리 키도록 답변을 편집하겠습니다
yprez

194

Django <= 1.6을 사용하는 경우 Kamagatos 솔루션을 권장합니다. 모델 모듈의 끝에 신호를 가져옵니다.

Django의 향후 버전 (> = 1.7)의 경우 권장되는 방법은 앱의 config ready () 함수 에서 신호 모듈을 가져 오는 것입니다 .

my_app/apps.py

from django.apps import AppConfig

class MyAppConfig(AppConfig):
    name = 'my_app'

    def ready(self):
        import my_app.signals

my_app/__init__.py

default_app_config = 'my_app.apps.MyAppConfig'

7
또한 1.7 문서에서는 때때로 준비가 여러 번 호출 될 수 있으므로 신호가 중복되지 않도록 고유 식별자를 신호 커넥터 호출에 첨부합니다. request_finished.connect (my_callback, dispatch_uid = "my_unique_identifier") 여기서 dispatch_uid는 일반적으로 문자열입니다. 그러나 모든 해시 가능한 객체가 될 수 있습니다. docs.djangoproject.com/en/1.7/topics/signals/…
Emeka

13
이것은 받아 들여진 대답이어야합니다! uwsgi를 사용하여 배포 할 때 허용 대답은 위의 오류가 발생합니다
패트릭

2
흠, django 2에서는 작동하지 않습니다. 모델을 준비 상태로 직접 가져 오면 괜찮습니다. 신호에서 모델을 가져오고 준비된 신호를 가져 doesn't declare an explicit app_label
오면

@Aldarun 당신은 INSTALLED_APPS 안에 'my_app.apps.MyAppConfig'를 넣을 수 있습니다.
Ramil Aglyautdinov

26

문제를 해결하려면 모델 정의 후에 신호를 가져 오기만하면됩니다. 그게 다야.


2
이것이 가장 쉬운 방법이며 순환 종속성 없이는 이것이 작동 할 줄은 몰랐습니다. 감사!
bradenm

2
훌륭한. 이게 내 대답보다 낫다. 나는 그것이 순환 수입을 일으키지 않는 이유를 정말로 이해하지
못하지만

솔루션은 Eclipse에서 활성화 된 autopep8 플러그인으로 작동하지 않습니다.
ramusus

5

또한 signal.py 파일에 신호를 넣고 모든 신호를로드하는이 코드 스 니펫도 있습니다.

# import this in url.py file !

import logging

from importlib import import_module

from django.conf import settings

logger = logging.getLogger(__name__)

signal_modules = {}

for app in settings.INSTALLED_APPS:
    signals_module = '%s.signals' % app
    try:
        logger.debug('loading "%s" ..' % signals_module)
        signal_modules[app] = import_module(signals_module)
    except ImportError as e:
        logger.warning(
            'failed to import "%s", reason: %s' % (signals_module, str(e)))

이것은 프로젝트 용이며 앱 수준에서 작동하는지 확실하지 않습니다.


이것은 (tasks.py 같은) 다른 패턴에 맞는로 지금까지 내가 좋아하는 솔루션입니다
dalore

1
당신이 urls.py 수입되지 않습니다 쉘 시작하고 신호를 연결하지 않을 경우,이 하나에 문제가 발견
dalore

예, 내 대답은 구식입니다. 요즘 django에 AppConfig 클래스가있는 것 같습니다. 마지막으로 django를 사용했을 때 버전 1.3이었습니다. 주변 조사를 제안합니다.
aisbaa 2014 년

1
우리는 여전히 1.6이고 그래서 모든 signal.py를 모델로 옮겨야했습니다. 그렇지 않으면 셀러리와 관리 명령이 선택되지 않았습니다
dalore

5

이전 장고 버전에서의 신호를 넣어 잘 될 것 __init__.py에 어쩌면 나models.py 좋습니다 (최종 모델은 내 취향에 따라 크게 될 것입니다).

Django 1.9에서는 신호를 signals.py파일 에 배치 하고apps.py 하고 모델을로드 한 후로드 될 위치에를 .

apps.py :

from django.apps import AppConfig


class PollsConfig(AppConfig):
    name = 'polls'

    def ready(self):
        from . import signals  # NOQA

당신은 또한 당신의 신호를 나눌 수 있습니다 signals.pyhandlers.py라는 이름의 모델 내의 다른 폴더에 signals뿐만 아니라,하지만 나를 위해 그것은 단지 엔지니어링을 통해입니다. 신호 배치 살펴보기


3

나는 당신이 그렇게하여 신호가 등록되어 어딘가에서 찾을 수 있다고 생각합니다. 정상적으로 models.py 파일에 신호를 넣습니다.


예, 모델 파일 내부로 신호를 이동하면 문제가 해결됩니다. 그러나 내 model.py 파일은 모든 클래스, 관리자 및 모델 규칙으로 매우 큽니다.
Mo J. Mughrabi

1
관리자는 내 경험에서 좀 더 쉽게 끌어낼 수 있습니다. Managers.py ftw.
Issac Kelly

3

이것은 별도의 신호가있는 경우에만 적용됩니다. signals.py 파일에

@EricMarcos의 답변에 완전히 동의하지만 django 문서 는 default_app_config 변수를 사용하지 말라고 명시 적으로 조언 한다고 명시해야 합니다 (잘못된 것은 아닙니다). 현재 버전의 경우 올바른 방법은 다음과 같습니다.

my_app / apps.py

from django.apps import AppConfig

class MyAppConfig(AppConfig):
    name = 'my_app'

    def ready(self):
        import my_app.signals

settings.py

(설치된 앱에 앱 이름 만있는 것이 아니라 AppConfig에 대한 상대 경로가 있는지 확인하십시오)

INSTALLED_APPS = [
    'my_app.apps.MyAppConfig',
    # ...
]

1

대안은 콜백 함수를 가져 와서 signals.py연결하는 것입니다.models.py .

signal.py

def pre_save_callback_function(sender, instance, **kwargs):
    # Do stuff here

model.py

# Your imports here
from django.db.models.signals import pre_save
from yourapp.signals import pre_save_callback_function

class YourModel:
    # Model stuff here
pre_save.connect(pre_save_callback_function, sender=YourModel)

추신 : 가져 오기 YourModel에서 signals.py재귀를 만들 것이다; 사용하다sender .

Ps2 : 콜백 함수에 인스턴스를 다시 저장하면 재귀가 생성됩니다. .save메서드 에서 제어 인수를 만들어 제어 할 수 있습니다 .

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.