Merged in issue660 (pull request #268)
fix issue660 --HG-- branch : pytest-2.7
This commit is contained in:
		
						commit
						6591d7a209
					
				|  | @ -1,6 +1,11 @@ | ||||||
| 2.7.1.dev (compared to 2.7.0) | 2.7.1.dev (compared to 2.7.0) | ||||||
| ----------------------------- | ----------------------------- | ||||||
| 
 | 
 | ||||||
|  | - fix issue660: properly report scope-mismatch-access errors | ||||||
|  |   independently from ordering of fixture arguments.  Also | ||||||
|  |   avoid the pytest internal traceback which does not provide | ||||||
|  |   information to the user. Thanks Holger Krekel. | ||||||
|  | 
 | ||||||
| - streamlined and documented release process.  Also all versions | - streamlined and documented release process.  Also all versions | ||||||
|   (in setup.py and documentation generation) are now read |   (in setup.py and documentation generation) are now read | ||||||
|   from _pytest/__init__.py. Thanks Holger Krekel. |   from _pytest/__init__.py. Thanks Holger Krekel. | ||||||
|  |  | ||||||
|  | @ -1356,12 +1356,7 @@ class FixtureRequest(FuncargnamesCompatAttr): | ||||||
|         try: |         try: | ||||||
|             val = cache[cachekey] |             val = cache[cachekey] | ||||||
|         except KeyError: |         except KeyError: | ||||||
|             __tracebackhide__ = True |             self._check_scope(self.fixturename, self.scope, scope) | ||||||
|             if scopemismatch(self.scope, scope): |  | ||||||
|                 raise ScopeMismatchError("You tried to access a %r scoped " |  | ||||||
|                     "resource with a %r scoped request object" %( |  | ||||||
|                     (scope, self.scope))) |  | ||||||
|             __tracebackhide__ = False |  | ||||||
|             val = setup() |             val = setup() | ||||||
|             cache[cachekey] = val |             cache[cachekey] = val | ||||||
|             if teardown is not None: |             if teardown is not None: | ||||||
|  | @ -1392,6 +1387,7 @@ class FixtureRequest(FuncargnamesCompatAttr): | ||||||
|                 if argname == "request": |                 if argname == "request": | ||||||
|                     class PseudoFixtureDef: |                     class PseudoFixtureDef: | ||||||
|                         cached_result = (self, [0], None) |                         cached_result = (self, [0], None) | ||||||
|  |                         scope = "function" | ||||||
|                     return PseudoFixtureDef |                     return PseudoFixtureDef | ||||||
|                 raise |                 raise | ||||||
|         # remove indent to prevent the python3 exception |         # remove indent to prevent the python3 exception | ||||||
|  | @ -1435,16 +1431,7 @@ class FixtureRequest(FuncargnamesCompatAttr): | ||||||
|         subrequest = SubRequest(self, scope, param, param_index, fixturedef) |         subrequest = SubRequest(self, scope, param, param_index, fixturedef) | ||||||
| 
 | 
 | ||||||
|         # check if a higher-level scoped fixture accesses a lower level one |         # check if a higher-level scoped fixture accesses a lower level one | ||||||
|         if scope is not None: |         subrequest._check_scope(argname, self.scope, scope) | ||||||
|             __tracebackhide__ = True |  | ||||||
|             if scopemismatch(self.scope, scope): |  | ||||||
|                 # try to report something helpful |  | ||||||
|                 lines = subrequest._factorytraceback() |  | ||||||
|                 raise ScopeMismatchError("You tried to access the %r scoped " |  | ||||||
|                     "fixture %r with a %r scoped request object, " |  | ||||||
|                     "involved factories\n%s" %( |  | ||||||
|                     (scope, argname, self.scope, "\n".join(lines)))) |  | ||||||
|             __tracebackhide__ = False |  | ||||||
| 
 | 
 | ||||||
|         # clear sys.exc_info before invoking the fixture (python bug?) |         # clear sys.exc_info before invoking the fixture (python bug?) | ||||||
|         # if its not explicitly cleared it will leak into the call |         # if its not explicitly cleared it will leak into the call | ||||||
|  | @ -1458,6 +1445,18 @@ class FixtureRequest(FuncargnamesCompatAttr): | ||||||
|                                                   subrequest.node) |                                                   subrequest.node) | ||||||
|         return val |         return val | ||||||
| 
 | 
 | ||||||
|  |     def _check_scope(self, argname, invoking_scope, requested_scope): | ||||||
|  |         if argname == "request": | ||||||
|  |             return | ||||||
|  |         if scopemismatch(invoking_scope, requested_scope): | ||||||
|  |             # try to report something helpful | ||||||
|  |             lines = self._factorytraceback() | ||||||
|  |             pytest.fail("ScopeMismatch: you tried to access the %r scoped " | ||||||
|  |                 "fixture %r with a %r scoped request object, " | ||||||
|  |                 "involved factories\n%s" %( | ||||||
|  |                 (requested_scope, argname, invoking_scope, "\n".join(lines))), | ||||||
|  |                 pytrace=False) | ||||||
|  | 
 | ||||||
|     def _factorytraceback(self): |     def _factorytraceback(self): | ||||||
|         lines = [] |         lines = [] | ||||||
|         for fixturedef in self._get_fixturestack(): |         for fixturedef in self._get_fixturestack(): | ||||||
|  | @ -1518,6 +1517,7 @@ scopenum_function = scopes.index("function") | ||||||
| def scopemismatch(currentscope, newscope): | def scopemismatch(currentscope, newscope): | ||||||
|     return scopes.index(newscope) > scopes.index(currentscope) |     return scopes.index(newscope) > scopes.index(currentscope) | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| class FixtureLookupError(LookupError): | class FixtureLookupError(LookupError): | ||||||
|     """ could not return a requested Fixture (missing or invalid). """ |     """ could not return a requested Fixture (missing or invalid). """ | ||||||
|     def __init__(self, argname, request, msg=None): |     def __init__(self, argname, request, msg=None): | ||||||
|  | @ -1867,6 +1867,7 @@ class FixtureDef: | ||||||
|         for argname in self.argnames: |         for argname in self.argnames: | ||||||
|             fixturedef = request._get_active_fixturedef(argname) |             fixturedef = request._get_active_fixturedef(argname) | ||||||
|             result, arg_cache_key, exc = fixturedef.cached_result |             result, arg_cache_key, exc = fixturedef.cached_result | ||||||
|  |             request._check_scope(argname, request.scope, fixturedef.scope) | ||||||
|             kwargs[argname] = result |             kwargs[argname] = result | ||||||
|             if argname != "request": |             if argname != "request": | ||||||
|                 fixturedef.addfinalizer(self.finish) |                 fixturedef.addfinalizer(self.finish) | ||||||
|  |  | ||||||
|  | @ -906,6 +906,27 @@ class TestFixtureUsages: | ||||||
|             "*1 error*" |             "*1 error*" | ||||||
|         ]) |         ]) | ||||||
| 
 | 
 | ||||||
|  |     def test_receives_funcargs_scope_mismatch_issue660(self, testdir): | ||||||
|  |         testdir.makepyfile(""" | ||||||
|  |             import pytest | ||||||
|  |             @pytest.fixture(scope="function") | ||||||
|  |             def arg1(): | ||||||
|  |                 return 1 | ||||||
|  | 
 | ||||||
|  |             @pytest.fixture(scope="module") | ||||||
|  |             def arg2(arg1): | ||||||
|  |                 return arg1 + 1 | ||||||
|  | 
 | ||||||
|  |             def test_add(arg1, arg2): | ||||||
|  |                 assert arg2 == 2 | ||||||
|  |         """) | ||||||
|  |         result = testdir.runpytest() | ||||||
|  |         result.stdout.fnmatch_lines([ | ||||||
|  |             "*ScopeMismatch*involved factories*", | ||||||
|  |             "* def arg2*", | ||||||
|  |             "*1 error*" | ||||||
|  |         ]) | ||||||
|  | 
 | ||||||
|     def test_funcarg_parametrized_and_used_twice(self, testdir): |     def test_funcarg_parametrized_and_used_twice(self, testdir): | ||||||
|         testdir.makepyfile(""" |         testdir.makepyfile(""" | ||||||
|             import pytest |             import pytest | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue