From 2d4a387fc67e53ba39497018abd736e9e0308d6f Mon Sep 17 00:00:00 2001 From: Magnus Wahlberg Date: Fri, 22 Dec 2023 11:43:15 +0100 Subject: [PATCH 1/3] Fix failing snakemake smoke tests (#47) * Update github-actions.yml * Update github-actions.yml * Update github-actions.yml * Revert "Update github-actions.yml" This reverts commit b8b82e568f823fa8030e244c7b2515b7ad9d3813. * Revert "Update github-actions.yml" This reverts commit d27e6a8b9d65561717536cd2111f8b91891cc558. * Revert "Update github-actions.yml" This reverts commit 6cde84f1188a7f3aae910c72b1c53fbd9c80f219. * Update github-actions.yml * Update github-actions.yml --- .github/workflows/github-actions.yml | 95 +++++++++++++++++----------- 1 file changed, 59 insertions(+), 36 deletions(-) diff --git a/.github/workflows/github-actions.yml b/.github/workflows/github-actions.yml index 2f686b17..5b3ff8a6 100644 --- a/.github/workflows/github-actions.yml +++ b/.github/workflows/github-actions.yml @@ -8,26 +8,33 @@ jobs: steps: - name: Check out repository code uses: actions/checkout@v3 - - name: Training Association Testing smoke test - uses: snakemake/snakemake-github-action@v1.24.0 + - uses: mamba-org/setup-micromamba@v1.4.3 with: - directory: 'example' - snakefile: 'pipelines/training_association_testing.snakefile' - args: '-j 2 -n' + environment-name: deeprvat-gh-action + environment-file: ${{ github.workspace }}/deeprvat_env_no_gpu.yml + cache-environment: true + cache-downloads: true + - name: Smoketest training_association_testing pipeline + run: | + python -m snakemake -n -j 2 --directory ${{ github.workspace }}/example \ + --snakefile ${{ github.workspace }}/pipelines/training_association_testing.snakefile --show-failed-logs + shell: micromamba-shell {0} - name: Link pretrained models run: cd ${{ github.workspace }}/example && ln -s ../pretrained_models - - name: Association Testing Pretrained Smoke Test - uses: snakemake/snakemake-github-action@v1.24.0 - with: - directory: 'example' - snakefile: 'pipelines/association_testing_pretrained.snakefile' - args: '-j 2 -n' - - name: Seed Gene Discovery Smoke Test - uses: snakemake/snakemake-github-action@v1.24.0 - with: - directory: 'example' - snakefile: 'pipelines/seed_gene_discovery.snakefile' - args: '-j 2 -n' + shell: bash -el {0} + - name: Smoketest association_testing_pretrained pipeline + run: | + python -m snakemake -n -j 2 --directory ${{ github.workspace }}/example \ + --snakefile ${{ github.workspace }}/pipelines/association_testing_pretrained.snakefile --show-failed-logs + shell: micromamba-shell {0} + - name: Copy seed gene discovery snakemake config + run: cd ${{ github.workspace }}/example && cp ../deeprvat/seed_gene_discovery/config.yaml . + shell: bash -el {0} + - name: Smoketest seed_gene_discovery pipeline + run: | + python -m snakemake -n -j 2 --directory ${{ github.workspace }}/example \ + --snakefile ${{ github.workspace }}/pipelines/seed_gene_discovery.snakefile --show-failed-logs + shell: micromamba-shell {0} DeepRVAT-Pipeline-Tests: runs-on: ubuntu-latest @@ -76,21 +83,32 @@ jobs: steps: - name: Check out repository code uses: actions/checkout@v3 - - name: Preprocessing Smoke Test With QC - uses: snakemake/snakemake-github-action@v1.24.0 + - uses: mamba-org/setup-micromamba@v1.4.3 with: - directory: 'example/preprocess' - snakefile: 'pipelines/preprocess_with_qc.snakefile' - args: '-j 2 -n --configfile pipelines/config/deeprvat_preprocess_config.yaml' - stagein: 'touch example/preprocess/workdir/reference/GRCh38.primary_assembly.genome.fa' + environment-name: deeprvat-preprocess-gh-action + environment-file: ${{ github.workspace }}/deeprvat_preprocessing_env.yml + cache-environment: true + cache-downloads: true - - name: Preprocessing Smoke Test No QC - uses: snakemake/snakemake-github-action@v1.24.0 - with: - directory: 'example/preprocess' - snakefile: 'pipelines/preprocess_no_qc.snakefile' - args: '-j 2 -n --configfile pipelines/config/deeprvat_preprocess_config.yaml' - stagein: 'touch example/preprocess/workdir/reference/GRCh38.primary_assembly.genome.fa' + - name: Fake fasta data + if: steps.cache-fasta.outputs.cache-hit != 'true' + run: | + cd ${{ github.workspace }}/example/preprocess && touch workdir/reference/GRCh38.primary_assembly.genome.fa + + - name: Run preprocessing pipeline no qc Smoke Test + run: | + python -m snakemake -n -j 2 --directory ${{ github.workspace }}/example/preprocess \ + --snakefile ${{ github.workspace }}/pipelines/preprocess_no_qc.snakefile \ + --configfile ${{ github.workspace }}/pipelines/config/deeprvat_preprocess_config.yaml --show-failed-logs + shell: micromamba-shell {0} + + + - name: Preprocessing pipeline with qc Smoke Test + run: | + python -m snakemake -n -j 2 --directory ${{ github.workspace }}/example/preprocess \ + --snakefile ${{ github.workspace }}/pipelines/preprocess_with_qc.snakefile \ + --configfile ${{ github.workspace }}/pipelines/config/deeprvat_preprocess_config.yaml --show-failed-logs + shell: micromamba-shell {0} DeepRVAT-Annotation-Pipeline-Smoke-Tests: @@ -98,19 +116,24 @@ jobs: steps: - name: Check out repository code uses: actions/checkout@v3 - - name: Annotations Smoke Test - uses: snakemake/snakemake-github-action@v1.25.1 + - uses: mamba-org/setup-micromamba@v1.4.3 with: - directory: 'example/annotations' - snakefile: 'pipelines/annotations.snakefile' - args: '-j 2 -n --configfile pipelines/config/deeprvat_annotation_config.yaml' + environment-name: deeprvat-preprocess-gh-action + environment-file: ${{ github.workspace }}/deeprvat_preprocessing_env.yml + cache-environment: true + cache-downloads: true + - name: Annotations Smoke Test + run: | + python -m snakemake -n -j 2 --directory ${{ github.workspace }}/example/annotations \ + --snakefile ${{ github.workspace }}/pipelines/annotations.snakefile \ + --configfile ${{ github.workspace }}/pipelines/config/deeprvat_annotation_config.yaml --show-failed-logs + shell: micromamba-shell {0} DeepRVAT-Preprocessing-Pipeline-Tests-No-QC: runs-on: ubuntu-latest needs: DeepRVAT-Preprocessing-Pipeline-Smoke-Tests steps: - - name: Check out repository code uses: actions/checkout@v3 - uses: mamba-org/setup-micromamba@v1.4.3 From fc14c51a362418f11212f84ba49593b3b363c75c Mon Sep 17 00:00:00 2001 From: Kayla Meyer <129152803+meyerkm@users.noreply.github.com> Date: Fri, 22 Dec 2023 11:44:46 +0100 Subject: [PATCH 2/3] Feature snakemake modular (#41) * making snakemake runners modular Additional snakemake runners for only running training, only association testing, and full train + association testing pipelines. * bug-fix pretrained model path * Adding additional snakemake pipeline run option to readthedocs * update train snakefile pipeline from PR #42 * bug-fix model path for snakemake pipeline runners * bug-fix f string syntax * Update github-actions.yml * Update github-actions.yml * Update github-actions.yml * Revert "Update github-actions.yml" This reverts commit b8b82e568f823fa8030e244c7b2515b7ad9d3813. * Revert "Update github-actions.yml" This reverts commit d27e6a8b9d65561717536cd2111f8b91891cc558. * Revert "Update github-actions.yml" This reverts commit 6cde84f1188a7f3aae910c72b1c53fbd9c80f219. * Update github-actions.yml * Update github-actions.yml * fix-model path string variable in rules --------- Co-authored-by: Magnus Wahlberg --- docs/usage.md | 14 + .../association_dataset.snakefile | 12 + .../association_testing/burdens.snakefile | 74 +++++ .../regress_eval.snakefile | 63 ++++ .../association_testing_pretrained.snakefile | 188 +----------- pipelines/run_training.snakefile | 51 +++ pipelines/training/config.snakefile | 29 ++ pipelines/training/train.snakefile | 64 ++++ pipelines/training/training_dataset.snakefile | 37 +++ .../training_association_testing.snakefile | 290 +----------------- 10 files changed, 363 insertions(+), 459 deletions(-) create mode 100644 pipelines/association_testing/association_dataset.snakefile create mode 100644 pipelines/association_testing/burdens.snakefile create mode 100644 pipelines/association_testing/regress_eval.snakefile create mode 100644 pipelines/run_training.snakefile create mode 100644 pipelines/training/config.snakefile create mode 100644 pipelines/training/train.snakefile create mode 100644 pipelines/training/training_dataset.snakefile diff --git a/docs/usage.md b/docs/usage.md index 93361782..5d7c9170 100644 --- a/docs/usage.md +++ b/docs/usage.md @@ -56,6 +56,20 @@ Replace `[path_to_deeprvat]` with the path to your clone of the repository. Note that the example data is randomly generated, and so is only suited for testing whether the `deeprvat` package has been correctly installed. +### Run the training pipeline on some example data + +```shell +mkdir example +cd example +ln -s [path_to_deeprvat]/example/* . +snakemake -j 1 --snakefile [path_to_deeprvat]/pipelines/run_training.snakefile +``` + +Replace `[path_to_deeprvat]` with the path to your clone of the repository. + +Note that the example data is randomly generated, and so is only suited for testing whether the `deeprvat` package has been correctly installed. + + ### Run the association testing pipeline with pretrained models ```shell diff --git a/pipelines/association_testing/association_dataset.snakefile b/pipelines/association_testing/association_dataset.snakefile new file mode 100644 index 00000000..0e63e53f --- /dev/null +++ b/pipelines/association_testing/association_dataset.snakefile @@ -0,0 +1,12 @@ + +rule association_dataset: + input: + config = '{phenotype}/deeprvat/hpopt_config.yaml' + output: + '{phenotype}/deeprvat/association_dataset.pkl' + threads: 4 + shell: + 'deeprvat_associate make-dataset ' + + debug + + '{input.config} ' + '{output}' diff --git a/pipelines/association_testing/burdens.snakefile b/pipelines/association_testing/burdens.snakefile new file mode 100644 index 00000000..550390fa --- /dev/null +++ b/pipelines/association_testing/burdens.snakefile @@ -0,0 +1,74 @@ + +rule link_burdens: + priority: 1 + input: + checkpoints = lambda wildcards: [ + f'{model_path}/repeat_{repeat}/best/bag_{bag}.ckpt' + for repeat in range(n_repeats) for bag in range(n_bags) + ], + dataset = '{phenotype}/deeprvat/association_dataset.pkl', + data_config = '{phenotype}/deeprvat/hpopt_config.yaml', + model_config = model_path / 'config.yaml', + output: + '{phenotype}/deeprvat/burdens/chunk{chunk}.linked' + threads: 8 + shell: + ' && '.join([ + ('deeprvat_associate compute-burdens ' + + debug + + ' --n-chunks '+ str(n_burden_chunks) + ' ' + f'--link-burdens ../../../{phenotypes[0]}/deeprvat/burdens/burdens.zarr ' + '--chunk {wildcards.chunk} ' + '--dataset-file {input.dataset} ' + '{input.data_config} ' + '{input.model_config} ' + '{input.checkpoints} ' + '{wildcards.phenotype}/deeprvat/burdens'), + 'touch {output}' + ]) + +rule compute_burdens: + priority: 10 + input: + reversed = model_path / "reverse_finished.tmp", + checkpoints = lambda wildcards: [ + model_path / f'repeat_{repeat}/best/bag_{bag}.ckpt' + for repeat in range(n_repeats) for bag in range(n_bags) + ], + dataset = '{phenotype}/deeprvat/association_dataset.pkl', + data_config = '{phenotype}/deeprvat/hpopt_config.yaml', + model_config = model_path / 'config.yaml', + output: + '{phenotype}/deeprvat/burdens/chunk{chunk}.finished' + threads: 8 + shell: + ' && '.join([ + ('deeprvat_associate compute-burdens ' + + debug + + ' --n-chunks '+ str(n_burden_chunks) + ' ' + '--chunk {wildcards.chunk} ' + '--dataset-file {input.dataset} ' + '{input.data_config} ' + '{input.model_config} ' + '{input.checkpoints} ' + '{wildcards.phenotype}/deeprvat/burdens'), + 'touch {output}' + ]) + +rule reverse_models: + input: + checkpoints = expand(model_path / 'repeat_{repeat}/best/bag_{bag}.ckpt', + bag=range(n_bags), repeat=range(n_repeats)), + model_config = model_path / 'config.yaml', + data_config = Path(phenotypes[0]) / "deeprvat/hpopt_config.yaml", + output: + temp(model_path / "reverse_finished.tmp") + threads: 4 + shell: + " && ".join([ + ("deeprvat_associate reverse-models " + "{input.model_config} " + "{input.data_config} " + "{input.checkpoints}"), + "touch {output}" + ]) \ No newline at end of file diff --git a/pipelines/association_testing/regress_eval.snakefile b/pipelines/association_testing/regress_eval.snakefile new file mode 100644 index 00000000..bcb3f369 --- /dev/null +++ b/pipelines/association_testing/regress_eval.snakefile @@ -0,0 +1,63 @@ + +rule evaluate: + input: + associations = expand('{{phenotype}}/deeprvat/repeat_{repeat}/results/burden_associations.parquet', + repeat=range(n_repeats)), + config = '{phenotype}/deeprvat/hpopt_config.yaml', + output: + "{phenotype}/deeprvat/eval/significant.parquet", + "{phenotype}/deeprvat/eval/all_results.parquet" + threads: 1 + shell: + 'deeprvat_evaluate ' + + debug + + '--use-seed-genes ' + '--n-repeats {n_repeats} ' + '--correction-method FDR ' + '{input.associations} ' + '{input.config} ' + '{wildcards.phenotype}/deeprvat/eval' + +rule all_regression: + input: + expand('{phenotype}/deeprvat/repeat_{repeat}/results/burden_associations.parquet', + phenotype=phenotypes, type=['deeprvat'], repeat=range(n_repeats)), + +rule combine_regression_chunks: + input: + expand('{{phenotype}}/deeprvat/repeat_{{repeat}}/results/burden_associations_{chunk}.parquet', chunk=range(n_regression_chunks)), + output: + '{phenotype}/deeprvat/repeat_{repeat}/results/burden_associations.parquet', + threads: 1 + shell: + 'deeprvat_associate combine-regression-results ' + '--model-name repeat_{wildcards.repeat} ' + '{input} ' + '{output}' + +rule regress: + input: + config = "{phenotype}/deeprvat/hpopt_config.yaml", + chunks = lambda wildcards: expand( + ('{{phenotype}}/deeprvat/burdens/chunk{chunk}.' + + ("finished" if wildcards.phenotype == phenotypes[0] else "linked")), + chunk=range(n_burden_chunks) + ), + phenotype_0_chunks = expand( + phenotypes[0] + '/deeprvat/burdens/chunk{chunk}.finished', + chunk=range(n_burden_chunks) + ), + output: + temp('{phenotype}/deeprvat/repeat_{repeat}/results/burden_associations_{chunk}.parquet'), + threads: 2 + shell: + 'deeprvat_associate regress ' + + debug + + '--chunk {wildcards.chunk} ' + '--n-chunks ' + str(n_regression_chunks) + ' ' + '--use-bias ' + '--repeat {wildcards.repeat} ' + + do_scoretest + + '{input.config} ' + '{wildcards.phenotype}/deeprvat/burdens ' #TODO make this w/o repeats + '{wildcards.phenotype}/deeprvat/repeat_{wildcards.repeat}/results' \ No newline at end of file diff --git a/pipelines/association_testing_pretrained.snakefile b/pipelines/association_testing_pretrained.snakefile index 702302f0..d7aaa006 100644 --- a/pipelines/association_testing_pretrained.snakefile +++ b/pipelines/association_testing_pretrained.snakefile @@ -5,19 +5,27 @@ configfile: 'config.yaml' debug_flag = config.get('debug', False) phenotypes = config['phenotypes'] phenotypes = list(phenotypes.keys()) if type(phenotypes) == dict else phenotypes +training_phenotypes = config["training"].get("phenotypes", phenotypes) n_burden_chunks = config.get('n_burden_chunks', 1) if not debug_flag else 2 n_regression_chunks = config.get('n_regression_chunks', 40) if not debug_flag else 2 +n_trials = config['hyperparameter_optimization']['n_trials'] n_bags = config['training']['n_bags'] if not debug_flag else 3 n_repeats = config['n_repeats'] debug = '--debug ' if debug_flag else '' do_scoretest = '--do-scoretest ' if config.get('do_scoretest', False) else '' -pretrained_model_path = Path(config.get("pretrained_model_path", "pretrained_models")) +tensor_compression_level = config['training'].get('tensor_compression_level', 1) +model_path = Path(config.get("pretrained_model_path", "pretrained_models")) wildcard_constraints: repeat="\d+", trial="\d+", +include: "training/config.snakefile" +include: "association_testing/association_dataset.snakefile" +include: "association_testing/burdens.snakefile" +include: "association_testing/regress_eval.snakefile" + rule all: input: expand("{phenotype}/deeprvat/eval/significant.parquet", @@ -25,69 +33,6 @@ rule all: expand("{phenotype}/deeprvat/eval/all_results.parquet", phenotype=phenotypes) -rule evaluate: - input: - associations = expand('{{phenotype}}/deeprvat/repeat_{repeat}/results/burden_associations.parquet', - repeat=range(n_repeats)), - config = '{phenotype}/deeprvat/hpopt_config.yaml', - output: - "{phenotype}/deeprvat/eval/significant.parquet", - "{phenotype}/deeprvat/eval/all_results.parquet" - threads: 1 - shell: - 'deeprvat_evaluate ' - + debug + - '--use-seed-genes ' - '--n-repeats {n_repeats} ' - '--correction-method FDR ' - '{input.associations} ' - '{input.config} ' - '{wildcards.phenotype}/deeprvat/eval' - -rule all_regression: - input: - expand('{phenotype}/deeprvat/repeat_{repeat}/results/burden_associations.parquet', - phenotype=phenotypes, type=['deeprvat'], repeat=range(n_repeats)), - -rule combine_regression_chunks: - input: - expand('{{phenotype}}/deeprvat/repeat_{{repeat}}/results/burden_associations_{chunk}.parquet', chunk=range(n_regression_chunks)), - output: - '{phenotype}/deeprvat/repeat_{repeat}/results/burden_associations.parquet', - threads: 1 - shell: - 'deeprvat_associate combine-regression-results ' - '--model-name repeat_{wildcards.repeat} ' - '{input} ' - '{output}' - -rule regress: - input: - config = "{phenotype}/deeprvat/hpopt_config.yaml", - chunks = lambda wildcards: expand( - ('{{phenotype}}/deeprvat/burdens/chunk{chunk}.' + - ("finished" if wildcards.phenotype == phenotypes[0] else "linked")), - chunk=range(n_burden_chunks) - ), - phenotype_0_chunks = expand( - phenotypes[0] + '/deeprvat/burdens/chunk{chunk}.finished', - chunk=range(n_burden_chunks) - ), - output: - temp('{phenotype}/deeprvat/repeat_{repeat}/results/burden_associations_{chunk}.parquet'), - threads: 2 - shell: - 'deeprvat_associate regress ' - + debug + - '--chunk {wildcards.chunk} ' - '--n-chunks ' + str(n_regression_chunks) + ' ' - '--use-bias ' - '--repeat {wildcards.repeat} ' - + do_scoretest + - '{input.config} ' - '{wildcards.phenotype}/deeprvat/burdens ' #TODO make this w/o repeats - '{wildcards.phenotype}/deeprvat/repeat_{wildcards.repeat}/results' - rule all_burdens: input: [ @@ -97,97 +42,11 @@ rule all_burdens: for c in range(n_burden_chunks) ] -rule link_burdens: - priority: 1 - input: - checkpoints = lambda wildcards: [ - f'{pretrained_model_path}/repeat_{repeat}/best/bag_{bag}.ckpt' - for repeat in range(n_repeats) for bag in range(n_bags) - ], - dataset = '{phenotype}/deeprvat/association_dataset.pkl', - data_config = '{phenotype}/deeprvat/hpopt_config.yaml', - model_config = pretrained_model_path / 'config.yaml', - output: - '{phenotype}/deeprvat/burdens/chunk{chunk}.linked' - threads: 8 - shell: - ' && '.join([ - ('deeprvat_associate compute-burdens ' - + debug + - ' --n-chunks '+ str(n_burden_chunks) + ' ' - f'--link-burdens ../../../{phenotypes[0]}/deeprvat/burdens/burdens.zarr ' - '--chunk {wildcards.chunk} ' - '--dataset-file {input.dataset} ' - '{input.data_config} ' - '{input.model_config} ' - '{input.checkpoints} ' - '{wildcards.phenotype}/deeprvat/burdens'), - 'touch {output}' - ]) - -rule compute_burdens: - priority: 10 - input: - reversed = pretrained_model_path / "reverse_finished.tmp", - checkpoints = lambda wildcards: [ - pretrained_model_path / f'repeat_{repeat}/best/bag_{bag}.ckpt' - for repeat in range(n_repeats) for bag in range(n_bags) - ], - dataset = '{phenotype}/deeprvat/association_dataset.pkl', - data_config = '{phenotype}/deeprvat/hpopt_config.yaml', - model_config = pretrained_model_path / 'config.yaml', - output: - '{phenotype}/deeprvat/burdens/chunk{chunk}.finished' - threads: 8 - shell: - ' && '.join([ - ('deeprvat_associate compute-burdens ' - + debug + - ' --n-chunks '+ str(n_burden_chunks) + ' ' - '--chunk {wildcards.chunk} ' - '--dataset-file {input.dataset} ' - '{input.data_config} ' - '{input.model_config} ' - '{input.checkpoints} ' - '{wildcards.phenotype}/deeprvat/burdens'), - 'touch {output}' - ]) - rule all_association_dataset: input: expand('{phenotype}/deeprvat/association_dataset.pkl', phenotype=phenotypes) -rule association_dataset: - input: - config = '{phenotype}/deeprvat/hpopt_config.yaml' - output: - '{phenotype}/deeprvat/association_dataset.pkl' - threads: 4 - shell: - 'deeprvat_associate make-dataset ' - + debug + - '{input.config} ' - '{output}' - -rule reverse_models: - input: - checkpoints = expand(pretrained_model_path / 'repeat_{repeat}/best/bag_{bag}.ckpt', - bag=range(n_bags), repeat=range(n_repeats)), - model_config = pretrained_model_path / 'config.yaml', - data_config = Path(phenotypes[0]) / "deeprvat/hpopt_config.yaml", - output: - temp(pretrained_model_path / "reverse_finished.tmp") - threads: 4 - shell: - " && ".join([ - ("deeprvat_associate reverse-models " - "{input.model_config} " - "{input.data_config} " - "{input.checkpoints}"), - "touch {output}" - ]) - rule all_config: input: seed_genes = expand('{phenotype}/deeprvat/seed_genes.parquet', @@ -196,32 +55,3 @@ rule all_config: phenotype=phenotypes), baseline = expand('{phenotype}/deeprvat/baseline_results.parquet', phenotype=phenotypes), - -rule config: - input: - config = 'config.yaml', - baseline = lambda wildcards: [ - str(Path(r['base']) / wildcards.phenotype / r['type'] / - 'eval/burden_associations.parquet') - for r in config['baseline_results'] - ] - output: - seed_genes = '{phenotype}/deeprvat/seed_genes.parquet', - config = '{phenotype}/deeprvat/hpopt_config.yaml', - baseline = '{phenotype}/deeprvat/baseline_results.parquet', - threads: 1 - params: - baseline_results = lambda wildcards, input: ''.join([ - f'--baseline-results {b} ' - for b in input.baseline - ]) - shell: - ( - 'deeprvat_config update-config ' - '--phenotype {wildcards.phenotype} ' - '{params.baseline_results}' - '--baseline-results-out {output.baseline} ' - '--seed-genes-out {output.seed_genes} ' - '{input.config} ' - '{output.config}' - ) diff --git a/pipelines/run_training.snakefile b/pipelines/run_training.snakefile new file mode 100644 index 00000000..0e10d79e --- /dev/null +++ b/pipelines/run_training.snakefile @@ -0,0 +1,51 @@ +from pathlib import Path + +configfile: 'config.yaml' + +debug_flag = config.get('debug', False) +phenotypes = config['phenotypes'] +phenotypes = list(phenotypes.keys()) if type(phenotypes) == dict else phenotypes +training_phenotypes = config["training"].get("phenotypes", phenotypes) + +n_burden_chunks = config.get('n_burden_chunks', 1) if not debug_flag else 2 +n_regression_chunks = config.get('n_regression_chunks', 40) if not debug_flag else 2 +n_trials = config['hyperparameter_optimization']['n_trials'] +n_bags = config['training']['n_bags'] if not debug_flag else 3 +n_repeats = config['n_repeats'] +debug = '--debug ' if debug_flag else '' +do_scoretest = '--do-scoretest ' if config.get('do_scoretest', False) else '' +tensor_compression_level = config['training'].get('tensor_compression_level', 1) +model_path = Path("models") +n_parallel_training_jobs = config["training"].get("n_parallel_jobs", 1) + +wildcard_constraints: + repeat="\d+", + trial="\d+", + +include: "training/config.snakefile" +include: "training/training_dataset.snakefile" +include: "training/train.snakefile" + +rule all: + input: + expand( model_path / 'repeat_{repeat}/best/bag_{bag}.ckpt', + bag=range(n_bags), repeat=range(n_repeats)), + model_path / "config.yaml" + +rule all_training_dataset: + input: + input_tensor = expand('{phenotype}/deeprvat/input_tensor.zarr', + phenotype=training_phenotypes, repeat=range(n_repeats)), + covariates = expand('{phenotype}/deeprvat/covariates.zarr', + phenotype=training_phenotypes, repeat=range(n_repeats)), + y = expand('{phenotype}/deeprvat/y.zarr', + phenotype=training_phenotypes, repeat=range(n_repeats)) + +rule all_config: + input: + seed_genes = expand('{phenotype}/deeprvat/seed_genes.parquet', + phenotype=phenotypes), + config = expand('{phenotype}/deeprvat/hpopt_config.yaml', + phenotype=phenotypes), + baseline = expand('{phenotype}/deeprvat/baseline_results.parquet', + phenotype=phenotypes), \ No newline at end of file diff --git a/pipelines/training/config.snakefile b/pipelines/training/config.snakefile new file mode 100644 index 00000000..3c58a39d --- /dev/null +++ b/pipelines/training/config.snakefile @@ -0,0 +1,29 @@ + +rule config: + input: + config = 'config.yaml', + baseline = lambda wildcards: [ + str(Path(r['base']) / wildcards.phenotype / r['type'] / + 'eval/burden_associations.parquet') + for r in config['baseline_results'] + ] + output: + seed_genes = '{phenotype}/deeprvat/seed_genes.parquet', + config = '{phenotype}/deeprvat/hpopt_config.yaml', + baseline = '{phenotype}/deeprvat/baseline_results.parquet', + threads: 1 + params: + baseline_results = lambda wildcards, input: ''.join([ + f'--baseline-results {b} ' + for b in input.baseline + ]) + shell: + ( + 'deeprvat_config update-config ' + '--phenotype {wildcards.phenotype} ' + '{params.baseline_results}' + '--baseline-results-out {output.baseline} ' + '--seed-genes-out {output.seed_genes} ' + '{input.config} ' + '{output.config}' + ) \ No newline at end of file diff --git a/pipelines/training/train.snakefile b/pipelines/training/train.snakefile new file mode 100644 index 00000000..c747fd1f --- /dev/null +++ b/pipelines/training/train.snakefile @@ -0,0 +1,64 @@ + +rule link_config: + input: + model_path / 'repeat_0/config.yaml' + output: + model_path / 'config.yaml' + threads: 1 + shell: + "ln -s repeat_0/config.yaml {output}" + + +rule best_training_run: + input: + expand(model_path / 'repeat_{{repeat}}/trial{trial_number}/config.yaml', + trial_number=range(n_trials)), + output: + checkpoints = expand(model_path / 'repeat_{{repeat}}/best/bag_{bag}.ckpt', + bag=range(n_bags)), + config = model_path / 'repeat_{repeat}/config.yaml' + threads: 1 + shell: + ( + 'deeprvat_train best-training-run ' + + debug + + '{model_path}/repeat_{wildcards.repeat} ' + '{model_path}/repeat_{wildcards.repeat}/best ' + '{model_path}/repeat_{wildcards.repeat}/hyperparameter_optimization.db ' + '{output.config}' + ) + +rule train: + input: + config = expand('{phenotype}/deeprvat/hpopt_config.yaml', + phenotype=training_phenotypes), + input_tensor = expand('{phenotype}/deeprvat/input_tensor.zarr', + phenotype=training_phenotypes), + covariates = expand('{phenotype}/deeprvat/covariates.zarr', + phenotype=training_phenotypes), + y = expand('{phenotype}/deeprvat/y.zarr', + phenotype=training_phenotypes), + output: + expand(model_path / 'repeat_{repeat}/trial{trial_number}/config.yaml', + repeat=range(n_repeats), trial_number=range(n_trials)), + expand(model_path / 'repeat_{repeat}/trial{trial_number}/finished.tmp', + repeat=range(n_repeats), trial_number=range(n_trials)) + params: + phenotypes = " ".join( + [f"--phenotype {p} " + f"{p}/deeprvat/input_tensor.zarr " + f"{p}/deeprvat/covariates.zarr " + f"{p}/deeprvat/y.zarr" + for p in training_phenotypes]) + shell: + f"parallel --jobs {n_parallel_training_jobs} --halt now,fail=1 --results train_repeat{{{{1}}}}_trial{{{{2}}}}/ " + 'deeprvat_train train ' + + debug + + '--trial-id {{2}} ' + "{params.phenotypes} " + 'config.yaml ' + '{model_path}/repeat_{{1}}/trial{{2}} ' + '{model_path}/repeat_{{1}}/hyperparameter_optimization.db "&&" ' + 'touch {model_path}/repeat_{{1}}/trial{{2}}/finished.tmp ' + "::: " + " ".join(map(str, range(n_repeats))) + " " + "::: " + " ".join(map(str, range(n_trials))) diff --git a/pipelines/training/training_dataset.snakefile b/pipelines/training/training_dataset.snakefile new file mode 100644 index 00000000..66903b85 --- /dev/null +++ b/pipelines/training/training_dataset.snakefile @@ -0,0 +1,37 @@ + +rule training_dataset: + input: + config = '{phenotype}/deeprvat/hpopt_config.yaml', + training_dataset = '{phenotype}/deeprvat/training_dataset.pkl' + output: + input_tensor = directory('{phenotype}/deeprvat/input_tensor.zarr'), + covariates = directory('{phenotype}/deeprvat/covariates.zarr'), + y = directory('{phenotype}/deeprvat/y.zarr') + threads: 8 + priority: 50 + shell: + ( + 'deeprvat_train make-dataset ' + + debug + + '--compression-level ' + str(tensor_compression_level) + ' ' + '--training-dataset-file {input.training_dataset} ' + '{input.config} ' + '{output.input_tensor} ' + '{output.covariates} ' + '{output.y}' + ) + +rule training_dataset_pickle: + input: + '{phenotype}/deeprvat/hpopt_config.yaml' + output: + '{phenotype}/deeprvat/training_dataset.pkl' + threads: 1 + shell: + ( + 'deeprvat_train make-dataset ' + '--pickle-only ' + '--training-dataset-file {output} ' + '{input} ' + 'dummy dummy dummy' + ) \ No newline at end of file diff --git a/pipelines/training_association_testing.snakefile b/pipelines/training_association_testing.snakefile index 1e887e1e..60384eaf 100644 --- a/pipelines/training_association_testing.snakefile +++ b/pipelines/training_association_testing.snakefile @@ -15,12 +15,20 @@ n_repeats = config['n_repeats'] debug = '--debug ' if debug_flag else '' do_scoretest = '--do-scoretest ' if config.get('do_scoretest', False) else '' tensor_compression_level = config['training'].get('tensor_compression_level', 1) +model_path = Path("models") n_parallel_training_jobs = config["training"].get("n_parallel_jobs", 1) wildcard_constraints: repeat="\d+", trial="\d+", +include: "training/config.snakefile" +include: "training/training_dataset.snakefile" +include: "training/train.snakefile" +include: "association_testing/association_dataset.snakefile" +include: "association_testing/burdens.snakefile" +include: "association_testing/regress_eval.snakefile" + rule all: input: expand("{phenotype}/deeprvat/eval/significant.parquet", @@ -28,69 +36,6 @@ rule all: expand("{phenotype}/deeprvat/eval/all_results.parquet", phenotype=phenotypes) -rule evaluate: - input: - associations = expand('{{phenotype}}/deeprvat/repeat_{repeat}/results/burden_associations.parquet', - repeat=range(n_repeats)), - config = '{phenotype}/deeprvat/hpopt_config.yaml', - output: - "{phenotype}/deeprvat/eval/significant.parquet", - "{phenotype}/deeprvat/eval/all_results.parquet" - threads: 1 - shell: - 'deeprvat_evaluate ' - + debug + - '--use-seed-genes ' - '--n-repeats {n_repeats} ' - '--correction-method FDR ' - '{input.associations} ' - '{input.config} ' - '{wildcards.phenotype}/deeprvat/eval' - -rule all_regression: - input: - expand('{phenotype}/deeprvat/repeat_{repeat}/results/burden_associations.parquet', - phenotype=phenotypes, type=['deeprvat'], repeat=range(n_repeats)), - -rule combine_regression_chunks: - input: - expand('{{phenotype}}/deeprvat/repeat_{{repeat}}/results/burden_associations_{chunk}.parquet', chunk=range(n_regression_chunks)), - output: - '{phenotype}/deeprvat/repeat_{repeat}/results/burden_associations.parquet', - threads: 1 - shell: - 'deeprvat_associate combine-regression-results ' - '--model-name repeat_{wildcards.repeat} ' - '{input} ' - '{output}' - -rule regress: - input: - config = "{phenotype}/deeprvat/hpopt_config.yaml", - chunks = lambda wildcards: ( - [] if wildcards.phenotype == phenotypes[0] - else expand('{{phenotype}}/deeprvat/burdens/chunk{chunk}.linked', - chunk=range(n_burden_chunks)) - ), - phenotype_0_chunks = expand( - phenotypes[0] + '/deeprvat/burdens/chunk{chunk}.finished', - chunk=range(n_burden_chunks) - ), - output: - temp('{phenotype}/deeprvat/repeat_{repeat}/results/burden_associations_{chunk}.parquet'), - threads: 2 - shell: - 'deeprvat_associate regress ' - + debug + - '--chunk {wildcards.chunk} ' - '--n-chunks ' + str(n_regression_chunks) + ' ' - '--use-bias ' - '--repeat {wildcards.repeat} ' - + do_scoretest + - '{input.config} ' - '{wildcards.phenotype}/deeprvat/burdens ' #TODO make this w/o repeats - '{wildcards.phenotype}/deeprvat/repeat_{wildcards.repeat}/results' - rule all_burdens: input: [ @@ -100,165 +45,16 @@ rule all_burdens: for c in range(n_burden_chunks) ] -rule link_burdens: - priority: 1 - input: - checkpoints = lambda wildcards: [ - f'models/repeat_{repeat}/best/bag_{bag}.ckpt' - for repeat in range(n_repeats) for bag in range(n_bags) - ], - dataset = '{phenotype}/deeprvat/association_dataset.pkl', - data_config = '{phenotype}/deeprvat/hpopt_config.yaml', - model_config = 'models/config.yaml', - output: - '{phenotype}/deeprvat/burdens/chunk{chunk}.linked' - threads: 8 - shell: - ' && '.join([ - ('deeprvat_associate compute-burdens ' - + debug + - ' --n-chunks '+ str(n_burden_chunks) + ' ' - f'--link-burdens ../../../{phenotypes[0]}/deeprvat/burdens/burdens.zarr ' - '--chunk {wildcards.chunk} ' - '--dataset-file {input.dataset} ' - '{input.data_config} ' - '{input.model_config} ' - '{input.checkpoints} ' - '{wildcards.phenotype}/deeprvat/burdens'), - 'touch {output}' - ]) - -rule compute_burdens: - priority: 10 - input: - reversed = "models/reverse_finished.tmp", - checkpoints = lambda wildcards: [ - f'models/repeat_{repeat}/best/bag_{bag}.ckpt' - for repeat in range(n_repeats) for bag in range(n_bags) - ], - dataset = '{phenotype}/deeprvat/association_dataset.pkl', - data_config = '{phenotype}/deeprvat/hpopt_config.yaml', - model_config = 'models/config.yaml', - output: - '{phenotype}/deeprvat/burdens/chunk{chunk}.finished' - threads: 8 - shell: - ' && '.join([ - ('deeprvat_associate compute-burdens ' - + debug + - ' --n-chunks '+ str(n_burden_chunks) + ' ' - '--chunk {wildcards.chunk} ' - '--dataset-file {input.dataset} ' - '{input.data_config} ' - '{input.model_config} ' - '{input.checkpoints} ' - '{wildcards.phenotype}/deeprvat/burdens'), - 'touch {output}' - ]) - rule all_association_dataset: input: expand('{phenotype}/deeprvat/association_dataset.pkl', phenotype=phenotypes) -rule association_dataset: - input: - config = '{phenotype}/deeprvat/hpopt_config.yaml' - output: - '{phenotype}/deeprvat/association_dataset.pkl' - threads: 4 - shell: - 'deeprvat_associate make-dataset ' - + debug + - '{input.config} ' - '{output}' - -rule reverse_models: - input: - checkpoints = expand('models/repeat_{repeat}/best/bag_{bag}.ckpt', - bag=range(n_bags), repeat=range(n_repeats)), - model_config = 'models/config.yaml', - data_config = Path(phenotypes[0]) / "deeprvat/hpopt_config.yaml", - output: - "models/reverse_finished.tmp" - threads: 4 - shell: - " && ".join([ - ("deeprvat_associate reverse-models " - "{input.model_config} " - "{input.data_config} " - "{input.checkpoints}"), - "touch {output}" - ]) - rule all_training: input: - expand('models/repeat_{repeat}/best/bag_{bag}.ckpt', + expand(model_path / 'repeat_{repeat}/best/bag_{bag}.ckpt', bag=range(n_bags), repeat=range(n_repeats)), - "models/config.yaml" - -rule link_config: - input: - 'models/repeat_0/config.yaml' - output: - "models/config.yaml" - threads: 1 - shell: - "ln -s repeat_0/config.yaml {output}" - -rule best_training_run: - input: - expand('models/repeat_{{repeat}}/trial{trial_number}/config.yaml', - trial_number=range(n_trials)), - output: - checkpoints = expand('models/repeat_{{repeat}}/best/bag_{bag}.ckpt', - bag=range(n_bags)), - config = 'models/repeat_{repeat}/config.yaml' - threads: 1 - shell: - ( - 'deeprvat_train best-training-run ' - + debug + - 'models/repeat_{wildcards.repeat} ' - 'models/repeat_{wildcards.repeat}/best ' - 'models/repeat_{wildcards.repeat}/hyperparameter_optimization.db ' - '{output.config}' - ) - -rule train: - input: - config = expand('{phenotype}/deeprvat/hpopt_config.yaml', - phenotype=training_phenotypes), - input_tensor = expand('{phenotype}/deeprvat/input_tensor.zarr', - phenotype=training_phenotypes), - covariates = expand('{phenotype}/deeprvat/covariates.zarr', - phenotype=training_phenotypes), - y = expand('{phenotype}/deeprvat/y.zarr', - phenotype=training_phenotypes), - output: - expand('models/repeat_{repeat}/trial{trial_number}/config.yaml', - repeat=range(n_repeats), trial_number=range(n_trials)), - expand('models/repeat_{repeat}/trial{trial_number}/finished.tmp', - repeat=range(n_repeats), trial_number=range(n_trials)) - params: - phenotypes = " ".join( - [f"--phenotype {p} " - f"{p}/deeprvat/input_tensor.zarr " - f"{p}/deeprvat/covariates.zarr " - f"{p}/deeprvat/y.zarr" - for p in training_phenotypes]) - shell: - f"parallel --jobs {n_parallel_training_jobs} --halt now,fail=1 --results train_repeat{{{{1}}}}_trial{{{{2}}}}/ " - 'deeprvat_train train ' - + debug + - '--trial-id {{2}} ' - "{params.phenotypes} " - 'config.yaml ' - 'models/repeat_{{1}}/trial{{2}} ' - "models/repeat_{{1}}/hyperparameter_optimization.db '&&' " - "touch models/repeat_{{1}}/trial{{2}}/finished.tmp " - "::: " + " ".join(map(str, range(n_repeats))) + " " - "::: " + " ".join(map(str, range(n_trials))) + model_path / "config.yaml" rule all_training_dataset: input: @@ -269,43 +65,6 @@ rule all_training_dataset: y = expand('{phenotype}/deeprvat/y.zarr', phenotype=training_phenotypes, repeat=range(n_repeats)) -rule training_dataset: - input: - config = '{phenotype}/deeprvat/hpopt_config.yaml', - training_dataset = '{phenotype}/deeprvat/training_dataset.pkl' - output: - input_tensor = directory('{phenotype}/deeprvat/input_tensor.zarr'), - covariates = directory('{phenotype}/deeprvat/covariates.zarr'), - y = directory('{phenotype}/deeprvat/y.zarr') - threads: 8 - priority: 50 - shell: - ( - 'deeprvat_train make-dataset ' - + debug + - '--compression-level ' + str(tensor_compression_level) + ' ' - '--training-dataset-file {input.training_dataset} ' - '{input.config} ' - '{output.input_tensor} ' - '{output.covariates} ' - '{output.y}' - ) - -rule training_dataset_pickle: - input: - '{phenotype}/deeprvat/hpopt_config.yaml' - output: - '{phenotype}/deeprvat/training_dataset.pkl' - threads: 1 - shell: - ( - 'deeprvat_train make-dataset ' - '--pickle-only ' - '--training-dataset-file {output} ' - '{input} ' - 'dummy dummy dummy' - ) - rule all_config: input: seed_genes = expand('{phenotype}/deeprvat/seed_genes.parquet', @@ -314,32 +73,3 @@ rule all_config: phenotype=phenotypes), baseline = expand('{phenotype}/deeprvat/baseline_results.parquet', phenotype=phenotypes), - -rule config: - input: - config = 'config.yaml', - baseline = lambda wildcards: [ - str(Path(r['base']) / wildcards.phenotype / r['type'] / - 'eval/burden_associations.parquet') - for r in config['baseline_results'] - ] - output: - seed_genes = '{phenotype}/deeprvat/seed_genes.parquet', - config = '{phenotype}/deeprvat/hpopt_config.yaml', - baseline = '{phenotype}/deeprvat/baseline_results.parquet', - threads: 1 - params: - baseline_results = lambda wildcards, input: ''.join([ - f'--baseline-results {b} ' - for b in input.baseline - ]) - shell: - ( - 'deeprvat_config update-config ' - '--phenotype {wildcards.phenotype} ' - '{params.baseline_results}' - '--baseline-results-out {output.baseline} ' - '--seed-genes-out {output.seed_genes} ' - '{input.config} ' - '{output.config}' - ) From 21a8e465d68f2526c12832e270be1a453c7f44af Mon Sep 17 00:00:00 2001 From: Magnus Wahlberg Date: Tue, 30 Jan 2024 08:45:50 +0100 Subject: [PATCH 3/3] Remove unsused directories in preprocessing pipeline (#49) --- docs/annotations.md | 2 +- docs/preprocessing.md | 22 ++++++------------- .../config/deeprvat_preprocess_config.yaml | 17 +++++--------- pipelines/preprocessing/preprocess.snakefile | 2 -- 4 files changed, 14 insertions(+), 29 deletions(-) diff --git a/docs/annotations.md b/docs/annotations.md index 6ba208b5..a9532220 100644 --- a/docs/annotations.md +++ b/docs/annotations.md @@ -22,7 +22,7 @@ BCFtools as well as HTSlib should be installed on the machine, will be installed by the pipeline together with the [plugins](https://www.ensembl.org/info/docs/tools/vep/script/vep_plugins.html) for primateAI and spliceAI. Annotation data for CADD, spliceAI and primateAI should be downloaded. The path to the data may be specified in the corresponding [config file](https://github.com/PMBio/deeprvat/blob/main/pipelines/config/deeprvat_annotation_config.yaml). Download path: -- [CADD](http://cadd.gs.washington.edu/download): "All possible SNVs of GRCh38/hg38" and "gnomad.genomes.r3.0.indel.tsv.gz" incl. their Tabix Indices +- [CADD](https://cadd.bihealth.org/download): "All possible SNVs of GRCh38/hg38" and "gnomad.genomes.r3.0.indel.tsv.gz" incl. their Tabix Indices - [SpliceAI](https://basespace.illumina.com/s/otSPW8hnhaZR): "genome_scores_v1.3"/"spliceai_scores.raw.snv.hg38.vcf.gz" and "spliceai_scores.raw.indel.hg38.vcf.gz" - [PrimateAI](https://basespace.illumina.com/s/yYGFdGih1rXL) PrimateAI supplementary data/"PrimateAI_scores_v0.2_GRCh38_sorted.tsv.bgz" diff --git a/docs/preprocessing.md b/docs/preprocessing.md index 6db444d8..502b1e62 100644 --- a/docs/preprocessing.md +++ b/docs/preprocessing.md @@ -50,17 +50,18 @@ An example file is included in this repo: [example config](https://github.com/PM # What chromosomes should be processed included_chromosomes : [21,22] +# The format of the name of the "raw" vcf files +vcf_files_list: vcf_files_list.txt + +# Number of threads to use in the preprocessing script, separate from snakemake threads +preprocess_threads: 16 + # If you need to run a cmd to load bcf and samtools specify it here, see example bcftools_load_cmd : # module load bcftools/1.10.2 && samtools_load_cmd : # module load samtools/1.9 && # Path to where you want to write results and intermediate data working_dir: workdir -# Path to ukbb data -data_dir: data - -# These paths are all relative to the data dir -metadata_dir_name: metadata # These paths are all relative to the working dir # Here will the finished preprocessed files end up @@ -75,23 +76,14 @@ sparse_dir_name : sparse # Expected to be found in working_dir/reference_dir reference_fasta_file : GRCh38.primary_assembly.genome.fa -# The format of the name of the "raw" vcf files -vcf_files_list: vcf_files_list.txt - -# Number of threads to use in the preprocessing script, separate from snakemake threads -preprocess_threads: 16 - # You can specify a different zcat cmd for example gzcat here, default zcat -zcat_cmd: gzcat +zcat_cmd: ``` The config above would use the following directory structure: ```shell parent_directory -|-- data -| |-- metadata -| `-- vcf `-- workdir |-- norm | |-- bcf diff --git a/pipelines/config/deeprvat_preprocess_config.yaml b/pipelines/config/deeprvat_preprocess_config.yaml index 1a0173a9..0f1b146c 100644 --- a/pipelines/config/deeprvat_preprocess_config.yaml +++ b/pipelines/config/deeprvat_preprocess_config.yaml @@ -1,17 +1,18 @@ # What chromosomes should be processed included_chromosomes : [21,22] +# The format of the name of the "raw" vcf files +vcf_files_list: vcf_files_list.txt + +# Number of threads to use in the preprocessing script, separate from snakemake threads +preprocess_threads: 16 + # If you need to run a cmd to load bcf and samtools specify it here, see example bcftools_load_cmd : # module load bcftools/1.10.2 && samtools_load_cmd : # module load samtools/1.9 && # Path to where you want to write results and intermediate data working_dir: workdir -# Path to ukbb data -data_dir: data - -# These paths are all relative to the data dir -metadata_dir_name: metadata # These paths are all relative to the working dir # Here will the finished preprocessed files end up @@ -26,11 +27,5 @@ sparse_dir_name : sparse # Expected to be found in working_dir/reference_dir reference_fasta_file : GRCh38.primary_assembly.genome.fa -# The format of the name of the "raw" vcf files -vcf_files_list: vcf_files_list.txt - -# Number of threads to use in the preprocessing script, separate from snakemake threads -preprocess_threads: 16 - # You can specify a different zcat cmd for example gzcat here, default zcat zcat_cmd: \ No newline at end of file diff --git a/pipelines/preprocessing/preprocess.snakefile b/pipelines/preprocessing/preprocess.snakefile index 87320690..bc9e4702 100644 --- a/pipelines/preprocessing/preprocess.snakefile +++ b/pipelines/preprocessing/preprocess.snakefile @@ -11,9 +11,7 @@ zcat_cmd = config.get("zcat_cmd") or "zcat" preprocessing_cmd = "deeprvat_preprocess" working_dir = Path(config["working_dir"]) -data_dir = Path(config["data_dir"]) preprocessed_dir = working_dir / config["preprocessed_dir_name"] -metadata_dir = data_dir / config["metadata_dir_name"] reference_dir = working_dir / config["reference_dir_name"] preprocess_threads = config["preprocess_threads"]