forked from dwinter/accuMUlate
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathboost_input_utils.cc
322 lines (283 loc) · 14.8 KB
/
boost_input_utils.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
/*
* boost_input_utils.cc
*
* Created on: 12/12/14
* Author: Steven Wu
* Modified DJW Aug-15
*/
#include <numeric>
#include "version.h"
#include "parsers.h"
#include "boost_input_utils.h"
void validate(boost::any& v, const vector<string>& values, nfreqs* target_type, int){
nfreqs result;
vector<double> pi;
for(vector<string>::const_iterator it = values.begin();
it != values.end(); ++it){
stringstream ss(*it);
copy(istream_iterator<double>(ss), istream_iterator<double>(), back_inserter(pi));
}
if(pi.size() != 4){
throw boost::program_options::invalid_option_value("Must specify 4 (and only 4) nucleotide frequencies");
}
if( accumulate(pi.begin(), pi.end(), 0.0) != 1.0){
throw boost::program_options::invalid_option_value("Nucleotide frequencies don't sum to 1.0"); }
result.freqs = pi;
v= result;
}
namespace BoostUtils {
using namespace std;
namespace po = boost::program_options;
static bool file_exists(const std::string &name) {
struct stat buffer;
return (stat(name.c_str(), &buffer) == 0);
}
SampleMap ParseSamples(boost::program_options::variables_map &vm, BamTools::SamHeader &header){
//OK. We want to store data form each sample in a unique position in
//a ReadDataVector. The caller also assumes that the ancestral sequence
//is in the 0th poistion of the ReadDataVector.
//The sample info in the BAM file is all in the ReadGroup headers, and
//we want to be able to exclude some samples.So, we ask users to provide
//a list of included samples.
//To acheive all this we start parsing through every read group in the
//BAM. If it's one to include, we add it the index map and remove it
//from the list of samples to include. If it's not in our include list
//we warn the user we are skipping some of the data.
SampleMap name_map;
vector<string> keepers = vm["sample-name"].as< vector<string> >();
string anc_tag = vm["ancestor"].as<string>();
uint16_t sindex = 1; //ancestor has sindex==0
bool ancestor_in_BAM = false;
for(auto it = header.ReadGroups.Begin(); it!= header.ReadGroups.End(); it++){
if(it->HasSample()){
auto samp_in_keepers = find(keepers.begin(), keepers.end(), it->Sample);
if (samp_in_keepers == keepers.end()){
if(it->Sample == anc_tag ){
name_map[it->Sample] = 0;
ancestor_in_BAM = true;
}
else {
name_map[it->Sample] = MAX_UINT32 ;
cerr << "Warning: excluding data from '" << it->Sample <<
"' which is included in the BAM file but not the list of included samples" << endl;
}
}
else {
auto s = name_map.find(it->Sample);
if( s == name_map.end()){//samples can have multiple readgroups...
name_map[it->Sample] = distance(keepers.begin(), samp_in_keepers) + 1;
sindex ++;
// keepers.erase(find(keepers.begin(),keepers.end(),it->Sample));
//NOTE: with erase, effectively remove all double+ samples
//Fixed version == #sample-name M28 40 44 50 531
}
}
}
}
//Once we've built the index map we can check if we now about every
//sample in the BAM file and if we have set the ancesoe
if(!ancestor_in_BAM){
cerr << "Error: No data for ancestral sample '" << anc_tag <<
"' in the specifified BAM file. Check the sample tags match" << endl;
exit(5);
}
if( (keepers.size()+1) != sindex ){
cerr << "Sample(s) note persent in BAM file: ";
for(auto s: keepers){
cerr << s << " ";
}
cerr << endl;
exit(6);
}
// And now.. go back over the read groups to map RG->sample index
SampleMap samples;
for(auto it = header.ReadGroups.Begin(); it!= header.ReadGroups.End(); it++){
if(it->HasSample()){
samples[it->ID] = name_map[it->Sample];
// cout << it->ID << " " << it->Sample << " " << name_map[it->Sample] << endl;
}
}
return samples;
}
void check_args(boost::program_options::variables_map &vm){
// Is the experimental design one of the ones we can handle?
int ploidy_ancestor = vm["ploidy-ancestor"].as<int>();
int polidy_descendant = vm["ploidy-descendant"].as<int>();
if (ploidy_ancestor > 2 or ploidy_ancestor < 1){
throw po::invalid_option_value("accuMUlate can't only deal with haploid or diploid ancestral samples");
}
if (polidy_descendant > 2 or polidy_descendant < 1){
throw po::invalid_option_value("accuMUlate can't only deal with haploid or descendant samples");
}
if (ploidy_ancestor == 1 and polidy_descendant == 2){
throw po::invalid_option_value("accuMUlate has no model for a haploid->diploid MA experiemt");
}
//Do we have the right over-dispersion params set
if (ploidy_ancestor == 1 or polidy_descendant == 1){
if(not vm.count("phi-haploid")){
throw po::invalid_option_value("Must specify phi-haploid (overdispersion for haploid sequencing)");
}
}
if (ploidy_ancestor == 2 or polidy_descendant == 2){
if(not vm.count("phi-diploid")){
throw po::invalid_option_value("Must specify phi-diploid (overdispersion for diploid sequencing)");
}
}
}
void ParseCommandLineInput(int argc, char **argv, boost::program_options::variables_map &vm) {
po::options_description cmd("Command line options");
cmd.add_options()
("help,h", "Print a help message")
("version,v", "Print the version number")
("bam,b", po::value<string>()->required(), "Path to BAM file")
("bam-index,x", po::value<string>()->default_value(""), "Path to BAM index, (defalult is <bam_path>.bai")
("reference,r", po::value<string>()->required(), "Path to reference genome")
("ancestor,a", po::value<string>()->required(), "Ancestor RG sample ID")
("sample-name,s", po::value<vector <string> >()->required(), "Sample tags to include")
("qual,q", po::value<int>()->default_value(13), "Base quality cuttoff")
("mapping-qual,m", po::value<int>()->default_value(13), "Mapping quality cuttoff")
("prob,p", po::value<double>()->default_value(0.1), "Mutaton probability cut-off")
("out,o", po::value<string>()->default_value(""), "Out file name (default is std out)")
("intervals,i", po::value<string>(), "Path to bed file")
("config,c", po::value<string>(), "Path to config file")
("header", po::value<string>()->default_value(""), "Alternative header")
("theta", po::value<double>()->required(), "theta")
("nfreqs", po::value< nfreqs >()->multitoken(), "Nucleotide frequencies")
("mu", po::value<double>()->required(), "Experiment-long mutation rate")
("seq-error", po::value<double>()->required(), "Probability of sequencing error")
("ploidy-ancestor", po::value<int>()->default_value(2), "Polidy of ancestor (1 or 2)")
("ploidy-descendant", po::value<int>()->default_value(2), "Ploidy of descendant (1 or 2)")
("phi-haploid", po::value<double>(), "Over-dispersion for haploid sequencing")
("phi-diploid", po::value<double>(), "Over-dispersion for diploid sequencing");
po::store(po::parse_command_line(argc, argv, cmd), vm);
if (vm.count("help")) {
cout << cmd << endl;
exit(-1);
}
if (vm.count("version")) {
cout << "accuMUlate " << VERSION_STRING << endl;
exit(-1);
}
if (vm.count("config")) {
string config_file = vm["config"].as<string>();
if( !file_exists(config_file) ){
std::cout << "ERROR: config file '" << config_file<< "' does not exist" << std::endl;
exit(2);
}
ifstream config_stream(vm["config"].as<string>());
po::store(po::parse_config_file(config_stream, cmd, false), vm);
}
vm.notify();
check_args(vm);
}
void ParseDenominateCommandline(int argc, char **argv, boost::program_options::variables_map &vm) {
uint32_t infinite_int = std::numeric_limits<uint32_t>::max();
double infinite_double = std::numeric_limits<double>::infinity();
po::options_description cmd("Command line options (not: all options can be set via configuration file)");
cmd.add_options()
("help,h", "Print a help message")
("version,v", "Print the version number")
("bam,b", po::value<string>()->required(), "Path to BAM file")
("bam-index,x", po::value<string>()->default_value(""), "Path to BAM index, (defalult is <bam_path>.bai")
("reference,r", po::value<string>()->required(), "Path to reference genome")
("config,c", po::value<string>(), "Path to config file")
("intervals,i", po::value<string>(), "Path to bed file")
("ancestor,a", po::value<string>(), "Ancestor RG sample ID")
("sample-name,s", po::value<vector <string> >()->required(), "Sample tags")
("qual,q", po::value<int>()->default_value(13), "Base quality cuttoff")
("mapping-qual,m", po::value<int>()->default_value(13), "Mapping quality cuttoff")
("prob,p", po::value<double>()->default_value(0.1), "Prob quality cuttoff")
("header", po::value<string>()->default_value(""), "Alternative header")
//Model params
("theta", po::value<double>()->required(), "theta")
("nfreqs", po::value< nfreqs >()->multitoken(), "Nucleotide frequencies")
("mu", po::value<double>()->required(), "Experiment-long mutation rate")
("seq-error", po::value<double>()->required(), "Probability of sequencing error")
("ploidy-ancestor", po::value<int>()->default_value(2), "Polidy of ancestor (1 or 2)")
("ploidy-descendant", po::value<int>()->default_value(2), "Ploidy of descendant (1 or 2)")
("phi-haploid", po::value<double>(), "Over-dispersion for haploid sequencing")
("phi-diploid", po::value<double>(), "Over-dispersion for diploid sequencing")
//statistical criterial, all should have default value of 0 or infinity
("min-depth", po::value<uint32_t>()->default_value(0), "Mimimum sequencing depth for a site to be included")
("max-depth", po::value<uint32_t>()->default_value(infinite_int), "Maximum sequencing depth for a site to be included")
("min-mutant-strand", po::value<uint32_t>()->default_value(0), "Minimum number of alleles supporting the mutant on each strand ")
("max-anc-in-mutant", po::value<uint32_t>()->default_value(infinite_int), "Maximum number of ancestral alleles in mutant sample")
("max-mutant-in-anc", po::value<uint32_t>()->default_value(infinite_int), "Maximum number of mutant alleles in ancestral samples")
("max-MQ-AD", po::value<double>()->default_value(infinite_double), "Maximum value of the AD test for mapping quality differences")
("max-insert-AD", po::value<double>()->default_value(infinite_double), "Maximum value of the AD test for insert length differences")
("min-strand-pval", po::value<double>()->default_value(0), "Minimum p-value for strand bias")
("min-mapping-pval", po::value<double>()->default_value(0), "Minimum p-value for paired-mapping bias");
po::store(po::parse_command_line(argc, argv, cmd), vm);
if (vm.count("help")) {
cout << cmd << endl;
exit(-1);
}
if (vm.count("version")) {
cout << "denominate " << VERSION_STRING << endl;
exit(-1);
}
if (vm.count("config")) {
string config_file = vm["config"].as<string>();
if(!file_exists(config_file) ){
std::cout << "ERROR: config file '" << config_file<< "' does not exist" << std::endl;
exit(2);
}
ifstream config_stream(config_file);
po::store(po::parse_config_file(config_stream, cmd, false), vm);
}
vm.notify();
check_args(vm);
}
// Set up everything that has to be refered to by reference
void ExtractInputVariables(boost::program_options::variables_map &vm,
BamTools::BamReader &experiment, BamTools::RefVector &references,
BamTools::SamHeader &header, LocalBamToolsUtils::Fasta &reference_genome){
string ref_file = vm["reference"].as<string>();
string index_path = vm["bam-index"].as<string>();
string bam_path = vm["bam"].as<string>();
string header_path = vm["header"].as<string>();
if (index_path == "") {
index_path = bam_path + ".bai";
}
if (!file_exists(bam_path )) {
std::cerr << "ERROR: BAM file '" << bam_path << "' does not exist" << std::endl;
exit(2);
}
experiment.Open(bam_path);
experiment.OpenIndex(index_path);
references = experiment.GetReferenceData();
if (header_path != ""){
ifstream h(header_path);
stringstream new_header;
new_header << h.rdbuf();
header = BamTools::SamHeader(new_header.str().c_str());
}
else {
header = experiment.GetHeader();
}
if (!file_exists(ref_file )) {
std::cerr << "ERROR: reference file '" << ref_file << "' does not exist" << std::endl;
exit(2);
}
if (!file_exists(ref_file + ".fai")) {
reference_genome.Open(ref_file);
reference_genome.CreateIndex(ref_file + ".fai");
}
else {
reference_genome.Open(ref_file, ref_file + ".fai");
}
}
ModelParams CreateModelParams(boost::program_options::variables_map vm) {
ModelParams params = {
vm["theta"].as<double>(),
vm["nfreqs"].as< nfreqs >().freqs,
vm["mu"].as<double>(),
vm["seq-error"].as<double>(),
vm["phi-haploid"].as<double>(),
vm["phi-diploid"].as<double>(),
vm["ploidy-ancestor"].as<int>(),
vm["ploidy-descendant"].as<int>()
};
return params;
}
}