Mercurial > repos > peterjc > seq_primer_clip
changeset 5:15b8ae0b3529 draft
Uploaded v0.0.10c, renamed folder and README file.
author | peterjc |
---|---|
date | Thu, 26 Sep 2013 10:33:37 -0400 |
parents | c6bb189bf7c0 |
children | 834cf6618a14 |
files | tools/primers/README.rst tools/primers/repository_dependencies.xml tools/primers/seq_primer_clip.py tools/primers/seq_primer_clip.xml tools/seq_primer_clip/README.rst tools/seq_primer_clip/repository_dependencies.xml tools/seq_primer_clip/seq_primer_clip.py tools/seq_primer_clip/seq_primer_clip.xml |
diffstat | 8 files changed, 627 insertions(+), 626 deletions(-) [+] |
line wrap: on
line diff
--- a/tools/primers/README.rst Tue Sep 17 11:57:34 2013 -0400 +++ /dev/null Thu Jan 01 00:00:00 1970 +0000 @@ -1,119 +0,0 @@ -Galaxy tool to primer clip (trim) FASTA, FASTQ or SFF reads -=========================================================== - -This tool is copyright 2011-2013 by Peter Cock, The James Hutton Institute -(formerly SCRI, Scottish Crop Research Institute), UK. All rights reserved. -See the licence text below (MIT licence). - -This tool is a short Python script (using the Galaxy library functions and -Biopython). It is available from the Galaxy Tool Shed here: -http://toolshed.g2.bx.psu.edu/view/peterjc/seq_primer_clip - - -Automated Installation -====================== - -This should be straightforward using the Galaxy Tool Shed, which should be -able to automatically install the dependency on Biopython, and then install -this tool and run its unit tests. - - -Manual Installation -=================== - -There are just two files to install: - -* seq_primer_clip.py (the Python script) -* seq_primer_clip.xml (the Galaxy tool definition) - -The suggested location is a new tools/primers folder. You will also need to -modify the tools_conf.xml file to tell Galaxy to offer the tool:: - - <tool file="primers/seq_primer_clip.xml" /> - -If you wish to run the unit tests, also add this to tools_conf.xml.sample -and move/copy the test-data files under Galaxy's test-data folder. Then:: - - $ ./run_functional_tests.sh -id seq_primer_clip - -You will also need to install Biopython 1.54 or later. That's it. - - -History -======= - -======= ====================================================================== -Version Changes -------- ---------------------------------------------------------------------- -v0.0.1 - Initial version (not publicly released) -v0.0.2 - Sort primers by length (longest and therefore most specific first) -v0.0.3 - Consider missing bases at start/end of read as mismatches -v0.0.4 - Apply minimum length to sequences with no match too -v0.0.5 - Count clipped & non-matched short reads separately, length bug fixes -v0.0.6 - Added some functional tests -v0.0.7 - Added error check for bad filename arguments -v0.0.8 - Record version of Python script when run from Galaxy. - - Check for errors using Python script's return code. -v0.0.9 - Moved test data to workaround Galaxy Tool Shed limititation. -v0.0.10 - Include links to Tool Shed in help text and this README file. - - Use reStructuredText for this README file. - - Adopted standard MIT licence. - - Automated installation of Biopython dependency. - - Development moved to GitHub, https://github.com/peterjc/pico_galaxy -======= ====================================================================== - - -Developers -========== - -This script and related tools were initially developed on the following hg branches: -http://bitbucket.org/peterjc/galaxy-central/src/fasta_filter -http://bitbucket.org/peterjc/galaxy-central/src/tools - -Development has now moved to a dedicated GitHub repository: -https://github.com/peterjc/pico_galaxy/tree/master/tools - -For making the "Galaxy Tool Shed" http://toolshed.g2.bx.psu.edu/ tarball use -the following command from the Galaxy root folder:: - - $ tar -czf seq_primer_clip.tar.gz tools/primers/README.rst tools/primers/seq_primer_clip.* tools/primers/repository_dependencies.xml test-data/dop_primers.fasta test-data/MID4_GLZRM4E04_rnd30* - -Check this worked:: - - $ tar -tzf seq_primer_clip.tar.gz - tools/primers/README.rst - tools/primers/seq_primer_clip.xml - tools/primers/seq_primer_clip.py - tools/primers/repository_dependencies.xml - test-data/dop_primers.fasta - test-data/MID4_GLZRM4E04_rnd30.fasta - test-data/MID4_GLZRM4E04_rnd30.fastqsanger - test-data/MID4_GLZRM4E04_rnd30_fclip.fasta - test-data/MID4_GLZRM4E04_rnd30_fclip.fastqsanger - test-data/MID4_GLZRM4E04_rnd30_fclip.sff - test-data/MID4_GLZRM4E04_rnd30_frclip.fasta - test-data/MID4_GLZRM4E04_rnd30_frclip.fastqsanger - test-data/MID4_GLZRM4E04_rnd30_frclip.sff - test-data/MID4_GLZRM4E04_rnd30.sff - - -Licence (MIT) -============= - -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is -furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in -all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN -THE SOFTWARE.
--- a/tools/primers/repository_dependencies.xml Tue Sep 17 11:57:34 2013 -0400 +++ /dev/null Thu Jan 01 00:00:00 1970 +0000 @@ -1,6 +0,0 @@ -<?xml version="1.0"?> -<repositories description="This requires Biopython as a dependency."> -<!-- Leave out the tool shed and revision to get the current - tool shed and latest revision at the time of upload --> -<repository changeset_revision="2f6c871cfa35" name="package_biopython_1_61" owner="biopython" toolshed="http://testtoolshed.g2.bx.psu.edu" /> -</repositories>
--- a/tools/primers/seq_primer_clip.py Tue Sep 17 11:57:34 2013 -0400 +++ /dev/null Thu Jan 01 00:00:00 1970 +0000 @@ -1,358 +0,0 @@ -#!/usr/bin/env python -"""Looks for the given primer sequences and clips matching SFF reads. - -Takes eight command line options, input read filename, input read format, -input primer FASTA filename, type of primers (forward, reverse or reverse- -complement), number of mismatches (currently only 0, 1 and 2 are supported), -minimum length to keep a read (after primer trimming), should primer-less -reads be kept (boolean), and finally the output sequence filename. - -Both the primer and read sequences can contain IUPAC ambiguity codes like N. - -This supports FASTA, FASTQ and SFF sequence files. Colorspace reads are not -supported. - -The mismatch parameter does not consider gapped alignemnts, however the -special case of missing bases at the very start or end of the read is handled. -e.g. a primer sequence CCGACTCGAG will match a read starting CGACTCGAG... -if one or more mismatches are allowed. - -This can also be used for stripping off (and optionally filtering on) barcodes. - -Note that only the trim/clip values in the SFF file are changed, not the flow -information of the full read sequence. - -This script is copyright 2011-2013 by Peter Cock, The James Hutton Institute -(formerly the Scottish Crop Research Institute, SCRI), UK. All rights reserved. -See accompanying text file for licence details (MIT/BSD style). - -This is version 0.0.8 of the script. Currently it uses Python's regular -expression engine for finding the primers, which for my needs is fast enough. -""" -import sys -import re -from galaxy_utils.sequence.fasta import fastaReader, fastaWriter -from galaxy_utils.sequence.fastq import fastqReader, fastqWriter - -if "-v" in sys.argv or "--version" in sys.argv: - print "v0.0.5" - sys.exit(0) - -def stop_err(msg, err=1): - sys.stderr.write(msg) - sys.exit(err) - -try: - from Bio.Seq import reverse_complement - from Bio.SeqIO.SffIO import SffIterator, SffWriter -except ImportError: - stop_err("Requires Biopython 1.54 or later") -try: - from Bio.SeqIO.SffIO import ReadRocheXmlManifest -except ImportError: - #Prior to Biopython 1.56 this was a private function - from Bio.SeqIO.SffIO import _sff_read_roche_index_xml as ReadRocheXmlManifest - -#Parse Command Line -try: - in_file, seq_format, primer_fasta, primer_type, mm, min_len, keep_negatives, out_file = sys.argv[1:] -except ValueError: - stop_err("Expected 8 arguments, got %i:\n%s" % (len(sys.argv)-1, " ".join(sys.argv))) - -if in_file == primer_fasta: - stop_err("Same file given as both primer sequences and sequences to clip!") -if in_file == out_file: - stop_err("Same file given as both sequences to clip and output!") -if primer_fasta == out_file: - stop_err("Same file given as both primer sequences and output!") - -try: - mm = int(mm) -except ValueError: - stop_err("Expected non-negative integer number of mismatches (e.g. 0 or 1), not %r" % mm) -if mm < 0: - stop_err("Expected non-negtive integer number of mismatches (e.g. 0 or 1), not %r" % mm) -if mm not in [0,1,2]: - raise NotImplementedError - -try: - min_len = int(min_len) -except ValueError: - stop_err("Expected non-negative integer min_len (e.g. 0 or 1), not %r" % min_len) -if min_len < 0: - stop_err("Expected non-negtive integer min_len (e.g. 0 or 1), not %r" % min_len) - - -if keep_negatives.lower() in ["true", "yes", "on"]: - keep_negatives = True -elif keep_negatives.lower() in ["false", "no", "off"]: - keep_negatives = False -else: - stop_err("Expected boolean for keep_negatives (e.g. true or false), not %r" % keep_negatives) - - -if primer_type.lower() == "forward": - forward = True - rc = False -elif primer_type.lower() == "reverse": - forward = False - rc = False -elif primer_type.lower() == "reverse-complement": - forward = False - rc = True -else: - stop_err("Expected foward, reverse or reverse-complement not %r" % primer_type) - - -ambiguous_dna_values = { - "A": "A", - "C": "C", - "G": "G", - "T": "T", - "M": "ACM", - "R": "AGR", - "W": "ATW", - "S": "CGS", - "Y": "CTY", - "K": "GTK", - "V": "ACGMRSV", - "H": "ACTMWYH", - "D": "AGTRWKD", - "B": "CGTSYKB", - "X": ".", #faster than [GATCMRWSYKVVHDBXN] or even [GATC] - "N": ".", - } - -ambiguous_dna_re = {} -for letter, values in ambiguous_dna_values.iteritems(): - if len(values) == 1: - ambiguous_dna_re[letter] = values - else: - ambiguous_dna_re[letter] = "[%s]" % values - - -def make_reg_ex(seq): - return "".join(ambiguous_dna_re[letter] for letter in seq) - -def make_reg_ex_mm(seq, mm): - if mm > 2: - raise NotImplementedError("At most 2 mismatches allowed!") - seq = seq.upper() - yield make_reg_ex(seq) - for i in range(1,mm+1): - #Missing first/last i bases at very start/end of sequence - for reg in make_reg_ex_mm(seq[i:], mm-i): - yield "^" + reg - for reg in make_reg_ex_mm(seq[:-i], mm-i): - yield "$" + reg - if mm >= 1: - for i,letter in enumerate(seq): - #We'll use a set to remove any duplicate patterns - #if letter not in "NX": - pattern = seq[:i] + "N" + seq[i+1:] - assert len(pattern) == len(seq), "Len %s is %i, len %s is %i" \ - % (pattern, len(pattern), seq, len(seq)) - yield make_reg_ex(pattern) - if mm >=2: - for i,letter in enumerate(seq): - #We'll use a set to remove any duplicate patterns - #if letter not in "NX": - for k,letter in enumerate(seq[i+1:]): - #We'll use a set to remove any duplicate patterns - #if letter not in "NX": - pattern = seq[:i] + "N" + seq[i+1:i+1+k] + "N" + seq[i+k+2:] - assert len(pattern) == len(seq), "Len %s is %i, len %s is %i" \ - % (pattern, len(pattern), seq, len(seq)) - yield make_reg_ex(pattern) - -def load_primers_as_re(primer_fasta, mm, rc=False): - #Read primer file and record all specified sequences - primers = set() - in_handle = open(primer_fasta, "rU") - reader = fastaReader(in_handle) - count = 0 - for record in reader: - if rc: - seq = reverse_complement(record.sequence) - else: - seq = record.sequence - #primers.add(re.compile(make_reg_ex(seq))) - count += 1 - for pattern in make_reg_ex_mm(seq, mm): - primers.add(pattern) - in_handle.close() - #Use set to avoid duplicates, sort to have longest first - #(so more specific primers found before less specific ones) - primers = sorted(set(primers), key=lambda p: -len(p)) - return count, re.compile("|".join(primers)) #make one monster re! - - - -#Read primer file and record all specified sequences -count, primer = load_primers_as_re(primer_fasta, mm, rc) -print "%i primer sequences" % count - -short_neg = 0 -short_clipped = 0 -clipped = 0 -negs = 0 - -if seq_format.lower()=="sff": - #SFF is different because we just change the trim points - if forward: - def process(records): - global short_clipped, short_neg, clipped, negs - for record in records: - left_clip = record.annotations["clip_qual_left"] - right_clip = record.annotations["clip_qual_right"] - seq = str(record.seq)[left_clip:right_clip].upper() - result = primer.search(seq) - if result: - #Forward primer, take everything after it - #so move the left clip along - if len(seq) - result.end() >= min_len: - record.annotations["clip_qual_left"] = left_clip + result.end() - clipped += 1 - yield record - else: - short_clipped += 1 - elif keep_negatives: - if len(seq) >= min_len: - negs += 1 - yield record - else: - short_neg += 1 - else: - def process(records): - global short_clipped, short_neg, clipped, negs - for record in records: - left_clip = record.annotations["clip_qual_left"] - right_clip = record.annotations["clip_qual_right"] - seq = str(record.seq)[left_clip:right_clip].upper() - result = primer.search(seq) - if result: - #Reverse primer, take everything before it - #so move the right clip back - new_len = result.start() - if new_len >= min_len: - record.annotations["clip_qual_right"] = left_clip + new_len - clipped += 1 - yield record - else: - short_clipped += 1 - elif keep_negatives: - if len(seq) >= min_len: - negs += 1 - yield record - else: - short_neg += 1 - - in_handle = open(in_file, "rb") - try: - manifest = ReadRocheXmlManifest(in_handle) - except ValueError: - manifest = None - in_handle.seek(0) - out_handle = open(out_file, "wb") - writer = SffWriter(out_handle, xml=manifest) - writer.write_file(process(SffIterator(in_handle))) - #End of SFF code -elif seq_format.lower().startswith("fastq"): - in_handle = open(in_file, "rU") - out_handle = open(out_file, "w") - reader = fastqReader(in_handle) - writer = fastqWriter(out_handle) - if forward: - for record in reader: - seq = record.sequence.upper() - result = primer.search(seq) - if result: - #Forward primer, take everything after it - cut = result.end() - record.sequence = seq[cut:] - if len(record.sequence) >= min_len: - record.quality = record.quality[cut:] - clipped += 1 - writer.write(record) - else: - short_clipped += 1 - elif keep_negatives: - if len(record) >= min_len: - negs += 1 - writer.write(record) - else: - short_negs += 1 - else: - for record in reader: - seq = record.sequence.upper() - result = primer.search(seq) - if result: - #Reverse primer, take everything before it - cut = result.start() - record.sequence = seq[:cut] - if len(record.sequence) >= min_len: - record.quality = record.quality[:cut] - clipped += 1 - writer.write(record) - else: - short_clipped += 1 - elif keep_negatives: - if len(record) >= min_len: - negs += 1 - writer.write(record) - else: - short_negs += 1 -elif seq_format.lower()=="fasta": - in_handle = open(in_file, "rU") - out_handle = open(out_file, "w") - reader = fastaReader(in_handle) - writer = fastaWriter(out_handle) - #Following code is identical to that for FASTQ but without editing qualities - if forward: - for record in reader: - seq = record.sequence.upper() - result = primer.search(seq) - if result: - #Forward primer, take everything after it - cut = result.end() - record.sequence = seq[cut:] - if len(record.sequence) >= min_len: - clipped += 1 - writer.write(record) - else: - short_clipped += 1 - elif keep_negatives: - if len(record) >= min_len: - negs += 1 - writer.write(record) - else: - short_negs += 1 - else: - for record in reader: - seq = record.sequence.upper() - result = primer.search(seq) - if result: - #Reverse primer, take everything before it - cut = result.start() - record.sequence = seq[:cut] - if len(record.sequence) >= min_len: - clipped += 1 - writer.write(record) - else: - short_clipped += 1 - elif keep_negatives: - if len(record) >= min_len: - negs += 1 - writer.write(record) - else: - short_negs += 1 -else: - stop_err("Unsupported file type %r" % seq_format) -in_handle.close() -out_handle.close() - -print "Kept %i clipped reads," % clipped -print "discarded %i short." % short_clipped -if keep_negatives: - print "Kept %i non-matching reads," % negs - print "discarded %i short." % short_neg
--- a/tools/primers/seq_primer_clip.xml Tue Sep 17 11:57:34 2013 -0400 +++ /dev/null Thu Jan 01 00:00:00 1970 +0000 @@ -1,143 +0,0 @@ -<tool id="seq_primer_clip" name="Primer clip sequences" version="0.0.10"> - <description>Trim off 5' or 3' primers</description> - <requirements> - <requirement type="package" version="1.61">biopython</requirement> - <requirement type="python-module">Bio</requirement> - </requirements> - <version_command interpreter="python">seq_primer_clip.py --version</version_command> - <command interpreter="python"> -seq_primer_clip.py $input_file $input_file.ext $primer_fasta $primer_type $mm $min_len $keep_negatives $output_file - </command> - <stdio> - <!-- Anything other than zero is an error --> - <exit_code range="1:" /> - <exit_code range=":-1" /> - </stdio> - <inputs> - <param name="input_file" type="data" format="fasta,fastq,sff" label="Sequence file to clip" description="FASTA, FASTQ, or SFF format."/> - <param name="primer_fasta" type="data" format="fasta" label="FASTA file containing primer(s)"/> - <param name="primer_type" type="select" label="Type of primers"> - <option value="Forward">Forward (5') primers</option> - <option value="Reverse">Reverse (3') primers (given with respect to the forward strand)</option> - <option value="Reverse-complement">Reverse (3') primers (given with respect to the reverse strand)</option> - </param> - <param name="mm" type="integer" value="0" label="How many mismatches to allow? (0, 1 or 2)"> - <validator type="in_range" min="0" max="2" /> - </param> - <param name="keep_negatives" type="boolean" value="false" label="Keep reads with no matched primer"/> - <param name="min_len" type="integer" label="Minimum length for (clipped) sequences " value="1"/> - </inputs> - <outputs> - <data name="output_file" format="data" label="$primer_type primer clipped"> - <!-- TODO - Replace this with format="input:input_fastq" if/when that works --> - <change_format> - <when input_dataset="input_file" attribute="extension" value="sff" format="sff" /> - <when input_dataset="input_file" attribute="extension" value="fasta" format="fasta" /> - <when input_dataset="input_file" attribute="extension" value="fastq" format="fastq" /> - <when input_dataset="input_file" attribute="extension" value="fastqsanger" format="fastqsanger" /> - <when input_dataset="input_file" attribute="extension" value="fastqsolexa" format="fastqsolexa" /> - <when input_dataset="input_file" attribute="extension" value="fastqillumina" format="fastqillumina" /> - <when input_dataset="input_file" attribute="extension" value="fastqcssanger" format="fastqcssanger" /> - </change_format> - </data> - </outputs> - <tests> - <test> - <param name="input_file" value="MID4_GLZRM4E04_rnd30.fasta" ftype="fasta" /> - <param name="primer_fasta" value="dop_primers.fasta" /> - <param name="primer_type" value="Forward" /> - <param name="mm" value="2" /> - <param name="keep_negatives" value="false" /> - <param name="min_len" value="35" /> - <output name="output_file" file="MID4_GLZRM4E04_rnd30_fclip.fasta" ftype="fasta" /> - </test> - <test> - <param name="input_file" value="MID4_GLZRM4E04_rnd30.fastqsanger" ftype="fastqsanger" /> - <param name="primer_fasta" value="dop_primers.fasta" /> - <param name="primer_type" value="Forward" /> - <param name="mm" value="2" /> - <param name="keep_negatives" value="false" /> - <param name="min_len" value="35" /> - <output name="output_file" file="MID4_GLZRM4E04_rnd30_fclip.fastqsanger" ftype="fastqsanger" /> - </test> - <test> - <param name="input_file" value="MID4_GLZRM4E04_rnd30.sff" ftype="sff" /> - <param name="primer_fasta" value="dop_primers.fasta" /> - <param name="primer_type" value="Forward" /> - <param name="mm" value="2" /> - <param name="keep_negatives" value="false" /> - <param name="min_len" value="35" /> - <output name="output_file" file="MID4_GLZRM4E04_rnd30_fclip.sff" ftype="sff" /> - </test> - <test> - <param name="input_file" value="MID4_GLZRM4E04_rnd30_fclip.fasta" ftype="fasta" /> - <param name="primer_fasta" value="dop_primers.fasta" /> - <param name="primer_type" value="Reverse" /> - <param name="mm" value="2" /> - <param name="keep_negatives" value="true" /> - <param name="min_len" value="35" /> - <output name="output_file" file="MID4_GLZRM4E04_rnd30_frclip.fasta" ftype="fasta" /> - </test> - <test> - <param name="input_file" value="MID4_GLZRM4E04_rnd30_fclip.fastqsanger" ftype="fastqsanger" /> - <param name="primer_fasta" value="dop_primers.fasta" /> - <param name="primer_type" value="Reverse" /> - <param name="mm" value="2" /> - <param name="keep_negatives" value="true" /> - <param name="min_len" value="35" /> - <output name="output_file" file="MID4_GLZRM4E04_rnd30_frclip.fastqsanger" ftype="fastqsanger" /> - </test> - <test> - <param name="input_file" value="MID4_GLZRM4E04_rnd30_fclip.sff" ftype="sff" /> - <param name="primer_fasta" value="dop_primers.fasta" /> - <param name="primer_type" value="Reverse" /> - <param name="mm" value="2" /> - <param name="keep_negatives" value="true" /> - <param name="min_len" value="35" /> - <output name="output_file" file="MID4_GLZRM4E04_rnd30_frclip.sff" ftype="sff" /> - </test> - </tests> - <requirements> - <requirement type="python-module">Bio</requirement> - </requirements> - <help> - -**What it does** - -Looks for the given primer sequences (within the existing clipped sequence) and -further clips the reads to remove the primers and any preceding/trailing sequence. - -Reads containing a forward primer are reduced to just the sequence after (and -excluding) the forward primer. - -Reads containing a reverse primer are reduced to just the sequence before (and -excluding) the reverse primer. - -Degenerate primers can be specified using the standard IUPAC ambiguity codes, -thus a primer with an N would match A, C, T or G (or any of the IUPAC ambiguity -codes) and so on. - -Note that for SFF files only the clip/trim positions are edited - you will still -be able to extract the original full read (with any adapter sequence and poor -quality sequence) if you need to. - -.. class:: warningmark - -**Note**. This tool was initially written for Roche 454 data, and should also -work fine on Sanger or Ion Torrent as well. However, it is probably too slow -for use on large Illumina datasets. - - -**Citation** - -This tool uses Biopython. If you use this tool in scientific work leading to a -publication, please cite: - -Cock et al 2009. Biopython: freely available Python tools for computational -molecular biology and bioinformatics. Bioinformatics 25(11) 1422-3. -http://dx.doi.org/10.1093/bioinformatics/btp163 pmid:19304878. - -This tool is available to install into other Galaxy Instances via the Galaxy -Tool Shed at http://toolshed.g2.bx.psu.edu/view/peterjc/seq_primer_clip - </help> -</tool>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/tools/seq_primer_clip/README.rst Thu Sep 26 10:33:37 2013 -0400 @@ -0,0 +1,120 @@ +Galaxy tool to primer clip (trim) FASTA, FASTQ or SFF reads +=========================================================== + +This tool is copyright 2011-2013 by Peter Cock, The James Hutton Institute +(formerly SCRI, Scottish Crop Research Institute), UK. All rights reserved. +See the licence text below (MIT licence). + +This tool is a short Python script (using the Galaxy library functions and +Biopython). It is available from the Galaxy Tool Shed here: +http://toolshed.g2.bx.psu.edu/view/peterjc/seq_primer_clip + + +Automated Installation +====================== + +This should be straightforward using the Galaxy Tool Shed, which should be +able to automatically install the dependency on Biopython, and then install +this tool and run its unit tests. + + +Manual Installation +=================== + +There are just two files to install: + +* seq_primer_clip.py (the Python script) +* seq_primer_clip.xml (the Galaxy tool definition) + +The suggested location is a new tools/seq_primer_clip folder. You will also +need to modify the tools_conf.xml file to tell Galaxy to offer the tool:: + + <tool file="seq_primer_clip/seq_primer_clip.xml" /> + +If you wish to run the unit tests, also add this to tools_conf.xml.sample +and move/copy the test-data files under Galaxy's test-data folder. Then:: + + $ ./run_functional_tests.sh -id seq_primer_clip + +You will also need to install Biopython 1.54 or later. That's it. + + +History +======= + +======= ====================================================================== +Version Changes +------- ---------------------------------------------------------------------- +v0.0.1 - Initial version (not publicly released) +v0.0.2 - Sort primers by length (longest and therefore most specific first) +v0.0.3 - Consider missing bases at start/end of read as mismatches +v0.0.4 - Apply minimum length to sequences with no match too +v0.0.5 - Count clipped & non-matched short reads separately, length bug fixes +v0.0.6 - Added some functional tests +v0.0.7 - Added error check for bad filename arguments +v0.0.8 - Record version of Python script when run from Galaxy. + - Check for errors using Python script's return code. +v0.0.9 - Moved test data to workaround Galaxy Tool Shed limititation. +v0.0.10 - Include links to Tool Shed in help text and this README file. + - Use reStructuredText for this README file. + - Adopted standard MIT licence. + - Automated installation of Biopython dependency. + - Development moved to GitHub, https://github.com/peterjc/pico_galaxy + - Renamed folder and adopted README.rst naming. +======= ====================================================================== + + +Developers +========== + +This script and related tools were initially developed on the following hg branches: +http://bitbucket.org/peterjc/galaxy-central/src/fasta_filter +http://bitbucket.org/peterjc/galaxy-central/src/tools + +Development has now moved to a dedicated GitHub repository: +https://github.com/peterjc/pico_galaxy + +For making the "Galaxy Tool Shed" http://toolshed.g2.bx.psu.edu/ tarball use +the following command from the Galaxy root folder:: + + $ tar -czf seq_primer_clip.tar.gz tools/seq_primer_clip/README.rst tools/seq_primer_clip/seq_primer_clip.* tools/seq_primer_clip/repository_dependencies.xml test-data/dop_primers.fasta test-data/MID4_GLZRM4E04_rnd30* + +Check this worked:: + + $ tar -tzf seq_primer_clip.tar.gz + tools/seq_primer_clip/README.rst + tools/seq_primer_clip/seq_primer_clip.xml + tools/seq_primer_clip/seq_primer_clip.py + tools/seq_primer_clip/repository_dependencies.xml + test-data/dop_primers.fasta + test-data/MID4_GLZRM4E04_rnd30.fasta + test-data/MID4_GLZRM4E04_rnd30.fastqsanger + test-data/MID4_GLZRM4E04_rnd30_fclip.fasta + test-data/MID4_GLZRM4E04_rnd30_fclip.fastqsanger + test-data/MID4_GLZRM4E04_rnd30_fclip.sff + test-data/MID4_GLZRM4E04_rnd30_frclip.fasta + test-data/MID4_GLZRM4E04_rnd30_frclip.fastqsanger + test-data/MID4_GLZRM4E04_rnd30_frclip.sff + test-data/MID4_GLZRM4E04_rnd30.sff + + +Licence (MIT) +============= + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE.
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/tools/seq_primer_clip/repository_dependencies.xml Thu Sep 26 10:33:37 2013 -0400 @@ -0,0 +1,6 @@ +<?xml version="1.0"?> +<repositories description="This requires Biopython as a dependency."> +<!-- Leave out the tool shed and revision to get the current + tool shed and latest revision at the time of upload --> +<repository changeset_revision="2f6c871cfa35" name="package_biopython_1_61" owner="biopython" toolshed="http://testtoolshed.g2.bx.psu.edu" /> +</repositories>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/tools/seq_primer_clip/seq_primer_clip.py Thu Sep 26 10:33:37 2013 -0400 @@ -0,0 +1,358 @@ +#!/usr/bin/env python +"""Looks for the given primer sequences and clips matching SFF reads. + +Takes eight command line options, input read filename, input read format, +input primer FASTA filename, type of primers (forward, reverse or reverse- +complement), number of mismatches (currently only 0, 1 and 2 are supported), +minimum length to keep a read (after primer trimming), should primer-less +reads be kept (boolean), and finally the output sequence filename. + +Both the primer and read sequences can contain IUPAC ambiguity codes like N. + +This supports FASTA, FASTQ and SFF sequence files. Colorspace reads are not +supported. + +The mismatch parameter does not consider gapped alignemnts, however the +special case of missing bases at the very start or end of the read is handled. +e.g. a primer sequence CCGACTCGAG will match a read starting CGACTCGAG... +if one or more mismatches are allowed. + +This can also be used for stripping off (and optionally filtering on) barcodes. + +Note that only the trim/clip values in the SFF file are changed, not the flow +information of the full read sequence. + +This script is copyright 2011-2013 by Peter Cock, The James Hutton Institute +(formerly the Scottish Crop Research Institute, SCRI), UK. All rights reserved. +See accompanying text file for licence details (MIT/BSD style). + +This is version 0.0.8 of the script. Currently it uses Python's regular +expression engine for finding the primers, which for my needs is fast enough. +""" +import sys +import re +from galaxy_utils.sequence.fasta import fastaReader, fastaWriter +from galaxy_utils.sequence.fastq import fastqReader, fastqWriter + +if "-v" in sys.argv or "--version" in sys.argv: + print "v0.0.5" + sys.exit(0) + +def stop_err(msg, err=1): + sys.stderr.write(msg) + sys.exit(err) + +try: + from Bio.Seq import reverse_complement + from Bio.SeqIO.SffIO import SffIterator, SffWriter +except ImportError: + stop_err("Requires Biopython 1.54 or later") +try: + from Bio.SeqIO.SffIO import ReadRocheXmlManifest +except ImportError: + #Prior to Biopython 1.56 this was a private function + from Bio.SeqIO.SffIO import _sff_read_roche_index_xml as ReadRocheXmlManifest + +#Parse Command Line +try: + in_file, seq_format, primer_fasta, primer_type, mm, min_len, keep_negatives, out_file = sys.argv[1:] +except ValueError: + stop_err("Expected 8 arguments, got %i:\n%s" % (len(sys.argv)-1, " ".join(sys.argv))) + +if in_file == primer_fasta: + stop_err("Same file given as both primer sequences and sequences to clip!") +if in_file == out_file: + stop_err("Same file given as both sequences to clip and output!") +if primer_fasta == out_file: + stop_err("Same file given as both primer sequences and output!") + +try: + mm = int(mm) +except ValueError: + stop_err("Expected non-negative integer number of mismatches (e.g. 0 or 1), not %r" % mm) +if mm < 0: + stop_err("Expected non-negtive integer number of mismatches (e.g. 0 or 1), not %r" % mm) +if mm not in [0,1,2]: + raise NotImplementedError + +try: + min_len = int(min_len) +except ValueError: + stop_err("Expected non-negative integer min_len (e.g. 0 or 1), not %r" % min_len) +if min_len < 0: + stop_err("Expected non-negtive integer min_len (e.g. 0 or 1), not %r" % min_len) + + +if keep_negatives.lower() in ["true", "yes", "on"]: + keep_negatives = True +elif keep_negatives.lower() in ["false", "no", "off"]: + keep_negatives = False +else: + stop_err("Expected boolean for keep_negatives (e.g. true or false), not %r" % keep_negatives) + + +if primer_type.lower() == "forward": + forward = True + rc = False +elif primer_type.lower() == "reverse": + forward = False + rc = False +elif primer_type.lower() == "reverse-complement": + forward = False + rc = True +else: + stop_err("Expected foward, reverse or reverse-complement not %r" % primer_type) + + +ambiguous_dna_values = { + "A": "A", + "C": "C", + "G": "G", + "T": "T", + "M": "ACM", + "R": "AGR", + "W": "ATW", + "S": "CGS", + "Y": "CTY", + "K": "GTK", + "V": "ACGMRSV", + "H": "ACTMWYH", + "D": "AGTRWKD", + "B": "CGTSYKB", + "X": ".", #faster than [GATCMRWSYKVVHDBXN] or even [GATC] + "N": ".", + } + +ambiguous_dna_re = {} +for letter, values in ambiguous_dna_values.iteritems(): + if len(values) == 1: + ambiguous_dna_re[letter] = values + else: + ambiguous_dna_re[letter] = "[%s]" % values + + +def make_reg_ex(seq): + return "".join(ambiguous_dna_re[letter] for letter in seq) + +def make_reg_ex_mm(seq, mm): + if mm > 2: + raise NotImplementedError("At most 2 mismatches allowed!") + seq = seq.upper() + yield make_reg_ex(seq) + for i in range(1,mm+1): + #Missing first/last i bases at very start/end of sequence + for reg in make_reg_ex_mm(seq[i:], mm-i): + yield "^" + reg + for reg in make_reg_ex_mm(seq[:-i], mm-i): + yield "$" + reg + if mm >= 1: + for i,letter in enumerate(seq): + #We'll use a set to remove any duplicate patterns + #if letter not in "NX": + pattern = seq[:i] + "N" + seq[i+1:] + assert len(pattern) == len(seq), "Len %s is %i, len %s is %i" \ + % (pattern, len(pattern), seq, len(seq)) + yield make_reg_ex(pattern) + if mm >=2: + for i,letter in enumerate(seq): + #We'll use a set to remove any duplicate patterns + #if letter not in "NX": + for k,letter in enumerate(seq[i+1:]): + #We'll use a set to remove any duplicate patterns + #if letter not in "NX": + pattern = seq[:i] + "N" + seq[i+1:i+1+k] + "N" + seq[i+k+2:] + assert len(pattern) == len(seq), "Len %s is %i, len %s is %i" \ + % (pattern, len(pattern), seq, len(seq)) + yield make_reg_ex(pattern) + +def load_primers_as_re(primer_fasta, mm, rc=False): + #Read primer file and record all specified sequences + primers = set() + in_handle = open(primer_fasta, "rU") + reader = fastaReader(in_handle) + count = 0 + for record in reader: + if rc: + seq = reverse_complement(record.sequence) + else: + seq = record.sequence + #primers.add(re.compile(make_reg_ex(seq))) + count += 1 + for pattern in make_reg_ex_mm(seq, mm): + primers.add(pattern) + in_handle.close() + #Use set to avoid duplicates, sort to have longest first + #(so more specific primers found before less specific ones) + primers = sorted(set(primers), key=lambda p: -len(p)) + return count, re.compile("|".join(primers)) #make one monster re! + + + +#Read primer file and record all specified sequences +count, primer = load_primers_as_re(primer_fasta, mm, rc) +print "%i primer sequences" % count + +short_neg = 0 +short_clipped = 0 +clipped = 0 +negs = 0 + +if seq_format.lower()=="sff": + #SFF is different because we just change the trim points + if forward: + def process(records): + global short_clipped, short_neg, clipped, negs + for record in records: + left_clip = record.annotations["clip_qual_left"] + right_clip = record.annotations["clip_qual_right"] + seq = str(record.seq)[left_clip:right_clip].upper() + result = primer.search(seq) + if result: + #Forward primer, take everything after it + #so move the left clip along + if len(seq) - result.end() >= min_len: + record.annotations["clip_qual_left"] = left_clip + result.end() + clipped += 1 + yield record + else: + short_clipped += 1 + elif keep_negatives: + if len(seq) >= min_len: + negs += 1 + yield record + else: + short_neg += 1 + else: + def process(records): + global short_clipped, short_neg, clipped, negs + for record in records: + left_clip = record.annotations["clip_qual_left"] + right_clip = record.annotations["clip_qual_right"] + seq = str(record.seq)[left_clip:right_clip].upper() + result = primer.search(seq) + if result: + #Reverse primer, take everything before it + #so move the right clip back + new_len = result.start() + if new_len >= min_len: + record.annotations["clip_qual_right"] = left_clip + new_len + clipped += 1 + yield record + else: + short_clipped += 1 + elif keep_negatives: + if len(seq) >= min_len: + negs += 1 + yield record + else: + short_neg += 1 + + in_handle = open(in_file, "rb") + try: + manifest = ReadRocheXmlManifest(in_handle) + except ValueError: + manifest = None + in_handle.seek(0) + out_handle = open(out_file, "wb") + writer = SffWriter(out_handle, xml=manifest) + writer.write_file(process(SffIterator(in_handle))) + #End of SFF code +elif seq_format.lower().startswith("fastq"): + in_handle = open(in_file, "rU") + out_handle = open(out_file, "w") + reader = fastqReader(in_handle) + writer = fastqWriter(out_handle) + if forward: + for record in reader: + seq = record.sequence.upper() + result = primer.search(seq) + if result: + #Forward primer, take everything after it + cut = result.end() + record.sequence = seq[cut:] + if len(record.sequence) >= min_len: + record.quality = record.quality[cut:] + clipped += 1 + writer.write(record) + else: + short_clipped += 1 + elif keep_negatives: + if len(record) >= min_len: + negs += 1 + writer.write(record) + else: + short_negs += 1 + else: + for record in reader: + seq = record.sequence.upper() + result = primer.search(seq) + if result: + #Reverse primer, take everything before it + cut = result.start() + record.sequence = seq[:cut] + if len(record.sequence) >= min_len: + record.quality = record.quality[:cut] + clipped += 1 + writer.write(record) + else: + short_clipped += 1 + elif keep_negatives: + if len(record) >= min_len: + negs += 1 + writer.write(record) + else: + short_negs += 1 +elif seq_format.lower()=="fasta": + in_handle = open(in_file, "rU") + out_handle = open(out_file, "w") + reader = fastaReader(in_handle) + writer = fastaWriter(out_handle) + #Following code is identical to that for FASTQ but without editing qualities + if forward: + for record in reader: + seq = record.sequence.upper() + result = primer.search(seq) + if result: + #Forward primer, take everything after it + cut = result.end() + record.sequence = seq[cut:] + if len(record.sequence) >= min_len: + clipped += 1 + writer.write(record) + else: + short_clipped += 1 + elif keep_negatives: + if len(record) >= min_len: + negs += 1 + writer.write(record) + else: + short_negs += 1 + else: + for record in reader: + seq = record.sequence.upper() + result = primer.search(seq) + if result: + #Reverse primer, take everything before it + cut = result.start() + record.sequence = seq[:cut] + if len(record.sequence) >= min_len: + clipped += 1 + writer.write(record) + else: + short_clipped += 1 + elif keep_negatives: + if len(record) >= min_len: + negs += 1 + writer.write(record) + else: + short_negs += 1 +else: + stop_err("Unsupported file type %r" % seq_format) +in_handle.close() +out_handle.close() + +print "Kept %i clipped reads," % clipped +print "discarded %i short." % short_clipped +if keep_negatives: + print "Kept %i non-matching reads," % negs + print "discarded %i short." % short_neg
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/tools/seq_primer_clip/seq_primer_clip.xml Thu Sep 26 10:33:37 2013 -0400 @@ -0,0 +1,143 @@ +<tool id="seq_primer_clip" name="Primer clip sequences" version="0.0.10"> + <description>Trim off 5' or 3' primers</description> + <requirements> + <requirement type="package" version="1.61">biopython</requirement> + <requirement type="python-module">Bio</requirement> + </requirements> + <version_command interpreter="python">seq_primer_clip.py --version</version_command> + <command interpreter="python"> +seq_primer_clip.py $input_file $input_file.ext $primer_fasta $primer_type $mm $min_len $keep_negatives $output_file + </command> + <stdio> + <!-- Anything other than zero is an error --> + <exit_code range="1:" /> + <exit_code range=":-1" /> + </stdio> + <inputs> + <param name="input_file" type="data" format="fasta,fastq,sff" label="Sequence file to clip" description="FASTA, FASTQ, or SFF format."/> + <param name="primer_fasta" type="data" format="fasta" label="FASTA file containing primer(s)"/> + <param name="primer_type" type="select" label="Type of primers"> + <option value="Forward">Forward (5') primers</option> + <option value="Reverse">Reverse (3') primers (given with respect to the forward strand)</option> + <option value="Reverse-complement">Reverse (3') primers (given with respect to the reverse strand)</option> + </param> + <param name="mm" type="integer" value="0" label="How many mismatches to allow? (0, 1 or 2)"> + <validator type="in_range" min="0" max="2" /> + </param> + <param name="keep_negatives" type="boolean" value="false" label="Keep reads with no matched primer"/> + <param name="min_len" type="integer" label="Minimum length for (clipped) sequences " value="1"/> + </inputs> + <outputs> + <data name="output_file" format="data" label="$primer_type primer clipped"> + <!-- TODO - Replace this with format="input:input_fastq" if/when that works --> + <change_format> + <when input_dataset="input_file" attribute="extension" value="sff" format="sff" /> + <when input_dataset="input_file" attribute="extension" value="fasta" format="fasta" /> + <when input_dataset="input_file" attribute="extension" value="fastq" format="fastq" /> + <when input_dataset="input_file" attribute="extension" value="fastqsanger" format="fastqsanger" /> + <when input_dataset="input_file" attribute="extension" value="fastqsolexa" format="fastqsolexa" /> + <when input_dataset="input_file" attribute="extension" value="fastqillumina" format="fastqillumina" /> + <when input_dataset="input_file" attribute="extension" value="fastqcssanger" format="fastqcssanger" /> + </change_format> + </data> + </outputs> + <tests> + <test> + <param name="input_file" value="MID4_GLZRM4E04_rnd30.fasta" ftype="fasta" /> + <param name="primer_fasta" value="dop_primers.fasta" /> + <param name="primer_type" value="Forward" /> + <param name="mm" value="2" /> + <param name="keep_negatives" value="false" /> + <param name="min_len" value="35" /> + <output name="output_file" file="MID4_GLZRM4E04_rnd30_fclip.fasta" ftype="fasta" /> + </test> + <test> + <param name="input_file" value="MID4_GLZRM4E04_rnd30.fastqsanger" ftype="fastqsanger" /> + <param name="primer_fasta" value="dop_primers.fasta" /> + <param name="primer_type" value="Forward" /> + <param name="mm" value="2" /> + <param name="keep_negatives" value="false" /> + <param name="min_len" value="35" /> + <output name="output_file" file="MID4_GLZRM4E04_rnd30_fclip.fastqsanger" ftype="fastqsanger" /> + </test> + <test> + <param name="input_file" value="MID4_GLZRM4E04_rnd30.sff" ftype="sff" /> + <param name="primer_fasta" value="dop_primers.fasta" /> + <param name="primer_type" value="Forward" /> + <param name="mm" value="2" /> + <param name="keep_negatives" value="false" /> + <param name="min_len" value="35" /> + <output name="output_file" file="MID4_GLZRM4E04_rnd30_fclip.sff" ftype="sff" /> + </test> + <test> + <param name="input_file" value="MID4_GLZRM4E04_rnd30_fclip.fasta" ftype="fasta" /> + <param name="primer_fasta" value="dop_primers.fasta" /> + <param name="primer_type" value="Reverse" /> + <param name="mm" value="2" /> + <param name="keep_negatives" value="true" /> + <param name="min_len" value="35" /> + <output name="output_file" file="MID4_GLZRM4E04_rnd30_frclip.fasta" ftype="fasta" /> + </test> + <test> + <param name="input_file" value="MID4_GLZRM4E04_rnd30_fclip.fastqsanger" ftype="fastqsanger" /> + <param name="primer_fasta" value="dop_primers.fasta" /> + <param name="primer_type" value="Reverse" /> + <param name="mm" value="2" /> + <param name="keep_negatives" value="true" /> + <param name="min_len" value="35" /> + <output name="output_file" file="MID4_GLZRM4E04_rnd30_frclip.fastqsanger" ftype="fastqsanger" /> + </test> + <test> + <param name="input_file" value="MID4_GLZRM4E04_rnd30_fclip.sff" ftype="sff" /> + <param name="primer_fasta" value="dop_primers.fasta" /> + <param name="primer_type" value="Reverse" /> + <param name="mm" value="2" /> + <param name="keep_negatives" value="true" /> + <param name="min_len" value="35" /> + <output name="output_file" file="MID4_GLZRM4E04_rnd30_frclip.sff" ftype="sff" /> + </test> + </tests> + <requirements> + <requirement type="python-module">Bio</requirement> + </requirements> + <help> + +**What it does** + +Looks for the given primer sequences (within the existing clipped sequence) and +further clips the reads to remove the primers and any preceding/trailing sequence. + +Reads containing a forward primer are reduced to just the sequence after (and +excluding) the forward primer. + +Reads containing a reverse primer are reduced to just the sequence before (and +excluding) the reverse primer. + +Degenerate primers can be specified using the standard IUPAC ambiguity codes, +thus a primer with an N would match A, C, T or G (or any of the IUPAC ambiguity +codes) and so on. + +Note that for SFF files only the clip/trim positions are edited - you will still +be able to extract the original full read (with any adapter sequence and poor +quality sequence) if you need to. + +.. class:: warningmark + +**Note**. This tool was initially written for Roche 454 data, and should also +work fine on Sanger or Ion Torrent as well. However, it is probably too slow +for use on large Illumina datasets. + + +**Citation** + +This tool uses Biopython. If you use this tool in scientific work leading to a +publication, please cite: + +Cock et al 2009. Biopython: freely available Python tools for computational +molecular biology and bioinformatics. Bioinformatics 25(11) 1422-3. +http://dx.doi.org/10.1093/bioinformatics/btp163 pmid:19304878. + +This tool is available to install into other Galaxy Instances via the Galaxy +Tool Shed at http://toolshed.g2.bx.psu.edu/view/peterjc/seq_primer_clip + </help> +</tool>