Skip to content

Commit fc40a65

Browse files
committed
Fixed django#17159 -- Validated returned number of next|previous_page_number
Thanks mehta.apurva at gmail.com for the report and the initial patch and neaf for the complete patch.
1 parent ef906b1 commit fc40a65

File tree

4 files changed

+35
-19
lines changed

4 files changed

+35
-19
lines changed

django/core/paginator.py

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -132,10 +132,10 @@ def has_other_pages(self):
132132
return self.has_previous() or self.has_next()
133133

134134
def next_page_number(self):
135-
return self.number + 1
135+
return self.paginator.validate_number(self.number + 1)
136136

137137
def previous_page_number(self):
138-
return self.number - 1
138+
return self.paginator.validate_number(self.number - 1)
139139

140140
def start_index(self):
141141
"""

docs/releases/1.5.txt

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -154,6 +154,16 @@ Unicode parameters (``password``, ``salt`` or ``encoded``). If any of the
154154
hashing methods need byte strings, you can use the
155155
:func:`~django.utils.encoding.smart_str` utility to encode the strings.
156156

157+
Validation of previous_page_number and next_page_number
158+
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
159+
160+
When using :doc:`object pagination </topics/pagination>`,
161+
the ``previous_page_number()`` and ``next_page_number()`` methods of the
162+
:class:`~django.core.paginator.Page` object did not check if the returned
163+
number was inside the existing page range.
164+
It does check it now and raises an :exc:`InvalidPage` exception when the number
165+
is either too low or too high.
166+
157167
Features deprecated in 1.5
158168
==========================
159169

docs/topics/pagination.txt

Lines changed: 10 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -253,13 +253,19 @@ Methods
253253

254254
.. method:: Page.next_page_number()
255255

256-
Returns the next page number. Note that this is "dumb" and will return the
257-
next page number regardless of whether a subsequent page exists.
256+
Returns the next page number.
257+
258+
.. versionchanged:: 1.5
259+
260+
Raises :exc:`InvalidPage` if next page doesn't exist.
258261

259262
.. method:: Page.previous_page_number()
260263

261-
Returns the previous page number. Note that this is "dumb" and will return
262-
the previous page number regardless of whether a previous page exists.
264+
Returns the previous page number.
265+
266+
.. versionchanged:: 1.5
267+
268+
Raises :exc:`InvalidPage` if previous page doesn't exist.
263269

264270
.. method:: Page.start_index()
265271

tests/modeltests/pagination/tests.py

Lines changed: 13 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,7 @@ def test_first_page(self):
4545
self.assertFalse(p.has_previous())
4646
self.assertTrue(p.has_other_pages())
4747
self.assertEqual(2, p.next_page_number())
48-
self.assertEqual(0, p.previous_page_number())
48+
self.assertRaises(InvalidPage, p.previous_page_number)
4949
self.assertEqual(1, p.start_index())
5050
self.assertEqual(5, p.end_index())
5151

@@ -63,7 +63,7 @@ def test_last_page(self):
6363
self.assertFalse(p.has_next())
6464
self.assertTrue(p.has_previous())
6565
self.assertTrue(p.has_other_pages())
66-
self.assertEqual(3, p.next_page_number())
66+
self.assertRaises(InvalidPage, p.next_page_number)
6767
self.assertEqual(1, p.previous_page_number())
6868
self.assertEqual(6, p.start_index())
6969
self.assertEqual(9, p.end_index())
@@ -104,20 +104,20 @@ def test_orphans(self):
104104

105105
def test_paginate_list(self):
106106
# Paginators work with regular lists/tuples, too -- not just with QuerySets.
107-
paginator = Paginator([1, 2, 3, 4, 5, 6, 7, 8, 9], 5)
107+
paginator = Paginator([1, 2, 3, 4, 5, 6, 7, 8, 9], 3)
108108
self.assertEqual(9, paginator.count)
109-
self.assertEqual(2, paginator.num_pages)
110-
self.assertEqual([1, 2], paginator.page_range)
111-
p = paginator.page(1)
112-
self.assertEqual("<Page 1 of 2>", unicode(p))
113-
self.assertEqual([1, 2, 3, 4, 5], p.object_list)
109+
self.assertEqual(3, paginator.num_pages)
110+
self.assertEqual([1, 2, 3], paginator.page_range)
111+
p = paginator.page(2)
112+
self.assertEqual("<Page 2 of 3>", unicode(p))
113+
self.assertEqual([4, 5, 6], p.object_list)
114114
self.assertTrue(p.has_next())
115-
self.assertFalse(p.has_previous())
115+
self.assertTrue(p.has_previous())
116116
self.assertTrue(p.has_other_pages())
117-
self.assertEqual(2, p.next_page_number())
118-
self.assertEqual(0, p.previous_page_number())
119-
self.assertEqual(1, p.start_index())
120-
self.assertEqual(5, p.end_index())
117+
self.assertEqual(3, p.next_page_number())
118+
self.assertEqual(1, p.previous_page_number())
119+
self.assertEqual(4, p.start_index())
120+
self.assertEqual(6, p.end_index())
121121

122122
def test_paginate_misc_classes(self):
123123
# Paginator can be passed other objects with a count() method.

0 commit comments

Comments
 (0)