Skip to content

Commit cf36696

Browse files
authored
Convert cmd arguments to strings, remove --vanilla, display executed command and release v0.0.6. (#7)
1 parent 03cc983 commit cf36696

12 files changed

+88
-39
lines changed

.pre-commit-config.yaml

+1
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,7 @@ repos:
5454
flake8-print,
5555
flake8-pytest-style,
5656
flake8-todo,
57+
flake8-typing-imports,
5758
flake8-unused-arguments,
5859
pep8-naming,
5960
pydocstyle,

CHANGES.rst

+7
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,13 @@ chronological order. Releases follow `semantic versioning <https://semver.org/>`
66
all releases are available on `Anaconda.org <https://anaconda.org/pytask/pytask-r>`_.
77

88

9+
0.0.6 - 2021-01-16
10+
------------------
11+
12+
- :gh:`7` converts command line arguments automatically to string and removes the
13+
``--vanilla`` option as the default.
14+
15+
916
0.0.5 - 2020-10-30
1017
------------------
1118

README.rst

+20-11
Original file line numberDiff line numberDiff line change
@@ -98,6 +98,7 @@ for a ``"source"`` key in the dictionary and, secondly, under the key ``0``.
9898

9999
.. code-block:: python
100100
101+
@pytask.mark.r
101102
@pytask.mark.depends_on({"source": "script.r", "input": "input.rds"})
102103
def task_run_r_script():
103104
pass
@@ -106,6 +107,7 @@ for a ``"source"`` key in the dictionary and, secondly, under the key ``0``.
106107
# or
107108
108109
110+
@pytask.mark.r
109111
@pytask.mark.depends_on({0: "script.r", "input": "input.rds"})
110112
def task_run_r_script():
111113
pass
@@ -114,6 +116,7 @@ for a ``"source"`` key in the dictionary and, secondly, under the key ``0``.
114116
# or two decorators for the function, if you do not assign a name to the input.
115117
116118
119+
@pytask.mark.r
117120
@pytask.mark.depends_on({"source": "script.r"})
118121
@pytask.mark.depends_on("input.rds")
119122
def task_run_r_script():
@@ -123,13 +126,12 @@ for a ``"source"`` key in the dictionary and, secondly, under the key ``0``.
123126
Command Line Arguments
124127
~~~~~~~~~~~~~~~~~~~~~~
125128

126-
The decorator can be used to pass command line arguments to ``Rscript`` which is, by
127-
default, only the ``--vanilla`` flag. If you want to pass arguments to the script via
128-
the command line, use
129+
The decorator can be used to pass command line arguments to ``Rscript``. See the
130+
following example.
129131

130132
.. code-block:: python
131133
132-
@pytask.mark.r(["--vanilla", "value"])
134+
@pytask.mark.r("value")
133135
@pytask.mark.depends_on("script.r")
134136
@pytask.mark.produces("out.rds")
135137
def task_run_r_script():
@@ -140,7 +142,7 @@ And in your ``script.r``, you can intercept the value with
140142
.. code-block:: r
141143
142144
args <- commandArgs(trailingOnly=TRUE)
143-
arg <- args[1] # ``arg`` holds ``"value"``
145+
arg <- args[1] # holds ``"value"``
144146
145147
146148
Parametrization
@@ -153,13 +155,23 @@ The following task executes two R scripts which produce different outputs.
153155

154156
.. code-block:: python
155157
158+
from src.config import BLD, SRC
159+
160+
156161
@pytask.mark.r
157162
@pytask.mark.parametrize(
158-
"depends_on, produces", [("script_1.r", "1.rds"), ("script_2.r", "2.rds")]
163+
"depends_on, produces",
164+
[(SRC / "script_1.r", BLD / "1.rds"), (SRC / "script_2.r", BLD / "2.rds")],
159165
)
160166
def task_execute_r_script():
161167
pass
162168
169+
And the R script includes something like
170+
171+
.. code-block:: r
172+
173+
args <- commandArgs(trailingOnly=TRUE)
174+
produces <- args[1] # holds the path
163175
164176
If you want to pass different command line arguments to the same R script, you have to
165177
include the ``@pytask.mark.r`` decorator in the parametrization just like with
@@ -170,10 +182,7 @@ include the ``@pytask.mark.r`` decorator in the parametrization just like with
170182
@pytask.mark.depends_on("script.r")
171183
@pytask.mark.parametrize(
172184
"produces, r",
173-
[
174-
("output_1.rds", (["--vanilla", "1"],)),
175-
("output_2.rds", (["--vanilla", "2"],)),
176-
],
185+
[(BLD / "output_1.rds", "1"), (BLD / "output_2.rds", "2")],
177186
)
178187
def task_execute_r_script():
179188
pass
@@ -199,7 +208,7 @@ The plugin is a convenient wrapper around
199208
200209
import subprocess
201210
202-
subprocess.run(["Rscript", "--vanilla", "script.r"], check=True)
211+
subprocess.run(["Rscript", "script.r"], check=True)
203212
204213
to which you can always resort to when the plugin does not deliver functionality you
205214
need.

setup.cfg

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
[bumpversion]
2-
current_version = 0.0.5
2+
current_version = 0.0.6
33
parse = (?P<major>\d+)\.(?P<minor>\d+)(\.(?P<patch>\d+))(\-?((dev)?(?P<dev>\d+))?)
44
serialize =
55
{major}.{minor}.{patch}dev{dev}

setup.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33

44
setup(
55
name="pytask-r",
6-
version="0.0.5",
6+
version="0.0.6",
77
packages=find_packages(where="src"),
88
package_dir={"": "src"},
99
entry_points={"pytask": ["pytask_r = pytask_r.plugin"]},

src/pytask_r/__init__.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -1 +1 @@
1-
__version__ = "0.0.5"
1+
__version__ = "0.0.6"

src/pytask_r/collect.py

+33-9
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,9 @@
22
import copy
33
import functools
44
import subprocess
5-
from pathlib import Path
65
from typing import Iterable
76
from typing import Optional
7+
from typing import Sequence
88
from typing import Union
99

1010
from _pytask.config import hookimpl
@@ -24,15 +24,14 @@ def r(options: Optional[Union[str, Iterable[str]]] = None):
2424
One or multiple command line options passed to Rscript.
2525
2626
"""
27-
if options is None:
28-
options = ["--vanilla"]
29-
elif isinstance(options, str):
30-
options = [options]
27+
options = [] if options is None else _to_list(options)
28+
options = [str(i) for i in options]
3129
return options
3230

3331

3432
def run_r_script(r):
3533
"""Run an R script."""
34+
print("Executing " + " ".join(r) + ".") # noqa: T001
3635
subprocess.run(r, check=True)
3736

3837

@@ -75,9 +74,8 @@ def pytask_collect_task_teardown(session, task):
7574

7675

7776
def _get_node_from_dictionary(obj, key, fallback=0):
78-
if isinstance(obj, Path):
79-
pass
80-
elif isinstance(obj, dict):
77+
"""Get node from dictionary."""
78+
if isinstance(obj, dict):
8179
obj = obj.get(key) or obj.get(fallback)
8280
return obj
8381

@@ -99,4 +97,30 @@ def _prepare_cmd_options(session, task, args):
9997
10098
"""
10199
source = _get_node_from_dictionary(task.depends_on, session.config["r_source_key"])
102-
return ["Rscript", source.value.as_posix(), *args]
100+
return ["Rscript", source.path.as_posix(), *args]
101+
102+
103+
def _to_list(scalar_or_iter):
104+
"""Convert scalars and iterables to list.
105+
106+
Parameters
107+
----------
108+
scalar_or_iter : str or list
109+
110+
Returns
111+
-------
112+
list
113+
114+
Examples
115+
--------
116+
>>> _to_list("a")
117+
['a']
118+
>>> _to_list(["b"])
119+
['b']
120+
121+
"""
122+
return (
123+
[scalar_or_iter]
124+
if isinstance(scalar_or_iter, str) or not isinstance(scalar_or_iter, Sequence)
125+
else list(scalar_or_iter)
126+
)

src/pytask_r/parametrize.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,4 +8,4 @@ def pytask_parametrize_kwarg_to_marker(obj, kwargs):
88
"""Attach parametrized r arguments to the function with a marker."""
99
if callable(obj):
1010
if "r" in kwargs:
11-
mark.r(*kwargs.pop("r"))(obj)
11+
mark.r(kwargs.pop("r"))(obj)

tests/test_collect.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ def task_dummy():
2424
@pytest.mark.parametrize(
2525
"r_args, expected",
2626
[
27-
(None, ["--vanilla"]),
27+
(None, []),
2828
("--some-option", ["--some-option"]),
2929
(["--a", "--b"], ["--a", "--b"]),
3030
],
@@ -132,7 +132,7 @@ def test_prepare_cmd_options(args, r_source_key):
132132
session.config = {"r_source_key": r_source_key}
133133

134134
node = DummyClass()
135-
node.value = Path("script.r")
135+
node.path = Path("script.r")
136136
task = DummyClass()
137137
task.depends_on = {r_source_key: node}
138138
task.name = "task"

tests/test_parallel.py

+10-10
Original file line numberDiff line numberDiff line change
@@ -82,15 +82,14 @@ def test_parallel_parametrization_over_source_file(runner, tmp_path):
8282

8383
source = """
8484
import pytask
85+
from pathlib import Path
86+
87+
SRC = Path(__file__).parent
8588
8689
@pytask.mark.depends_on("script.r")
87-
@pytask.mark.parametrize(
88-
"produces, r",
89-
[
90-
("output_1.rds", (["--vanilla", "1"],)),
91-
("output_2.rds", (["--vanilla", "2"],)),
92-
],
93-
)
90+
@pytask.mark.parametrize("produces, r", [
91+
(SRC / "0.rds", (1, SRC / "0.rds")), (SRC / "1.rds", (1, SRC / "1.rds"))
92+
])
9493
def task_execute_r_script():
9594
pass
9695
"""
@@ -99,8 +98,9 @@ def task_execute_r_script():
9998
r_script = """
10099
Sys.sleep(2)
101100
args <- commandArgs(trailingOnly=TRUE)
102-
number <- args[2]
103-
saveRDS(number, file=paste0("output_", number, ".rds"))
101+
number <- args[1]
102+
produces <- args[2]
103+
saveRDS(number, file=produces)
104104
"""
105105
tmp_path.joinpath("script.r").write_text(textwrap.dedent(r_script))
106106

@@ -109,7 +109,7 @@ def task_execute_r_script():
109109
assert result.exit_code == 0
110110
duration_normal = time.time() - start
111111

112-
for name in ["output_1.rds", "output_2.rds"]:
112+
for name in ["0.rds", "1.rds"]:
113113
tmp_path.joinpath(name).unlink()
114114

115115
start = time.time()

tests/test_parametrize.py

+9-3
Original file line numberDiff line numberDiff line change
@@ -43,12 +43,17 @@ def task_run_r_script():
4343

4444
@needs_rscript
4545
@pytest.mark.end_to_end
46-
def test_parametrize_r_options(tmp_path):
46+
def test_parametrize_r_options_and_product_paths(tmp_path):
4747
task_source = """
4848
import pytask
49+
from pathlib import Path
50+
51+
SRC = Path(__file__).parent
4952
5053
@pytask.mark.depends_on("script.r")
51-
@pytask.mark.parametrize("produces, r", [("0.rds", "0"), ("1.rds", "1")])
54+
@pytask.mark.parametrize("produces, r", [
55+
(SRC / "0.rds", (0, SRC / "0.rds")), (SRC / "1.rds", (1, SRC / "1.rds"))
56+
])
5257
def task_execute_r_script():
5358
pass
5459
"""
@@ -57,7 +62,8 @@ def task_execute_r_script():
5762
r_script = """
5863
args <- commandArgs(trailingOnly=TRUE)
5964
number <- args[1]
60-
saveRDS(number, file=paste0(number, ".rds"))
65+
produces <- args[2]
66+
saveRDS(number, file=produces)
6167
"""
6268
tmp_path.joinpath("script.r").write_text(textwrap.dedent(r_script))
6369

tox.ini

+2
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,8 @@ warn-symbols =
4545
addopts = --doctest-modules
4646
filterwarnings =
4747
ignore: the imp module is deprecated in favour of importlib
48+
ignore: The (parser|symbol) module is deprecated and will be removed in future
49+
ignore: Using or importing the ABCs from 'collections' instead of from
4850
junit_family = xunit2
4951
markers =
5052
wip: Tests that are work-in-progress.

0 commit comments

Comments
 (0)