-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathpromod-tk
executable file
·167 lines (141 loc) · 6.11 KB
/
promod-tk
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
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
import tkinter as tk
import tkinter.filedialog
import tkinter.messagebox
import re
import os
from builder import errors
from builder import build_macrocomplex as bm
from Bio import SeqIO, PDB
def run_promod():
pass
def _parse_stoichiometry(input_file):
'''
Input file with one element in the fasta per line, stoichiometry indicated by commas (,)
'''
if(not input_file): return input_file
stoic = {}
for line in open(input_file):
print(line)
result = re.search('^(.+),([0-9]+)$', line.strip())
if (not result):
raise Exception('Non valid stoichimetry')
stoic[result.group(1)] = int(result.group(2))
return stoic
def select_directory(entry):
folder = tk.filedialog.askdirectory(parent=top, initialdir='.', title='Select folder')
entry.set(folder)
return None
def select_file(entry):
folder = tk.filedialog.askopenfilename(parent=top, initialdir='.', title='Select file')
entry.set(folder)
return None
def build(thres_label, dist_label, output_label, stoic_label,
fasta_label, folder_label, initial_label, button):
threshold = float(thres_label.get()) if thres_label.get() else 0.95
distance = float(dist_label.get()) if dist_label.get() else 1
output_file = output_label.get()
start = initial_label.get()
folder = folder_label.get()
stoic = _parse_stoichiometry(stoic_label.get())
fasta_file = fasta_label.get()
folder = folder_label.get()
if(not folder or not fasta_file or not output_file):
tkinter.messagebox.showerror('Missing elements', 'Please fill the input folder, the output folder and the fasta file')
return
sequences = list(SeqIO.parse(fasta_file, 'fasta'))
parser = PDB.PDBParser(QUIET=1)
structures = list()
for file in os.listdir(folder):
if file.endswith('.pdb'):
path = os.path.join(folder, file[:-4])
pdb = parser.get_structure(path, path+'.pdb')
structures.append(pdb)
try:
model = bm.build_complex(threshold=threshold,
stoichiometry=stoic,
sequences=list(sequences),
structures=structures,
distance=distance,
verbose=False,
initial=start)
except errors.PDB_disagrees_fasta as e:
tkinter.messagebox.showerror('PDB or Fasta error', 'Sequence in the PDB doesn\'t match with any in the fasta')
except ValueError:
tkinter.messagebox.showerror('PInput error', 'Input folder has less than 2 PDB files')
except errors.chain_in_stoic_not_in_fasta as e:
tkinter.messagebox.showerror('Stoichiometry error', 'Stoichiometry ask for an inexistent sequence in the fasta')
else:
tkinter.messagebox.showinfo('Done', 'Process complete. Model saved in the indicated folder')
print('Finished...')
print('Saving complex...')
io = PDB.PDBIO()
io.set_structure(model)
if(not os.path.exists(output_file)):
os.mkdir(output_file)
io.save(os.path.join(output_file, 'final_model.pdb'))
pass
top = tk.Tk()
top.title('Welcome to Promod FC')
#top.geometry('800x600')
## Row 1: Directory with pdbs
name = tk.Label(top, text='PDB directory')
folder_label = tk.StringVar()
log = tk.Entry(top, textvariable=folder_label)
run = tk.Button(top, text='PDB directory', command=lambda:select_directory(folder_label))
name.grid(column=0, row=0)
run.grid(column=2, row=0)
log.grid(column=1, row=0)
## Row 2: Fasta file
output = tk.Label(top, text='Output directory')
output_label = tk.StringVar()
output_entry = tk.Entry(top, textvariable=output_label)
select_output_button = tk.Button(top, text='Output directory', command=lambda:select_directory(output_label))
output.grid(column=0, row=1)
output_entry.grid(column=1, row=1)
select_output_button.grid(column=2, row=1)
## Row 2: Fasta file
pdbs = tk.Label(top, text='Fasta file')
fasta_label = tk.StringVar()
fasta_entry = tk.Entry(top, textvariable=fasta_label)
select_fasta_button = tk.Button(top, text='Fasta file', command=lambda:select_file(fasta_label))
pdbs.grid(column=0, row=2)
fasta_entry.grid(column=1, row=2)
select_fasta_button.grid(column=2, row=2)
## Row 3: Stoichiometry
stoic = tk.Label(top, text='Stoichiometry file*')
stoic_label = tk.StringVar()
stoic_entry = tk.Entry(top, textvariable=stoic_label)
select_stoic_button = tk.Button(top, text='Stoic. directory', command=lambda:select_file(stoic_label))
stoic.grid(column=0, row=3)
stoic_entry.grid(column=1, row=3)
select_stoic_button.grid(column=2, row=3)
## Row 4: Threshold
threshold = tk.Label(top, text='Threshold*')
thres_label = tk.StringVar()
thres_entry = tk.Entry(top, textvariable=thres_label)
#select_thres_button = tk.Button(top, text='Select directory', command=lambda:select_file(thres_label))
threshold.grid(column=0, row=4)
thres_entry.grid(column=1, row=4)
## Row 5: Distance
distance = tk.Label(top, text='Distance*')
dist_label = tk.StringVar()
dist_entry = tk.Entry(top, textvariable=dist_label)
distance.grid(column=0, row=5)
dist_entry.grid(column=1, row=5)
## Row 6: Initial structure
initial = tk.Label(top, text='Initial structure *')
initial_label = tk.StringVar()
initial_entry = tk.Entry(top, textvariable=initial_label)
select_initial_button = tk.Button(top, text='Select initial file', command=lambda:select_file(initial_label))
initial.grid(column=0, row=6)
initial_entry.grid(column=1, row=6)
select_initial_button.grid(column=2, row=6)
run_button = tk.Button(top, text='Build complex', command=lambda: build(thres_label, dist_label, output_label,
stoic_label, fasta_label,
folder_label, initial_label, run_button))
run_button.grid(column=1, row=7)
optional = tk.Label(top, text='Elementes marked with * are optional')
optional.grid(column=1, row=8)
top.mainloop()