[pytest-dev] Parametrized scope question

holger krekel holger at merlinux.eu
Tue Dec 15 05:17:04 EST 2015


Hi Alex,

On Tue, Dec 15, 2015 at 09:18 +0000, Alex Netes wrote:
> Hi holger,
> 
> Thanks for your help.
> 
> > Hi Alex,
> > 
> > On Mon, Dec 14, 2015 at 15:50 +0000, Alex Netes wrote:
> >> Hello guys,
> >>
> >> I'm new to Pytest and I encounter something I cannot explain.
> > 
> > I also hit many things which i can not explain, even with pytest and maybe even with this very mail.
> > 
> 
> I'll try to explain my intention in behind my code, instead of showing my nitty code.
> 
> >> I'm trying to give fixture fixt_func() a parameter fixt_prm and expect the fixture to be called only
> >> once as it defined with 'class' scope, but the fixture is called twice as it ignores the scope. What am I
> >> missing?
> >>
> >>
> >> @pytest.fixture(scope='class')
> >> def fixt_func(request, resource, fixt_prm):
> >>     print fixt_prm
> >>
> >> class TestA():
> >>     @pytest.mark.parametrize('resource', ['x'], scope='class')
> >>     @pytest.mark.parametrize(fixt_prm ', ['x'], scope='class')
> >>     @pytest.mark.parametrize('prm', ['a', 'b'])
> >>     def test_a(self, prm, fixt_func)
> >>         assert True
> > 
> > You are doing something i wasn't aware is possible. You pass a parameter to the fixture function
> > fixt_func through parametrization but as a fixture.
> > In any case, if we modify your example to:
> > 
> >     import pytest
> >     @pytest.fixture(scope='class')
> >     def fixt_func(request, fixt_prm):
> >         print "fixt_func"
> > 
> >     class TestA():
> >         @pytest.mark.parametrize('fixt_prm', ['x', 'y'], scope='class')
> >         def test_a(self, fixt_func):
> >             assert True
> > 
> > this will also call fixt_func twice for the two executing tests.
> > It's argubaly "correct" behaviour from a certain point of view.
> > fixt_prm is class-scoped so each parameter instance exists as a different "class-level" value so we
> > interpret it to mean each class-scoped fixture function needs to be executed with both values.
> > 
> > Just imagine we wouldn't parametrize on the function directly but through a fixture function:
> > 
> >     import pytest
> >     @pytest.fixture(scope='class')
> >     def fixt_func(request, fixt_prm):
> >         print "fixt_func"
> > 
> >     @pytest.fixture(scope='class', params=["x", "y"])
> >     def fixt_prm(request):
> >         print "fixt_prm"
> > 
> >     class TestA():
> >         def test_a(self, fixt_func):
> >             assert True
> > 
> > Here it's maybe more obvious why this executes both fixture functions twice.
> > 
> > however, i am not sure about your precise example above.  I can see why you expect the two
> > different "prm" values (and thus test functions) to execute with the same class-level fixtures.  Maybe
> > others can chime in and say if they consider your example a bug or a "usual" behaviour.
> > 
> 
> Your examples makes sense. I'm trying to do something more complex and maybe I look at it in a wrong
> way. I want to define a fixture "fixt_func" so it would be able to receive a parameter defined by different
> test classes. Moreover I want "fixt_func" to have Class scope, so I can call finalize when all tests of the
> same class finished running. That's why I came up with the above "solution". 

I think you could use a marker on the class or even just a class attribute:

    import pytest
    
    @pytest.fixture(scope="class")
    def fix(request):
        return request.cls.attr * 10

    class TestA:
        attr = 1

        def test_one(self, fix):
            assert 0

    class TestB:
        attr = 2

        def test_one(self, fix):
            assert 0

The "request" object gives you back references into the context
of the test which requests fixtures.

HTH,
holger


More information about the pytest-dev mailing list