001/*
002 *                    BioJava development code
003 *
004 * This code may be freely distributed and modified under the
005 * terms of the GNU Lesser General Public Licence.  This should
006 * be distributed with the code.  If you do not have a copy,
007 * see:
008 *
009 *      http://www.gnu.org/copyleft/lesser.html
010 *
011 * Copyright for this code is held jointly by the individual
012 * authors.  These should be listed in @author doc comments.
013 *
014 * For more information on the BioJava project and its aims,
015 * or to join the biojava-l mailing list, visit the home page
016 * at:
017 *
018 *      http://www.biojava.org/
019 *
020 */
021package org.biojava.nbio.genome.parsers.twobit;
022
023
024import org.slf4j.Logger;
025import org.slf4j.LoggerFactory;
026
027import java.io.File;
028import java.io.IOException;
029import java.io.InputStream;
030import java.io.RandomAccessFile;
031import java.util.HashMap;
032
033/**
034 * downloaded from http://storage.bioinf.fbb.msu.ru/~roman/TwoBitParser.java
035 *
036 * Class is a parser of UCSC Genome Browser file format .2bit used to store
037 * nucleotide sequence information. This class extends InputStream and can
038 * be used as it after choosing one of names of containing sequences. This
039 * parser can be used to do some work like UCSC tool named twoBitToFa. For
040 * it just run this class with input file path as single parameter and set
041 * stdout stream into output file. If you have any problems or ideas don't
042 * hesitate to contact me through email: rsutormin[at]gmail.com.
043 * @author Roman Sutormin
044 */
045public class TwoBitParser extends InputStream {
046
047        private static final Logger logger = LoggerFactory.getLogger(TwoBitParser.class);
048
049        public int DEFAULT_BUFFER_SIZE = 10000;
050        //
051        private RandomAccessFile raf;
052        private File f;
053        private boolean reverse = false;
054        private String[] seq_names;
055        private HashMap<String,Long> seq2pos = new HashMap<>();
056        private String cur_seq_name;
057        private long[][] cur_nn_blocks;
058        private long[][] cur_mask_blocks;
059        private long cur_seq_pos;
060        private long cur_dna_size;
061        private int cur_nn_block_num;
062        private int cur_mask_block_num;
063        private int[] cur_bits;
064        private byte[] buffer;
065        private long buffer_size;
066        private long buffer_pos;
067        private long start_file_pos;
068        private long file_pos;
069        //
070        private static final char[] bit_chars = {
071                        'T','C','A','G'
072        };
073
074        public TwoBitParser(File f) throws Exception {
075                this.f = f;
076                raf = new RandomAccessFile(f,"r");
077                long sign = readFourBytes();
078                if(sign==0x1A412743) {
079                        logger.debug("2bit: Normal number architecture");
080                }
081                else if(sign==0x4327411A) {
082                        reverse = true;
083                        logger.debug("2bit: Reverse number architecture");
084                }
085                else throw new Exception("Wrong start signature in 2BIT format");
086                readFourBytes();
087                int seq_qnt = (int)readFourBytes();
088                readFourBytes();
089                seq_names = new String[seq_qnt];
090                for(int i=0;i<seq_qnt;i++) {
091                        int name_len = raf.read();
092                        char[] chars = new char[name_len];
093                        for(int j=0;j<name_len;j++) chars[j] = (char)raf.read();
094                        seq_names[i] = String.valueOf(chars);
095                        long pos = readFourBytes();
096                        seq2pos.put(seq_names[i],pos);
097                        logger.debug("2bit: Sequence name=[{}], pos={}", seq_names[i], pos);
098                }
099        }
100        private long readFourBytes() throws Exception {
101                long ret = 0;
102                if(!reverse) {
103                        ret = raf.read();
104                        ret += raf.read()*0x100;
105                        ret += raf.read()*0x10000;
106                        ret += raf.read()*0x1000000;
107                }
108                else {
109                        ret = raf.read()*0x1000000;
110                        ret += raf.read()*0x10000;
111                        ret += raf.read()*0x100;
112                        ret += raf.read();
113                }
114                return ret;
115        }
116        public String[] getSequenceNames() {
117                String[] ret = new String[seq_names.length];
118                System.arraycopy(seq_names,0,ret,0,seq_names.length);
119                return ret;
120        }
121        /**
122         * Method open nucleotide stream for sequence with given name.
123         * @param seq_name name of sequence (one of returned by getSequenceNames()).
124         * @throws Exception
125         */
126        public void setCurrentSequence(String seq_name) throws Exception {
127                if(cur_seq_name!=null) {
128                        throw new Exception("Sequence ["+cur_seq_name+"] was not closed");
129                }
130                if(seq2pos.get(seq_name)==null) {
131                        throw new Exception("Sequence ["+seq_name+"] was not found in 2bit file");
132                }
133                cur_seq_name = seq_name;
134                long pos = seq2pos.get(seq_name);
135                raf.seek(pos);
136                long dna_size = readFourBytes();
137                logger.debug("2bit: Sequence name=[{}], dna_size={}", cur_seq_name, dna_size);
138                cur_dna_size = dna_size;
139                int nn_block_qnt = (int)readFourBytes();
140                cur_nn_blocks = new long[nn_block_qnt][2];
141                for(int i=0;i<nn_block_qnt;i++) {
142                        cur_nn_blocks[i][0] = readFourBytes();
143                }
144                for(int i=0;i<nn_block_qnt;i++) {
145                        cur_nn_blocks[i][1] = readFourBytes();
146                }
147
148                for(int i=0;i<nn_block_qnt;i++) {
149                        logger.debug("NN-block: [{},{}] ", cur_nn_blocks[i][0], cur_nn_blocks[i][1]);
150                }
151
152                int mask_block_qnt = (int)readFourBytes();
153                cur_mask_blocks = new long[mask_block_qnt][2];
154                for(int i=0;i<mask_block_qnt;i++) {
155                        cur_mask_blocks[i][0] = readFourBytes();
156                }
157                for(int i=0;i<mask_block_qnt;i++) {
158                        cur_mask_blocks[i][1] = readFourBytes();
159                }
160
161                for(int i=0;i<mask_block_qnt;i++) {
162                        logger.debug("[{},{}] ", cur_mask_blocks[i][0], cur_mask_blocks[i][1]);
163                }
164
165                readFourBytes();
166                start_file_pos = raf.getFilePointer();
167                reset();
168        }
169        /**
170         * Method resets current position to the begining of sequence stream.
171         */
172        @Override
173        public synchronized void reset() throws IOException {
174                cur_seq_pos = 0;
175                cur_nn_block_num = (cur_nn_blocks.length>0)?0:-1;
176                cur_mask_block_num = (cur_mask_blocks.length>0)?0:-1;
177                cur_bits = new int[4];
178                file_pos = start_file_pos;
179                buffer_size = 0;
180                buffer_pos = -1;
181        }
182        /**
183         * @return number (starting from 0) of next readable nucleotide in sequence stream.
184         */
185        public long getCurrentSequencePosition() {
186                if(cur_seq_name==null) throw new RuntimeException("Sequence is not set");
187                return cur_seq_pos;
188        }
189        public void setCurrentSequencePosition(long pos) throws IOException {
190                if(cur_seq_name==null) throw new RuntimeException("Sequence is not set");
191                if(pos>cur_dna_size) throw new RuntimeException(
192                                "Postion is too high (more than "+cur_dna_size+")");
193                if(cur_seq_pos>pos) {
194                        reset();
195                }
196                skip(pos-cur_seq_pos);
197        }
198        private void loadBits() throws IOException {
199                if((buffer==null)||(buffer_pos<0)||(file_pos<buffer_pos)||
200                                (file_pos>=buffer_pos+buffer_size)) {
201                        if((buffer==null)||(buffer.length!=DEFAULT_BUFFER_SIZE)) {
202                                buffer = new byte[DEFAULT_BUFFER_SIZE];
203                        }
204                        buffer_pos = file_pos;
205                        buffer_size = raf.read(buffer);
206                }
207                int cur_byte = buffer[(int)(file_pos-buffer_pos)]& 0xff;
208                for(int i=0;i<4;i++) {
209                        cur_bits[3-i] = cur_byte%4;
210                        cur_byte /= 4;
211                }
212        }
213        /**
214         * Method reads 1 nucleotide from sequence stream. You should set current sequence
215         * before use it.
216         */
217        @Override
218        public int read() throws IOException {
219                if(cur_seq_name==null) throw new IOException("Sequence is not set");
220                if(cur_seq_pos==cur_dna_size) {
221                        logger.debug("End of sequence (file position:{})", raf.getFilePointer());
222                        return -1;
223                }
224                int bit_num = (int)cur_seq_pos%4;
225                if(bit_num==0) {
226                        loadBits();
227                }
228                else if(bit_num==3) {
229                        file_pos++;
230                }
231                char ret = 'N';
232                if((cur_nn_block_num>=0)&&
233                                (cur_nn_blocks[cur_nn_block_num][0]<=cur_seq_pos)) {
234                        if(cur_bits[bit_num]!=0) {
235                                throw new IOException("Wrong data in NN-block ("+cur_bits[bit_num]+") "+
236                                                "at position "+cur_seq_pos);
237                        }
238                        if(cur_nn_blocks[cur_nn_block_num][0]+cur_nn_blocks[cur_nn_block_num][1]==cur_seq_pos+1) {
239                                cur_nn_block_num++;
240                                if(cur_nn_block_num>=cur_nn_blocks.length) {
241                                        cur_nn_block_num = -1;
242                                }
243                        }
244                        ret = 'N';
245                }
246                else {
247                        ret = bit_chars[cur_bits[bit_num]];
248                }
249                if((cur_mask_block_num>=0)&&
250                                (cur_mask_blocks[cur_mask_block_num][0]<=cur_seq_pos)) {
251                        ret = Character.toLowerCase(ret);
252                        if(cur_mask_blocks[cur_mask_block_num][0]+cur_mask_blocks[cur_mask_block_num][1]==cur_seq_pos+1) {
253                                cur_mask_block_num++;
254                                if(cur_mask_block_num>=cur_mask_blocks.length) {
255                                        cur_mask_block_num = -1;
256                                }
257                        }
258                }
259                cur_seq_pos++;
260                return ret;
261        }
262        /**
263         * Method skips n nucleotides in sequence stream. You should set current sequence
264         * before use it.
265         */
266        @Override
267        public synchronized long skip(long n) throws IOException {
268                if(cur_seq_name==null) throw new IOException("Sequence is not set");
269                if(n<4) {
270                        int ret = 0;
271                        while((ret<n)&&(read()>=0)) ret++;
272                        return ret;
273                }
274                if(n>cur_dna_size-cur_seq_pos) {
275                        n = cur_dna_size-cur_seq_pos;
276                }
277                cur_seq_pos += n;
278                file_pos = start_file_pos+(cur_seq_pos/4);
279                raf.seek(file_pos);
280                if((cur_seq_pos%4)!=0) {
281                        loadBits();
282                }
283                while((cur_nn_block_num>=0)&&
284                                (cur_nn_blocks[cur_nn_block_num][0]+cur_nn_blocks[cur_nn_block_num][1]<=cur_seq_pos)) {
285                        cur_nn_block_num++;
286                        if(cur_nn_block_num>=cur_nn_blocks.length) cur_nn_block_num = -1;
287                }
288                while((cur_mask_block_num>=0)&&
289                                (cur_mask_blocks[cur_mask_block_num][0]+cur_mask_blocks[cur_mask_block_num][1]<=cur_seq_pos)) {
290                        cur_mask_block_num++;
291                        if(cur_mask_block_num>=cur_mask_blocks.length) cur_mask_block_num = -1;
292                }
293                return n;
294        }
295        /**
296         * Method closes current sequence and it's necessary to invoke it before setting
297         * new current sequence.
298         */
299        @Override
300        public void close() throws IOException {
301                cur_seq_name = null;
302                cur_nn_blocks = null;
303                cur_mask_blocks = null;
304                cur_seq_pos = -1;
305                cur_dna_size = -1;
306                cur_nn_block_num = -1;
307                cur_mask_block_num = -1;
308                cur_bits = null;
309                buffer_size = 0;
310                buffer_pos = -1;
311                file_pos = -1;
312                start_file_pos = -1;
313        }
314        @Override
315        public int available() throws IOException {
316                if(cur_seq_name==null) throw new IOException("Sequence is not set");
317                return (int)(cur_dna_size-cur_seq_pos);
318        }
319        /**
320         * Method closes random access file descriptor. You can't use any reading methods
321         * after it.
322         * @throws Exception
323         */
324        public void closeParser() throws Exception {
325                raf.close();
326        }
327        public File getFile() {
328                return f;
329        }
330        public String loadFragment(long seq_pos,int len) throws IOException {
331                if(cur_seq_name==null) throw new IOException("Sequence is not set");
332                setCurrentSequencePosition(seq_pos);
333                char[] ret = new char[len];
334                int i = 0;
335                for(;i<len;i++) {
336                        int ch = read();
337                        if(ch<0) break;
338                        ret[i] = (char)ch;
339                }
340                return new String(ret,0,i);
341        }
342        public void printFastaSequence() throws IOException {
343                if(cur_seq_name==null) throw new RuntimeException("Sequence is not set");
344                printFastaSequence(cur_dna_size-cur_seq_pos);
345        }
346        public void printFastaSequence(long len) throws IOException {
347                if(cur_seq_name==null) throw new RuntimeException("Sequence is not set");
348                logger.info(">{} pos={}, len={}", cur_seq_name, cur_seq_pos, len);
349                char[] line = new char[60];
350                boolean end = false;
351                long qnt_all = 0;
352                while(!end) {
353                        int qnt = 0;
354                        for(;(qnt<line.length)&&(qnt_all<len);qnt++,qnt_all++) {
355                                int ch = read();
356                                if(ch<0) {
357                                        end = true;
358                                        break;
359                                }
360                                line[qnt] = (char)ch;
361                        }
362                        if(qnt>0) {
363                                logger.info(new String(line,0,qnt));
364                        }
365                        if(qnt_all>=len) end = true;
366                }
367        }
368        public static void main(String[] args) throws Exception {
369                if(args.length==0) {
370                        logger.info("Usage: <program> <input.2bit> [<seq_name> [<start> [<length>]]]");
371                        logger.info("Resulting fasta data will be written in stdout.");
372                        return;
373                }
374                try (TwoBitParser p = new TwoBitParser(new File(args[0]))) {
375                        if (args.length == 1) {
376                                String[] names = p.getSequenceNames();
377                                for (int i = 0; i < names.length; i++) {
378                                        p.setCurrentSequence(names[i]);
379                                        p.printFastaSequence();
380                                        p.close();
381                                }
382                        } else {
383                                String name = args[1];
384                                p.setCurrentSequence(name);
385                                if (args.length > 2) {
386                                        long start = Long.parseLong(args[2]);
387                                        p.skip(start);
388                                }
389                                if (args.length > 3) {
390                                        long len = Long.parseLong(args[3]);
391                                        p.printFastaSequence(len);
392                                } else {
393                                        p.printFastaSequence();
394                                }
395                                p.close();
396                        }
397                        p.closeParser();
398                }
399        }
400}
401