|
|
|
from django.contrib.contenttypes.fields import GenericRelation
|
|
|
|
from django.contrib.contenttypes.models import ContentType
|
|
|
|
from django.db import models
|
|
|
|
from django.db.models import OneToOneField, Count
|
|
|
|
|
|
|
|
from common.utils import lazyproperty
|
|
|
|
from .models import LabeledResource
|
|
|
|
|
|
|
|
__all__ = ['LabeledMixin']
|
|
|
|
|
|
|
|
|
|
|
|
class LabeledMixin(models.Model):
|
|
|
|
labels = GenericRelation(LabeledResource, object_id_field='res_id', content_type_field='res_type')
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
abstract = True
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def label_model(cls):
|
|
|
|
pk_field = cls._meta.pk
|
|
|
|
model = cls
|
|
|
|
if isinstance(pk_field, OneToOneField):
|
|
|
|
model = pk_field.related_model
|
|
|
|
return model
|
|
|
|
|
|
|
|
@lazyproperty
|
|
|
|
def real(self):
|
|
|
|
pk_field = self._meta.pk
|
|
|
|
if isinstance(pk_field, OneToOneField):
|
|
|
|
return getattr(self, pk_field.name)
|
|
|
|
return self
|
|
|
|
|
|
|
|
@property
|
|
|
|
def res_labels(self):
|
|
|
|
return self.real.labels
|
|
|
|
|
|
|
|
@res_labels.setter
|
|
|
|
def res_labels(self, value):
|
|
|
|
self.real.labels.set(value, bulk=False)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def filter_resources_by_labels(cls, resources, label_ids, rel='all'):
|
|
|
|
if rel == 'all':
|
|
|
|
return cls._get_filter_res_by_labels_m2m_all(resources, label_ids)
|
|
|
|
else:
|
|
|
|
return cls._get_filter_res_by_labels_m2m_in(resources, label_ids)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def _get_filter_res_by_labels_m2m_in(cls, resources, label_ids):
|
|
|
|
return resources.filter(label_id__in=label_ids)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def _get_filter_res_by_labels_m2m_all(cls, resources, label_ids):
|
|
|
|
if len(label_ids) == 1:
|
|
|
|
return cls._get_filter_res_by_labels_m2m_in(resources, label_ids)
|
|
|
|
|
|
|
|
resources = resources.filter(label_id__in=label_ids) \
|
|
|
|
.values('res_id') \
|
|
|
|
.order_by('res_id') \
|
|
|
|
.annotate(count=Count('res_id')) \
|
|
|
|
.values('res_id', 'count') \
|
|
|
|
.filter(count=len(label_ids))
|
|
|
|
return resources
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def get_labels_filter_attr_q(cls, value, match):
|
|
|
|
res_type = ContentType.objects.get_for_model(cls.label_model())
|
|
|
|
resources = LabeledResource.objects.all().filter(res_type=res_type)
|
|
|
|
if not value:
|
|
|
|
return None
|
|
|
|
|
|
|
|
if match != 'm2m_all':
|
|
|
|
resources = cls._get_filter_res_by_labels_m2m_in(resources, value)
|
|
|
|
else:
|
|
|
|
resources = cls._get_filter_res_by_labels_m2m_all(resources, value)
|
|
|
|
res_ids = set(resources.values_list('res_id', flat=True))
|
|
|
|
return models.Q(id__in=res_ids)
|