-
Notifications
You must be signed in to change notification settings - Fork 211
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
Changes from 1 commit
e94ff49
54b2296
5d7502f
09aa815
c62e363
2408e9a
9033da2
b8dd925
146e65a
b2db7dc
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,4 +1,3 @@ | ||
case>=1.3.1 | ||
pytest>=6.2.5,<8 | ||
pytest-django>=4.5.2 | ||
pytest-benchmark | ||
|
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, | ||
|
@@ -20,6 +25,9 @@ | |
) | ||
|
||
|
||
SENTINEL = object() | ||
|
||
|
||
@pytest.fixture(scope='session', autouse=True) | ||
def setup_default_app_trap(): | ||
from celery._state import set_default_app | ||
|
@@ -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): | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. As far as I can tell, the Not sure if duplicating these ~100 lines of code is the best way forward but that does the job There was a problem hiding this comment. Choose a reason for hiding this commentThe 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 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. or we can handle this in a separate PR There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The version from Celery is defined under the tests directory ( There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. So I have 2 approaches:
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: | ||
|
There was a problem hiding this comment.
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?