-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathpygrep.py
115 lines (97 loc) · 3.75 KB
/
pygrep.py
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
from itertools import zip_longest
from joblib import Parallel, delayed
from typing import Tuple
from tqdm import tqdm
import numpy as np
import shelve
import gzip
import sys
import os
class FastQ:
dictionary = None
tipo = None
prefix = None
def __init__(self, filename: str, cpus: int = -1, nodump: bool = True
) -> None:
self.filename = filename
self.cpus = cpus
self.nodump = nodump
self.parser()
@property
def filename(self):
return self.__filename
@filename.setter
def filename(self, filename):
if 'gz' in filename:
self.open = gzip.open
self.tipo = 'gz'
self.prefix = filename[:filename.find('.fastq.gz')]
else:
self.open = open
self.prefix = filename[:filename.rfind('.')]
self.__filename = filename
def yielder(self):
with self.open(self.filename, 'r') as infile:
args = [iter(infile)] * 4
for entry in zip_longest(*args, fillvalue=''):
if self.tipo is None:
yield entry
else:
yield entry.decode('utf-8')
@staticmethod
def to_dict(seq: str) -> Tuple[str, str]:
name = seq[0].strip().split()[0][1:]
return name, ''.join(seq)
def parser(self):
if not os.path.isfile('%s.shelve' % self.prefix):
d = Parallel(n_jobs=self.cpus)(delayed(self.to_dict)(seq)
for seq in tqdm(self.yielder(),
desc="Parsing file")
)
self.dictionary = dict(d)
if not self.nodump:
eprint('Dumping sequences in shelve')
with shelve.open('%s.shelve' % self.prefix) as dic:
dic.update(d)
eprint("Done!!")
else:
eprint("Shelve exists... using it")
with shelve.open('%s.shelve' % self.prefix) as dic:
self.dictionary = dict(dic)
def parse_pattern(pattern):
if os.path.isfile(pattern):
with open(pattern) as p:
return p.read().strip().split('\n')
else:
return [pattern]
def print_it(key, dictionary):
print(dictionary[key])
def eprint(*args, **kwargs):
print(*args, file=sys.stderr, **kwargs)
def main(filename: str, cpus: int, pattern: str, inverse: bool, nodump: bool):
fastq = FastQ(filename=filename, cpus=cpus, nodump=nodump)
seqs = fastq.dictionary
pattern = parse_pattern(pattern)
if inverse:
eprint('Subsetting inverse match')
subset = np.setdiff1d(np.array(list(seqs.keys())), np.array(pattern))
else:
eprint('Subsetting direct match')
subset = np.intersect1d(np.array(list(seqs.keys())), np.array(pattern))
for key in tqdm(subset, total=len(subset), desc='Subsetting'):
print(seqs[key])
if __name__ == '__main__':
import argparse
parser = argparse.ArgumentParser()
parser.add_argument('fastq', help="FastQ file with sequences")
parser.add_argument('-c', '--cpus', help="Number of cpus to use", type=int,
default=-1)
parser.add_argument('-v', '--inverse_match', action='store_true',
help="Return all matches not found in pattern")
parser.add_argument('-p', '--pattern', action='store', required=True,
help="Pattern string or file to find")
parser.add_argument('-d', '--nodump', action='store_false',
help="Create persisting object (shelve)")
args = parser.parse_args()
main(filename=args.fastq, cpus=args.cpus, inverse=args.inverse_match,
pattern=args.pattern, nodump=args.nodump)