-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathnetmhccons_1_1_python_interface.3.1.1.py
109 lines (92 loc) · 4.23 KB
/
netmhccons_1_1_python_interface.3.1.1.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
import os
from subprocess import Popen, PIPE
import tempfile
from allele_info.allele_info import is_user_defined_allele, MHCIAlleleData
from iedbtools_utilities.sequence_io import SequenceOutput
from iedbtools_utilities.sequence_manipulation import split_sequence
from logging import getLogger
logger = getLogger(__name__)
EXECUTABLE_NAME = 'netMHCcons'
EXECUTABLE_DIR_PATH = os.path.normpath(os.path.join(os.path.abspath(__file__), os.path.pardir))
EXECUTABLE_FULL_PATH = os.path.join(EXECUTABLE_DIR_PATH, EXECUTABLE_NAME)
def predict_sequence(sequence, allele_length_pair):
'''Given one protein sequence, break it up into peptides, return their predicted binding scores.'''
allele, binding_length = allele_length_pair
peptide_list = split_sequence(sequence, binding_length)
peptides_list = [peptide_list[i:i+500] for i in range(len(peptide_list))[::500]]
scores = ()
for peptide_list in peptides_list:
scores += predict_peptide_list(peptide_list, allele_length_pair)
return scores
def parse_pickpocket_output(content):
'''Given pickpocket output, returns a list of scores'''
scores = []
for lines in content.split('\n'):
if 'PEPLIST' in lines:
data_list = lines.split()
if data_list[0].isdigit():
scores.append(float(data_list[5]))
return scores
def predict_peptide_list(peptide_list, allele_length_pair):
'''This routine can be directly called so that you do not make a file for each prediction.'''
infile = tempfile.NamedTemporaryFile(prefix='netmhccons_', suffix='_input', delete=False, mode='w')
for peptide in peptide_list:
infile.write(peptide + "\n")
infile.close()
allele, binding_length = allele_length_pair
# Temporary fix.
stripped_allele_name = strip_allele_name(allele)
user_defined_allele = is_user_defined_allele(allele)
if user_defined_allele:
# is the user_defined_allele a seuqnce list?
fasta_allele = SequenceOutput().to_fasta(allele)
usermhcfile = tempfile.NamedTemporaryFile(
prefix='netmhccons_', suffix='_usermhc', delete=False, mode='w')
usermhcfile.write(fasta_allele)
usermhcfile.close()
if user_defined_allele:
cmd = [
EXECUTABLE_FULL_PATH, '-hlaseq', usermhcfile.name, '-length', str(binding_length),
'-inptype', '1', '-f', infile.name
]
else:
cmd = [
EXECUTABLE_FULL_PATH, '-a', stripped_allele_name, '-length', str(binding_length),
'-inptype', '1', '-f', infile.name
]
logger.info('Calling netmhccons executable:\n%s', ' '.join(cmd))
p = Popen(cmd, stdout=PIPE, stderr=PIPE)
#process_status_code = p.wait()
analysis_results, ignored_stderr = p.communicate()
process_status_code = p.returncode
# process_status_code = f.close()
if process_status_code != 0:
msg = 'Error calling netmhccons executable:\n{}'.format(' '.join(cmd))
logger.error(msg)
raise Exception(msg)
analysis_results = analysis_results.decode()
scores = parse_pickpocket_output(analysis_results)
os.remove(infile.name)
if user_defined_allele:
os.remove(usermhcfile.name)
if (len(peptide_list) != len(scores)):
msg = "len(peptide_list) != len(scores) -- {} != {}".format(len(peptide_list), len(scores))
logger.error('%s\n%s', msg, analysis_results)
raise Exception(msg)
if cmd == None: # According to python doc, return status of '0' is actually 'None' using os.popen.
msg = "netMHCconsPredictor did not execute. Command used:\n{}".format(cmd)
raise Exception(msg)
return tuple(scores)
def strip_allele_name(allele_name):
""" | *brief*: Temporary hack to get the allele name right for netmhccons executable.
| *author*: Dorjee
| *created*: 2016-09-13
TODO: A more permanent solution would be to create a column in the database for canonical allele name.
"""
miad = MHCIAlleleData()
species = miad.get_species_for_allele_name(allele_name=allele_name)
if species in ['macaque', 'pig']:
stripped_allele_name = allele_name.replace('*',':')
else:
stripped_allele_name = allele_name.replace('*', '')
return stripped_allele_name