Source code for django_elasticsearch_dsl_drf.tests.test_filtering

"""
Test filtering backend.
"""

from __future__ import absolute_import

import unittest
import uuid

from django.core.management import call_command

from nine.versions import DJANGO_GTE_1_10

import pytest

from rest_framework import status

from books import constants
import factories

from .base import BaseRestFrameworkTestCase

if DJANGO_GTE_1_10:
    from django.urls import reverse
else:
    from django.core.urlresolvers import reverse

__title__ = 'django_elasticsearch_dsl_drf.tests.test_filtering'
__author__ = 'Artur Barseghyan <artur.barseghyan@gmail.com>'
__copyright__ = '2017 Artur Barseghyan'
__license__ = 'GPL 2.0/LGPL 2.1'
__all__ = (
    'TestFiltering',
)


@pytest.mark.django_db
[docs]class TestFiltering(BaseRestFrameworkTestCase): """Test filtering.""" pytestmark = pytest.mark.django_db @classmethod
[docs] def setUpClass(cls): """Set up.""" # Counts are primarily taken into consideration. Don't create Book # objects without `state`. If you don't know which state to use, use # ``constants.BOOK_PUBLISHING_STATUS_REJECTED``. cls.published_count = 10 cls.published = factories.BookFactory.create_batch( cls.published_count, **{ 'state': constants.BOOK_PUBLISHING_STATUS_PUBLISHED, } ) cls.in_progress_count = 10 cls.in_progress = factories.BookFactory.create_batch( cls.in_progress_count, **{ 'state': constants.BOOK_PUBLISHING_STATUS_IN_PROGRESS, } ) cls.prefix_count = 2 cls.prefix = 'DelusionalInsanity' cls.prefixed = factories.BookFactory.create_batch( cls.prefix_count, **{ 'title': '{} {}'.format(cls.prefix, uuid.uuid4()), 'state': constants.BOOK_PUBLISHING_STATUS_REJECTED, } ) cls.no_tags_count = 5 cls.no_tags = factories.BookWithoutTagsFactory.create_batch( cls.no_tags_count, **{ 'state': constants.BOOK_PUBLISHING_STATUS_REJECTED, } ) cls.geo_origin = factories.BookWithoutTagsFactory.create( **{ 'state': constants.BOOK_PUBLISHING_STATUS_REJECTED, 'lat': 48.8549, 'lon': 2.3000, } ) cls.geo_in_count = 5 cls.geo_distance = '1km' cls.geo_in = factories.BookWithoutTagsFactory.create_batch( cls.geo_in_count, **{ 'state': constants.BOOK_PUBLISHING_STATUS_REJECTED, 'lat': 48.8570, 'lon': 2.3005, } ) cls.all_count = ( cls.published_count + cls.in_progress_count + cls.prefix_count + cls.no_tags_count + cls.geo_in_count + 1 # geo_origin ) cls.base_url = reverse('bookdocument-list', kwargs={}) cls.base_publisher_url = reverse('publisherdocument-list', kwargs={}) call_command('search_index', '--rebuild', '-f')
def _field_filter_value(self, field_name, value, count): """Field filter value. Usage example: >>> self._field_filter_value( >>> 'title__wildcard', >>> self.prefix[3:-3], >>> self.prefix_count >>> ) """ url = self.base_url[:] data = {} response = self.client.get( url + '?{}={}'.format(field_name, value), data ) self.assertEqual(response.status_code, status.HTTP_200_OK) self.assertEqual( len(response.data['results']), count ) def _field_filter_term(self, field_name, filter_value): """Field filter term. Example: http://localhost:8000/api/articles/?tags=children """ self.authenticate() url = self.base_url[:] data = {} # Should contain 22 results response = self.client.get(url, data) self.assertEqual(response.status_code, status.HTTP_200_OK) self.assertEqual(len(response.data['results']), self.all_count) # Should contain only 10 results filtered_response = self.client.get( url + '?{}={}'.format(field_name, filter_value), data ) self.assertEqual(filtered_response.status_code, status.HTTP_200_OK) self.assertEqual( len(filtered_response.data['results']), self.published_count ) @pytest.mark.webtest
[docs] def test_field_filter_geo_distance(self): """Field filter term. Example: http://localhost:8000/api/books/?location__geo_distance=1km|48.8549|2.3000 """ self.authenticate() __params = '{}|{}|{}'.format(self.geo_distance, self.geo_origin.lat, self.geo_origin.lon) url = self.base_url[:] + '?{}={}'.format('location__geo_distance', __params) data = {} response = self.client.get(url, data) self.assertEqual(response.status_code, status.HTTP_200_OK) # Should contain only 6 results self.assertEqual(len(response.data['results']), self.geo_in_count + 1)
[docs] def test_field_filter_term(self): """Field filter term.""" return self._field_filter_term( 'state', constants.BOOK_PUBLISHING_STATUS_PUBLISHED )
[docs] def test_field_filter_term_explicit(self): """Field filter term.""" return self._field_filter_term( 'state__term', constants.BOOK_PUBLISHING_STATUS_PUBLISHED )
[docs] def test_field_filter_range(self): """Field filter range. Example: http://localhost:8000/api/users/?age__range=16|67 """ lower_id = self.published[0].id upper_id = self.published[-1].id value = '{}|{}'.format(lower_id, upper_id) return self._field_filter_value( 'id__range', value, self.published_count )
[docs] def test_field_filter_range_with_boost(self): """Field filter range. Example: http://localhost:8000/api/users/?age__range=16|67|2.0 """ lower_id = self.published[0].id upper_id = self.published[-1].id value = '{}|{}|{}'.format(lower_id, upper_id, '2.0') return self._field_filter_value( 'id__range', value, self.published_count )
[docs] def test_field_filter_prefix(self): """Test filter prefix. Example: http://localhost:8000/api/articles/?tags__prefix=bio """ return self._field_filter_value( 'title__prefix', self.prefix, self.prefix_count )
[docs] def test_field_filter_in(self): """Test filter in. Example: http://localhost:8000/api/articles/?id__in=1|2|3 """ return self._field_filter_value( 'id__in', '|'.join([str(__b.id) for __b in self.prefixed]), self.prefix_count )
def _field_filter_terms_list(self, field_name, in_values, count): """Field filter terms. Example: http://localhost:8000/api/articles/?id=1&id=2&id=3 """ url = self.base_url[:] data = {} url_parts = ['{}={}'.format(field_name, val) for val in in_values] response = self.client.get( url + '?{}'.format('&'.join(url_parts)), data ) self.assertEqual(response.status_code, status.HTTP_200_OK) self.assertEqual( len(response.data['results']), count )
[docs] def test_field_filter_terms_list(self): """Test filter terms.""" return self._field_filter_terms_list( 'id', [str(__b.id) for __b in self.prefixed], self.prefix_count )
[docs] def test_field_filter_terms_string(self): """Test filter terms. Example: http://localhost:8000/api/articles/?id__terms=1|2|3 """ return self._field_filter_value( 'id__terms', '|'.join([str(__b.id) for __b in self.prefixed]), self.prefix_count )
[docs] def test_field_filter_exists_true(self): """Test filter exists true. Example: http://localhost:8000/api/articles/?tags__exists=true """ return self._field_filter_value( 'tags__exists', 'true', self.all_count )
[docs] def test_field_filter_exists_false(self): """Test filter exists. Example: http://localhost:8000/api/articles/?non_existent__exists=false """ return self._field_filter_value( 'non_existent_field__exists', 'false', self.all_count )
[docs] def test_field_filter_wildcard(self): """Test filter wildcard. Example: http://localhost:8000/api/articles/?title__wildcard=*elusional* """ return self._field_filter_value( 'title__wildcard', '*{}*'.format(self.prefix[1:6]), self.prefix_count )
[docs] def test_field_filter_exclude(self): """Test filter exclude. Example: http://localhost:8000/api/articles/?tags__exclude=children """ return self._field_filter_value( 'state__exclude', constants.BOOK_PUBLISHING_STATUS_PUBLISHED, self.all_count - self.published_count )
[docs] def test_field_filter_isnull_true(self): """Test filter isnull true. Example: http://localhost:8000/api/articles/?null_field__isnull=true """ self._field_filter_value( 'null_field__isnull', 'true', self.all_count ) self._field_filter_value( 'tags__isnull', 'true', 0 )
[docs] def test_field_filter_isnull_false(self): """Test filter isnull true. Example: http://localhost:8000/api/articles/?tags__isnull=false """ self._field_filter_value( 'null_field__isnull', 'false', 0 ) self._field_filter_value( 'tags__isnull', 'false', self.all_count )
[docs] def test_field_filter_endswith(self): """Test filter endswith. Example: http://localhost:8000/api/articles/?state__endswith=lished """ return self._field_filter_value( 'state__endswith', constants.BOOK_PUBLISHING_STATUS_PUBLISHED[4:], self.published_count )
[docs] def test_field_filter_contains(self): """Test filter contains. Example: http://localhost:8000/api/articles/?state__contains=lishe """ return self._field_filter_value( 'state__contains', constants.BOOK_PUBLISHING_STATUS_PUBLISHED[4:-2], self.published_count )
def _field_filter_gte_lte(self, field_name, value, lookup, boost=None): """Field filter gt/gte/lt/lte. Example: http://localhost:8000/api/users/?id__gt=10 http://localhost:8000/api/users/?id__gte=10 http://localhost:8000/api/users/?id__lt=10 http://localhost:8000/api/users/?id__lte=10 """ url = self.base_url[:] data = {} if boost is not None: url += '?{}__{}={}|{}'.format(field_name, lookup, value, boost) else: url += '?{}__{}={}'.format(field_name, lookup, value) response = self.client.get( url, data ) self.assertEqual(response.status_code, status.HTTP_200_OK) __mapping = { 'gt': self.assertGreater, 'gte': self.assertGreaterEqual, 'lt': self.assertLess, 'lte': self.assertLessEqual, } __func = __mapping.get(lookup) if callable(__func): for obj in response.data['results']: __func( obj['id'], value )
[docs] def test_field_filter_gt(self): """Field filter gt. Example: http://localhost:8000/api/users/?id__gt=10 :return: """ return self._field_filter_gte_lte('id', self.in_progress[0].id, 'gt')
[docs] def test_field_filter_gt_with_boost(self): """Field filter gt with boost. Example: http://localhost:8000/api/users/?id__gt=10|2.0 :return: """ # TODO: check boost value return self._field_filter_gte_lte( 'id', self.in_progress[0].id, 'gt', '2.0' )
[docs] def test_field_filter_gte(self): """Field filter gte. Example: http://localhost:8000/api/users/?id__gte=10 :return: """ return self._field_filter_gte_lte('id', self.in_progress[0].id, 'gte')
[docs] def test_field_filter_lt(self): """Field filter lt. Example: http://localhost:8000/api/users/?id__lt=10 :return: """ return self._field_filter_gte_lte('id', self.in_progress[0].id, 'lt')
[docs] def test_field_filter_lt_with_boost(self): """Field filter lt with boost. Example: http://localhost:8000/api/users/?id__lt=10|2.0 :return: """ # TODO: check boost value return self._field_filter_gte_lte( 'id', self.in_progress[0].id, 'lt', '2.0' )
[docs] def test_field_filter_lte(self): """Field filter lte. Example: http://localhost:8000/api/users/?id__lte=10 :return: """ return self._field_filter_gte_lte('id', self.in_progress[0].id, 'lte')
[docs] def test_ids_filter(self): """Test ids filter. Example: http://localhost:8000/api/articles/?ids=68|64|58 http://localhost:8000/api/articles/?ids=68&ids=64&ids=58 """ __ids = [str(__obj.id) for __obj in self.published] return self._field_filter_value( 'ids', '|'.join(__ids), self.published_count )
if __name__ == '__main__': unittest.main()