web-dev-qa-db-ja.com

関連するオブジェクトでカスタムマネージャーを使用する方法

カスタムマネージャーがいます。関連オブジェクトに使用したい。ドキュメントで se_for_related_fields を見つけました。しかし、それは私がそれを使用した方法では機能しません:

class RandomQueryset(models.query.QuerySet):

    def randomize(self):       
        count = self.count()
        random_index = random.randint(0, count - 1)
        return self.all()[random_index]


class RandomManager(models.Manager):

    use_for_related_fields = True

    def get_query_set(self):
        return RandomQueryset(self.model, using=self._db)

    def randomize(self):
        return self.get_query_set().randomize()

私はそれを1つのモデルに使用しました:

>>> post = PostPages.default_manager.filter(image_gallery__isnull=False).distinct().randomize()

そしてm2m関連オブジェクトで同じことをしようとしました:

>>> post.image_gallery.randomize()

エラーが発生しました:

AttributeError: 'ManyRelatedManager' object has no attribute 'randomize'

私のやり方でカスタムマネージャーを使用することは可能ですか?もしそうなら、それをどのように機能させるのですか?

編集する

私のモデル:

class ShivaImage(models.Model, ImageResizing):
    image = models.ImageField(upload_to='img')
    slide_show = models.BooleanField() 
    title = models.CharField(max_length=100)
    text = models.TextField(max_length=400)
    ordering = models.IntegerField(blank=True, null=True)

    objects = RandomManager()


class PostPages(models.Model):
    image_gallery = models.ManyToManyField(ShivaImage, blank=True,
                                       related_name='gallery',)
    # all the other fields... 

    objects = RandomManager()
42
I159

マネージャーでuse_for_related_fieldsTrueに設定すると、定義したモデルを指すからのすべての関係で使用可能になりますデフォルトのマネージャーとしてのこのマネージャー。これは文書化されています ここ

class MyManager(models.Manager):
    use_for_related_fields = True
    # ...

PostPagesモデル(またはpost_image_galleryを通じて参照されるモデルと呼ばれるもの)ではなく、Galleryモデルでのみ有効になっていると思います。この不動産マネージャーに追加の機能を持たせたい場合は、Galleryモデルにuse_for_related_fields = Trueのカスタムデフォルトマネージャーを追加する必要があります。

26

トピックを完全にするために、Django 1.7(最終的に)は カスタムリバースマネージャーを使用して をサポートしているので、そのようなことを行うことができます(Django docs):

from Django.db import models

class Entry(models.Model):
    objects = models.Manager()  # Default Manager
    entries = EntryManager()    # Custom Manager

b = Blog.objects.get(id=1)
b.entry_set(manager='entries').all()
34
Serafeim

Django 2.0 use_for_related_fieldsは廃止予定です https://docs.djangoproject.com/en/2.0/releases/1.10/#manager-use-for-related- fields-and-inheritance-changes

base_manager_nameを使用する必要があります: https://docs.djangoproject.com/en/2.0/ref/models/options/#Django.db.models.Options.base_manager_name

更新されたドキュメント: https://docs.djangoproject.com/en/2.0/topics/db/managers/#using-managers-for-related-object-access

class MyModel(models.Model):
    field1 = ...
    field2 = ...
    special_manager = MyManager()

    class Meta:
        base_manager_name = 'special_manager'
11

また、カスタムマネージャーでは、関連するマネージャーから呼び出されるカスタムフィルターを実装するときに、self.get_query_set()プロキシメソッドを介してクエリセットにアクセスしてください。

class EventManager(models.Manager):

    use_for_related_fields = True

    def visible_events(self):
        today = datetime.date.today()
        # don't do this !!! 
        # unsuitable for related managers as could retrieve extraneous objects
        # qs = super(EventManager, self).get_query_set()
        # Use queryset proxy method as follows, instead:
        qs = self.get_query_set()
        qs = qs.filter(visible_from__lte=today, visible_to__gte=today)
        return qs


class Event(models.Model):

    visible_from = models.DateField(_(u'visible from'), null=False, blank=False)
    visible_to = models.DateField(_(u'visible to'), null=False, blank=False)
    concepts = models.ManyToManyField(Concept, through='ConceptEventRegistration')

    objects = EventManager()

使用例:

my_concept = Concept.objects.get(id=1)
# retrieve all events related to the object
my_concept.event_set.all()
# retrieve all visible events related to the object
my_concept.event_set.visible_events()
2
Mario Orlandi