Skip to content

Commit

Permalink
Fix Django + cron using the wrong connector
Browse files Browse the repository at this point in the history
  • Loading branch information
ewjoachim committed Aug 16, 2024
1 parent cfe0aa6 commit 66ac98e
Show file tree
Hide file tree
Showing 7 changed files with 50 additions and 48 deletions.
18 changes: 0 additions & 18 deletions docs/howto/advanced/sync_defer.md
Original file line number Diff line number Diff line change
Expand Up @@ -55,24 +55,6 @@ app = App(connector=SQLAlchemyPsycopg2Connector())
app.open(engine)
```

## Having multiple apps

If you need to have multiple connectors interact with the tasks, you can
create multiple synchronized apps with {py:meth}`App.with_connector`:

```
import procrastinate
app = procrastinate.App(
connector=procrastinate.PsycopgConnector(...),
)
sync_app = app.with_connector(
connector=procrastinate.SyncPsycopgConnector(...),
)
```

## Procrastinate's automatic connector selection

Async connectors are able to summon their synchronous counterpart when needed
Expand Down
4 changes: 2 additions & 2 deletions docs/howto/django/scripts.md
Original file line number Diff line number Diff line change
Expand Up @@ -21,8 +21,8 @@ def main():
django.setup()
# By default, the app uses the Django database connection, which is unsuitable
# for the worker.
app = app.with_connector(app.connector.get_worker_connector())
app.run_worker()
with app.replace_connector(app.connector.get_worker_connector()):
app.run_worker()

if __name__ == "__main__":
main()
Expand Down
22 changes: 11 additions & 11 deletions docs/howto/django/tests.md
Original file line number Diff line number Diff line change
Expand Up @@ -24,8 +24,8 @@ def app():
# Replace the connector in the current app
# Note that this fixture gives you the app back for convenience, but it's
# the same instance as you'd get with `procrastinate.contrib.django.app`.
with procrastinate_app.current_app.replace_connector(in_memory) as app_with_connector:
yield app_with_connector
with procrastinate_app.current_app.replace_connector(in_memory) as app:
yield app

def test_my_task(app):
# Run the task
Expand Down Expand Up @@ -126,8 +126,8 @@ class TestingTaskClass(TransactionTestCase):
my_task.defer(a=1, b=2)

# Start worker
app = app.with_connector(app.connector.get_worker_connector())
app.run_worker(wait=False, install_signal_handlers=False, listen_notify=False)
with app.replace_connector(app.connector.get_worker_connector())
app.run_worker(wait=False, install_signal_handlers=False, listen_notify=False)

# Check task has been executed
assert ProcrastinateJob.objects.filter(task_name="my_task").status == "succeeded"
Expand All @@ -144,20 +144,20 @@ def test_task():
my_task.defer(a=1, b=2)

# Start worker
app = app.with_connector(app.connector.get_worker_connector())
app.run_worker(wait=False, install_signal_handlers=False, listen_notify=False)
with app.replace_connector(app.connector.get_worker_connector())
app.run_worker(wait=False, install_signal_handlers=False, listen_notify=False)

# Check task has been executed
assert ProcrastinateJob.objects.filter(task_name="my_task").status == "succeeded"

# Or with a fixture
@pytest.fixture
def worker(transactional_db):
def _():
app = app.with_connector(app.connector.get_worker_connector())
app.run_worker(wait=False, install_signal_handlers=False, listen_notify=False)
return app
return _
with app.replace_connector(app.connector.get_worker_connector())
def f():
app.run_worker(wait=False, install_signal_handlers=False, listen_notify=False)
return app
yield f

def test_task(worker):
# Run tasks
Expand Down
4 changes: 2 additions & 2 deletions docs/howto/production/testing.md
Original file line number Diff line number Diff line change
Expand Up @@ -17,8 +17,8 @@ def app():
# Replace the connector in the current app
# Note that this fixture gives you the app back for covenience,
# but it's the same instance as `my_app`.
with my_app.replace_connector(in_memory) as app_with_connector:
yield app_with_connector
with my_app.replace_connector(in_memory) as app:
yield app


def test_my_task(app):
Expand Down
26 changes: 19 additions & 7 deletions procrastinate/app.py
Original file line number Diff line number Diff line change
Expand Up @@ -94,12 +94,18 @@ def __init__(

self._register_builtin_tasks()

def with_connector(self, connector: connector_module.BaseConnector) -> App:
def with_connector(
self,
connector: connector_module.BaseConnector,
) -> App:
"""
Create another app instance sychronized with this one, with a different
connector. For all things regarding periodic tasks, the original app
(and its original connector) will be used, even when the new app's
methods are used.
connector.
.. deprecated:: 2.14.0
Use `replace_connector` instead. Because this method creates a new
app that references the same tasks, and the task have a link
back to the app, using this method can lead to unexpected behavior.
Parameters
----------
Expand All @@ -109,7 +115,7 @@ def with_connector(self, connector: connector_module.BaseConnector) -> App:
Returns
-------
:
A new compatible app.
A new app with the same tasks.
"""
app = App(
connector=connector,
Expand All @@ -126,6 +132,12 @@ def replace_connector(
) -> Iterator[App]:
"""
Replace the connector of the app while in the context block, then restore it.
The context variable is the same app as this method is called on.
>>> with app.replace_connector(new_connector) as app2:
... ...
... # app and app2 are the same object
Parameters
----------
Expand All @@ -134,8 +146,8 @@ def replace_connector(
Yields
-------
`App`
A new compatible app.
:
A context manager that yields the same app with the new connector.
"""
old_connector = self.connector
self.connector = connector
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
from __future__ import annotations

import asyncio
import contextlib

from django.core.management.base import BaseCommand

Expand All @@ -24,6 +25,11 @@ def add_arguments(self, parser):

def handle(self, *args, **kwargs):
kwargs = {k: v for k, v in kwargs.items() if k not in self._django_options}
context = contextlib.nullcontext()

if isinstance(app.connector, django_connector.DjangoConnector):
kwargs["app"] = app.with_connector(app.connector.get_worker_connector())
asyncio.run(cli.execute_command(kwargs))
kwargs["app"] = app
context = app.replace_connector(app.connector.get_worker_connector())

with context:
asyncio.run(cli.execute_command(kwargs))
14 changes: 8 additions & 6 deletions tests/integration/contrib/django/test_models.py
Original file line number Diff line number Diff line change
Expand Up @@ -111,12 +111,14 @@ def my_task(timestamp):
pass

django_app = procrastinate.contrib.django.app
app = django_app.with_connector(django_app.connector.get_worker_connector())
async with app.open_async():
try:
await asyncio.wait_for(app.run_worker_async(), timeout=0.1)
except asyncio.TimeoutError:
pass
with django_app.replace_connector(
django_app.connector.get_worker_connector()
) as app:
async with app.open_async():
try:
await asyncio.wait_for(app.run_worker_async(), timeout=0.1)
except asyncio.TimeoutError:
pass

periodic_defers = []
async for element in models.ProcrastinatePeriodicDefer.objects.values().all():
Expand Down

0 comments on commit 66ac98e

Please sign in to comment.