fast-bwa/bwakit/bwa-postalt.js

525 lines
18 KiB
JavaScript
Raw Normal View History

2014-09-20 11:23:54 +08:00
/*****************************************************************
* The K8 Javascript interpreter is required to run this script. *
* *
* Source code: https://github.com/attractivechaos/k8 *
* Binary: http://sourceforge.net/projects/lh3/files/k8/ *
* *
* Data file used for generating GRCh38 ALT alignments: *
* *
* http://sourceforge.net/projects/bio-bwa/files/ *
*****************************************************************/
/******************
*** From k8.js ***
******************/
2014-09-23 22:33:15 +08:00
// Parse command-line options. A BSD getopt() clone in javascript.
2014-09-20 11:23:54 +08:00
var getopt = function(args, ostr) {
var oli; // option letter list index
if (typeof(getopt.place) == 'undefined')
getopt.ind = 0, getopt.arg = null, getopt.place = -1;
if (getopt.place == -1) { // update scanning pointer
if (getopt.ind >= args.length || args[getopt.ind].charAt(getopt.place = 0) != '-') {
getopt.place = -1;
return null;
}
if (getopt.place + 1 < args[getopt.ind].length && args[getopt.ind].charAt(++getopt.place) == '-') { // found "--"
++getopt.ind;
getopt.place = -1;
return null;
}
}
var optopt = args[getopt.ind].charAt(getopt.place++); // character checked for validity
if (optopt == ':' || (oli = ostr.indexOf(optopt)) < 0) {
if (optopt == '-') return null; // if the user didn't specify '-' as an option, assume it means null.
if (getopt.place < 0) ++getopt.ind;
return '?';
}
if (oli+1 >= ostr.length || ostr.charAt(++oli) != ':') { // don't need argument
getopt.arg = null;
if (getopt.place < 0 || getopt.place >= args[getopt.ind].length) ++getopt.ind, getopt.place = -1;
} else { // need an argument
if (getopt.place >= 0 && getopt.place < args[getopt.ind].length)
getopt.arg = args[getopt.ind].substr(getopt.place);
else if (args.length <= ++getopt.ind) { // no arg
getopt.place = -1;
if (ostr.length > 0 && ostr.charAt(0) == ':') return ':';
return '?';
} else getopt.arg = args[getopt.ind]; // white space
getopt.place = -1;
++getopt.ind;
}
return optopt;
}
2014-09-23 22:33:15 +08:00
// reverse a string
2014-09-20 11:23:54 +08:00
Bytes.prototype.reverse = function()
{
for (var i = 0; i < this.length>>1; ++i) {
var tmp = this[i];
this[i] = this[this.length - i - 1];
this[this.length - i - 1] = tmp;
}
}
2014-09-23 22:33:15 +08:00
// reverse complement a DNA string
2014-09-20 11:23:54 +08:00
Bytes.prototype.revcomp = function()
{
if (Bytes.rctab == null) {
var s1 = 'WSATUGCYRKMBDHVNwsatugcyrkmbdhvn';
var s2 = 'WSTAACGRYMKVHDBNwstaacgrymkvhdbn';
Bytes.rctab = [];
for (var i = 0; i < 256; ++i) Bytes.rctab[i] = 0;
for (var i = 0; i < s1.length; ++i)
Bytes.rctab[s1.charCodeAt(i)] = s2.charCodeAt(i);
}
for (var i = 0; i < this.length>>1; ++i) {
var tmp = this[this.length - i - 1];
this[this.length - i - 1] = Bytes.rctab[this[i]];
this[i] = Bytes.rctab[tmp];
}
2015-08-27 22:44:08 +08:00
if (this.length&1)
2014-09-20 11:23:54 +08:00
this[this.length>>1] = Bytes.rctab[this[this.length>>1]];
}
2014-09-23 22:33:15 +08:00
// create index for a list of intervals for fast interval queries; ported from bedidx.c in samtools
function intv_ovlp(intv, bits)
2014-09-20 11:23:54 +08:00
{
if (typeof bits == "undefined") bits = 13;
intv.sort(function(a,b) {return a[0]-b[0];});
// create the index
var idx = [], max = 0;
for (var i = 0; i < intv.length; ++i) {
var b = intv[i][0]>>bits;
var e = (intv[i][1]-1)>>bits;
if (b != e) {
for (var j = b; j <= e; ++j)
if (idx[j] == null) idx[j] = i;
} else if (idx[b] == null) idx[b] = i;
max = max > e? max : e;
}
// closure
return function(_b, _e) {
var x = _b >> bits;
if (x > max) return [];
var off = idx[x];
if (off == null) {
var i;
for (i = ((_e - 1) >> bits) - 1; i >= 0; --i)
if (idx[i] != null) break;
off = i < 0? 0 : idx[i];
}
var ovlp = [];
for (var i = off; i < intv.length && intv[i][0] < _e; ++i)
if (intv[i][1] > _b) ovlp.push(intv[i]);
return ovlp;
}
}
2014-09-23 22:33:15 +08:00
var re_cigar = /(\d+)([MIDSHN])/g;
/******************************
*** Generate ALT alignment ***
******************************/
// given a pos on ALT and the ALT-to-REF CIGAR, find the pos on REF
function cigar2pos(cigar, pos)
2014-09-20 11:23:54 +08:00
{
var x = 0, y = 0;
for (var i = 0; i < cigar.length; ++i) {
var op = cigar[i][0], len = cigar[i][1];
if (op == 'M') {
if (y <= pos && pos < y + len)
return x + (pos - y);
x += len, y += len;
} else if (op == 'D') {
x += len;
} else if (op == 'I') {
if (y <= pos && pos < y + len)
return x;
y += len;
} else if (op == 'S' || op == 'H') {
if (y <= pos && pos < y + len)
return -1;
y += len;
}
}
return -1;
}
2014-09-23 22:33:15 +08:00
// Parse a hit. $s is an array that looks something like ["chr1", "+12345", "100M", 5]
// Return an object keeping various information about the alignment.
function parse_hit(s, opt)
2014-09-20 11:23:54 +08:00
{
var h = {};
h.ctg = s[0];
h.start = parseInt(s[1].substr(1)) - 1;
h.rev = (s[1].charAt(0) == '-');
h.cigar = s[2];
h.NM = parseInt(s[3]);
h.hard = false;
var m, l_ins, n_ins, l_del, n_del, l_match, l_skip, l_clip;
l_ins = l_del = n_ins = n_del = l_match = l_skip = l_clip = 0;
while ((m = re_cigar.exec(h.cigar)) != null) {
var l = parseInt(m[1]);
if (m[2] == 'M') l_match += l;
else if (m[2] == 'D') ++n_del, l_del += l;
else if (m[2] == 'I') ++n_ins, l_ins += l;
else if (m[2] == 'N') l_skip += l;
else if (m[2] == 'H' || m[2] == 'S') {
l_clip += l;
if (m[2] == 'H') h.hard = true;
}
}
h.end = h.start + l_match + l_del + l_skip;
h.NM = h.NM > l_del + l_ins? h.NM : l_del + l_ins;
h.score = Math.floor((opt.a * l_match - (opt.a + opt.b) * (h.NM - l_del - l_ins) - opt.o * (n_del + n_ins) - opt.e * (l_del + l_ins)) / opt.a + .499);
h.l_query = l_match + l_ins + l_clip;
return h;
}
function print_buffer(buf2, fp_hla, hla) // output alignments
{
if (buf2.length == 0) return;
2014-10-26 10:13:31 +08:00
for (var i = 0; i < buf2.length; ++i)
print(buf2[i].join("\t"));
if (fp_hla != null) {
var name = buf2[0][0] + '/' + (buf2[0][1]>>6&3) + ((buf2[0][1]&16)? '-' : '+');
for (var x in hla) {
if (fp_hla[x] != null);
fp_hla[x].write('@' + name + '\n' + buf2[0][9] + '\n+\n' + buf2[0][10] + '\n');
}
}
}
function collect_hla_hits(idx, ctg, start, end, hla) // collect reads hit to HLA genes
{
var m, ofunc = idx[ctg];
if (ofunc == null) return;
var ovlp_alt = ofunc(start, end);
for (var i = 0; i < ovlp_alt.length; ++i)
if ((m = /^(HLA-[^\s\*]+)\*\d+/.exec(ovlp_alt[i][2])) != null)
hla[m[1]] = true;
}
2014-10-03 09:49:29 +08:00
function bwa_postalt(args)
2014-09-20 11:23:54 +08:00
{
var version = "r985";
var c, opt = { a:1, b:4, o:6, e:1, min_mapq:10, min_sc:90, max_nm_sc:10, min_pa_ratio:1 };
while ((c = getopt(args, 'vp:r:')) != null) {
if (c == 'p') opt.pre = getopt.arg;
2014-10-21 02:17:56 +08:00
else if (c == 'r') opt.min_pa_ratio = parseFloat(getopt.arg);
else if (c == 'v') { print(version); exit(0); }
2014-10-15 04:31:40 +08:00
}
2014-10-20 08:53:47 +08:00
if (opt.min_pa_ratio > 1.) opt.min_pa_ratio = 1.;
2014-10-15 04:31:40 +08:00
2014-10-03 09:49:29 +08:00
if (args.length == getopt.ind) {
print("");
2014-10-15 04:31:40 +08:00
print("Usage: k8 bwa-postalt.js [options] <alt.sam> [aln.sam]\n");
print("Options: -p STR prefix of output files containting sequences matching HLA genes [null]");
2014-10-21 12:22:59 +08:00
print(" -r FLOAT reduce mapQ to 0 if not overlapping lifted best and pa<FLOAT ["+opt.min_pa_ratio+"]");
print(" -v show version number");
2014-10-03 09:49:29 +08:00
print("");
2014-10-21 12:22:59 +08:00
print("Note: This script extracts the XA tag, lifts the mapping positions of ALT hits to");
2014-10-03 09:49:29 +08:00
print(" the primary assembly, groups them and then estimates mapQ across groups. If");
print(" a non-ALT hit overlaps a lifted ALT hit, its mapping quality is set to the");
print(" smaller between its original mapQ and the adjusted mapQ of the ALT hit. If");
print(" multiple ALT hits are lifted to the same position, they will yield new SAM");
print(" lines with the same mapQ.");
print("");
exit(1);
}
var aux = new Bytes(); // used for reverse and reverse complement
var buf = new Bytes(); // line reading buffer
2014-10-03 09:49:29 +08:00
// read ALT-to-REF alignment
var intv_alt = {}, intv_pri = {}, hla_ctg = {}, is_alt = {}, hla_chr = null;
2014-10-03 09:49:29 +08:00
var file = new File(args[getopt.ind]);
2014-09-20 11:23:54 +08:00
while (file.readline(buf) >= 0) {
var line = buf.toString();
if (line.charAt(0) == '@') continue;
var t = line.split("\t");
if (t.length < 11) continue; // incomplete lines
is_alt[t[0]] = true;
2014-10-01 08:50:26 +08:00
var pos = parseInt(t[3]) - 1;
2014-09-20 11:23:54 +08:00
var flag = parseInt(t[1]);
if ((flag&4) || t[2] == '*') continue;
2014-10-01 08:50:26 +08:00
var m, cigar = [], l_qaln = 0, l_tlen = 0, l_qclip = 0;
if ((m = /^(HLA-[^\s\*]+)\*\d+/.exec(t[0])) != null) { // read HLA contigs
if (hla_ctg[m[1]] == null) hla_ctg[m[1]] = 0;
++hla_ctg[m[1]];
hla_chr = t[2];
}
2014-09-20 11:23:54 +08:00
while ((m = re_cigar.exec(t[5])) != null) {
var l = parseInt(m[1]);
cigar.push([m[2] != 'H'? m[2] : 'S', l]); // convert hard clip to soft clip
if (m[2] == 'M') l_qaln += l, l_tlen += l;
else if (m[2] == 'I') l_qaln += l;
2014-09-20 11:23:54 +08:00
else if (m[2] == 'S' || m[2] == 'H') l_qclip += l;
2014-10-01 08:50:26 +08:00
else if (m[2] == 'D' || m[2] == 'N') l_tlen += l;
2014-09-20 11:23:54 +08:00
}
var j = flag&16? cigar.length-1 : 0;
var start = cigar[j][0] == 'S'? cigar[j][1] : 0;
2014-10-01 08:50:26 +08:00
if (intv_alt[t[0]] == null) intv_alt[t[0]] = [];
intv_alt[t[0]].push([start, start + l_qaln, l_qaln + l_qclip, t[2], flag&16? true : false, pos - 1, cigar, pos + l_tlen]);
2014-10-03 09:49:29 +08:00
if (intv_pri[t[2]] == null) intv_pri[t[2]] = [];
intv_pri[t[2]].push([pos, pos + l_tlen, t[0]]);
2014-09-20 11:23:54 +08:00
}
file.close();
2014-10-03 09:49:29 +08:00
var idx_alt = {}, idx_pri = {};
for (var ctg in intv_alt) idx_alt[ctg] = intv_ovlp(intv_alt[ctg]);
for (var ctg in intv_pri) idx_pri[ctg] = intv_ovlp(intv_pri[ctg]);
2014-09-20 11:23:54 +08:00
// initialize the list of HLA contigs
var fp_hla = null;
if (opt.pre) {
fp_hla = {};
for (var h in hla_ctg)
fp_hla[h] = new File(opt.pre + '.' + h + '.fq', "w");
}
2014-09-20 11:23:54 +08:00
// process SAM
var buf2 = [], hla = {};
2014-09-20 11:23:54 +08:00
file = args.length - getopt.ind >= 2? new File(args[getopt.ind+1]) : new File();
while (file.readline(buf) > 0) {
2014-09-20 11:23:54 +08:00
var m, line = buf.toString();
2014-09-20 11:23:54 +08:00
if (line.charAt(0) == '@') { // print and then skip the header line
print(line);
continue;
}
var t = line.split("\t");
t[1] = parseInt(t[1]); t[3] = parseInt(t[3]); t[4] = parseInt(t[4]);
2014-09-23 22:33:15 +08:00
// print bufferred reads
if (buf2.length && (buf2[0][0] != t[0] || (buf2[0][1]&0xc0) != (t[1]&0xc0))) {
print_buffer(buf2, fp_hla, hla);
buf2 = [], hla = {};
2014-09-20 11:23:54 +08:00
}
2014-10-03 06:04:36 +08:00
// skip unmapped lines
if (t[1]&4) {
2014-10-03 04:23:55 +08:00
buf2.push(t);
continue;
}
2014-09-23 22:33:15 +08:00
// parse the reported hit
2014-09-20 11:23:54 +08:00
var NM = (m = /\tNM:i:(\d+)/.exec(line)) == null? '0' : m[1];
var flag = t[1];
var h = parse_hit([t[2], ((flag&16)?'-':'+') + t[3], t[5], NM], opt);
if (t[2] == hla_chr) collect_hla_hits(idx_pri, h.ctg, h.start, h.end, hla);
2014-10-03 06:04:36 +08:00
if (h.hard) { // the following does not work with hard clipped alignments
2014-09-20 11:23:54 +08:00
buf2.push(t);
continue;
}
2014-10-03 06:04:36 +08:00
var hits = [h];
2014-09-23 22:33:15 +08:00
// parse hits in the XA tag
2014-10-03 06:04:36 +08:00
if ((m = /\tXA:Z:(\S+)/.exec(line)) != null) {
var XA_strs = m[1].split(";");
for (var i = 0; i < XA_strs.length; ++i)
if (XA_strs[i] != '') // as the last symbol in an XA tag is ";", the last split is an empty string
hits.push(parse_hit(XA_strs[i].split(","), opt));
}
// check if there are ALT hits
var has_alt = false;
for (var i = 0; i < hits.length; ++i)
if (is_alt[hits[i].ctg] != null) {
2014-10-03 06:04:36 +08:00
has_alt = true;
break;
}
2014-10-03 09:49:29 +08:00
if (!has_alt) {
buf2.push(t);
continue;
}
2014-09-20 11:23:54 +08:00
2014-09-23 22:33:15 +08:00
// lift mapping positions to the primary assembly
2014-10-03 05:11:37 +08:00
var n_rpt_lifted = 0, rpt_lifted = null;
2014-09-20 11:23:54 +08:00
for (var i = 0; i < hits.length; ++i) {
2014-10-03 05:11:37 +08:00
var a, h = hits[i];
2014-09-20 11:23:54 +08:00
2014-10-03 06:04:36 +08:00
if (idx_alt[h.ctg] == null || (a = idx_alt[h.ctg](h.start, h.end)) == null || a.length == 0)
2014-10-03 05:11:37 +08:00
continue;
2014-09-20 11:23:54 +08:00
// find the approximate position on the primary assembly
var lifted = [];
for (var j = 0; j < a.length; ++j) {
var s, e;
if (!a[j][4]) { // ALT is mapped to the forward strand of the primary assembly
s = cigar2pos(a[j][6], h.start);
e = cigar2pos(a[j][6], h.end - 1) + 1;
} else {
s = cigar2pos(a[j][6], a[j][2] - h.end);
e = cigar2pos(a[j][6], a[j][2] - h.start - 1) + 1;
}
if (s < 0 || e < 0) continue; // read is mapped to clippings in the ALT-to-chr alignment
s += a[j][5]; e += a[j][5];
lifted.push([a[j][3], (h.rev!=a[j][4]), s, e]);
if (i == 0) ++n_rpt_lifted;
}
2014-10-01 08:50:26 +08:00
if (i == 0 && n_rpt_lifted == 1) rpt_lifted = lifted[0].slice(0);
2014-10-03 05:11:37 +08:00
if (lifted.length) hits[i].lifted = lifted;
2014-09-20 11:23:54 +08:00
}
2014-10-03 06:04:36 +08:00
// prepare for hits grouping
2014-09-20 11:23:54 +08:00
for (var i = 0; i < hits.length; ++i) { // set keys for sorting
2014-10-03 06:04:36 +08:00
if (hits[i].lifted != null) // TODO: only the first element in lifted[] is used
2014-09-20 11:23:54 +08:00
hits[i].pctg = hits[i].lifted[0][0], hits[i].pstart = hits[i].lifted[0][2], hits[i].pend = hits[i].lifted[0][3];
else hits[i].pctg = hits[i].ctg, hits[i].pstart = hits[i].start, hits[i].pend = hits[i].end;
hits[i].i = i; // keep the original index
}
2014-10-03 06:04:36 +08:00
// group hits based on the lifted positions on non-ALT sequences
if (hits.length > 1) {
hits.sort(function(a,b) { return a.pctg != b.pctg? (a.pctg < b.pctg? -1 : 1) : a.pstart - b.pstart });
var last_chr = null, end = 0, g = -1;
for (var i = 0; i < hits.length; ++i) {
if (last_chr != hits[i].pctg) ++g, last_chr = hits[i].pctg, end = 0;
else if (hits[i].pstart >= end) ++g;
hits[i].g = g;
end = end > hits[i].pend? end : hits[i].pend;
}
} else hits[0].g = 0;
// find the index and group id of the reported hit; find the size of the reported group
var reported_g = null, reported_i = null, n_group0 = 0;
if (hits.length > 1) {
for (var i = 0; i < hits.length; ++i)
if (hits[i].i == 0)
reported_g = hits[i].g, reported_i = i;
for (var i = 0; i < hits.length; ++i)
if (hits[i].g == reported_g)
++n_group0;
} else {
if (is_alt[hits[0].ctg] == null) { // no need to go through the following if the single hit is non-ALT
buf2.push(t);
continue;
}
reported_g = reported_i = 0, n_group0 = 1;
}
2014-09-20 11:23:54 +08:00
2014-09-23 22:33:15 +08:00
// re-estimate mapping quality if necessary
2014-09-20 11:39:59 +08:00
var mapQ, ori_mapQ = t[4];
2014-09-23 22:33:15 +08:00
if (n_group0 > 1) {
2014-09-20 11:39:59 +08:00
var group_max = [];
for (var i = 0; i < hits.length; ++i) {
var g = hits[i].g;
if (group_max[g] == null || group_max[g][0] < hits[i].score)
group_max[g] = [hits[i].score, g];
}
if (group_max.length > 1)
group_max.sort(function(x,y) {return y[0]-x[0]});
if (group_max[0][1] == reported_g) { // the best hit is the hit reported in SAM
mapQ = group_max.length == 1? 60 : 6 * (group_max[0][0] - group_max[1][0]);
} else mapQ = 0;
mapQ = mapQ < 60? mapQ : 60;
2014-10-03 04:23:55 +08:00
if (idx_alt[t[2]] == null) mapQ = mapQ < ori_mapQ? mapQ : ori_mapQ;
else mapQ = mapQ > ori_mapQ? mapQ : ori_mapQ;
2014-09-20 11:39:59 +08:00
} else mapQ = t[4];
2014-09-20 11:23:54 +08:00
// find out whether the read is overlapping HLA genes
if (hits[reported_i].pctg == hla_chr) {
var rpt_start = 1<<30, rpt_end = 0;
for (var i = 0; i < hits.length; ++i) {
var h = hits[i];
if (h.g == reported_g) {
rpt_start = rpt_start < h.pstart? rpt_start : h.pstart;
rpt_end = rpt_end > h.pend ? rpt_end : h.pend;
}
}
collect_hla_hits(idx_pri, hla_chr, rpt_start, rpt_end, hla);
}
// adjust the mapQ of the primary hits
if (n_rpt_lifted <= 1) {
2014-10-21 12:16:41 +08:00
var l = n_rpt_lifted == 1? rpt_lifted : null;
2014-09-20 11:23:54 +08:00
for (var i = 0; i < buf2.length; ++i) {
2014-10-21 02:17:56 +08:00
var s = buf2[i], is_ovlp = true;
2014-10-21 12:16:41 +08:00
if (l != null) {
if (l[0] != s[2]) is_ovlp = false; // different chr
else if (((s[1]&16) != 0) != l[1]) is_ovlp = false; // different strand
else {
var start = s[3] - 1, end = start;
while ((m = re_cigar.exec(t[5])) != null)
if (m[2] == 'M' || m[2] == 'D' || m[2] == 'N')
end += parseInt(m[1]);
if (!(start < l[3] && l[2] < end)) is_ovlp = false; // no overlap
}
2014-10-21 12:16:41 +08:00
} else is_ovlp = false;
// get the "pa" tag if present
2014-10-20 08:53:47 +08:00
var om = -1, pa = 10.;
for (var j = 11; j < s.length; ++j)
if ((m = /^om:i:(\d+)/.exec(s[j])) != null)
om = parseInt(m[1]);
2014-10-20 08:53:47 +08:00
else if ((m = /^pa:f:(\S+)/.exec(s[j])) != null)
pa = parseFloat(m[1]);
2014-10-21 02:17:56 +08:00
if (is_ovlp) { // overlapping the lifted hit
2014-09-20 11:23:54 +08:00
if (om > 0) s[4] = om;
s[4] = s[4] < mapQ? s[4] : mapQ;
2014-10-20 08:53:47 +08:00
} else if (pa < opt.min_pa_ratio) { // not overlapping; has a small pa
if (om < 0) s.push("om:i:" + s[4]);
s[4] = 0;
2014-09-20 11:23:54 +08:00
}
}
}
// generate lifted_str
for (var i = 0; i < hits.length; ++i) {
if (hits[i].lifted && hits[i].lifted.length) {
var u = '', lifted = hits[i].lifted;
for (var j = 0; j < lifted.length; ++j)
u += lifted[j][0] + "," + lifted[j][2] + "," + lifted[j][3] + "," + (lifted[j][1]?'-':'+') + ";";
hits[i].lifted_str = u;
}
}
2014-09-23 22:33:15 +08:00
// stage the reported hit
2014-09-20 11:23:54 +08:00
t[4] = mapQ;
2014-09-20 11:39:59 +08:00
if (n_group0 > 1) t.push("om:i:"+ori_mapQ);
2014-09-20 11:23:54 +08:00
if (hits[reported_i].lifted_str) t.push("lt:Z:" + hits[reported_i].lifted_str);
buf2.push(t);
2014-09-23 22:33:15 +08:00
// stage the hits generated from the XA tag
var cnt = 0, rs = null, rq = null; // rq: reverse quality; rs: reverse complement sequence
var rg = (m = /\t(RG:Z:\S+)/.exec(line)) != null? m[1] : null;
2014-09-20 11:23:54 +08:00
for (var i = 0; i < hits.length; ++i) {
if (hits[i].g != reported_g || i == reported_i) continue;
if (idx_alt[hits[i].ctg] == null) continue;
var s = [t[0], 0, hits[i].ctg, hits[i].start+1, mapQ, hits[i].cigar, t[6], t[7], t[8]];
2014-11-10 23:16:53 +08:00
if (t[6] == '=' && s[2] != t[2]) s[6] = t[2];
2014-10-01 08:50:26 +08:00
// print sequence/quality and set the rev flag
if (hits[i].rev == hits[reported_i].rev) {
s.push(t[9], t[10]);
2014-10-15 04:57:57 +08:00
s[1] = flag | 0x800;
} else { // we need to write the reverse sequence
if (rs == null || rq == null) {
aux.length = 0;
aux.set(t[9], 0); aux.revcomp(); rs = aux.toString();
aux.set(t[10],0); aux.reverse(); rq = aux.toString();
}
2014-10-01 08:50:26 +08:00
s.push(rs, rq);
2014-10-15 04:57:57 +08:00
s[1] = (flag ^ 0x10) | 0x800;
2014-10-01 08:50:26 +08:00
}
2014-09-20 11:23:54 +08:00
s.push("NM:i:" + hits[i].NM);
if (hits[i].lifted_str) s.push("lt:Z:" + hits[i].lifted_str);
if (rg != null) s.push(rg);
2014-09-20 11:23:54 +08:00
buf2.push(s);
}
}
print_buffer(buf2, fp_hla, hla);
2014-09-20 11:23:54 +08:00
file.close();
if (fp_hla != null)
for (var h in fp_hla)
fp_hla[h].close();
2014-09-20 11:23:54 +08:00
buf.destroy();
2014-10-03 09:49:29 +08:00
aux.destroy();
2014-09-20 11:23:54 +08:00
}
bwa_postalt(arguments);