From 655ab0bf8b8af9962e40360b40c7e5b55c2092f6 Mon Sep 17 00:00:00 2001 From: Tom Dalton Date: Mon, 23 Oct 2017 17:49:49 +0100 Subject: [PATCH] Address more review comments, fix massive bug I reintroduced in the node-splitting code :-/ --- _pytest/fixtures.py | 51 ++++----------------- _pytest/nodes.py | 37 +++++++++++++++ changelog/2836.bug | 4 +- testing/{test_fixtures.py => test_nodes.py} | 6 +-- 4 files changed, 51 insertions(+), 47 deletions(-) create mode 100644 _pytest/nodes.py rename testing/{test_fixtures.py => test_nodes.py} (71%) diff --git a/_pytest/fixtures.py b/_pytest/fixtures.py index 9b2abc94f..3a1321664 100644 --- a/_pytest/fixtures.py +++ b/_pytest/fixtures.py @@ -1,12 +1,12 @@ from __future__ import absolute_import, division, print_function -import sys - -from py._code.code import FormattedExcinfo - -import py -import warnings import inspect +import sys +import warnings + +import py +from py._code.code import FormattedExcinfo + import _pytest from _pytest._code.code import TerminalRepr from _pytest.compat import ( @@ -15,9 +15,11 @@ from _pytest.compat import ( is_generator, isclass, getimfunc, getlocation, getfuncargnames, safe_getattr, + FuncargnamesCompatAttr, ) +from _pytest.nodes import ischildnode from _pytest.outcomes import fail, TEST_OUTCOME -from _pytest.compat import FuncargnamesCompatAttr + if sys.version_info[:2] == (2, 6): from ordereddict import OrderedDict @@ -1130,40 +1132,7 @@ class FixtureManager: else: return tuple(self._matchfactories(fixturedefs, nodeid)) - @classmethod - def _splitnode(cls, nodeid): - """Split a nodeid into constituent 'parts'. - - Node IDs are strings, and can be things like: - '' - 'testing/code' - 'testing/code/test_excinfo.py' - 'testing/code/test_excinfo.py::TestFormattedExcinfo::()' - - Return values are lists e.g. - [''] - ['testing', 'code'] - ['testing', 'code', 'test_excinfo.py'] - ['testing', 'code', 'test_excinfo.py', 'TestFormattedExcinfo', '()'] - """ - parts = nodeid.split(py.path.local.sep) - # Replace single last element 'test_foo.py::Bar::()' with multiple elements 'test_foo.py', 'Bar', '()' - parts[-1:] = parts[-1].split("::") - return parts - - @classmethod - def _ischildnode(cls, baseid, nodeid): - """Return True if the nodeid is a child node of the baseid. - - E.g. 'foo/bar::Baz::()' is a child of 'foo', 'foo/bar' and 'foo/bar::Baz', but not of 'foo/blorp' - """ - base_parts = cls._splitnode(baseid) - node_parts = cls._splitnode(nodeid) - if len(node_parts) < len(base_parts): - return False - return node_parts[:len(base_parts)] == base_parts - def _matchfactories(self, fixturedefs, nodeid): for fixturedef in fixturedefs: - if self._ischildnode(fixturedef.baseid, nodeid): + if ischildnode(fixturedef.baseid, nodeid): yield fixturedef diff --git a/_pytest/nodes.py b/_pytest/nodes.py new file mode 100644 index 000000000..c28c63180 --- /dev/null +++ b/_pytest/nodes.py @@ -0,0 +1,37 @@ +import py + + +def _splitnode(nodeid): + """Split a nodeid into constituent 'parts'. + + Node IDs are strings, and can be things like: + '' + 'testing/code' + 'testing/code/test_excinfo.py' + 'testing/code/test_excinfo.py::TestFormattedExcinfo::()' + + Return values are lists e.g. + [] + ['testing', 'code'] + ['testing', 'code', 'test_excinfo.py'] + ['testing', 'code', 'test_excinfo.py', 'TestFormattedExcinfo', '()'] + """ + if nodeid == '': + # If there is no root node at all, return an empty list so the caller's logic can remain sane + return [] + parts = nodeid.split(py.path.local.sep) + # Replace single last element 'test_foo.py::Bar::()' with multiple elements 'test_foo.py', 'Bar', '()' + parts[-1:] = parts[-1].split("::") + return parts + + +def ischildnode(baseid, nodeid): + """Return True if the nodeid is a child node of the baseid. + + E.g. 'foo/bar::Baz::()' is a child of 'foo', 'foo/bar' and 'foo/bar::Baz', but not of 'foo/blorp' + """ + base_parts = _splitnode(baseid) + node_parts = _splitnode(nodeid) + if len(node_parts) < len(base_parts): + return False + return node_parts[:len(base_parts)] == base_parts diff --git a/changelog/2836.bug b/changelog/2836.bug index a6e36a34b..afa1961d7 100644 --- a/changelog/2836.bug +++ b/changelog/2836.bug @@ -1,3 +1 @@ -Attempted fix for https://github.com/pytest-dev/pytest/issues/2836 - -Improves the handling of "nodeid"s to be more aware of path and class separators. +Match fixture paths against actual path segments in order to avoid matching folders which share a prefix. diff --git a/testing/test_fixtures.py b/testing/test_nodes.py similarity index 71% rename from testing/test_fixtures.py rename to testing/test_nodes.py index 8d595fe52..6f4540f99 100644 --- a/testing/test_fixtures.py +++ b/testing/test_nodes.py @@ -1,6 +1,6 @@ import pytest -from _pytest import fixtures +from _pytest import nodes @pytest.mark.parametrize("baseid, nodeid, expected", ( @@ -13,6 +13,6 @@ from _pytest import fixtures ('foo/bar::TestBaz::()', 'foo/bar::TestBop::()', False), ('foo/bar', 'foo/bar::TestBop::()', True), )) -def test_fixturemanager_ischildnode(baseid, nodeid, expected): - result = fixtures.FixtureManager._ischildnode(baseid, nodeid) +def test_ischildnode(baseid, nodeid, expected): + result = nodes.ischildnode(baseid, nodeid) assert result is expected