Opened 11 months ago

Closed 8 months ago

Last modified 4 months ago

#35945 closed New feature (fixed)

Add async interface to Paginator

Reported by: smiling-watermelon Owned by: wookkl
Component: Core (Other) Version: 5.1
Severity: Normal Keywords: Paginator, async, SynchronousOnlyOperation
Cc: Jon Janzen Triage Stage: Ready for checkin
Has patch: yes Needs documentation: no
Needs tests: no Patch needs improvement: no
Easy pickings: no UI/UX: no

Description

If you provide a QuerySet into a paginator from an async context, the Paginator cannot retrieve count property of the QuerySet resulting in an error.
This can be avoided by making an async version of page method that would use an async version of count property that would refer to acount method of an object, if it exists.

As of now we have to use sync_to_async to run the pagination code in production.

Error traceback below:

(Partially omitted)
Traceback (most recent call last):
  File "/app/MyProjectName/operations/XYZ/read.py", line 23, in get_XYZ_model
    XYZ_list, total = await paginate_queryset(
                           ^^^^^^^^^^^^^^^^^^^^^^^^
  File "/app/MyProjectName/operations/utils/pagination.py", line 12, in paginate_queryset
    page = paginator.page(qs)
               ^^^^^^^^^^^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/django/core/paginator.py", line 89, in page
    number = self.validate_number(number)
             ^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/django/core/paginator.py", line 70, in validate_number
    if number > self.num_pages:
                ^^^^^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/django/utils/functional.py", line 47, in __get__
    res = instance.__dict__[self.name] = self.func(instance)
                                         ^^^^^^^^^^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/django/core/paginator.py", line 116, in num_pages
    if self.count == 0 and not self.allow_empty_first_page:
       ^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/django/utils/functional.py", line 47, in __get__
    res = instance.__dict__[self.name] = self.func(instance)
                                         ^^^^^^^^^^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/django/core/paginator.py", line 110, in count
    return c()
           ^^^
  File "/app/.venv/lib/python3.12/site-packages/django/db/models/query.py", line 620, in count
    return self.query.get_count(using=self.db)
           ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/django/db/models/sql/query.py", line 630, in get_count
    return obj.get_aggregation(using, {"__count": Count("*")})["__count"]
           ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/django/db/models/sql/query.py", line 616, in get_aggregation
    result = compiler.execute_sql(SINGLE)
             ^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/cachalot/monkey_patch.py", line 37, in inner
    return original(compiler, *args, **kwargs)
           ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/cachalot/monkey_patch.py", line 96, in inner
    return _get_result_or_execute_query(
           ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/cachalot/monkey_patch.py", line 64, in _get_result_or_execute_query
    result = execute_query_func()
             ^^^^^^^^^^^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/cachalot/monkey_patch.py", line 80, in <lambda>
    execute_query_func = lambda: original(compiler, *args, **kwargs)
                                 ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/django/db/models/sql/compiler.py", line 1572, in execute_sql
    cursor = self.connection.cursor()
             ^^^^^^^^^^^^^^^^^^^^^^^^
  File "/app/.venv/lib/python3.12/site-packages/django/utils/asyncio.py", line 24, in inner
    raise SynchronousOnlyOperation(message)
django.core.exceptions.SynchronousOnlyOperation: You cannot call this from an async context - use a thread or sync_to_async.

P.S. I'm 99.99% certain that cachalot is not at fault here at all, I can create a small repro-code example, if necessary.
P.P.S. I'm uncertain what type of issue this is. It's partially a feature request, partially a bug. I leave the decision on this matter to maintainers.

Change History (18)

comment:1 by Sarah Boyce, 11 months ago

Cc: Jon Janzen added
Summary: Paginator doesn't work in async modeAdd async interface to Paginator
Triage Stage: UnreviewedAccepted
Type: UncategorizedNew feature

Django has limited async support and needing to use sync_to_async is expected/documented, so this is a new feature

comment:2 by wookkl, 11 months ago

Hello, would it be okay if I contributed to this new feature?

comment:3 by wookkl, 11 months ago

Owner: set to wookkl
Status: newassigned

comment:4 by wookkl, 10 months ago

Has patch: set
Last edited 10 months ago by wookkl (previous) (diff)

comment:5 by Sarah Boyce, 10 months ago

Needs documentation: set
Patch needs improvement: set

in reply to:  5 comment:6 by wookkl, 10 months ago

Replying to Sarah Boyce:

I have refactored code based on your feedback and prepared docs and release notes. Please review them.😀

comment:7 by Jacob Walls, 10 months ago

Needs documentation: unset
Patch needs improvement: unset

Thanks for the update. (You can unset these checkboxes yourself when you've finished addressing feedback.)

comment:8 by Sarah Boyce, 10 months ago

Patch needs improvement: set

comment:9 by Jacob Walls, 10 months ago

Patch needs improvement: unset

comment:10 by Sarah Boyce, 9 months ago

Patch needs improvement: set

comment:11 by wookkl, 9 months ago

Patch needs improvement: unset

comment:12 by Sarah Boyce, 8 months ago

Patch needs improvement: set

comment:13 by wookkl, 8 months ago

Patch needs improvement: unset

comment:14 by Sarah Boyce, 8 months ago

Needs tests: set

comment:15 by wookkl, 8 months ago

Needs tests: unset

comment:16 by Sarah Boyce, 8 months ago

Triage Stage: AcceptedReady for checkin

comment:17 by Sarah Boyce <42296566+sarahboyce@…>, 8 months ago

Resolution: fixed
Status: assignedclosed

In 2ae3044d:

Fixed #35945 -- Added async interface to Paginator.

comment:18 by GitHub <noreply@…>, 4 months ago

In 426be740:

Refs #35844, #35945 -- Used asgiref.sync.iscoroutinefunction() instead of deprecated asyncio.iscoroutinefunction().

Follow up to bd3b1dfa2422e02ced3a894adb7544e42540c97d.
Introduced in 2ae3044d9d4dfb8371055513e440e0384f211963.

Fixes DeprecationWarning:

'asyncio.iscoroutinefunction' is deprecated and slated for removal
in Python 3.16; use inspect.iscoroutinefunction() instead.

Note: See TracTickets for help on using tickets.
Back to Top