Skip to content

Commit

Permalink
Added submodule rampler, implemented racon wrapper, updated fragment …
Browse files Browse the repository at this point in the history
…correction to expect dual overlaps, added SAM tags to headers of polished targets, updated README and unit tests
  • Loading branch information
rvaser committed Mar 8, 2018
1 parent a6ddd48 commit 7295b9b
Show file tree
Hide file tree
Showing 13 changed files with 258 additions and 123 deletions.
3 changes: 3 additions & 0 deletions .gitmodules
Original file line number Diff line number Diff line change
Expand Up @@ -13,3 +13,6 @@
[submodule "vendor/googletest"]
path = vendor/googletest
url = https://github.com/google/googletest
[submodule "vendor/rampler"]
path = vendor/rampler
url = https://github.com/rvaser/rampler
14 changes: 14 additions & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ set(CMAKE_CXX_STANDARD 11)
set(CMAKE_CXX_STANDARD_REQUIRED ON)

option(racon_build_tests "Build racon unit tests" OFF)
option(racon_build_wrapper "Build racon wrapper" OFF)

add_executable(racon
src/main.cpp
Expand Down Expand Up @@ -46,3 +47,16 @@ if (racon_build_tests)
target_link_libraries(racon_test bioparser spoa thread_pool pthread
edlib_static gtest_main)
endif(racon_build_tests)

if (racon_build_wrapper)
set(racon_path ${PROJECT_BINARY_DIR}/bin/racon)
set(rampler_path ${PROJECT_BINARY_DIR}/vendor/rampler/bin/rampler)
configure_file(${PROJECT_SOURCE_DIR}/scripts/racon_wrapper.py
${PROJECT_BINARY_DIR}/${CMAKE_FILES_DIRECTORY}/racon_wrapper)
file(COPY ${PROJECT_BINARY_DIR}/${CMAKE_FILES_DIRECTORY}/racon_wrapper
DESTINATION ${PROJECT_BINARY_DIR}/bin
FILE_PERMISSIONS OWNER_READ OWNER_EXECUTE GROUP_READ GROUP_EXECUTE
WORLD_READ WORLD_EXECUTE)

add_subdirectory(vendor/rampler)
endif(racon_build_wrapper)
27 changes: 23 additions & 4 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,11 +9,13 @@ Consensus module for raw de novo DNA assembly of long uncorrected reads.
## Description
Racon is intended as a standalone consensus module to correct raw contigs generated by rapid assembly methods which do not include a consensus step. The goal of Racon is to generate genomic consensus which is of similar or better quality compared to the output generated by assembly methods which employ both error correction and consensus steps, while providing a speedup of several times compared to those methods. It supports data produced by both Pacific Biosciences and Oxford Nanopore Technologies.

Racon can be used as a polishing tool after the assembly with either Illumina data or data produced by third generation of sequencing. The type of data inputed is automatically detected.
Racon can be used as a polishing tool after the assembly with **either Illumina data or data produced by third generation of sequencing**. The type of data inputed is automatically detected.

Racon takes as input only three files: contigs in FASTA/FASTQ format, reads in FASTA/FASTQ format and overlaps/alignments between the reads and the contigs in MHAP/PAF/SAM format. Output is a set of polished contigs in FASTA format printed to stdout. All input files can be compressed with gzip.
Racon takes as input only three files: contigs in FASTA/FASTQ format, reads in FASTA/FASTQ format and overlaps/alignments between the reads and the contigs in MHAP/PAF/SAM format. Output is a set of polished contigs in FASTA format printed to stdout. All input files **can be compressed with gzip**.

Racon can also be used as a read error-correction tool. In this scenario, the MHAP/PAF/SAM file needs to contain pairwise overlaps between reads without any dual or self overlaps.
Racon can also be used as a read error-correction tool. In this scenario, the MHAP/PAF/SAM file needs to contain pairwise overlaps between reads **with dual overlaps**.

A **wrapper script** is also available to enable easier usage to the end-user for large datasets. It has the same interface as racon but adds two additional features from the outside. Sequences can be **subsampled** to decrease the total execution time (accuracy might be lower) while target sequences can be **split** into smaller chunks and run sequentially to decrease memory consumption. Both features can be run at the same time as well.

## Dependencies
1. gcc 4.8+ or clang 3.4+
Expand All @@ -37,8 +39,12 @@ Optionally, you can run `sudo make install` to install racon executable to your

***Note***: if you omitted `--recursive` from `git clone`, run `git submodule update --init --recursive` before proceeding with compilation.

To build unit tests add `-Dracon_build_tests=ON` while running `cmake`. After installation, an executable named `racon_test` will be created in `build/bin`.

To build the wrapper script add `-Dracon_build_wrapper=ON` while running `cmake`. After installation, an executable named `racon_wrapper` (python script) will be created in `build/bin`.

## Usage
Usage of racon is as following:
Usage of `racon` is as following:

racon [options ...] <sequences> <overlaps> <target sequences>

Expand Down Expand Up @@ -86,6 +92,19 @@ Usage of racon is as following:
-h, --help
prints the usage

`racon_test` is run without any parameters.

Usage of `racon_wrapper` equals the one of `racon` with two additional parameters:

...
options:
--split <int>
split target sequences into chunks of desired size in bytes
--subsample <int> <int>
subsample sequences to desired coverage (2nd argument) given the
reference length (1st argument)
...

## Contact information

For additional information, help and bug reports please send an email to one of the following: ivan.sovic@irb.hr, robert.vaser@fer.hr, mile.sikic@fer.hr, nagarajann@gis.a-star.edu.sg
Expand Down
189 changes: 189 additions & 0 deletions scripts/racon_wrapper.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,189 @@
#!/usr/bin/env python

from __future__ import print_function
import os, sys, time, shutil, argparse, subprocess

def eprint(*args, **kwargs):
print(*args, file=sys.stderr, **kwargs)

#*******************************************************************************

class RaconWrapper:

__racon = '@racon_path@'
__rampler = '@rampler_path@'

def __init__(self, sequences, overlaps, target_sequences, split, subsample,
include_unpolished, fragment_correction, window_length, quality_threshold,
error_threshold, match, mismatch, gap, threads):

self.sequences = os.path.abspath(sequences)
self.subsampled_sequences = None
self.overlaps = os.path.abspath(overlaps)
self.target_sequences = os.path.abspath(target_sequences)
self.split_target_sequences = []
self.chunk_size = split
self.reference_length, self.coverage = subsample if subsample is not None\
else (None, None)
self.include_unpolished = include_unpolished
self.fragment_correction = fragment_correction
self.window_length = window_length
self.quality_threshold = quality_threshold
self.error_threshold = error_threshold
self.match = match
self.mismatch = mismatch
self.gap = gap
self.threads = threads
self.work_directory = os.path.dirname(os.path.realpath(__file__)) +\
'/racon_work_directory_' + str(time.time())

try:
os.makedirs(self.work_directory)
except OSError:
if (not os.path.isdir(self.work_directory)):
eprint('[RaconWrapper::run] error: unable to create work directory!')
sys.exit(1)

def __del__(self):
try:
shutil.rmtree(self.work_directory)
except OSError:
eprint('[RaconWrapper::run] warning: unable to clean work directory!')

def run(self):
# run preprocess
eprint('[RaconWrapper::run] preparing data with rampler')
if (self.reference_length is not None and self.coverage is not None):
try:
p = subprocess.Popen([RaconWrapper.__rampler, '-o', self.work_directory,
'subsample', self.sequences, self.reference_length, self.coverage])
except OSError:
eprint('[RaconWrapper::run] error: unable to run rampler!')
sys.exit(1)
p.communicate()
if (p.returncode != 0):
sys.exit(1)

base_name = os.path.basename(self.sequences).split('.')[0]
extension = '.fasta' if (self.sequences.endswith('.fasta') or\
self.sequences.endswith('.fasta.gz') or\
self.sequences.endswith('.fa') or\
self.sequences.endswith('.fa.gz')) else\
'.fastq'
self.subsampled_sequences = os.path.join(self.work_directory, base_name) +\
'_' + self.coverage + 'x' + extension
if (not os.path.isfile(self.subsampled_sequences)):
eprint('[RaconWrapper::run] error: unable to find subsampled sequences!')
sys.exit(1)
else:
self.subsampled_sequences = self.sequences

if (self.chunk_size is not None):
try:
p = subprocess.Popen([RaconWrapper.__rampler, '-o', self.work_directory,
'split', self.target_sequences, self.chunk_size])
except OSError:
eprint('[RaconWrapper::run] error: unable to run rampler!')
sys.exit(1)
p.communicate()
if (p.returncode != 0):
sys.exit(1)

base_name = os.path.basename(self.target_sequences).split('.')[0]
extension = '.fasta' if (self.target_sequences.endswith('.fasta') or\
self.target_sequences.endswith('.fasta.gz') or\
self.target_sequences.endswith('.fa') or\
self.target_sequences.endswith('.fa.gz')) else\
'.fastq'

i = 0
while (True):
target_sequences_part = os.path.join(self.work_directory, base_name) +\
'_' + str(i) + extension
if (not os.path.isfile(target_sequences_part)):
break
self.split_target_sequences.append(target_sequences_part)
i += 1
if (len(self.split_target_sequences) == 0):
eprint('[RaconWrapper::run] error: unable to find split target sequences!')
sys.exit(1)
else:
self.split_target_sequences.append(self.target_sequences)

racon_params = [RaconWrapper.__racon]
if (self.include_unpolished == True): racon_params.append('-u')
if (self.fragment_correction == True): racon_params.append('-f')
racon_params.extend(['-w', str(self.window_length),
'-q', str(self.quality_threshold),
'-e', str(self.error_threshold),
'-m', str(self.match),
'-x', str(self.mismatch),
'-g', str(self.gap),
'-t', str(self.threads),
self.subsampled_sequences, self.overlaps, ""])

for target_sequences_part in self.split_target_sequences:
eprint('[RaconWrapper::run] processing data with racon')
racon_params[-1] = target_sequences_part
try:
p = subprocess.Popen(racon_params)
except OSError:
eprint('[RaconWrapper::run] error: unable to run racon!')
sys.exit(1)
p.communicate()
if (p.returncode != 0):
sys.exit(1)

self.subsampled_sequences = None
self.split_target_sequences = []

#*******************************************************************************

if __name__ == '__main__':

parser = argparse.ArgumentParser(description='''Racon_wrapper encapsulates
racon and adds two additional features from the outside to enable easier
usage to the end-user. Sequences can now be subsampled to decrease the
total execution time (accuracy might be lower) while target
sequences can be split into smaller chunks and run sequentially to
decrease memory consumption. Both features can be run at the same time
as well! The usage equals the one of racon.''',
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
parser.add_argument('sequences', help='''input file in FASTA/FASTQ format
(can be compressed with gzip) containing sequences used for correction''')
parser.add_argument('overlaps', help='''input file in MHAP/PAF/SAM format
(can be compressed with gzip) containing overlaps between sequences and
target sequences''')
parser.add_argument('target_sequences', help='''input file in FASTA/FASTQ
format (can be compressed with gzip) containing sequences which will be
corrected''')
parser.add_argument('--split', help='''split target sequences into chunks of
desired size in bytes''')
parser.add_argument('--subsample', nargs=2, help='''subsample sequences to
desired coverage (2nd argument) given the reference length (1st argument)''')
parser.add_argument('-u', '--include-unpolished', action='store_true',
help='''output unpolished target sequences''')
parser.add_argument('-f', '--fragment-correction', action='store_true',
help='''perform fragment correction instead of contig polishing
(overlaps file should not contain dual/self overlaps!)''')
parser.add_argument('-w', '--window-length', default=500, help='''size of
window on which POA is performed''')
parser.add_argument('-q', '--quality-threshold', default=10.0,
help='''threshold for average base quality of windows used in poa''')
parser.add_argument('-e', '--error-threshold', default=0.3, help='''maximum
allowed error rate used for filtering overlaps''')
parser.add_argument('-m', '--match', default=5, help='''score for matching
bases''')
parser.add_argument('-x', '--mismatch', default=-4, help='''score for
mismatching bases''')
parser.add_argument('-g', '--gap', default=-8, help='''gap penalty (must be
negative)''')
parser.add_argument('-t', '--threads', default=1, help='''number of threads''')

args = parser.parse_args()

racon = RaconWrapper(args.sequences, args.overlaps, args.target_sequences,
args.split, args.subsample, args.include_unpolished,
args.fragment_correction, args.window_length, args.quality_threshold,
args.error_threshold, args.match, args.mismatch, args.gap, args.threads)
racon.run()
6 changes: 2 additions & 4 deletions src/main.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
#include "sequence.hpp"
#include "polisher.hpp"

static const char* version = "v1.1.1";
static const char* version = "v1.2.0";

static struct option options[] = {
{"include-unpolished", no_argument, 0, 'u'},
Expand Down Expand Up @@ -128,9 +128,7 @@ void help() {
"\n"
" options:\n"
" -u, --include-unpolished\n"
" output unpolished target sequences (each sequence contains\n"
" a tag in its header (C:<float>) which represents the\n"
" percentage of polished windows)\n"
" output unpolished target sequences\n"
" -f, --fragment-correction\n"
" perform fragment correction instead of contig polishing\n"
" (overlaps file should not contain dual/self overlaps!)\n"
Expand Down
Loading

0 comments on commit 7295b9b

Please sign in to comment.