Skip to content

Commit

Permalink
Use pytest data to extract the parametrised decoration
Browse files Browse the repository at this point in the history
Rather than try and parse out the parametrized portion of the nodeid (delimited by square brackets but possibly containing square brackets), use native [pytest item attributes](https://docs.pytest.org/en/6.2.x/reference.html#function) to separate out the decoration.

Better solution for microsoft#17357, fixing microsoft#17676.
  • Loading branch information
mjpieters committed Oct 31, 2021
1 parent 9bf27f2 commit 4a011e4
Show file tree
Hide file tree
Showing 2 changed files with 29 additions and 39 deletions.
54 changes: 15 additions & 39 deletions pythonFiles/testing_tools/adapter/pytest/_pytest_item.py
Original file line number Diff line number Diff line change
Expand Up @@ -158,9 +158,21 @@ def parse_item(
# Skip plugin generated tests
if kind is None:
return None, None
(nodeid, parents, fileid, testfunc, parameterized) = _parse_node_id(
item.nodeid, kind
)

if kind == "function" and item.originalname and item.originalname != item.name:
# split out parametrized decorations `node[params]`) before parsing
# and manually attach parametrized portion back in when done.
parameterized = item.name[len(item.originalname) :]
(parentid, parents, fileid, testfunc, _) = _parse_node_id(
item.nodeid[: -len(parameterized)], kind
)
nodeid = "{}{}".format(parentid, parameterized)
parents = [(parentid, item.originalname, kind)] + parents
else:
(nodeid, parents, fileid, testfunc, parameterized) = _parse_node_id(
item.nodeid, kind
)

# Note: testfunc does not necessarily match item.function.__name__.
# This can result from importing a test function from another module.

Expand Down Expand Up @@ -434,32 +446,6 @@ def _parse_node_id(
)


def _find_left_bracket(nodeid):
"""Return tuple of part before final bracket open, separator [, and the remainder.
Notes:
Testcase names in case of parametrized tests are wrapped in [<test-case-name>].
Examples:
dirname[sometext]/dirname/testfile.py::testset::testname[testcase]
=> ('dirname[sometext]/dirname/testfile.py::testset::testname', '[', 'testcase]')
dirname/dirname/testfile.py::testset::testname[testcase]
=> ('dirname/dirname/testfile.py::testset::testname', '[', 'testcase]')
dirname/dirname/testfile.py::testset::testname[testcase[x]]
=> ('dirname/dirname/testfile.py::testset::testname', '[', 'testcase[x]]')
"""
if not nodeid.endswith("]"):
return nodeid, "", ""
bracketcount = 0
for index, char in enumerate(nodeid[::-1]):
if char == "]":
bracketcount += 1
elif char == "[":
bracketcount -= 1
if bracketcount == 0:
n = len(nodeid) - 1 - index
return nodeid[:n], nodeid[n], nodeid[n + 1 :]
return nodeid, "", ""


def _iter_nodes(
testid,
kind,
Expand All @@ -473,16 +459,6 @@ def _iter_nodes(
if len(nodeid) > len(testid):
testid = "." + _pathsep + testid

if kind == "function" and nodeid.endswith("]"):
funcid, sep, parameterized = _find_left_bracket(nodeid)
if not sep:
raise should_never_reach_here(
nodeid,
# ...
)
yield (nodeid, sep + parameterized, "subtest")
nodeid = funcid

parentid, _, name = nodeid.rpartition("::")
if not parentid:
if kind is None:
Expand Down
14 changes: 14 additions & 0 deletions pythonFiles/tests/testing_tools/adapter/pytest/test_discovery.py
Original file line number Diff line number Diff line change
Expand Up @@ -530,6 +530,7 @@ def test_modifyitems(self):
stub,
nodeid="test_spam.py::SpamTests::test_one",
name="test_one",
originalname="test_one",
location=("test_spam.py", 12, "SpamTests.test_one"),
fspath=adapter_util.PATH_JOIN(testroot, "test_spam.py"),
function=FakeFunc("test_one"),
Expand All @@ -538,6 +539,7 @@ def test_modifyitems(self):
stub,
nodeid="test_spam.py::SpamTests::test_other",
name="test_other",
originalname="test_other",
location=("test_spam.py", 19, "SpamTests.test_other"),
fspath=adapter_util.PATH_JOIN(testroot, "test_spam.py"),
function=FakeFunc("test_other"),
Expand All @@ -546,6 +548,7 @@ def test_modifyitems(self):
stub,
nodeid="test_spam.py::test_all",
name="test_all",
originalname="test_all",
location=("test_spam.py", 144, "test_all"),
fspath=adapter_util.PATH_JOIN(testroot, "test_spam.py"),
function=FakeFunc("test_all"),
Expand All @@ -554,6 +557,7 @@ def test_modifyitems(self):
stub,
nodeid="test_spam.py::test_each[10-10]",
name="test_each[10-10]",
originalname="test_each",
location=("test_spam.py", 273, "test_each[10-10]"),
fspath=adapter_util.PATH_JOIN(testroot, "test_spam.py"),
function=FakeFunc("test_each"),
Expand All @@ -562,6 +566,7 @@ def test_modifyitems(self):
stub,
nodeid=relfile2 + "::All::BasicTests::test_first",
name="test_first",
originalname="test_first",
location=(relfile2, 31, "All.BasicTests.test_first"),
fspath=adapter_util.PATH_JOIN(testroot, relfile2),
function=FakeFunc("test_first"),
Expand All @@ -570,6 +575,7 @@ def test_modifyitems(self):
stub,
nodeid=relfile2 + "::All::BasicTests::test_each[1+2-3]",
name="test_each[1+2-3]",
originalname="test_each",
location=(relfile2, 62, "All.BasicTests.test_each[1+2-3]"),
fspath=adapter_util.PATH_JOIN(testroot, relfile2),
function=FakeFunc("test_each"),
Expand Down Expand Up @@ -781,6 +787,7 @@ def test_finish(self):
stub,
nodeid=relfile + "::SpamTests::test_spam",
name="test_spam",
originalname="test_spam",
location=(relfile, 12, "SpamTests.test_spam"),
fspath=adapter_util.PATH_JOIN(testroot, relfile),
function=FakeFunc("test_spam"),
Expand Down Expand Up @@ -992,6 +999,7 @@ def test_nested_brackets(self):
stub,
nodeid=relfile + "::SpamTests::test_spam[a-[b]-c]",
name="test_spam[a-[b]-c]",
originalname="test_spam",
location=(relfile, 12, "SpamTests.test_spam[a-[b]-c]"),
fspath=adapter_util.PATH_JOIN(testroot, relfile),
function=FakeFunc("test_spam"),
Expand Down Expand Up @@ -1054,6 +1062,7 @@ def test_nested_suite(self):
stub,
nodeid=relfile + "::SpamTests::Ham::Eggs::test_spam",
name="test_spam",
originalname="test_spam",
location=(relfile, 12, "SpamTests.Ham.Eggs.test_spam"),
fspath=adapter_util.PATH_JOIN(testroot, relfile),
function=FakeFunc("test_spam"),
Expand Down Expand Up @@ -1120,6 +1129,7 @@ def test_windows(self):
# pytest always uses "/" as the path separator in node IDs:
nodeid="X/Y/Z/test_Eggs.py::SpamTests::test_spam",
name="test_spam",
originalname="test_spam",
# normal path separator (contrast with nodeid):
location=(relfile, 12, "SpamTests.test_spam"),
# path separator matches location:
Expand Down Expand Up @@ -1152,6 +1162,7 @@ def test_windows(self):
stub,
nodeid=fileid + "::test_spam",
name="test_spam",
originalname="test_spam",
location=(locfile, 12, "test_spam"),
fspath=fspath,
function=FakeFunc("test_spam"),
Expand Down Expand Up @@ -1412,6 +1423,7 @@ def test_mysterious_parens(self):
stub,
nodeid=relfile + "::SpamTests::()::()::test_spam",
name="test_spam",
originalname="test_spam",
location=(relfile, 12, "SpamTests.test_spam"),
fspath=adapter_util.PATH_JOIN(testroot, relfile),
function=FakeFunc("test_spam"),
Expand Down Expand Up @@ -1472,6 +1484,7 @@ def test_imported_test(self):
stub,
nodeid=relfile + "::SpamTests::test_spam",
name="test_spam",
originalname="test_spam",
location=(srcfile, 12, "SpamTests.test_spam"),
fspath=adapter_util.PATH_JOIN(testroot, relfile),
function=FakeFunc("test_spam"),
Expand All @@ -1480,6 +1493,7 @@ def test_imported_test(self):
stub,
nodeid=relfile + "::test_ham",
name="test_ham",
originalname="test_ham",
location=(srcfile, 3, "test_ham"),
fspath=adapter_util.PATH_JOIN(testroot, relfile),
function=FakeFunc("test_spam"),
Expand Down

0 comments on commit 4a011e4

Please sign in to comment.