Skip to content

Commit

Permalink
Preserve existing query params in PaginatorMixin
Browse files Browse the repository at this point in the history
Previously, generation of next/previous links would discard any existing
query parameters. This commit introduces a dependency on URLObject, which
is used to intelligently parse and modify URLs to ensure existing params
are preserved.

Addresses issues #107
  • Loading branch information
j4mie committed Jan 5, 2012
1 parent b745d0c commit 18535c7
Show file tree
Hide file tree
Showing 4 changed files with 17 additions and 2 deletions.
6 changes: 4 additions & 2 deletions djangorestframework/mixins.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
from django.core.paginator import Paginator
from django.db.models.fields.related import ForeignKey
from django.http import HttpResponse
from urlobject import URLObject

from djangorestframework import status
from djangorestframework.renderers import BaseRenderer
Expand Down Expand Up @@ -659,11 +660,12 @@ def get_limit(self):

def url_with_page_number(self, page_number):
""" Constructs a url used for getting the next/previous urls """
url = "%s?page=%d" % (self.request.path, page_number)
url = URLObject.parse(self.request.get_full_path())
url = url.add_query_param('page', page_number)

limit = self.get_limit()
if limit != self.limit:
url = "%s&limit=%d" % (url, limit)
url = url.add_query_param('limit', limit)

return url

Expand Down
11 changes: 11 additions & 0 deletions djangorestframework/tests/mixins.py
Original file line number Diff line number Diff line change
Expand Up @@ -237,3 +237,14 @@ def test_page_range(self):
response = MockPaginatorView.as_view()(request)
content = json.loads(response.content)
self.assertEqual(response.status_code, status.NOT_FOUND)

def test_existing_query_parameters_are_preserved(self):
""" Tests that existing query parameters are preserved when
generating next/previous page links """
request = self.req.get('/paginator/?foo=bar&another=something')
response = MockPaginatorView.as_view()(request)
content = json.loads(response.content)
self.assertEqual(response.status_code, status.OK)
self.assertTrue('foo=bar' in content['next'])
self.assertTrue('another=something' in content['next'])
self.assertTrue('page=2' in content['next'])
1 change: 1 addition & 0 deletions requirements.txt
Original file line number Diff line number Diff line change
Expand Up @@ -3,3 +3,4 @@

Django>=1.2
coverage>=3.4
URLObject>=0.6.0
1 change: 1 addition & 0 deletions setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@
package_dir={'djangorestframework': 'djangorestframework'},
package_data = {'djangorestframework': ['templates/*', 'static/*']},
test_suite = 'djangorestframework.runtests.runcoverage.main',
install_requires=['URLObject>=0.6.0'],
classifiers = [
'Development Status :: 4 - Beta',
'Environment :: Web Environment',
Expand Down

0 comments on commit 18535c7

Please sign in to comment.