Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Option to incorporate additional metadata from provided SAM header file #95

Merged
merged 27 commits into from
Jul 12, 2024
Merged
Show file tree
Hide file tree
Changes from 16 commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
18cf836
add --header optional argument
tkchafin Jun 26, 2024
7a11e6b
install nf-core/samtools/reheader module
tkchafin Jun 26, 2024
6fcc80a
install nf-core/samtools/reheader module
tkchafin Jun 26, 2024
55172c7
default config for samtools_reheader
tkchafin Jun 28, 2024
7a460b0
switch to custom samtools reheader wrapper in local module
tkchafin Jun 28, 2024
eeb3b29
switch to custom samtools reheader wrapper in local module
tkchafin Jun 28, 2024
0e58ade
added subworkflow to insert custom sam header before cram generation
tkchafin Jun 28, 2024
6e065b9
example header template for profile/test
tkchafin Jun 28, 2024
23bcb91
remove test prints
tkchafin Jun 28, 2024
79aef7d
remove test prints and minor cleanup
tkchafin Jun 28, 2024
36079fe
remove test prints and minor cleanup
tkchafin Jun 28, 2024
c6568fe
docs for samtools_reheader
tkchafin Jun 28, 2024
af7636e
prettier linting
tkchafin Jun 28, 2024
755731e
moved assets/{example}.header.sam to hosted test_data
tkchafin Jul 1, 2024
1593a8d
add reheader steps to -profile test runs
tkchafin Jul 1, 2024
4ab46b0
fix reheader to only replace SQ lines; maintains input order
tkchafin Jul 4, 2024
a1725a0
Update modules/local/samtools_replaceheader.nf
tkchafin Jul 5, 2024
b6e6477
dynamic output but this time it works
tkchafin Jul 5, 2024
8ade9cf
simplify samtools_replaceheader inputs
tkchafin Jul 5, 2024
a8abc89
remove some of the back-and-forth with bai placeholders
tkchafin Jul 5, 2024
db54501
cleanup un-needed line
tkchafin Jul 5, 2024
f5b3edc
Update subworkflows/local/align_short.nf
tkchafin Jul 11, 2024
a7c125b
Update modules/local/samtools_replaceheader.nf
tkchafin Jul 11, 2024
7a714fc
Update subworkflows/local/convert_stats.nf
tkchafin Jul 11, 2024
ca86a50
Update modules/local/samtools_replaceheader.nf
tkchafin Jul 11, 2024
15e8d7b
streamline prep for samtools_view call
tkchafin Jul 12, 2024
3aca264
remove intermediate steps before emit
tkchafin Jul 12, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions conf/modules.config
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,12 @@ process {
ext.prefix = { "${meta.id}.merge" }
}

// If custom header provided, this is inserted in place of existing
// @HD and @SQ lines, while preserving any other header entries
withName: SAMTOOLS_REHEADER {
ext.prefix = { "${meta.id}.reheader" }
}

withName: SAMTOOLS_COLLATETOFASTA {
ext.args = { (params.use_work_dir_as_temp ? "-T." : "") }
}
Expand Down
1 change: 1 addition & 0 deletions conf/test.config
Original file line number Diff line number Diff line change
Expand Up @@ -24,4 +24,5 @@ params {

// Fasta references
fasta = "https://tolit.cog.sanger.ac.uk/test-data/Meles_meles/assembly/release/mMelMel3.1_paternal_haplotype/GCA_922984935.2.subset.fasta.gz"
header = "https://tolit.cog.sanger.ac.uk/test-data/Meles_meles/assembly/release/mMelMel3.1_paternal_haplotype/GCA_922984935.2.subset.header.sam"
}
4 changes: 4 additions & 0 deletions docs/output.md
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,10 @@ The filtered PacBio reads are aligned with `MINIMAP2_ALIGN`. The sorted and merg

## Alignment post-processing

### External metadata

If provided using the `--header` option, all output alignments (`*.cram`) will include any additional metadata supplied as a SAM header template, replacing the existing _@HD_ and _@SD_ entries (note that this behaviour can be altered by modifying the `ext.args` for `SAMTOOLS_REHEADER` in `modules.config`.

### Statistics

The output alignments, along with the index, are used to calculate mapping statistics. Output files are generated using `SAMTOOLS_STATS`, `SAMTOOLS_FLAGSTAT` and `SAMTOOLS_IDXSTATS`.
Expand Down
2 changes: 2 additions & 0 deletions docs/usage.md
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,8 @@ work # Directory containing the nextflow working files
# Other nextflow hidden files, eg. history of pipeline runs and old logs.
```

You can also optionally supply a template SAM header using the `--header` option to add or modify metadata associated with the assembly, which will be incorporated into the output alignments.

### Updating the pipeline

When you run the above command, Nextflow automatically pulls the pipeline code from GitHub and stores it as a cached version. When running the pipeline after this, it will always use the cached version if available - even if the pipeline has been updated since. To make sure that you're running the latest version of the pipeline, make sure that you regularly update the cached version of the pipeline:
Expand Down
59 changes: 59 additions & 0 deletions modules/local/samtools_replaceheader.nf
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
process SAMTOOLS_REHEADER {
tag "$meta.id"
label 'process_single'

conda "bioconda::samtools=1.17"
container "${ workflow.containerEngine == 'singularity' && !task.ext.singularity_pull_docker_container ?
'https://depot.galaxyproject.org/singularity/samtools:1.17--h00cdaf9_0' :
'biocontainers/samtools:1.17--h00cdaf9_0' }"

input:
tuple val(meta), path(file), path(header)

output:
tuple val(meta), path("*.${meta.suffix}"), emit: bam
path "versions.yml", emit: versions

when:
task.ext.when == null || task.ext.when

script:
def prefix = task.ext.prefix ?: "${meta.id}"
def suffix = "${meta.suffix}"

if ("$file" == "${prefix}.${suffix}") error "Input and output names are the same, use \"task.ext.prefix\" to disambiguate!"
"""
# Replace SQ lines with those from external template
( samtools view --no-PG --header-only ${file} | \\
grep -v ^@SQ && grep ^@SQ ${header} ) > .temp.header.sam
tkchafin marked this conversation as resolved.
Show resolved Hide resolved

# custom sort for readability (retain order of insertion but sort groups by tag)
( grep ^@HD .temp.header.sam && \
grep ^@SQ .temp.header.sam && \
grep ^@RG .temp.header.sam && \
grep ^@PG .temp.header.sam && \
if grep -q -E -v '^@HD|^@SQ|^@RG|^@PG' .temp.header.sam; then \
grep -v -E '^@HD|^@SQ|^@RG|^@PG' .temp.header.sam; \
fi; ) > .temp.sorted.header.sam
tkchafin marked this conversation as resolved.
Show resolved Hide resolved

# Insert new header into file
samtools reheader .temp.sorted.header.sam ${file} > ${prefix}.${suffix}

cat <<-END_VERSIONS > versions.yml
"${task.process}":
samtools: \$(echo \$(samtools --version 2>&1) | sed 's/^.*samtools //; s/Using.*\$//')
END_VERSIONS
"""

stub:
def prefix = task.ext.prefix ?: "${meta.id}"
def suffix = file.name.split('\\.')[-1]
"""
touch ${prefix}.${suffix}

cat <<-END_VERSIONS > versions.yml
"${task.process}":
samtools: \$(echo \$(samtools --version 2>&1) | sed 's/^.*samtools //; s/Using.*\$//')
END_VERSIONS
"""
}
1 change: 1 addition & 0 deletions nextflow.config
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ params {
vector_db = "${projectDir}/assets/vectorDB.tar.gz"
bwamem2_index = null
fasta = null
header = null

// Execution options
use_work_dir_as_temp = false
Expand Down
9 changes: 8 additions & 1 deletion nextflow_schema.json
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,8 @@
"type": "string",
"format": "directory-path",
"description": "The output directory where the results will be saved. You have to use absolute paths to storage on Cloud infrastructure.",
"fa_icon": "fas fa-folder-open"
"fa_icon": "fas fa-folder-open",
"default": "./results"
},
"vector_db": {
"type": "string",
Expand Down Expand Up @@ -64,6 +65,12 @@
"description": "Path to directory or tar.gz archive for pre-built BWAMEM2 index.",
"format": "path",
"fa_icon": "fas fa-bezier-curve"
},
"header": {
"type": "string",
"format": "path",
"description": "Optional template header file for BAM/CRAM outputs",
"fa_icon": "far fa-file-code"
}
}
},
Expand Down
29 changes: 26 additions & 3 deletions workflows/readmapping.nf
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,13 @@ if (params.fasta) { ch_fasta = Channel.fromPath(params.fasta) } else { exit 1, '
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
*/

//
// MODULE: Local modules
//

include { SAMTOOLS_REHEADER } from '../modules/local/samtools_replaceheader'


//
// SUBWORKFLOW: Consisting of a mix of local and nf-core/modules
//
Expand Down Expand Up @@ -48,7 +55,6 @@ include { CONVERT_STATS } from '../subworkflows/local/convert_st
include { UNTAR } from '../modules/nf-core/untar/main'
include { CUSTOM_DUMPSOFTWAREVERSIONS } from '../modules/nf-core/custom/dumpsoftwareversions/main'


/*
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
RUN MAIN WORKFLOW
Expand Down Expand Up @@ -125,14 +131,31 @@ workflow READMAPPING {
ALIGN_ONT ( PREPARE_GENOME.out.fasta, ch_reads.ont )
ch_versions = ch_versions.mix ( ALIGN_ONT.out.versions )


// gather alignments
ch_aligned_bams = Channel.empty()
| mix( ALIGN_HIC.out.bam )
| mix( ALIGN_ILLUMINA.out.bam )
| mix( ALIGN_HIFI.out.bam )
| mix( ALIGN_CLR.out.bam )
| mix( ALIGN_ONT.out.bam )
CONVERT_STATS ( ch_aligned_bams, PREPARE_GENOME.out.fasta )

// Optionally insert params.header information to bams
ch_reheadered_bams = Channel.empty()
if ( params.header ) {
ch_combined = ch_aligned_bams.map { meta, bam, _ ->
def suffix = bam instanceof List ? bam[0].getExtension() : bam.getExtension()
meta.suffix = suffix // add suffix to meta so output matches input type
tkchafin marked this conversation as resolved.
Show resolved Hide resolved
[meta, bam, file( params.header )]
tkchafin marked this conversation as resolved.
Show resolved Hide resolved
}
SAMTOOLS_REHEADER( ch_combined )
ch_reheadered_bams = SAMTOOLS_REHEADER.out.bam.map { bam -> bam + [[]] }
tkchafin marked this conversation as resolved.
Show resolved Hide resolved
ch_versions = ch_versions.mix ( SAMTOOLS_REHEADER.out.versions )
} else {
ch_reheadered_bams = ch_aligned_bams
}

// convert to cram and gather stats
CONVERT_STATS ( ch_reheadered_bams, PREPARE_GENOME.out.fasta )
ch_versions = ch_versions.mix ( CONVERT_STATS.out.versions )


Expand Down
Loading