Skip to content

Commit ae882f2

Browse files
authored
Add more test cases for parametrizations. (#221)
1 parent a313d44 commit ae882f2

File tree

8 files changed

+153
-58
lines changed

8 files changed

+153
-58
lines changed

README.rst

+10-10
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
.. image:: https://raw.githubusercontent.com/pytask-dev/pytask/main/docs/source/_static/images/pytask_w_text.png
2-
:target: https://pytask-dev.readthedocs.io/en/latest
2+
:target: https://pytask-dev.readthedocs.io/en/stable
33
:align: center
44
:width: 50%
55
:alt: pytask
@@ -27,7 +27,7 @@
2727
:target: https://pypi.org/project/pytask
2828

2929
.. image:: https://readthedocs.org/projects/pytask-dev/badge/?version=latest
30-
:target: https://pytask-dev.readthedocs.io/en/latest
30+
:target: https://pytask-dev.readthedocs.io/en/stable
3131

3232
.. image:: https://img.shields.io/github/workflow/status/pytask-dev/pytask/Continuous%20Integration%20Workflow/main
3333
:target: https://github.com/pytask-dev/pytask/actions?query=branch%3Amain
@@ -57,12 +57,12 @@ projects. Its features include:
5757
do not execute it.
5858

5959
- **Debug mode.** `Jump into the debugger
60-
<https://pytask-dev.readthedocs.io/en/latest/tutorials/how_to_debug.html>`_ if a task
60+
<https://pytask-dev.readthedocs.io/en/stable/tutorials/how_to_debug.html>`_ if a task
6161
fails, get feedback quickly, and be more productive.
6262

6363
- **Select tasks via expressions.** Run only a subset of tasks with `expressions and
6464
marker expressions
65-
<https://pytask-dev.readthedocs.io/en/latest/tutorials/how_to_select_tasks.html>`_
65+
<https://pytask-dev.readthedocs.io/en/stable/tutorials/how_to_select_tasks.html>`_
6666
known from pytest.
6767

6868
- **Easily extensible with plugins**. pytask is built on top of `pluggy
@@ -73,7 +73,7 @@ projects. Its features include:
7373
<https://github.com/pytask-dev/pytask-r>`_, and `Stata
7474
<https://github.com/pytask-dev/pytask-stata>`_ and more can be found `here
7575
<https://github.com/topics/pytask>`_. Read in `this tutorial
76-
<https://pytask-dev.readthedocs.io/en/latest/tutorials/how_to_use_plugins.html>`_ how
76+
<https://pytask-dev.readthedocs.io/en/stable/tutorials/how_to_use_plugins.html>`_ how
7777
to use and create plugins with a `cookiecutter
7878
<https://github.com/pytask-dev/cookiecutter-pytask-plugin>`_.
7979

@@ -105,7 +105,7 @@ example, installed via the `Microsoft Store <https://aka.ms/terminal>`_.
105105
To quickly set up a new project, use the `cookiecutter-pytask-project
106106
<https://github.com/pytask-dev/cookiecutter-pytask-project>`_ template or start from
107107
`other templates or example projects
108-
<https://pytask-dev.readthedocs.io/en/latest/how_to_guides/bp_templates_and_projects.html>`_.
108+
<https://pytask-dev.readthedocs.io/en/stable/how_to_guides/bp_templates_and_projects.html>`_.
109109

110110
.. end-installation
111111
@@ -144,16 +144,16 @@ To execute the task, enter ``pytask`` on the command-line
144144
Documentation
145145
-------------
146146

147-
The documentation can be found under https://pytask-dev.readthedocs.io/en/latest with
148-
`tutorials <https://pytask-dev.readthedocs.io/en/latest/tutorials/index.html>`_ and
147+
The documentation can be found under https://pytask-dev.readthedocs.io/en/stable with
148+
`tutorials <https://pytask-dev.readthedocs.io/en/stable/tutorials/index.html>`_ and
149149
guides for `best practices
150-
<https://pytask-dev.readthedocs.io/en/latest/how_to_guides/index.html>`_.
150+
<https://pytask-dev.readthedocs.io/en/stable/how_to_guides/index.html>`_.
151151

152152

153153
Changes
154154
-------
155155

156-
Consult the `release notes <https://pytask-dev.readthedocs.io/en/latest/changes.html>`_
156+
Consult the `release notes <https://pytask-dev.readthedocs.io/en/stable/changes.html>`_
157157
to find out about what is new.
158158

159159

docs/source/changes.rst

+1
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ all releases are available on `PyPI <https://pypi.org/project/pytask>`_ and
1515
- :pull:`218` removes ``depends_on`` and ``produces`` from the task function when
1616
parsed.
1717
- :pull:`219` removes some leftovers from pytest in :class:`~_pytask.mark.Mark`.
18+
- :pull:`221` adds more test cases for parametrizations.
1819
- :pull:`222` adds an automated Github Actions job for creating a list pytask plugins.
1920

2021

environment.yml

+1
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@ dependencies:
3434
- pydot
3535
- pytest
3636
- pytest-cov
37+
- pytest-xdist
3738
- tox-conda
3839

3940
# Documentation

setup.cfg

+3-3
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ name = pytask
33
description = In its highest aspirations, pytask tries to be pytest as a build system.
44
long_description = file: README.rst
55
long_description_content_type = text/x-rst
6-
url = https://pytask-dev.readthedocs.io/en/latest
6+
url = https://pytask-dev.readthedocs.io/en/stable
77
author = Tobias Raabe
88
author_email = [email protected]
99
license = MIT
@@ -26,8 +26,8 @@ classifiers =
2626
Topic :: Scientific/Engineering
2727
Topic :: Software Development :: Build Tools
2828
project_urls =
29-
Changelog = https://pytask-dev.readthedocs.io/en/latest/changes.html
30-
Documentation = https://pytask-dev.readthedocs.io/en/latest
29+
Changelog = https://pytask-dev.readthedocs.io/en/stable/changes.html
30+
Documentation = https://pytask-dev.readthedocs.io/en/stable
3131
Github = https://github.com/pytask-dev/pytask
3232
Tracker = https://github.com/pytask-dev/pytask/issues
3333

src/_pytask/parametrize.py

+7-2
Original file line numberDiff line numberDiff line change
@@ -95,8 +95,13 @@ def pytask_parametrize_task(
9595

9696
if len(markers) > 1:
9797
raise NotImplementedError(
98-
"Multiple parametrizations are currently not implemented since it is "
99-
"not possible to define products for tasks from a Cartesian product."
98+
"You cannot apply @pytask.mark.parametrize multiple times to a task. "
99+
"Use multiple for-loops, itertools.product or a different strategy to "
100+
"create all combinations of inputs and pass it to a single "
101+
"@pytask.mark.parametrize.\n\nFor improved readability, consider to "
102+
"move the creation of inputs into its own function as shown in the "
103+
"best-practices guide on parametrizations: https://pytask-dev.rtfd.io/"
104+
"en/stable/how_to_guides/bp_parametrizations.html."
100105
)
101106

102107
if has_marker(obj, "task"):

tests/test_collect.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -177,7 +177,7 @@ def test_pytask_collect_node_does_not_raise_error_if_path_is_not_normalized(
177177
if is_absolute:
178178
collected_node = tmp_path / collected_node
179179

180-
with warnings.catch_warnings() as record:
180+
with warnings.catch_warnings(record=True) as record:
181181
result = pytask_collect_node(session, task_path, collected_node)
182182
assert not record
183183

tests/test_live.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -169,7 +169,7 @@ def test_live_execution_displays_subset_of_table(capsys, tmp_path, n_entries_in_
169169

170170
live_manager.start()
171171
live.update_running_tasks(running_task)
172-
live_manager.stop()
172+
live_manager.stop(transient=False)
173173

174174
captured = capsys.readouterr()
175175
assert "Task" in captured.out

tests/test_parametrize.py

+129-41
Original file line numberDiff line numberDiff line change
@@ -3,12 +3,15 @@
33
import itertools
44
import textwrap
55
from contextlib import ExitStack as does_not_raise # noqa: N813
6+
from typing import NamedTuple
67

78
import _pytask.parametrize
89
import pytask
910
import pytest
1011
from _pytask.parametrize import _arg_value_to_id_component
12+
from _pytask.parametrize import _check_if_n_arg_names_matches_n_arg_values
1113
from _pytask.parametrize import _parse_arg_names
14+
from _pytask.parametrize import _parse_arg_values
1215
from _pytask.parametrize import _parse_parametrize_markers
1316
from _pytask.parametrize import pytask_parametrize_task
1417
from _pytask.pluginmanager import get_plugin_manager
@@ -54,14 +57,7 @@ def test_pytask_generate_tasks_1(session):
5457
def func(i, j): # noqa: U100
5558
pass
5659

57-
names_and_objs = pytask_parametrize_task(session, "func", func)
58-
59-
for (name, func), values in zip(
60-
names_and_objs, itertools.product(range(2), range(2))
61-
):
62-
assert name == f"func[{values[0]}-{values[1]}]"
63-
assert func.keywords["i"] == values[0]
64-
assert func.keywords["j"] == values[1]
60+
pytask_parametrize_task(session, "func", func)
6561

6662

6763
@pytest.mark.integration
@@ -72,16 +68,7 @@ def test_pytask_generate_tasks_2(session):
7268
def func(i, j, k): # noqa: U100
7369
pass
7470

75-
names_and_objs = pytask_parametrize_task(session, "func", func)
76-
77-
for (name, func), values in zip(
78-
names_and_objs,
79-
[(i, j, k) for i in range(2) for j in range(2) for k in range(2)],
80-
):
81-
assert name == f"func[{values[0]}-{values[1]}-{values[2]}]"
82-
assert func.keywords["i"] == values[0]
83-
assert func.keywords["j"] == values[1]
84-
assert func.keywords["k"] == values[2]
71+
pytask_parametrize_task(session, "func", func)
8572

8673

8774
@pytest.mark.integration
@@ -109,9 +96,32 @@ def func():
10996
(["i", "j"], ("i", "j")),
11097
],
11198
)
112-
def test_parse_argnames(arg_names, expected):
113-
parsed_argnames = _parse_arg_names(arg_names)
114-
assert parsed_argnames == expected
99+
def test_parse_arg_names(arg_names, expected):
100+
parsed_arg_names = _parse_arg_names(arg_names)
101+
assert parsed_arg_names == expected
102+
103+
104+
class TaskArguments(NamedTuple):
105+
a: int
106+
b: int
107+
108+
109+
@pytest.mark.unit
110+
@pytest.mark.parametrize(
111+
"arg_values, expected",
112+
[
113+
(["a", "b", "c"], [("a",), ("b",), ("c",)]),
114+
([(0, 0), (0, 1), (1, 0)], [(0, 0), (0, 1), (1, 0)]),
115+
([[0, 0], [0, 1], [1, 0]], [(0, 0), (0, 1), (1, 0)]),
116+
({"a": 0, "b": 1}, [("a",), ("b",)]),
117+
([TaskArguments(1, 2)], [(1, 2)]),
118+
([TaskArguments(a=1, b=2)], [(1, 2)]),
119+
([TaskArguments(b=2, a=1)], [(1, 2)]),
120+
],
121+
)
122+
def test_parse_arg_values(arg_values, expected):
123+
parsed_arg_values = _parse_arg_values(arg_values)
124+
assert parsed_arg_values == expected
115125

116126

117127
@pytest.mark.unit
@@ -267,28 +277,20 @@ def task_func(i):
267277

268278

269279
@pytest.mark.end_to_end
270-
@pytest.mark.xfail(strict=True, reason="Cartesian task product is disabled.")
271-
def test_two_parametrize_w_ids(tmp_path):
272-
tmp_path.joinpath("task_module.py").write_text(
273-
textwrap.dedent(
274-
"""
275-
import pytask
280+
def test_two_parametrize_w_ids(runner, tmp_path):
281+
source = """
282+
import pytask
276283
277-
@pytask.mark.parametrize('i', range(2), ids=["2.1", "2.2"])
278-
@pytask.mark.parametrize('j', range(2), ids=["1.1", "1.2"])
279-
def task_func(i, j):
280-
pass
281-
"""
282-
)
283-
)
284-
session = main({"paths": tmp_path})
284+
@pytask.mark.parametrize('i', range(2), ids=["2.1", "2.2"])
285+
@pytask.mark.parametrize('j', range(2), ids=["1.1", "1.2"])
286+
def task_func(i, j):
287+
pass
288+
"""
289+
tmp_path.joinpath("task_module.py").write_text(textwrap.dedent(source))
290+
result = runner.invoke(cli, [tmp_path.as_posix()])
285291

286-
assert session.exit_code == 0
287-
assert len(session.tasks) == 4
288-
for task, id_ in zip(
289-
session.tasks, ["[1.1-2.1]", "[1.1-2.2]", "[1.2-2.1]", "[1.2-2.2]"]
290-
):
291-
assert id_ in task.name
292+
assert result.exit_code == ExitCode.COLLECTION_FAILED
293+
assert "You cannot apply @pytask.mark.parametrize multiple" in result.output
292294

293295

294296
@pytest.mark.end_to_end
@@ -430,3 +432,89 @@ def task_example(produces):
430432
session = main({"paths": tmp_path})
431433
assert session.exit_code == 0
432434
assert session.tasks[0].function.__wrapped__.pytaskmark == []
435+
436+
437+
@pytest.mark.end_to_end
438+
def test_parametrizing_tasks_with_namedtuples(runner, tmp_path):
439+
source = """
440+
from typing import NamedTuple
441+
import pytask
442+
from pathlib import Path
443+
444+
445+
class Task(NamedTuple):
446+
i: int
447+
produces: Path
448+
449+
450+
@pytask.mark.parametrize('i, produces', [
451+
Task(i=1, produces="1.txt"), Task(produces="2.txt", i=2),
452+
])
453+
def task_write_numbers_to_file(produces, i):
454+
produces.write_text(str(i))
455+
"""
456+
tmp_path.joinpath("task_module.py").write_text(textwrap.dedent(source))
457+
458+
result = runner.invoke(cli, [tmp_path.as_posix()])
459+
460+
assert result.exit_code == 0
461+
for i in range(1, 3):
462+
assert tmp_path.joinpath(f"{i}.txt").read_text() == str(i)
463+
464+
465+
@pytest.mark.end_to_end
466+
def test_parametrization_with_different_n_of_arg_names_and_arg_values(runner, tmp_path):
467+
source = """
468+
import pytask
469+
470+
@pytask.mark.parametrize('i, produces', [(1, "1.txt"), (2, 3, "2.txt")])
471+
def task_write_numbers_to_file(produces, i):
472+
produces.write_text(str(i))
473+
"""
474+
tmp_path.joinpath("task_module.py").write_text(textwrap.dedent(source))
475+
476+
result = runner.invoke(cli, [tmp_path.as_posix()])
477+
478+
assert result.exit_code == ExitCode.COLLECTION_FAILED
479+
assert "Task 'task_write_numbers_to_file' is parametrized with 2" in result.output
480+
481+
482+
@pytest.mark.unit
483+
@pytest.mark.parametrize(
484+
"arg_names, arg_values, name, expectation",
485+
[
486+
pytest.param(
487+
("a",),
488+
[(1,), (2,)],
489+
"task_name",
490+
does_not_raise(),
491+
id="normal one argument parametrization",
492+
),
493+
pytest.param(
494+
("a", "b"),
495+
[(1, 2), (3, 4)],
496+
"task_name",
497+
does_not_raise(),
498+
id="normal two argument argument parametrization",
499+
),
500+
pytest.param(
501+
("a",),
502+
[(1, 2), (2,)],
503+
"task_name",
504+
pytest.raises(ValueError, match="Task 'task_name' is parametrized with 1"),
505+
id="error with one argument parametrization",
506+
),
507+
pytest.param(
508+
("a", "b"),
509+
[(1, 2), (3, 4, 5)],
510+
"task_name",
511+
pytest.raises(ValueError, match="Task 'task_name' is parametrized with 2"),
512+
id="error with two argument argument parametrization",
513+
),
514+
],
515+
)
516+
def test_check_if_n_arg_names_matches_n_arg_values(
517+
arg_names, arg_values, name, expectation
518+
):
519+
with expectation:
520+
_check_if_n_arg_names_matches_n_arg_values(arg_names, arg_values, name)

0 commit comments

Comments
 (0)