-
Notifications
You must be signed in to change notification settings - Fork 5
/
predict.py
executable file
·531 lines (501 loc) · 23.3 KB
/
predict.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
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
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
#Authors : Maria-Anna Trapotsi, Layla Hosseini-Gerami and Lewis Mervin
#Contact: [email protected] and [email protected]
#Supervisor : Dr. A. Bender
#All rights reserved 2021
#Protein Target Prediction using SAR data from PubChem and ChEMBL_28
#Molecular Descriptors : 2048bit circular Binary Fingerprints (Rdkit) - ECFP_4
#Dependencies : rdkit, sklearn, standardiser
### predict.py ###
#Output a matrix of probabilities [computed as the mean predicted class probabilities of
#the trees in the forest (where the class probability of a single tree is the fraction of
#samples of the same class in a leaf)], or user-specified Random probability thresholds to
#produce binary predictions for an input list of smiles/sdfs. Predictions are generated
#for the [filtered] models using a reliability-density neighbourhood Applicability Domain
#(AD) analysis from: doi.org/10.1186/s13321-016-0182-y
#standard libraries
import bz2
import cPickle
import csv
import glob
import math
import multiprocessing
import os
import sys
import time
import zipfile
from multiprocessing import Pool
from optparse import OptionParser
from os import path
#third-party libraries
import numpy as np
import pandas as pd
from rdkit import Chem
from rdkit import DataStructs
from rdkit.Chem import AllChem
from rdkit.Chem import Descriptors
from scipy.stats import percentileofscore
# suppress warnings
import warnings
warnings.filterwarnings("ignore")
#optionparser options
parser = OptionParser()
parser.add_option('-f', dest='inf', help='Input smiles or sdf file (required)', metavar='FILE')
parser.add_option('-d', '--smiles_delim', default=' ', type=str, dest='delim', help='Input file (smiles) delimiter char (default: white space " ")')
parser.add_option('--smiles_column', default=0, type=int, dest='smicol', help='Input file (smiles) delimiter column (default: 0)')
parser.add_option('--smiles_id_column', default=1, type=int, dest='idcol', help='Input file (smiles) ID column (default: 1)')
parser.add_option('-o', dest='off', default=None, help='Optional output prediction file name', metavar='FILE')
parser.add_option('-t', '--transpose', action='store_true', default=False, dest='transpose', help='Transpose output (rows are compounds, columns are targets)')
parser.add_option('-n', '--ncores', default=1, type=int, dest='ncores', help='No. cores (default: 1)')
parser.add_option('-b', '--bioactivity', default=None, type=str, dest='bioactivity', help='Bioactivity threshold (can use multiple split by ",". E.g. "100,10"')
parser.add_option('-p', '--proba', default=None, type=float, dest='proba', help='RF probability threshold (default: None)')
parser.add_option('--ad', default='90', type=str, dest='ad', help='Applicability Domain (AD) filter using percentile of weights [float]. Default: 90 (integer for percentile)')
parser.add_option('--known_flag', action='store_true', default=False, dest='known', help='Set known activities (annotate duplicates betweem input to train with correct label)')
parser.add_option('--orthologues', action='store_true', default=False, dest='ortho', help='Set to use orthologue bioactivity data in model generation')
parser.add_option('--organism', dest='organism', default=None, type=str, help='Organism filter (multiple can be specified using commas ",")')
parser.add_option('--target_class', dest='targetclass', default=None, type=str, help='Target classification filter')
parser.add_option('--min_size', dest='minsize', default=10, type=int, help='Minimum number of actives used in model generation (default: 10)')
parser.add_option('--performance_filter', default=None, type=str, dest='p_filt', help='Comma-seperated performance filtering using following nomenclature: validation_set[tsscv,l50so,l50po],metric[bedroc,roc,prauc,brier],performance_threshold[float]. E.g "tsscv,bedroc,0.5"')
parser.add_option('--se_filter', action='store_true', default=False, dest='se_filt', help='Optional setting to restrict to models which do not require Sphere Exclusion (SE)')
parser.add_option('--training_log', action='store_true', default=False, dest='train_log', help='Optional setting to add training_details to the prediction file (large increase in output file size)')
parser.add_option('--ntrees', dest='ntrees', default=None, type=int, help='Specify the minimum number of trees for warm-start random forest models (N.B Potential large latency/memory cost)')
parser.add_option('--preprocess_off', dest='preproc', action='store_false', default=True, help='Turn off preprocessing using the flatkinson (eTox) standardizer (github.com/flatkinson/standardiser), size filter (100 >= Mw >= 1000 and organic mol check (C count >= 1)')
parser.add_option('--std_dev', dest='std', action='store_true', default=False, help='Turn on matrix calculation for the standard deviation of prediction across the trees in the forest')
parser.add_option('--percentile', dest='percentile', action='store_true', default=False, help='Turn on matrix calculation for the percentile of AD compounds')
parser.add_option('--model_dir', dest='model_dir', default=None, type=str, help='Path to directory containing models, if not default')
parser.add_option('--ad_smiles', action='store_true', default=False, dest='ad_smiles', help='Set to output SMILES of highest-weighted AD compound in percentile mode')
(options, args) = parser.parse_args()
#if tab delimited then set to \t
if options.delim == 'tab': options.delim = '\t'
def introMessage():
print '=============================================================================================='
print ' Authors: Maria-Anna Trapotsi, Layla Hosseini-Gerami and Lewis Mervin\n Email: [email protected], [email protected]\n Supervisor: Dr. A. Bender'
print ' Address: Centre For Molecular Informatics, Dept. Chemistry, Lensfield Road, Cambridge CB2 1EW'
print '==============================================================================================\n'
return
#check smiles of sdf (if not quit)
def check_Input():
global options
extension = options.inf.split('.')[-1]
if extension not in ['smi','smiles','sdf']:
print ' Warning [-f]: File type not "smi", "smiles" or "sdf". Interpreting input as SMILES'
return extension
#check input & set OS directory pointers
def check_set_working_env():
global options
if os.name == 'nt': sep = '\\'
else: sep = '/'
input_extension = check_Input()
try:
ad_settings = int(options.ad)
if 0 >= ad_settings >= 100: raise ValueError
except ValueError:
print ' Input Error [--ad]: Percentile weight not integer between 0-100%. Please Check parameters'
sys.exit()
if options.model_dir:
pidgin_dir = options.model_dir
else:
pidgin_dir = os.path.dirname(os.path.abspath(__file__))
if options.ortho:
mod_dir = pidgin_dir + sep + 'ortho' + sep
if not os.path.isdir(mod_dir):
print ' Orthologue Error [--orthologues]: Orthologues directory is not present. Please download from here: '
else: mod_dir = pidgin_dir + sep + 'no_ortho' + sep
return input_extension, sep, pidgin_dir, mod_dir, ad_settings
#filter models using user specification supplied via option parser
def get_Models():
global mod_dir, sep, options
target_count = 0
model_info = [l.split('\t')[1:] for l in open(mod_dir + 'training_log.txt').read().splitlines()]
if options.ntrees: model_info = [m[:2]+[str(options.ntrees)]+m[3:] if idx > 0 else m \
for idx, m in enumerate(model_info)]
model_info = {l[0] : l for l in model_info}
uniprot_info = [i.split('\t') for i in open(mod_dir + 'uniprot_information.txt').read().splitlines()[1:]]
mid_uniprots = dict()
if options.p_filt:
val_dict = {'tsscv':7, 'l50po':12, 'l50so':17}
metric_dict = dict(zip(['bedroc','roc','prauc','brier'],range(4)))
try:
(validation,metric,perf_threshold) = [pf.lstrip() for pf in options.p_filt.split(',')]
train_row_idx = val_dict[validation] + metric_dict[metric]
perf_threshold = float(perf_threshold)
print ' Filtering models for a minimum ' + metric + ' performance of ' \
+ str(perf_threshold) + ' during ' + validation + ' validaiton'
except (KeyError,ValueError):
print ' Input Error [--performance_filter]: Use format ' \
'validation_set[tsscv,l50so,l50po],metric[bedroc,roc,prauc,brier],' \
'threshold[float].\n E.g "bedroc,tsscv,0.5"\n...exiting'
sys.exit()
if options.organism: orgs = map(str.lower, options.organism.split(','))
for row in uniprot_info:
#filter bioactivity/organism/targetclass/minsize/se/performance (if set)
if options.bioactivity and row[8] not in options.bioactivity.split(','): continue
if options.organism and (row[4] == '' \
or not any([org.lstrip() in row[4].lower() for org in orgs])): continue
if options.targetclass and row[3] not in options.targetclass: continue
if sum(map(int,row[9:11])) < options.minsize: continue
if options.se_filt and int(row[13]) > 0: continue
if options.p_filt:
try:
if float(model_info[row[-1]][train_row_idx].split(',')[0]) < perf_threshold: continue
except (KeyError, ValueError): continue
#if passes all filters then add to mid->uniprot dictionary
try: mid_uniprots[row[-1]].append(row)
except KeyError: mid_uniprots[row[-1]] = [row]
target_count +=1
if len(mid_uniprots) == 0:
print ' Warning: No eligable models using current filters...exiting'
sys.exit()
return mid_uniprots, model_info, target_count
#preprocess rdkit molecule
def preprocessMolecule(inp):
def checkC(mm):
mwt = Descriptors.MolWt(mm)
for atom in mm.GetAtoms():
if atom.GetAtomicNum() == 6 and 100 <= mwt <= 1000: return True
return False
def checkHm(mm):
for atom in mm.GetAtoms():
if atom.GetAtomicNum() in [2,10,13,18]: return False
if 21 <= atom.GetAtomicNum() <= 32: return False
if 36 <= atom.GetAtomicNum() <= 52: return False
if atom.GetAtomicNum() >= 54: return False
return True
try: std_mol = standardise.run(inp)
except standardise.StandardiseException: return None
if not std_mol or checkHm(std_mol) == False or checkC(std_mol) == False: return None
else: return std_mol
#preprocess exception to catch
class MolFromSmilesError(Exception):
'raise due to "None" from Chem.MolFromSmiles'
#preprocess exception to catch
class PreprocessViolation(Exception):
'raise due to preprocess violation'
#calculate 2048bit morgan fingerprints, radius 2, for smiles or sdf input
def calcFingerprints(input,qtype='smiles'):
if qtype == 'smiles': m = Chem.MolFromSmiles(input)
else: m = input
if not m: raise MolFromSmilesError(' None mol in function')
if options.preproc:
m = preprocessMolecule(m)
if not m: raise PreprocessViolation(' Molecule preprocessing violation')
fp = AllChem.GetMorganFingerprintAsBitVect(m,2, nBits=2048)
binary = fp.ToBitString()
if qtype == 'sdf': return Chem.MolToSmiles(m), map(int,list(binary)), fp
else: return map(int,list(binary)), fp
#calculate fingerprints for chunked array of smiles
def arrayFP(inp):
idx, i = inp
i = i.split(options.delim)
try:
fp, mol = calcFingerprints(i[options.smicol])
outfp = fp
outmol = mol
try: outsmi_id = i[options.idcol]
except IndexError:
outsmi_id = i[options.smicol]
except PreprocessViolation:
print ' SMILES preprocessing violation (line ' + str(idx+1) + ' will be removed): ' + i[options.smicol]
outfp = None
outmol = None
outsmi_id = None
except MolFromSmilesError:
print ' SMILES parse error (line ' + str(idx+1) + '): ' + i[options.smicol]
outfp = None
outmol = None
outsmi_id = None
return outfp, outmol, outsmi_id
#import user smiles query
def importQuerySmiles(in_file):
query = open(in_file).read().splitlines()
query = zip(range(len(query)), query)
chunksize = max(1, int(len(query) / (options.ncores)))
pool = Pool(processes=options.ncores) # set up resources
jobs = pool.imap(arrayFP, query, chunksize)
matrix = []
processed_mol = []
processed_id = []
for i, result in enumerate(jobs):
percent = (float(i)/float(len(query)))*100 + 1
sys.stdout.write(' Processing molecules: %3d%%\r' % percent)
sys.stdout.flush()
if result[0]:
matrix.append(result[0])
processed_mol.append(result[1])
processed_id.append(result[2])
pool.close()
pool.join()
sys.stdout.write(' Processing molecules: 100%')
matrix = np.array(matrix)
return matrix, processed_mol, processed_id
#preprocess exception to catch
class SdfNoneMolError(Exception):
'raised due to "None" mol during enumeration through Chem.SDMolSupplier'
#import user query for sdf
def importQuerySDF(in_file):
outfp = []
outid= []
outmol = []
query = Chem.SDMolSupplier(in_file)
for idx, m in enumerate(suppl):
sys.stdout.write(' Importing SDF file. Compound number: %s\r' % idx)
sys.stdout.flush()
try:
if not m: raise SdfNoneMolError('None mol')
smi, fp, mol = calcFingerprints(m,qtype='sdf')
try: outid.append(m.GetProp('_Name'))
except KeyError: outid.append(smi)
outfp.append(fp)
outmol.append(mol)
except SdfNoneMolError: print ' SDF parse error (compound index: ' + str(idx) + ')'
print
return np.array(outfp,dtype=np.uint8),outmol,outid
#unzip a pkl model
def open_Model(mod):
global mod_dir, options
with bz2.BZ2File(mod_dir + 'pkls' + sep + mod + '.pkl.bz2', 'rb') as bzfile:
clf = cPickle.load(bzfile)
#if set, increase number of trees in forest
if options.ntrees and clf.n_estimators < options.ntrees:
clf.set_params(n_estimators=options.ntrees)
return clf
#import the training data similarity, bias and standard deviation file for given model
def getAdData(model_name):
global mod_dir, sep
actual_mid = model_name.split('/')[-1].split('.pkl.bz2')[0]
ad_file = mod_dir + 'ad_analysis' + sep + actual_mid + '.pkl.bz2'
with bz2.BZ2File(ad_file, 'rb') as bzfile:
try:
ad_data = cPickle.load(bzfile)
except:
print(ad_file)
return ad_data
#perform AD analysis using full similarity weighted threshold [similarity/(bias * std_dev]
#adapted from DOI:10.1186/s13321-016-0182-y
def doSimilarityWeightedAdAnalysis(model_name):
global rdkit_mols, ad_settings
ad_idx = []
known = []
ad_data = getAdData(model_name)
required_threshold = np.percentile(ad_data[:,5],ad_settings)
for mol_idx, m in enumerate(rdkit_mols):
ad_flag = True
#only check for known compounds if set in options (True means check)
if options.known: k_flag = True
else: k_flag = False
for training_instance in ad_data:
sim = DataStructs.TanimotoSimilarity(m,training_instance[0])
#check if input=train & need to check input=train
if sim == 1.0 and k_flag == True:
known.append([mol_idx,training_instance[1]])
k_flag = False
weight = sim/(training_instance[2]*training_instance[3])
#if comp in AD & no comp already in AD
if weight >= required_threshold and ad_flag == True:
ad_idx.append(mol_idx)
ad_flag = False
#if compound is in AD and no need to check accross all comps for known then break
if k_flag == False and ad_flag == False: break
return ad_idx, np.array(known)
#get smiles from training set of model
def get_training_smiles(model_name):
target_name = model_name.split('_')[0].replace('SE,', '').replace(',SE', '')
thresh = model_name.split('_')[-1]
zip_filename = (
mod_dir + 'bioactivity_dataset' + sep + target_name + '.smi.zip'
)
training_filename = target_name + '.smi'
with zipfile.ZipFile(zip_filename, 'r') as zip_file:
with zip_file.open(training_filename) as training_file:
training_data = pd.read_csv(training_file, sep='\t',
low_memory=False)
target_smiles = training_data['Smiles']
target_thresh_smiles = list(
target_smiles[-np.isnan(training_data[thresh + '_Flag'])]
)
return target_thresh_smiles
#return percentile AD analysis for [similarity/(bias * std_dev] vs training data
def doPercentileCalculation(model_name):
global rdkit_mols
#expensive to unzip training file - so only done if smiles requested
if options.ad_smiles:
smiles = get_training_smiles(model_name)
ad_data = getAdData(model_name)
def calcPercentile(rdkit_mol):
sims = DataStructs.BulkTanimotoSimilarity(rdkit_mol,ad_data[:,0])
bias = ad_data[:,2].astype(float)
std_dev = ad_data[:,3].astype(float)
scores = ad_data[:,5].astype(float)
weights = sims / (bias * std_dev)
critical_weight = weights.max()
percentile = percentileofscore(scores,critical_weight)
if options.ad_smiles:
critical_smiles = smiles[np.argmax(weights)]
result = percentile, critical_smiles
else:
result = percentile, None
return result
ret = [calcPercentile(x) for x in rdkit_mols]
return model_name, ret
#prediction runner for percentile calculation
def performPercentileCalculation(models):
print ' Starting percentile calculation...'
input_len = len(models)
percentile_results = np.empty(input_len, dtype=object)
pool = Pool(processes=options.ncores)
chunksize = max(1, int(input_len / (10 * options.ncores)))
jobs = pool.imap_unordered(doPercentileCalculation, models, chunksize)
for i, result in enumerate(jobs):
progress = str(i+1) + '/' + str(input_len)
percent = '%3d%%\r' % (float(i)/float(input_len)*100 + 1)
sys.stdout.write(' Performing percentile calculation: ' + progress + ', ' + percent)
sys.stdout.flush()
if result is not None: percentile_results[i] = result
pool.close()
pool.join()
sys.stdout.write(' Performing percentile calculation: ' + progress + ', 100%')
print '\n Percentile calculation completed!'
return percentile_results
#calculate standard deviation for an input compound
def getStdDev(clf, querymatrix):
std_dev = []
for tree in range(len(clf.estimators_)):
std_dev.append(clf.estimators_[tree].predict_proba(querymatrix)[:,1])
std_dev = np.clip(np.std(std_dev,axis=0),0.001,None)
return std_dev
#raw or binary prediction worker
def doTargetPrediction(model_name):
global ad_settings
try:
#percentile ad analysis calculated from [near_neighbor_similarity/(bias*deviation)]
clf = open_Model(model_name)
ret = np.zeros(len(querymatrix))
ret.fill(np.nan)
try:
ad_idx, known = doSimilarityWeightedAdAnalysis(model_name)
except: return model_name, ret
#if no mols in AD then return
if len(ad_idx) == 0: return model_name, ret
probs = clf.predict_proba(querymatrix[ad_idx])[:,1].clip(0.001,0.999)
#return the standard deviation if turned on
if options.std:
std_dev = getStdDev(clf,querymatrix)
ret[ad_idx] = std_dev[ad_idx]
return model_name, ret
#will only have known if was set on
if len(known) > 0: probs[known[:,0]] = known[:,1]
if threshold: ret[ad_idx] = map(int,probs > threshold)
else: ret[ad_idx] = probs
except IOError: return None
return model_name, ret
#prediction runner for prediction or standard deviation calculation
def performTargetPrediction(models):
print ' Starting classification...'
input_len = len(models)
prediction_results = []
pool = Pool(processes=options.ncores, initializer=initPool, initargs=(querymatrix,options.proba,))
chunksize = max(1, int(input_len / (10 * options.ncores)))
jobs = pool.imap_unordered(doTargetPrediction, models, chunksize)
for i, result in enumerate(jobs):
progress = str(i+1) + '/' + str(input_len)
percent = '%3d%%\r' % ((float(i)/float(input_len))*100 + 1)
sys.stdout.write(' Performing classification on query molecules: ' + progress + ', ' + percent)
sys.stdout.flush()
if result is not None: prediction_results.append(result)
pool.close()
pool.join()
sys.stdout.write(' Performing classification on query molecules: ' + progress + ', 100%')
print '\n Classification completed!'
return prediction_results
#write out results
def writeOutResults(results, mode='predictions'):
global query_id, mid_uniprots, model_info, options
out_desc = ''
if options.transpose:
out_desc += 'transposed '
if mode == 'ad-percentiles':
out_desc += 'applicability domain percentiles '
elif mode == 'ad-smiles':
out_desc += 'applicability domain SMILES '
else:
out_desc += 'predictions '
prefix = options.inf
timestr = time.strftime('%Y%m%d-%H%M%S')
if options.off: output_name = options.off
else: output_name = prefix + '_out_' + timestr + mode + '.txt'
print ' Writing ' + out_desc + 'to file: ' + output_name
with open(output_name, 'wb') as out_file:
out_writer = csv.writer(out_file, delimiter='\t')
if options.transpose:
# rows are compounds, columns are targets (at a threshold)
out_data = [['Compound'] + query_id]
for mid, preds in results:
for uniprot_rows in mid_uniprots[mid]:
model_name = uniprot_rows[0] + "_" + uniprot_rows[8]
out_data += [[model_name] + list(preds)]
out_data = np.transpose(out_data)
for row in out_data: out_writer.writerow(row)
else:
# rows are targets (at a threshold), columns are compounds
out_data = []
for mid, preds in results:
for uniprot_rows in mid_uniprots[mid]:
if options.train_log: out_data += [uniprot_rows + model_info[mid][1:] + list(preds)]
else: out_data += [uniprot_rows + list(preds)]
header = ['Uniprot', 'Name', 'Gene_ID', 'Protein_Classification',
'Organism', 'PDB_IDs', 'DisGeNET_0.06',
'ChEMBL_First_Publication', 'Activity_Threshold',
'Actives', 'Orthologue_Actives', 'Inactives',
'Orthologue_Inactives', 'Sphere_Exclusion_Inactives',
'Ratio', 'Model_ID']
if options.train_log:
header.extend(model_info['MODEL_ID'][1:])
header.extend(query_id)
out_writer.writerow(header)
for row in out_data: out_writer.writerow(row)
return
#nt (Windows) compatibility initializer for the pool
def initPool(querymatrix_, threshold_=None):
global querymatrix, threshold
querymatrix = querymatrix_
threshold = threshold_
#main
if __name__ == '__main__':
introMessage()
print ' Predicting Targets for input: ' + options.inf
print ' Using ' + str(options.ncores) + ' core(s)'
if options.ntrees: print 'Latency warning: Number of trees will be increased to minimum: ' + str(options.ntrees)
print ' Using bioactivity thresholds (IC50/EC50/Ki/Kd) of: ' + str(options.bioactivity)
print ' Using orthologues: ' + str(options.ortho)
if options.organism: print ' Organism filter: ' + options.organism
if options.targetclass: print ' Target class filter: ' + options.targetclass
if options.minsize: print ' Minimum number of actives in training: ' + str(options.minsize)
if options.se_filt: print ' Filtering out models with Sphere Exclusion (SE)'
if options.train_log: print ' Training log will be added to output'
#set up environment
input_extension, sep, pidgin_dir, mod_dir, ad_settings = check_set_working_env()
#if preprocessing import standardizer
if options.preproc: from standardiser import standardise
#gather the models required and their information
mid_uniprots, model_info, target_count = get_Models()
print ' Total number of protein targets: ' + str(target_count)
print ' Total number of distinct models: ' + str(len(mid_uniprots))
print ' Using p(activity) threshold of: ' + str(options.proba)
print ' Importing query (calculating ECFP_4 fingerprints)'
#import user query files
if input_extension == 'sdf': querymatrix,rdkit_mols,query_id = importQuerySDF(options.inf)
else: querymatrix,rdkit_mols,query_id = importQuerySmiles(options.inf)
print ' Total number of query molecules: ' + str(len(querymatrix))
#perform target prediction on (filtered) models (using model ids)
if options.percentile:
results = performPercentileCalculation(mid_uniprots.keys())
else:
results = performTargetPrediction(mid_uniprots.keys())
#write out
if options.percentile:
percentile_results = [(mid, [cresult[0] for cresult in mresult]) for (mid, mresult) in results]
writeOutResults(percentile_results, 'ad-percentiles')
if options.ad_smiles:
smiles_results = [(mid, [cresult[1] for cresult in mresult]) for (mid, mresult) in results]
writeOutResults(smiles_results, 'ad-smiles')
else:
writeOutResults(results)