From 9c81017e7d00bf494769ba50fe5519adb94988dc Mon Sep 17 00:00:00 2001 From: Thomas Sibley Date: Thu, 15 Aug 2024 10:50:11 -0700 Subject: [PATCH] snakemake-style-guide: Recommend *raw* triple-quoted strings for shell blocks This greatly enhances the readability of the shell block in Snakemake's log messages and avoids accidentally interpreting escapes too early. --- src/reference/snakemake-style-guide.rst | 30 +++++++++++++++++-------- 1 file changed, 21 insertions(+), 9 deletions(-) diff --git a/src/reference/snakemake-style-guide.rst b/src/reference/snakemake-style-guide.rst index 074890d5..c38708cf 100644 --- a/src/reference/snakemake-style-guide.rst +++ b/src/reference/snakemake-style-guide.rst @@ -102,7 +102,7 @@ For example, do this: params: name = lambda _: config["name"] shell: - """ + r""" echo {params.name:q} """ @@ -152,7 +152,7 @@ the ``:q`` modifier for interpolation: params: file = "filename with spaces.txt" shell: - """ + r""" wc -l {params.file:q} """ @@ -169,24 +169,28 @@ string. Snakemake will interpolate it correctly: params: files = ["a.txt", "b.txt", "c.txt"] shell: - """ + r""" wc -l {params.files:q} """ -Use triple-quoted command definitions -===================================== +.. _use-triple-quoted-command-definitions: + +Use raw, triple-quoted shell blocks +=================================== -Using triple-quoted (``"""`` or ``'''``) command definitions make it +Using raw, triple-quoted (``r"""`` or ``r'''``) ``shell`` blocks makes it much easier to build readable commands with one-option per line. It also avoids any nested quoting issues if you need to use literal single or -double quotes in your command. +double quotes in your command. The command will remain readable in +Snakemake's logging messages because it'll look like the source form +(e.g. with backslashes and newlines retained instead of collapsed). Example: .. code-block:: python shell: - """ + r""" augur parse \ --sequences {input:q} \ --fields {params.fields:q} \ @@ -194,6 +198,14 @@ Example: --output-metadata {output.metadata:q} """ +.. hint:: + If you're converting interpreted strings to raw strings (e.g. + ``"""`` to ``r"""``), make sure to check that they're not relying on + `escape sequences`_ like ``\n``, ``\t``, or ``\\`` to be interpreted by + Python before the shell (Bash) sees them. + +.. _escape sequences: https://docs.python.org/3/reference/lexical_analysis.html#escape-sequences + Log standard out and error output to log files and the terminal =============================================================== @@ -212,7 +224,7 @@ Example: log: "logs/filter.txt" shell: - """ + r""" augur filter \ --metadata {input.metadata} \ --output-metadata {output.metadata} 2>&1 | tee {log}