Python monkeypatch.setattr () с приспособлением pytest в области видимости модуля - PullRequest
0 голосов
/ 28 декабря 2018

Прежде всего, соответствующая часть моего каталога проекта выглядит следующим образом:

└── my_package
    ├── my_subpackage
    │   ├── my_module.py
    |   └── other_module.py
    └── tests
        └── my_subpackage
            └── unit_test.py

Я пишу несколько тестов в unit_test.py, которые требуют проверки внешнего ресурса на уровне модуля.Я хотел бы использовать pytest fixture с областью действия уровня модуля и pytest monkeypatch для достижения этой цели.Вот фрагмент того, что я пробовал в unit_test.py:

import unittest.mock as mock
import pytest
from my_package.my_subpackage.my_module import MyClass


@pytest.fixture(scope='function')
def external_access(monkeypatch):
    external_access = mock.MagicMock()
    external_access.get_something = mock.MagicMock(
        return_value='Mock was used.')
    monkeypatch.setattr(
        'my_package.my_subpackage.my_module.ExternalAccess.get_something',
        external_access.get_something)


def test_get_something(external_access):
    instance = MyClass()
    instance.get_something()
    assert instance.data == 'Mock was used.'

Все работает просто отлично.Но когда я пытаюсь изменить строку 8 с @pytest.fixture(scope='function') на @pytest.fixture(scope='module'), я получаю следующую ошибку:

ScopeMismatch: You tried to access the 'function' scoped fixture 'monkeypatch' with a 'module' scoped request object, involved factories
my_package\tests\unit_test.py:7:  def external_access(monkeypatch)
..\..\Anaconda3\envs\py37\lib\site-packages\_pytest\monkeypatch.py:20:  def monkeypatch()

Кто-нибудь знает, как сделать monkeypatch с областью действия уровня модуля?

InЕсли кто-то хочет знать, это то, как выглядят оба модуля.

my_module.py

from my_package.my_subpackage.other_module import ExternalAccess


class MyClass(object):
    def __init__(self):
        self.external_access = ExternalAccess()
        self.data = None

    def get_something(self):
        self.data = self.external_access.get_something()

other_module.py

class ExternalAccess(object):
    def get_something(self):
        return 'Call to external resource.'

1 Ответ

0 голосов
/ 28 декабря 2018

Я нашел эту проблему , которая направляла путь.Мне нужно было внести несколько изменений в решение для области уровня модуля.unit_test.py теперь выглядит так:

import unittest.mock as mock

import pytest

from my_package.my_subpackage.my_module import MyClass


@pytest.fixture(scope='module')
def monkeymodule():
    from _pytest.monkeypatch import MonkeyPatch
    mpatch = MonkeyPatch()
    yield mpatch
    mpatch.undo()

@pytest.fixture(scope='module')
def external_access(monkeymodule):
    external_access = mock.MagicMock()
    external_access.get_something = mock.MagicMock(
        return_value='Mock was used.')
    monkeymodule.setattr(
        'my_package.my_subpackage.my_module.ExternalAccess.get_something',
        external_access.get_something)


def test_get_something(external_access):
    instance = MyClass()
    instance.get_something()
    assert instance.data == 'Mock was used.'
...