Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update Django/Python support #458

Closed
1 change: 0 additions & 1 deletion requirements/test.txt
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
case>=1.3.1
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This one might deserve its own separate PR... Thoughts?

pytest>=6.2.5,<8
pytest-django>=4.5.2
pytest-benchmark
Expand Down
119 changes: 119 additions & 0 deletions t/conftest.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,8 @@
import sys
import types
from contextlib import contextmanager
from unittest.mock import MagicMock, Mock, patch

import pytest

# we have to import the pytest plugin fixtures here,
Expand All @@ -20,6 +25,9 @@
)


SENTINEL = object()


@pytest.fixture(scope='session', autouse=True)
def setup_default_app_trap():
from celery._state import set_default_app
Expand All @@ -31,6 +39,117 @@ def app(celery_app):
return celery_app


@contextmanager
def module_context_manager(*names):
"""Mock one or modules such that every attribute is a :class:`Mock`."""
yield from _module(*names)


def _module(*names):
prev = {}

class MockModule(types.ModuleType):

def __getattr__(self, attr):
setattr(self, attr, Mock())
return types.ModuleType.__getattribute__(self, attr)

mods = []
for name in names:
try:
prev[name] = sys.modules[name]
except KeyError:
pass
mod = sys.modules[name] = MockModule(name)
mods.append(mod)
try:
yield mods
finally:
for name in names:
try:
sys.modules[name] = prev[name]
except KeyError:
try:
del sys.modules[name]
except KeyError:
pass


class _patching:

def __init__(self, monkeypatch, request):
self.monkeypatch = monkeypatch
self.request = request

def __getattr__(self, name):
return getattr(self.monkeypatch, name)

def __call__(self, path, value=SENTINEL, name=None,
new=MagicMock, **kwargs):
value = self._value_or_mock(value, new, name, path, **kwargs)
self.monkeypatch.setattr(path, value)
return value

def object(self, target, attribute, *args, **kwargs):
return _wrap_context(
patch.object(target, attribute, *args, **kwargs),
self.request)

def _value_or_mock(self, value, new, name, path, **kwargs):
if value is SENTINEL:
value = new(name=name or path.rpartition('.')[2])
for k, v in kwargs.items():
setattr(value, k, v)
return value

def setattr(self, target, name=SENTINEL, value=SENTINEL, **kwargs):
# alias to __call__ with the interface of pytest.monkeypatch.setattr
if value is SENTINEL:
value, name = name, None
return self(target, value, name=name)

def setitem(self, dic, name, value=SENTINEL, new=MagicMock, **kwargs):
# same as pytest.monkeypatch.setattr but default value is MagicMock
value = self._value_or_mock(value, new, name, dic, **kwargs)
self.monkeypatch.setitem(dic, name, value)
return value

def modules(self, *mods):
modules = []
for mod in mods:
mod = mod.split('.')
modules.extend(reversed([
'.'.join(mod[:-i] if i else mod) for i in range(len(mod))
]))
modules = sorted(set(modules))
return _wrap_context(module_context_manager(*modules), self.request)


def _wrap_context(context, request):
ret = context.__enter__()

def fin():
context.__exit__(*sys.exc_info())
request.addfinalizer(fin)
return ret


@pytest.fixture()
def patching(monkeypatch, request):
"""Monkeypath.setattr shortcut.
Example:
.. code-block:: python
>>> def test_foo(patching):
>>> # execv value here will be mock.MagicMock by default.
>>> execv = patching('os.execv')
>>> patching('sys.platform', 'darwin') # set concrete value
>>> patching.setenv('DJANGO_SETTINGS_MODULE', 'x.settings')
>>> # val will be of type mock.MagicMock by default
>>> val = patching.setitem('path.to.dict', 'KEY')
"""
return _patching(monkeypatch, request)


@pytest.fixture(autouse=True)
def test_cases_shortcuts(request, app, patching):
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As far as I can tell, the patching fixture was only thing that was coming from the case dependency. I took the current implementation from Celery (found here celery/celery#7077)

Not sure if duplicating these ~100 lines of code is the best way forward but that does the job

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

we might be using that from celery instead of case if those are already available there

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

or we can handle this in a separate PR

Copy link
Contributor Author

@browniebroke browniebroke Mar 3, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The version from Celery is defined under the tests directory (celery/t), which not included in the wheel distribution, so it's not available in the version we have installed... I tried to install from the source distribution, which seems to have the t directory included, but couldn't get to import the fixture.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So I have 2 approaches:

  1. Extract the commit as is from this PR Remove case dependency and copy Celery fixture #462 - just works but fixture is duplicated with Celery
  2. Use the fixture from Celery PoC: Remove case dependency and use Celery fixture #461 - not working right now, so I assume it needs more work in Celery... Perhaps move the fixture from t to under celery to be included in the distribution, but I'm not sure about the side effects of doing so.

Approach 1 is more practical, and can unblock this PR immediately. Approach 2 is more purist but needs more work and guidance from one of the Celery maintainers.

if request.instance:
Expand Down