Skip to content

Commit 5b91510

Browse files
committed
Implement smooth transition to attributes on nodes.
1 parent adf9376 commit 5b91510

File tree

7 files changed

+74
-6
lines changed

7 files changed

+74
-6
lines changed

src/_pytask/click.py

+3-2
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424

2525
from _pytask import __version__ as version
2626
from _pytask.console import console
27+
from _pytask.console import create_panel_title
2728

2829
if TYPE_CHECKING:
2930
from collections.abc import Sequence
@@ -109,7 +110,7 @@ def format_help(
109110
console.print(
110111
Panel(
111112
commands_table,
112-
title="[bold #f2f2f2]Commands[/]",
113+
title=create_panel_title("Commands"),
113114
title_align="left",
114115
border_style="grey37",
115116
)
@@ -244,7 +245,7 @@ def _print_options(group_or_command: Command | DefaultGroup, ctx: Context) -> No
244245
console.print(
245246
Panel(
246247
options_table,
247-
title="[bold #f2f2f2]Options[/]",
248+
title=create_panel_title("Options"),
248249
title_align="left",
249250
border_style="grey37",
250251
)

src/_pytask/collect.py

+11
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
import os
88
import sys
99
import time
10+
import warnings
1011
from contextlib import suppress
1112
from pathlib import Path
1213
from typing import TYPE_CHECKING
@@ -385,6 +386,16 @@ def pytask_collect_node( # noqa: C901, PLR0912
385386
"""
386387
node = node_info.value
387388

389+
if isinstance(node, (PNode, PProvisionalNode)) and not hasattr(node, "attributes"):
390+
warnings.warn(
391+
"PNode and PProvisionalNode will require an 'attributes' field starting "
392+
"with pytask v0.6.0. It is a dictionary with any type of key and values "
393+
"similar to PTask. See https://tinyurl.com/pytask-custom-nodes for more "
394+
"information about adjusting your custom nodes.",
395+
stacklevel=1,
396+
category=FutureWarning,
397+
)
398+
388399
if isinstance(node, DirectoryNode):
389400
if node.root_dir is None:
390401
node.root_dir = path

src/_pytask/console.py

+6-1
Original file line numberDiff line numberDiff line change
@@ -296,10 +296,15 @@ def create_summary_panel(
296296

297297
return Panel(
298298
grid,
299-
title="[bold #f2f2f2]Summary[/]",
299+
title=create_panel_title("Summary"),
300300
expand=False,
301301
style="none",
302302
border_style=outcome_enum.FAIL.style
303303
if counts[outcome_enum.FAIL]
304304
else outcome_enum.SUCCESS.style,
305305
)
306+
307+
308+
def create_panel_title(title: str) -> Text:
309+
"""Create a title for a panel."""
310+
return Text(title, style="bold #f2f2f2")

src/_pytask/node_protocols.py

-2
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,6 @@ class PNode(Protocol):
2121
"""Protocol for nodes."""
2222

2323
name: str
24-
attributes: dict[Any, Any]
2524

2625
@property
2726
def signature(self) -> str:
@@ -117,7 +116,6 @@ class PProvisionalNode(Protocol):
117116
"""
118117

119118
name: str
120-
attributes: dict[Any, Any]
121119

122120
@property
123121
def signature(self) -> str:

src/_pytask/nodes.py

+8
Original file line numberDiff line numberDiff line change
@@ -162,6 +162,8 @@ class PathNode(PPathNode):
162162
Name of the node which makes it identifiable in the DAG.
163163
path
164164
The path to the file.
165+
attributes: dict[Any, Any]
166+
A dictionary to store additional information of the task.
165167
166168
"""
167169

@@ -220,6 +222,8 @@ class PythonNode(PNode):
220222
objects. The function should return either an integer or a string.
221223
node_info
222224
The infos acquired while collecting the node.
225+
attributes: dict[Any, Any]
226+
A dictionary to store additional information of the task.
223227
224228
Examples
225229
--------
@@ -304,6 +308,8 @@ class PickleNode(PPathNode):
304308
Name of the node which makes it identifiable in the DAG.
305309
path
306310
The path to the file.
311+
attributes: dict[Any, Any]
312+
A dictionary to store additional information of the task.
307313
308314
"""
309315

@@ -353,6 +359,8 @@ class DirectoryNode(PProvisionalNode):
353359
root_dir
354360
The pattern is interpreted relative to the path given by ``root_dir``. If
355361
``root_dir = None``, it is the directory where the path is defined.
362+
attributes: dict[Any, Any]
363+
A dictionary to store additional information of the task.
356364
357365
"""
358366

src/_pytask/warnings.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@
1212
from rich.panel import Panel
1313

1414
from _pytask.console import console
15+
from _pytask.console import create_panel_title
1516
from _pytask.pluginmanager import hookimpl
1617
from _pytask.warnings_utils import WarningReport
1718
from _pytask.warnings_utils import catch_warnings_for_item
@@ -82,7 +83,9 @@ def pytask_log_session_footer(session: Session) -> None:
8283
"""Log warnings at the end of a session."""
8384
if session.warnings:
8485
renderable = _WarningsRenderable(session.warnings)
85-
panel = Panel(renderable, title="Warnings", style="warning")
86+
panel = Panel(
87+
renderable, title=create_panel_title("Warnings"), style="warning"
88+
)
8689
console.print(panel)
8790

8891

tests/test_node_protocols.py

+42
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
def test_node_protocol_for_custom_nodes(runner, tmp_path):
1414
source = """
1515
from typing import Annotated
16+
from typing import Any
1617
from pytask import Product
1718
from attrs import define
1819
from pathlib import Path
@@ -22,6 +23,7 @@ class CustomNode:
2223
name: str
2324
value: str
2425
signature: str = "id"
26+
attributes: dict[Any, Any] = {}
2527
2628
def state(self):
2729
return self.value
@@ -43,12 +45,14 @@ def task_example(
4345
result = runner.invoke(cli, [tmp_path.as_posix()])
4446
assert result.exit_code == ExitCode.OK
4547
assert tmp_path.joinpath("out.txt").read_text() == "text"
48+
assert "FutureWarning" not in result.output
4649

4750

4851
@pytest.mark.end_to_end
4952
def test_node_protocol_for_custom_nodes_with_paths(runner, tmp_path):
5053
source = """
5154
from typing import Annotated
55+
from typing import Any
5256
from pytask import Product
5357
from pathlib import Path
5458
from attrs import define
@@ -60,6 +64,7 @@ class PickleFile:
6064
path: Path
6165
value: Path
6266
signature: str = "id"
67+
attributes: dict[Any, Any] = {}
6368
6469
def state(self):
6570
return str(self.path.stat().st_mtime)
@@ -87,3 +92,40 @@ def task_example(
8792
result = runner.invoke(cli, [tmp_path.as_posix()])
8893
assert result.exit_code == ExitCode.OK
8994
assert tmp_path.joinpath("out.txt").read_text() == "text"
95+
96+
97+
@pytest.mark.end_to_end
98+
def test_node_protocol_for_custom_nodes_adding_attributes(runner, tmp_path):
99+
source = """
100+
from typing import Annotated
101+
from pytask import Product
102+
from attrs import define
103+
from pathlib import Path
104+
105+
@define
106+
class CustomNode:
107+
name: str
108+
value: str
109+
signature: str = "id"
110+
111+
def state(self):
112+
return self.value
113+
114+
def load(self, is_product):
115+
return self.value
116+
117+
def save(self, value):
118+
self.value = value
119+
120+
def task_example(
121+
data = CustomNode("custom", "text"),
122+
out: Annotated[Path, Product] = Path("out.txt"),
123+
) -> None:
124+
out.write_text(data)
125+
"""
126+
tmp_path.joinpath("task_module.py").write_text(textwrap.dedent(source))
127+
128+
result = runner.invoke(cli, [tmp_path.as_posix()])
129+
assert result.exit_code == ExitCode.OK
130+
assert tmp_path.joinpath("out.txt").read_text() == "text"
131+
assert "FutureWarning" in result.output

0 commit comments

Comments
 (0)