filters.py 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. import django_filters
  2. from dcim.forms import MACAddressField
  3. from django import forms
  4. from django.conf import settings
  5. from django.db import models
  6. from extras.models import Tag
  7. def multivalue_field_factory(field_class):
  8. """
  9. Given a form field class, return a subclass capable of accepting multiple values. This allows us to OR on multiple
  10. filter values while maintaining the field's built-in validation. Example: GET /api/dcim/devices/?name=foo&name=bar
  11. """
  12. class NewField(field_class):
  13. widget = forms.SelectMultiple
  14. def to_python(self, value):
  15. if not value:
  16. return []
  17. return [
  18. # Only append non-empty values (this avoids e.g. trying to cast '' as an integer)
  19. super(field_class, self).to_python(v) for v in value if v
  20. ]
  21. return type('MultiValue{}'.format(field_class.__name__), (NewField,), dict())
  22. #
  23. # Filters
  24. #
  25. class MultiValueCharFilter(django_filters.MultipleChoiceFilter):
  26. field_class = multivalue_field_factory(forms.CharField)
  27. class MultiValueDateFilter(django_filters.MultipleChoiceFilter):
  28. field_class = multivalue_field_factory(forms.DateField)
  29. class MultiValueDateTimeFilter(django_filters.MultipleChoiceFilter):
  30. field_class = multivalue_field_factory(forms.DateTimeField)
  31. class MultiValueNumberFilter(django_filters.MultipleChoiceFilter):
  32. field_class = multivalue_field_factory(forms.IntegerField)
  33. class MultiValueTimeFilter(django_filters.MultipleChoiceFilter):
  34. field_class = multivalue_field_factory(forms.TimeField)
  35. class MACAddressFilter(django_filters.CharFilter):
  36. field_class = MACAddressField
  37. class MultiValueMACAddressFilter(django_filters.MultipleChoiceFilter):
  38. field_class = multivalue_field_factory(MACAddressField)
  39. class TreeNodeMultipleChoiceFilter(django_filters.ModelMultipleChoiceFilter):
  40. """
  41. Filters for a set of Models, including all descendant models within a Tree. Example: [<Region: R1>,<Region: R2>]
  42. """
  43. def get_filter_predicate(self, v):
  44. # null value filtering
  45. if v is None:
  46. return {self.field_name.replace('in', 'isnull'): True}
  47. return super().get_filter_predicate(v)
  48. def filter(self, qs, value):
  49. value = [node.get_descendants(include_self=True) if not isinstance(node, str) else node for node in value]
  50. return super().filter(qs, value)
  51. class NumericInFilter(django_filters.BaseInFilter, django_filters.NumberFilter):
  52. """
  53. Filters for a set of numeric values. Example: id__in=100,200,300
  54. """
  55. pass
  56. class NullableCharFieldFilter(django_filters.CharFilter):
  57. """
  58. Allow matching on null field values by passing a special string used to signify NULL.
  59. """
  60. def filter(self, qs, value):
  61. if value != settings.FILTERS_NULL_CHOICE_VALUE:
  62. return super().filter(qs, value)
  63. qs = self.get_method(qs)(**{'{}__isnull'.format(self.field_name): True})
  64. return qs.distinct() if self.distinct else qs
  65. class TagFilter(django_filters.ModelMultipleChoiceFilter):
  66. """
  67. Match on one or more assigned tags. If multiple tags are specified (e.g. ?tag=foo&tag=bar), the queryset is filtered
  68. to objects matching all tags.
  69. """
  70. def __init__(self, *args, **kwargs):
  71. kwargs.setdefault('field_name', 'tags__slug')
  72. kwargs.setdefault('to_field_name', 'slug')
  73. kwargs.setdefault('conjoined', True)
  74. kwargs.setdefault('queryset', Tag.objects.all())
  75. super().__init__(*args, **kwargs)
  76. #
  77. # FilterSets
  78. #
  79. class NameSlugSearchFilterSet(django_filters.FilterSet):
  80. """
  81. A base class for adding the search method to models which only expose the `name` and `slug` fields
  82. """
  83. q = django_filters.CharFilter(
  84. method='search',
  85. label='Search',
  86. )
  87. def search(self, queryset, name, value):
  88. if not value.strip():
  89. return queryset
  90. return queryset.filter(
  91. models.Q(name__icontains=value) |
  92. models.Q(slug__icontains=value)
  93. )
  94. #
  95. # Update default filters
  96. #
  97. FILTER_DEFAULTS = django_filters.filterset.FILTER_FOR_DBFIELD_DEFAULTS
  98. FILTER_DEFAULTS.update({
  99. models.AutoField: {
  100. 'filter_class': MultiValueNumberFilter
  101. },
  102. models.CharField: {
  103. 'filter_class': MultiValueCharFilter
  104. },
  105. models.DateField: {
  106. 'filter_class': MultiValueDateFilter
  107. },
  108. models.DateTimeField: {
  109. 'filter_class': MultiValueDateTimeFilter
  110. },
  111. models.DecimalField: {
  112. 'filter_class': MultiValueNumberFilter
  113. },
  114. models.EmailField: {
  115. 'filter_class': MultiValueCharFilter
  116. },
  117. models.FloatField: {
  118. 'filter_class': MultiValueNumberFilter
  119. },
  120. models.IntegerField: {
  121. 'filter_class': MultiValueNumberFilter
  122. },
  123. models.PositiveIntegerField: {
  124. 'filter_class': MultiValueNumberFilter
  125. },
  126. models.PositiveSmallIntegerField: {
  127. 'filter_class': MultiValueNumberFilter
  128. },
  129. models.SlugField: {
  130. 'filter_class': MultiValueCharFilter
  131. },
  132. models.SmallIntegerField: {
  133. 'filter_class': MultiValueNumberFilter
  134. },
  135. models.TimeField: {
  136. 'filter_class': MultiValueTimeFilter
  137. },
  138. models.URLField: {
  139. 'filter_class': MultiValueCharFilter
  140. },
  141. })