From 1a74fe1a641b847d1d0c8f8a1c88017b630cd1f3 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Tue, 5 Nov 2024 14:46:36 +0100 Subject: [PATCH 01/32] Minor text change --- pages/git.qmd | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pages/git.qmd b/pages/git.qmd index 215e1a78..2d8843f6 100644 --- a/pages/git.qmd +++ b/pages/git.qmd @@ -35,7 +35,7 @@ There are many benefits of using Git in your research project: to collaborate by tracking all edits made by each person. It will also handle any potential conflicting edits. - Using a cloud-based repository hosting service (the one you push your commits - to), like _e.g._ [GitHub](https://github.com/) or + to), like [GitHub](https://github.com/) or [Bitbucket](https://bitbucket.org/), adds additional features, such as being able to discuss the project, comment on edits, or report issues. - If at some point your project will be published GitHub or Bitbucket (or From 958038ebdb1de63a19f9ed386f0ef8e94211c5ca Mon Sep 17 00:00:00 2001 From: John Sundh Date: Wed, 6 Nov 2024 08:13:29 +0100 Subject: [PATCH 02/32] Minor change to wording --- pages/git.qmd | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/pages/git.qmd b/pages/git.qmd index 2d8843f6..42451df5 100644 --- a/pages/git.qmd +++ b/pages/git.qmd @@ -516,10 +516,10 @@ displays the difference on a word-per-word basis rather than line-per-line. ::: {.callout-note} Git is constantly evolving, along with some of its commands. The `checkout` -command was previously used for switching between branches, but this -functionality now has the dedicated (and clearer) `switch` command for this. -If you've previously learned using `checkout` instead you can keep doing that -without any issues, as the `checkout` command itself hasn't changed. +command was previously used for switching between branches, but now there's the +dedicated (and clearer) `switch` command for this functionality. If you've +previously learned using `checkout` instead you can keep doing that without any +issues, as the `checkout` command itself hasn't changed. ::: Now, let's assume that we have tested our code and the alignment analysis is run From 9760e0157cf51a04c517e708f2ba2be209506c1a Mon Sep 17 00:00:00 2001 From: John Sundh Date: Wed, 6 Nov 2024 08:14:48 +0100 Subject: [PATCH 03/32] Minor word change --- pages/git.qmd | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pages/git.qmd b/pages/git.qmd index 42451df5..c0ebc07d 100644 --- a/pages/git.qmd +++ b/pages/git.qmd @@ -540,7 +540,7 @@ git merge test_alignment ``` Run `git log --graph --all --oneline` again to see how the merge commit brings -back the changes made in `test_alignment` to `main`. +the changes made in `test_alignment` into `main`. ::: {.callout-tip} If working on different features or parts of an analysis on different From d857eb146da0de3ca006f2fd2314563b9ccb79a0 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Fri, 8 Nov 2024 09:56:05 +0100 Subject: [PATCH 04/32] Update conda instructions for new macs --- home_precourse.qmd | 24 +++++++++++++++++------- 1 file changed, 17 insertions(+), 7 deletions(-) diff --git a/home_precourse.qmd b/home_precourse.qmd index c9af8913..d1b76d55 100644 --- a/home_precourse.qmd +++ b/home_precourse.qmd @@ -284,18 +284,28 @@ into some problems with certain Conda packages that have not yet been built for the ARM64 architecture. The [Rosetta](https://support.apple.com/en-us/HT211861) software allows ARM64 Macs to use software built for the old AMD64 architecture, which means you can always fall back on creating AMD/Intel-based environments -and use them in conjunction with Rosetta. This is how you do it: +and use them in conjunction with Rosetta. This can be done by specifying +`--subdir osx-64` when creating the environment, _e.g._: + +```bash +conda env create -f --subdir osx-64 +``` + +or + +```bash +conda create -n myenv --subdir osx-64 +``` + +::: {.callout-important} +To make sure that subsequent installations into this environment also use the +`osx-64` architecture, activate the environment and then run: ```bash -CONDA_SUBDIR=osx-64 -conda activate conda config --env --set subdir osx-64 ``` +::: -The first command creates the Intel-based environment, while the last one -makes sure that subsequent commands are also using the Intel architecture. If -you don't want to remember and do this manually each time you want to use -AMD64/Rosetta you can check out [this bash script](https://github.com/fasterius/dotfiles/blob/main/scripts/intel-conda-env.sh). ## Installing Snakemake From e7f8742e5077b1ff693e7201f086d57b0c111ee4 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Fri, 8 Nov 2024 12:51:39 +0100 Subject: [PATCH 05/32] Minor rewording --- pages/git.qmd | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/pages/git.qmd b/pages/git.qmd index c0ebc07d..806b8bd1 100644 --- a/pages/git.qmd +++ b/pages/git.qmd @@ -719,9 +719,11 @@ git remote add origin git@github.com:user/git_tutorial.git your local Git clone. The short name of the default remote is usually "_origin_" by convention. -::: {.callout-note} +::: {.callout-note} Make sure you've used an SSH address (_i.e._ starting with `git@github.com` -rather than an HTTPS address (starting with `https://github.com`)! +rather than an HTTPS address (starting with `https://github.com`)! Also make +sure you've set up ssh-keys as described in the +[github-setup](../home_precourse.html#github-setup) in the pre-course material. ::: - We have not yet synced the local and remote repositories, though, we've simply @@ -897,8 +899,8 @@ a shorthand for `git fetch` followed by `git merge FETCH_HEAD` (where That's quite a few concepts and commands you've just learnt! It can be a bit hard to keep track of everything and the connections between local and remote -Git repositories and how you work with them, but hopefully the following figure -will give you a short visual summary: +Git repositories and how you work with them. The following figure will give you +a short visual summary: ![](images/git_overview_remote.png){ width=600px } From e4b1a2294a39909e7a31e8bb42e4603edec873f3 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Fri, 8 Nov 2024 12:52:27 +0100 Subject: [PATCH 06/32] Fix misspelling --- pages/git.qmd | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pages/git.qmd b/pages/git.qmd index 806b8bd1..f66b57fa 100644 --- a/pages/git.qmd +++ b/pages/git.qmd @@ -918,7 +918,7 @@ repositories and how to sync them: ### Remote branches -Remote branches work much in the same way a local branches, but you have to +Remote branches work much in the same way as local branches, but you have to push them separately; you might have noticed that GitHub only listed our repository as having one branch (you can see this by going to the _Code_ tab). This is because we only pushed our `main` branch to the remote. Let's create From aa8b9acad7e10f33bf84009009e1196e72e89fa6 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Tue, 12 Nov 2024 21:19:57 +0100 Subject: [PATCH 07/32] Fix callout --- pages/conda.qmd | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/pages/conda.qmd b/pages/conda.qmd index 3d552e98..5500efb1 100644 --- a/pages/conda.qmd +++ b/pages/conda.qmd @@ -41,8 +41,7 @@ A Conda _environment_ is essentially a directory that is added to your PATH and that contains a specific collection of packages that you have installed. Packages are symlinked between environments to avoid unnecessary duplication. -::: {.callout-} -**Different Conda flavours** +::: {.callout-note title="Different Conda flavours"} You may come across several flavours of Conda. There's _Miniconda_, which is the installer for Conda. The second is _Anaconda_, which is a distribution of not only Conda, but also over 150 scientific Python packages curated by the From a2ca8ba8cf80c380c0f388070e0eed3b8852f405 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Tue, 12 Nov 2024 21:20:31 +0100 Subject: [PATCH 08/32] Remove duplicate wording --- pages/conda.qmd | 17 +++++++++-------- 1 file changed, 9 insertions(+), 8 deletions(-) diff --git a/pages/conda.qmd b/pages/conda.qmd index 5500efb1..5d2f08ad 100644 --- a/pages/conda.qmd +++ b/pages/conda.qmd @@ -51,14 +51,15 @@ not even use. Then, lastly, there's the _Miniforge_ flavour that we're using here, which is a community-driven version of Conda that's highly popular within the scientific community. -The difference between Miniconda and Miniforge is that the former points to -points to the `default` channel by default (which requires an Anaconda license -for commercial purposes), while the latter points to the community-maintained -`conda-forge` channel by default. While Conda is created and owned by Anaconda -the company, Conda itself is open source - it's the `default` channel that is -proprietary. The `conda-forge` and `bioconda` channels (two of the largest -channels outside of `default`) are community-driven. Confusing? Yes. If you -want this information more in-depth you can read this [blog post by Anaconda](https://www.anaconda.com/blog/is-conda-free). +The difference between Miniconda and Miniforge is that the former points to the +`default` channel by default (which requires an Anaconda license for commercial +purposes), while the latter points to the community-maintained `conda-forge` +channel by default. While Conda is created and owned by Anaconda the company, +Conda itself is open source - it's the `default` channel that is proprietary. +The `conda-forge` and `bioconda` channels (two of the largest channels outside +of `default`) are community-driven. Confusing? Yes. If you want this information +more in-depth you can read this [blog post by +Anaconda](https://www.anaconda.com/blog/is-conda-free). ::: ## The basics From abee898812d092721e43b0c9720194fbe59bcde2 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Wed, 13 Nov 2024 08:37:08 +0100 Subject: [PATCH 09/32] Add details about conda channels in basics section --- pages/conda.qmd | 47 +++++++++++++++++++++++++++++++++++++++-------- 1 file changed, 39 insertions(+), 8 deletions(-) diff --git a/pages/conda.qmd b/pages/conda.qmd index 5d2f08ad..2fa95dd9 100644 --- a/pages/conda.qmd +++ b/pages/conda.qmd @@ -80,12 +80,43 @@ called _Project A_. - Let's make our first Conda environment: ```bash -conda create -n project_a -c bioconda fastqc +conda create -n project_a fastqc ``` -This will create an environment called `project_a`, containing FastQC from the -Bioconda channel. Conda will list the packages that will be installed and ask -for your confirmation. +This will create an environment called `project_a`, containing FastQC. + +You should see something like this printed to the terminal: + +``` +Channels: + - conda-forge + - bioconda + - defaults +Platform: osx-arm64 # <- Your platform may differ +``` + +This is Conda telling you which channels it is looking in for the package you +requested. If you followed the +[pre-course](../home_precourse.html##configuring-conda) instructions and added +the `conda-forge` and `bioconda` channels to your conda configuration, you +should see them listed here. If you had not configured Conda to use these +channels, you would have to specify them when installing packages, _e.g._ `conda +install -c bioconda fastqc`. + +Further down you will see something like this: + +``` +The following NEW packages will be INSTALLED: + + fastqc bioconda/noarch::fastqc-0.12.1-hdfd78af_0 + +``` + +This shows that the `fastqc` package will be installed from the `bioconda` +channel (the `noarch` part shows that the package is not specific to any +computer architecture). In the example above we see that version `0.12.1` will +be installed. The `hdfd78af_0` part is a unique build string that is used to +differentiate between different builds of the same package version. - Once it is done, you can activate the environment: @@ -142,7 +173,7 @@ activated. install`. Make sure that `project_a` is the active environment first. ```bash -conda install -c bioconda multiqc +conda install multiqc ``` - If we don't specify the package version, the latest available version will be @@ -150,13 +181,13 @@ conda install -c bioconda multiqc - Run the following to see what versions are available: ```bash -conda search -c bioconda multiqc +conda search multiqc ``` - Now try to install a different version of MultiQC, _e.g._: ```bash -conda install -c bioconda multiqc=1.13 +conda install multiqc=1.13 ``` Read the information that Conda displays in the terminal. It probably asks if @@ -196,7 +227,7 @@ conda create -n project_old -c bioconda bbmap=37.10 Now let's try to remove an installed package from the active environment: -``` +```bash conda remove multiqc ``` From d516d5a700411939ccacaa6baea2ca277189af06 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Wed, 13 Nov 2024 08:40:49 +0100 Subject: [PATCH 10/32] Fix link --- pages/conda.qmd | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pages/conda.qmd b/pages/conda.qmd index 2fa95dd9..fb6504ab 100644 --- a/pages/conda.qmd +++ b/pages/conda.qmd @@ -97,7 +97,7 @@ Platform: osx-arm64 # <- Your platform may differ This is Conda telling you which channels it is looking in for the package you requested. If you followed the -[pre-course](../home_precourse.html##configuring-conda) instructions and added +[pre-course](../home_precourse.html#configuring-conda) instructions and added the `conda-forge` and `bioconda` channels to your conda configuration, you should see them listed here. If you had not configured Conda to use these channels, you would have to specify them when installing packages, _e.g._ `conda From 4c48bdc981c662da5acc768af7d53f9c4af911d1 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Wed, 13 Nov 2024 08:46:21 +0100 Subject: [PATCH 11/32] Minor update to instructions --- pages/conda.qmd | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/pages/conda.qmd b/pages/conda.qmd index fb6504ab..d1555e62 100644 --- a/pages/conda.qmd +++ b/pages/conda.qmd @@ -321,9 +321,9 @@ Conda installation path. - Activate the environment! -- Now we can run the code for the MRSA project found in `code/run_qc.sh`, - either by running `bash code/run_qc.sh` or by opening the `run_qc.sh` file - and executing each line in the terminal one by one. Do this! +- Now we can run the code for the MRSA project found in `code/run_qc.sh` + by running `bash code/run_qc.sh`. Do this! (You can also open the file + and run the commands manually if you prefer.) This should download the project FASTQ files and run FastQC on them (as mentioned above). From 98f7ec7a7a8a7dd8b8a384972b1867c454752b46 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Wed, 13 Nov 2024 21:23:03 +0100 Subject: [PATCH 12/32] Add callout about M1 python --- pages/conda.qmd | 13 ++++++++++++- 1 file changed, 12 insertions(+), 1 deletion(-) diff --git a/pages/conda.qmd b/pages/conda.qmd index d1555e62..9218fadb 100644 --- a/pages/conda.qmd +++ b/pages/conda.qmd @@ -500,7 +500,18 @@ directory of the currently active environment. When you create a new Conda environment you can choose to install a specific version of Python in that environment as well. As an example, create an -environment containing Python version `3.5` by running: +environment containing Python version `3.5`: + +::: {.callout-note} + +Python versions older than `v3.8.5` are not available for Macs with the M-series +chips, so if you are using one of those you will need to add `--subdir osx-64` +to the command, _e.g._: + +```bash +conda create --subdir osx-64 -n py35 python=3.5 +``` +::: ```bash conda create -n py35 python=3.5 From beba69e19a687bf7f3f1dd43b86eb1cb194d1e12 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Wed, 13 Nov 2024 21:57:49 +0100 Subject: [PATCH 13/32] Remove deprecated flags --- pages/snakemake.qmd | 23 ++++++++++------------- 1 file changed, 10 insertions(+), 13 deletions(-) diff --git a/pages/snakemake.qmd b/pages/snakemake.qmd index bce62a9a..ac4b7ed2 100644 --- a/pages/snakemake.qmd +++ b/pages/snakemake.qmd @@ -101,19 +101,16 @@ will convert all characters in the set `[a-z]` to the corresponding character in the set `[A-Z]`, and then send the output to `a.upper.txt`". Now let's run our first Snakemake workflow. When a workflow is executed -Snakemake tries to generate a set of target files. Target files can be -specified via the command line (or, as you will see later, in several other -ways). Here we ask Snakemake to make the file `a.upper.txt`. We can specify -the file containing our rules with `-s` but since the default behaviour of -Snakemake is to look for a file called `Snakefile` in either the working -directory or in a subdirectory called `workflow/` we don't need to specify -that here. It's good practice to first run with the flag `-n` (or -`--dry-run`), which will show what Snakemake plans to do without actually -running anything, and you also need to specify how many cores to be used for -the workflow with `--cores` or `-c`. For now, you only need 1 so set `-c 1`. -You can also use the flag `-p`, for showing the shell commands that it will -execute, and the flag `-r` for showing the reason for running a specific -rule. `snakemake --help` will show you all available flags. +Snakemake tries to generate a set of target files. Target files can be specified +via the command line (or, as you will see later, in several other ways). Here we +ask Snakemake to make the file `a.upper.txt`. We can specify the file containing +our rules with `-s` but since the default behaviour of Snakemake is to look for +a file called `Snakefile` in either the working directory or in a subdirectory +called `workflow/` we don't need to specify that here. It's good practice to +first run with the flag `-n` (or `--dry-run`), which will show what Snakemake +plans to do without actually running anything. You can also use the flag `-p`, +for showing the shell commands that it will execute. `snakemake --help` will +show you all available flags. ```no-highlight $ snakemake -n -c 1 -r -p a.upper.txt From 73c161d621d439e89d571e604e5b5d3a53275b86 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Wed, 13 Nov 2024 21:58:04 +0100 Subject: [PATCH 14/32] Move callout for snakemake env --- home_precourse.qmd | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/home_precourse.qmd b/home_precourse.qmd index 74f5ca29..5573f74a 100644 --- a/home_precourse.qmd +++ b/home_precourse.qmd @@ -313,19 +313,19 @@ We will use Conda environments for the set up of this tutorial, but don't worry if you don't understand exactly what everything does - you'll learn all the details at the course. First make sure you're currently situated inside the tutorials directory (`workshop-reproducible-research/tutorials`) and then create -the Conda environment like so: +and activate the Conda environment with the commands below: + +::: {.callout-caution title="ARM64 users"} +Some of the packages in this environment are not available for the ARM64 +architecture, so you'll have to add `--subdir osx-64` to the `conda env create` +command. See the [instructions above](#conda-on-new-macs) for more details. +::: ```bash conda env create -f snakemake/environment.yml -n snakemake-env conda activate snakemake-env ``` -::: {.callout-caution title="ARM64 users"} -Some of the packages in this environment is not available for the ARM64 -architecture, so you'll have to follow the [instructions -above](#conda-on-new-macs). -::: - Check that Snakemake is installed correctly, for example by executing `snakemake --help`. This should output a list of available Snakemake settings. If you get `bash: snakemake: command not found` then you need to go back and ensure that From f25d2504322cd03c99aaf20e45a43bd6d099b4e0 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Wed, 13 Nov 2024 22:00:38 +0100 Subject: [PATCH 15/32] Update dry run output --- pages/snakemake.qmd | 29 +++++++++++++++++------------ 1 file changed, 17 insertions(+), 12 deletions(-) diff --git a/pages/snakemake.qmd b/pages/snakemake.qmd index ac4b7ed2..ba94ccf7 100644 --- a/pages/snakemake.qmd +++ b/pages/snakemake.qmd @@ -113,32 +113,37 @@ for showing the shell commands that it will execute. `snakemake --help` will show you all available flags. ```no-highlight -$ snakemake -n -c 1 -r -p a.upper.txt +$ snakemake -n -p a.upper.txt Building DAG of jobs... Job stats: -job count min threads max threads ---------------------- ------- ------------- ------------- -convert_to_upper_case 1 1 1 -total 1 1 1 +job count +--------------------- ------- +convert_to_upper_case 1 +total 1 -[Mon Oct 25 16:48:43 2021] +[Wed Nov 13 21:59:01 2024] rule convert_to_upper_case: input: a.txt output: a.upper.txt jobid: 0 reason: Missing output files: a.upper.txt - resources: tmpdir=/var/folders/p0/6z00kpv16qbf_bt52y4zz2kc0000gp/T + resources: tmpdir= tr [a-z] [A-Z] < a.txt > a.upper.txt - + Job stats: -job count min threads max threads ---------------------- ------- ------------- ------------- -convert_to_upper_case 1 1 1 -total 1 1 1 +job count +--------------------- ------- +convert_to_upper_case 1 +total 1 + +Reasons: + (check individual jobs above for details) + output files have to be generated: + convert_to_upper_case This was a dry-run (flag -n). The order of jobs does not reflect the order of execution. ``` From 138e4dc7fbbec267e564d951675bdcdd25992fcd Mon Sep 17 00:00:00 2001 From: John Sundh Date: Thu, 14 Nov 2024 07:27:44 +0100 Subject: [PATCH 16/32] Start rewrite of basics section --- pages/snakemake.qmd | 255 +++++++++++++++++++++++++------------------- 1 file changed, 146 insertions(+), 109 deletions(-) diff --git a/pages/snakemake.qmd b/pages/snakemake.qmd index ba94ccf7..96965e85 100644 --- a/pages/snakemake.qmd +++ b/pages/snakemake.qmd @@ -101,20 +101,36 @@ will convert all characters in the set `[a-z]` to the corresponding character in the set `[A-Z]`, and then send the output to `a.upper.txt`". Now let's run our first Snakemake workflow. When a workflow is executed -Snakemake tries to generate a set of target files. Target files can be specified -via the command line (or, as you will see later, in several other ways). Here we -ask Snakemake to make the file `a.upper.txt`. We can specify the file containing -our rules with `-s` but since the default behaviour of Snakemake is to look for -a file called `Snakefile` in either the working directory or in a subdirectory -called `workflow/` we don't need to specify that here. It's good practice to -first run with the flag `-n` (or `--dry-run`), which will show what Snakemake -plans to do without actually running anything. You can also use the flag `-p`, -for showing the shell commands that it will execute. `snakemake --help` will -show you all available flags. +Snakemake tries to generate a set of **target files**. Target files can be +specified via the command line or, as you will see later, in several other ways. +But importantly you will always have to specify what you want Snakemake to +generate, in one way or another. The basic Snakemake command line looks like +this: -```no-highlight -$ snakemake -n -p a.upper.txt +```bash +snakemake [target files ...] +``` + +where `` are flags that modify the behaviour of Snakemake and +`[target files ...]` are the files that you want Snakemake to generate. + +Let's ask Snakemake to generate the file `a.upper.txt`, this is our target file. +For the arguments part we'll use the `-s` flag to specify the file containing +our rules and the `-c` flag to set the number of cores to use. It's good +practice to first run with the flag `-n` (or `--dry-run`), which will show what +Snakemake plans to do without actually running anything. You can also use the +flag `-p`, for showing the shell commands that it will execute. `snakemake +--help` will show you all available flags. + +The command thus becomes: +```bash +snakemake -s Snakefile -c 1 -n -p a.upper.txt +``` + +You should see something like this: + +```no-highlight Building DAG of jobs... Job stats: job count @@ -148,43 +164,81 @@ Reasons: This was a dry-run (flag -n). The order of jobs does not reflect the order of execution. ``` -You can see that Snakemake plans to run one job: the rule `convert_to_upper_case` -with `a.txt` as input and `a.upper.txt` as output. The reason for doing this is -that it's missing the file `a.upper.txt`. Now execute the workflow without the -`-n` flag and check that the contents of `a.upper.txt` is as expected. Then try -running the same command again. What do you see? It turns out that Snakemake -only reruns jobs if there have been changes to either **the input files, or the -workflow itself**. This is how Snakemake ensures that everything in the -workflow is up to date. We will get back to this shortly. +You can see that Snakemake plans to run one job: the rule +`convert_to_upper_case` with `a.txt` as input and `a.upper.txt` as output. The +reason for doing this is that it's missing the file `a.upper.txt`. Now run the +command without the `-n` flag and check that the contents of `a.upper.txt` is as +expected. -What if we ask Snakemake to generate the file `b.upper.txt`? +We used the `-s` flag to point to the file containing the rules. In fact, +Snakemake will look for a file named `Snakefile` in the current directory (or in +a subdirectory called `workflow`) by default, which means we can omit this flag +for this example. If we also omit the `-c` flag Snakemake will use all available +cores, which won't make a difference for this small workflow. The command can +thus be simplified to: -```no-highlight -$ snakemake -n -c 1 -r -p b.upper.txt +```bash +snakemake -p a.upper.txt +``` -Building DAG of jobs... -MissingRuleException: -No rule to produce b.upper.txt (if you use input functions make sure that they don't raise unexpected exceptions). +Try running this simplified command. What do you see? It turns out that +Snakemake only reruns jobs if there have been changes to either **the input +files, or the workflow itself**. Here neither the `a.txt` input, nor the +`Snakefile` have been changed so Snakemake determines that everything in the +workflow is up to date and there is nothing to do. + +What if we ask Snakemake to generate the file `b.upper.txt`? Run the following: + +```bash +snakemake -n -p b.upper.txt +``` + +You will see that Snakemake complains that there is no rule to produce +`b.upper.txt`. This is because we have only defined a rule for `a.upper.txt`. We +could copy the rule to make a similar one for `b.txt`, but that would be a bit +cumbersome. Here is where named wildcards come in; one of the most powerful +features of Snakemake. Simply change the input and output directives from + +```python + output: + "a.upper.txt" + input: + "a.txt" +``` + +to + +```python + output: + "{some_name}.upper.txt" + input: + "{some_name}.txt" ``` -That didn't work well. We could copy the rule to make a similar one for -`b.txt`, but that would be a bit cumbersome. Here is where named wildcards come -in; one of the most powerful features of Snakemake. Simply change the input -from `input: "a.txt"` to `input: "{some_name}.txt"` and the output to `output: -"{some_name}.upper.txt"`. Now try asking for `b.upper.txt` again. +Now try asking for `b.upper.txt` again. Tada! What happens here is that Snakemake looks at all the rules it has available (actually only one in this case) and tries to assign values to all -wildcards so that the targeted files can be generated. In this case it was -quite simple, you can see that it says that `wildcards: some_name=b`, but for -large workflows and multiple wildcards it can get much more complex. Named -wildcards is what enables a workflow (or single rules) to be efficiently -generalized and reused between projects or shared between people. - -It seems we have the first part of our workflow working, now it's time to make -the second rule for concatenating the outputs from `convert_to_upper_case`. The -rule structure will be similar; the only difference is that here we have two -inputs instead of one. This can be expressed in two ways, either with named +wildcards so that the targeted files can be generated. In this case you can see +that it says `wildcards: some_name=b`, which means that the `some_name` wildcard +was assigned the value `b`. This was quite simple but for large workflows and +multiple wildcards it can get much more complex. Named wildcards is what enables +a workflow (or single rules) to be efficiently generalized and reused between +projects or shared between people. + +Remember that we're trying to make a workflow that will convert two files to +uppercase and concatenate them. It seems we have the first part of our workflow +working, now it's time to make the second rule for concatenating the outputs +from `convert_to_upper_case`. + +The syntax for concatenating two files in Bash is : + +```bash +cat first_file.txt second_file.txt > output_file.txt` +``` + +The structure of this second rule will be similar to the first but here we have +two inputs instead of one. This can be expressed in two ways, either with named inputs like this: ```python @@ -216,29 +270,19 @@ Snakemake workflows. The parser will complain, but sometimes the error message can be difficult to interpret. ::: -Now try to construct this rule yourself and name it `concatenate_a_and_b`. -The syntax for concatenating two files in Bash is -`cat first_file.txt second_file.txt > output_file.txt`. Call the output `c.txt`. -Run the workflow in Snakemake and validate that the output looks as expected. +Now try to construct this second rule yourself. Add it to the `Snakefile` and +name the rule `concatenate_a_and_b`. Use `a.upper.txt` and `b.upper.txt` as +input and call the output `c.txt`. Check the solution below if you get stuck. -Wouldn't it be nice if our workflow could be used for _any_ files, not just -`a.txt` and `b.txt`? We can achieve this by using named wildcards (or in other -ways as we will discuss later). As we've mentioned, Snakemake looks at all the -rules it has available and tries to assign values to all wildcards so that the -targeted files can be generated. We therefore have to name the output file in -a way so that it also contains information about which input files it should be -based on. Try to figure out how to do this yourself. If you're stuck you can -look at the spoiler below, but spend some time on it before you look. Also -rename the rule to `concatenate_files` to reflect its new more general use. ::: {.callout-tip collapse="true" title="Click to show"} ```python -rule concatenate_files: +rule concatenate_a_and_b: output: - "{first}_{second}.txt" + "c.txt" input: - "{first}.upper.txt", - "{second}.upper.txt" + "a.upper.txt", + "b.upper.txt" shell: """ cat {input[0]} {input[1]} > {output} @@ -246,66 +290,59 @@ rule concatenate_files: ``` ::: -We can now control which input files to use by the name of the file we ask -Snakemake to generate. Run the workflow without the flag `-n` (or `--dry-run`) -to execute both rules, providing one core with `-c 1` (or `--cores 1`): - -```no-highlight -$ snakemake a_b.txt -c 1 - -Building DAG of jobs... -Using shell: /bin/bash -Provided cores: 1 (use --cores to define parallelism) -Rules claiming more threads will be scaled down. -Job stats: -job count min threads max threads ---------------------- ------- ------------- ------------- -concatenate_files 1 1 1 -convert_to_upper_case 2 1 1 -total 3 1 1 - -Select jobs to execute... +Now try to generate the file `c.txt` with your workflow by running: -[Mon Oct 25 16:51:52 2021] -rule convert_to_upper_case: - input: b.txt - output: b.upper.txt - jobid: 2 - wildcards: some_name=b - resources: tmpdir=/var/folders/p0/6z00kpv16qbf_bt52y4zz2kc0000gp/T +```bash +snakemake -p c.txt +``` -[Mon Oct 25 16:51:53 2021] -Finished job 2. -1 of 3 steps (33%) done -Select jobs to execute... +Validate that the output looks as expected. -[Mon Oct 25 16:51:53 2021] -rule convert_to_upper_case: - input: a.txt - output: a.upper.txt - jobid: 1 - wildcards: some_name=a - resources: tmpdir=/var/folders/p0/6z00kpv16qbf_bt52y4zz2kc0000gp/T +Good job! You've now created a simple Snakemake workflow that converts two files +to uppercase and concatenates them. -[Mon Oct 25 16:51:53 2021] -Finished job 1. -2 of 3 steps (67%) done -Select jobs to execute... +Wouldn't it be nice if our workflow could be used for _any_ files, not just +`a.txt` and `b.txt`? We can achieve this by generalizing the +`concatenate_a_and_b` rule using named wildcards. + +As we've mentioned, Snakemake looks at all the rules it has available and tries +to assign values to all wildcards so that the targeted files can be generated. +We therefore have to name the output file in a way so that it contains +information about which input files it should be based on. Try to figure out how +to modify the `concatenate_a_and_b` rule yourself. If you're stuck you can look +at the solution below, but spend some time on it before you look. Also rename +the `concatenate_a_and_b` rule to `concatenate_files` to reflect its new more +general use. -[Mon Oct 25 16:51:53 2021] +::: {.callout-tip collapse="true" title="Click to show"} +```python rule concatenate_files: - input: a.upper.txt, b.upper.txt - output: a_b.txt - jobid: 0 - wildcards: first=a, second=b - resources: tmpdir=/var/folders/p0/6z00kpv16qbf_bt52y4zz2kc0000gp/T + output: + "{first}_{second}.txt" + input: + "{first}.upper.txt", + "{second}.upper.txt" + shell: + """ + cat {input[0]} {input[1]} > {output} + """ +``` +::: -[Mon Oct 25 16:51:53 2021] -Finished job 0. -3 of 3 steps (100%) done +We can now control which input files to use by the name of the file we ask +Snakemake to generate. Run the workflow to create a file `a_b.txt`: + +```bash +snakemake -p a_b.txt ``` -Neat! +You will see that Snakemake assigned the values `a` and `b` to the wildcards +`{first}` and `{second}`, respectively. This is because the output of the +`concatenate_files` rule is defined as `"{first}_{second}.txt"`, which means +that when we ask for `a_b.txt`, the `{first}` wildcard is assigned `a` and the +`{second}` wildcard is assigned `b`. Try to generate the file `b_a.txt` and see +what happens. + ::: {.callout-tip} You can name a file whatever you want in a Snakemake workflow, but you will From 6bcbf1e801ef3bd05dcf0b60fe4b3b8f83c4d3f4 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Fri, 15 Nov 2024 10:18:03 +0100 Subject: [PATCH 17/32] Change snakemake command line --- pages/snakemake.qmd | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pages/snakemake.qmd b/pages/snakemake.qmd index 96965e85..4551d417 100644 --- a/pages/snakemake.qmd +++ b/pages/snakemake.qmd @@ -406,7 +406,7 @@ Let's run the workflow with the updated rule. Remove the file `a_b.txt` or add `-f` to the Snakemake command to force a re-run: ```bash -snakemake a_b.txt -c 1 -f +snakemake -f a_b.txt ``` If you added the `print` statement to the function you should see the From fbe91e5a1b172c3f34961a8bf51a616d2f1976e0 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Fri, 15 Nov 2024 11:12:31 +0100 Subject: [PATCH 18/32] Update visualization section --- pages/snakemake.qmd | 129 +++++++++++++++++++++++++++----------------- 1 file changed, 79 insertions(+), 50 deletions(-) diff --git a/pages/snakemake.qmd b/pages/snakemake.qmd index 4551d417..20cfe0b1 100644 --- a/pages/snakemake.qmd +++ b/pages/snakemake.qmd @@ -521,81 +521,84 @@ The main difference here is that now each node is a job instead of a rule. You can see that the wildcards used in each job are also displayed. Another difference is the dotted lines around the nodes. A dotted line is Snakemake's way of indicating that this rule doesn't need to be rerun in order to generate -`a_b.txt`. Validate this by running `snakemake -n -r a_b.txt` and it should say +`a_b.txt`. Validate this by running `snakemake -n a_b.txt` and it should say that there is nothing to be done. - We've discussed before that one of the main purposes of using a WfMS is that it automatically makes sure that everything is up to date. This is done by recursively checking that outputs are always newer than inputs for all the rules involved in the generation of your target files. Now try to change the contents of `a.txt` to some other text and save it. What do you think will -happen if you run `snakemake -n -r a_b.txt` again? +happen if you run `snakemake -n a_b.txt` again? ::: {.callout-tip collapse="true" title="Click to show"} ```no-highlight -$ snakemake -n -r a_b.txt - Building DAG of jobs... Job stats: -job count min threads max threads ---------------------- ------- ------------- ------------- -concatenate_files 1 1 1 -convert_to_upper_case 1 1 1 -total 2 1 1 +job count +--------------------- ------- +concatenate_files 1 +convert_to_upper_case 1 +total 2 -[Mon Oct 25 17:00:02 2021] +[Fri Nov 15 10:27:22 2024] rule convert_to_upper_case: input: a.txt output: a.upper.txt jobid: 1 reason: Updated input files: a.txt wildcards: some_name=a - resources: tmpdir=/var/folders/p0/6z00kpv16qbf_bt52y4zz2kc0000gp/T + resources: tmpdir= -[Mon Oct 25 17:00:02 2021] +[Fri Nov 15 10:27:22 2024] rule concatenate_files: input: a.upper.txt, b.upper.txt output: a_b.txt jobid: 0 reason: Input files updated by another job: a.upper.txt wildcards: first=a, second=b - resources: tmpdir=/var/folders/p0/6z00kpv16qbf_bt52y4zz2kc0000gp/T + resources: tmpdir= Job stats: -job count min threads max threads ---------------------- ------- ------------- ------------- -concatenate_files 1 1 1 -convert_to_upper_case 1 1 1 -total 2 1 1 +job count +--------------------- ------- +concatenate_files 1 +convert_to_upper_case 1 +total 2 + +Reasons: + (check individual jobs above for details) + input files updated by another job: + concatenate_files + updated input files: + convert_to_upper_case This was a dry-run (flag -n). The order of jobs does not reflect the order of execution. ``` ::: Were you correct? Also generate the job graph and compare to the one generated -above. What's the difference? Now rerun without `-n` and validate that -`a_b.txt` contains the new text (don't forget to specify `-c 1`). Note that -Snakemake doesn't look at the contents of files when trying to determine what has -changed, only at the timestamp for when they were last modified. +above. What's the difference? Now rerun without `-n` and validate that `a_b.txt` +contains the new text. Note that Snakemake doesn't look at the contents of files +when trying to determine what has changed, only at the timestamp for when they +were last modified. We've seen that Snakemake keeps track of if files in the workflow have changed, and automatically makes sure that any results depending on such files are -regenerated. What about if the rules themselves are changed? It turns out that -since version 7.8.0 Snakemake keeps track of this automatically. +regenerated. What about if the rules themselves are changed? Since version 7.8.0 +Snakemake keeps track of this automatically. -Let's say that we want to modify the rule `concatenate_files` to also include -which files were concatenated. +Modify the rule `concatenate_files` to also include which files were +concatenated. ```python rule concatenate_files: output: "{first}_{second}.txt" input: - "{first}.upper.txt", - "{second}.upper.txt" + concat_input shell: """ echo 'Concatenating {input}' | cat - {input[0]} {input[1]} > {output} @@ -611,7 +614,14 @@ it should read from standard input). Lastly, the output from `cat` is sent to `{output}`. ::: -If you now run the workflow as before you should see: +If you now run: + +```bash +snakemake -n a_b.txt +``` + + you should see: + ```bash rule concatenate_files: input: a.upper.txt, b.upper.txt @@ -621,21 +631,28 @@ rule concatenate_files: wildcards: first=a, second=b ``` -Because although no files involved in the workflow have been changed, Snakemake -recognizes that the workflow code itself has been modified and this triggers -a re-run. +Although no files involved in the workflow have been changed, Snakemake +recognizes that the workflow code itself has been modified and this triggers a +re-run. -Snakemake is aware of changes to four categories of such "rerun-triggers": -"input" (changes to rule input files), "params" (changes to the rule `params` section), -"software-env" (changes to Conda environment files specified by the `conda:` -directive) and "code" (changes to code in the `shell:`, `run:`, `script:` and -`notebook:` directives). +Snakemake is aware of changes to five categories of such "rerun-triggers": + +- "input" (changes to rule input files), +- "params" (changes to the rule `params` section) +- "software-env" (changes to Conda environment files specified by the `conda:` directive) +- "code" (changes to code in the `shell:`, `run:`, `script:` and `notebook:` directives) +- "mtime" (changes to the modification time of input files) + +Using the flag `--rerun-triggers` you can specify which of these categories +should trigger a re-run. For example, `--rerun-triggers input` will only +re-run the workflow if the specified input files have changed and `--rerun-triggers +input params` will re-run if either the input files or the `params` section of +the rule has changed. Prior to version 7.8.0, only changes to the modification time of input files would trigger automatic re-runs. To run Snakemake with this previous behaviour you can use the setting `--rerun-triggers mtime` at the command line. -Change the `shell:` section of the `concatenate_files` rule back to the previous -version, then try running: `snakemake -n -r a_b.txt --rerun-triggers mtime` and +Try running: `snakemake --rerun-triggers mtime -n a_b.txt` and you should again see `Nothing to be done (all requested files are present and up to date).` You can also export information on how all files were generated (when, by which @@ -643,18 +660,18 @@ rule, which version of the rule, and by which commands) to a tab-delimited file like this: ```bash -snakemake a_b.txt -c 1 -D > summary.tsv +snakemake a_b.txt -D > summary.tsv ``` The content of `summary.tsv` is shown in the table below: -
-| output_file | date | rule | version | log-file(s) | input-file(s) | shellcmd | status | plan | -|----------------|-----------------------------|--------------------------|-------------|-----------------|----------------------------|------------------------------------------|--------------------------------|-------------------| -| a_b.txt | Mon Oct 25 17:01:46 2021 | concatenate_files | - | | a.upper.txt,b.upper.txt | cat a.upper.txt b.upper.txt > a_b.txt | rule implementation changed | update pending | -| a.upper.txt | Mon Oct 25 17:01:46 2021 | convert_to_upper_case | - | | a.txt | tr [a-z] [A-Z] < a.txt > a.upper.txt | ok | no update | -| b.upper.txt | Mon Oct 25 17:01:46 2021 | convert_to_upper_case | - | | b.txt | tr [a-z] [A-Z] < b.txt > b.upper.txt | ok | no update | -
+| output_file | date | rule | log-file(s) | input-file(s) | shellcmd | status | plan | +|-------------|-----------------------|-------------------------|-------------|------------------------|-------------------------------------------|----------------------------|-----------------| +| a_b.txt | Fri Nov 15 10:32:04 2024 | concatenate_files | | a.upper.txt,b.upper.txt | cat a.upper.txt b.upper.txt > a_b.txt | rule implementation changed | update pending | +| a.upper.txt | Fri Nov 15 10:32:04 2024 | convert_to_upper_case | | a.txt | tr [a-z] [A-Z] < a.txt > a.upper.txt | ok | no update | +| b.upper.txt | Thu Nov 14 00:04:06 2024 | convert_to_upper_case | | b.txt | tr [a-z] [A-Z] < b.txt > b.upper.txt | ok | no update | + +: summary.tsv {.responsive .sm} You can see in the second last column that the rule implementation for `a_b.txt` has changed. The last column shows if Snakemake plans to regenerate the files @@ -668,6 +685,19 @@ since it's easy to delete if you don't need it anymore and everything is contained in the project directory. Just be sure to add it to `.gitignore` so that you don't end up tracking it with git. +Let's take a look at the final type of graph that Snakemake can generate, this +is done with the `--filegraph` flag and will show the rules and their input and +output files. + +```bash +snakemake --filegraph a_b.txt | dot -Tpng > filegraph.png +``` + +In terms of details/complexity the filegraph is an intermediate between the +rulegraph (`--rulegraph`) and the jobgraph (`--dag`). It shows the rules to run +with their respective input and output files (shown with wildcards), but doesn't +show the jobs that will be executed. + By now you should be familiar with the basic functionality of Snakemake, and you can build advanced workflows with only the features we have discussed here. There's a lot we haven't covered though, in particular when it comes to making @@ -681,8 +711,7 @@ the other tutorials instead. ::: {.callout-note title="Quick recap"} In this section we've learned: -- How to use `--dag` and `--rulegraph` for visualizing the job and rule - graphs, respectively. +- How to use `--rulegraph`, `--dag` and `--filegraph` for visualizing the workflow. - How Snakemake reruns relevant parts of the workflow after there have been changes. - How Snakemake tracks changes to files and code in a workflow From 47e4ac96ea1af35dbb223770749a91fef09d76f8 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Fri, 15 Nov 2024 22:04:24 +0100 Subject: [PATCH 19/32] Fix invalid escape sequence --- pages/snakemake.qmd | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/pages/snakemake.qmd b/pages/snakemake.qmd index 20cfe0b1..3a82e8b0 100644 --- a/pages/snakemake.qmd +++ b/pages/snakemake.qmd @@ -743,10 +743,7 @@ a per-rule basis, by specifying something like `conda: rule-specific-env.yml` in the rule definition and running Snakemake with the `--use-conda` flag. The given rule will then be run in the Conda environment specified in `rule-specific-env.yml` that will be created and -activated on the fly by Snakemake. Note that by default Snakemake uses -`conda` to generate the rule-specific environments. This behaviour can be -changed by running with `--conda-frontend conda`, which will force -Snakemake to use `conda` instead. +activated on the fly by Snakemake. ::: Let's start by generating the rule graph so that we get an overview of the From 777abfeefbec79ca38ede177f1fa7783c49dc427 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Fri, 15 Nov 2024 22:05:06 +0100 Subject: [PATCH 20/32] Revert "Fix invalid escape sequence" This reverts commit 47e4ac96ea1af35dbb223770749a91fef09d76f8. --- pages/snakemake.qmd | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/pages/snakemake.qmd b/pages/snakemake.qmd index 3a82e8b0..20cfe0b1 100644 --- a/pages/snakemake.qmd +++ b/pages/snakemake.qmd @@ -743,7 +743,10 @@ a per-rule basis, by specifying something like `conda: rule-specific-env.yml` in the rule definition and running Snakemake with the `--use-conda` flag. The given rule will then be run in the Conda environment specified in `rule-specific-env.yml` that will be created and -activated on the fly by Snakemake. +activated on the fly by Snakemake. Note that by default Snakemake uses +`conda` to generate the rule-specific environments. This behaviour can be +changed by running with `--conda-frontend conda`, which will force +Snakemake to use `conda` instead. ::: Let's start by generating the rule graph so that we get an overview of the From 1edfc86e34a9faf751a975179eeac984b9ba83ec Mon Sep 17 00:00:00 2001 From: John Sundh Date: Fri, 15 Nov 2024 22:05:37 +0100 Subject: [PATCH 21/32] Fix invalid escape sequence --- tutorials/snakemake/snakefile_mrsa.smk | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tutorials/snakemake/snakefile_mrsa.smk b/tutorials/snakemake/snakefile_mrsa.smk index 00db0b8b..c07af13a 100644 --- a/tutorials/snakemake/snakefile_mrsa.smk +++ b/tutorials/snakemake/snakefile_mrsa.smk @@ -122,7 +122,7 @@ rule align_to_genome: Align a fastq file to a genome index using Bowtie 2. """ output: - "results/bam/{sample_id,\w+}.bam" + "results/bam/{sample_id,\\w+}.bam" input: "data/{sample_id}.fastq.gz", "results/bowtie2/NCTC8325.1.bt2", From 38b908e176721da86c2d652a3c19a5f775e67437 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Sat, 16 Nov 2024 22:31:51 +0100 Subject: [PATCH 22/32] Update up to Generalising workflows section --- pages/snakemake.qmd | 294 +++++++++++++++++++++++++++----------------- 1 file changed, 178 insertions(+), 116 deletions(-) diff --git a/pages/snakemake.qmd b/pages/snakemake.qmd index 20cfe0b1..f15310d4 100644 --- a/pages/snakemake.qmd +++ b/pages/snakemake.qmd @@ -730,7 +730,7 @@ will be spent improving and making this workflow more flexible! ::: {.callout-tip} This section will leave a little more up to you compared to the previous one. If you get stuck at some point the final workflow after all the -modifications is available in `tutorials/git/Snakefile`. +modifications is available in `tutorials/containers/Snakefile`. ::: You are probably already in your `snakemake-env` environment, otherwise @@ -743,10 +743,7 @@ a per-rule basis, by specifying something like `conda: rule-specific-env.yml` in the rule definition and running Snakemake with the `--use-conda` flag. The given rule will then be run in the Conda environment specified in `rule-specific-env.yml` that will be created and -activated on the fly by Snakemake. Note that by default Snakemake uses -`conda` to generate the rule-specific environments. This behaviour can be -changed by running with `--conda-frontend conda`, which will force -Snakemake to use `conda` instead. +activated on the fly by Snakemake. ::: Let's start by generating the rule graph so that we get an overview of the @@ -759,11 +756,11 @@ snakemake -s snakefile_mrsa.smk --rulegraph | dot -T png > rulegraph_mrsa.png There's another difference in this command compared to the one we've used before, namely that we don't define a target. In the toy example we used -`a_b.txt` as a target, and the wildcards were resolved based on that. -How come that we don't need to do that here? It turns out that by default -Snakemake targets the first rule in a workflow. By convention, we call this rule -`all` and let it serve as a rule for aggregating the main outputs of the -workflow. +`a_b.txt` as a target, and the wildcards were resolved based on that. How come +that we don't need to do that here? This is because by default Snakemake uses +the first rule in a workflow as its target, unless a target is specified on the +command line. By convention, this rule is called `all` and serves as a rule +for aggregating the main outputs of the workflow. ![](images/rulegraph_mrsa.svg) @@ -787,62 +784,21 @@ downstream analysis will be based on. ![](images/dag_mrsa.svg) -Now try to run the whole workflow. Hopefully you see something like this. +Now try to run the whole workflow. As before you can specify `-p` to have +Snakemake print the shell commands for rules and you can control how many cores +to use with `-c`. If you don't specify number of cores Snakemake will use all +available cores on your machine. Let's run the workflow with one core: -```no-highlight -Building DAG of jobs... -Using shell: /bin/bash -Provided cores: 1 (use --cores to define parallelism) -Rules claiming more threads will be scaled down. -Job stats: -job count min threads max threads --------------------- ------- ------------- ------------- -align_to_genome 3 1 1 -all 1 1 1 -fastqc 3 1 1 -generate_count_table 1 1 1 -generate_rulegraph 1 1 1 -get_SRA_by_accession 3 1 1 -get_genome_fasta 1 1 1 -get_genome_gff3 1 1 1 -index_genome 1 1 1 -multiqc 1 1 1 -sort_bam 3 1 1 -total 19 1 1 - -Select jobs to execute... - -[Mon Oct 25 17:13:47 2021] -rule get_genome_fasta: - output: data/ref/NCTC8325.fa.gz - jobid: 6 - resources: tmpdir=/var/folders/p0/6z00kpv16qbf_bt52y4zz2kc0000gp/T - ---2021-10-25 17:13:48-- ftp://ftp.ensemblgenomes.org/pub/bacteria/release-37/fasta/bacteria_18_collection/staphylococcus_aureus_subsp_aureus_nctc_8325/dna//Staphylococcus_aureus_subsp_aureus_nctc_8325.ASM1342v1.dna_rm.toplevel.fa.gz - => ‘data/ref/NCTC8325.fa.gz’ -Resolving ftp.ensemblgenomes.org (ftp.ensemblgenomes.org)... 193.62.197.75 -Connecting to ftp.ensemblgenomes.org (ftp.ensemblgenomes.org)|193.62.197.75|:21... connected. -Logging in as anonymous ... Logged in! -==> SYST ... done. ==> PWD ... done. -. -. -[lots of stuff] -. -. -localrule all: - input: results/tables/counts.tsv, results/multiqc/multiqc.html, results/rulegraph.png - jobid: 0 - resources: tmpdir=/var/folders/p0/6z00kpv16qbf_bt52y4zz2kc0000gp/T - -[Mon Oct 25 17:14:38 2021] -Finished job 0. -19 of 19 steps (100%) done +```bash +snakemake -s snakefile_mrsa.smk -c 1 -p ``` -After everything is done, the workflow will have resulted in a bunch of files in -the directories `data/` and `results/`. Take some time to look through the -structure, in particular the quality control reports in `results/multiqc/` and -the count table in `results/tables/`. +After everything is done, the workflow will have generated fastq files for each +sample under `data/` and reference genome files under `data/ref`. The `results/` +directory will contain the output from the different steps in the workflow, +divided into subdirectories. Take some time to look through the structure, in +particular the quality control reports in `results/multiqc/` and the count table +in `results/tables/`. ::: {.callout-note title="Quick recap"} In this section we've learned: @@ -874,10 +830,13 @@ rule some_rule: """ ``` -Most of the programs are run with default settings in the MRSA workflow and +In the toy example above, the params directive has one keyword called `cutoff` +which is set to `2.5` and this value can be accessed in the shell command with +`{params.cutoff}`. + +In the MRSA workflow most of the programs are run with default settings and don't use the `params:` directive. However, the `get_SRA_by_accession` rule -is an exception. Here the remote address for each of the files to download -is passed to the shell directive via: +is an exception. Let's take a look at this part of the workflow: ```python def get_sample_url(wildcards): @@ -903,6 +862,11 @@ rule get_SRA_by_accession: ``` +Here `params` has a keyword called `url` which is set to a function called +`get_sample_url` defined just above the rule. The function returns a URL for the +sample that the rule is supposed to download and this URL is then accessed in +the `shell:` directive with `{params.url}`. + You may recognize this from [the basics](#the-basics) of this tutorial where we used input functions to generate strings and lists of strings for the `input:` section of a rule. Using a function to return values based on the wildcards also @@ -945,9 +909,15 @@ rule get_SRA_by_accession: ``` ::: -Now run through the workflow. Because there's been changes to the `get_SRA_by_accession` -rule this will trigger a re-run of the rule for all three accessions. In addition -all downstream rules that depend on output from `get_SRA_by_accession` are re-run. +Now run the workflow again with: + +```bash +snakemake -s snakefile_mrsa.smk -p -c 1 +``` + +Because there's been changes to the `get_SRA_by_accession` rule this will +trigger a re-run of the rule for all three accessions. In addition all +downstream rules that depend on output from `get_SRA_by_accession` are re-run. As you can see the parameter values we set in the `params` section don't have to be static, they can be any Python expression. In particular, Snakemake @@ -995,9 +965,16 @@ max_reads: 25000 ``` If we now run Snakemake with `--configfile config.yml`, it will parse this file -to form the `config` dictionary. If you want to overwrite a parameter value, -_e.g._ for testing, you can still use the `--config KEY=VALUE` flag, as in -`--config max_reads=1000`. +to form the `config` dictionary. Try it out by running: + +```bash +snakemake -s snakefile_mrsa.smk --configfile config.yml -p -c 1 +``` + +Configuration settings that you specify with `--config KEY=VALUE` on the command +line will overwrite settings specified in a config file. This can be useful if +you want to run the workflow with a different setting for a specific parameter +for testing purposes. ::: {.callout-tip} Rather than supplying the config file from the command line you could also @@ -1055,7 +1032,7 @@ all the rules, only the ones with interesting output. probably the most important log to save. Now add a log file to some or all of the rules above. A good place to save them -to would be `results/logs/rule_name/`. In order to avoid that multiple jobs +to would be `results/logs//`. In order to avoid that multiple jobs write to the same files Snakemake requires that all output and log files contain the same wildcards, so be sure to include any wildcards used in the rule in the log name as well, _e.g._ `{some_wildcard}.log`. @@ -1105,9 +1082,15 @@ featureCounts -t gene -g gene_id -a gff_file -o output_file input_files 2>{log} bowtie2-build input_file index_dir > {log} ``` -Now rerun the whole workflow. Do the logs contain what they should? Note how -much easier it is to follow the progression of the workflow when the rules write -to logs instead of to the terminal. +Now rerun the whole workflow with: + +```bash +snakemake -s snakefile_mrsa.smk -p -c 1 --configfile config.yml +``` + +Do the logs contain what they should? Note how much easier it is to follow the +progression of the workflow when the rules write to logs instead of to the +terminal. ::: {.callout-tip} If you have a rule with a shell directive in which several commands are run @@ -1136,7 +1119,7 @@ from `align_to_genome` is a BAM file, which contains information about all the reads for a sample and where they map in the genome. For downstream processing we need this file to be sorted by genome coordinates. This is what the rule `sort_bam` is for. We therefore end up with both `results/bam/{sample_id}.bam` -and `results/bam/{sample_id}.sorted.bam`. +and `results/bam/{sample_id}.sorted.bam` as you can see if you list the contents of `results/bam`. In Snakemake we can mark an output file as temporary like this: @@ -1145,32 +1128,34 @@ output: temp("...") ``` The file will then be deleted as soon as all jobs where it's an input have -finished. Now do this for the output of `align_to_genome`. We have to rerun the -rule for it to trigger, so use `-R align_to_genome`. It should look something -like this: +finished. Let's use the `temp` notation for for the output of `align_to_genome`, modify the rule so that the `output:` directive looks like this: -```no-highlight -. -. -rule sort_bam: - input: results/bam/SRR935090.bam - output: results/bam/SRR935090.sorted.bam - jobid: 2 - wildcards: sample_id=SRR935090 +```python + output: + temp("results/bam/{sample_id,\\w+}.bam") +``` + +We have to force a rerun of the rule for it to trigger, so we use `--forcerun align_to_genome` to do this. Run the workflow with: -Removing temporary output file results/bam/SRR935090.bam. -Finished job 2. -. -. +```bash +snakemake -s snakefile_mrsa.smk -p -c 1 --configfile config.yml --forcerun align_to_genome +``` + +Take a look at the output printed to your terminal. After the rule `sort_bam` you should see something like this: + +```no-highlight +Removing temporary output results/bam/SRR935092.bam. ``` +If you list the contents of `results/bam` you should see that it now only +contains the sorted BAM files. + ::: {.callout-tip} -Sometimes you may want to trigger removal of temporary files without -actually rerunning the jobs. You can then use the `--delete-temp-output` -flag. In some cases you may instead want to run only parts of a workflow -and therefore want to prevent files marked as temporary from being deleted -(because the files are needed for other parts of the workflow). In such -cases you can use the `--notemp` flag. +If you want to prevent Snakemake from deleting files marked as temporary you can +run with the `--notemp` command line flag which will cause Snakemake to ignore +any `temp()` declarations. If you at any point want to delete remaining files +that are declared as temporary in your workflow you can instead run with the +`--delete-temp-output` which will delete all remaining `temp` files. ::: Snakemake has a number of options for marking files: @@ -1193,11 +1178,11 @@ In this section we've learned: - How to mark an output file as temporary for automatic removal. ::: -## Targets +## The expand function We've mentioned that Snakemake rules take either strings or a list of strings as -input, and that we can use any Python expression in Snakemake workflows. -Here we'll show how these features help us condense the code of rules. +input, and that we can use any Python expression in Snakemake workflows. Here +we'll show how these features help us condense the code of rules. Consider the rule `align_to_genome` below. @@ -1239,7 +1224,7 @@ input: This will take the elements of the list of substrings one by one, and insert that element in the place of `{substr}`. Since this type of aggregating rules are quite common, Snakemake also has a more compact way of achieving the -same thing. +same thing with the built-in function `expand()`. ```python input: @@ -1248,13 +1233,61 @@ input: substr = ["1", "2", "3", "4", "rev.1", "rev.2"]) ``` +Here `expand` will take the string `"results/bowtie2/NCTC8325.{substr}.bt2"` and +insert each of the values in the list `["1", "2", "3", "4", "rev.1", "rev.2"]` +in place of `{substr}`. The result is a list of strings that can be used as +input to the rule. This is a very powerful feature that can save you a lot of +typing and make your code more readable. + ::: {.callout-caution} -When using expand() like this, `substr` is not a wildcard because it is +When using expand() like this, `substr` is **not** a wildcard because it is resolved to the values explicitly given inside the expand expression. ::: Now change in the rules `index_genome` and `align_to_genome` to use the -`expand()` expression. +`expand()` expression. Click the solution below if you need help. + +::: {.callout-tip collapse="true" title="Click to show"} +```python +rule index_genome: + """ + Index a genome using Bowtie 2. + """ + output: + expand("results/bowtie2/NCTC8325.{substr}.bt2", + substr=["1", "2", "3", "4","rev.1", "rev.2"]) + input: + "data/ref/NCTC8325.fa.gz" + log: + "results/logs/index_genome/NCTC8325.log" + shell: + """ + # Bowtie2 cannot use .gz, so unzip to a temporary file first + gunzip -c {input} > tempfile + bowtie2-build tempfile results/bowtie2/NCTC8325 > {log} + + # Remove the temporary file + rm tempfile + """ + +rule align_to_genome: + """ + Align a fastq file to a genome index using Bowtie 2. + """ + output: + temp("results/bam/{sample_id,\\w+}.bam") + input: + "data/{sample_id}.fastq.gz", + expand("results/bowtie2/NCTC8325.{substr}.bt2", + substr=["1", "2", "3", "4", "rev.1", "rev.2"]) + log: + "results/logs/align_to_genome/{sample_id}.log" + shell: + """ + bowtie2 -x results/bowtie2/NCTC8325 -U {input[0]} > {output} 2> {log} + """ +``` +::: In the workflow we decide which samples to run by including the SRR ids in the names of the inputs to the rules `multiqc` and `generate_count_table`: @@ -1303,7 +1336,26 @@ input: expand("results/fastqc/{sample_id}_fastqc.zip", sample_id = SAMPLES) ``` -See if you can update the `generate_count_table` rule in the same manner! +See if you can update the `generate_count_table` rule in the same manner. If you +need help, click the solution below. + +::: {.callout-tip collapse="true" title="Click to show"} +```python +ule generate_count_table: + """ + Generate a count table using featureCounts. + """ + output: + "results/tables/counts.tsv" + input: + bams = expand("results/bam/{sample_id}.sorted.bam", sample_id = SAMPLES), + annotation = "data/ref/NCTC8325.gff3.gz" + shell: + """ + featureCounts -t gene -g gene_id -a {input.annotation} -o {output} {input.bams} + """ +``` +::: ::: {.callout-note title="Quick recap"} In this section we've learned: @@ -1322,7 +1374,7 @@ rule index_genome: Index a genome using Bowtie 2. """ output: - index = expand("results/bowtie2/NCTC8325.{substr}.bt2", + expand("results/bowtie2/NCTC8325.{substr}.bt2", substr = ["1", "2", "3", "4", "rev.1", "rev.2"]) input: "data/NCTC8325.fa.gz" @@ -1340,23 +1392,24 @@ rule index_genome: ``` There is a temporary file here called `tempfile` which is the uncompressed -version of the input, since Bowtie2 cannot use compressed files. There are -a number of drawbacks with having files that aren't explicitly part of the -workflow as input/output files to rules: +version of the input fasta file. This file is created when the rule starts ( +since Bowtie2 cannot use compressed files) but is not needed upon completion. +There are a number of drawbacks with having files that aren't explicitly part of +the workflow as input/output files to rules: - Snakemake cannot clean up these files if the job fails, as it would do for normal output files. - If several jobs are run in parallel there is a risk that they write to `tempfile` at the same time. This can lead to very scary results. - Sometimes we don't know the names of all the files that a program can - generate. It is, for example, not unusual that programs leave some kind of - error log behind if something goes wrong. + generate and we may only be interested in a few of them. All of these issues can be dealt with by using the `shadow` option for a rule. The shadow option results in that each execution of the rule is run in an isolated temporary directory (located in `.snakemake/shadow/` by default). There are a few options for `shadow` (for the full list of these options see the [Snakemake docs](https://snakemake.readthedocs.io/en/stable/snakefiles/rules.html#shadow-rules)). + The most simple is `shadow: "minimal"`, which means that the rule is executed in an empty directory that the input files to the rule have been symlinked into. For the rule below, that means that the only file available would be `input.txt`. @@ -1379,11 +1432,20 @@ rule some_rule: """ ``` -Try this out for the rules where we have to "manually" deal with files that -aren't tracked by Snakemake (`multiqc`, `index_genome`). Also remove the shell -commands that remove temporary files from those rules, as they are no longer -needed. Now rerun the workflow and validate that the temporary files don't show -up in your working directory. +Try this out for the rules `multiqc` and `index_genome` where we have to +"manually" deal with files that aren't tracked by Snakemake. Add `shadow: +"minimal"` to these rules and also delete the shell commands that remove +temporary files from those rules (the last lines in the shell directives that +begin with `rm`), as they are no longer needed. By editing the shell directive +you will trigger a rerun of these two rules, as well as all rules that depend on +them. So run the workflow with: + +```bash +snakemake -s snakefile_mrsa.smk -p -c 1 --configfile config.yml +``` + +and make sure that you see neither `tempfile`, nor the `multiqc_data` directory +in your current working directory after the workflow has finished. ::: {.callout-tip} Some people use the shadow option for almost every rule and some never From ec7cf937643a49df5c5d821699cba0529bab9bba Mon Sep 17 00:00:00 2001 From: John Sundh Date: Sat, 16 Nov 2024 23:03:17 +0100 Subject: [PATCH 23/32] Add filename to code blocks --- pages/snakemake.qmd | 30 +++++++++++++++--------------- 1 file changed, 15 insertions(+), 15 deletions(-) diff --git a/pages/snakemake.qmd b/pages/snakemake.qmd index f15310d4..6b02b0fe 100644 --- a/pages/snakemake.qmd +++ b/pages/snakemake.qmd @@ -838,7 +838,7 @@ In the MRSA workflow most of the programs are run with default settings and don't use the `params:` directive. However, the `get_SRA_by_accession` rule is an exception. Let's take a look at this part of the workflow: -```python +```{.python filename="snakefile_mrsa.smk"} def get_sample_url(wildcards): samples = { "SRR935090": "https://figshare.scilifelab.se/ndownloader/files/39539767", @@ -892,7 +892,7 @@ need help, click to show the solution below. ::: {.callout-tip collapse="true" title="Click to show"} -```python +```{.python filename="snakefile_mrsa.smk"} rule get_SRA_by_accession: """ Retrieve a single-read FASTQ file @@ -925,7 +925,7 @@ provides a global dictionary of configuration parameters called `config`. Let's modify `get_SRA_by_accession` to look something like this in order to make use of this dictionary: -```python +```{.python filename="snakefile_mrsa.smk"} rule get_SRA_by_accession: """ Retrieve a single-read FASTQ file @@ -960,8 +960,8 @@ projects. Like several other files used in these tutorials, this file should be in [YAML format](https://en.wikipedia.org/wiki/YAML). Create the file below and save it as `config.yml`. -```yaml -max_reads: 25000 +```{.yaml filename="config.yml"} +max_reads: 20000 ``` If we now run Snakemake with `--configfile config.yml`, it will parse this file @@ -1045,7 +1045,7 @@ a flag. For example, in the `align_to_genome` rule, it could look like this (Bowtie2 writes log info to standard error): -```python +```{.python filename="snakefile_mrsa.smk"} rule align_to_genome: """ Align a fastq file to a genome index using Bowtie 2. @@ -1130,7 +1130,7 @@ output: temp("...") The file will then be deleted as soon as all jobs where it's an input have finished. Let's use the `temp` notation for for the output of `align_to_genome`, modify the rule so that the `output:` directive looks like this: -```python +```{.python filename="snakefile_mrsa.smk"} output: temp("results/bam/{sample_id,\\w+}.bam") ``` @@ -1186,7 +1186,7 @@ we'll show how these features help us condense the code of rules. Consider the rule `align_to_genome` below. -```python +```{.python filename="snakefile_mrsa.smk"} rule align_to_genome: """ Align a fastq file to a genome index using Bowtie 2. @@ -1214,7 +1214,7 @@ list of these files instead. If you're familiar with Python you could do this with [list comprehensions](https://docs.python.org/3/tutorial/datastructures.html#list-comprehensions) like this: -```python +```{.python filename="snakefile_mrsa.smk"} input: "data/{sample_id}.fastq.gz", [f"results/bowtie2/NCTC8325.{substr}.bt2" for @@ -1226,7 +1226,7 @@ that element in the place of `{substr}`. Since this type of aggregating rules are quite common, Snakemake also has a more compact way of achieving the same thing with the built-in function `expand()`. -```python +```{.python filename="snakefile_mrsa.smk"} input: "data/{sample_id}.fastq.gz", expand("results/bowtie2/NCTC8325.{substr}.bt2", @@ -1248,7 +1248,7 @@ Now change in the rules `index_genome` and `align_to_genome` to use the `expand()` expression. Click the solution below if you need help. ::: {.callout-tip collapse="true" title="Click to show"} -```python +```{.python filename="snakefile_mrsa.smk"} rule index_genome: """ Index a genome using Bowtie 2. @@ -1292,7 +1292,7 @@ rule align_to_genome: In the workflow we decide which samples to run by including the SRR ids in the names of the inputs to the rules `multiqc` and `generate_count_table`: -```python +```{.python filename="snakefile_mrsa.smk"} rule generate_count_table: output: "results/tables/counts.tsv" @@ -1331,7 +1331,7 @@ SAMPLES = ["SRR935090", "SRR935091", "SRR935092"] Now use `expand()` in `multiqc` and `generate_count_table` to use `SAMPLES` for the sample ids. For the `multiqc` rule it could look like this: -```python +```{.python filename="snakefile_mrsa.smk"} input: expand("results/fastqc/{sample_id}_fastqc.zip", sample_id = SAMPLES) ``` @@ -1340,7 +1340,7 @@ See if you can update the `generate_count_table` rule in the same manner. If you need help, click the solution below. ::: {.callout-tip collapse="true" title="Click to show"} -```python +```{.python filename="snakefile_mrsa.smk"} ule generate_count_table: """ Generate a count table using featureCounts. @@ -1368,7 +1368,7 @@ In this section we've learned: Take a look at the `index_genome` rule below: -```python +```{.python filename="snakefile_mrsa.smk"} rule index_genome: """ Index a genome using Bowtie 2. From 797e0daa3d6c3e3671d6d2e4ce2fec4cc582c593 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Sun, 17 Nov 2024 22:22:13 +0100 Subject: [PATCH 24/32] Remove rulegraph rule --- tutorials/snakemake/snakefile_mrsa.smk | 14 +------------- 1 file changed, 1 insertion(+), 13 deletions(-) diff --git a/tutorials/snakemake/snakefile_mrsa.smk b/tutorials/snakemake/snakefile_mrsa.smk index c07af13a..eb879261 100644 --- a/tutorials/snakemake/snakefile_mrsa.smk +++ b/tutorials/snakemake/snakefile_mrsa.smk @@ -8,7 +8,6 @@ rule all: input: "results/tables/counts.tsv", "results/multiqc/multiqc.html", - "results/rulegraph.png" def get_sample_url(wildcards): samples = { @@ -163,15 +162,4 @@ rule generate_count_table: shell: """ featureCounts -t gene -g gene_id -a {input.annotation} -o {output} {input.bams} - """ - -rule generate_rulegraph: - """ - Generate a rulegraph for the workflow. - """ - output: - "results/rulegraph.png" - shell: - """ - snakemake --snakefile snakefile_mrsa.smk --config max_reads=0 --rulegraph | dot -Tpng > {output} - """ + """ \ No newline at end of file From f0afd644e82aca911dcc6c482673ffa21154b8c8 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Mon, 18 Nov 2024 07:07:09 +0100 Subject: [PATCH 25/32] Update starting point for MRSA workflow --- tutorials/snakemake/snakefile_mrsa.smk | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/tutorials/snakemake/snakefile_mrsa.smk b/tutorials/snakemake/snakefile_mrsa.smk index eb879261..f008f702 100644 --- a/tutorials/snakemake/snakefile_mrsa.smk +++ b/tutorials/snakemake/snakefile_mrsa.smk @@ -1,5 +1,5 @@ from snakemake.utils import min_version -min_version("5.3.0") +min_version("8.0.0") rule all: """ @@ -7,7 +7,7 @@ rule all: """ input: "results/tables/counts.tsv", - "results/multiqc/multiqc.html", + "results/multiqc/multiqc.html" def get_sample_url(wildcards): samples = { @@ -27,7 +27,7 @@ rule get_SRA_by_accession: url = get_sample_url shell: """ - wget -O - {params.url} | seqtk sample - 25000 | gzip -c > {output[0]} + wget -q -O - {params.url} | seqtk sample - 25000 | gzip -c > {output[0]} """ rule fastqc: From e21277d7b08864eb298e82912d6904dbaba9a0c7 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Tue, 19 Nov 2024 07:33:51 +0100 Subject: [PATCH 26/32] Streamline MRSA generalization --- pages/snakemake.qmd | 526 +++++++++++++++++++++++++------------------- 1 file changed, 294 insertions(+), 232 deletions(-) diff --git a/pages/snakemake.qmd b/pages/snakemake.qmd index 6b02b0fe..4cd9d5a9 100644 --- a/pages/snakemake.qmd +++ b/pages/snakemake.qmd @@ -762,7 +762,41 @@ the first rule in a workflow as its target, unless a target is specified on the command line. By convention, this rule is called `all` and serves as a rule for aggregating the main outputs of the workflow. -![](images/rulegraph_mrsa.svg) +```{dot} +//| echo: false +//| label: MRSA_rulegraph +digraph MRSA_rulegraph { + graph[bgcolor=white, margin=0]; + node[shape=box, style=rounded, fontname=sans, fontsize=14, penwidth=2]; + edge[penwidth=2, color=grey]; + 0[label = "all", color = "0.07 0.6 0.85", style="rounded"]; + 1[label = "generate_count_table", color = "0.20 0.6 0.85", style="rounded"]; + 2[label = "sort_bam", color = "0.60 0.6 0.85", style="rounded"]; + 3[label = "align_to_genome", color = "0.00 0.6 0.85", style="rounded"]; + 4[label = "get_SRA_by_accession", color = "0.27 0.6 0.85", style="rounded"]; + 5[label = "index_genome", color = "0.47 0.6 0.85", style="rounded"]; + 6[label = "get_genome_fasta", color = "0.33 0.6 0.85", style="rounded"]; + 7[label = "get_genome_gff3", color = "0.40 0.6 0.85", style="rounded"]; + 8[label = "multiqc", color = "0.53 0.6 0.85", style="rounded"]; + 9[label = "fastqc", color = "0.13 0.6 0.85", style="rounded"]; + 8 -> 0 + 1 -> 0 + 7 -> 1 + 2 -> 1 + 3 -> 2 + 5 -> 3 + 4 -> 3 + 6 -> 5 + 9 -> 8 + 4 -> 9 +} +``` + +As you can see from the `snakefile_mrsa.smk` and the rulegraph, the input to the +`all` rule is the output from the rules `multiqc` and `generate_count_table`. +These rules in turn have their own inputs and outputs and by targeting the `all` +rule Snakemake determines what needs to be run by resolving the graph from +bottom to top. Now take some time and look through the workflow file and try to understand how the rules fit together. Use the rule graph as aid. The rules represent a quite @@ -782,12 +816,60 @@ will be used to generate a count table, _i.e._ a table that shows how many reads map to each gene for each sample. This count table is then what the downstream analysis will be based on. -![](images/dag_mrsa.svg) - -Now try to run the whole workflow. As before you can specify `-p` to have -Snakemake print the shell commands for rules and you can control how many cores -to use with `-c`. If you don't specify number of cores Snakemake will use all -available cores on your machine. Let's run the workflow with one core: +```{dot} +//| echo: false +//| label: MRSA_jobgraph +digraph MRSA_jobgraph { + graph[bgcolor=white, margin=0]; + node[shape=box, style=rounded, fontname=sans, fontsize=10, penwidth=2]; + edge[penwidth=2, color=grey]; + 0[label = "all", color = "0.07 0.6 0.85", style="rounded"]; + 1[label = "generate_count_table", color = "0.20 0.6 0.85", style="rounded"]; + 2[label = "sort_bam", color = "0.60 0.6 0.85", style="rounded"]; + 3[label = "align_to_genome", color = "0.00 0.6 0.85", style="rounded"]; + 4[label = "get_SRA_by_accession\nsample_id: SRR935090", color = "0.27 0.6 0.85", style="rounded"]; + 5[label = "index_genome", color = "0.47 0.6 0.85", style="rounded"]; + 6[label = "get_genome_fasta", color = "0.33 0.6 0.85", style="rounded"]; + 7[label = "sort_bam", color = "0.60 0.6 0.85", style="rounded"]; + 8[label = "align_to_genome", color = "0.00 0.6 0.85", style="rounded"]; + 9[label = "get_SRA_by_accession\nsample_id: SRR935091", color = "0.27 0.6 0.85", style="rounded"]; + 10[label = "sort_bam", color = "0.60 0.6 0.85", style="rounded"]; + 11[label = "align_to_genome", color = "0.00 0.6 0.85", style="rounded"]; + 12[label = "get_SRA_by_accession\nsample_id: SRR935092", color = "0.27 0.6 0.85", style="rounded"]; + 13[label = "get_genome_gff3", color = "0.40 0.6 0.85", style="rounded"]; + 14[label = "multiqc", color = "0.53 0.6 0.85", style="rounded"]; + 15[label = "fastqc", color = "0.13 0.6 0.85", style="rounded"]; + 16[label = "fastqc", color = "0.13 0.6 0.85", style="rounded"]; + 17[label = "fastqc", color = "0.13 0.6 0.85", style="rounded"]; + 1 -> 0 + 14 -> 0 + 2 -> 1 + 7 -> 1 + 10 -> 1 + 13 -> 1 + 3 -> 2 + 4 -> 3 + 5 -> 3 + 6 -> 5 + 8 -> 7 + 9 -> 8 + 5 -> 8 + 11 -> 10 + 12 -> 11 + 5 -> 11 + 15 -> 14 + 16 -> 14 + 17 -> 14 + 4 -> 15 + 9 -> 16 + 12 -> 17 +} +``` + +Now let's run the whole workflow. As before, specify `-p` to have Snakemake +print the shell commands for rules and use `-c` to control how many cores to run +with. If you don't specify number of cores Snakemake will use all available +cores on your machine. Let's run the workflow with one core: ```bash snakemake -s snakefile_mrsa.smk -c 1 -p @@ -796,9 +878,7 @@ snakemake -s snakefile_mrsa.smk -c 1 -p After everything is done, the workflow will have generated fastq files for each sample under `data/` and reference genome files under `data/ref`. The `results/` directory will contain the output from the different steps in the workflow, -divided into subdirectories. Take some time to look through the structure, in -particular the quality control reports in `results/multiqc/` and the count table -in `results/tables/`. +divided into subdirectories. Take some time to look through the structure. ::: {.callout-note title="Quick recap"} In this section we've learned: @@ -808,7 +888,7 @@ In this section we've learned: - Which output files the MRSA workflow produces. ::: -## Parameters +## Parameters and configuration In a typical bioinformatics project, considerable efforts are spent on tweaking parameters for the various programs involved. It would be inconvenient if you @@ -857,7 +937,7 @@ rule get_SRA_by_accession: url = get_sample_url shell: """ - wget -O - {params.url} | seqtk sample - 25000 | gzip -c > {output[0]} + wget -q -O - {params.url} | seqtk sample - 25000 | gzip -c > {output[0]} """ ``` @@ -867,7 +947,7 @@ Here `params` has a keyword called `url` which is set to a function called sample that the rule is supposed to download and this URL is then accessed in the `shell:` directive with `{params.url}`. -You may recognize this from [the basics](#the-basics) of this tutorial where we +You may recognize this from [The basics](#the-basics) of this tutorial where we used input functions to generate strings and lists of strings for the `input:` section of a rule. Using a function to return values based on the wildcards also works for `params:`. Here `sample_id` is a wildcard which in this specific @@ -904,7 +984,7 @@ rule get_SRA_by_accession: max_reads = 20000 shell: """ - wget -O - {params.url} | seqtk sample - {params.max_reads} | gzip -c > {output[0]} + wget -q -O - {params.url} | seqtk sample - {params.max_reads} | gzip -c > {output[0]} """ ``` ::: @@ -919,11 +999,10 @@ Because there's been changes to the `get_SRA_by_accession` rule this will trigger a re-run of the rule for all three accessions. In addition all downstream rules that depend on output from `get_SRA_by_accession` are re-run. -As you can see the parameter values we set in the `params` section don't have -to be static, they can be any Python expression. In particular, Snakemake -provides a global dictionary of configuration parameters called `config`. -Let's modify `get_SRA_by_accession` to look something like this in order to -make use of this dictionary: +As you can see the parameter values we set in the `params` section don't have to +be static, they can be any Python expression. In particular, Snakemake provides +a global dictionary of configuration parameters called `config`. Let's modify +`get_SRA_by_accession` in order to make use of this dictionary: ```{.python filename="snakefile_mrsa.smk"} rule get_SRA_by_accession: @@ -937,7 +1016,7 @@ rule get_SRA_by_accession: max_reads = config["max_reads"] shell: """ - wget -L {params.url} | seqtk sample - {params.max_reads} | gzip -c > {output[0]} + wget -q -O - {params.url} | seqtk sample - {params.max_reads} | gzip -c > {output[0]} """ ``` @@ -958,10 +1037,10 @@ project-specific settings, sample ids and so on. This also enables us to write the Snakefile in a more general manner so that it can be better reused between projects. Like several other files used in these tutorials, this file should be in [YAML format](https://en.wikipedia.org/wiki/YAML). Create the file below and -save it as `config.yml`. +save it as `config.yml` (here we change back to using 25000 reads): ```{.yaml filename="config.yml"} -max_reads: 20000 +max_reads: 25000 ``` If we now run Snakemake with `--configfile config.yml`, it will parse this file @@ -1021,13 +1100,14 @@ a little differently by Snakemake. For example, it's shown in the file summary when using `-D` and unlike other output files it's not deleted if jobs fail which of course is necessary for debugging purposes. It's also a good way to clarify the purpose of the file. We probably don't need to save logs for -all the rules, only the ones with interesting output. +all the rules, only the ones with interesting output: - `get_genome_fasta` and `get_genome_gff3` would be good to log since they are dependent on downloading files from an external server. - `multiqc` aggregates quality control data for all the samples into one html report, and the log contains information about which samples were aggregated. - `index_genome` outputs some statistics about the genome indexing. +- `generate_count_table` outputs information about input files, settings and a summary of results. - `align_to_genome` outputs important statistics about the alignments. This is probably the most important log to save. @@ -1042,8 +1122,7 @@ log to contain. Some of the programs we use send their log information to standard out, some to standard error and some let us specify a log file via a flag. -For example, in the `align_to_genome` rule, it could look like this (Bowtie2 -writes log info to standard error): +For example, `bowtie2` used in the `align_to_genome` rule writes log info to standard error so we use `2>{log}` to redirect this to the log file: ```{.python filename="snakefile_mrsa.smk"} rule align_to_genome: @@ -1051,7 +1130,7 @@ rule align_to_genome: Align a fastq file to a genome index using Bowtie 2. """ output: - "results/bam/{sample_id,\w+}.bam" + "results/bam/{sample_id,\\w+}.bam" input: "data/{sample_id}.fastq.gz", "results/bowtie2/NCTC8325.1.bt2", @@ -1068,21 +1147,19 @@ rule align_to_genome: """ ``` -To save some time you can use the info below. +For programs used in the rules `multiqc`, `get_genome_fasta`, `get_genome_gff3`, +`index_genome` and `generate_count_table` you can use the table below if you +need help with how to redirect output to the log file. -```bash -# wget has a -o flag for specifying the log file -wget remote_file -O output_file -o {log} +| Rule | Program | Redirect flag | +|-----------|----------------|----------------| +| `multiqc` | `multiqc` | `2>{log}` | +| `generate_count_table` | `featureCounts` | `2>{log}` | +| `get_genome_fasta`/`get_genome_gff3` | `wget` | `-o {log}` | +| `index_genome` | `bowtie2-build` | `>{log}` | -# MultiQC and featureCounts write to standard error so we redirect with "2>" -multiqc -n output_file input_files 2> {log} -featureCounts -t gene -g gene_id -a gff_file -o output_file input_files 2>{log} -# Bowtie2-build redirects to standard out so we use ">" -bowtie2-build input_file index_dir > {log} -``` - -Now rerun the whole workflow with: +After adding logfiles to the rules, rerun the whole workflow with: ```bash snakemake -s snakefile_mrsa.smk -p -c 1 --configfile config.yml @@ -1128,11 +1205,12 @@ output: temp("...") ``` The file will then be deleted as soon as all jobs where it's an input have -finished. Let's use the `temp` notation for for the output of `align_to_genome`, modify the rule so that the `output:` directive looks like this: +finished. Let's use the `temp` notation for for the output of `align_to_genome`, +modify the rule so that the `output:` directive looks like this: ```{.python filename="snakefile_mrsa.smk"} output: - temp("results/bam/{sample_id,\\w+}.bam") + temp("results/bam/{sample_id,\\w+}.bam") ``` We have to force a rerun of the rule for it to trigger, so we use `--forcerun align_to_genome` to do this. Run the workflow with: @@ -1158,10 +1236,8 @@ that are declared as temporary in your workflow you can instead run with the `--delete-temp-output` which will delete all remaining `temp` files. ::: -Snakemake has a number of options for marking files: +Snakemake has a number of additional options for marking files: -- `temp("...")`: The output file should be deleted once it's no longer needed - by any rules. - `protected("...")`: The output file should be write-protected. Typically used to protect files that require a huge amount of computational resources from being accidentally deleted. @@ -1192,7 +1268,7 @@ rule align_to_genome: Align a fastq file to a genome index using Bowtie 2. """ output: - "results/bam/{sample_id}.bam" + temp("results/bam/{sample_id,\\w+}.bam") input: "data/{sample_id}.fastq.gz", "results/bowtie2/NCTC8325.1.bt2", @@ -1201,30 +1277,17 @@ rule align_to_genome: "results/bowtie2/NCTC8325.4.bt2", "results/bowtie2/NCTC8325.rev.1.bt2", "results/bowtie2/NCTC8325.rev.2.bt2" + log: + "results/logs/align_to_genome/{sample_id}.log" shell: """ - bowtie2 -x results/bowtie2/NCTC8325 -U {input[0]} > {output} + bowtie2 -x results/bowtie2/NCTC8325 -U {input[0]} > {output} 2>{log} """ ``` Here we have seven inputs; the FASTQ file with the reads and six files with similar file names from the Bowtie2 genome indexing. Instead of writing all -the filenames we can tidy this up by using a Python expression to generate a -list of these files instead. If you're familiar with Python you could do -this with [list comprehensions](https://docs.python.org/3/tutorial/datastructures.html#list-comprehensions) -like this: - -```{.python filename="snakefile_mrsa.smk"} -input: - "data/{sample_id}.fastq.gz", - [f"results/bowtie2/NCTC8325.{substr}.bt2" for - substr in ["1", "2", "3", "4", "rev.1", "rev.2"]] -``` - -This will take the elements of the list of substrings one by one, and insert -that element in the place of `{substr}`. Since this type of aggregating -rules are quite common, Snakemake also has a more compact way of achieving the -same thing with the built-in function `expand()`. +the filenames we can tidy this up by using the built-in function `expand()`. ```{.python filename="snakefile_mrsa.smk"} input: @@ -1234,16 +1297,11 @@ input: ``` Here `expand` will take the string `"results/bowtie2/NCTC8325.{substr}.bt2"` and -insert each of the values in the list `["1", "2", "3", "4", "rev.1", "rev.2"]` -in place of `{substr}`. The result is a list of strings that can be used as -input to the rule. This is a very powerful feature that can save you a lot of +insert each of the values in the `substr` list: `["1", "2", "3", "4", "rev.1", +"rev.2"]` in place of `{substr}`. The result is a list of strings that is used +as input to the rule. This is a very powerful feature that can save you a lot of typing and make your code more readable. -::: {.callout-caution} -When using expand() like this, `substr` is **not** a wildcard because it is -resolved to the values explicitly given inside the expand expression. -::: - Now change in the rules `index_genome` and `align_to_genome` to use the `expand()` expression. Click the solution below if you need help. @@ -1284,11 +1342,13 @@ rule align_to_genome: "results/logs/align_to_genome/{sample_id}.log" shell: """ - bowtie2 -x results/bowtie2/NCTC8325 -U {input[0]} > {output} 2> {log} + bowtie2 -x results/bowtie2/NCTC8325 -U {input[0]} > {output} 2>{log} """ ``` ::: +Great! Let's put the `expand` function to more use! + In the workflow we decide which samples to run by including the SRR ids in the names of the inputs to the rules `multiqc` and `generate_count_table`: @@ -1312,28 +1372,22 @@ rule multiqc: ``` -The output files from these two rules, `results/multiqc.html` and -`results/tables/counts.tsv`, are in turn specified as input to the `all` rule at -the top of the file. Because the first rule is targeted by default when we run -Snakemake on the command line (like we mentioned in the [MRSA workflow -section](#the-mrsa-workflow)) this is what triggers the rules to run on each of -the three samples. - However, this is a potential source of errors since it's easy to change in one place and forget to change in the other. Because we can use Python code "everywhere" let's instead define a list of sample ids and put at the very top of the Snakefile, just before the rule `all`: -```python -SAMPLES = ["SRR935090", "SRR935091", "SRR935092"] +```{.python filename="snakefile_mrsa.smk"} +sample_ids = ["SRR935090", "SRR935091", "SRR935092"] ``` -Now use `expand()` in `multiqc` and `generate_count_table` to use `SAMPLES` for +Now use `expand()` in `multiqc` and `generate_count_table` to use `sample_ids` for the sample ids. For the `multiqc` rule it could look like this: ```{.python filename="snakefile_mrsa.smk"} input: - expand("results/fastqc/{sample_id}_fastqc.zip", sample_id = SAMPLES) + expand("results/fastqc/{sample_id}_fastqc.zip", + sample_id = sample_ids) ``` See if you can update the `generate_count_table` rule in the same manner. If you @@ -1341,22 +1395,30 @@ need help, click the solution below. ::: {.callout-tip collapse="true" title="Click to show"} ```{.python filename="snakefile_mrsa.smk"} -ule generate_count_table: +rule generate_count_table: """ Generate a count table using featureCounts. """ output: "results/tables/counts.tsv" input: - bams = expand("results/bam/{sample_id}.sorted.bam", sample_id = SAMPLES), + bams = expand("results/bam/{sample_id}.sorted.bam", sample_id = sample_ids), annotation = "data/ref/NCTC8325.gff3.gz" + log: + "results/logs/generate_count_table.log" shell: """ - featureCounts -t gene -g gene_id -a {input.annotation} -o {output} {input.bams} + featureCounts -t gene -g gene_id -a {input.annotation} -o {output} {input.bams} 2>{log} """ ``` ::: +Now run the workflow again with: + +```bash +snakemake -s snakefile_mrsa.smk -p -c 1 --configfile config.yml +``` + ::: {.callout-note title="Quick recap"} In this section we've learned: @@ -1384,7 +1446,7 @@ rule index_genome: """ # Bowtie2 cannot use .gz, so unzip to a temporary file first gunzip -c {input} > tempfile - bowtie2-build tempfile results/bowtie2/NCTC8325 >{log} + bowtie2-build tempfile results/bowtie2/NCTC8325 > {log} # Remove the temporary file rm tempfile @@ -1463,123 +1525,104 @@ In this section we've learned: - How to use the shadow option to handle files that are not tracked by Snakemake. ::: -## Generalising workflows +## Generalizing the MRSA workflow -It's a good idea to separate project-specific parameters from the -actual implementation of the workflow. This allows anyone using the -workflow to modify its behaviour without changing the underlying code, -making the workflow more general. +You've now improved the MRSA workflow by making it more readable, adding +logfiles a config file etc. However, the workflow is not very general since it's +hard-coded to work with the three samples `SRR935090`, `SRR935091` and +`SRR935092` and the genome `NCTC8325`. If someone would want to use the workflow +with other samples or genomes they would have to change the code in the +Snakefile, which is not optimal. Instead it's a good idea to separate +project-specific parameters from the actual implementation of the workflow. This +allows anyone using the workflow to modify its behaviour without changing the +underlying code, making the workflow more general. -In order to generalize our RNA-seq analysis workflow we should move all -project-specific information to `config.yml`. This means that we want the -config file to: +In this section we will go through how to make the workflow more general and +flexible by moving the sample ids and URLs to a configuration file and turning the reference genome into a wildcard. -- Specify which samples to run. -- Specify which genome to align to and where to download its sequence and - annotation files. -- (Contain any other parameters we might need to make it into a general - workflow, _e.g._ to support both paired-end and single-read sequencing) +### Specifying samples in a configuration file -::: {.callout-note} -Putting all configuration in `config.yml` will break the -`generate_rulegraph` rule. You can fix it either by replacing -`--config max_reads=0` with `--configfile=config.yml` in the shell -command of that rule in the Snakefile, or by adding -`configfile: "config.yml"` to the top of the Snakefile (as mentioned -in a previous tip). -::: - -The first point is straightforward; rather than using `SAMPLES = ["..."]` in -the Snakefile we define it as a parameter in `config.yml`. You can either add -it as a list similar to the way it was expressed before by adding: +Rather than using `sample_ids = ["..."]` in the Snakefile let's define it as a +parameter in `config.yml`. Remove the line we added to the top of +`snakefile_mrsa.smk` that defines `sample_ids` and add the following to +`config.yml`: -```yaml -SAMPLES: ["SRR935090", "SRR935091", "SRR935092"] +```{.yaml filename="config.yml"} +samples: + SRR935090: "https://figshare.scilifelab.se/ndownloader/files/39539767" + SRR935091: "https://figshare.scilifelab.se/ndownloader/files/39539770" + SRR935092: "https://figshare.scilifelab.se/ndownloader/files/39539773" ``` -To `config.yml`, or you can use this YAML notation (whether you -choose `SAMPLES` or `sample_ids` as the name of the entry doesn't matter, -you will just have to reference the same name in the config dictionary -inside the workflow): +Now when passing `--configfile config.yml`on the command line +`config["samples"]` will be a dictionary where the keys are the sample ids and +the values are the URLs to the FASTQ files. -```yaml -sample_ids: - - SRR935090 - - SRR935091 - - SRR935092 -``` +Next, modify the `get_sample_url` function to use this dictionary instead of the +hard-coded values: -Change the workflow to reference `config["sample_ids"]` (if using the latter -example) instead of `SAMPLES`, as in: - -```bash -expand("results/fastqc/{sample_id}_fastqc.zip", - sample_id = config["sample_ids"]) +```{.python filename="snakefile_mrsa.smk"} +def get_sample_url(wildcards): + return config["samples"][wildcards.sample_id] ``` -Remove the line with `SAMPLES = ["SRR935090", "SRR935091", "SRR935092"]` that we -added to the top of `snakefile_mrsa.smk` in the [targets section](#targets). - -Do a dry-run afterwards to make sure that everything works as expected. - -You may remember from the [parameters section](#parameters) part of this -tutorial that we're using a function to return the URL of the FASTQ files to -download for each sample: +Now change the `multiqc` and `generate_count_table` rules that use the `expand` function in the input directive to use `config["samples"].keys()` to expand the `sample_id` wildcard. This is how it should look for the `multiqc` rule: ```python -def get_sample_url(wildcards): - samples = { - "SRR935090": "https://figshare.scilifelab.se/ndownloader/files/39539767", - "SRR935091": "https://figshare.scilifelab.se/ndownloader/files/39539770", - "SRR935092": "https://figshare.scilifelab.se/ndownloader/files/39539773" - } - return samples[wildcards.sample_id] + input: + expand("results/fastqc/{sample_id}_fastqc.zip", + sample_id = config["samples"].keys()) ``` -Here the URLs of each sample_id is hard-coded in the `samples` dictionary -inside the function. To generalize this function we can move the definition -to the config file, placing it for example under an entry that we call -`sample_urls` like this: +Try to change the `generate_count_table` rule in the same way. Check the solution below if you need help. -```yaml -sample_urls: - SRR935090: "https://figshare.scilifelab.se/ndownloader/files/39539767" - SRR935091: "https://figshare.scilifelab.se/ndownloader/files/39539770" - SRR935092: "https://figshare.scilifelab.se/ndownloader/files/39539773" +::: {.callout-tip collapse="true" title="Click to show"} +```{.python filename="snakefile_mrsa.smk"} +rule generate_count_table: + """ + Generate a count table using featureCounts. + """ + output: + "results/tables/counts.tsv" + input: + bams = expand("results/bam/{sample_id}.sorted.bam", sample_id = config["samples"].keys()), + annotation = "data/ref/NCTC8325.gff3.gz" + log: + "results/logs/generate_count_table.log" + shell: + """ + featureCounts -t gene -g gene_id -a {input.annotation} -o {output} {input.bams} 2>{log} + """ ``` -This is what's called 'nested' key/value pairs, meaning that each sample_id --> URL pair becomes nested under the config key `sample_urls`. So in order -to access the URL of _e.g._ `SRR935090` we would use -`config["sample_urls"]["SRR935090"]`. This means that you will have to -update the `get_sample_url` function to: +To see that your new implementation works you can do a forced rerun: -```python -def get_sample_url(wildcards): - return config["sample_urls"][wildcards.sample_id] +```bash +snakemake -s snakefile_mrsa.smk -c 1 --configfile config.yml --forcerun generate_count_table multiqc get_SRA_by_accession ``` -Now the function uses the global `config` dictionary to return URLs for each -sample_id. Again, do a dry-run to see that the new implementation works. - ::: {.callout-tip} + If you were to scale up this workflow with more samples it could become -impractical to have to define the URLs by hand in the config file. A -tip then is to have a separate file where samples are listed in one column -and the URLs (or file paths) in another column. With a few lines of python -code you could then read that list at the start of the workflow and add -each sample to the config dictionary. +impractical to have to define the sample ids and URLs by hand in the config +file. A tip then is to have a separate file where samples are listed in one +column and the URLs (or file paths) in another column. With a few lines of +python code you could then read that list at the start of the workflow and add +each sample to the config dictionary. See +[below](#reading-samples-from-a-file-instead-of-hard-coding-them) for an example +of how to do this. + ::: -Now let's take a look at the genome reference used in the workflow. In the -`get_genome_fasta` and `get_genome_gff3` rules we have hard-coded FTP paths -to the FASTA GFF annotation file for the genome `NCTC8325`. We can -generalize this in a similar fashion to what we did with the -`get_SRA_by_accession` rule. Let's add a nested entry called `genomes` to -the config file that will hold the genome id and FTP paths to the FASTA and -GFF file: +## Generalizing the genome reference -```yaml +Now we'll take a look at how to generalize the genome reference used in the +workflow. + +In the `get_genome_fasta` and `get_genome_gff3` rules we have hard-coded FTP +paths to the FASTA and GFF annotation file for the genome `NCTC8325`. Let's move this information to the configuration file, and also add information for another genome, `ST398`. Add the following to `config.yml`: + +```{.yaml filename="config.yml"} genomes: NCTC8325: fasta: ftp://ftp.ensemblgenomes.org/pub/bacteria/release-37/fasta/bacteria_18_collection/staphylococcus_aureus_subsp_aureus_nctc_8325/dna//Staphylococcus_aureus_subsp_aureus_nctc_8325.ASM1342v1.dna_rm.toplevel.fa.gz @@ -1589,16 +1632,15 @@ genomes: gff3: ftp://ftp.ensemblgenomes.org/pub/bacteria/release-37/gff3/bacteria_18_collection/staphylococcus_aureus_subsp_aureus_st398//Staphylococcus_aureus_subsp_aureus_st398.ASM958v1.37.gff3.gz ``` -As you can see this is very similar to what with did with `sample_urls`, -just that we have one more nested level. Now to access the FTP path to the -FASTA file for genome id `NCTC8325` we can use -`config["genomes"]["NCTC8325"]["fasta"]`. +As you can see this is very similar to what we did with `samples`, it's just +that we have one more nested level. Now to access the FTP path to the FASTA file +for genome id `NCTC8325` we can use `config["genomes"]["NCTC8325"]["fasta"]`. Let's now look at how to do the mapping from genome id to FASTA path in the rule `get_genome_fasta`. This is how the rule currently looks (if you have added the log section as previously described). -```python +```{.python filename="snakefile_mrsa.smk"} rule get_genome_fasta: """ Retrieve the sequence in fasta format for a genome. @@ -1613,15 +1655,14 @@ rule get_genome_fasta: """ ``` -We don't want the hard-coded genome id `NCTC8325`, so replace that with a -wildcard, say `{genome_id}` (remember to add the wildcard to the `log:` -directive as well). We now need to supply the remote paths to the FASTA file -for a given genome id. Because we've added this information to the -config file we just need to pass it to the rule in some way, and just like -in the `get_SRA_by_accession` rule we'll use a function to do the job: +Replace the hard-coded `NCTC8325` with a wildcard `{genome_id}` in both the +`input` and `log` directives. We now need to supply the remote paths to the +FASTA file for a given genome id. Because we've added this information to the +config file we just need to pass it to the rule in some way, and just like in +the `get_SRA_by_accession` rule we'll use a function to do the job: -```python +```{.python filename="snakemfile_mrsa.smk"} def get_fasta_path(wildcards): return config["genomes"][wildcards.genome_id]["fasta"] @@ -1666,31 +1707,25 @@ rule get_genome_gff3: ``` ::: -Also change in `index_genome` to use a wildcard rather than a hard-coded genome -id. Here you will run into a complication if you have followed the previous -instructions and use the `expand()` expression. We want the list to expand to -`["results/bowtie2/{genome_id}.1.bt2", "results/bowtie2/{genome_id}.2.bt2", ...]`, -_i.e._ only expanding the wildcard referring to the Bowtie2 index. To keep the -`genome_id` wildcard from being expanded we have to "mask" it with double curly -brackets: `{{genome_id}}`. In addition, we need to replace the hard-coded -`results/bowtie2/NCTC8325` in the shell directive of the rule with the genome id -wildcard. Inside the shell directive the wildcard object is accessed with this -syntax: `{wildcards.genome_id}`, so the Bowtie2-build command should be: +Also replace `NCTC8325` with `{genome_id}` in the `log`, `input` and `output` of +the `index_genome` rule. Here you will run into a complication if you have +followed the previous instructions and use the `expand()` expression. We want +the list of output files be `["results/bowtie2/{genome_id}.1.bt2", +"results/bowtie2/{genome_id}.2.bt2", ...]`, _i.e._ only expanding the `substr` +wildcard in the Bowtie2 index. To prevent Snakemake from trying to expand the +`genome_id` wildcard we have to "mask" it with double curly brackets: +`{{genome_id}}`. + +In addition, we need to replace the hard-coded `results/bowtie2/NCTC8325` in the +shell directive of the rule with the genome id wildcard. Inside the shell +directive the wildcard object is accessed with this syntax: +`{wildcards.genome_id}`, so the Bowtie2-build command should be: ```bash bowtie2-build tempfile results/bowtie2/{wildcards.genome_id} > {log} ``` -Note that this will only work if the `{genome_id}` wildcard can be resolved to -something defined in the config (currently `NCTC8325` or `ST398`). If you try to -generate a FASTA file for a genome id not defined in the config Snakemake will -complain, even at the dry-run stage. - -Finally, remember that any wildcards need to be present both in the `output:` -and `log:` directives? This means we have to update the `log:` directive in -`index_genome` as well. The final rule should look like this: - -```python +```{.python filename="snakefile_mrsa.smk"} rule index_genome: """ Index a genome using Bowtie 2. @@ -1715,31 +1750,62 @@ Good job! The rules `get_genome_fasta`, `get_genome_gff3` and `index_genome` can now download and index *any genome* as long as we provide valid links in the config file. -However, we need to define somewhere which genome id we actually want to use -when running the workflow. This needs to be done both in `align_to_genome` and -`generate_count_table`. Do this by introducing a parameter in `config.yml` -called `"genome_id"` (you can set it to either `NCTC8325` or `ST398`), _e.g._: +Only two rules remain where the `NCTC8325` genome id is hard-coded: `align_to_genome` and `generate_count_table`. We will use these rules to define which genome id we actually want to use when running the workflow. First add a parameter to `config.yml` +called `"genome_id"` and set it to `NCTC8325`: -```yaml +```{.yaml filename="config.yml"} genome_id: "NCTC8325" ``` -Now we can resolve the `genome_id` wildcard from the config. See below for an -example for `align_to_genome`. Here the `substr` wildcard gets expanded from a -list while `genome_id` gets expanded from the config file. +Now we can resolve the `genome_id` wildcard from the config. For +`align_to_genome` we change the input directive to: ```python input: "data/{sample_id}.fastq.gz", - index = expand("results/bowtie2/{genome_id}.{substr}.bt2", + expand("results/bowtie2/{genome_id}.{substr}.bt2", genome_id = config["genome_id"], substr = ["1", "2", "3", "4", "rev.1", "rev.2"]) ``` -Also change the hard-coded genome id in the `generate_count_table` input in a -similar manner: +Here the `substr` wildcard gets expanded from a list while `genome_id` gets +expanded from the config dictionary. Also change the hard-coded `NCTC8325` in +the `shell:` directive of `align_to_genome` so that the genome_id is inserted +directly from the config, like this: ```python +shell: + """ + bowtie2 -x results/bowtie2/{config[genome_id]} -U {input[0]} > {output} 2>{log} + """ +``` + +The final rule should look like this: + +```{.python filename="snakefile_mrsa.smk"} +rule align_to_genome: + """ + Align a fastq file to a genome index using Bowtie 2. + """ + output: + temp("results/bam/{sample_id,\\w+}.bam") + input: + "data/{sample_id}.fastq.gz", + expand("results/bowtie2/{genome_id}.{substr}.bt2", + genome_id = config["genome_id"], + substr = ["1", "2", "3", "4", "rev.1", "rev.2"]) + log: + "results/logs/align_to_genome/{sample_id}.log" + shell: + """ + bowtie2 -x results/bowtie2/{config[genome_id]} -U {input[0]} > {output} 2>{log} + """ +``` + +Now let's change the hard-coded genome id in the `generate_count_table` input in +a similar manner. The final rule should look like this: + +```{.python filename="snakefile_mrsa.smk"} rule generate_count_table: """ Generate a count table using featureCounts. @@ -1749,7 +1815,7 @@ rule generate_count_table: "results/tables/counts.tsv.summary" input: bams=expand("results/bam/{sample_id}.sorted.bam", - sample_id = config["sample_ids"]), + sample_id = config["samples"].keys()), annotation=expand("data/ref/{genome_id}.gff3.gz", genome_id = config["genome_id"]) log: @@ -1760,6 +1826,14 @@ rule generate_count_table: """ ``` +To make sure that the workflow works as expected, run it with: + +```bash +snakemake -s snakefile_mrsa.smk -p -c 1 --configfile config.yml +``` + +Try changing the `genome_id` in the config file to `ST398` and rerun the workflow. You should now see that the workflow downloads the genome files for `ST398` and aligns the samples to that genome instead. + In general, we want the rules as far downstream as possible in the workflow to be the ones that determine what the wildcards should resolve to. In our case this is `align_to_genome` and `generate_count_table`. You can think of it like @@ -1768,18 +1842,6 @@ to determine how it can use all the available rules to generate it. Here the `align_to_genome` rule says "I need this genome index to align my sample to" and then it's up to Snakemake to determine how to download and build the index. -One last thing is to change the hard-coded `NCTC8325` in the `shell:` directive -of `align_to_genome`. Bowtie2 expects the index name supplied with the `-x` flag -to be without the ".*.bt2" suffix so we can't use `-x {input.index}`. Instead -we'll insert the genome_id directly from the config like this: - -```bash -shell: - """ - bowtie2 -x results/bowtie2/{config[genome_id]} -U {input[0]} > {output} 2>{log} - """ -``` - ::: {.callout-note title="Summary"} Well done! You now have a complete Snakemake workflow with a number of excellent features: From 5dba2e3dbeca9d379d5052923bd3ca0c7ace06a4 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Tue, 19 Nov 2024 14:46:40 +0100 Subject: [PATCH 27/32] Rewrite snakemake tutorial --- pages/snakemake.qmd | 404 ++++++++++++++++++++++++++++---------------- 1 file changed, 257 insertions(+), 147 deletions(-) diff --git a/pages/snakemake.qmd b/pages/snakemake.qmd index 4cd9d5a9..7348756b 100644 --- a/pages/snakemake.qmd +++ b/pages/snakemake.qmd @@ -736,16 +736,6 @@ modifications is available in `tutorials/containers/Snakefile`. You are probably already in your `snakemake-env` environment, otherwise activate it (use `conda info --envs` if you are unsure). -::: {.callout-tip} -Here we have one Conda environment for executing the whole Snakemake -workflow. Snakemake also supports using explicit Conda environments on -a per-rule basis, by specifying something like `conda: -rule-specific-env.yml` in the rule definition and running Snakemake with -the `--use-conda` flag. The given rule will then be run in the Conda -environment specified in `rule-specific-env.yml` that will be created and -activated on the fly by Snakemake. -::: - Let's start by generating the rule graph so that we get an overview of the workflow. Here we have to specify the file with the rules using the `-s` flag to Snakemake since the path to the file differs from the default. @@ -794,8 +784,8 @@ digraph MRSA_rulegraph { As you can see from the `snakefile_mrsa.smk` and the rulegraph, the input to the `all` rule is the output from the rules `multiqc` and `generate_count_table`. -These rules in turn have their own inputs and outputs and by targeting the `all` -rule Snakemake determines what needs to be run by resolving the graph from +These rules in turn have their own inputs from other rules and by targeting the +`all` rule Snakemake determines what needs to be run by resolving the graph from bottom to top. Now take some time and look through the workflow file and try to understand how @@ -899,9 +889,9 @@ keyword. ```python rule some_rule: output: - "..." + "results/some_output.txt" input: - "..." + "some_input.txt" params: cutoff=2.5 shell: @@ -939,7 +929,6 @@ rule get_SRA_by_accession: """ wget -q -O - {params.url} | seqtk sample - 25000 | gzip -c > {output[0]} """ - ``` Here `params` has a keyword called `url` which is set to a function called @@ -1084,14 +1073,14 @@ this. Just as we define `input` and `output` in a rule we can also define ```python rule some_rule: output: - "..." + "results/some_output.txt" input: - "..." + "some_input.txt" log: - "..." + "logs/some_log.txt" shell: """ - echo 'Converting {input} to {output}' > {log} + some_program {input} {output} > {log} """ ``` @@ -1102,10 +1091,10 @@ which of course is necessary for debugging purposes. It's also a good way to clarify the purpose of the file. We probably don't need to save logs for all the rules, only the ones with interesting output: -- `get_genome_fasta` and `get_genome_gff3` would be good to log since they are - dependent on downloading files from an external server. - `multiqc` aggregates quality control data for all the samples into one html report, and the log contains information about which samples were aggregated. +- `get_genome_fasta` and `get_genome_gff3` would be good to log since they are + dependent on downloading files from an external server. - `index_genome` outputs some statistics about the genome indexing. - `generate_count_table` outputs information about input files, settings and a summary of results. - `align_to_genome` outputs important statistics about the alignments. This is @@ -1154,15 +1143,114 @@ need help with how to redirect output to the log file. | Rule | Program | Redirect flag | |-----------|----------------|----------------| | `multiqc` | `multiqc` | `2>{log}` | -| `generate_count_table` | `featureCounts` | `2>{log}` | | `get_genome_fasta`/`get_genome_gff3` | `wget` | `-o {log}` | | `index_genome` | `bowtie2-build` | `>{log}` | +| `generate_count_table` | `featureCounts` | `2>{log}` | + + +If you need help, click to show the solution below for the rules. + +::: {.callout-tip collapse="true" title="Click to show"} +```{.python filename="snakefile_mrsa.smk"} +rule multiqc: + """ + Aggregate all FastQC reports into a MultiQC report. + """ + output: + html = "results/multiqc/multiqc.html", + stats = "results/multiqc/multiqc_general_stats.txt" + input: + "results/fastqc/SRR935090_fastqc.zip", + "results/fastqc/SRR935091_fastqc.zip", + "results/fastqc/SRR935092_fastqc.zip" + log: + "results/logs/multiqc.log" + shell: + """ + # Run multiQC and keep the html report + multiqc -n multiqc.html {input} 2>{log} + mv multiqc.html {output.html} + mv multiqc_data/multiqc_general_stats.txt {output.stats} + + # Remove the other directory that multiQC creates + rm -rf multiqc_data + """ + +rule get_genome_fasta: + """ + Retrieve the sequence in fasta format for a genome. + """ + output: + "data/ref/NCTC8325.fa.gz" + log: + "results/logs/get_genome_fasta.log" + shell: + """ + wget -o {log} ftp://ftp.ensemblgenomes.org/pub/bacteria/release-37/fasta/bacteria_18_collection/staphylococcus_aureus_subsp_aureus_nctc_8325/dna//Staphylococcus_aureus_subsp_aureus_nctc_8325.ASM1342v1.dna_rm.toplevel.fa.gz -O {output} + """ + +rule get_genome_gff3: + """ + Retrieve annotation in gff3 format for a genome. + """ + output: + "data/ref/NCTC8325.gff3.gz" + log: + "results/logs/get_genome_gff3.log" + shell: + """ + wget -o {log} ftp://ftp.ensemblgenomes.org/pub/bacteria/release-37/gff3/bacteria_18_collection/staphylococcus_aureus_subsp_aureus_nctc_8325//Staphylococcus_aureus_subsp_aureus_nctc_8325.ASM1342v1.37.gff3.gz -O {output} + """ + +rule index_genome: + """ + Index a genome using Bowtie 2. + """ + output: + "results/bowtie2/NCTC8325.1.bt2", + "results/bowtie2/NCTC8325.2.bt2", + "results/bowtie2/NCTC8325.3.bt2", + "results/bowtie2/NCTC8325.4.bt2", + "results/bowtie2/NCTC8325.rev.1.bt2", + "results/bowtie2/NCTC8325.rev.2.bt2" + input: + "data/ref/NCTC8325.fa.gz" + log: + "results/logs/index_genome.log" + shell: + """ + # Bowtie2 cannot use .gz, so unzip to a temporary file first + gunzip -c {input} > tempfile + bowtie2-build tempfile results/bowtie2/NCTC8325 >{log} + # Remove the temporary file + rm tempfile + """ + +rule generate_count_table: + """ + Generate a count table using featureCounts. + """ + output: + "results/tables/counts.tsv" + input: + bams = ["results/bam/SRR935090.sorted.bam", + "results/bam/SRR935091.sorted.bam", + "results/bam/SRR935092.sorted.bam"], + annotation = "data/ref/NCTC8325.gff3.gz" + log: + "results/logs/generate_count_table.log" + shell: + """ + featureCounts -t gene -g gene_id -a {input.annotation} -o {output} {input.bams} 2>{log} + """ +``` +::: After adding logfiles to the rules, rerun the whole workflow with: ```bash -snakemake -s snakefile_mrsa.smk -p -c 1 --configfile config.yml +snakemake -s snakefile_mrsa.smk --configfile config.yml -p -c 1 ``` Do the logs contain what they should? Note how much easier it is to follow the @@ -1201,7 +1289,9 @@ and `results/bam/{sample_id}.sorted.bam` as you can see if you list the contents In Snakemake we can mark an output file as temporary like this: ```python -output: temp("...") +rule some_rule: + output: + temp("results/some_output.txt") ``` The file will then be deleted as soon as all jobs where it's an input have @@ -1216,7 +1306,7 @@ modify the rule so that the `output:` directive looks like this: We have to force a rerun of the rule for it to trigger, so we use `--forcerun align_to_genome` to do this. Run the workflow with: ```bash -snakemake -s snakefile_mrsa.smk -p -c 1 --configfile config.yml --forcerun align_to_genome +snakemake -s snakefile_mrsa.smk --configfile config.yml -p -c 1 --forcerun align_to_genome ``` Take a look at the output printed to your terminal. After the rule `sort_bam` you should see something like this: @@ -1256,9 +1346,8 @@ In this section we've learned: ## The expand function -We've mentioned that Snakemake rules take either strings or a list of strings as -input, and that we can use any Python expression in Snakemake workflows. Here -we'll show how these features help us condense the code of rules. +Now that we have a working workflow with logs and temporary files, let's look at +a way to make the code more readable. Consider the rule `align_to_genome` below. @@ -1289,11 +1378,9 @@ Here we have seven inputs; the FASTQ file with the reads and six files with similar file names from the Bowtie2 genome indexing. Instead of writing all the filenames we can tidy this up by using the built-in function `expand()`. -```{.python filename="snakefile_mrsa.smk"} -input: - "data/{sample_id}.fastq.gz", - expand("results/bowtie2/NCTC8325.{substr}.bt2", - substr = ["1", "2", "3", "4", "rev.1", "rev.2"]) +```{.python} +expand("results/bowtie2/NCTC8325.{substr}.bt2", + substr = ["1", "2", "3", "4", "rev.1", "rev.2"]) ``` Here `expand` will take the string `"results/bowtie2/NCTC8325.{substr}.bt2"` and @@ -1302,8 +1389,9 @@ insert each of the values in the `substr` list: `["1", "2", "3", "4", "rev.1", as input to the rule. This is a very powerful feature that can save you a lot of typing and make your code more readable. -Now change in the rules `index_genome` and `align_to_genome` to use the -`expand()` expression. Click the solution below if you need help. +Now modify the rules `index_genome` and `align_to_genome` to use the `expand()` +expression in the output and input, respectively. Click the solution below if +you need help. ::: {.callout-tip collapse="true" title="Click to show"} ```{.python filename="snakefile_mrsa.smk"} @@ -1313,7 +1401,7 @@ rule index_genome: """ output: expand("results/bowtie2/NCTC8325.{substr}.bt2", - substr=["1", "2", "3", "4","rev.1", "rev.2"]) + substr=["1", "2", "3", "4","rev.1", "rev.2"]) input: "data/ref/NCTC8325.fa.gz" log: @@ -1322,7 +1410,7 @@ rule index_genome: """ # Bowtie2 cannot use .gz, so unzip to a temporary file first gunzip -c {input} > tempfile - bowtie2-build tempfile results/bowtie2/NCTC8325 > {log} + bowtie2-build tempfile results/bowtie2/NCTC8325 >{log} # Remove the temporary file rm tempfile @@ -1349,33 +1437,33 @@ rule align_to_genome: Great! Let's put the `expand` function to more use! -In the workflow we decide which samples to run by including the SRR ids in the -names of the inputs to the rules `multiqc` and `generate_count_table`: +The `multiqc` and `generate_count_table` rules take as input the fastqc zip files, and the sorted bam files for all three samples, respectively. See the excerpt below: ```{.python filename="snakefile_mrsa.smk"} -rule generate_count_table: - output: - "results/tables/counts.tsv" - input: - bams = ["results/bam/SRR935090.sorted.bam", - "results/bam/SRR935091.sorted.bam", - "results/bam/SRR935092.sorted.bam"], -... rule multiqc: - output: - html = "results/multiqc/multiqc.html", - stats = "results/multiqc/multiqc_general_stats.txt" + ... input: "results/fastqc/SRR935090_fastqc.zip", "results/fastqc/SRR935091_fastqc.zip", "results/fastqc/SRR935092_fastqc.zip" +... + +rule generate_count_table: + ... + input: + bams = ["results/bam/SRR935090.sorted.bam", + "results/bam/SRR935091.sorted.bam", + "results/bam/SRR935092.sorted.bam"], +... ``` -However, this is a potential source of errors since it's easy to change in one -place and forget to change in the other. Because we can use Python code -"everywhere" let's instead define a list of sample ids and put at the very -top of the Snakefile, just before the rule `all`: +Having the files for each sample written explicitly in these rules is how +Snakemake knows to run the workflow for all three samples. However, this is also +a potential source of errors since it's easy to change in one place and forget +to change in the other. Because we can use Python code "everywhere" let's +instead define a list of sample ids and put at the top of the Snakefile, just +before the rule `all`: ```{.python filename="snakefile_mrsa.smk"} sample_ids = ["SRR935090", "SRR935091", "SRR935092"] @@ -1384,9 +1472,9 @@ sample_ids = ["SRR935090", "SRR935091", "SRR935092"] Now use `expand()` in `multiqc` and `generate_count_table` to use `sample_ids` for the sample ids. For the `multiqc` rule it could look like this: -```{.python filename="snakefile_mrsa.smk"} -input: - expand("results/fastqc/{sample_id}_fastqc.zip", +```{.python} + input: + expand("results/fastqc/{sample_id}_fastqc.zip", sample_id = sample_ids) ``` @@ -1416,14 +1504,16 @@ rule generate_count_table: Now run the workflow again with: ```bash -snakemake -s snakefile_mrsa.smk -p -c 1 --configfile config.yml +snakemake -s snakefile_mrsa.smk --configfile config.yml -p -c 1 --forcerun align_to_genome index_genome multiqc generate_count_table ``` +to force a rerun of the modified rules and make sure that the workflow still +works as expected. + ::: {.callout-note title="Quick recap"} In this section we've learned: -- How to use the `expand()` expression to create a list with file names, - inserting all provided wildcard values. +- How to use the `expand()` expression to make the code more readable. ::: ## Shadow rules @@ -1446,7 +1536,7 @@ rule index_genome: """ # Bowtie2 cannot use .gz, so unzip to a temporary file first gunzip -c {input} > tempfile - bowtie2-build tempfile results/bowtie2/NCTC8325 > {log} + bowtie2-build tempfile results/bowtie2/NCTC8325 >{log} # Remove the temporary file rm tempfile @@ -1473,24 +1563,20 @@ There are a few options for `shadow` (for the full list of these options see the [Snakemake docs](https://snakemake.readthedocs.io/en/stable/snakefiles/rules.html#shadow-rules)). The most simple is `shadow: "minimal"`, which means that the rule is executed in -an empty directory that the input files to the rule have been symlinked into. -For the rule below, that means that the only file available would be `input.txt`. -The shell commands would generate the files `some_other_junk_file` and -`output.txt`. Lastly, Snakemake will move the output file (`output.txt`) to its -"real" location and remove the whole shadow directory. We therefore never have -to think about manually removing `some_other_junk_file`. +an empty shadow directory that the input files to the rule have been symlinked into. +For the rule below, that means that `some_input.txt` would be symlinked into a shadow directory under `.snakemake/shadow/`. The shell commands would generate the file `some_other_junk_file` in this shadow directory, as well as the output file `results/some_output.txt`. When the rule finishes, the shadow directory is removed which means we wouldn't have to think about manually removing `some_other_junk_file`. ```python rule some_rule: output: - "output.txt" + "results/some_output.txt" input: - "input.txt" + "some_input.txt" shadow: "minimal" shell: """ touch some_other_junk_file - cp {input} {output} + some_program {input} {output} """ ``` @@ -1498,12 +1584,57 @@ Try this out for the rules `multiqc` and `index_genome` where we have to "manually" deal with files that aren't tracked by Snakemake. Add `shadow: "minimal"` to these rules and also delete the shell commands that remove temporary files from those rules (the last lines in the shell directives that -begin with `rm`), as they are no longer needed. By editing the shell directive -you will trigger a rerun of these two rules, as well as all rules that depend on -them. So run the workflow with: +begin with `rm`), as they are no longer needed. Check the solution below if you +need help. + +::: {.callout-tip collapse="true" title="Click to show"} +```{.python filename="snakefile_mrsa.smk"} +rule multiqc: + """ + Aggregate all FastQC reports into a MultiQC report. + """ + output: + html = "results/multiqc/multiqc.html", + stats = "results/multiqc/multiqc_general_stats.txt" + input: + expand("results/fastqc/{sample_id}_fastqc.zip", + sample_id = sample_ids) + log: + "results/logs/multiqc.log" + shadow: "minimal" + shell: + """ + # Run multiQC and keep the html report + multiqc -n multiqc.html {input} 2>{log} + mv multiqc.html {output.html} + mv multiqc_data/multiqc_general_stats.txt {output.stats} + """ + +rule index_genome: + """ + Index a genome using Bowtie 2. + """ + output: + expand("results/bowtie2/NCTC8325.{substr}.bt2", + substr=["1", "2", "3", "4","rev.1", "rev.2"]) + input: + "data/ref/NCTC8325.fa.gz" + log: + "results/logs/index_genome/NCTC8325.log" + shadow: "minimal" + shell: + """ + # Bowtie2 cannot use .gz, so unzip to a temporary file first + gunzip -c {input} > tempfile + bowtie2-build tempfile results/bowtie2/NCTC8325 >{log} + """ +``` +::: + +Now run the workflow again with: ```bash -snakemake -s snakefile_mrsa.smk -p -c 1 --configfile config.yml +snakemake -s snakefile_mrsa.smk --configfile config.yml -p -c 1 ``` and make sure that you see neither `tempfile`, nor the `multiqc_data` directory @@ -1528,8 +1659,8 @@ In this section we've learned: ## Generalizing the MRSA workflow You've now improved the MRSA workflow by making it more readable, adding -logfiles a config file etc. However, the workflow is not very general since it's -hard-coded to work with the three samples `SRR935090`, `SRR935091` and +logfiles, a config file etc. However, the workflow is not very general since +it's hard-coded to work with the three samples `SRR935090`, `SRR935091` and `SRR935092` and the genome `NCTC8325`. If someone would want to use the workflow with other samples or genomes they would have to change the code in the Snakefile, which is not optimal. Instead it's a good idea to separate @@ -1542,10 +1673,8 @@ flexible by moving the sample ids and URLs to a configuration file and turning t ### Specifying samples in a configuration file -Rather than using `sample_ids = ["..."]` in the Snakefile let's define it as a -parameter in `config.yml`. Remove the line we added to the top of -`snakefile_mrsa.smk` that defines `sample_ids` and add the following to -`config.yml`: +Remove the `sample_ids = ["SRR935090", "SRR935091", "SRR935092"]` line we added +to the top of `snakefile_mrsa.smk` and add the following to `config.yml`: ```{.yaml filename="config.yml"} samples: @@ -1585,7 +1714,8 @@ rule generate_count_table: output: "results/tables/counts.tsv" input: - bams = expand("results/bam/{sample_id}.sorted.bam", sample_id = config["samples"].keys()), + bams = expand("results/bam/{sample_id}.sorted.bam", + sample_id = config["samples"].keys()), annotation = "data/ref/NCTC8325.gff3.gz" log: "results/logs/generate_count_table.log" @@ -1594,11 +1724,12 @@ rule generate_count_table: featureCounts -t gene -g gene_id -a {input.annotation} -o {output} {input.bams} 2>{log} """ ``` +::: -To see that your new implementation works you can do a forced rerun: +To see that your new implementation works you can do a forced rerun of the modified rules: ```bash -snakemake -s snakefile_mrsa.smk -c 1 --configfile config.yml --forcerun generate_count_table multiqc get_SRA_by_accession +snakemake -s snakefile_mrsa.smk --configfile config.yml -p -c 1 --forcerun generate_count_table multiqc get_SRA_by_accession ``` ::: {.callout-tip} @@ -1614,7 +1745,7 @@ of how to do this. ::: -## Generalizing the genome reference +### Generalizing the genome reference Now we'll take a look at how to generalize the genome reference used in the workflow. @@ -1725,6 +1856,8 @@ directive the wildcard object is accessed with this syntax: bowtie2-build tempfile results/bowtie2/{wildcards.genome_id} > {log} ``` +The final rule should look like this: + ```{.python filename="snakefile_mrsa.smk"} rule index_genome: """ @@ -1742,7 +1875,7 @@ rule index_genome: """ # Bowtie2 cannot use .gz, so unzip to a temporary file first gunzip -c {input} > tempfile - bowtie2-build tempfile results/bowtie2/{wildcards.genome_id} > {log} + bowtie2-build tempfile results/bowtie2/{wildcards.genome_id} >{log} """ ``` @@ -1815,9 +1948,9 @@ rule generate_count_table: "results/tables/counts.tsv.summary" input: bams=expand("results/bam/{sample_id}.sorted.bam", - sample_id = config["samples"].keys()), + sample_id = config["samples"].keys()), annotation=expand("data/ref/{genome_id}.gff3.gz", - genome_id = config["genome_id"]) + genome_id = config["genome_id"]) log: "results/logs/generate_count_table.log" shell: @@ -1829,14 +1962,14 @@ rule generate_count_table: To make sure that the workflow works as expected, run it with: ```bash -snakemake -s snakefile_mrsa.smk -p -c 1 --configfile config.yml +snakemake -s snakefile_mrsa.smk --configfile config.yml -p -c 1 ``` Try changing the `genome_id` in the config file to `ST398` and rerun the workflow. You should now see that the workflow downloads the genome files for `ST398` and aligns the samples to that genome instead. In general, we want the rules as far downstream as possible in the workflow to be the ones that determine what the wildcards should resolve to. In our case -this is `align_to_genome` and `generate_count_table`. You can think of it like +`align_to_genome` and `generate_count_table`. You can think of it like the rule that really "needs" the file asks for it, and then it's up to Snakemake to determine how it can use all the available rules to generate it. Here the `align_to_genome` rule says "I need this genome index to align my sample to" and @@ -1858,15 +1991,24 @@ excellent features: reproduce either via Git, Snakemake's `--archive` option or a Docker image. ::: -## Reading samples from a file instead of hard-coding them +## Extra material + +If you want to read more about Snakemake in general you can find several +resources here: + +- The Snakemake documentation is available on [ReadTheDocs](https://snakemake.readthedocs.io/en/stable/#). +- Here is another (quite in-depth) [tutorial](https://snakemake.readthedocs.io/en/stable/tutorial/tutorial.html#tutorial). +- If you have questions, check out [stack overflow](https://stackoverflow.com/questions/tagged/snakemake). + +### Reading samples from a file instead of hard-coding them So far we've specified the samples to use in the workflow either as a hard-coded list in the Snakefile, or as a list in the configuration file. This is of course impractical for large real-world examples. Here we'll just quickly show how you -could supply the samples instead via a tab-separated file. For example you could -create a file called `samples.tsv` with the following content: +could supply the samples to the MRSA workflow via a tab-separated file. For +example you could create a file called `samples.tsv` with the following content: -``` +```{.txt filename="samples.tsv"} SRR935090 https://figshare.scilifelab.se/ndownloader/files/39539767 SRR935091 https://figshare.scilifelab.se/ndownloader/files/39539770 SRR935092 https://figshare.scilifelab.se/ndownloader/files/39539773 @@ -1877,7 +2019,7 @@ fastq file. Now in order to read this into the workflow we need to use a few lines of python code. Since you can mix python code with rule definitions in Snakemake we'll just add the following lines to the top of the Snakefile: -```python +```{.python filename="snakefile_mrsa.smk"} # define an empty 'samples' dictionary samples = {} # read the sample list file and populate the dictionary @@ -1895,7 +2037,7 @@ url for `SRR935090` we can use `samples["SRR935090"]`. For example, the `get_sample_url` function can now be written as: -```python +```{.python filename="snakefile_mrsa.smk"} def get_sample_url(wildcards): return samples[wildcards.sample_id] ``` @@ -1903,25 +2045,9 @@ def get_sample_url(wildcards): We can also use the `samples` dictionary in `expand()`, for example in the `multiqc` rule: ```python -rule multiqc: - """ - Aggregate all FastQC reports into a MultiQC report. - """ - output: - html="results/multiqc/multiqc.html", - stats="results/multiqc/multiqc_general_stats.txt" input: - expand("results/fastqc/{sample_id}_fastqc.zip", sample_id = samples.keys()) - log: - "results/logs/multiqc/multiqc.log" - shadow: "minimal" - shell: - """ - # Run multiQC and keep the html report - multiqc -n multiqc.html {input} 2> {log} - mv multiqc.html {output.html} - mv multiqc_data/multiqc_general_stats.txt {output.stats} - """ + expand("results/fastqc/{sample_id}_fastqc.zip", + sample_id = samples.keys()) ``` Now this depends on there being a `samples.tsv` file in the working directory. @@ -1949,21 +2075,6 @@ with open(config["sample_list"], "r") as fhin: This way, anyone can take our Snakefile and just update the path to their own `sample_list` using the config file. -::: {.callout-note title="Quick recap"} -In this section we've learned: - -- How to generalize a Snakemake workflow. -::: - -## Extra material - -If you want to read more about Snakemake in general you can find several -resources here: - -- The Snakemake documentation is available on [ReadTheDocs](https://snakemake.readthedocs.io/en/stable/#). -- Here is another (quite in-depth) [tutorial](https://snakemake.readthedocs.io/en/stable/tutorial/tutorial.html#tutorial). -- If you have questions, check out [stack overflow](https://stackoverflow.com/questions/tagged/snakemake). - ### Using containers in Snakemake Snakemake also supports defining an Apptainer or Docker container for each rule @@ -1971,7 +2082,7 @@ Snakemake also supports defining an Apptainer or Docker container for each rule during the course). Analogous to using a rule-specific Conda environment, specify `container: "docker://some-account/rule-specific-image"` in the rule definition. Instead of a link to a container image, it is also possible to -provide the path to a `*.sif` file (= a _Singularity image file_). When +provide the path to an image file somwhere on your filesystem. When executing Snakemake, add the `--software-deployment-method apptainer` (or the shorthand `--sdm apptainer`) flag to the command line. For the given rule, an Apptainer container will then be created from the image or file that is provided @@ -2009,12 +2120,11 @@ Start your Snakemake workflow with the following command: snakemake --software-deployment-method apptainer ``` -Feel free to modify the MRSA workflow according to this example. As Apptainer -is a container software that was developed for HPC clusters, and for example the -Mac version is still a beta version, it might not work to run your updated -Snakemake workflow with Apptainer locally on your computer. -In the next section we explain how you can run Snakemake workflows on UPPMAX -where Apptainer is pre-installed. +Feel free to modify the MRSA workflow according to this example. As Apptainer is +a container software that was developed for HPC clusters with only limited +support on *e.g.* Macs, it might not work to run your updated Snakemake workflow +with Apptainer locally on your computer. In the next section we explain how you +can run Snakemake workflows on UPPMAX where Apptainer is pre-installed. ### Running Snakemake workflows on HPC clusters @@ -2042,9 +2152,9 @@ things on the cluster or even logging out of the cluster. For short workflows with only a few rules that need the same compute resources in terms of CPU (cores) and memory, you can submit the entire workflow as a job directly to the SLURM scheduler, or start an interactive job (in your `tmux` or -`screen` session) and run your Snakemake workflow as you would do that on your -local machine. Make sure to give your job enough time to finish running all -rules of your Snakemake workflow. +`screen` session), and run your Snakemake workflow as you would on your local +machine. Make sure to give your job enough time to finish running all rules of +your Snakemake workflow. If you choose this option, you don't need to install anything from the plugin catalogue. However, your workflow may not run as efficiently as it could if you @@ -2110,7 +2220,7 @@ rule testrule: resources: runtime = 60, mem_mb = 16000, - cpus_per_task = 4 + tasks = 4 shell: """ uname -a > {output} @@ -2119,7 +2229,7 @@ rule testrule: This rule uses the standard resource `runtime` to set the maximum allowed time (in minutes) for the rule, sets the memory requirement with `mem_mb` and the -number of requested CPUs with `cpus_per_task`. In this example the rule will have a +number of requested CPUs with `tasks`. In this example the rule will have a time limit of 60 minutes, will require 16G of RAM and 4 CPUs. Some clusters also require you to specify the **partition** you want to run your @@ -2137,7 +2247,7 @@ rule testrule: resources: runtime = 60, mem_mb = 16000, - cpus_per_task = 4, + tasks = 4, slurm_partition: "shared" shell: """ @@ -2153,14 +2263,14 @@ Snakemake without changing the workflow code. For example, you could create a `dardel` folder (_e.g._ in the root of your workflow) with a `config.yaml` file that contains the following: -```yaml +```{.yaml filename="dardel/config.yaml"} executor: "slurm" jobs: 100 default-resources: slurm_account: "naiss-2023-01-001" slurm_partition: "shared" mem_mb: 16000 - cpus_per_task: 4 + tasks: 1 runtime: 60 ``` @@ -2178,24 +2288,24 @@ the command line call much more succinct. To set rule-specific resources in the configuration profile, you can add a `set_resources:` section to the `config.yaml` file: -```yaml +```{.yaml filename="dardel/config.yaml"} executor: "slurm" jobs: 100 default-resources: slurm_account: "naiss-2023-01-001" slurm_partition: "shared" mem_mb: 16000 - cpus_per_task: 4 + tasks: 4 runtime: 60 set_resources: index_genome: runtime: 240 mem_mb: 32000 - cpus_per_task: 8 + tasks: 8 align_to_genome: runtime: 120 mem_mb: 24000 - cpus_per_task: 6 + tasks: 6 ``` In this example, the `index_genome` rule will have a runtime of 240 minutes, From d80cf5583626d92380177a1fe8c6e6c04ad8704e Mon Sep 17 00:00:00 2001 From: John Sundh Date: Tue, 19 Nov 2024 14:48:30 +0100 Subject: [PATCH 28/32] Update finished Snakemake workflow Remove rulegraph output and update config params for better consistency --- tutorials/containers/Snakefile | 42 +++++++------------ .../code/supplementary_material.qmd | 2 - tutorials/containers/config.yml | 2 +- tutorials/containers/environment.yml | 2 +- 4 files changed, 16 insertions(+), 32 deletions(-) diff --git a/tutorials/containers/Snakefile b/tutorials/containers/Snakefile index de562cc2..5af76853 100644 --- a/tutorials/containers/Snakefile +++ b/tutorials/containers/Snakefile @@ -1,5 +1,5 @@ from snakemake.utils import min_version -min_version("5.3.0") +min_version("8.0.0") configfile: "config.yml" @@ -10,11 +10,10 @@ rule all: input: "results/tables/counts.tsv", "results/multiqc.html", - "results/rulegraph.png", "results/supplementary.html" def get_sample_url(wildcards): - return config["sample_urls"][wildcards.sample_id] + return config["samples"][wildcards.sample_id] rule get_SRA_by_accession: """ @@ -40,7 +39,7 @@ rule get_SRA_by_accession: url = get_sample_url shell: """ - wget -o {log} -O - {params.url} | seqtk sample - {params.max_reads} | gzip -c > {output[0]} + wget -q -o {log} -O - {params.url} | seqtk sample - {params.max_reads} | gzip -c > {output[0]} """ rule fastqc: @@ -71,14 +70,14 @@ rule multiqc: html="results/multiqc.html", stats="results/multiqc/multiqc_general_stats.txt" input: - expand("results/fastqc/{sample_id}_fastqc.zip", sample_id = config["sample_ids"]) + expand("results/fastqc/{sample_id}_fastqc.zip", sample_id = config["samples"].keys()) log: "results/logs/multiqc/multiqc.log" shadow: "minimal" shell: """ # Run multiQC and keep the html report - multiqc -n multiqc.html {input} 2> {log} + multiqc -n multiqc.html {input} 2>{log} mv multiqc.html {output.html} mv multiqc_data/multiqc_general_stats.txt {output.stats} """ @@ -134,7 +133,7 @@ rule index_genome: """ # Bowtie2 cannot use .gz, so unzip to a temporary file first gunzip -c {input} > tempfile - bowtie2-build tempfile results/bowtie2/{wildcards.genome_id} > {log} + bowtie2-build tempfile results/bowtie2/{wildcards.genome_id} >{log} """ rule align_to_genome: @@ -144,10 +143,10 @@ rule align_to_genome: output: # Here the sample_id wildcard is constrained with \w+ to match only # 'word characters', i.e. letters, numbers and underscore - temp("results/bam/{sample_id,\w+}.bam") + temp("results/bam/{sample_id,\\w+}.bam") input: - fastq = "data/{sample_id}.fastq.gz", - index = expand("results/bowtie2/{genome_id}.{substr}.bt2", + "data/{sample_id}.fastq.gz", + expand("results/bowtie2/{genome_id}.{substr}.bt2", genome_id=config["genome_id"], substr=["1", "2", "3", "4", "rev.1", "rev.2"]) log: @@ -155,7 +154,7 @@ rule align_to_genome: genome_id = config["genome_id"]) shell: """ - bowtie2 -x results/bowtie2/{config[genome_id]} -U {input.fastq} > {output} 2>{log} + bowtie2 -x results/bowtie2/{config[genome_id]} -U {input[0]} > {output} 2>{log} """ rule sort_bam: @@ -163,9 +162,9 @@ rule sort_bam: Sort a bam file. """ output: - "{prefix}.sorted.bam" + "results/bam/{sample_id}.sorted.bam" input: - "{prefix}.bam" + "results/bam/{sample_id}.bam" shell: """ samtools sort {input} > {output} @@ -179,7 +178,7 @@ rule generate_count_table: "results/tables/counts.tsv", "results/tables/counts.tsv.summary" input: - bams=expand("results/bam/{sample_id}.sorted.bam", sample_id = config["sample_ids"]), + bams=expand("results/bam/{sample_id}.sorted.bam", sample_id = config["samples"].keys()), annotation=expand("data/ref/{genome_id}.gff3.gz", genome_id = config["genome_id"]) log: "results/logs/generate_count_table.log" @@ -198,7 +197,6 @@ rule make_supplementary: counts = "results/tables/counts.tsv", summary_file = "results/tables/counts.tsv.summary", multiqc_file = "results/multiqc/multiqc_general_stats.txt", - rulegraph = "results/rulegraph.png" log: "results/logs/make_supplementary.log" params: @@ -210,19 +208,7 @@ rule make_supplementary: -P counts_file:../{input.counts} \ -P multiqc_file:../{input.multiqc_file} \ -P summary_file:../{input.summary_file} \ - -P rulegraph_file:../{input.rulegraph} \ -P srr_ids:"{params.SRR_IDs}" \ -P gsm_ids:"{params.GSM_IDs}" mv code/supplementary_material.html {output} - """ - -rule generate_rulegraph: - """ - Generate a rulegraph for the workflow. - """ - output: - "results/rulegraph.png" - shell: - """ - snakemake --rulegraph --configfile config.yml | dot -Tpng > {output} - """ + """ \ No newline at end of file diff --git a/tutorials/containers/code/supplementary_material.qmd b/tutorials/containers/code/supplementary_material.qmd index fc8d6c8c..989e6b35 100644 --- a/tutorials/containers/code/supplementary_material.qmd +++ b/tutorials/containers/code/supplementary_material.qmd @@ -9,7 +9,6 @@ params: counts_file: "results/tables/counts.tsv" multiqc_file: "results/multiqc/multiqc_general_stats.txt" summary_file: "results/tables/counts.tsv.summary" - rulegraph_file: "results/rulegraph.png" srr_ids: "SRR935090 SRR935091 SRR935092" gsm_ids: "GSM1186459 GSM1186460 GSM1186461" --- @@ -27,7 +26,6 @@ library("GEOquery") counts_file <- params$counts_file multiqc_file <- params$multiqc_file summary_file <- params$summary_file -rulegraph_file <- params$rulegraph_file srr_ids <- unlist(strsplit(params$srr_ids, " ")) gsm_ids <- unlist(strsplit(params$gsm_ids, " ")) ``` diff --git a/tutorials/containers/config.yml b/tutorials/containers/config.yml index 73e41615..3fd9764b 100644 --- a/tutorials/containers/config.yml +++ b/tutorials/containers/config.yml @@ -4,7 +4,7 @@ sample_ids_geo: ["GSM1186459", "GSM1186460", "GSM1186461"] series_id_geo: "GSE48896" # URLs to gzipped fastq files for each sample in remote repository -sample_urls: +samples: SRR935090: "https://figshare.scilifelab.se/ndownloader/files/39539767" SRR935091: "https://figshare.scilifelab.se/ndownloader/files/39539770" SRR935092: "https://figshare.scilifelab.se/ndownloader/files/39539773" diff --git a/tutorials/containers/environment.yml b/tutorials/containers/environment.yml index ee440128..95e54aa1 100644 --- a/tutorials/containers/environment.yml +++ b/tutorials/containers/environment.yml @@ -5,7 +5,7 @@ dependencies: - python - fastqc - seqtk - - snakemake + - snakemake>=8 - multiqc - jupyter - bowtie2 From 415c970516e7c0acec16259c8e6ed74ddd77c1d5 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Tue, 19 Nov 2024 22:26:49 +0100 Subject: [PATCH 29/32] Set smaller fontsize for code --- assets/css/styles.scss | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/assets/css/styles.scss b/assets/css/styles.scss index 183fa179..e36624ac 100644 --- a/assets/css/styles.scss +++ b/assets/css/styles.scss @@ -116,6 +116,10 @@ $footer-fg: DimGray; /* code */ +pre, code, .sourceCode { + font-size: 0.75rem; +} + pre { line-height: 1.4; background-color: $code-block-bg; From cbe1e23f6604f847d155fb9424aedb8dd3a7b039 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Tue, 19 Nov 2024 23:07:46 +0100 Subject: [PATCH 30/32] Set min Snakemake version 8 --- tutorials/snakemake/environment.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tutorials/snakemake/environment.yml b/tutorials/snakemake/environment.yml index 6de7ac7e..b3a19c8b 100644 --- a/tutorials/snakemake/environment.yml +++ b/tutorials/snakemake/environment.yml @@ -6,7 +6,7 @@ dependencies: - python - fastqc - seqtk - - snakemake + - snakemake>=8 - multiqc - bowtie2 - tbb From 7d7728ae06de63707e74bce88d8fd1b8c49c1774 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Wed, 20 Nov 2024 00:58:39 +0100 Subject: [PATCH 31/32] Update snakemake lecture --- lectures/snakemake/snakemake.qmd | 189 ++++++++++++++++--------------- 1 file changed, 97 insertions(+), 92 deletions(-) diff --git a/lectures/snakemake/snakemake.qmd b/lectures/snakemake/snakemake.qmd index b7d217c5..ba2789f0 100644 --- a/lectures/snakemake/snakemake.qmd +++ b/lectures/snakemake/snakemake.qmd @@ -98,12 +98,10 @@ digraph snakemake_dag { 15[label = "fastqc", color = "0.00 0.6 0.85", style="rounded"]; 16[label = "fastqc", color = "0.00 0.6 0.85", style="rounded"]; 17[label = "fastqc", color = "0.00 0.6 0.85", style="rounded"]; - 18[label = "generate_rulegraph", color = "0.33 0.6 0.85", style="rounded"]; - 19[label = "make_supplementary", color = "0.39 0.6 0.85", style="rounded"]; + 18[label = "make_supplementary", color = "0.39 0.6 0.85", style="rounded"]; 1 -> 0 14 -> 0 18 -> 0 - 19 -> 0 2 -> 1 7 -> 1 10 -> 1 @@ -124,9 +122,8 @@ digraph snakemake_dag { 4 -> 15 9 -> 16 12 -> 17 - 1 -> 19 - 14 -> 19 - 18 -> 19 + 1 -> 18 + 14 -> 18 } ``` @@ -178,7 +175,7 @@ digraph snakemake_dag { ## Example: sequence trimming Using a bash-script: -```{.bash code-line-numbers="|1-2,10|3|4-5|6-7|8-9"} +```{.bash filename="trimfastq.sh" code-line-numbers="|1-2,10|3|4-5|6-7|8-9"} for input in *.fastq do sample=$(echo ${input} | sed 's/.fastq//') @@ -201,7 +198,7 @@ $ bash trimfastq.sh ## Example: sequence trimming {auto-animate=true} Using Snakemake rules: -```{.python code-line-numbers="|1,6|3|2,8|4,5,9,10|7"} +```{.python filename="Snakefile" code-line-numbers="|1,6|3|2,8|4,5,9,10|7"} rule trim_fastq: output: temp("{sample}.trimmed.fastq") input: "{sample}.fastq" @@ -235,38 +232,79 @@ $ snakemake -c 1 a.trimmed.fastq.gz b.trimmed.fastq.gz ```{.default code-line-numbers=false} $ snakemake -c 1 a.trimmed.fastq.gz b.trimmed.fastq.gz -Provided cores: 1 +Building DAG of jobs... +Using shell: /bin/bash +Provided cores: 1 (use --cores to define parallelism) Rules claiming more threads will be scaled down. -Job counts: -count jobs -2 gzip -2 trim_fastq -4 -rule trim_fastq: +Job stats: +job count +---------- ------- +gzip 2 +trim_fastq 2 +total 4 +Select jobs to execute... +Execute 1 jobs... + +[Tue Nov 19 23:09:00 2024] +localrule trim_fastq: + input: b.fastq + output: b.trimmed.fastq + jobid: 3 + reason: Missing output files: b.trimmed.fastq + wildcards: sample=b + resources: tmpdir=/var/folders/wb/jf9h8kw11b734gd98s6174rm0000gp/T + +[Tue Nov 19 23:09:01 2024] +Finished job 3. +1 of 4 steps (25%) done +Select jobs to execute... +Execute 1 jobs... + +[Tue Nov 19 23:09:01 2024] +localrule gzip: + input: b.trimmed.fastq + output: b.trimmed.fastq.gz + jobid: 2 + reason: Missing output files: b.trimmed.fastq.gz; Input files updated by another job: b.trimmed.fastq + wildcards: sample=b + resources: tmpdir=/var/folders/wb/jf9h8kw11b734gd98s6174rm0000gp/T + +[Tue Nov 19 23:09:02 2024] +Finished job 2. +2 of 4 steps (50%) done +Removing temporary output b.trimmed.fastq. +Select jobs to execute... +Execute 1 jobs... + +[Tue Nov 19 23:09:02 2024] +localrule trim_fastq: input: a.fastq output: a.trimmed.fastq + jobid: 1 + reason: Missing output files: a.trimmed.fastq wildcards: sample=a - 1 of 4 steps (25%) done + resources: tmpdir=/var/folders/wb/jf9h8kw11b734gd98s6174rm0000gp/T -rule gzip: +[Tue Nov 19 23:09:02 2024] +Finished job 1. +3 of 4 steps (75%) done +Select jobs to execute... +Execute 1 jobs... + +[Tue Nov 19 23:09:02 2024] +localrule gzip: input: a.trimmed.fastq output: a.trimmed.fastq.gz + jobid: 0 + reason: Missing output files: a.trimmed.fastq.gz; Input files updated by another job: a.trimmed.fastq wildcards: sample=a -Removing temporary output file a.trimmed.fastq. -2 of 4 steps (50%) done - -rule trim_fastq: - input: b.fastq - output: b.trimmed.fastq - wildcards: sample=b -3 of 4 steps (75%) done + resources: tmpdir=/var/folders/wb/jf9h8kw11b734gd98s6174rm0000gp/T -rule gzip: - input: b.trimmed.fastq - output: b.trimmed.fastq.gz - wildcards: sample=b -Removing temporary output file b.trimmed.fastq. +[Tue Nov 19 23:09:03 2024] +Finished job 0. 4 of 4 steps (100%) done +Removing temporary output a.trimmed.fastq. +Complete log: .snakemake/log/2024-11-19T230900.634412.snakemake.log ``` ## Getting into the Snakemake mindset @@ -531,11 +569,11 @@ rule trim_fastq: ```{dot} digraph snakemake_dag { graph[bgcolor=white, margin=1]; - node[shape=box, style=rounded, fontname=sans, fontsize=10, penwidth=2]; + node[shape=box, style=rounded, fontname=sans, fontsize=18, penwidth=2]; edge[penwidth=2, color=grey]; 0[label = "make_supplementary", color = "black", style="rounded"]; 1[label = "generate_count_table", color = "black", style="rounded"]; - 2[label = "sort_bam\nprefix: results/bam/SRR935090", color = "black", style="rounded"]; + 2[label = "sort_bam\nsample_id: SRR935090", color = "black", style="rounded"]; 3[label = "align_to_genome", color = "black", style="rounded"]; 4[label = "get_SRA_by_accession\nsample_id: SRR935090", color = "black", style="rounded"]; 5[label = "index_genome", color = "black", style="rounded"]; @@ -543,10 +581,8 @@ digraph snakemake_dag { 7[label = "get_genome_gff3\ngenome_id: NCTC8325", color = "black", style="rounded"]; 8[label = "multiqc", color = "black", style="rounded"]; 9[label = "fastqc", color = "black", style="rounded"]; - 10[label = "generate_rulegraph", color = "black", style="rounded"]; 1 -> 0 8 -> 0 - 10 -> 0 2 -> 1 7 -> 1 3 -> 2 @@ -561,9 +597,7 @@ digraph snakemake_dag { ## {auto-animate=true} -- The tutorial contains a workflow to download and map RNA-seq reads against a - reference genome. -- Here we ask for [results/supplementary.html]{.green}, which is an Quarto +- Here we ask for [results/supplementary.html]{.green}, which is a Quarto report generated by the rule `make_supplementary`: ::: {data-id="code1"} @@ -577,11 +611,11 @@ $ snakemake -c 1 results/supplementary.html ```{dot} digraph snakemake_dag { graph[bgcolor=white, margin=1]; - node[shape=box, style=rounded, fontname=sans, fontsize=10, penwidth=2]; + node[shape=box, style=rounded, fontname=sans, fontsize=18, penwidth=2]; edge[penwidth=2, color=grey]; 0[label = "make_supplementary", color = "red", style="rounded"]; 1[label = "generate_count_table", color = "black", style="rounded"]; - 2[label = "sort_bam\nprefix: results/bam/SRR935090", color = "black", style="rounded"]; + 2[label = "sort_bam\nsample_id: SRR935090", color = "black", style="rounded"]; 3[label = "align_to_genome", color = "black", style="rounded"]; 4[label = "get_SRA_by_accession\nsample_id: SRR935090", color = "black", style="rounded"]; 5[label = "index_genome", color = "black", style="rounded"]; @@ -589,10 +623,8 @@ digraph snakemake_dag { 7[label = "get_genome_gff3\ngenome_id: NCTC8325", color = "black", style="rounded"]; 8[label = "multiqc", color = "black", style="rounded"]; 9[label = "fastqc", color = "black", style="rounded"]; - 10[label = "generate_rulegraph", color = "black", style="rounded"]; 1 -> 0 8 -> 0 - 10 -> 0 2 -> 1 7 -> 1 3 -> 2 @@ -607,10 +639,6 @@ digraph snakemake_dag { ## {auto-animate=true} -- The tutorial contains a workflow to download and map RNA-seq reads against a - reference genome. -- Here we ask for [results/supplementary.html]{.green}, which is an Quarto - report generated by the rule `make_supplementary`: - If the timestamp of a file upstream in the workflow is updated... ::: {data-id="code1"} @@ -623,11 +651,11 @@ $ touch results/bowtie2/NCTC8325.1.bt2 ```{dot} digraph snakemake_dag { graph[bgcolor=white, margin=1]; - node[shape=box, style=rounded, fontname=sans, fontsize=10, penwidth=2]; + node[shape=box, style=rounded, fontname=sans, fontsize=18, penwidth=2]; edge[penwidth=2, color=grey]; 0[label = "make_supplementary", color = "black", style="rounded"]; 1[label = "generate_count_table", color = "black", style="rounded"]; - 2[label = "sort_bam\nprefix: results/bowtie2/SRR935090", color = "black", style="rounded"]; + 2[label = "sort_bam\nsample_id: SRR935090", color = "black", style="rounded"]; 3[label = "align_to_genome", color = "black", style="rounded"]; 4[label = "get_SRA_by_accession\nsample_id: SRR935090", color = "black", style="rounded"]; 5[label = "index_genome*", color = "cyan", style="rounded"]; @@ -635,10 +663,8 @@ digraph snakemake_dag { 7[label = "get_genome_gff3\ngenome_id: NCTC8325", color = "black", style="rounded"]; 8[label = "multiqc", color = "black", style="rounded"]; 9[label = "fastqc", color = "black", style="rounded"]; - 10[label = "generate_rulegraph", color = "black", style="rounded"]; 1 -> 0 8 -> 0 - 10 -> 0 2 -> 1 7 -> 1 3 -> 2 @@ -653,11 +679,6 @@ digraph snakemake_dag { ## {auto-animate=true} -- The tutorial contains a workflow to download and map RNA-seq reads against a - reference genome. -- Here we ask for [results/supplementary.html]{.green}, which is an Quarto - report generated by the rule `make_supplementary`: -- If the timestamp of a file upstream in the workflow is updated... - Snakemake detects a file change and only reruns the necessary rules. ::: {data-id="code1"} @@ -670,11 +691,11 @@ $ snakemake -c 1 results/supplementary.html ```{dot} digraph snakemake_dag { graph[bgcolor=white, margin=1]; - node[shape=box, style=rounded, fontname=sans, fontsize=10, penwidth=2]; + node[shape=box, style=rounded, fontname=sans, fontsize=18, penwidth=2]; edge[penwidth=2, color=grey]; 0[label = "make_supplementary", color = "red", style="rounded"]; 1[label = "generate_count_table", color = "black", style="rounded"]; - 2[label = "sort_bam\nprefix: results/bowtie2/SRR935090", color = "black", style="rounded"]; + 2[label = "sort_bam\nsample_id: SRR935090", color = "black", style="rounded"]; 3[label = "align_to_genome", color = "black", style="rounded"]; 4[label = "get_SRA_by_accession\nsample_id: SRR935090", fontcolor="grey", fillcolor="lightgrey", color = "grey", style="rounded,dashed,filled"]; 5[label = "index_genome", color = "grey", fillcolor="lightgrey", fontcolor="grey", style="rounded,dashed,filled"]; @@ -682,10 +703,8 @@ digraph snakemake_dag { 7[label = "get_genome_gff3\ngenome_id: NCTC8325", color = "grey", fontcolor="grey", fillcolor="lightgrey", style="rounded,dashed,filled"]; 8[label = "multiqc", fontcolor="grey", fillcolor="lightgrey", color = "grey", style="rounded,dashed,filled"]; 9[label = "fastqc", color = "grey", fillcolor="lightgrey", fontcolor="grey", style="rounded,dashed,filled"]; - 10[label = "generate_rulegraph", background_color="black", fontcolor="grey", color = "grey", fillcolor="lightgrey", style="rounded,dashed,filled"]; 1 -> 0 8 -> 0 - 10 -> 0 2 -> 1 7 -> 1 3 -> 2 @@ -716,7 +735,6 @@ rule trim_fastq: ## {auto-animate=true} -- Rules are typically named and have input and/or output directives - Logfiles help with debugging and leave a "paper trail" ```{.python code-line-numbers="4,7"} @@ -732,8 +750,6 @@ rule trim_fastq: ## {auto-animate=true} -- Rules are typically named and have input and/or output directives -- Logfiles help with debugging and leave a "paper trail" - Params can be used to pass on settings ```{.python code-line-numbers="5-7,10"} @@ -752,9 +768,6 @@ rule trim_fastq: ## {auto-animate=true} -- Rules are typically named and have input and/or output directives -- Logfiles help with debugging and leave a "paper trail" -- Params can be used to pass on settings - The `threads` directive specify maximum number of threads for a rule - You can also define `resources` such as disk/memory requirements and runtime @@ -778,11 +791,6 @@ rule trim_fastq: ## {auto-animate=true} -- Rules are typically named and have input and/or output directives -- Logfiles help with debugging and leave a "paper trail" -- Params can be used to pass on settings -- The `threads` directive specify maximum number of threads for a rule -- You can also define `resources` such as disk/memory requirements and runtime - Rules can be executed in separate software environments using either the `conda` or `container` directive @@ -808,16 +816,10 @@ rule trim_fastq: ## {auto-animate=true} -- Rules are typically named and have input and/or output directives -- Logfiles help with debugging and leave a "paper trail" -- Params can be used to pass on settings -- The `threads` directive specify maximum number of threads for a rule -- You can also define `resources` such as disk/memory requirements and runtime - Rules can be executed in separate software environments using either the `conda` or `container` directive -`envs/seqtk.yaml` -```{.yaml} +```{.yaml filename="envs/seqtk.yaml"} name: seqtk channels: - bioconda @@ -829,47 +831,50 @@ dependencies: [https://snakemake.readthedocs.io/en/stable/snakefiles/rules.html](https://snakemake.readthedocs.io/en/stable/snakefiles/rules.html) -## Snakemake commandline +## Snakemake commandline {auto-animate=true} -::: {.fragment} - Generate the output of the first rule in Snakefile ```{.bash code-line-numbers=false} $ snakemake -s Snakefile ``` -::: -::: {.fragment} +## Snakemake commandline {auto-animate=true} + - Run the workflow in dry mode and print shell commands ```{.bash code-line-numbers=false} -$ snakemake -n -p +$ snakemake -s Snakefile -n -p ``` -::: -::: {.fragment} +## Snakemake commandline {auto-animate=true} + - Execute the workflow with 8 cores ```{.bash code-line-numbers=false} -$ snakemake --cores 8 +$ snakemake -s Snakefile -c 8 -p ``` -::: -::: {.fragment} +## Snakemake commandline {auto-animate=true} + - Specify a configuration file ```{.bash code-line-numbers=false} -$ snakemake --configfile config.yaml +$ snakemake --configfile config.yaml -s Snakefile -c 8 -p ``` -::: -::: {.fragment} +## Snakemake commandline {auto-animate=true} + - Run rules with specific conda environments or Docker/Apptainer containers ```{.bash code-line-numbers=false} -$ snakemake --software-deployment-method conda -$ snakemake --software-deployment-method apptainer +$ snakemake --configfile config.yaml -s Snakefile -c 8 -p \ + --software-deployment-method conda +``` + +```{.bash code-line-numbers=false} +$ snakemake --configfile config.yaml -s Snakefile -c 8 -p \ + --software-deployment-method apptainer ``` -::: # Questions? From f035c42470289bf8b78ce5156a969eb5c572be54 Mon Sep 17 00:00:00 2001 From: John Sundh Date: Wed, 20 Nov 2024 01:01:08 +0100 Subject: [PATCH 32/32] Update picture --- lectures/introduction/images/john-sundh.png | Bin 58706 -> 76524 bytes 1 file changed, 0 insertions(+), 0 deletions(-) diff --git a/lectures/introduction/images/john-sundh.png b/lectures/introduction/images/john-sundh.png index a9f2343ba1c7cc8c1c906c23ef5e089147be9450..5c86c6fb4d375791a5d616af566e1f94f69caf57 100644 GIT binary patch literal 76524 zcmbUI1who>^9KwswICtgwSXwnOM`%bbazWFOT*GFDN1)E9nv8nof6UwBApgWrwH$Y zdA;}d{-5VrT)wk&<})*AX3m^*Rt-^Bl)^$MLk9o=STfS$s_mTs zIH^jB0m_FdJ^%nn%1~_?GX(_zBYXf2fQEn#K!j6Q(sl?q*EAuVc1FMhAi}U;=(YzW3NW>OMI?A!uieh`>XfQwy#iyHvA9gzCdDx!Bl`Y-x%Fzs(I zgYSH&5&ZywVD{gh1#|rL2GK7X06>LXLT2aS1V7h(}65DqRd7XYBx74jY~b;Dj-#|Z$yz`uGU08*ck004M8mg?Hh+6wXl z5ST5iu_??1%Ia=wf2~!}T>wtnLY<94?zT2|P6F;iwAT^>aQcePMhm(Yakdts)mBgj zJ%BkvK|HMNtn9SH=pYbC(9zUPKvi7w2OR!Mh}Oc{*4iOqiRU4Gg;e#m>eKg1^7xevSVJPWZZ|gCr#3xL>OR<_P*Pwv5bgeSc5j z+J?9=7nqypw~&nZ-(r$B#!eOwV7B&fmyNILQyA{@kJWN?yQn?X(HU+})7ZuZD$K*j&(F!u#>vYA=HvZg{9i(UYmre?QnQCbO;zFHu`_r2 zBY^O7x$Y8bP=|kVe-7AK3iEKmy}erUaQe6AD}ih7KcPQd`rj4>uj3^NgBaT=!=nyW zg*w4(T&~ilVQ1+q%<)6P{}lM`!FNp$U^XyEHG5-t`hF(k57yt({|qJvn2qm;y?+k; z4ZAKec&>gdAYpd+uKLr_s}lQT=x?Z+i-{8yq7F~KlB1=$rJb>jFnr(s0sfD{zwy5V zq4PUb!WAmS(>{zeqS-a<$fmqpJ|12e-6N_ z0B-ZY>fsLwcpCo?3C{mh;{Scs{ok|uUNb*e>s3~S!E9g-?w<+y$H?!X{86;#-;%%M_S^se5!l}X ze=AXd!s82XpVxgKUcJ|S>>td(5!YSg`mkhg?C5kI6(QQIMeO1Tf&MyOU9T+Aj|y|K zgIw=mlEyYp(4Y17llfc!)k^!`M=h_)>L1eAoc|D2g~C$<-y45u{qgZX1g|IfGo;ra z{zFI=YAWIJeGmN8`XB7ypr7-&>Xd)%Ab%{X>zzRR&;G{4#|Lk3TtBzC-`(v`sXt6y zN8@*%fA!k`k>+3Hf2;UebYhmqPT$wj^0@&0@0Z>R+P zH1cbEfnPv>1np1u-;m$C$A7NXze6pdEg%Ln^-zOXv9ZmMmGb8`=ex!KlKRUqTtxZm zWC~xc_7;{9wX0)|v!$!0v&Z+d!d0;T|H%AXn<^B3JAj{#OyNcK<8&nh|KR=)b^lZ7 z-}`e{Qh2;^AI<7h6-O@ z`B~VxS-|}2Kfm~4z22YxlKUf)S0mpKyjMN%k57NeJaB}36f`|QIyuaoD>|wm{vbYBP$^0AeyEFbgBws`RV*ic$^T7Da%%9A^0Sd;}(C?SG zA3j|({v~z2vVL6V{_%y~Pt>)*A23sh0Q~H0;rw4W4^bQEKN{7K&G7r#@#=dE0r(xk zT!_}4#T060>|*0g%l1#{?-lk>C4y}KWl#{#gk0qs=J=P2D{KF((SNw{pBDcqe{J#K z$_0Pj&;_pQ>w5*6{;y*GxmEo)=tnre&-afnW3K1&oeTe}M;IMm>Oa3W5=I{eioFE@ zL;y14qU!Dl@+~^0WqPhl+?q?6j-SnGO_e$3%SNt=(>T8{S$*8qgQz1N{9JGo zfkvWv3JE5Si+Dp~-5(Vhkt!IE0xc2%L2a_pN~qW_E1t`2oqJS$mazmY^}_r*U0!`` zVEt(686q{{DM~Cu6qpxsBlr zgkdVBFLT~lNz|%_$oP`1t$;Sy(vp+HR7zZP^cj0gD?|HCs1g{&%akKE#FFFv%zC)@9Nh zK*^*A_XK+s4fobd852zy-|wx_W_+~4cWO%>q=d+Hm;CNo9`P6z6!qpHN7PgcMgtc zf&~+#$D~rw#^60~3T*28p;0~;6YmwIb}PQ^8AiZ-_w}mQ8!Ixr3C&AKVjC70-$tb8 zHuoL3t!Zz`5ewPrD%**8c zVz}iwZXEypCW3B@T0NC=u4rryOB(zOy~x{O$|y)*4pFp6twE&XBf0?YW-9Ex+9~&* zx5L`|4y8ohBy7>xFN)(C!4EM_CaJrK&Q#0dA?zj!o-M&TF}srx195C#-LVDB-r~;2 zV<|9Ilu2&*`Ivv$g^Rqo+(8-cM#M1Cid)jddd!B0bP-w9pxc7W-OzBOSn zrdg@1BLWc>4UueqVQpPbc&CzfSDy(H{e`AUCu5mfGS(e%no=&8L3fYj(;G8Y$e1|| zMHir856q{$%!@M3-iI8lY;5CvF9+u2y{SH%-LYb553u%9!Lrs7?JBV%kYnlzj#k$u zD1Yov&@<6o=Twq?3*%Oc_zF`3BZR3}d}ZG!NQ;pW66}BWHT05kuv)XC8!w-A@1dDZ z?h5;p+>&S8<3Y>1YD8744<#HTpGoQ~3(tCH4QKV^pOboPEN%A`Yg%N7RW zG@bpB<@h@F>yh&C2X9i|KvB%o40Pf_FP=1r4fBy>2GEN;qpNTEwkLyfgJ$H17wx*W zxvESEf(hE{a;2{yAG4a=Y>#L3A=cS9Q3q63omJ{qR(e+`yNVcdysP%py`fwP&d>>| zEhOw{mo10klA7R$KaXD46zd@lEP2yC3IiXBW|PkZ8WSiFPni(TynoTCW(zQTorn^p zi~2~6HOMEZ7Li2dVz2KI1%Im-7^+uNqtXg}W? zjdAbtJszDIO@CD~hzBZth*o%+S6QB(&zgn@gbfxN-dE2?fUB$dnbgS+3=v zh`($}pLXRNaA>Y0$Moq;s9}$zUb?NYRxr=DEtfJH z8!8*RIFkd>xaVw&YnvZ$GkW`Irsp=3sn8v{!Me-Bgttj^w*pVr-%(37e|4R2bRnNy zgjVeuc^mG{#_d;&*WSol<&(HCgvGWW!Le$a$cvlA>yC|C+-xOVAbt&K+F@n{#ySvgzX$AN=@c_IFY4J!F2h ztH&sr6$p~}<@cW*D7%Otw-oNB_6t3LZI0TD?R! z>*S2Eg3d*oNR&O^6e;N75d4}K3W5On>sv!shUn2r9X`FMs;J0$LOZG4DO=XJ%}qoG zb;f1yrG4U0k;?}itba&v0PL1UY3Yz2+vP_tGTC3GrOKx%9xMh^8}w2@)-^#AcgfN{ zC#|`06s-*k{2AH__cdwS_ZjF81lDwDCY{UCoMzufk`vfH5sBei3oe9-puAPUDSOJa zg{qG!ZYx1)x@2WzB3@>(2Yd7qzokOir)LWB1A6!a&%L7-`t9!Im|l`f6377-QA3!h zjd2(EIxi(S3&i4r4u=s^bnXW_hj__u<8*{14-u+(*M>iT zRe4$-elqw1)6+=^tIwsxczuBAS;QeTpPPOgS*1wdfjZ(0qO-zoZnzwQPv@geCTCM& z(b>Gf{o3;+?gLY*+NoQ`<#HlS2*gAAQCjIs;kop0*SHhOFxLY8U(-el_0WIZKi;(< zAuN=_Z}{L+;y&gW(an^@zzuW(8aDcKaEX-k9BEqlpth1$yH3`(PoX7sEw5|h+ZE(g zO*8Xg%=v9Qrql8of*FFczUqn5^fV_W;e8XQKV|X_?Zd2M4I8tw?LyWni7Mkp56NKq z(zITSU@?KJ;k?EOK@cULGKnujqdh^<6Rl*WSRz?L&rMe7Rl&ziJ!cK_mm?Gh5EVt@ z=ugUA0tvbU{qinaI*;-4sb59B3vWZ{wW~sd7D_KZJyuYTXSDONh7l|BKiP%^E8k_m zk!s)Lt?58)%F+$A&=Os)*TbyzlP8q3^k3yxVK}!Sg(e@`&y9tDp89NIGrd;2o9tCx z4@p4EHx`!hh;|{gO#vT+3)xotDl~>@s(J|*t%tGnm2e$<8jWVrP_cvjJtz&2OMHdU zNVUrrmUTI&ouA0ZSKtJBg0~x%#_wDt?;@d>trEplOY0$UG3xvD@e!R1ZjgU9zi~>+ zUy`BG%qJcTwr4Xa0k{S~J!MgiGNqCru;4c>r;Y7doCuql_V0m-iW_a08L?o(+W7VoG zLySp)1J(r!MfL!~;&3A$cQAV>gen7KP6$^-?`&>Z_?bE(Hj{#Gi%p{eytl)R9z52f zZ4S>700#;RB4KEYg-B2ZR&1aT7ik$$`3Y{PbnKuV4Q4Psvv8cChcR-G(C(4sE{lK$9JEDSxrpc&ij!umv$T`>rZ@uBJ}+w9WO1$j)MPRgjws;D`}<<5S!WwL?d&IMVP zQDl@<2&Eh_4r%QPP@a*6JyBXO^_9a7rcDM>+@K@~8QO^x%7DRclcYJ9VaMkkRH8|n z-ekGeBYQB@;`VX~vU`MtQnrKK0x0ZWH#vce8k*!q(H=XY{s|P5M9=G~c51NhX@_t_ zFD;O{0KI}{2uBv1K6%$oNDAMr;?9RJRJTbmv&7|SEr57Ul zFE7KroX0%e$q%^CRK*3e(Yee&C%F?Qi;}$!<<{vqRXaj+D0)#67C^vJvU2pvU7=b8 z1%QSkR_-Y|9mahZTtQ15$7UT&FOri@vwAlPdABuVmzPlwS)8(MSN;*;5{AaZ%!%YA zh_sqP^GUZ<%v#~$hnwm-h)WVcRc$WZ+pMwbuO;2?S$###N&#C?*NkH2?4XG{4%z2+ zwTD4(iym0#u~L(+6%;T?B-H3yBDFU6BfVp}Pb(E8=IMYbfG!m(7%$Uh&eS-v|l4IE?Lf0k3r*zPB? zJ-!_ADGf;R$~=90-HcRWem@>$)3uZ|)7W`)&rwu!R@P?=_9*Ll^UT2x>mV2XoHfinK9t!WhH{AN-9sT4!MU&! zwScCH`bI&MmS86BFb*fh9SF^{CO}odmqB(`=)8ExR098Aiv1(eLgXEm<-xcydV*fU z<=dsy)Y2el7IT#F}fjqc}5R z4&MVb`+NZ)sKXFdv{5MjX3|>_S~BMclrOvwQ7t&zyGw#mZd>S<-gLp`)-ZxSwl zA2xEu8Uv9C|A|GzOI+J|kK8w*eadbq#rsQm8;10U{z=#=EZ)Pf8+s7eGDH%ooHkon z%28tzNbkS)%Q3*;>WoV<^*@l00Y@q{(eUEhic&dCREB4r+^c%MiCVQgPzNdTy5nm{M<#FioKgWCe^x1;b22#HL=6ZZ*2`E*wn3A2B$;WS^K=}`8WAZ7Q&^O4m+ z-X)!P`Ma^3PAX1|fvP#oPNr3p_PCBHrvg5Qd49X<%9R|}t4XZhXe^DbK7=eLNwIbr zRTa_|?plX(k6djG=A26*?Br@WoV}PDr|y$@qZ=J#lpo3Z88HG< z0)#_=+N!$uh?V^{c7Tay>i0H%I2{WLW|5J{oO*J~+!Y2ms>j8v`WgHL2XiD4=oyF| ztixZAyvCnk@izqqIcm14)+;7^FxxAnF~wfyb?Ao|q4}q=IIYna)^~CfI>f0FBv)cQ zrtg;OL3}=>uMP0AQ$Jd+yoVV-h+f#z)wF{+BU1OI-h&t@`l*32Z&A_|$@v>oK-tic zHNhqAOOc0x6=Dq`j#;HsI`hkSd2THCBZDO3AB}Bpi?51hO89l7&jjFR+mJ8SViF!cyuPQEwHK&k+xsGeRGIde0E} z)!OuhHeNgp=(7B&LE+B4UNN79XuisVZqkUFpnup5&&vivPY0KdZ&@7g<Teb0trj%L>bt`5|IS=tZnq>_yZYk$X;gL1>G z1(O&>L6qktF9nOv?Sr)CSg7mgn;nC;i%&76-q1>`5-|4~$tlt1oYdYl5*OZ!HI=>1 z9D6!ONH`#1A^0(3b!<(GRH)6tEeR$3Brfxux%CobyLXrSE_dPB=CJ18IAXu!#VK|_9Qs1D5TwbkAnF6aiE4Fi}EIjo&`*8_Y z{ieo?2i&G5f>vEQ880ju||S((>2pME`YD&ug*@_tPqzrrJaATt*s$)I=$GS5ZN zJJeTle&VOvG2FazfR`#!pIH#Ue((`|bVR#`1vUY4C(9ZGvUqfg@=}P}2icsK@T{w{ z8hQMu8(NulHXTrUfd^_r+T{y9Z;`U_=}cncFg~I)mW5fRT4Q4C&@!kA@R=b?-*iRF zvz>Ps1UK>rqP7JKSXN`ncIGI(YR(Dq4WM*-#!34*ts5(_7{}OE)dXZzW{B z&$fKT_uf-f_cDY@e=*44$rUgthqjW>vc1pW9WV$(ajuQrNE={x)zw*jXOP&c07^EPdKn6e0c?;JGJO+< z5Pv2Zjm0?4@6ofZ>Yw5wVIZ|}S-ogE&UKrPAifi|ity#b`yFAyVuGajqzVHe8G^Fx z$0&QR5+7NUH)-nj$%X{mYhJv>+UAtHXV!7{$b%aJ6=~#svXKJgs9jxdp9={|Xuvp_ zh4>YEYUc~~@oaPd8UVjyD%Td%vXkn~{Xo`$V;Jm9XBcDs6zx&NnrIC_Q|65l)Uf)-+Q#qaByEaoyr)Zpnc8(K4j*p!kvZHyxgWM;o35A7IWt%j; zcnO7KQC=Eg`qnV}jZJakn~pS3CF0tYv_(vn^<$TSI+~`UMn?qxWMdDihBLDCpYF5^ z>`TRLf{Y34^pc~UYDYWH7!vfE*6Wx@4_?i)FpYr6`AKP?$m1f*ig$0f)cM@Kh-dAFp>sCkJOv`j0wcwpsdN5@D{#6oh=NeZIs};51v91#< zZtNc9Kz z=AU-R5vmkhZ7#2zF>Hrdr2!=ZcU;}Y;}6!9dO z(Ge$Y;`(()Q)vT%A64E7K%;B1@hSj_`&;dIK7~=`VB;gcK+JfoF+>(1#3>zQ#empe zLWr+DrZGXq`Xt3qmuJzYT}7J!Imd)N{?Ou(Q!XpO1+6P}@>$!--aGrkIKH>OoaE-{ z^_D2XN9M!YJ|#^E^((V^Bg%YjHrAY(&u5{h$swuQBku{5EVwOt7!U0qxIUOF+9}29 zLaYmCh>IJ-G*X@*q}| zctci{irKq6MH~BQ2=g$l#%=^grgi{N-`t>dD*vQOUJ>|-Nl-Y1V{J~%xRJ1a>&V!~ z%_6s}yG@44tjblaT7jnAp|^qrih`0nt{Wl}!!jngn1%Q@hr5W+h$L8I5RqlTo2y9Z z^yA_*?<()*Q5$u_n8f)Q^(FfgeVDU{Onw=0)4fs_16dR%hezz0)|1MUI?aL}6kk#6 zav#nPJv%Q6TRS-^xgYkbU1 ztxPLj3_igt3d4&aY-fw3bqiC)VuOj0xuYXpT1F{WfO+7;lSCpp z?Nn1KDr?r6Hgjr((5&qD?(5Z!7qsQ$2ccXLnJxyl9KOUBV#vwNU8uqfrCCG$2llrI zSe(&5&Nba@*in0ZNhs=9zSyuu>AZS$Q&t*}wjf?#^pKFEg#{5bhC-=%_LdA-r<8B| z)oN%7!_^Y2V)Y%@%6w(LIi{&Yjp(H)DgAA>NRy8rHzw&q;kSqo+X%;clr`B@+SDDV zw5mm%Tb8DWW^}`P>~){*2DeAgJD03qKgT-LD$nS?_)zl3S2r)ogOf+)<88tgTMj(E zhvQN{L@5S=s}RH`JSuZh%?I1b)`OnYdVt`|Z1J1|1dq$mp2R)T^B%TuvfdbwZ(rN? zeW7>tbkbb3k5)1no_+%J4gd<%!I7wjkHjS~9+cc4IG)|~Fh5Rvo$eB(XtB-fUPty? zZv)mtL+;uXbEXp%@mR?tsfmZe&F)+mo?&k}w& zpqfg~)YwS3fS2r>d7n7SghU=~DJ7fAD5KbzCwWAKI%z~;Egj*^0BQ2mae&{4D$YpcQz}t zq?JwP$q5+E2#6Wo>xrBy4vDKCa2e2z2zYwSx3zmvY)6kS6?b>1FVveDbMGTi8QVK5 zE_*6stZ9Vz)eo-tL5Z5hUzI=t-y4@OpsaD+w(y8W*VUlC zmyi08W3$|P>osF1M%2+wcFfzG$?lo`2CrV zg@~wbElTBDek@KJrCq@nd9vjDF0=jN+-^(Tj$A{enW_5n1aXx1t0aaG#?G#m+4?QG zBZjTPWr6ASPVY27`Yo!A&MwT+J>m|5#6POWNa8i1iFV@@Nx+=IjB=Hcx5pAHeo=5F zY95DU(78M+SsW{Y7hVu7<$SYoAXJqM{$G!#eOM<(;a?`+g-H;Pq=sF!$8OnM{t-*!-nhA1c2%R)VQg*B!sI#1Dgq{G|-F3 zs;Ox&k;+7@%$lmYw<|bysy(`#7iDQ;`r|UTDi1ezVz&V3U$;)H-Nn_$&kX&R`{FP=J5T}je8$rAtG3VxBb5PnsTXRS5ggzQunsZG2k&Rc6iF|6 zUQ_Tu^vP>ECXr6r)}ID-;^2Zf6>G-Q3guf{H6496-5@9heTRfR+jowcs2}2pf@ zL&MX$OEb;KK~*xnOACX2pK*l+lRe6b13mUUpyy->8;=WUJ?>2Ckk{i-KqAWMrC_Vb|=2hPN!0YoOUPV7fggV4yXbmH!lr1F_7EI6;IctA%iks6DRf zUUZm2kh^*D)g5PLf_$VM`Vfu zc#yGhcr^T3FmILD;vn*|nWzXAFmZ?%vs`>{{uamVLu858s34t;^lGMr^D_Npe)W^I z%Ma)-^X@t_2caauqX9OoFqJ(!R*9Y zmjQlX@Z#<8MOs|KY;L67aEhw>DE=wDvrntqeRtyBZlzGf^R{!zXyvd9;d2@N``+OW z%DO1C1*Tfl{qNj0w7!g-OkbXHhGOQtjk96p`V`X)^JjWs!hiozpEVLCyQUc}hiJu< zHC3|xEMF_G?T+Mroj9yKTK}!r4ljau*kQlDA49Ay30pbjyss^MBC~4d^Ke~`*MTJF z(wi`Hvel81k+t-w3K1VlnCLwx?@d(zJ$*+eJPY?>Gu{_=Bk=}gEN{*k8g$tWhG=aNb-U3Yi^)z zb;htT%FJ!vvXF~^q9D5Y1j)4aNWY)Z(6ktl%_^p=;9@5Kh@Zvy!POSQrn<#49FPVrMJ_IY$?^hiqon_k?ueC6iZ*z`5NXsr$Z&#)m zXha$CZe?Yom#sl~B%3qO!j3}b@Zv*SO>~oStS^CBa)6|JELjieg<88+;Zh zmD2DIZ(GApNsfN=d@wBqqi5J$^)_?j=f|u8g(W<{(sq_xSu3?0MPpzNKNn^)${sYI(0^MU(>ExVgz;;Ofw~HK-l! z#In5*+A%Qkz8%Wj*Qjse%UpWEG7`M- z7ua_A&I!OF9j7{7ZnUF^aR^1-9C{Z*x3dS$fYMguuLRT&LI)>J1+wmDfHGTC74~NJ z=9)xJMmUAqOV}Bv4u^s}qh`;V}d>TNyqO!{#!b zvZt&!Ww}50$$F-b{M*AxhFp0wRCkOI6)ws(?R(Lz`mg)(HD28A=XY774^jqhCpEEI zp_K&_vQ)MrzBe>o<$I&A?GQ;%Yi)!dEsMZ*Y`7hgT~E3bpgtwB&G-Z)ozBzr2uXZC z^vRoi92I7v6;4zYi1AvrsG#v~>)A%Gd!tV1oyjf4W#l-x5l&8S6z&nB(<=#;+M%@+WJ4Y7N9YrtvF}cTcc0lA0uxFvGY(d z(Aw27x`oYHt$jXVJN^~??R*fC)FJGgzxS1wc5`y3kjd9+OZ_kNArbeEO?*CqhVOge zZeMQEy26&iUgc5>5^k99t&=M^)7~>`Y$`kHSqMD&jD0^OuuSs&X!{%ME2{3Vum!qN zI~?Tk(>giLz?1yQL2KnG#{OFoIgJ~jYOBpwf!LwhlP|+oYQ|egVL%-m-aE-inWy~4 zaYWREh*Mb@56HtGMYR=Oq{0u%QQb#OYem3a5z<_yZwr?@ z+ZXK%SYLEh?fsaS(9*)*QoS>=d^mms+;hm;&*e4;5OO)BD{a5wV2;-d0t*(3>E*?` zWpiN?_lZd%AtdABJ(AaL8kiX&eSOcW*@02{%lSsDok~$M-DA!K2h!-9%CWq=9$}%4 z>o*W5xNTod9JSl-Bil~VChF08EM967v_0zz9BB)#Xmw(un%vy&!@dOd5Ld9r;|dY+ z76s3zv+0xeuN--DGAYV<8WH(b4r#?i^gp59cZ%SOkKpII(?H^~zOv2jO0K&O|NqmP zm7*;uwSCnEDEe z&B?T72%d$a&&dskyvcX3?0z=$4dr8iLzoUFh3Mi;RYPu%d;;`*a2Q0_Q1tc76Xaor|#4-F<9z4ijOkflE|Q?l);m~D0qcv}xAMs!agAQXH>XJJpc2wXZV zqaFnHAk8=6ZsQp@Y`k$5Kcx_eWgWQb{uOv@2+#9_q-zX~DM|`Ctpm9R^YFrpi!V9w zQ>9NuR!yRG?HhGn*3y-`JLmpOWwJ6zgVdXe*gmX&%Vs3wvJXXCUu?3n(liT%A+Y&R z+xKIl`!qGhVQmbY##!i7N_@z*^xj>bI5|(NTci1qQ{l7sD!H=#NaFR{5GdH4wp`X) zMk9#(d9rMtLxTnX?ac^GOpfyzhmfcgQe#%&^9k@++cVM%4yjT6fQ%Tu4w)cQJ?529 z@10ov{42|qYlh1%`Zd=#=fM2k^qG3~ZOjR8mu614<(u-Erx%qa9Gj1d?EC6(ePnwP znNYVa6@_$f*!MuEl3FnG%RJ5PjGoKMFf>C?=DCwO=50<`&{}eRxLRc2;qjrDPs2sA&-7 z_-^ixZPkhsT7WX4s;MT;Lk?o@&wAMzJKH z6b{#SI!qZpd(6UKC4?hr8Lj7#adCCTvRkrGW0g$2H%TP;35oX-4Z6gDA3Uvjk+VTe zPQ>79Ke1@O{9>NKChQMbBrj6NNQ}5z`$KUB((Y1@#EjB;WRf~OZXr|q0>f=Dp7e9v zabFHNiJ*a!8nB_G*;nB0O-5H+^))iw3(ASTxV$WwSDp2UGGwerm??F`EJGZ-8MW$} z=D3%eG)6VJY6N@Q1KaD2_Z=dTUiaDC;zu2+Y^|RoMxKN8sBYBqEEw=|wP<}%uEsMj zmd{r+n?JZ*J6LbNZ9jHfBD7~1_eIxxFj|W{5?yK>MG{7Ze3fVuBO0t%#f1pHOwfu6RZ2|`AL!tDkbf5a z#qDpg}epG~h#8sg;;GH?M9(D0X*Q&aEpw*|z84Tk)$*k=J zaWljT8m`2W#T>bjpF3_f)M4h@@1h?NPg(@Csma#f+FE1+Jcd0+Zk&6+n3bKo&$qgz zMBVd_;fqs$qM^^pgVZql=V*#PU*8Hfy(h~(?nl2FfInNd)wjFFO@xS_FVl_X^-g>= zel#U!Xo+f61yv!H7M=A?WD!|>*3CQC?9|xIMZ3q&fz7?Qed#vHoF8YsyjwF#2`&U(9Ae-i6QZU4X|<(fM{0C2ij3cIAVRo zLoT}qq%Yh9chn0Aa(bR1ax1K5jRFbi!*4{(^0Rw~N5CGBjbg~mo~F<3a}prWPKr_! z&TB3sVsJw!LZ(f9Z-cz;239g?tBTrY7;Mfs)TE2r5vwzBtch+%w}{@E59&nF5OZ^F z^S#e1?1593rZexkMm41|Y_;|*@@dxPcmVd3+GUvI2^GV)7j7wONLI*l?+9cP34C6< zy0Y83@gX61AyJ`f=HQn&}zZuVOtDJoQ0V+drjy_q1pt}twz;V3+ z`SrwpgL8_Rmbe%Ot44Gvuv*+e&BcuHym@9w9+Tm{nV6-Yf0%hqimE)@yKP$THm&H~ zjDhjGnf2JaoFxfGgY}KzhU2@aL>$Uocqm#&H-OB1m&G;Z5;??eatI042J*1#ypQ?j z`q66XV;0UHo)i)LH(HvH*HV2}?q6)ZI~X$|KsNY9Vrt6(d8_C}a<&2tPuXquJS6GP zt5L*##r?7w_N4*6Fss34#N)=!8(2UoiBhb%Uxaoftqe9)YDUm9{tW|DhBwunBs6Oq zV=KIg-at~V%E4c;2A__4i32*$c62g>3o3;OrzE8k_dHPJU|sm z#@MG{NTK8W>xm?FtM%;#sp@Dc!H+tDrn(1!z@1V=zepK&8D*2#OSji~jBVrH+Ts|)uSjCox4-{H>lX@52lj?}&+^0!UsOV}sl{n}{a-h3R_j}9j z<${83syLSJz;35*EUD7IPlEoMTj`LXLENyu_|ydy^eLMQ`c8TxPCte@3OdEXaCR_( zoqG&K|2pgw@6(f5{>io3>m(j`pTgIb~Y(@~Bbx^C5Liy7$z6P=`jGMfcQ#|~T zY=TVR16<3RZzq0|N5j*i1ll7smSY22O0$;xw>GS+R&g%o7T@MCj9)Ao*i6=u@Nyu> zAJ}NRJtZNu9R&akqCswm!O6y?vmrgHjW0DHZFECi&FVX9Wnk7%1sy*ueRcEMM}0}w zIOinBzH*m4rDT0b%sGutX5&#taUo@j4vxei#p^cAL2;-X5L@506!_JQMjl2&8aO?# zZ*W`7dUSXOeD)Zx9!nbjk9?XKnLZdWsO2>i9zX>3=U#o2_p6ghtgF0`56$)7jKT=l zaONo6NQ|rFi>NwWW%N0py|X*@&{c0V@tudsIOTC|kuqbBfK6LhJP1AAsOp@KW7(h( z;Dlm{>!{1y|0#{akn&dalI{?`=)#eDZDdotb6@otB7Ljb!szwPq1_ zb~Hc4sLNq9P9aUBsp`my-0~3Qc!@GyW9y%sU2=g?j`)~LYT3E-`aVU2`smM=10vJ zLnUQX6&HvaaND4!ATQk?x<`h=gmWvvkb!;D@ABLuBu0BeDp4QRywlvqE>r$Zi|j}c z{|8+B#@oA#w^-~mDN&U!cJAyx#~K%6=zlzMhlRRLQ=dQXP9Hnw<19ve9RDL8cwZRh zmn?NCUr1cAC8GG46NFyYNvocqC9u}E_l8weJgz^k&&?}aC_ys+C1W0v^gLDxiCW4MU$;?qiS3v2-~J#S}>{)KX9>w&%o#7!z`N&UERx z`(-;02CcEtw#>h|eOFC`j$wMVN_jAt&9JAbZreW7@jNrrw;w1wjk2BVEm+3DNm|;$ zKXe}pUF<8eIYiY2m`-`-XbX5j?e1lb>fxjn?}71<0n~(NKs*zF&vZT_Zt{sgE1+)c z9Z+Q20SyqGkrFD0-s3gsnu23LfpCidMW!Ht-$R?XvyxU$7FjDinUgyZIo-0c0TZ*v zzSQEwDF!14anQ)VF)*aoW3TR9k#VIe9Sb-ZUd3^7#66^q@8y0=w8OpMqn3G-71cui zhP=4u!=mDHFYbvX~K~%99 z4IbF3CQHm=wZl;#%XJJ!T@V|1>a1cqrtilsQBmvS9^ypI#IJ4S=vui3|p9ILE>h(5yjp!b{`M@o}mX~AKI8uysS`XapQm4ma3EE<9O#idMn6QoFey&D~tjK8-mJb;p#_0%(Wl%ZV%`L8I!4clsG_Xxd=zK&JaEx6%BcrS_fLgL<@GfW z7R+4@zJ}tP!7*Q&Id8_H@k+_Oe+GSkgWk>njz6d3TM;Xe;8ziO8(AJ-5h-IUXazKH^XG-t1@IW~jPuo^tZz&6#aTA~ZN=@;6k*Qk)U zm84iZ%`ZM+{NrErik=a1NioTAHu^mWM1wTN{jKFzQPSi*bq09+jJS)oA zvY!EgzOL>xbqOHL&5&&|r#v5lBz1M0#%L7Ljw8#)9`q(xtB{vf&Rj?@LZF5w@xFSC z{r@1aN1o0JMpU+RQ@mMuX96B6+g51qIW>^&_4VD|I3hK3i(udx2vP=_9+FwO^x+xY zz(yj2AJ&M8>P>LF^d6Kzr!fL_OidNY#5w|0fn1~_imYy(w(6roi%gEyjYP;^no*g>p{Ep7M|jZe4jmKXNEu~C zMEo|D=-j>bkolH)Usg5w-SzRHkrDwFcuVr0>4`X~`H1%o$;G#Ph6MANGxrhiBRRz# zp$%Ny>+*@5htHKTJMfq+(VeHy;$3abSvtlHP@Rz< zK0La`Q3AVE()N)#bdL0%&%m(1!7Ae==a}u*?$R$oZl3Gvjh5&U`kD9kXs)C{EU!G@ z7E05rb9)6y(dk7!cC5l#cx|X7{y$Ic$qkrU?);j=B0})XY}v9$(tZW+Y;)6BG$uc$(|wzYif-2htDs0Cle>3U9AaVHWn+``a>*5Y&g_`I zU`2u|EvbU#m#b@~6WNQTWEv)_5!j6F=bi%Yw)7&EtTunE5x{M;m#U6|l}`t8GNCfW z#zv!Zwbq52)T@l8Dk$fiOh?m36g-4yX=K#Sj)@BA3s;zj{N7&K z%m*`W*Tt{brSZ#W52Ntzcj6H{slUthfAQ8k@dpn-hL?LLgfxc0n0aKziP(ZXK|Fd+ zaCQke)cz=1d+qr4?g6x>MR^=`XFdj_go*T8lN~xc%Pdxt-V?V4l>mj=jO#Gg3AuG2 z(Bk#ea2*?Cl=883?8@OF?@iVXk8(q1;+){lhu>7`Um-?NIT5e|<#QG*&Pb`;LgCt0 zQM>_GQyz0-002M$Nkl&LhTzA}vH+c$*83bW=G@&6paKhBS5SW;d9-#sz3* zpjl(S=q~plE5aW=1f#AJlP)qXWkCjA%H#Uw`5wt%V!=e%Jb6f@;qd5GRu-t>VI5Xx za+JDYra_Wl<*|bX(M%+&oCSzrz72{bhW> zj*9>3r+*ONr06s?L!k*_D305doL@fjp-&0utF^po9Wi*=i1nyn-FNr3}QCt%wF~|S9J~Lcz3m!DV zExIMgtwL?yYer!Sgt@8HgW$%6SGX`AkF}L~zyjS$>%h=msN`xrIUK>(aAS4SEInLc zZ*rW~wKl=w93?M_NGP5*Z*qv)sfg5ES$7!iib4|&~07%bZ?Ll!7_-v+|4q1iGm z7@NxEmZuho-GVx-kMlIbUZb;as!cUfBa~W54}7hHkr*6_7}6ZM=!&u{W`XZRi9pK? z&Oihg#%}|@3Wmb~b9|)|Tgu^sSsEoEcE_}cYMO9d!eBY0Sg%$9T|K}sMGhboK%p|d z8cC@@qXEi}HZou4j8dpTrY=?H9-Owrx_StVn-}H$BB?>4ND-6U`o|F_?qtKn_jM1i`?4mG`lV&xwjncU%_hGt$ID-ly19P&lw>gxDSw|EZ6aKEHkdoLwAh9w zlnIQARBC$E+HLkWQ{r}gcM+dLD?H6nBXIHxH~WPVkY^@zaAj^RGzDXt$Ad1MPlG0m zF4u37bG5syM%s(DPXkn#<`kt2MUda9xd;!lr=r1Uqz>+p!@*@CLq*oBNX1la4ci4O zDj$A#P|CrB{>IlJsh*Dtl_%=BP|8$9E|$osa8}t=CFb*`0u?gWsm>e=Ugx-<;Lz}UXDpIkz=jnG5I+1l#Xc#P%-Y~+$`Rp+dSj!EM9SD+qa>8 z-@E;7wn;t0`>c;GK(+E~1z1h>9x0(EhS4VLa26kb^5gi{w?2rwTl>5w*JjK!vk!JF zI(`hAxMe}9-JKBy40**mzcwIgn0bBQ+r-P(2!@q|Yl+Xs12RvjLIrWC(pmY{9rYnAKT40np=^ns}G( z4USu)YpRBFnppSv&ULPrAkJmx8a@_Fert}~(`2&zYl)I?@yMorgvMeJgr?!3q?s!< z$U#{#0-;6aMl?B>7GB4)sf8Mg41T$I1Och+8o*eJ22h~pfB|x)=fNxmH|5L7-+YZo z3Rs}{>)i&>K-SKSFPP(qad6MS|ZOsXe$4vyl{g9mZJ=kFlrvVpb8&YD*_aB)Om#T>Nv;O>eU zCuCm7(0bBAl)8!WIv!jKdWC!dpegIVR&J!r+7h*-0^9iH>LEOLl)}OZ%cy4cn6Y*QbGSks1+ub<4hG%#X{;iw5SX+-> zC7R0awSqFkm;@NtZAwN(u(YCPBfF(x^TaE%6J-E~lqAh1lqY>)o36i#b=B-Sg|Ez9 zplr_Oq;C{io@BvdJGmYQ((xWkb}E2844WaPMI=^F+%hD^T?$9yqc6p!kMIm<#oG0o@O@)1!eX(|DKP&Qp_f{RDAUU2NYj^ z@gTnUJ@UrL&u%h1(WEYIPHHa^t4zt$=LTGidc;{66%e&Tfydx_#)&`bM+fd+LJ#og z9K4Ml66hzjL+-|9*+qv_S*UNG%x5(1K^{FK{~opJG3Qn__PQM7kHWEddF&2vZlSd9 z^lid}6zGz(XHBE6h>&&ZT8tg@D!5Kk#apt9U5+G@I=H(wT{-U6;BX1T*^us0mPV4? z9k)y#b%RqG;n!YUq)%%{d2o)&Ash4x`#d-wGrL-Gpj+{T_RKP8W?gYt00T~7a{U71 z$|Fr_|d6BFnj(G!NOjC~s|{_-*FYB!bNc5K_zcP;3JUivjC`0fvP;MHl!N z6+OG-9`` z$Wx=mtY<-n&$l@uU{2}WY>$FaerwNya)V+9rlub?q%n%(!x_cL9?CqXRO@4+l$+Ji z8P{Pu=Gw%#KLf*^i}(x{N@q~z({^T1C^HnElgzRw`r2^sv7~m!*^A=X^Z?4&13|X; z?#4$n``^0vX8h`3{cC)NAd&0pot6fZq*^Lf#zg;vd^n|ntPH%}kA?-9yW<~r#Gu4J zl6HmBn}@EYQKL}|MuV|kfr<0@R!#w2Ebb5-u&*C2pU@2LPK5y}nWpZE31-E|+%WUh6MKyQn0c;j zDTL1(oOu#ao`A#J%2{6@spUlsyyo?Fbu_at{IN`0N+h-?cR$an=ZQqTH&j%%7@!AV!5K8X0XH2S;<4H4YJ(6-`# z{73&VZtZvDul>@$%35jIfEf2hNC^aGD1iORdA=EA&=9iVgV8RLx2wHGhdj>IQ+TrK z$=`>$1s+fm(D1f1^t3#r1E|cv?RMJPVxQ9HAvI`9x_nCu)O4RqX7xzYk9SGen$;fj zwVaS$r0#63K6x(aaSgXgq!58nj2TKg=5HO`P!D9dy0+&NbT_$Q21OA{31#=-=@Hv^ zN(k`k`ZUu$8#H0qnPvsb2IV;c$2eEoXNs+2ivZ>c4;XnyPIa5@-6NjXIpUz)ld}uP z&SpXsvb({Ph?v7xc6UJ^2r65q0WmiDT8op~+swsu*qV0W&de^bNCuQ}l0-}c)T9<5 zfIL^lwIGmCriSvR;@I2H+Ol3Osy}@FmAxCHjVleJ*+G@hL(adU1k9(->Ec+?IzZ&} z%2dVlL`IFF$n)uZaJ1JoAnu^hB05x;*X%K&c?IQblGkmL8k>NoHu;mjWxl))O6CRv zm6kLj@W4P(7!+KL8lha=!#^@3fXCog7%fJ`RCjmp z#_#>!{}J!K{k^!ygJ5!}HSvi%$$NO@1}N;lLyuQ**SM)ijbUmmMDEy=bjr+Cbk1YQ zxTZ$h!P7ZL1ML=`)Mw0#aGGe)JMcR)Mp}fct9M8tjVm1FZ5G0sER? z8K@wkJa!XfPGy2;t5`0a{Lcaa(@O--52?E-M=m;V@-r%4x_(wd1TUe4qMzt(@39=R zShN=aC9RMpfM$`OO4KaTY*PWHR+TWi>u) zvy|@M-QRg1$Ksf{za`l%D2!(-JRhRGyvHVe3A}RYN^z7GUS*Rimk-VHl%)~JZk=(O z5qHU56KNgjfbyx#{p~8ArSn~#JZY7Pd2whV_ z?Q0&b4Q$RvOk)IPgwbSC2mLZ{zfEs(*{3IN?l$uP;SEw(L$bpN0zSmS^T zv`nz!(p6wi1LcAU43{zG^%>7>)4dwJ8C2+wi&zRP;Gw7^FLytI#Bt z7Nk)r)&kURIye>8vt=D;R;;+DIbVQ|1(wGdgjp?u9+{iw`lk36WG8ID@aaLU8zhL! zsT6#WX;aW){-UjJ5k*1m)qP~)+LM(;{&}0fhG4hvev_LhYDa!H1K+xdzNwaAdT|J_FTN+}op<6SoP*axN|V=-Jah z0|DiJM*KEunJbhL0L%+fm0h9+oS4C&4VMc~C3JL`T+CmXl zU67jvclQJ4m5bsN!Ivn!51=l~@o1qFnJhWGng%=>%yC}qif!d(7T2`bA(;l6L}S*c zXO~jx%QKok?(fC}iWS>bjP?#Usi`%PCoZ| zeIL{ZCEg|g-(r`i#6G1i@}&xk%HKCWk9IxJ@GvcG9-xxE9P%d3dS*3VF4TbKBv;NR`=d`4@KL^ zNlZbqo3m3rC5okcTbFmOK2P9M<{RB|ehmdM&$ht{$QC!MMDkV8r3{~?%8-J0r$Z0U zY-G5N0{H+rBS$%nNM%dnf(p`vmdPdMWIK_jHkhnHY!t<9&z>)!B6~XUjQn$jbGNGG zf^FUGY*MOTUBoj4eQ`}D!ax$)dT2-snCfzpqLn+Fp(_=rO}c1t4#eb*s+|uBv30Rd zo;P<$YyI*({!4{GSe_7G!Dvlml|c*hq`6B(bd**ITCxg$%wW#NW{W!?;9L9SUI`iW zAgGiC_5~DQ1@>KeYB)5&w>vHybF#yV>#0;4udn$KLlPFV{be5xo-d2OLh)(wl49A z>n~L*K^uH9AlQ!_@zS^6h@GQb^l}sPP(f=_*7k7<(ijU4ce&Se{AOIvcrG&Cg)8J% z=3w#_NScT5F+Sy8#IQVqRb9)?w#>z6Fl$&=R%@VnKEjpYR*(>MfbX2s3_)@LV}a5t z&$J?F8{+;g1~BTh5M7T5;sWSuTSV<9xUFsx$$(De0kKNYdmSMyLJ81{d@HKqwQewm z4YCbBdNm)Mjqp89Mtb4HPJkZ%wa1qMXTI6vf#hrmO<}{o>`v4DX1S5i*bK7d-kbaI zOkE<#Hipaj@D)=-?ndaqwY#L|SjPO_E!Wuxs=~-8SqJl#-qUnEJ_Uz&&dKbk zaCuEo#~^IUeB4=$rmew)f2AR)FR#&%$hX&4Ni*7KKJqS=tIR^XPtYeBDKoccRpE>U zI7+djajZ{dM{Q{M4o-bMk4+v^w$mkx#C*0$gI>LON(4`bAmw2_yG0Bx$30EYkqJS@ z-94UH$3fmL9}>yh7YSJ9u*L_DEi+SmNaJjAz_d@i_Z(tR5Xx*6I&j6e=s)@JlwdhSCXcYaL|gFAeYF1f_DHHp&9iqGJh zQT7?i?MW!QRu#-r57jHtz9kHc5sqE;3GScoO)whn$k z2Z6^Yu%qn25bq~K*3Fp$`dX1yvQ-VYuGbP_fn>MDxSPQgTxtX_2rhGunUXIS3=CCr zwVJq<}ugJBZOGr0&WzLwT-PiJv|F za)sg9e(8A#H_(u#9fNq%8G)c3bUwE1<-;%H4uP2G{ifNRctFo2 zNLtMCdCeZeGIB0>2Xpk6$=&wlZkh@)SJ?@vr1puH;W$FGiaKLJ3rdg^w72IiIUi*=a$kQskRkagb~>Qn{28iojvJMcP)BlFvCFX z3wB3?9cKfcPUC9_cd71BAQ`csm+WRXWoMPla~qfk$T8tzAIDTi`qUlXFG@l>6%pW?!TI7KiTblCy7FRN>|#qjD4|GPov2 zc+HJfXai0I{>zIK){eZrUWj${+~3RdmvI-$+M%ViHl-bOK;Y-GSR@JD?qsjESs3DT zbdyXsgrI6$ta`~kvu)B95Qas$8rt2$m00l^vw%as?cu$k!Q2P&VFW2mt;=M5pJ1s_ zq&UC{Di!#T+hVF9*a&ywg96<+xS~g4!O=)VQYCF3Fjt1ui_Qi=c@iVm3K0N(_#L6! zw_KYDBxzoi@2K*d}OBMnZNv>(IGi7raM1Jq@0p#dZ@`BH%wrdjw>oif{k0;}^( zGL68`7Ew+Y(l39X@Bs2UC%mX>OHOI@nQVk8ndeT*z`QKBm`U(Sq6SuT2JXYcRZ^8* zYG?Mt;t?znBtj|qoli0r%B!3Zq#`X*?j>L4PH4jj1JcUO+iG+F=(a^IQQb{p*yY*7Bp)Gdl`975`(~N=! z<8h3_)sxRZ&B~pffn7GDZPdH0*GZfmpKu@`UV|8FMz+bF#5M|fES4wSdqN}v#fuTN z(_b;hgQC)G_$03K+7=GBuuzvp&(_6VH(EP!$_|JA^;fhW_TrdBiThl~#}#>!N{3zm z^MrGDlDdmj#VRn-IhHzOv3i#rKTm3`O8yg*NR(I~AU&ohbCVUIV#4I)zEW|OAs1Y3 zh_(7)W>4*D(+BEtLlmXb8~DT`4<%83TD(t%HlUkDq=O8}EE0X_|mxWpjLX%qf&SGXzA&VQ@XhZ3f6H8A8XJGp^Mdb+y6~*(gC4Bv7DB-1sTP-sf_)bT6b+wrC>XBF9&!%sQJ^;h|LE z2(q@ZV0$kgJp3$K}HtA zZ{3+*!SsvU%+2Kn@I zCXa-3n>K2|edjK%AjgHFRj(cIf*Pr`_M6roM#Q$;(kH|5b?4Lr`Q%JVBQTpyN`XK%2U8h{VW zuJT?VpqSf>SQA`J+ya=E0EQ)LpYT(_tF0)2)Zp@!CA)ZWHE!j%QOU76CjXV^Pw`wr zkz>fY!re6r8_y@ufd!xKLOXm2xVvLkRkBHbUSXLI#3STGZ}@eJoXa5n2KnI`-UQ3Y zKy+b>jBA2D6e-v^dCfursc;M2Fbm{E`A)Ac;h~i8$j{!|B4eQ2WH4P(4FS(-8h25g zOjCrz^qh@@loB(GAh3Wxuqx43gl1D5d3A8;+4i3d8q{7P`4* zM(WInT^4OVvjXnstiY1z4$OG2+7x8y!4(_qhG-Kt`^29G^uYkf6qyq>Z);HJDK`rn zZ)|o3r>>wwKI6i2c<&`Gfo?-$jK0^jo26;@SoKe!LeLpt0p%j9XKWs%*MeVrbk(v2 zs&Wq%$Bc(rnv{<>=gXfp2>XxiR~>WDIWa^D+!iY zPvtO@mycrp?k}+B9L+_s>Y9Ms!$OuQafj_YyX;yim5z`h%9HDZ@;W*GUxU_DTVwlZ1L%F=ARNbP;gu*bFd{_vPvxvEV-Umz0Ja5dGlehJ(1J5W}AA<@a zEpJlUb>V*Awb6@p$fxT<%g~cDZdL!`@lUFd`{jx#jYm6Kb|16IB3gFtnNH(B4Z-zu zcX6%{pM(fV-~kiL^XELp{@e7hFgDR&*Dep8pcl`dKZ*Ujx8pYJ=-MpsT7-?DHI#-q z`|a8}f0a4A&sa}x!5pn(@}NW9{-dvc#xvyJWVi7FIemIt3{depdO*R>eiFDA^Fo)9 z*AUueKgQH&kN}R^#|zE%(PZTgwP$IKccG~dS3{B?1n7Z8O)gf)N*ZJkXz|Ee_Q2<~ zfx;U!f+MPo^v}sU^f_W*>z3!Nxhz-2ax$OF&D|kvjP4T^TTl~QAr*qp4)*+A%q%A_ zU9jR})!YiWKS_We)0=bYPn zpI&x%W;@HSNNFhsaw0KSi7i`-5+#u>I00e=0a6hJhF~Ed0>pqn8E|AIa)2N~5;+b< zC5TN+B1My;ShP#-a=Ei@W_H@0+wVQS=J$KvdAS!<-r2L~+;iUYFHf&{epa|2txb_h zA)a_3tLYEE^(X1UA>JfpawFyL?Nj#wqde|G+Of6LZ~pRU(tq&tKb>Q0B5ZR$-9``;0#1ACZbPOfmdGr9_-bsg8Kjvqyru+Mc z>G=Lp`Yg)g1%iN#0ahk7j@cl8Rv9Q^K7@K&Ag>2Sw-^YVKP)8zjt#6^*vC7F4?8H! zj}8eNq*=EzxLUok{{jw$g{Aqlh-kJ<0Fq72>j0U`opuy6Ky-PloJs(t+6C4`VDr$y z)Bu46X!k@)tny@O)fU8OjyRCHu@y?%fYk8zk46R3ve=4hHC`;p8AdM%$@8 z8x<$!3lFAT0V2SMmJDFVs385@J#3_6uB|agLcfDSb`(>UOG?N|iV?EFwyX--jJrXZ zaK%Es?dskpvaAEvPtFoK4}eCX4o?mi#r;Msv?drt8bmIXHWn819f}i}1{=5hkw*i? zK{ug^iHa9IHmVTT0P34iokgmj7s2v=2Q3-be35Cum; zhXpUf(kBV4wCay(NIHECDl(S2h0K0(bbxh+3~jpKJi;9JQI{g$I0UAthhPAXaUR5E zrCl=*_V8#OVb3ORVP)+gts+WnP%~kMcx5d3Y`CH?JWjcj&r)AS2KRu=q3XmZi`bG6 z=B@H&GHrJUZ|L=FX?mU5Zsc@x)20Dh0dHv)cRXt@Fl$vQK#(+R>It>|g9Q(z>Uv*9 zB*;3j!59!9#T&Md=PSGG6ClIw!H>6ehmm6rjAag&o})v{&s$#!uJbZ`(@=tJ4@^k^@Xlj-(MG!^UujO zjJpi^w~p8Wl*_33ItwC8s{+uIOvJusvc?`0=s;oZ1va7C$fahIVGR&*K~@(z*eY16 zb3~JkEuw6vv}x)NvAC9BvLk$n=dP_^N}vDCT)KMYa=P{KF4l!nhz<6Y*gUsMSthQ> z%JCfLznRxwP5;S1`cnGr&1-4r-rYpO0Tx|Z8xJF5!UsRN1vhD>Prmr6bh!60y>Wjh z-TJ{>1gc@)h zI25K(K6Nddwth^;af7ui!odzmp#H&!4-mU{(&ep?m?ic@01$wMD~}Gy z(Z_}=*R0i#iG5}a5U&nOl6< zv=o#1kGjEqc~6l>Cxu5?Bmhm#T)_=kwPB?Vz!DZqjkPw}6%$xECQ#z!Qo%uhF}F#x}hLNrTU0g@t=kO?pegcJsYH~=LuE&YTArJO*T2k6Ccqw%{4 zHg$k>(y%%X&35|hYrN}Hv$?X43ENaPnUMX~#)SCI0SH5U!fNB&=MWFr=tr%>V#Rb#1O6ob080Z@&fQrUUxLOX>ahZ>0zC+)5iS zznXTZ7E&LN&tn2iP+d?aC)0P|znh+2nN2gW?*-ylPg(c^GV5zMucY16pGyDT|M*Xt zg)76D(G41KBzDQE{pk42OE0C@p1YDB>>sBuzex|_jpg(d6kL=(#>rEF6K>2+(iiP! zTE28CqWH!ryKVO6F$(7Gci(4z^XVmiBjPuG2%`PY7vF%ZETvE0xSlqbXHY&N3=oA0 z5DqxSdVrGzHmoR8(tF0*fJ2BXP;xp4)$pJx<16+kc11c9R8LACvNcE$A|TgFQNwPn zsOPq^*E(M;El@zrQP%wt*6?$Q@53$)Y~+@gGL{<4$hK*i+=fsNaZHuTDXdU1g+T;W zM@e9AjZ1|yW|aQ8>Mu)K2DpHE5uLFrf>c5z4WyY9;Y|nd^7#*Prb_T zt0+zTX}C#~Z05O5a_APa`victhgD=xQzRQ!V{MntZKu_(^BDd=qS(rVba{D&%!PC5 z*{c`WyJq_D|JQ$-?(Ch?WEyLYVU6xDz;JJmR;(zimVG^9UsSY;0L45??9;Ejlop@9 z356xs4=U?^*}B1i^ax`F7~NBK6{&A1+gz*OqGB!git<-H2Ui@xhuKrYA)>IGilAHs z5@)in{fhMnF5#K25PxFKf|)v5_ahZm)|&ARP};IOZ0bQE1y&11R|ECSdwUPkz5Cb< zTPNulMWlzZTY0jEzkC2YC_yo!MlC?1{W2!NB0^BZOVsOZeWc*X)zOjUz-Fo#v}>&xY&@QN&&psdjg4M~mr zchg7z;t!zQgz_N@H4qc-!YM3VQbvI*Q{SNkrH~Tq{nDfc?}?35b{;U7m#$wZT)MKgmYSrr589atOGKJvyrWk? zhxYtF0MSb4FVR93-~2iNm*&>f3s)|rrEw$8Up<#@y!^|^dDE#u&8{DO@7w9_!-I5g zdnE{TBnM90voK9#n z``nGI>1mXmXKy@{zVqF0rgv`LO4n;QQGTlFwHKbo)Ky9U{2R1)C0bD5!`9M5x&&pH zP*kky!(Pn5QN}%*!?4Dyw7Qxi;nmWtT1N^7#Ix7tmub@#U{#`(W5OKD{zrB<_tADO z@VaS6jj0o;#0e+yrPp%4pACu{qtKP%!1#KhsECp^Y`FLms&jC`0d?=VkFg~-ztMJ_ zAqJu*NTlsNVjI+rII;B8uRi-*k8giSt%Q@X>s6q#h~hCQjo^}RgO(Ks;2*Gg=_f#e zC@~-}$&?!Hk5kB^RTMn~Xk;K$41SGNk2!!yx2-I?A@IgTc`#JgDuZ=c$1IC9!9XKQ ziHRDtZ3XgaEEw}k)@FqZz_bbRPkV2_jXk=T7N5VGo_g&wU?=9E-Q6^~z812#rnVCh zWEMas+#|5mfR+kC{n7p&3MEd5ciuxmYvPUDiVZ4aX_z3V?fSK+q0pGHpaM0Xsj1E2 zlzCX`Jo`D1b7GFaO+xJ3T7QI<-6a%l_t-hOUo~RiZQpb zXiTNs_wEMkKH%QElG`7=3y_msit>2-JKv@|J56p_>o#+F^!Oxw;^vF#^>_X}U0Ak! zX(Qc!u#?_;beNvnT1eNo)=UPlW1Xy(u z;E?}NZWZJF$6kdp&R$1AIcqQA%HsJs&hdJcEB^`5ii!qHN%r-=TK~f}vvign-)&hX zklTYpuicnvK>M8--{O5u8FG9jM8h(XZ}it%}$RkFo)Zm5`Y&h9cgFxrjL^0?l*Zs%6kE$B7Ysef2!gg~D@Yjc~sPPLTsBo>!Jo zI@u5I22#OpOgGvmTflxBHVj`v$yr7A>=4sY#AR+~wc(fF|HgaiZ~W^&pVl`o5ol*X zybc0ljpx=97C7&l98q%fxPc`P>D>s7iYOoyQ>6cDi=;8e^GH8(1w4 z4m&*aI8C3jW|Vfi#=PszN9j9%{N?o0r#_Pw0Q7@{J**&`VH>u8+ZNxQTihg2AFn6! zQpT%LI81N z;$O(*>=R%|0FHxqd4Of3L>-+@<&}kX9)yWZivWmbWg9iw?tq#NYyHXp(bkA z01wlwqN$i-#Ij|J2m(ccYT`Fmu>B4u4lsg4ae1aLW7lZFcs0k!TkLdm45e>fSR1IjrR3#|7rT@(f#zyjce(jeDM#`fB09vkXCTQy!q}Ap=izYgLiMG z^NXwLUw!eD>D&f&bFmtgG}1%;AHxM2<{yf&9PWjO)K;0aLHNLAy#sMsjPHWVvUujY z87gY(J)00L-)jVVGnxh33ZfSuSY3dfJvFGwik@sO_rmhQnh%hdhxiY)7t6`*%&N(C z$obJ}31^JlTSUSwV`@1S!T~sM;?V(^4V7^MHa)gr-V^{POYou|!uHSs0kADLU9(jL zj|6Ms93@oDVm0id)BJAxDaDIvFv`Xv73`j!nL*k^*|F(z8%kc`e9(fY^5VND?C%g3 zdV>Ae+P_7F^~u$BEDWo7>x^L_v{F?-ma=|0@i;(Ii#}>1b%TP=g3(WFAwn zC+*4=Cjhys20KwyckwnsXW1G8PDr9uFNd-a3@B#axv$ zPcS!*vD{cRPX~mY>KyrcmvA2R#&&H&sKfNvfoP1W%y@#E{)CeJIxHGkDHdpQ{dfNM zf1Jip(69X2pQRVCUZmR(<(2@-MUjOFv`tJ@TLACM`g+7aSVwFGc8vz7skYllO>L`q z@RU^}9z*Qo5D<@vg=tS<+S;aEmwOkj~1Sx0Jhz;UgLDX<$}In8=kP;yVO`$N9@4jK4~iB%6~`AsnN;xaw96c>1yl?fXEBfjhYgfZHCWfTqJWybC8`AeKYQ~|D$Osao%i2P&;9&w;L;^RORB{4 z%$FMibVzGZEmCMW9x(T9EGC-^*gKixNJkJ}H>Q{Nup*REBz3e*!c~;P%ZMpv{@UoU ziG|0G_X~?NVaql$%nm}J9)aDhv~qC^H!^AeAeBx569LvpnjAwo*085e!^PT|9JQNT zSaI*+BMQPy0MO?sr2!&w9iYdY5!;x9;(+Ymyt5X_*_b1XhV`_(l=a#y7p2umBsWl8 zz*Q8pf1YB6XwQcz7G29-kj1V=#<;jo4FGqb-5n}wXH+X)YC`ST83Rh3d!hx3 z$pptNTYGoFKrK&dk{{!QGO~!?fX`++Sc+Xz-9y2!jubo3jrg7S7T+7>t=-T8^Yk#k zlnlN>tSG`NN|1U>(CZngX>J13#G+{cXNbbqYT+bcGse_nFfP|16>CN;Ai}I;C|BT@ z#LJp&A~yyzEoP!@ChI@F`yu_szMnq2^)J(nU;d9$onq7SX&1yrGfTio?s;6Qh5j+# zI?72Qmx`>Y7hpQ(G+ez*1)v1UKoXS{HD@a|wZ+&n( zJp$^Ms4o62EA;#`PX$HqVk!v=&owsNYaijs!i@K+r=CISuq3Ud3Aw8%&|lDk~S+ zjnp6_VJz#;=l_I;&W^a#K3!}$^6M<|QQUK@y zu*k`BCqA2LU4Z(lW0WBn7CEarma#{gJ*Qu;7 zdh+sc7(PTtO%beMutDs|$ z9dHlie?=mILiyZHQE|X&nZ9@Hy)b`$08T)$zx|uvO@He*e=RMdlo=jbn!rkfw@eiD$5r(ah=vuyzlw z#CHde$UI|BAT(CPbRSAkR+aGhX-?)9il)1ZiT0`w6&$ zgrTXvON5MpFdfRIc1fW7-6C?Y2%uJo-KfK*dWcI!-PY(1Jrr@}Zd*81fQ2&3t+5eS zv@g#?*|ZiE0HOl&qeYO%7zf>OBecT2fnnUZOmm*NHTWd!Vo)rOrZkPnnSaIHH>eYn z)e{BTX2bl)1`F8${t*#NZazED0N7NKf#+xlH=flj0820Fs;@(Z49Kv2X9w#*A8eai zTnAtfU9d%pSjI%SG6wyuUwtNBxU`*yc)X08P!W|%nc<)*N9=m~!hHJP!~4_|xI{HY zVjG~YI$11xUnP+-3NpL!ERGd2Beew%V7{A2ut1*cg35iAK@W*F*um{;fxN5K4*RK} zcr_@}NUFDg2|9Ei*$@rO6ypp)&}4gpmwJ3h>-UmJWQ7^ zUQb{A{V%28{`Y?c+cpT%B+(HnI);mxvEk;4?2+2e9k*AP(s${qBO5-%gSfU1y<bxTL>(h>*r94SJDKWs|uIOQ+(V*`U04u9vH=-%`0Pk zh#KbgTlWETim53b%vG>jwdN=i2NEz{fFP6GhKLb;fZHN`0*VIq=>lo$C`8uLExfJv-GID^QK zP9B2{#2&DJa_S=7p-*YsAxXHdiFtrk0%)YtB8&2|NTzP!oRwD%H5h@ARzNa>Fic<~ zB4o^Xm?3*$qnk<0rXq~MZ_g8*Wkv39LBtZ>fh+`BQh}e-4eE`+7+Ii1+|7%ULaeqE z0AvAV+?EVvN;V6|oH96TJZLzM02~(6D&LkKwE$iZF+{-CyCs&HI5`4qP#nE!ETkK2 zmPMb1wXNwi8p4EfC3`gj03DNMe_TL~;$rv_RC;w`2^kR^GMgMD=Dc$+zDRxD)zsd{ ztq3*rx~OjWMB%LAK?B85Ji6f`{=rvYPj6j3pKdO1<7U1cN|%Vh#jq*{%q9sq%Do{f zKpky@=zRr*9!g6GD@$$VLc0C-2dVegH`BlI+kaPym~ldB0eTY;QBlyQ3GxO&N$7>5 zpacT^D_9Ax(uS+a{H0E10)!yY5CagFO)MZM>!z*H&8MK|jm_JP0@qIGjn80ktd*O@ zu%V2AN_ahY_NZ@wtbLBKRUJ$xjNK5_vc!pf(9K;yd$$sNa96IKU)@Soy_q;ROj9?R z5*EyID3VC5QZ1 zaE&b|R%OnHA{bAuh^8_~`doNOi_}^x0I;L96rrMn0mT)`(JTFXzy5FjmgSQyq-;93 z@t)vYCJ?iM4o2Y28tZ`{Jj?SD z5M^wCCuAf|e1=|1OKn&!b+J^FrY+4z*U`qm^wqDY$;&s=7Szw2ykJqxPl~{-1K@_Z zR>x9C;%^6d*Q`BYOHs{g)$`91VfG;X*MIl#q`&`P{pV>7mKzPCW$CO>l)%Cwpq$E| z3?Ecut;}CD>P}PBS6}}=A(%JQEayxBMBpJW<|*gnAF>NC#1!VO#i0~tCU3^e1HGaP zFm}v-02o=iRD%<|{? z&+^kZQgt5r7T{v@;}H@wYXeUcxKt-bLzoEVOF+XaVgEIqSr(vijXUZa5V%avZC}kEP zA)*QlJz;Z{ccVZDOD=Xu-CTY}iw^8s)-;8-tAK5h;7B?IaHLjMFfv#aSBQaq$a?`9 z*c1{m&nhz&09ZxFx?{6Ux&$@zIlv3?q)@GBFvr4jPl3Ke>XAYjAd3uYa5gFQA{rKg zx%ExVYKRq_GobR~G1{pLLTE2IE)*SPwd_P7r;4X$zak>bufKXD{lOpozvSuKC9qm3UNaufjTPs2zY7+%D*16`8LRM9@p|5Ec79PxfjlFW5cES3@l$Qotprl zEHpZR4jNW1fI8|`0Q@R-Qc(lLy|*&1dp5VYk>0~%w9?+AL+Gf03dXD1g0GE~#)MCuRI?#x}Sq57Yr+tfw~IAPRggBUBe~djcaw zqZI*0HW{E(+qaR5tQ~@aESj;Q5@B|wAsEUzWgHIL{RY?&ZH)a0vzB(nIYa{^#FPd1 z4tAlsusGP2UC?c)(kDh>7B{3V2Cp*ArY4~TD%n#i>`;e!Z=KAcYSr6h9^#2atT*5-}!Qb zK7I3g`kmkXpVHO;;=fFb0HT{yhRP{w>2(@FSuJ-Y5~=_TL}p2N#GDAKF^ffxCBhbo z$^Gu{eKGyQb5}Em3g=l1PVQ!1u}J__1=6HXv!f047n!CZUZs_FWM3&O_vYF8-uq0C zcs8H&a7e#XROq6obTD7Z1=TU^HS9eCiP;Cm4DHclJcNdoRwnS!hNuCD;r9^1kf*IY z#4I@n(-@r1;F~6hZA&v&F<{Ak0}x3h*9Odn7iJA$fpEe?<^g1l#)hSZoaxE%GSfIT z9vc-@0?-YUEHGGwb^M$F17-|J{~Yq2iwg3??Zilc0IJJ4LbNc4zr-fO^P#Gwj6>Nh z;oc<`rq6}8FB$>3(TjmGm~A^bNsV19tr+2Nr}#bzl0|BBnPPpoznhkcyRBl*I(Y9r z%3y-Eupw!!NR}--6o%<*8`ORqkxPyTnuCupJ%96JI(KlC-umPJJ$)I@GNdEU9HKUWfG$i zQ$P!A$DI&p?ak4JD430n29&mOBHV;x|Ifew&(gpBOFx$uN<@v3xl#tGikcf`-@J{X z;^8o2%&2EU&4ghq6sh(bOjW)1Z2G-F{%ZQ{CCZHw*wny;C)JaZ>E$zS#WoAvjs0X_ zxEIV6Wf`tWzFj)E8KRdlz5=d5>U&0IGoW>DQeqE(bFU-!45Z~DG6Mh@Q=5njfwJ1X z>HMe717JmFfmo?X31P8IYF+u7o3xdy*=uVxa~4acQ-d)Ai!G3Lg5`@4O#oS4%es2y z0orm8BY&4km5^DcS^P+GXMqtE0{D~>16 zFQ5(pIzT$m)vC@rRsDL8Um-SLIRMt9>U-`AnI`#nIprVXL%gYwgWPkbB{@wJY-~Gqwf4~j@ ztAFjUri)vfsk?VK3aX5tXgY+X#uncj%0-?~CnQcpB|N$7=TnKu!2jcq{xDs`WOnV^ zMri-W?aJa~B#3!SD{wko$B%EloeJ|pBLN*-tl%_6E3M0n?NhvTrRnWle0Q8uQff7hSI0V`&;t4>qc|4nO3 zD97{cCr7Zqpb{z}ZivVTHI(u&`Wi0pB4)Y?T2&QFn`vY7MjF2Oxd2jwu}4x*khYJa zV-x5hxnOF>lX%Id4dQ0(iYFqA%4@`|vT-W2RjOsk`iCsufGlWDTmpxHtyXPU&sTox zbI7{HM*J`TIQ?%Q+)JPPcYY0S@J!^c5#*zZcSC;c@OQtLDv#hg0N=9r-R2ypbm=C#-jxS7T%C)={uQc0Q z&_=2&N0)MQqj0R~M;L1HL2N7%s8s=o%WA;(b{%N~Liy?wB-2By?tuv^4xS{n^=P1Q zbctf>;dZqk#Ta&|$3~IBNU*4&&PF)_?g)iLqvo>-c`|Tw*DOL(i{KaH8E{Y97%*zq zGI-NmYzS>xAYH=z4?85l3MJ(-y+$^x#0BE&m1{%>_B3A{G27WC6%Zgre?=B976Szzl_BTlu zY=JRY)rCchjdC3)W++eyrOQ*Kw0QKa!vNx8`C{%+4>rn8Q=Stj5C9@!^@Y#Af?f7F z-6T}>y?^lA>5WUTrc0lHEj@MlBB@1Nsf0;S^P2#o{MUw+?&2W$@au1<{{6cofLTc;F%f`*xbWU(;}tk+^F)F{9IVl%-cclbd5no{66H~S#^pMD2xK|n`Yu4_ zRQc?Ae)f63h#EzsnX_W0Y~OOa2JXjRV;YfovJ&hG*F&u2*jyZ3DZjr+tK1xC zvR6`QOWab$9&;4YMpYj`X!7|6g@+j#N| zw{$V0ii&19VgP!W$?9Nq8E1g8T0>)hAKnH?2tYvw9Z{8gs8$Y}6=+JNDY-!W!Of(O zpcBtAeglYL!?7{5y-W}hfZAtNVK3l#86X-Bp&G5dL%0zo0Pu)uzUpfkz=_ba;UDLX z5n0AWYPkkx0mzOmpDIx6$Cr}C<3UPINi7x>6mrLzCPn^GxGr3|ffZ{fZC~F^oqYmq z{`l{wulzl-YcFi1`qM8*lB|F$TP^JE5}z?hSFmqyKYNX}MQ+5>GKK}SLNH6%e(VQV zfH`)WbRh@GzOL^AIdh6`miV9E+z#1PGpK9v5|lorIv^XY5|ULIiD zC@5?*AghsEcdJc)zAfxsZA~B@w7Upftn68&!5LxMuH~m*{4Q@h5sNWP<0SfQBOq}po_QysTHF-CSP#^ec;MxUPH#8 zPR(6_ow2H02xuY`L;eegzX_baoR^26^Y8sTdzRLVkL6{9nABseexoQjK~wH=ku^h} zZGyaI#HcJ%4vNcl?yKS=8f_50zQh!<(26;7u1S~`EYYA{HFS#%4H$c{VMh6R4BFV^4+RZKZ51^U6km(<59(%pr+!6gdF zlT%FQ&Le<_u@rG4i~zbNYM05T-64Ao14@hpisU`q48$H;kcw1=AZ=}`ib4uG1Aupc zwWA9aZR4nj?lZV2%goP>u;l0A&OP!?4~fn?MegiE&9?BIk=?=C*z4Cq)lEESCtPpF zu8thZKg-AFDDxxQEI?^kH+@t_-dTmH4ziXQk6x^~t(CM|y3BO|7eH?>k1n1*Bz5Km z8j28n=j<}Os}x>8sUkr3EKA1fVjZCiEa26X<{v(MK-TnqstXY&O5nLS@E!tT{^HM| zuKt|^t)Fvhn}3&CrGk$HV*0zk>9nY#m*`zJeaB1@Ff?F{blmV{8a!q0pNgW3f3=C+*U`{H@v$|lMAQNyKnj4t#)1*dI1{9~ zF#OIaK4#MeB4|hIt0yvd%&W%!I}Ogiv-w4`uQa<31u~$Z9kPn9sHVbaw-E zVc?wa95|o-=8dhDLIAX$Bdf!qi3mfD2<{v$sieBZAUIbkQ;|TGX{yFlSZJZIhS*YI z-bc6Zrky+QrMJI+i|t716TkX}bRIkLm__aowRVgt;}GRn1h!I*9?VlHaR*sgrFE4U z;{{%Ny0nqnN_GMYEYkJJv8xlTq*bKK_b%MTfMi#R1wgC9D}w3C6JQ*x0Xrf{gt9U( ztL<11O*{JZv#*BevH$2!I(hUk&Ml$zl^_JMDM6IDFZVeMzZ)p?xbPC@S1l z5ZQHv|08oLvq9pm!o$-6Tmh5$5QJ?Wn?4#cvNq6&boVNyqz*^|ZacO=+i#6g;6(3G%!gfI+j@D2OO!B8u^@ zT4anIbGcN8vtAb>qBVu}dV#9(BYOjYM_>zcGnH#?{bI!8TBg++!?W|NVXi8JlqTJV z80a;w3C}nStnRyq@A+rX@g>rDFS zpcPMR5KOH#5gN(9d4K*Ndn2)e0-<2C`*8Ci>q7%}g=M@eP8=W^u=rXHOryydkG3@f zW}q6_Z0&sG1R@lYMK$8ShKxKd>w|?y>?oV1-P7!eG7DQ(fn(r?TpGmOhbqEIV!?zR zjpt4lh?3ZYnLIfFOZMYpqO57v;MvO1oI)fS_G#Ga9)Z=}-N%fVO(MGAAQ3|mBYF)4 z(>x_&76k_8$PAdg9yb&z#3dIx`VA`=nsI?)-}r?Zaj?+c$nh%T!j^>(;3}pw8Mc|7tW^|%8yo*6Rx!xqSb{?sV48!e7<$p?48?&yPZ3?m5P_ohZUobGT6sDFhpiw zS|=P4`P@qCl_()U>=H--hC<$ze4UQw?n3<>HoB~Dk zG@EAZ*v}6M=r{POM#i&>Uq_JvGMFoC-baA@@x%4X0rc!|o|^-^<}4$|Dw?t?bS#DH z1~RZ?mIOA;u?Tl_A0%M#39f}xvp;PJWfug6ti&}EK?d*k!0i+%(3225jWEOK0WhJeKEz_nZh*$` z5rt69Tva5p=-UKV3MuP|=UYRTRYb60fl;cC?J>bvMHm<<^S2%w8-SN_f((x)l`HM< z3us~4pqdWKQVrMX2tr^%?edT`c=|IeV5X{KckVOx@X7(q2G?7YcX+bPvo$s8dR1J= zsA8ntK=mLaj6ZYr3aA25ST{dAApoizf(I-IjiW$Qo6soN8%M6sw|jcnh_{9;-H*p#7-+#9ogP4tSRMmNoo2`wOK=|%WMQH2{Z z4!b5rcOtk0Q$^Os^T}CSB*N9pJo3Z&$N{VgY<=FSyWw82{>qCIBZIw7-kb(NjP9zy zWo_Fv0P#k1l*C%!j(u<~J#q?r7KjDb{I?i0dzXLbUidi!c;7`B`*)ZLLIX&WOa)#E z8X+mM-?oW$e};Uou4HWv6;hid>i!quXb>18X@R{EFK%#M?NFd`8)_l^GII zdlqc(V!tdB>``$LP%zs@&9F*VIxM-6P+kZ%z=;w?6&)8)z;LnxqCq19wm{-{rd4YO z(LUz{m4l*j3D^7xVEOCDi4)!vs;1@G-M@e~2F|$Tg346%feO%8&W%^(;=kT${l z)5Icc&Qpgh>b7O=jBzc{K-*psW;9!0!i_Z0+qE+@1d|K^z#E9AP9f!W-Th~jnei&4 zmTOOx9(z#ZZ+_xFJpbZT>>>7V*2KN>@ZEcRcpvvA2UOql-{OQb|I7B)`y1P$xPs>v zLUe>n*kchw*VPIH^)&c@g5Y-(8Tt^+E;W@kOS->tf5**&iB86fSVs?mnNU*Jfw37H z$_efVK*2vZ&PA?s(=z4OM@+8wBr42mHdA`I5Eb-{m|KFJ5UEd#HME%OSNAo zD(g^ycMt+>KB-);@)|!2&_7~3AV*|gaIb+bO$I(PIg*KbzP3ZtQ6fFbb7P^$n1I%x zXw*%kK%HHQj-Z$}>28t|GbWI=hI}epwe+W9hu%wI_W4##)TOEI)v?+_%M^2a&>f?> zw)gka3;{G{pv2!tl=(L%wMSfOAL>ywUJ~S4F`FE_~DB)hTpTv>6h$YC_Q;m^iB$5&$2dcszvlzQAMP&k>d!Saig^>m_5f=?_|!k+RRf_k93QX-Q=wYBzL5CBzbWMdVC zdB(>A!lI13H~?6iCEX@!upyWtBku3@Xn4&wPF5KOKX(3ClnORioia_n40Y1GSs_GDcKr;HcRk#O@)oO%PB^imtsG#iB+mgAKTiHI`c2EKg-pnf3VaAhiz;C0?lwi(W#0J*GS; z#ctEW$m%WvEb=<7w|Y;+2+e$5MFv=R1%w)7RcH}=LA^@svlP%^)ExkE0g9>oX2+wo z?emJGuz2e>WC+{_5Q#h@DR4d=-P=i<=eKB{h%AikEMM0nH%s(Zmrd0MYd;Eg0_U&H zJX2dnFBqqXk00APH53_Jh7aK+ig4u#s?B3pw3>`5KTI0nC8O5Z01yCJhy6qOq^|6b z%1%3;rCe{kEp>e#+=f~Y(mKKSD;F+vfm97(crpip&zFaLl<$-KBH-o#>i3R2=eqvx z-vY9~d7kfbTfZkw6YGkE35yNmA7KBc0u0Yaguzv>fuO=~ZHR%&W*vZ6v6q9IUE+P? zSy0_@WC@E5v)se$qS9f_F)a&DCifrb1Ip!9C*~w93bI3Wnl|=Z%`;Iq0EdDp>Zh=Q z4z52+>LLH&Eka3fJ#xytfX40Q9GutY3*mh|K3)J_%hYTjHNSz@en35jNVY^S)XXM3 zs*|d2=^$&6wP{*+_zK=!#F{K7L_ECtBy=+P`6>0H2|+}z%}5!q=>=& z2gqq^c+)6G1*mfu;$W$KJ(fTk>jLFQgt4900g-&xs2DO{gRhn8i#C4Yx&r32LK*Rd z%~M{CPMvTa9XXoTY}Y!*<__q9)c&8_^NeV;Et}b|zRozI=&=VG z*UQC@SQ&sZ-v|G8k9_5adnKF*AHMue*f@J$e!c|8Lm2A!Nl1lro#ISVe9;7J$54%c zMMTpw+;T)wKyC7x!%U^p8R6JM9Achbs>fT&R4=S<1}GAi1A47RyCD>b7WT);ze2-x zkPbG-GPU~-K^|>3*}}(jXw_3tIBk{T(BNDHVk|b|d<9S|t|-)uK@@9pO)7^vS)Bqb zl+@58e}=ND`>Sj_zknjcIgO*oX`f&ewP@X`%6_)Zf=Ocnf>OfLyHG{@sY&%!Y7Su6 zGhly-C@_K(L*%jQ`%Gzx%nTM%wyzBs8NG*$&_UKbB0&73hxds;JxXgEI6Za=3}+SV zD5ukm!;+jD#-yfF{65~vAsVuF(qiA>AAk(KL9bC6KxMfAYFx@(-#{EWBrU%}abaDi zR+^ujT#lgn9=$Dy2@2(J3{cv(X+)IRF%FGsEE(exS=DtYB^xME>g!jO5J8QAQ+Npb zu6O}x$}s>rxC$v-bLY}0e&Q$L7~3>t!Q=$8%Z2nXN~yVHtbhK$90=W`oa*L;EI%iw z!hW9<9?#ET`8@<)#Y!Dbn)f2^#K=TM7dd=*)*xmf;_*MLHR+78ZEy`@sLXH2C>3}; zbR|Mp71bisSQ&&6aRy1CGzRsNF}6^-HtxvzdE{3nD68tCka$z(3YF9LHjRgAJ!|pA zjK+Aaq!wbB6VPz(pp9nQCyt?rTNm+-Hc?bw1c2;(V_?iO_n4kt3QYHW4=0{uw^k0l zVl_5MZQKTW4L02hl(Y3yY{hEcn0}y;ZR9E|orjEye5pxph&M^2IXy}O>C|X7g}0z; zI%w>`smoCpb&Hsb25kO>I%y)c?(zu{3)}P2&Le_)8-9d#@gVk*1{gS1MqY`D%5 z4wCgHKmo`CU>7iz+5Jezk8I8Y$)^wLxl7^XvGz{pb4o}kAQYB`9u(ZFl?D=7mtY4s z_Y_K;;#`9&I+rhQAfsl4VaS+WBN2s(MRO80JH2k!c+)}BV|>&$pvW{yq>RIA@gX|# zY^g3I0%(*~*%(&EBL_NxHvilA{BJ)CupT}?C0G*uk30;o$@_;FF?eaSJ6D+F%wVuzJ|& z`XEXl;B$F8$jZh~L{1eNrkm|#rT9Vh#TMB}fdYE~&EesrwDb5bNC42l2BrE{0?+NP zW3jj^U#iHsVITiZby?)Hk}(SjO!Z0DvL|^o91<@TuCN#zf(aV}rDdpt9fGaXQQ9R4J#AkhV0@1H zVf3kEoqOExaQ~19JLFc@q{D{Q=HXNzb{pBqJb>1r!qr0AZ2<`MDB0cctk{Koyw z_Rhb~Qd0l+a8J)3avfQDer|qFex88lOM>P((1GCQ$ONoEloS|}!;e7ZI%1m+Az?@R zuEiAR*Dk=xKzg90NH9+wSGpw^oLe}$P;AO4G{F;iO)XbTS0`mZX!oD{R73x@71y*b9m&4LoHjYyHwU8aEEhrH+B&hEE8m(GdDz8 zeiyg#0vff(d;9brV`H5mess*Zbtv=z7IVXuw@xvqH891rG0RPmLDD6irrTAP>maXV z6(E%Hgd)#_VB@_g-rN~tLW2rez0xAtQ5lG}6G%)mI(>+;xo1yd))u+D$(ov^sN$(v zu75!ICd!P+G>t+yr23-T!p=vx5iRCpjqQVc3J|wht1$sPGfPB;!LeqUR~heIi)sn+?}upUfjdSNG%v7D3_2tb`!3j^9nP|$*8Z}Pg_*)ue6dqd4R*5&C2kTsVApq* zl|C#@vr}dJLZ)2AqBwjBv@A`nQ=y&(p|;p?L_=$=5dyRpjecLBI<8(6p#}h9qhx_@ z3cxlUWDbN;TXivj@z40JK(tl-*nPaZGU(OAi?|F7ll2?O{zY$lMK747dj?Los6xBCy@p>&Vl~P<$1| z0WM5SA8~f{S2%clJGE+<>j2COz^JSp0r>jinR``ZJ4&c|XgU~1u*)W#$KV}3O-;Dc z%;pOAPY4B`&WPzisB0tZ?L)nn`|D`rh)KAT%Q#jnCs3l$^XLd_223jBnvZ94Yl{ju z<%*t73V1-dBE*yZ;-&$Fu7|*+xX33sYLS1mLPO zjo?n!r)ZMBEte?+n`Paa6Uct^#L>z^PhsZ<#!O9Ypw&T`)B@p$8PMS#hN`G2jm=e) zMWOW%NoCgz;`p_Fs(2}!>%|$N>KgUrL^6#NnH35z={Fs^iaXT_=}uOBm_?z;0oP}` zC@!KHZw%{bIE||lS&D@Z1w{>&e}%$blsWuju;T_=zHJEX182uYv07@QhHqo+9_c}~ z8PZ+=?moc0ckdqBKIT5si6)?GT$H9LAhJQ+g#F^inA$8gKQTk49>%0I0%8|{TPOdk zf+xslzbhGljE^rfZc zpt@QhBqn8uhDy4r^=6(}*jr0ch!d=(Yn}tGVvGCl;T#l??A0;Jcd_55#j@_++iTqC z9Ds8LJO^g?)6c#y4>t&|KS5Xd0OeHoF8~TWd2N%TcL%$sa4{*2mJ$cKi_dpF+=PN7 zSbs(+LW?F6lF>Pb3cJE2Oor2Rrbe$^V$n2vCrkmLMuur{hpz+l06fZqp@R_v!PG1Z zWF*xQ>D!NC+q&{*p{7-0B2*R}1W?2|V6IrrV5eab(>OQ!+y~Vu5I`!MW4>(Imy0S8 zs|=dh*XzkDLmhQQfSR*QCC8MgE-(9 zi)gix;vwVGc_o3-J7!S5DPZo$m|Vb#5&`UR63>ZRbbKy`{ZTP11Ag~JKy@G8H<8-w z1x|rnSM>xiHzNJ%kVJ)OjPB?>ht{>uG2&~0hj5NDuZ)mEG#Qt@smqH*{x4vT(qJq# zix4$_bdo(v+-ug-MQfa@ikx;xAR9RmKv6bkkUqpjJC87B@MQ*YYT77&!^ddlvIKa7 zl$`3^VvqmqmCH`B7yu5>l(TtL0m(_mJ(;jaHh7dMsExongu=Q&NJ>1Im{+_n6Qtn^ zd}3&H;_OZVz=$0r9`;c9rjZ#>0HDXn`pe86vWk3}EvV15G?abjYFIF&!i!7ftf81i zunY^%KCo~g@*LEbu^!V+X9yLwZGkyqFd2|UGRQs4Gl*UG>SaC<0NuNC9p{l{)^^fT zZy`h_y8$U*cW@eM4qW7#Q^FPZjK77lS+uajrW#Ia>_Y?jPdVNWn+AQFfO?GX&}3aI zcp(*g3KYI;;XGNob}p!IHa~ug#bnj6XWI|iNH`rov6+l5b_N(=X=AKy#F#!DR`?JA z3&4Tlu8s>xXRl2o(tN4oD5H>y5*~6}0Wn8fzjGXczz_{!66F~c7Y}6%r0*gg_}o$n zOA^mh|FANqHvc+YHtJsh7ef%ayt2H-iP&;s*+R4n>&<=lCN>IWcwjbO4dgvKbtfMMEzUhZJwjzq1&0g?Cg^0FSDLQ1d?S zb*&r|P+72(ARNRi)Vz5a{unVh}z>E>p5J zTb3|g~>i(P0dbsz%N02bvEWb%h>g1?{uey6fGg3Wj+(^`>yFtf^r zvB3!6(Zw7r5;z4wAc1FaKKHiNljr;VFv-Ch4MLAHLi{c@&)mY8*MiCdNQ}pV)QiZ5 z2PkrSaCXt+E6BOihzbX!AXT9-YU2fBgZK7G+C+(~EiI;E2H_Szg{C32+B zb8L$`MG|P8fScGLp5eBJ*u#BFX#G}_cOh**HHG*zULwK(5urzaViponL9HB)1ckDJ zAfONkV63vbg`B|PbKB*bn`Ip_u!yqP9zb>ef4mixf?H);tY5r>=ZzD z3^5LflECMw+26?Aa4y9Idl)OfS`o(+gds&v!#7I+oXe(+t>r@sY~Ydi@TuxZ63vl1 z6Yw=q?#2wO3tS}YlW}W-Ye8e|cvvUvKijWQF2@w15nzr-HW2|b`#Yz4@wZqhWP+(t z_DMaHk>-QTD5FD?Y75NBEf|=k=2a;WRAYp-A)Wzsagy2zm6v-2T&U1lwrBzkQtIDE z@rlYd+(!!9#r`X0@H$OLW&uy)v2qjasP1*=FJDXF|Ms^7sB~&bDGdsbxK_q-WN?f~ zig?sx@tX+zlUPxd-DGhkE?w-cw}mw$0_52k5!H=|IRJbkqPm6-NUI7P z<8PA&?=K>olFi#DtHOeH_7>A|U?}OBf+{tv92K%Os!*a#23DvwLJf@_mZ{j{+A0%P zCbm=UbU&>Sp;o?sALK_spx&2#q1RAs-GC;PE36VIjK?5O8?&6HQ3U`!aeHutjg@oE z9I*{6#1KxUj*?TvCmS2{ihYFZn2llM8IKtq8$3s#cTGJ6SmD9g6=Q=f6eD*~1oOT_ zL=hEc1Bz6b^-y}qAv1X&C5&^IC|KCpx&biU26^wLlP8?td-Ty=l;MdGER&xNjse5b#5J-vQ5;(>dQt#V;zXCM;D4H23pqoZ{B8P4Q-_IN{%1rkUIM3zJR z3^x2Rug$=#STmN(;Wim#0I!2~FizxS`I37+IdZ_3%5mzCChWOV%KZ179r>OE^jSpk zd_|P(IZhhj)dQs2g?~A#GR!f?V{8{2EfqJ*piL70oSr?hD7?l+^bl!G-MRMk&GdtB zeS>H&C@D7U0fRMOfgO!|$Tsa0+@m*gl_JyNHl^&>9HL4ar^gNr3!73{?rPX!8`H-2f!yQ%2>|{e}Hzi(<}>!TrjD$w6YqXhwvJlY1jj8`EK$ zxT+Rsvjp_&NVO2hRMv{QL|svupg#^YUxFkALS_ zoddCxc5(t-MyXkF{)M=~JgxhnLEFW!^pZudb%q%6Hl!n=RK(wdBJfP_nkoA41%+$PE-6t8E~jRUj=cGryo{ zHNY}-2RNwV$k2%fmqPShqYZ8VuLF(62BUm;03LoXFgoabayc-b{oda^yv|pCKi~7i z^Mji5-C6m`jZ(X2bwfPJ08vEn{0w`&hu>=r@IJoI9{#PkaY{^bY#jI0j3YHNjihmc zJCgnh){?M1l|9847MECbEfUx*L5AZY{ml5gK^M!74FblYn9A#Y*qE%*E;yN?9EAq} z0HP$t03k_?%)be>=py1+d&|t~XaYSzo>m}GXv5+l(uP@)d%I4qTg2m{xn~FhvM@0f zLofgvC>)G)yspiB9ubu`L&7UQ2~22YD+3_Kt`&jE96@}rv}=v)W^tm(ZYu>GC@7P& zcn1})+>UJjvVHdR*_Ue?RFmgx-pXvPDAI7XxtWZz%I!SIu?H|~#o)919wz^PHz* z!2~Yp9#q{>z5#hftlBlPeX4X_b3e zw6Tsj=<|K1ieY9EVZ0lg04qu}3h*T=s;_O|z@lU5t!oDNWKBh^0Nxyvav*RJ`J{4S z$-i?TX8|i;7|-{d|G&fE#6<0T%6ZddXsIhx@6D=ljRGm5nC2uZMzo=(RuZ*2 zi@%jY8l3C^LqNR0a3vH~?bxt_7EYD{){t-?vmic8;B_xOazt^ryx;9R_i%J5`?tap zG>5%>4sl{_Z9POe0lk4JwX%FJ)wZAP9gE!p=xgj*Ri)RPo<02A$>l)t;{cfhY5u$K z`FFm1lhetz~I-rL{(TN`m5 zjr#at2kJ_=BG=y`5!2ND8knlW5N=uCH#Vre$al(NRg{n>l&vPag=Cee&^T?r97{9J zJ?!Ytt^hg$zYY*w)NERW1+uw9uO3LLiZQB#h-?F(i1n>v>pp}6&mqI9@Qv|$MG!fJ zc1RK;jGM_jRHzdQV;NDUXk;3j-oo5wfHVu}q5?9qSY1OUd!OmURObm!Lh zm>-@z1AanWA|;AawTqMluoI_yB(mnz(Z2;M6~t49$(l$4c0Y&gOsHpc$EnQl{Otc`3hkRB7SJ2k&f>81^o)Lkl_OIZWU zcpH-LvmF5Byj?5T$@O;)Ll#B6IHi)jqRvENE96qw+4C~z3}OnOJFG#IUU!Yty;wi) zfCxhz%-U)?zrKkVlDVkRg2&FsXXWsb!^7FXz5gsWc#EBcA_$i6mP`iak{uYL8ON8Z~U z!oGn;7KTW?4We&74^Qp+#C1B!_WC1y00*9|0D13+$RdL8Er@#OB5!b=mvZ{-s1uu^5R4Jb< z2yA+CP0>pj!+wb#p^J^Wi-{t4TLa@=(vr}oFjyNSYMK*@M0j<+U zOxn<_3Ix}%Sl4dch}h&~Ok3-l=dn&G>kZPaTes3S36q+`Rxv~IaDbSA6YRxRY|L5? zvc@Tsu{L^=Yztr`>ki&Rpgj;`h{vu3K(~(w-NU9iOT`)k`N481E!+xAg6X`(oLK+> zKmbWZK~zgq*trdIVz7)$YXkBl35v%?l|kX|@9(Dv52225l>_8>DW0}!BhXlxG)vi4 z6-!I>+d4}n$9RgDU;G4JHn$jl1~{&hYvkenDz5Q~^Rrg5#y+^_T4;0>SnuHg1-6Ju zaE5}~l|>M{kC_tdfY;ZE*qbKD&ddg%uZd1;j4@^Y{$KYp0HV^l(k85~@ z*rxf+;C^!p+YpcedO?jjjqs=rk?Sq)XhB5F)v8dJb1LS;yr~b$C+Z+%If0F504RT` z56)jPSfy-%BjDTrX@kwQR#{|z-+U4CYrS6%gns9}JbeCHK+eB+JR+3i^^n?Fo1lq& z>oGS!pH?@w(*6$h7eJZ{9=tJ$k%mgH(pUEYBYAPxPMWK8ase+_q}(Bek166 znvK*`1lAuAHvm|WmQz4Z-Xsr=DmOR`9H+9fK|2Cbq)=4=ByiNW2ck?7gZm^D8qflL zl*w6R5t(00APeXh)IQR7SSMpV8n*QU(aiKRv*SdAJa*4}_wJ?{OhwTK6<{(3;RN7n zs-$wR6x$+%BF8zrR!`q}hNgzAVQp~TTr<}*UwbZ!IS6b=6b1Bq36&Wi;XX!xn1F&W z&r|Xr0PnDlh`0z#8G*6lyv6=HI1BLHkVu&k64%8e?7ys`_G&9Ty1zbO_)>JUP#(f6 zV=xf_6Dw<_Em%*arpa04c}wIv&jQ4(2f&^Jiw$C>3zQT9mvzf$Y{d!sfc)Lb_*ua_ z*HU9t{LcaUyYI3a|BFN0=Z^q4`zovU=)JT9W_DnK&S!b}jI#^+<73Fr%bXT+1;>ff z^90G1ncP6_gaW!4J&9-OnWtY!htxoL@bDo?iYg;Vneu_fW0FO&PaFFHP~0Pv<9@n) z;{_5g8({uon4t<#+4}_XEQ1(L%u{1z;U>S^fm7SD6@{WdFVx*J3j8Igvfd$!p>QH& zL}{UJYydl}U7%x;sX{{%TJc#xQCJtmmKuN}aJ8{;G$>TvMC1UbPz<56WOj7n+dX2f zAEU6XY+l3+hpB1zII0-wu?ib1>tceXHRV3>>CYnjUk%feYn-p4>+Z`v&jDE^HC9*T zcQ2&CQy|AAWuWQ+P!SF`#Pm2oypel(Cfp51`R;KRVDry@AAktN3OKT&e1QZ=0nYrW0O$-MfAlQxo6BB)&Y$hQedQF_MfPvc zapJ!GTY%M*<2|(7ieP@PN6uJ?7hZe;z`dQ`e&cmACZN&)T%SydX%@t>>Dsisyz!@V zWcfaNh;bZh;-V4ziQ>?tCmkDsOMFaWyb8_%fqQ;+ncqQjsBahLq(;1{A)`jI+2li! zcSJ;#R0*;H3xquH{EjKl%NNUjhSIzfmY97tK)x)(X&3W?f@<%gg1 ziTR#SI>*!;xbtiB_xxPn^K;JLGr!*7J^b636Y@JLkJtIx3HkE3T&xi(JUmCU&wZk- z1mI=D1P!%w98G}Jpp#srup#2c$}B~liQ8q;npa#@wR*Fr4Qresy?^KK`)Jd6-8L@) zz$j4w=P5E)jR2r7yT@J-Vu;8=CUA*ow-sA35tRWDX^~A~2zCgYB?IImUMb9Or^Gqf z{Ll3NS}m)W(8oJeV$%!39&I{EJ|AN;2*oUq29*0T06GRa((GCqPaP80xRQ>M0kv)F zj<)-{R*Vol7^iUvy3Hra=vZD}0Vo(3`Cie$kM(ptJX|B^@7ntAn!4V;3lL}bCR=;R zKNQiuks3-MyiSLRrbwxF3reX^9k35s+&wX~BgCeVw?zg;CKW+UVJNU2?7LD_DQ=W+ zMO(`Nn>b}S;}Mj*L3}k+q-o4NU5=L0&Ul}&P+Oj52t)0+kbSvc5dW z_p?BmPt@x?^4~o7?D_eE<-&SKIWwy4Jsd>-jQpNn?~(uJXDtRsn;9%FwR9l`T3>M; z6uwH@OZ1fBzUIWOEKj8iWJ2s8LeXfOYPcYxCfAawu(SfD10Yz)NfQ0)^K^_yrZWAh z2vFN?W)?O)1^aw(_cnG~dZs;1%u#LP7ii_ zfH8gn>ZV=RCMHs+d78M`Af{`TZ#H=E;1O<7Y`A=W%smu=y2$AJC~IRV?lk7SCP3r9 z93$rnKs6B1#8E93x~{C>u(UxZ&x_~ylN5f=Gx$ED-hhI9P4j(nO$C^oisj!6$bL4G z%XVUhj@DRrbKYz@peIzMkflo*O*XI$mR0~`9Ga}D6jH#eBA&_F3z!P^)~UD~tI))$ zA@RvLX2u;_iV9q0QmbY;53ikN-DfaWR%c0{L7%02Hu97VqNbYI;o2$^#q)`u{ERZ1Fvx-v#*gA3Z{pASaE* zGgxMKk02c=#& zX&!0JVXm+{Q&vDy59{={@SH^@L>_cNJz=Ts4%dyfWr9c}qT-s!)F0iui}M5P16sFL zV^qp)dlj({$Q`yIP>)y!Nq^=GKbvN7sTu}ary7vlg{#t(^)`I&lfCub^>d9~zx;2{ z@o&EmphWpSfE$^Zs#9r=;xMySf`H)Fy0gt~jPxJ@7HTU1^ELMd`EhOQXfQN%09CFa z_i6`@WLTarLiwD#4|n+v%Fg+07)WmHF!N&lqH#8q8Y0hr6*1M(ADiong@aT;53pK# zSR3-m<-q6P0$D!6oTd1klg*E_=jT8wpm|Pyo(m#|`@4V@=Mmm0`7eV zhUu{Ga!uJUnmxb~_FMKaCnEWNxbE(o>muM9!)$nE83BHU#Mrq70EgHQv{cMha>a>s z1P6Ld=_;K|B9nI77S@rCB{IMPcDq^@c>bVE>;hEMRtE#t#Av)B%4Zk({RALv4Gz;P zT+s4ZOe@l}<`hZbP*zyqgZpTTo5#*B9Em7KoXL4Z<_&7~1aO>)o9YDp-p{@}IRVSV z-!dS}$b0^`=LkG!S&^Rw9MAD{b{M3=wK_D=js;}H6{7?Y&(KSwEM%a@Dp>$?vnU{F zWB3@vxoXhp^X1EU}cFg$txG3vS9j&@wx=$SfC_; znaz?7nzf)S*VbAyI43ln8SPd@4z&QZsXN+Trx{0sFw4nRED{yWQY>%>(^3}z=`rRR zOiFe?su1Zl0U%j!f0zDgD;wt``Lahrh!z0a*e3=U*!_^>_zVHpeKM1LKqsUs7J-Eos=TAPTRG?ugga5x!ZBf?lnRv5RtNIG zwvH95{VYu3Oh8Pshc#aV@Hiki$BY?W-7pH0ExYO7qX)6iGXSNPA1wvjgwtAf))uao z`tOlC?b!5&Dj&ONj$bY;Oj3_SU$xkfY4!2t6CiVv{^rZiF>wG*iXfKy_gUc10oQxv z$627~My?;%sNl%nS#mbsIXXdBJzG@SA)9H2h1I~hEP%;-is_~e)$mA7)=?NVm0=C2 zQWtb`GE*udy93fQf_0$^u#Ge{XTx_1B>(VkR>SO&@oN@hT5N7+Whj#vrWvb!0=1YN zr}qy&08lmwTmc9WQpdz)EH5pmhj;HG=Yk;2W{zxtQx?6hH3D&>3zad4ZVr2J2gS`M zE43wpPoNkDtRi{<3p`tK$c8Yc>b&>u@1_ecyg-t2Jzc$gC4KqJUrtxAUPXCCaYX)g z{$L{VWCMrFHbLBr>5VtvNI&u1v+2h9^Z0~7Kom9$FWXsDi=`D81$g(>z#Z$nL5f^R z{=+lK9=osZqv!ZJBId9~+oEd{@1Of@@x_o;p}H-SIuG_852dz5ZuUndvI&Rdu9zS* z*Kz{{4)YlIGzdoK}1_1wmW!+h`p4WBY z_ZMdZ4rUOXCrhH{$+p^McXb@cU6ty}Mc?&B?mM3$_xv)cq*58G+m$%o={UA!%TilyPeKOcB|*FrSqw5^VP3>IfEkHd?LTc`t+STm?&l-(d*5U97(I zKytdveGSP=IQjLZ*W#4HlmD{kKCODs`LeR!z5MnpxcR!C%y|2x9Dw7ewpZW!+~-^1 z5<4CK zA(pP*SfAqk;-D`NUIpfgs?LD9j}7oUxT9Ijq_ z?X}hO&p$Uo0gm&#iM8vWNN_)&o?O^A%l}?~56g?Vm&d_D(TCM{IFoq1np$x2x!=jb zeySXWPvT^iIeoGdjq*m^EV>t9OP0B_y86jStFM0bYpa)D{#=CxOYaO$UcY|hmowg% zp1xSvHh8rY`gjDi+h0n;ZW(Y?4fzm6@tICVQog`?Dbl@0*8Pnn-EXaz&8Oo53D2#i z^7~W;NkWq=91N_TDx4U^^>(6MkDESoVKXGz3DXedzC3YpV5$E5(3zwb0%UsRa;l%%m97+ zUWQ{vnzB!Kt9W=_5t5KZDSxL3U3lX8)w}P!SweM*EzOr02A2;%{AhITci(!eN|6~P zA^t%GO?`U*@>`=Gd>n@rHRC}z8CQc4tofVOO?c&*mm=aMM-^@ubkolRm`4-M0pc_E^Wh z(+R^15&@j-f3A_^AyPlLfJL~^>Zc)59$su|+jrgJvGFBK{E zdi7a4;O&s+#>XFaywnRVI!FDTb`rc&&_Pb{^-<}-_Ed=Y5VwkmWxpv1V=o>FqEOLm3c z1}Cth$oBo_eY)u7iG1%DQtPZW91NOe<7}WPL@P(&ESiqfBtBdg>Dpx zxGR;lHa=BecP|1=YOOCEpW1S-O|qLwway-1tk3lg^dU77RIY(+CXDO4h@RB?{>Fec z6!y92w-MKUYa;OM|Mz#_dATHAoT$HJ1q~1b!0LhJ?lhQ?g0qr6KQK-Y)*mK;v))gc-yBJlvLf%}uD(t+ z?_d%UO0%TruE%4aqEI#>z$hRhUw%!NOPmCW`|ih}1YY8B`H`1Aijy%PVW{xx1)5W<|M1uOVT`)`*W0~UT7vjzs0npjPu?I!!-t7AFP;K{358~6 zqiXhx;@Ivdd9A>Lu!VphOJ!TaZiR#=!>PU0AL}g0!*Dd#IUGC{!yMq?aR1-|C#bj+ zq=3&Z(nDID)`ZsA(w=yD7G#*SM)38020?tR zk==6;Qd?tu<+pXwi}m3-K$w)mN;ZulA!947+ddJM7yf;!aq>b$)YRu-#aLU$#NcSo z{SJUdpvpsK&Hny3Pp@8o^PSbD%kKE?7Q*X}Og zXb;4h9;CkTt|nR$`yA~F}l>X5H9u+955Y(P!T`{vor?!p|O@VSImq;g;?miJfNl*eL_j9#D%2;grIb-|e z>WPX89`yOM><#M-#>j&{dQiQ?{s)X`Sb68zisGG7~<-Kob+cS4%8-xfnLbyw55pH z-Dy2Ga}%j7^PK__Uw*ktj|C%04F$dny2Z(39&LQwYkdxiqVX8muSaU$irB$iT`zXQ z=$gUMv*yaa;DtD6*(ZRI6UCFp(_ zzu$fMhI5*o$E<(wm#FW&iL93?I_O|_zVlQ3-l-wdCh6A9q8e>4|N~i$oAlQ2F9(4J<{tnVXNcXwcT-A3d z3imcy$h<~g?XI66Bnwa?J?=*+kXgl?Q(3d?XZIh9u4F9a7~lTv%IeAf+6(n4+2(Y0 zfLlt=qxOs{nD^D8pk?O`h|$$Prvba*cgj|wS665)G zZA`SyNWx3Fx;pywlhx*#=T^tko7Hr~ISYABzVM`NM!434CqqAH;`otqNV??kGW5df zL2^0DciX>rzZ$<_=V)4GS?T}Q&N}YJ+52gJCq&e&u>Kg$gNUv0+z_kHqK-RoNdY+4 z8adDg+gH)ZTHnhTJ?b1Io*Ehas3#{2@?a*L!GX{FUXo6&U|X%$_0zVL>)Z?p4Rj6) zjQi3LRnMj-Kt>#L8Qeb^TWiFHoRHAp8Ds+js6{_n5ZtpS4gvk<<-PB{JZpQmV$8#Z zTfk;Gz+@nh0LBODBs^~ggyXlbte$9tq)e@mF(rOHXdY+~H^Uv)B1WXj794P6K-5st zp{(n|ysM2buRhG;J(jaQHW@ zEo_g+G2i@k23H&n0yDNwq?b}#_fi`!zxC#bU}a5JpXUa8@BR1NHlFJ#WJFw#;-tH& z1Wt-Skx}>3#m~is&g9UKc)N$w|9Av4=af1YVzF}1G0 zbC3J|t}UixpCHbfq)U5VWH0kPMq^y7Q^dneZ* zDO9{;^}#QHv%1!zoqFN9A$WT6tzJ$h!j-h?FD7uhDB#V0<7PuRqHi0IL%#Py+*4WX zqrWyL2rWUIh<&-|o=Gy`z!KQ`N3R|7okHUshc-SZG^bS5xS4RT2N)Ppw1_4kGffsjML~(pK+jMNZfsnOuf|^$@cHL zmbK0+)L4UNh?Wr@Tu-F?Vjzojt!8q+(T|<~JS;6-ic9vp^Fp}nlD>`mQTS#|9Q@&R zI<|b5Hp9mh;0L@r6CBQF$YL7uOE3l(yvbne^qaO!&%tc{U@sBT^C${)X~dumm@gKK zCwdmqz98-zq1Q-hl51`aNpM-0?_KNr*U0WUxu0H$#b+Xz^ZI56j}mL4ViDpQ5VA`LaY6!o_uC?@}E^8Bagu4cP_7f^u6x~;M23gk|D>xsqN1S4Lt( zTn3_J3B%5#a9WMS;4GAZu90XOgy`K5dA&?@{XC=@Lg;rFB-O5$Xpxx@V)f=~Zl+6b zU&>GVo6_p;uKtT}esLD{q_MOBDA>X*-if3{+$z`)MNi}XPrK`}ZG?CLev~ubcZ0k= z&j37LHrQeHc5lT=E{0U^TzYf$jhCKYU84%TooGMPJUdS7b{+t6|1(v0`RM&Cr9Y-l zJpasyM?r$!#2>D2KiBcB?VYn3c-OD29<&&zs>urx&V}fuPQ(bBckmPSa=qWtvrk{V zkPrHFDV#5^o_V57G5h1pw>}$mF0uB`k<(oe!0ImWjIpF!9;W_$@aB(KXTR{RG1|7` zWShYioOjBCI}u{ae|9p^XSwSzpmV$lwS6Mv=`fE1?jJBF2UpH)8I2*_!pKOH1wH$J zgd_tD7bWfB27{47XiQ#8uX0=;2#@#M!ZbYB_qF@regT@%Sm$y@3XC9> zJFSVL({t2Uk{K>7>mGwcJ@g{gVI*TPYxrzh*BM|<&g3OT6Whm%;5e6p2vo7!pu7MG zQh4Fee7GjSKZu*^bbYzk0NjfbZCX^&c&BKQq?GqR{do1GAO3oE`OQoDG!Iw*=8wO= zdhwY&0yRJ{KM^PO?DRR`E-epGdh56TX9BZb2c#6OlVs!ayJf(AoNQzF1;CyOg8Lb- zibXrZ>g9lW;mgNYZ|50UwWa)Oxdb;8v5)r3!9BRKdhw-;ZJKoE24jSX&i5H{O!9S$ z8GjGa+anh865pqr+ip?3QpG1P17eYApH*(^WMSuTyz=>oC;xAYOZVJpY_iS-1#%CJ zj)(wfezj*_X0j&IQ?+sD%Ezk@t9iWNdwCZQ!MR%?gxhhBKZwWeIRh{KUT!VUxKWS? z5+Xbnz0_3F%5Dw1|4zE~NVl?yjK^hd>F-h^XB_PgQ9U11G2g+7T&^x_?K3=p^uut1 z*XCxyu!vm5P<@I*B?WMb)D^8ed-}-;DGx?2Kvf}?jxz6c=w}DTMZ*&8B5Uo|s4H!#&}Zh(Lasf^`Jh<-1d;iZo65!RNNZet>Sk_hFuM8Io91-$p^(ZTgmi~jP( z7fXGt+-eSdJ2Dtbp4GT9L)-;+dPATjA&yJE8@}5_1}*aOCf#cD^ieQP$KAVGUE;_3 zr(%Q4X2;0=NOk$PwkKJf)|ARUeSTD#08z5@c{iM_&X;@`d~D?hkA^dA1$>Z{oxI)sRN*AeNw&_n>uj{8vGgk8Vqc71Cc0 zTNgQ8!vS2Vt5{J;w1}jQ4g-uC1)!)iF$2Rzwc<}RXhVT$@-Bl!%rNALLyaCHhDdz+ zTv?<0`5E`_eUv}8q8Lr~OcpaMb|XE>$~Y$sk38&{s}Bpc-fW-QUh(ofz4q?W31MVxETs}A-FZN8XH9RnaEvVz8fTVKaSiWi7AT# zhT5QyS|q8HENX`Amrg!AJ24*S7*{50w*|CqMXIJ`8DykDcC_61@^i+K6sQ&&J{Knj z(+4A+I$$JAdbd#TMsCUD!A$9-%@B~D>oP`b<1*IuoJPct3Z%EF%GLOex|hV^FMXng zym9-J+`Z*#JpE$rIy28f=5fHJ-05ma>?X+ z4COfTAnrW~7ooM{6-LRnNlh}oy@C^>f}|@VvzrC-`wfE|oXSNPE+@?w$w*?*eQ|0v zeQ&l~#C3??J^ga9iZq-7}w{iXMiOTlm;R zZSX-uzI4WP(8IPD(FGNIdzgw*w0_P3J}9%v84o-h=B+O7jP9|4BVRyoTd4qis|ef9 z=cACBqure45$M-=>6faYQ-v%VPURY+Yn&Dk^^!F4`5==RFJQyvgpW;cwKZHZL##OA zYNIIO2OnHsy>lbtN&j|2lOt%UYIjRsQjcsBoN^IzlZqT3v>Rh1q%{s>*vZ1oqc(N` zgC2`$CKFlP?-~EViCeQrMwP^gsU4&<81;10s`*4XoM=c7RAv4L80%#cUfG27pAd{*)wt)jPfvJjOqb-A?*C~*IL%zPWtwl7|Q45 zJRDqE{rP|Ro7HZ5@bCTZcM6H0ohYS#O+Vh2f##EO`#(<-`-SU|@!8r2{-Z~C^MOko z&;Of(LO&nVyO1=x zm>ZH>z@D&09wLJ~Psp9{^;FR}rLA&?aB(!%7Fzi%{Jh}*eZ zgX!zNkY3#Tl+r5!XRAI@uiw{*r@u!QJmfhFpmZ@&_c+$(_7p9X2?uBPz# zH#;xtrumfD95HUBHXW7>#9+B!Ed|W=Sp9h(r{{Pd@mLRv=N;aUCw6{X)9RiDPOa2% zR|{YV1BCSQdB)K1qv}Aq5V0R0!+Dv7d(GDv2jTmWNgUTL=n!)LwwkoSt#QZEh0GDH z)W+Z4{X&R%Z1q2V_l>smf7;?ZkUPD!wS79Ic`gfIDb~9Mrap_=P>Jq#tm-kzx%KAU zG%$GJ$gd158)66k8L#`PY2iOI+_lq#CS*I3sG*8I zH`~N&MiSC_sn`hCjn~4?7P-yC)anBwkB%SgW*ZO}PiPZ0Q4bQQ4H9EJmY>lEW5>*M zb6Bu7Ct!R@i4#s$bBk5KpMQ`e{Q9RABmVf);`r_U3@>e%+)epbY>QyA9qqE(l5`SP*=>v1rR;Rju*3*u5uLaaI&?iKPt!t2*>hG^GT@4r`v?P0Q6Zz@Cv zzH<|noP4-N%Z?qdi&42N5De|l_f=&-g} z%N+f3PLIdIgrzOE>!3Ua*;osWva^ekNEF< z*LpnjVMwDfJx1pBcRmyL78R3lbS};qrKVf=T{^T4pNH*?-YkEC_`jbw;JtURPT`a1 zo;ueqhX;8Xa;HV;%eE^>DxYE52{T02{5YI=L++&R+1aw2+@A8RAt-J)+kM)NF=HBd zt_g6FG7po+cf+Zpq^Q-y0Pk0r#P}ZuSN$Lgh--V0sy}JYxk6%o!4f=3cp+{&`}lbw ztQQ3K7hnxw9VPSL%kMT%64S@X#1Mt``u=;LtUiuI-o7a(bHAOeRiMe@ymeCo3a;Mb zcRyok-G>5gr!s)(ZVv;rta}L=%D0zPJpCaOIRqkymg z3CChQ9wbhE$3PGT!kHn>{A&z$xK3%(ILneSvdjmmbWrmnviGPmLSt|{}OT3u9x{k;z8ubGT~1W1M)Tg7~x#Ts}F5>^>&EGckAyL zLgJ1L?DEIU8nZV(ZalW8pDyxd+y4}?%Vl`i8F8tL z`W$?K)N$ZF#%3NSy0D_z*N!S-RqEn}bJepb_0xDJZ81RbLK*i+oMvPku6x-q+gth3 z%_;e%aJ1dwjt4lOL5Xo4wJtnA3~cp*`RC6SbU7C61v5^TOm_A}1!i<8$gt+Rl;W}S z;}{STWQaPtlYu#v=5`tKckX-^b=+%jRb_uhFIu=|)9ts|e3T(=5rxd6A&yv+3Hhu& z#+U6Ok9v<|8pC#a&t4OBkFlM|V0itl%d2mMyeD3IF%B4~N`xOxJTaokQ^60ry_Muh zg_-lp68oOVSs~tjD#iG7Q$^^!v*`7QvocH(D+C|`g2Wh^qY^ZG2;ed{mv+79nG`<7 zRR>X53N1HBd*#}rg%eqLm;OAbEs54V3Hf}IAoW#$>e7i^?j-6NUoO3}J*NCo&nfq` z9y{c5$I`w#NdO4Gcnc7MyaKAoAb-erRMhR#rH@wcUhAxiTuz>sbNg9qaJ%t{rR>F_ zFBHIlOw^Q`$QG%zRZ^6KcQ32%QM(0d`6s*!`d4efXg zWxEXq_7nD?|AHY#mW~{vQX}navkvlx^?=CbIgmsUAN(djWl*>T)erEAh(X@4o;A*W za=(E(0u*-ylzS=f*Kgfjec1W7H?A*^WXNp0G5`bKURLEKS6mMvTZH=)r3qn}`3;F! zGU=L|!BZ9ZH5V61#!l^RWFbsyW`Ivm2t`h|3*pAuFjubxZWh5*Xk$^ zA>u~QK1v_c4{@}p1X-w5jSzN?j0-n#TZ4!&b|NU*S0J_#z#)0s>2UV$nlXWm7dRNB z_W-&^N(7_l{rVAB9lbjJe^N0MamPYZbEoq?YQFB>g4`{t(w=tBZcs%U0|G+Ge69M? zhdCk;@3lca(dL50{n&U@@an@NH?EnF#>fp)?_`m@JFD-#cB%EuZC4w{PXu?%tp9#{ z*B*p;4Q7Z&nlAO-0uq#<`hRO%AAd^ELF$@t-0S{7%`%OVaoR_gyf@f{S2xB-0 z>Ed!gb}ERdBD*!PooI?#ck~y0x|W`+KTPC44@CytIst!R-f#L>1ba&-sfcr7VLHvFYbBu%BMA2;k6^p zcCQ*+9OVqE;f=xBL=4p1Zf&B*P!$}xyxsT4n^#tEf9s>wx6XaOXGB*Pw_Q+dBwcji zGL^IouwEU!GdeV%;)rAOtp9QGWxvii_(|iIIK`v93W0Nsv58(m`o=|i9E;-lE&&)K zt&w@S=rr5BLgKj}Jd!5-(m20DAkM8=5>?<-^N%dm?Ug4286lL*Z`J|SrhaYfLw#~_ zS3+#NHyFtxj>|PD+EZ~@)@AMT`4TO3TwLz8w|=#H^UBTD7oU56=1t<@ln+|Ny*S{h z5baopa^K1yOd@Rti_KGUlFF{D55#H=K>--{DJ3Za#mg{<&`5C36=G^$7pC#79!X@-Y3=CO?+C!|ZIjv+sX>kcL46 zzSXdZc7=-0@M6^Sa4o>bF$5+cgNlMSeBwHQLU9k)Lwyjfx1*pRwu9oYfAae3#b=+1 z1GoJKen847dh{gRE3_NV(To^>#}dO0qbma~;4BHwQ2t`NW^<|BC5svTH z7cSg-k=!>U@J5u_m!O{SS9m!!?P0KKr52dt;rdRA^j_@w3;A62L^KYDpuX{Y z+<^`rXUiezFH0MzOfdCqztvtd3P`z~P)h*x{wV951X+RsEbxdg#Kg_IYq&h3Rk9xp z^t=^w;M*h-T#(P=E-o(O-uHd(nWH3L!EjwUq2qx{f?yF$xXSP_u~F% z%NC&BvhYtjWievlaWVJZ`Zg9Jou9Q!uisF{^UtvNU@QBIvg_8;e(k&rt{9`CxQDZ*X6uS_ZxPp{&Z%s(etA$%I z5S5JLN`v^4KNd$j*i=*^gUz9Qjxjohqt%Dxk5lCg{Ken=boKJ@Jh%FKMU*FiQt#o$ z=(P|{Xc;0{OvGR@H{(E7y~3Z}2wAxS7mj!6`{uzvd@Mh1t2sy{s8+x`9K zzM~c^SAB%!H%3ng6?k!fWWFOb+wXSXGmdMH$DsLla?1Q=rd`Tu6Y!pHSko9O1kFP zwSVnFh=2D^#(Jp$lFD&u%n2@RwdOmBG|$EjJA+FWwoNEbG-||C7cS0zME8Q(R=GA* z6B&)Oy(;x)ulpayNRGErVDnG@)qnNs>%Y9b`q>-5X-8`wgpOV^SeJZ+jK7d1phQu! z-|3a>7YgFi05zh*eJBr7_ioe!vYF1hPnSt-jE^_qS}?{QKnM*&czRSb4QMIv0a@`j zxeZnXCxpy&qK`j(clG&~pPgi%gYM_D(xXMh>DF8bS6h#T(bi7`>?o~0Z1Y6pUaaCC zZ49VDLoICY8JvKcHGP9M4uEIlNS!M~;{IC1#0G{!#j> z`=~AbX*_WlyM9RmG^5|pbr7-fPAX^5AZXtSb|5qe<2&vpqccaz{1^bd7w9N+y#$wA ztN-=?{+rbgU#qlL{oiXbhY2-)GJ^QEgV>xA4RvV&yBS300#w>5+)W>jdkrU)$@idf z?{{(!338%9#!>c#ohC+&NkO(f7TV5J_>O?VjgUzX=l4_A` z9$UTyaB%Ad->6)kZ{E!fL$Q1NOq)$NU;gTMUVSfvjF@+JFRF~G)OpVALWF?Nngkf{ zxi=PrC4DBpY^~O2fDVZ??Vb^pHNsuC=#V%XU5}{bn#uxCE~<;>bItcm08iYUD|~nm zlJf%#eo)etd*7=6Nf#{e@Ii-+eDKl7<&@HH0F3!TDm#z4bo3b&&s-1`4!f_`Y@$9EHC&4~f%t^;vc~K_-k^)N=o>lWogB0s zn+9w!S*r{l@C=DXh1qCMxKFd}Uj&l`KxkVEiNlf-*2}mSiLlI{oH}eU z`q^!)_xJK5mHB6vHYqzrqqCdnx(`e16gED2iXVH)9$cgX;sb-)PG(RA`Ls`$fFrM& z3RbGi^x!az{}*2-d|s zuRUkYLZ0D*0axXkYuE3uew+opySv$Li>De#3-=h=EVw25|6Q|i{nDZz1(cvQ`qp9@ z<2n(PxfyTcAQ0%QX-<)yd58F^L)#3mIInX4i~gK$+iz2ASl3le+)TysorL_Iwptm} zvcImGCiadBA`GXhCWMqpuCHo-%y+mL0vdfUgfT80X9QIJPWD^fj7uzL(SSg246gD9 z+d_3~^&kK0N2?#c{=wu6Q2QRlbrCj%r|YT(jO&h@swUoXROgGr5M?tK%WV>H*q_1W`Wj+` z3XP-h@Zv?1BofC36+-d^m|vX5_?NpwEQVCibG2=Sl)|wa^|6^xd<>ozbGt@KiP&zU zHOG75&V58&n@*@)8#MYCzGhI)LI)Uo^CS`R6%(6>7)l72#Ldq9b?^m;K;O9d*OR)S9`cmt4{(mj*s^p zH{!X{De2$~vF8a1{}kRkU&1^hBKbKJY)m-U2~DSrvo<$>=SzR|s_(%n?&3E|bKS4bb4$seSa<1-Wy%gkWQN_q=utdA7AP3~-3J46X~In^6e4EFfSJ zIrOW0i6&WSfYi#&3m}W-360qR02H7}L_t)7%$D?;o&~Yj?9zKTR=;}hok=u#wgMqk z5T_|t!YF!y-8JahzL}leBj_CY?)P= zF)8+h_XwzgJrnSz|Xx1zGjG2K3&%KBo!QDJ`Gmi+(zKmL7N*rfwd_vo}qt=d2gGd|3ucJ=Fj58|pW+-=mR zU}m@T5Psha7|o(LUNFE}&4;`2K^~m_h|&o?$HE_ZxD0tZtKildbpp6zTeCxqpvlM= zCavhVkP}hy`5Mn&+sSE$~9SZ4Vx({7tFQ5?}0jcj4r~v=)J0uBl0{kFSy+V+! zxPZ-jxUm9t<}kCt0UH;p1Q2_fEX3H1#@WLt@@$z(*E;^_hd+B`^*hg>n|cS2(mltZ zs;`XO;^1bO#ArN0V|2#sh7bc)Tf7!TU(;n=vU}V-xw$RCFb#%buSa2aOFi$o^&lCp z%zw?gW~fP0Au1Sd%I>SRAaRIK90J%7iu>8@IZ?C6W3mb=?PTjg7 zKnrgE4lI1_&C9F*<*$F> z6N1H%xc96nofQEH@)$2Ki{gEA#8}dT;e8y8YLC%)t8@Zux_FpUr6MS8ZUd)bK&uiEme3*~P$zI5rQ2~~#VwleAi#o9 zEq)8tkOoML@z}!XyZbTNIrctcu}$YN%=^L5f3QKn@VPBO2QAM1l7c?Vxv=@QZ~w`w!$Ip0qPh&$g77ThdZIN`3aE)dY`~*? za#nY16-}HDvviDlO%edD1%c|jP_oak>z|W)_^APe(I%w*cfb5~g_2Kn`sGsvI|9CWg|NfDES3M<-`5j5#&P2W z5Cd^)ck&JT4ut@6ppMgV-(*;I(;`*Owe}N*HY`Hg-}WpZ4k?-3ehf16lCuE5&UjmI zeK!{0&B8VZ64rO5#j++LZ1BOAcu3aUh9vd>!yA`Y-}_~!3T55zXYo@4HnJT?)oAT- za0qrfRR+d+=)$0qupu|S7;egSQ1Ef5Lv#d1 z%-@!=&2RqxzkJmVqnOu#rSl+4TMV-`Wyu^-^RVQ-`cjUm=} z`ehIZQa}>BKa}Yoo?AOL7gz3(mfV})FXH%jzxXh?W4IVi_ecEB{-w64cxi(Gq>aia z44lBqa7G06u7i5PnAS`QqZ)X&7Pzc6nYlGKbHJh3+}N}3CsnDO@;>B0>}3l)pnlXB z72(|3^K-1isnRXCvL^Ja0;yb&v)_w>Ih3Va;^cW7 zzWDWj{_0GjnLuKLTeI$N8RRjE;YKr|hTP|=SQyf@=&3gSKJ?Xsk4{_v z&3ZUC^@Td%(d!vU|JRRyx%%J#?#;G`7a$0^8Ov*5yk}A{=6iZJ1aaynCcrDeqj0ZO z$fg`q5&g7=vlJu28fo9-c(ohCU_|CK!~_qz*G2C|aA4-e#<4*mFb1^X7fytqjXOB^ z-kNS_Xx$I7%eeGIuhzd8qY&K3PDw=*01~`#zB73UPt?uZtE9pcm57z6eXsf&o3DKJ zk6tw}BFO}Qo6-!F?gWWZ5RSK|Ita2L`dTfBTCHAw{-s&0jlLf)Z_z@2X_ItfFWfs|%mNU(n=ST6^T9!nLU;z4lCbgz zc0%fj3O&{p5yGO!FGP(K(&Hw26^!i?>7hZW?KqU{QA8WH4dDrcZ-wdfYa9psC+WVZ z)ZKo2?cMiRfA+m!t$ux_%94$VKh!hu1y_VbvyDH5W%Nw$NORB=<1qCzj)>SJqzGrE z+w>jXwEzpE_FEUkrQT_iBnB^^*+THe*nEd@s7q@#3L`);-BOzhmc%jZ_;!(7Dg|_v zQz9s4;HQG;@#bf(9+bAZ8*@6_o;n$MHf|nE^&q9F7(81N*46dBRT00XBO`@jlpyZd;0q8%o` zy!_GX-+lLItN-+)chdD5M~KMgkP$|&22Wn1VFL5FJ!nZWbK4tzbeSu=WFdNlzet)j zLK-_x3^CVhYAwKA`~5tK>dm5JE=X+1xU}cpvwU9ObESC|^|DVY-03^1cyY7ZkP*d@SbTjUQ&fnXD-ooAE)K$DKzzyvHXfJ5_CBq! z$>06`Pp3NL3(q}2&h`iVw6jPH(n7Q_cg8|L+>Z0|3ef3Bj|_39E?EN`_ga0#UY8_x zh#8@9h4HfNr>h^*H6ujg(f`Li2XVUwrZA-c)}P04slR(A2EE<>z5n#R*H-_&9ltkn z1)fVcl|p*6jjZEE=(*X<8a#$*o4X~O&~fmh`>u`I5%Y~L5YBpn6PQ|C7ld6i2j4?j z16vRnqmn_!E`i9NS&-lRA;xZWFFZmJ?#F1x4cHIVMaJ|)#$pWVc=rivq#2I7NM)Qn zmk<;4GX}oz!{9txLF;Ij#_i7Rv)g#QR^RyjKYexFI(5cJYcL*MC@{7i^0^Gq3n$!7 z=OYSdvQ2()yn&x6$5X*$6fU-H0_$f0V_7Wf!Gb76<6hJ@Dngi|JwO>^fq*Rv0Lv{L7CCi~1xA7KPQ#;(cM{ffZF|VAJ>fk^!WDjik2f zK1A=&yd}-7S&D-K3+%+Y2Mb3D51;v;RQ8e z*0N{h@lxYvVoi#QxMqHy0Ve~sR-~N!z}eAUB*NaU46wq~BRPj*NMbrZhkrYNPJc1) zdFD~!=alD>CR&H6rlu7BeACJl-f$ixTvb0V(Qn=({d z(ztE?sl&w`GipWmTeKQ28pM}M0tkE8XV=r682Q?7K3V*UH6r4s;|i;7eZT_@#leOoE6!+s zYi>Sx5%-++8ntFoZQu!prmf+?x$ADV*;B5zf6C_CnzdELH+6Y%V?U03qTipWT$pjI zOsQSFc{X2qed_6UEp+2-jGMtH0Q+kKYw-~j<~!6SiZbq# z05FpzW+!PQ{Y(_C*FhA-Yv4UgR1R(})@PI4PQny^8HFVP{o@M_1<%Gj$mQ35rba+S?lrZ( zzWWN~l58Xt+oAERZFZ}z_#6VDV(XY(V8IT4b>U^X(1JOOY9k+_#CUl6_22uG zS1mr#E37yEy5?f=xat!nTtI}|9h*7gWPne%TM&FLYR9mv`!(VlOk-=iNXA8aCB1d2 zUE4ji!BSTTwuZoM7H6VnS@q6J8_pgg06NMCEQ6oj$cdQY3W?>zw%*B1@kbN}^)``u`l}zlRjNdA>5(JGUT|<|^@HY(%gUR+FDTT)h-~?8F`^@xV5y&hQOG+9`AIQS z1oHBl4DyAn<5o0IaEPEF;$pKHXY1Oc4Knw%@sq+{>M2-@_2r)BgrLU~y3Y`Oq)UIT zDduLJf>{Dyh+UgPyNlN-HDI6?^GS{H-r?I1oK0OF3HiL6;f=A94xTwx*4*jroY|!N z&2(OJOJ{aM(2G#KU7k9;TD>Ttz46<2(=xGT3W5nZ8H9B$dQA)3BxlREtf5w8sIvAY z(GY`cZiVt6;uw#ghe(hBPU9dZkEpaifZ~1#0RWlUvWO-)$87}^#>`|p#m%|xM89rT zyaK=+%QT`F2S^O#s5rrF^lNOy6ocqiN3VYO_1~;szw}=3@09*mQH+St5RYeYD2qQk zKzbf>tX)%Mh*^jT1N^84<|z+?ScKT66s2n)tv<@o{lTw(v-)>`{gc&S|L`|e8LHY$ z+*Rc3ew>K$0vThVvq<0xQKyg{&Xw?I{PJ^KbZLt?AjadIjk<0T`a8WYvcS55!-9~c zKu8+X#B@m6LEIiB{+FN^iCLf_vD_bQRPX}ge2LcZgDiG-X{!gdFS$?TPTc`>nO@ zI-@GI_OnLKk=QpM>xJ{1?<{rya}LPGv%9x_x}q@ WdQqY64F7ci0000StO&>uS)ve<0AYj>5AR{$W90N^4L=L-RlQUJ&DC0@ZjPh;=*jPLSYvv5M~MFBAl0-BNIsH z15C~g000{K(ZT*WKal6<?_01!^k@7iDG<<3=fuAC~28EsPoqkpK{9G%|Vj005J}`Hw&=0RYXHq~ibpyyzHQsFW8>#s~laM4*8xut5h5 z!4#~(4xGUqyucR%VFpA%3?#rj5JCpzfE)^;7?wd9RKPme1hudO8lVxH;SjXJF*pt9 z;1XPc>u?taU>Kgl7`%oF1VP9M6Ja4bh!J9r*dopd7nzO(B4J20l7OTj>4+3jBE`sZ zqynizYLQ(?Bl0bB6giDtK>Co|$RIL`{EECsF_eL_Q3KQhbwIhO9~z3rpmWi5G!I>X zmZEFX8nhlgfVQHi(M#xcbO3#dj$?q)F%D*o*1Pf{>6$SWH+$s3q(pv=X`qR|$iJF~TPzlc-O$C3+J1 z#CT#lv5;6stS0Uu9wDA3UMCI{Uz12A4#|?_P6{CkNG+sOq(0IRX`DyT~9-sA|ffUF>wk++Z!kWZ5P$;0Hg6gtI-;!FvmBvPc55=u2?Kjj3apE5$3psG>L zsh-pbs)#zDT1jo7c2F-(3)vyY4>O^>2$gY-Gd%Qm(Z8e zYv>2*=jns=cMJ`N4THx>VkjAF8G9M07`GWOnM|ey)0dgZR4~^v8<}UA514ONSSt1^ zd=-((5|uiYR+WC0=c-gyb5%dpd8!Lkt5pxHURHgkMpd&=fR^vEcAI*_=wwAG2sV%zY%w@v@XU~7=xdm1xY6*0;iwVIXu6TaXrs|dqbIl~ z?uTdNHFy_3W~^@g_pF#!K2~{F^;XxcN!DEJEbDF7 zS8PxlSDOr*I-AS3sI8l=#CDr)-xT5$k15hA^;2%zG3@;83hbKf2JJcaVfH2VZT8O{ z%p4LO);n}Nd~$Sk%yw*Wyz8XlG{dRHsl(}4XB%gsbDi@w7p6;)%MzD%mlsoQr;4X; zpL)xc%+^yMd)ZNTI#eJ*$O)i@o$z8)e??LqN_gLa_%;TM>o2SC_ zkmoO6c3xRt`@J4dvz#WL)-Y|z+r(Soy~}%GIzByR`p)SCKE^%*pL(B%zNWq+-#xw~ ze%5}Oeh2)X`#bu}{g3#+;d$~F@lFL`0l@*~0lk45fwKc^10MvL1f>Tx1&sx}1}_Xg z6+#RN4Ot&@lW)Km@*DYMGu&q^n$Z=?2%QyL8~QNJCQKgI5srq>2;UHXZ>IT7>CCnW zh~P(Th`1kV8JQRPeH1AwGO8}>QM6NZadh`A)~w`N`)9q5@sFvDxjWlxwsLl7tZHmh zY-8-3xPZ8-xPf?w_(k!T5_A(J3GIpG#Ms0=iQ{tu=WLoYoaCBRmULsT<=mpV7v|~C z%bs^USv6UZd^m-e5|^?+<%1wXP%juy<)>~<9TW0|n}ttBzM_qyQL(qUN<5P0omQ3h zINdvaL;7fjPeygdGYL;pD|wL_lDQ-EO;$wK-mK5raoH_7l$?~Dqf!lNmb5F^Ft;eT zPi8AClMUo~=55LwlZVRpxOiFd;3B_8yA~shQx|tGF!j;$toK>JuS&gYLDkTP@C~gS@r~shUu{a>bfJ1` z^^VQ7&C1OKHDNXFTgC{M|V%fo{xK_dk6MK@9S!GZ*1JJzrV5xZBjOk z9!NTH<(q(S+MDf~ceQX@Dh|Ry<-sT4rhI$jQ0Sq~!`#Eo-%($2E^vo}is5J@NVEf|KK?WT&2;PCq@=ncR8z zO#GQ^T~S@VXG71PKNocFOt)Y6$@AXlk6rM*aP%VgV%sIRORYVwJx6|U{ozQjTW{-S z_si{9Jg#)~P3t?+@6&(!YQWWV*Z9{iU7vZq@5byKw{9lg9JnRA_4s!7?H6|n?o8ZW zdXIRo{Jz@#>IeD{>VLHUv1Pz*;P_y`V9&!@5AO~Mho1hF|I>%z(nrik)gwkDjgOrl z9~%uCz4Bzvli{bbrxVZ0epdf^>vOB;-~HnIOV3#R*zgPai_gEVd8zYq@2jb=I>#f& zAH2?aJ@Kaet=n_=e1k2p{&Khp%4q!80<=ef!#L zueAby{*T_~>{%>lVj?0=IZemg_wW9E&TslY!h6-e_CcyWk*pe}22Xzkl=Ymp}i_&G8;Ga>;i$_wV0+b30F& zoDje;0!`g9p7;IJ*Uz86eEOT;{?AWeeml!V%bY;OJBCfu{c%1P&NlYeE`4u%tzY-A zU(P>VE?;{;6X3)Ll;k7^*mQ06*~~a7vpA6HfV8~-*+2coyT4c{Bd61Rf0!2`QikjG zT+fe)@b2BWx3>>~R{Q08c|zd#?|*T+eS_o_;T{+e1{M0}r$2oE`~Pik>+RjQZ{Ggu z=kI>~&Fwdmku%Mdaw>_E2)ZbmjG%G50Ud}2jIY<1zyJIH@rOVB z>mNV;tG<6`5DOKIl-P_NxRiUMgY`CgyS82XsC8?751_iMxT29k)%(;lu3dAT#z#L5I6y5vf@#*(vQAsCvt9^KcWxAZd z=<~-L00VJ$1Z0352%L?PNDxuo208%+0Z~AZX%MJBKK~>?<c>soCy-B}I7OPPQY zK*E3+s;!&HbAD$r%%7Hm!BxCKn|J!z+KhBdKl@=>6`nT zU)Smto+H-OwV4v9skpfdF%mc^fg#e?UJQ@(avycAmoM=4SGHEG(5AlaSA0Z=kTd}? z0Z?F&7!r7ZwfemM7W4D6ybUbfEkL(X&tE^&*cU43PUM0{7>Fz>2@)Uy0Rm140j!_| zBSAr6VoPaEIH10)S12W4KBrNp`E+W_HcFK;2d4;f0EZx?+V*|jLa~ne<>~w9FF#B+ z2`xd#^4wZ3n469xqL0f68vf+!J|9noQIFHc)_$_8LIv@6;Gpw$O6 zBy%E_l&DM;6cCPqj)a_%%z=p`%%e)0gThn4J@Aq3CZ}6s=sV;j0hj0VOhq094FO0< z+_krUxvrzy=JfsJ58wa&|Ki&-<~L@WD@ZOcFHaWf!~M^fd^@-Y^y}q$y=>ZWs9Ov` zEcqs3s0}7ZBmx>{Omcg8=%f39unv^NVr)<2`e_V%eEwd4`Coi9y^ZdxZ7HkpverFT z5`byGJuMF?v1=W5Kd)cDo`|zPY=3Tk_EzB<1Dih#Ui~aLP0jBOyhgx{8Q* ztFkp~K80Ua@PE&1lO9Kls>6tT)2QUVkB@0AdH|OW@aiRQxXsp@x$SG zd-wLRJRIhGN|}UD)HlCoR#s8~{-4--#^sT3!0!k~m%(u_h+&%Zmr{4h`R@pLy&N8vP2htu)q zWpSYO6C6UeNG4FDVXYLLZtoD$9k|@4tF1J-=YCVIOu3jfpnM; zH}Bqk`>S9}(+Q-c6WCRDcK=z>;fXEaIVL(Cz0e}R^5D78A;5rERO*({+7DS|lA3PcX*buN32SbH5ypDKC5L z8ROf#@6^K3>g~(pm;bn)f0xjhZ$vTzIg|#iYhBlu5ph|Y0lz7C6W*XtLi35HY=t9l z*TT?Rt zk`PDCNKBLx3kCULBw+&QxfHKfu{Yb#``72|AI9~g_XZFOpdNL28Z3w0sR%}JN#&I30Koup$UGcCc)7d#@XcR+ z^YEt&XAC!0^KqJP?&s6GUoY#~gC)teUDlV6bvwfm_g0|nKWsvQj?(#c1rt9_* zFF)oGG*fMpoFqcnSSZV!$Y27=4|=+7&9}Ob>6ipR#?E79$krpJC^XSr?ic*-tt@%2 z-9n~GrsH%F$)<7H*YkB#RU*nXvm_zj&R0HqV(0-bbU4oQG#~QRo2efT_s6$)Z!t_= z-3D|81UgJ*o|lvryw^$q!4UwajuYq1B!~foNC;qoh@7NMGy58*K6YzYT}i-WG-uH<~dR*o2&ysN4pTC~}z;^=P zYQ5C<)aqFTW<}63*ewW!ri84n0d2TNU?Cz%)toc}OalPb5S&GW30^TqNi>sBp15ZQFKTpJ(YnDVq7GV-w; zg>spvKu{eBVP-`1_2u~wpMDrN_Mso2`=zdXz4o&1XqJUUmiaiR2?F+3Tkk2;&Efdw z<}Nesqju|I4u~PlWtyj=-Kl_s8$!=fr1OkMBf$VNh!vj1#AO;Cx2OlFS00^BOD6oeGSDsHG2Ox0> zjX@q}PRw&T&G{~|_-HD@%i&O#;weU7_0v88&CzT-KiBO-!+YNl%_6q*OOOksnHHj{ zjWOCwZR&0cYUa4DHhRSAbaQu_PKRlkBnP5L1YxixTc=?+aQJ5W^GsGGP1B+})FT{1 zU8}A3BKtLTIB`%Sasbarzy9XClnW9*y?lPWJe}3I`gN!PQaRnuha0cny&6KE`S#{? zGau%ZN}`;k_uj#gu=l3g&bwi^z>q8g5X8jH2uv6Vj@S*!(IeE|!@Jei_c6|+U(BqU z=?F6)Za9D~oWMf8J=Y+_Ws*A%YkgBNkJZMN!jUs(;0S}^;DJtnKmfqu-p9vr{#4Qd z@buKz`S2jL1eZE4;^9J!IE&0gF`^Rq%(**M-M|zbEZ~pOoIM~Q1Suh5pbrd0={81; z9)Kw(Za`6Fp|niXT?XlG?`U(HPf`RD2aPdWG_&v#S{oAyiALM&%e6fdh?vhMo$^Uw zY7HWS2@^~A4(4igLr?OCB^{=li3@_N`sg)_NC1x3UOrLSA_X8g=zP9>{`AShMynJ+ zIrlNNZ4ksQzPX~`^%Yyd!j!TNi3R#d#8@sr9=bV#7=&&$X#xQa~Cd`fv7}3$81|k6=A^;*H{c(dkA$i{|y+k?@ zVaZgo2#HfdjB3qd$y|Wv21M(?5Lhg` zE0Hjy!MjCQ*KtwR`OS?d`TYF(IHjx}(V-Mx3G*5`$g?EL)2+*8VesjmB|!|%)x_ckxfe1``v=|+kX4jEg2K6fsM!wueKni$bT zy8&9D12ID|5{2s6cRn4s2n9HThl9BiAtFXtgonGksy6L=Z(F-QZ$CYse>=vPKoco} ztO^`7po5eY6n)cRNmKNC)!!rLLK2{Wz?=p)h#p43D3~d5kcX)wq7y(#LP|jn3a*J$;mCPpE`{by zOeu?G5vb5pnGe(LJQc~A7=fl#4i3ZB!&Cr<1)1H<^PJvpn>0NG8m1B;WFr9plUi7) zf;Fo{$7t86&w74Wzx*&$W?jNG<<#4*U3VK`X4V|y*b>QEpEzyb+r zUoUmEK)AcR&63ua$38TttllW)?R@!UK}>fKOUg^Zv^1ttYAAV@DT}0VGxG=ta0nm> z77TSCwe_{Y5`YGd01&Q2y*J~;sc87t>$YuLn}u%c^?ZKbuTKmmPv2NP_r12!Rcl0q zBcnKk2Q!PLgKLBi$EYxnod^gDp*XMwFo;7$xCam?KzB3rFd%X<*XsL3N0veYwwqJP z#0!yw280V35ruF;kj&MMj2YPjI9%8P5rD)Afd~l+GR+;0L~;@W%OY70oDyULKuV>^ zbX$(MclWvEl1jpY39q_do}QrYf+Q?a%CS#`sSUusPw6DgQ-Xxqdxog01{i^3*ww5f z>;&7x{PqJ*x1(Z=t#1!+9&8L`UdKAlYhABACjhX0$AC+}5MW8EZ!0pV+nasg>)t(F zT`eM)S=C-%E`)S@dm3FwZwY;jaX1}$n)l7>`m#sNH+RVb8F3Tq8aX2Y*l=_8;Tn=s zK&V^Qj;N?k3c>(x%w!I2Z^Ij6AFhx4dEYm%(c5}CKfOHvxLv+vJG{C5rAgx14AMw6aW{}#ECcs1rw&6C6ka(2_32v zp)kaj2oCez&Efs+%{$5&5SWP>s%_Wxsa-GRp13H+-d}XThyex%8c`igq$g<2gCk8+a$%znCDP^7!ajdPk>Ip!&t4m^NWBdBNZGFDI zJ)Caby^Rgb9k9E5_^4M3%Vo-#0igR(w=st5a5M0@wC&6D*KNJ@(Vw0_KYsd$e)$Bg znVRsCh+nQh?b{20MVN<27`R6SF{Q-AQ3tw%0XmX_CB$&-uuD>53@{G>hzNC82X#Z! zu<&wtKlvT_Ffa;aUf%!WKbN=feswv2y*_>)-2;+GphHeGOS+D)C`L%}dPfAK0|9wJ zP^aWzO9W(*jHsgz2kTmLK}%Wk-Oc^!aDR8YqtJGKWEzdoi6BHcvCuwh>*_viyr6Fp z36o4|0t=72*1fmqoHtL$!DEYO!+u)cCWUFahw9S3?w94}__J@{>1EsYeVC`fcXtoh zL)AXUWqY{4&qadSy1zU>)z)&(;eJ^!*YkCnmy#zy<^aTq-n5;)CXnJJAdK$&^W){~ zSIW89y$|IwL3}x$ZuRDnkhOn){_^?!<;M?y@~7YZ`j>M#Y`U)7MZ*I$e1tM5AUI5R-QsD$+P7Bs zbz7$N@aFCN+xzY5aeI21N-6Vn|M13#rET5fdVG6GiMQ*^)0e09a<*;u3{PIkM>K^V$1QEy6>Gcz!j9}qD z=gbJ_{hCG6R5+F6;Ur}uPDLh48H7MGco0KMqTBgD{muXK@ejXy|BG*39>1LbHj{gP z*ryxdQp(-w^f09ZVpfQ*qi*$b`RV1$?^_*^qW4SPzIFgc34&0I@Me44D+UUu@IWLa zb_k42*j@M5zLMMu9MIGO04>_y`<_Zb26vSRL>k&{XaG8Lb=Vz#dGqHVPJgwVaORR> zGM(nT)~+H*C?cu2r0x!&)`u`GG83?3OTdXi9hg!Q&q0b%PpPty>B*+=blT@%5) z%u3Pfn3fwGetrH}>ow&mPs{!7gNNJD2y;`lp7WF>t==m*kFI@0-$rko4sDoUpHfL- zo`aGAvL$Dgj3 z&)0p$d^*Z4?JZ@%LI~Ky-DvFRR=NTpF&SQXM_o#&g#5;H7 zK*AV9t#yZd`1w*Ee3+*8aQOCi{*61|9#4rg1f?>Q6>}E3&k*>Bpu33OCH|Z&aO2^ zjrg>;r+IeO9^}(>lM}P4c6AhWO?i^TZBEO>&71wQ|LcGL|NPtk>0i|Cd^_jqbS(4y z_U`W6Z{FXXPNy=ZIj6axyH1&elS*5+F?7zepMZK_x9c=bgVOWUM^G*Ea(jP&yt}Vk zYrAGD+-Gmg5K4b;C8n6HdszwiVV#qRHGJ&wu~r%imru>wdjn_UA~;hqwRie*P{A zpu1x*o7%RvPs3iinmeRYJik@=KsKYczXKnHnt_-x65T(j!Rl{6yj3mnUV|uf$(q-kBA`TOtfT~CC@3V4G0^6 z09Z&o0Fl#lOneJE33*AV-gV!BA(x`47t?#3nQqK5ph^gW<0Q*GgOwe+4Qzxwq_oP> zR~zAt95D)okel`nN zWnzi|kCd_-I-s!R#AtnV$F4{XkjP}{Mobu>JzDeAH{brFfA)X<$G`kPEk4zu_sa*6 z*!FgnN8!LxIWY^ODW{2$q+2u9ARZkOfFLEJOvyX~g8(LGBvB1XFw4P=oq|&Y5E-Qq z6?5D*ssWW$K%?$8V@&hOzy>H1A%F-*mR%7LOTOjf&qw6+}TSEYVKy*fDE`e^NKR8D&U|0cC)dNwe2n9}{}n3jBa`trS@L3DE07(NJtrCs;#z123> zD_bjgEKxAR)lHQVOUi&T5EIR=5Nh-0tS zOtOx#1Im<7;4$35Enu^e4FNDA0kI%z>z5h&X#4;C|NR$#`>+1RJmu5PX+J+7mh^CQ z{P6DGhi`s%fB)w8{yuSXw^E8>Oi-*F0tH0fn-E*#oA5%=@8p0|`sS{8$mI8SBLy(aY0_rB{? zmSaj^KYjddS^nam{K;J8vVD)b8XOe5#q;(2Nt+39LpKBnLu4js+jF2y=@6KZ6N!Y6 zKvzRV%(P7TK!D1F9L#;7qhaCK<6BxcBv*>z=4L_g?T7#R+uQ#t4Qg}-me8lu#1KPYVF-W|0U$Qh)-A~-l+{GX5SdCzT6Yoo z-QWHDzy8<%ugvXmd%Le!1irmHeDnVOFFySI?Yp=0;g|uXOq3EZg4t-DgoQ=NSfmUQ zNvZAC-Iw`*4qLk>8~1bWnfBftKt!(FHnt560H9jmcAXtW8WRyp5uSu4Pg(LZ&uw&X zdYq>~9IYZF(LCRu(qYtnqd?!T>-u8WA>!lje?ZK?`A0t!x!L!xAp2H7)%{6(4Rc5E z2=~lfx@mWv^R3_%h=I)w0OVCmM`WoiGu$9JkU=0|0E08>h;R!q&IuX;wj$h`$G~*E z+`joUhOE$*`NS*;gyzG;Os8pCZtsDHRSZYs1gvX&ku~aedD889<}8RDN)A3H7fHzB zhM{4;*)D0uJQ0QWsw|PIO!LF^b~(K*^CHBHBmzy#X`!s+X%HnY;sOZ4;2sd<-pw6R zJfsir-I>unv^7Er2+8x^;f-GPZ~v$N5pleE|IoUk+hLmSk9WWP?$_UZ`?J&CiBiEJ zPFZpaA~DFEP1Q{~3lVv3nSfJjqo3wXqms&C=F{D!?yeRDbqutTA;zd<@8|%CZFEnB zEEG%_X(_~cc;g6bO;A{ba@wv}94$w5PV@10o~EhHX`S)q%hS`Nd4TUf{Pyov`_KP$ zzVsjV{u5h4YkaVJdUEo6^xot^14HWoXl`?uy_We^#>~Uj~ zj}ICqp^PI*p7LUui3#1wOt%e6p6=fr_(;)pybwijy1#$_&D-yer?u zn>T+JNL%kIACkwu?>Tw6hU@D9W(;sEEOYloO~D%pbZ?nwM*h41^?zGGe|&p?=Yg&6 z$7T81hxgxn_;7c3d%C@u<^xILT$lt95hxM@c&nS4GK=XzroPn*Js|?&@i4IuNOwUL zgw{47OOgc+V@g*taGN+uVpH#bOJuFNGm{VeKMeCgtCy{2%lV@hr>>ALCCG*TAUeYktHm2K~o8#dq z%Mu9?AS|s;2EHhJ>_O1%(=k@Av zSC)I{1icwvx?KTGh`)aSmB zF$@GgyhuiP&?|!jN zZ_KdimdZB@>FM$Bp+BQGcV{FGH^AuQ>(}o;U4D9hcq2f&kBkI>QVJBgY>$_h&x<*_ zb{L8HI4|4fQS$Aqtri3W2v=mnL|o=V$pIO}ayi_-I~;DYYwcUgxa31Q-Q2%@mn7pK zMMD|ggW=*;s1Pr8--FbAU#-*4JyNt5Lgd!l=w0`wUE~Hl5oN)IB#uH*_w(iBA3nJu zCN}LrbbELI=YRTV?>~Gvo{p!}G0zL-NjNdEhetRf5Rw5h3L#oFk6<}Wr=(@~ra9-) zswAns;TUtt!O=nmqZ4$-$R(E2``W7}Ax;z|riAE*K@>vLgi21#8M!1$YO&07Io=S0 zv`quRVP58v6Vl$bs_M)6jJEIBuRr|0{N~U83q8DC_usKBhnrvCAHFNgyW{Q8Zs&Ki zEWqFh5pc~*IQ071^jRp)Wv<#B`m#LS-T(6L_Ro*=8^UaA84l^~O-gr^Pd|MBZ@hg8 zMv#I5(tRM9?oVI7|8#n@9Dg}eQdci1I6@osE%)9*5sl4lIAECWvdl|IBI^cx9o0GA zKfJ%af5%95uiN>tEX$jBKY#b|3j%$v&p;yq6*(;B7BRq9w>EJ?N+y`C%giJ|IKaXTg9s4Z+>DSA2^oDDBN69tV~61umN=ty4}+h~Wt_z)gW+ z66i`g!jNGWU`Pf;l5xrl5K96QfJ7-Tl8GhIQkjI8$-y~;2<6=RIAm!{-h0~WI#iz@ zzyA2s;oWzCL8+?8;rLA{w+|2B9;f$HK2Bv4P7WS{%>$Z%)7H)}Nn$-?wK30)e3pf&`JET_1n@I~D*tEH?}hJ_=8Bnr2S( z@vd7{+jl*Wx}s?~h)OsCLoy31^8Po!{>xwg*KZy^AnMEI@%NwqAe_E=`|DG_b@0Q@ zo17&T9M;CZTfdNLIvnymAs`SU-a@#$V;`<+ee1e~RX~=KhHh;<1MrmIhG8F@l1v$L zDd5#?3xmtcUPmuUjtOp#hg_D^>2x?Arg<*Yl*^Q*sJjkDW@d%}Hv<3&4}fqFKty!6 z&_DGdyguA5`_{g;{{3}d3+IG4S?-tlU8bbHCIlktWNJOa=W@49Z?2M(SVRvs z5@eR-&G#?cSNVtS%Y1sXq~nt3hd1Be=0nB|0zxq`U$ab zefZTs`zQbHpMUqC-OjfOqsI7CyN*`(&zm({PUSdHNqD(A0`trH)2A;jd72NWk{1YI zbYaS!$@QzeaO^>&!wAwg>w9UY08k=+7!Vl+!_1fm`Cw3I1NvxuB- zPKVR+a6HV*LXr%@5D<_8!W{y_DG3F-+E5*4&dkJYeRv=4&N)S^+qRD0LWd6x*Fb|X zVvryU^dKgplqj>fGa?{&HN;>J!4#BuDk)FqtcXFJb4DTnkW}2s!dRHI6q%QL=srF? zd~oCQR^8EB|MZ9L{f9sO)zAKkQXG!&r+ho5GEKSUi2y}h)tYTcE|jKnm-8VoSeQvL zW#Qso)mj7t00S_2GXzSKX5s>PdBy06Qqlnxy>Devs%t8z!z4HJ;glwh2y4UvfZza1 zoDXRk?ddFk_T4}Eq1^oL$GI6HZ4m-Ap5ISz4E=lhVPCiV`_s%x zSf+W-Ij52=k=GR=5r{|%gjB@FZZ>Sy-q+DahoCZuz`%!HAwbP-UnxKclVpHa>uTO} zp1*u)PtWI@C7q7PMCs=C{;*8P<033&Iv|Uy1sI6{ppOAW#3F(Lq3&iDfs!P`M{n9% zgaZKdzP>z;y1NZ+HFS7%)f#T07OnvRL_i6g!HtN7XqYDhCqzyT%Y=B64s !*orE}p@XTB!W#(bH-yXiZ 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