From 5ca857b49e3c60a74f37a3344f2d58a6c99dc39f Mon Sep 17 00:00:00 2001 From: Eric Kutschera Date: Mon, 12 Dec 2022 08:47:59 -0500 Subject: [PATCH] IRIS v2.0.0 --- .gitignore | 4 + IRIS/IRIS_annotate_ijc.py | 220 ++ IRIS/IRIS_append_cpm.py | 57 + IRIS/IRIS_append_sjc.py | 95 + IRIS/IRIS_epitope_post.py | 176 +- IRIS/IRIS_extract_sjc.py | 238 +++ IRIS/IRIS_formatting.py | 163 +- IRIS/IRIS_indexing.py | 24 +- IRIS/IRIS_makeqsub_mapping.py | 69 - IRIS/IRIS_makeqsub_rmats.py | 32 - IRIS/IRIS_makesubsh_extractsj.py | 55 + IRIS/IRIS_makesubsh_hla.py | 58 + IRIS/IRIS_makesubsh_mapping.py | 55 + IRIS/IRIS_makesubsh_rmats.py | 66 + IRIS/IRIS_makesubsh_rmatspost.py | 48 + IRIS/IRIS_ms_makedb.py | 5 +- IRIS/IRIS_parse_hla.py | 29 +- IRIS/IRIS_pep2epitope.py | 29 +- IRIS/IRIS_prediction.py | 202 +- IRIS/IRIS_screening.py | 554 ++++-- IRIS/IRIS_screening_cpm.py | 426 ++++ IRIS/IRIS_screening_novelss.py | 538 +++++ IRIS/IRIS_screening_plot.py | 84 +- IRIS/IRIS_screening_sjc.py | 276 +++ IRIS/IRIS_screening_sjcplot.py | 193 ++ IRIS/IRIS_seq2hla.py | 61 - IRIS/IRIS_sjc_matrix.py | 126 ++ IRIS/IRIS_translation.py | 305 ++- IRIS/IRIS_visual_summary.py | 1154 +++++++++++ IRIS/config.py | 14 +- IRIS/data/blacklist.brain_2020.txt | 369 ++++ IRIS_functions.md | 484 +++++ IRIS_modules.md | 271 --- README.md | 455 +++-- Snakefile | 1766 +++++++++++++++++ bin/IRIS | 465 ++++- conda.sh | 57 - conda_requirements_py2.txt | 7 + conda_requirements_py2_optional.txt | 10 + conda_requirements_py3.txt | 5 + conda_wrapper | 30 + docs/iris_diagram.png | Bin 0 -> 120132 bytes example/HLA_types/hla_exp.list | 22 - example/HLA_types/hla_patient.tsv | 22 - example/NEPC_test.para | 10 + example/SJ_matrices.tar.gz | Bin 20480 -> 0 bytes .../RL_100.matrix/JC.raw.input.SE.txt | 13 - .../SJ_matrices/RL_100.matrix/fromGTF.SE.txt | 13 - example/SJ_matrices/RL_100_rmatspost_list.txt | 1 - .../RL_150.matrix/JC.raw.input.SE.txt | 13 - .../SJ_matrices/RL_150.matrix/fromGTF.SE.txt | 13 - example/SJ_matrices/RL_150_rmatspost_list.txt | 1 - example/SJ_matrices/matrices.txt | 2 - example/SJ_matrices/samples.txt | 2 - example/Test.para | 10 - example/Test_simplified.para | 6 - example/exp_matrix_test.txt | 11 + example/gene_exp_file_list.txt | 22 - example/hla_patient_test.tsv | 10 + .../hla_types.list => hla_types_test.list} | 78 +- example/parameter_file_description.txt | 79 +- example/sjc_matrix/SJ_count.NEPC_example.txt | 31 + .../sjc_matrix/SJ_count.NEPC_example.txt.idx | 30 + .../splicing_matrix.SE.cov10.NEPC_example.txt | 11 + ...icing_matrix.SE.cov10.NEPC_example.txt.idx | 10 + google_drive_download.py | 187 ++ install | 318 +-- qsub/qsub.py | 163 -- qsub/requirements.txt | 1 - qsub/submit_qsub_and_wait.py | 63 - qsub/test | 21 - qsub/test_submit_qsub_and_wait.py | 60 - requirements.txt | 5 - run | 3 + run_example | 23 - run_iris | 238 --- scripts/check_read_lengths.py | 57 + scripts/count_iris_predict_tasks.py | 62 + scripts/prepare_iris_exp_matrix.py | 31 + scripts/prepare_iris_format.py | 57 + scripts/prepare_iris_sjc_matrix.py | 31 + scripts/write_param_file.py | 231 +++ set_env_vars.sh | 23 +- setup.py | 4 +- snakemake_config.yaml | 111 ++ snakemake_profile/.gitignore | 1 + snakemake_profile/cluster_commands.py | 384 ++++ snakemake_profile/cluster_commands_sge.py | 132 ++ snakemake_profile/cluster_status.py | 127 ++ snakemake_profile/cluster_submit.py | 149 ++ snakemake_profile/config.yaml | 36 + snakemake_profile/try_command.py | 40 + 92 files changed, 9998 insertions(+), 2245 deletions(-) create mode 100644 IRIS/IRIS_annotate_ijc.py create mode 100644 IRIS/IRIS_append_cpm.py create mode 100644 IRIS/IRIS_append_sjc.py create mode 100644 IRIS/IRIS_extract_sjc.py delete mode 100644 IRIS/IRIS_makeqsub_mapping.py delete mode 100644 IRIS/IRIS_makeqsub_rmats.py create mode 100644 IRIS/IRIS_makesubsh_extractsj.py create mode 100644 IRIS/IRIS_makesubsh_hla.py create mode 100644 IRIS/IRIS_makesubsh_mapping.py create mode 100644 IRIS/IRIS_makesubsh_rmats.py create mode 100644 IRIS/IRIS_makesubsh_rmatspost.py create mode 100644 IRIS/IRIS_screening_cpm.py create mode 100644 IRIS/IRIS_screening_novelss.py create mode 100644 IRIS/IRIS_screening_sjc.py create mode 100644 IRIS/IRIS_screening_sjcplot.py delete mode 100644 IRIS/IRIS_seq2hla.py create mode 100644 IRIS/IRIS_sjc_matrix.py create mode 100644 IRIS/IRIS_visual_summary.py create mode 100644 IRIS/data/blacklist.brain_2020.txt create mode 100644 IRIS_functions.md delete mode 100644 IRIS_modules.md create mode 100644 Snakefile delete mode 100644 conda.sh create mode 100644 conda_requirements_py2.txt create mode 100644 conda_requirements_py2_optional.txt create mode 100644 conda_requirements_py3.txt create mode 100755 conda_wrapper create mode 100644 docs/iris_diagram.png delete mode 100644 example/HLA_types/hla_exp.list delete mode 100644 example/HLA_types/hla_patient.tsv create mode 100644 example/NEPC_test.para delete mode 100644 example/SJ_matrices.tar.gz delete mode 100644 example/SJ_matrices/RL_100.matrix/JC.raw.input.SE.txt delete mode 100644 example/SJ_matrices/RL_100.matrix/fromGTF.SE.txt delete mode 100644 example/SJ_matrices/RL_100_rmatspost_list.txt delete mode 100644 example/SJ_matrices/RL_150.matrix/JC.raw.input.SE.txt delete mode 100644 example/SJ_matrices/RL_150.matrix/fromGTF.SE.txt delete mode 100644 example/SJ_matrices/RL_150_rmatspost_list.txt delete mode 100644 example/SJ_matrices/matrices.txt delete mode 100644 example/SJ_matrices/samples.txt delete mode 100644 example/Test.para delete mode 100644 example/Test_simplified.para create mode 100644 example/exp_matrix_test.txt delete mode 100644 example/gene_exp_file_list.txt create mode 100644 example/hla_patient_test.tsv rename example/{HLA_types/hla_types.list => hla_types_test.list} (63%) create mode 100644 example/sjc_matrix/SJ_count.NEPC_example.txt create mode 100644 example/sjc_matrix/SJ_count.NEPC_example.txt.idx create mode 100644 example/splicing_matrix/splicing_matrix.SE.cov10.NEPC_example.txt create mode 100644 example/splicing_matrix/splicing_matrix.SE.cov10.NEPC_example.txt.idx create mode 100644 google_drive_download.py delete mode 100644 qsub/qsub.py delete mode 100644 qsub/requirements.txt delete mode 100644 qsub/submit_qsub_and_wait.py delete mode 100755 qsub/test delete mode 100644 qsub/test_submit_qsub_and_wait.py delete mode 100644 requirements.txt create mode 100755 run delete mode 100755 run_example delete mode 100755 run_iris create mode 100644 scripts/check_read_lengths.py create mode 100644 scripts/count_iris_predict_tasks.py create mode 100644 scripts/prepare_iris_exp_matrix.py create mode 100644 scripts/prepare_iris_format.py create mode 100644 scripts/prepare_iris_sjc_matrix.py create mode 100644 scripts/write_param_file.py create mode 100644 snakemake_config.yaml create mode 100644 snakemake_profile/.gitignore create mode 100644 snakemake_profile/cluster_commands.py create mode 100644 snakemake_profile/cluster_commands_sge.py create mode 100644 snakemake_profile/cluster_status.py create mode 100644 snakemake_profile/cluster_submit.py create mode 100644 snakemake_profile/config.yaml create mode 100644 snakemake_profile/try_command.py diff --git a/.gitignore b/.gitignore index 2c7b77f..424dcaa 100644 --- a/.gitignore +++ b/.gitignore @@ -12,3 +12,7 @@ IRIS.egg-info/ build/ dist/ __pycache__/ +conda_env_2/ +conda_env_3/ +/.snakemake/ +/readline/ diff --git a/IRIS/IRIS_annotate_ijc.py b/IRIS/IRIS_annotate_ijc.py new file mode 100644 index 0000000..e2d39d2 --- /dev/null +++ b/IRIS/IRIS_annotate_ijc.py @@ -0,0 +1,220 @@ +import numpy as np +import sys +import os, glob, pyBigWig, argparse +from scipy import stats +import statsmodels.stats.weightstats as smw +from . import config +import warnings +warnings.filterwarnings("ignore") + +#python retreive_SJ_info.py test_sj.para SE event_list_test.txt sj_info +def read_SJMatrix_index(fn,outdir): + index = {} + for line in open(outdir+'/'+fn.split('/')[-1]+'.idx', 'r'): + ele = line.strip().split() + index[ele[0]] = int(ele[1]) + return index + +def fetch_SJMatrix(eid, fn, delim, index, head_only): + with open(fn, 'r') as f: + if head_only: + ele = f.readline().strip().split(delim) + retrieved_text = np.asarray([ x.split('.aln')[0] for x in ele ]) + else: + f.seek(index[eid], 0) + retrieved_text = np.asarray(f.readline().strip().split(delim)) + return retrieved_text + + +def loadParametersRow(filter_para, panel_list): + filter_cutoffs='' + if filter_para.strip()!='': + filter_cutoffs = map(float,filter_para.strip().split(' ')[0].split(',')) + filter_panel_list = filter_para.strip().split(' ')[1].split(',') + panel_list+=filter_panel_list + else: + filter_panel_list =[] + return filter_cutoffs, filter_panel_list, panel_list + + + +def readEventRow(row, header_line): + if header_line=='' or header_line==False: + rs=row.strip().split('\t') + return rs + else: + rs=row.strip().split('\t') + return dict(zip(header_line, rs)) + +def convertAS2SJevent(line_dict, splicing_event_type):#only for this script. Diff + as_event=line_dict['as_event'] + event_s=as_event.split(':') + if splicing_event_type=='SE': + event_row_list=[event_s[2]+':'+str(int(event_s[6])+1)+':'+event_s[4], event_s[2]+':'+str(int(event_s[5])+1)+':'+event_s[7], event_s[2]+':'+str(int(event_s[6])+1)+':'+event_s[7]] + + elif splicing_event_type=='A5SS':# Only use one junction for inc. Need to improve by updating db later + event_row_list=[line_dict['chr']+':'+str(int(event_s[5])+1)+':'+event_s[8], line_dict['chr']+':'+str(int(event_s[7])+1)+':'+event_s[8]] + + elif splicing_event_type=='A3SS': # Only use one junction for inc. Need to improve by updating db later + event_row_list=[line_dict['chr']+':'+str(int(event_s[9])+1)+':'+event_s[4],line_dict['chr']+':'+str(int(event_s[9])+1)+':'+event_s[6]] + + else: + exit('[Error] Splicing event type not supported. Exiting.') + return event_row_list, as_event + +def summarizeSJ2ASevent(event_list_fin, splicing_event_type, sig_junction, outdir, out_prefix): + fout_summary_fname=outdir+'/SJ.'+out_prefix+'.'+splicing_event_type+'.summary_by_sig_event.txt' + fout_summary=open(fout_summary_fname,'w') + for event_idx, event_row in enumerate(open(event_list_fin)): + if event_idx==0: + header_list=readEventRow(event_row,'') + continue + line_dict=readEventRow(event_row, header_list) + event_row_list, as_event=convertAS2SJevent(line_dict, splicing_event_type) + as_event_result=[] + as_event_result_list=[] + for k in event_row_list: + if k not in sig_junction: + as_event_result.append(False) + else: + as_event_result.append(True) + as_event_result_list.append(k) + if as_event_result[0]==as_event_result[1]==as_event_result[2]==True: + fout_summary.write(as_event+'\tAll junctions\t'+';'.join(as_event_result_list)+'\n') + elif as_event_result[0]==as_event_result[1]==as_event_result[2]==False: + continue + else: + if as_event_result[0]==as_event_result[1]!=as_event_result[2]: + fout_summary.write(as_event+'\tOnly alternative junctions\t'+';'.join(as_event_result_list)+'\n') + else: + fout_summary.write(as_event+'\tOther combination\t'+';'.join(as_event_result_list)+'\n') + fout_summary.close() + return fout_summary_fname + + + + +def main(args): + ###Loading Parameters#### + para_fin=args.parameter_file + splicing_event_type=args.splicing_event_type + if splicing_event_type!='SE': + exit('[Error] Invalid AS event type.') + event_list_fin=args.screening_result_event_list + outdir=args.outdir.rstrip('/') + + os.system('mkdir -p '+outdir) + fetching_sj_col=1 + out_prefix,db_dir,filter1_para,filter2_para,filter3_para=[l.strip() for l in open(para_fin)][:5] + db_dir=db_dir.rstrip('/') + if os.path.isdir(db_dir+'_sjc'): #automatically use db_sjc if in the same dir. Otherwise, use the user input db_dir + db_dir=db_dir+'_sjc' + panel_list=[out_prefix] + + filter1_cutoffs, filter1_panel_list, panel_list = loadParametersRow(filter1_para, panel_list) + filter2_cutoffs, filter2_panel_list, panel_list = loadParametersRow(filter2_para, panel_list) + filter3_cutoffs, filter3_panel_list, panel_list = loadParametersRow(filter3_para, panel_list) + tumor_dict=dict.fromkeys(filter2_panel_list,'') + tumor_dict[out_prefix]='' + pvalue_cutoff_normal=''; pvalue_cutoff_tumor='' + filter1_group_cutoff=''; filter2_group_cutoff=''; filter3_group_cutoff=''; + if filter1_cutoffs!='': + pvalue_cutoff_normal,filter1_group_cutoff=filter1_cutoffs[3:] + if filter2_cutoffs!='': + pvalue_cutoff_tumor,filter2_group_cutoff=filter2_cutoffs[3:] + if filter3_cutoffs!='': + pvalue_cutoff_normal,filter3_group_cutoff=filter3_cutoffs[3:] + + inc_read_cov_cutoff=int(args.inc_read_cov_cutoff)#2 + event_read_cov_cutoff=int(args.event_read_cov_cutoff)#10 + + ##Load IRIS reference panels to 'fin_list', 'index' + index={} + fin_list={} + for group_name in panel_list: + fin_list[group_name]=db_dir+'/'+group_name+'/sjc_matrix/SJ_count.'+group_name+'.txt' + for group in fin_list.keys(): + if not os.path.isfile(fin_list[group]+'.idx'): + exit('[Error] Need to index '+fin_list[group]) + index[group]=read_SJMatrix_index(fin_list[group],'/'.join(fin_list[group].split('/')[:-1])) + + print('[INFO] Done loading index '+' '.join(panel_list)) + tot=config.file_len(event_list_fin)-1 + fout_ijc=open(outdir+'/'+event_list_fin.split('/')[-1]+'.ijc_info.txt','w') + header_line=[] + sample_size={} + + header_line=['ijc_ratio', 'mean_ijc_by_group', 'percent_sample_imbalanced', 'sample_imbalanced_by_group'] + header_list=[] + print('[INFO] Retrieving inclusion junction info') + for event_idx, event_row in enumerate(open(event_list_fin)): + config.update_progress(event_idx/(0.0+tot)) + if event_idx==0: + header_list=readEventRow(event_row,'') + fout_ijc.write(event_row.strip()+'\t'+'\t'.join(header_line)+'\n') + continue + line_dict=readEventRow(event_row, header_list) + event_row_list, as_event=convertAS2SJevent(line_dict, splicing_event_type) + + forms={} + if splicing_event_type=='SE': + inc1, inc2, skp= event_row_list + forms={'inc1':inc1,'inc2':inc2,'skp':skp} + else: + exit('[Error] Invalid AS event type.') + #Initiate psi matrix by each row to 'sj' + + sj={} + for form in forms: + k=forms[form] + sj[form]={} + for group in panel_list: + random_key=index[group].keys()[0] + sample_names=map(str,fetch_SJMatrix(random_key,fin_list[group],'\t',index[group],True)[fetching_sj_col:]) + sample_size[group]=len(sample_names) + if k in index[group]: + sj[form][group]=map(int,fetch_SJMatrix(k,fin_list[group],'\t',index[group], False)[fetching_sj_col:]) + else: + sj[form][group]=[0]*sample_size[group] + + imbalance={} + total_sample={} + inc_ratio={} + i1_list={} + i2_list={} + for group in panel_list: + i1=map(int,sj['inc1'][group]) + i2=map(int,sj['inc2'][group]) + s=map(int,sj['skp'][group]) + i1_list[group]=[] + i2_list[group]=[] + imbalance[group]=0 + total_sample[group]=0 + for pidx in range(0, len(sj['inc1'][group])): + if i1[pidx]+i2[pidx]2 or ratio<0.5: + imbalance[group]+=1 + i1_list[group].append(i1[pidx]) + i2_list[group].append(i2[pidx]) + + i_by_group='|'.join([str(round(np.mean(i1_list[group]),1))+','+str(round(np.mean(i2_list[group]),1)) for group in panel_list]) + i1_sum=sum(sum(i1_list[g]) for g in panel_list) + i2_sum=sum(sum(i2_list[g]) for g in panel_list) + ratio=round(i1_sum/(i2_sum+0.01),3) + + imbalance_by_group='|'.join([str(imbalance[group])+','+str(total_sample[group]) for group in panel_list]) + imb_count=sum(imbalance[group] for group in panel_list) + tot_count=sum(total_sample[group] for group in panel_list) + imb_perc='-' + if tot_count!=0: + imb_perc=round(imb_count/(tot_count+0.0),3) + fout_ijc.write(event_row.rstrip()+'\t'+'\t'.join([as_event,str(ratio), i_by_group, str(imb_perc),imbalance_by_group])+'\n') + + fout_ijc.close() + + +if __name__ == '__main__': + main() diff --git a/IRIS/IRIS_append_cpm.py b/IRIS/IRIS_append_cpm.py new file mode 100644 index 0000000..d3bd688 --- /dev/null +++ b/IRIS/IRIS_append_cpm.py @@ -0,0 +1,57 @@ +import sys, glob, os +from . import config +#append annotation to all screening result and epitope result in a specified screening folder +def loadPTsummary(fin): + PT_event={} + for l in open(fin): + ls=l.strip().split('\t') + PT_event[ls[0]]=ls[1]+'|'+ls[2]+'|'+ls[3]+'|'+ls[4] + return PT_event + + +def annotateRAbyPTEvent(fin, event_col, CPM_event, fout_fname): + fout=open(fout_fname,'w') + for n,l in enumerate(open(fin)): + if n==0: + annotation='cpm_test_summary' + fout.write(l.strip()+'\t'+annotation+'\n') + continue + ls=l.strip().split('\t') + annotation='-' + if CPM_event.has_key(ls[event_col]): + annotation=CPM_event[ls[event_col]] + fout.write(l.strip()+'\t'+annotation+'\n') + fout.close() + +def main(args): + CPMsummary=args.cpm_summary + CPM_event=loadPTsummary(CPMsummary) + print('[INFO] Number of event in CPM screen summary:'+str(len(CPM_event))) + splicing_event_type=args.splicing_event_type + screening_result_dir=args.outdir.rstrip('/') + + epitope_file_junction_file_list= glob.glob(screening_result_dir+'/'+splicing_event_type+'.*/epitope_summary.junction-based.txt') + epitope_file_peptide_file_list= glob.glob(screening_result_dir+'/'+splicing_event_type+'.*/epitope_summary.peptide-based.txt') + extracellular_as_file_list= glob.glob(screening_result_dir+'/*'+splicing_event_type+'.*.ExtraCellularAS.txt') + screening_file_list=glob.glob(screening_result_dir+'/*.'+splicing_event_type+'.tier1.txt')+glob.glob(screening_result_dir+'/*.'+splicing_event_type+'.tier2tier3.txt') + + + for fin_fname in screening_file_list: + print '[INFO] Integrating CPM test result to', fin_fname + annotateRAbyPTEvent(fin_fname, 0, CPM_event, fin_fname+'.integratedCPM.txt') + + for fin_fname in extracellular_as_file_list: + print '[INFO] Integrating CPM test result to', fin_fname + annotateRAbyPTEvent(fin_fname, 0, CPM_event, fin_fname+'.integratedCPM.txt') + + for fin_fname in epitope_file_peptide_file_list: + print '[INFO] Integrating CPM test result to', fin_fname + annotateRAbyPTEvent(fin_fname, 1, CPM_event, fin_fname+'.integratedCPM.txt') + + for fin_fname in epitope_file_junction_file_list: + print '[INFO] Integrating CPM test result to', fin_fname + annotateRAbyPTEvent(fin_fname, 0, CPM_event, fin_fname+'.integratedCPM.txt') + + +if __name__ == '__main__': + main() diff --git a/IRIS/IRIS_append_sjc.py b/IRIS/IRIS_append_sjc.py new file mode 100644 index 0000000..3421e0d --- /dev/null +++ b/IRIS/IRIS_append_sjc.py @@ -0,0 +1,95 @@ +import sys, glob, os +from . import config +#append annotation to all screening result and epitope result in a specified screening folder +def loadPTsummary(fin): + PT_event={} + for l in open(fin): + ls=l.strip().split('\t') + PT_event[ls[0]]=ls[1]+'|'+ls[2] + return PT_event + +def loadIJCsummary(fin): + IJC_info={} + for i,l in enumerate(open(fin)): + if i==0: + continue + ls=l.strip().split('\t') + IJC_info[ls[0]]=ls[-4]+'\t'+ls[-3]+'\t'+ls[-2]+'\t'+ls[-1] + return IJC_info + +def annotateRAbyPTEvent(fin, event_col, PT_event, fout_fname, IJC_info): + fout=open(fout_fname,'w') + for n,l in enumerate(open(fin)): + if n==0: + annotation='sjc_test_summary' + if IJC_info!={}: + annotation+='\t'+'\t'.join(['ijc_ratio', 'mean_ijc_by_group', 'percent_sample_imbalanced', 'sample_imbalanced_by_group']) + fout.write(l.strip()+'\t'+annotation+'\n') + continue + ls=l.strip().split('\t') + annotation='-' + if PT_event.has_key(ls[event_col]): + annotation=PT_event[ls[event_col]] + if IJC_info!={}: + if IJC_info.has_key(ls[event_col]): + annotation+='\t'+IJC_info[ls[event_col]] + else: + annotation+='\t'+'\t'.join(['-']*4) + fout.write(l.strip()+'\t'+annotation+'\n') + fout.close() + +def main(args): + PTsummary=args.sjc_summary + PT_event=loadPTsummary(PTsummary) + print('[INFO] Number of event in SJ count screen summary:'+str(len(PT_event))) + splicing_event_type=args.splicing_event_type + screening_result_dir=args.outdir.rstrip('/') + + epitope_file_junction_file_list= glob.glob(screening_result_dir+'/'+splicing_event_type+'.*/epitope_summary.junction-based.txt') + epitope_file_peptide_file_list= glob.glob(screening_result_dir+'/'+splicing_event_type+'.*/epitope_summary.peptide-based.txt') + extracellular_as_file_list= glob.glob(screening_result_dir+'/*'+splicing_event_type+'.*.ExtraCellularAS.txt') + screening_file_list=glob.glob(screening_result_dir+'/*.'+splicing_event_type+'.tier1.txt')+glob.glob(screening_result_dir+'/*.'+splicing_event_type+'.tier2tier3.txt') + + #optional, if ijc imbalance + add_ijc_info=args.add_ijc_info + use_existing_result=args.use_existing_result#use existing ijc result + para_file=args.parameter_file + event_list_file=args.screening_result_event_list + inc_read_cov_cutoff=args.inc_read_cov_cutoff#2 + event_read_cov_cutoff=args.event_read_cov_cutoff#10 + IJC_info={} + if add_ijc_info or use_existing_result: + if para_file=='' or event_list_file=='': + exit('[Error] Specify parameters and event list file for retrieving inclusion junction information.') + + if add_ijc_info and use_existing_result==False: + cmd='IRIS annotate_ijc -p '+para_file+' --splicing-event-type '+splicing_event_type+' -e '+event_list_file+' -o '+screening_result_dir+' --inc-read-cov-cutoff '+str(inc_read_cov_cutoff)+' --event-read-cov-cutoff '+str(event_read_cov_cutoff) + print('[INFO] Annotating inclusion junction info to '+event_list_file) + print(cmd) + os.system(cmd) + if use_existing_result or add_ijc_info: + file_path=screening_result_dir+'/'+event_list_file.split('/')[-1]+'.ijc_info.txt' + if os.path.exists(file_path)==False: + exit('[Error] Result IJC annotation file not found in path '+file_path) + print('[INFO] Loading inclusion junction info from '+file_path) + IJC_info=loadIJCsummary(file_path) + + for fin_fname in screening_file_list: + print '[INFO] Integrating SJC test result to', fin_fname + annotateRAbyPTEvent(fin_fname, 0, PT_event, fin_fname+'.integratedSJC.txt',IJC_info) + + for fin_fname in extracellular_as_file_list: + print '[INFO] Integrating SJC test result to', fin_fname + annotateRAbyPTEvent(fin_fname, 0, PT_event, fin_fname+'.integratedSJC.txt',IJC_info) + + for fin_fname in epitope_file_peptide_file_list: + print '[INFO] Integrating SJC test result to', fin_fname + annotateRAbyPTEvent(fin_fname, 1, PT_event, fin_fname+'.integratedSJC.txt',IJC_info) + + for fin_fname in epitope_file_junction_file_list: + print '[INFO] Integrating SJC test result to', fin_fname + annotateRAbyPTEvent(fin_fname, 0, PT_event, fin_fname+'.integratedSJC.txt',IJC_info) + + +if __name__ == '__main__': + main() diff --git a/IRIS/IRIS_epitope_post.py b/IRIS/IRIS_epitope_post.py index bc9d49d..ee0eb8a 100644 --- a/IRIS/IRIS_epitope_post.py +++ b/IRIS/IRIS_epitope_post.py @@ -3,6 +3,7 @@ from . import config # SUPPORT -u mode. will parse both form + def parsePredFile(path_to_peptide_file,fin,fout_pass,IC50_cutoff,gene_name,junction,pred_med,length): seq_list=[] seq_list_seq=[] @@ -25,9 +26,30 @@ def parsePredFile(path_to_peptide_file,fin,fout_pass,IC50_cutoff,gene_name,junct if med_ic50<=IC50_cutoff: fout_pass.write('{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\n'.format(gene_name, junction,seq_list[int(l['seq_num'])-1],l['start'],l['end'],length,pred_med, l['allele'],l['peptide'],junc_pep,med_ic50)) -def writePositivePrediction(output_path, IC50_cutoff): +def retrievePrioritizedfromPrimary(output_path, splicing_event_type): + fin_list=glob.glob(output_path+'/tmp/prot.compared/skp/*.fa') + fin_list=fin_list+glob.glob(output_path+'/tmp/prot.compared/inc/*.fa') + pred_fin_list=[] + retrieve_path='/'.join(output_path.split('/')[:-1])+'/'+splicing_event_type+'.tier1' + for fin in fin_list: + event_name_list=fin.split('/')[-1].split('.') + prediction_result_fn=retrieve_path+'/tmp/pred/'+event_name_list[1].split('_')[0]+'/'+'.'.join(event_name_list[:-2])##WORKING ON + prediction_result_fin_list=glob.glob(prediction_result_fn+'*') + pred_fin_list+=prediction_result_fin_list + return pred_fin_list + +def writePositivePrediction(output_path, IC50_cutoff, splicing_event_type, retrieve_prioritized): print '[INFO] Collecting binding predictions:'+output_path.rstrip('/').split('/')[-1] - pred_fin_list=glob.glob(output_path+'/tmp/pred/*/*') + pred_fin_list=[] + if retrieve_prioritized: + if glob.glob(output_path+'/tmp/pred/*/*')!=[] and glob.glob('/'.join(output_path.split('/')[:-1])+'/'+splicing_event_type+'.tier1'+'/tmp/pred/*/*')==[]: + print '[INFO] Tier1 comparison result is empty. Retrieving tier2&tier3 prediction.' + pred_fin_list=glob.glob(output_path+'/tmp/pred/*/*') + else: + print '[INFO] Retrieving t prediction from tier1 comparison.' + pred_fin_list= retrievePrioritizedfromPrimary(output_path, splicing_event_type) + else: + pred_fin_list=glob.glob(output_path+'/tmp/pred/*/*') fout_pass=open(output_path+'/pred_filtered.score'+str(IC50_cutoff)+'.txt','w') tot=len(pred_fin_list)-1 for i,fin in enumerate(pred_fin_list): @@ -86,17 +108,69 @@ def loadGeneExp(gene_exp_matrix_fin): Exp[name]=map(str,[mean, Q1, Q3]) return Exp,['meanGeneExp','Q1GeneExp','Q3GeneExp'] +def buildJunctionKmer(seq, kmer_length, kmer_dict, name, keeper, anno): + if anno in keeper: + return kmer_dict + + else: + keeper[anno]='' + for k in kmer_length: + for i in range(len(seq)-k+1): + s=seq[i:i+k] + if s not in kmer_dict: + kmer_dict[s]=[] + kmer_dict[s].append(name) + return kmer_dict + +def epitopeUniquenessAnnotation(kmer_length, screening_dir): + kmer={} + keeper={} + peptide_fin_dir=screening_dir+'/*.*/tmp/prot/*' + peptide_fin_list=glob.glob(peptide_fin_dir) + print '[INFO] Total pool of splice junctions:',len(peptide_fin_list) + tot=len(peptide_fin_list)-1 + for m, peptide_fin in enumerate(peptide_fin_list): + config.update_progress(m/(0.0+tot)) + pep_gene_name=peptide_fin.split('/')[-1].split('_')[0] + anno='' + for l in open(peptide_fin): + if l[0]=='>': + anno=l.strip() + continue + ls=l.strip().upper() + kmer=buildJunctionKmer(ls, kmer_length, kmer, pep_gene_name, keeper, anno) + anno='' + print '[INFO] Total number of AS junction k mers from the analyzed sequencing data:',len(kmer) + return kmer + + +def loadKmer(kmer_file): + kmer={} + for l in open(kmer_file): + ls=l.strip().split('\t') + kmer[ls[0]]=ls[1].strip(';') + return kmer -def writePeptideSummary(output_path, screening_result_path, gene_exp_path, IC50_cutoff, sample_HLA, sample_list): +def writePeptideSummary(output_path, screening_result_path, gene_exp_path, IC50_cutoff, sample_HLA, sample_list, match_normal, uniqueness, epitope_len_list, db_dir): pep_dict={} - screening_result_dict, screening_result_header=loadScreening(screening_result_path) + kmer_dict={} + header_ext=[] if gene_exp_path: gene_exp_dict, gene_exp_header=loadGeneExp(gene_exp_path)#exp. or fpkm1. matrix + header_ext+=gene_exp_header + if match_normal: + print '[INFO] Loading kmers for checking canonical proteome: '+','.join(map(str,epitope_len_list)) + for kmer_len in epitope_len_list: + kmer_dict[kmer_len]=loadKmer(db_dir+'/resources/kmers/uniprot-all.fasta.'+str(kmer_len)+'mer_dict.txt') + header_ext+=['canonical_match'] + if uniqueness: + junction_kmer=epitopeUniquenessAnnotation(epitope_len_list, '/'.join(screening_result_path.split('/')[:-1])) + header_ext+=['uniqueness'] + screening_result_dict, screening_result_header=loadScreening(screening_result_path) + fout_screening=open(output_path+'/epitope_summary.peptide-based.txt','w') - if gene_exp_path: - fout_screening.write('\t'.join(['epitope','as_event','junction_peptide_form','num_hla','num_sample']+sample_list+screening_result_header+gene_exp_header)+'\n') - else: - fout_screening.write('\t'.join(['epitope','as_event','junction_peptide_form','num_hla','num_sample']+sample_list+screening_result_header)+'\n') + fout_screening.write('\t'.join(['epitope','as_event','junction_peptide_form','inclusion_form','num_hla','num_sample','hla_types']+sample_list+screening_result_header+header_ext)+'\n') + for l in open(output_path+'/pred_filtered.score'+str(IC50_cutoff)+'.txt'): ls=l.strip().split('\t') @@ -105,15 +179,18 @@ def writePeptideSummary(output_path, screening_result_path, gene_exp_path, IC50_ pred_score=ls[10] splicing=(ls[0]+'_'+ls[1]).replace('_',':') form=ls[2].split(':')[5][:3] + inc_form='' + if form=='inc': + inc_form=ls[2].split(':')[5][:4] if peptide+'|'+splicing+form not in pep_dict: - pep_dict[peptide+'|'+splicing+form]=[] - pep_dict[peptide+'|'+splicing+form].append(hla_type+'|'+pred_score) -# print len(pep_dict) + pep_dict[peptide+'|'+splicing+form+'|'+inc_form]=[] + pep_dict[peptide+'|'+splicing+form+'|'+inc_form].append(hla_type+'|'+pred_score) for k in pep_dict: ks=k.split('|') peptide=ks[0] splicing=ks[1][:-3] form=ks[1][-3:] + inc_form=ks[2] hla_info=pep_dict[k] hla={} for info in hla_info: @@ -122,7 +199,7 @@ def writePeptideSummary(output_path, screening_result_path, gene_exp_path, IC50_ pred_score=info_split[1] hla[hla_type]=pred_score num_hla=len(hla) - line= [peptide, splicing, form, str(num_hla)] + line= [peptide, splicing, form, inc_form, str(num_hla), ';'.join(sorted(hla.keys()))] patient_count=len(sample_list) for p in sample_list: patient_hla_info=[] @@ -135,11 +212,25 @@ def writePeptideSummary(output_path, screening_result_path, gene_exp_path, IC50_ patient_hla_info_line=';'.join(patient_hla_info) line.append(patient_hla_info_line) line.insert(4,str(patient_count)) + optional_annotations=[] if gene_exp_path: gene_exp_list=gene_exp_dict[splicing.split(':')[0]] - fout_screening.write('\t'.join(line+screening_result_dict[splicing]+gene_exp_list)+'\n') - else: - fout_screening.write('\t'.join(line+screening_result_dict[splicing])+'\n') + optional_annotations+=gene_exp_list + if match_normal: + matched_protein='NonCanonical' if peptide not in kmer_dict[len(peptide)] else kmer_dict[len(peptide)][peptide] + optional_annotations+=[matched_protein] + if uniqueness: + if peptide not in junction_kmer: + print 'error', ls[0] + else: + uniqueness_annotation='-' + if len(junction_kmer[peptide])==1: + uniqueness_annotation='unique:'+junction_kmer[peptide][0] + else: + uniqueness_annotation='multi:'+';'.join(junction_kmer[peptide]) + optional_annotations+=[uniqueness_annotation] + fout_screening.write('\t'.join(line+screening_result_dict[splicing]+optional_annotations)+'\n') + def writeJunctionSummary(output_path, screening_result_path, gene_exp_path, IC50_cutoff, sample_HLA, sample_list): junction_dict={} @@ -159,13 +250,9 @@ def writeJunctionSummary(output_path, screening_result_path, gene_exp_path, IC50 pred_score=ls[10] splicing=(ls[0]+'_'+ls[1]).replace('_',':') form=ls[2].split(':')[5][:3] - # if peptide+'|'+splicing+form not in pep_dict: - # pep_dict[peptide+'|'+splicing+form]=[] - # pep_dict[peptide+'|'+splicing+form].append(hla_type+'|'+pred_score) if splicing+form not in junction_dict: junction_dict[splicing+form]=[] junction_dict[splicing+form].append(hla_type+'|'+pred_score) -# print len(pep_dict) for k in junction_dict: splicing=k[:-3] form=k[-3:] @@ -192,28 +279,53 @@ def writeJunctionSummary(output_path, screening_result_path, gene_exp_path, IC50 patient_hla_info_line=';'.join(patient_hla_info) line.append(patient_hla_info_line) line.insert(3,str(patient_count)) + optional_annotations=[] if gene_exp_path: gene_exp_list=gene_exp_dict[splicing.split(':')[0]] - fout_screening.write('\t'.join(line+screening_result_dict[splicing]+gene_exp_list)+'\n') - else: - fout_screening.write('\t'.join(line+screening_result_dict[splicing])+'\n') + optional_annotations+=gene_exp_list + fout_screening.write('\t'.join(line+screening_result_dict[splicing]+optional_annotations)+'\n') def main(args): outdir=args.outdir + splicing_event_type=args.splicing_event_type IC50_cutoff=args.ic50_cut_off gene_exp_matrix=args.gene_exp_matrix + match_normal=True if args.no_match_to_canonical_proteome==False else False + uniqueness=True if args.no_uniqueness_annotation==False else False + prioritized_only=args.tier3_only + keep_exist=args.keep_exist + + write_positive=True + if keep_exist==True: + if os.path.exists(outdir+'/'+splicing_event_type+'.tier2tier3/pred_filtered.score'+str(IC50_cutoff)+'.txt') or os.path.exists(outdir+'/'+splicing_event_type+'.tier1/pred_filtered.score'+str(IC50_cutoff)+'.txt'): + write_positive=False - writePositivePrediction(outdir+'/primary', IC50_cutoff) - writePositivePrediction(outdir+'/prioritized', IC50_cutoff) + sample_list,sample_HLA=loadSampleHLA(args.mhc_by_sample) + analysis_name,db_dir=[l.strip() for l in open(args.parameter_fin)][0:2] + db_dir='/'.join(db_dir.rstrip('/').split('/')[:-1]) + analysis_name=analysis_name+'.'+splicing_event_type + screening_result_path=outdir+'/'+analysis_name + if config.file_len(screening_result_path+'.tier1.txt')==1 and prioritized_only==False: + prioritized_only=True + print "[INFO] No tier1 comparisons (tissue-matched normal) found. Use tier2tier3 only mode. " + + epitope_len_list=map(int,args.epitope_len_list.split(',')) + if write_positive: + if prioritized_only: + writePositivePrediction(outdir+'/'+splicing_event_type+'.tier2tier3', IC50_cutoff, splicing_event_type, False) + else: + writePositivePrediction(outdir+'/'+splicing_event_type+'.tier1', IC50_cutoff, splicing_event_type, False) + writePositivePrediction(outdir+'/'+splicing_event_type+'.tier2tier3', IC50_cutoff, splicing_event_type, True) + else: + print '[INFO] File(s) '+outdir+'/'+splicing_event_type+'.*/pred_filtered.score'+str(IC50_cutoff)+'.txt'+' exist. Skip the step generating this output.' + + if prioritized_only==False: + writePeptideSummary(outdir+'/'+splicing_event_type+'.tier1', outdir+'/'+analysis_name+'.tier1.txt', gene_exp_matrix, IC50_cutoff, sample_HLA, sample_list, match_normal, uniqueness, epitope_len_list, db_dir) + writePeptideSummary(outdir+'/'+splicing_event_type+'.tier2tier3', outdir+'/'+analysis_name+'.tier2tier3.txt', gene_exp_matrix, IC50_cutoff, sample_HLA, sample_list, match_normal, uniqueness, epitope_len_list, db_dir) - sample_list,sample_HLA=loadSampleHLA(args.mhc_by_sample) - analysis_name=[l.strip() for l in open(args.parameter_fin)][0] - screening_result_path=outdir+'/'+analysis_name+'.primary.txt' - writePeptideSummary(outdir+'/primary', outdir+'/'+analysis_name+'.primary.txt', gene_exp_matrix, IC50_cutoff, sample_HLA, sample_list) - writePeptideSummary(outdir+'/prioritized', outdir+'/'+analysis_name+'.prioritized.txt', gene_exp_matrix, IC50_cutoff, sample_HLA, sample_list) - - writeJunctionSummary(outdir+'/primary', outdir+'/'+analysis_name+'.primary.txt', gene_exp_matrix, IC50_cutoff, sample_HLA, sample_list) - writeJunctionSummary(outdir+'/prioritized', outdir+'/'+analysis_name+'.prioritized.txt', gene_exp_matrix, IC50_cutoff, sample_HLA, sample_list) + if prioritized_only==False: + writeJunctionSummary(outdir+'/'+splicing_event_type+'.tier1', outdir+'/'+analysis_name+'.tier1.txt', gene_exp_matrix, IC50_cutoff, sample_HLA, sample_list) + writeJunctionSummary(outdir+'/'+splicing_event_type+'.tier2tier3', outdir+'/'+analysis_name+'.tier2tier3.txt', gene_exp_matrix, IC50_cutoff, sample_HLA, sample_list) if __name__ == '__main__': diff --git a/IRIS/IRIS_extract_sjc.py b/IRIS/IRIS_extract_sjc.py new file mode 100644 index 0000000..406937e --- /dev/null +++ b/IRIS/IRIS_extract_sjc.py @@ -0,0 +1,238 @@ +import argparse +import pysam +import os +# Adopted by Yang Pan 2020.12.20 (panyang@ucla.edu) +# Author: Robert Wang, PhD Student (Xing Lab) +# Date: 2020.10.02 +# E-mail: robwang@pennmedicine.upenn.edu + +# This is a script that extracts splice junctions from a STAR-aligned BAM +# file and annotates each splice junction with the number of uniquely +# mapped reads that support that splice junction. Supporting reads for +# annotated splice junctions are by default only required to have a +# minimum overhang of 1 bp. Supporting reads for unannotated and +# canonical splice junctions (GT-AG/CT-AC; GC-AG/CT-GC; AT-AC/GT-AT) are +# by default only required to have a minimum overhang of 8 bp. Supporting +# reads for unannotated and non-canonical splice junctions however are +# by default required to have a minimum overhang of 10 bp. The resulting +# output file is a TSV with two fields: (1) a splice junction ID with +# format [chr#]:[start]:[end], (2) number of uniquely mapped reads +# supporting the splice junction. All genomic coordinates used in +# describing each splice junction are 1-based. Make sure that the BAM +# file and genome FASTA file have been indexed. + +# UPDATE (2020.10.02): Users have the option of strictly filtering +# for reads of a specified size using the -r option + +# Usage: +# python extract_SJ.py -i /path/to/BAM/file \ +# -g /path/to/annotation/GTF/file \ +# -a [minimum overhang length for annotated SJs, default: 1] +# -c [minimum overhang length for unannotated canonical SJs, default: 8] \ +# -n [minimum overhang length for unannotated non-canonical SJs, default: 10] \ +# -r [length of reads to keep when counting junction reads] \ +# -f /path/to/genome/fasta/file \ +# -o /path/to/output/file + +# Dependencies: +# * argparse +# * pysam + +def get_introns(blocks): + introns = [] + + # N blocks are associated with N-1 introns + for i in range(len(blocks)-1): + intronStart = blocks[i][1] + 1 + intronEnd = blocks[i+1][0] + introns += [(intronStart, intronEnd)] + + return introns + +def isCanonical(dn1, dn2): + # Construct tuple representing SJ dinucleotides + sjDN = (dn1, dn2) + + # Establish canonical SJs + canonicalSJ = [('GT', 'AG'), ('CT', 'AC'), ('GC', 'AG'), + ('CT', 'GC'), ('AT', 'AC'), ('GT', 'AT')] + + return sjDN in canonicalSJ + +def get_threshold(chr, introns, genome, annoSJ, minOverhang, minOverhangC, minOverhangNC): + isAnno = [] + isCanon = [] + + # Iterate through each intron + for i in range(len(introns)): + # Construct splice junction ID + sjInfo = [chr, introns[i][0], introns[i][1]] + sjID = ':'.join(map(str, sjInfo)) + + # Retrieve dinucleotides for splice junction + dn1 = genome.fetch(chr, introns[i][0]-1, introns[i][0]+1) + dn2 = genome.fetch(chr, introns[i][1]-2, introns[i][1]) + + isAnno.append(sjID in annoSJ) + isCanon.append(isCanonical(dn1, dn2)) + + if all(isAnno): + threshold = minOverhang + elif all(isCanon): + threshold = minOverhangC + else: + threshold = minOverhangNC + + return threshold + +def update_SJdb(read, sjDB, annoSJ, genome, minOverhang, minOverhangC, minOverhangNC): + # Get chromosome for read + chr = read.reference_name + + # Extract coordinates of blocks for each read + blocks = read.get_blocks() + introns = get_introns(blocks) + + # Determine threshold for read anchor lengths + threshold = get_threshold(chr, introns, genome, annoSJ, minOverhang, minOverhangC, minOverhangNC) + + # Compute anchor lengths + leftAnchorLen = blocks[0][1] - blocks[0][0] + rightAnchorLen = blocks[len(introns)][1] - blocks[len(introns)][0] + + # Only keep reads that satisfy appropriate anchor length + # threshold + if min(leftAnchorLen, rightAnchorLen) >= threshold: + # Iterate through introns and check if SJ is canonical + for i in range(len(introns)): + # Construct splice junction ID + sjInfo = [chr, introns[i][0], introns[i][1]] + sjID = ':'.join(map(str, sjInfo)) + + # Update sjDB with SJ + # Check if SJ exists in dictionary + if sjID in sjDB: + # Increment current read count by 1 + sjDB[sjID] += 1 + else: + # Add sjID to sjDB with one read + sjDB[sjID] = 1 + + # Return the updated dictionary + return sjDB + +def get_transcript_ID(infoString): + infoStringArr = infoString.split(';') + # Get position in the INFO string that has the transcript ID + idx = [i for i, s in enumerate(infoStringArr) if 'transcript_id' in s][0] + return infoStringArr[idx].split('"')[1] + +def build_anno_SJdb(gtfPath, chrList): + annoSJ = {} + + # Create dictionary object to store exons of a transcript + exons = {} + # Create dictionary object to store chr of a transcript + chrDict = {} + + # Read through every line of the gtfPath + with open(gtfPath) as f: + for line in f: + if not line.startswith('#'): + info = line.strip().split('\t') + if info[0] in chrList and info[2] == 'exon': + exonStart = int(info[3]) + exonEnd = int(info[4]) + transcriptID = get_transcript_ID(info[8]) + # Check if transcript is included in exons + if transcriptID in exons: + exons[transcriptID] += [(exonStart, exonEnd)] + else: + exons[transcriptID] = [(exonStart, exonEnd)] + + # Check if transcript is included in chrDict + if transcriptID not in chrDict: + chrDict[transcriptID] = info[0] + + # Extract splice junctions from exons + for transcriptID in exons: + chr = chrDict[transcriptID] + + # Sort the tuple of exons + exonList = sorted(exons[transcriptID]) + + # Build annoSJ + for i in range(len(exonList)-1): + sjID = ':'.join(map(str,[chr, exonList[i][1]+1, exonList[i+1][0]-1])) + if sjID not in annoSJ: + annoSJ[sjID] = 0 + + return annoSJ + +def write_output(sjDB, outfile): + f = open(outfile, 'w') + + # Iterate through all sjIDs in sjDB + for sjID in list(sjDB.keys()): + f.write('\t'.join(map(str,[sjID, sjDB[sjID]]))+'\n') + + f.close() + +def check_cigar(cigarString): + # Only keep read if CIGAR string only contains 'N' and 'M' + return set([i for i in cigarString if i.isalpha()]) == set(['N', 'M']) + +def main(args): + + # Parse command-line arguments + bamPath, gtfPath, fastaPath, outfile = args.bam_path, args.gtf, args.genome_fasta, args.outdir + minOverhang, minOverhangC, minOverhangNC, filterRL = int(args.minimum_overhang_length_annotated), int(args.minimum_overhang_length_unannotated_canonical), int(args.minimum_overhang_length_unannotated_noncanonical), int(args.read_length) + + if os.path.exists(bamPath+'.bai') == False: + pysam.index(bamPath) + # Create list of target chromosomes + chrList = ['chrX', 'chrY'] + for i in range(22): + chrList.append('chr' + str(i+1)) + + # Initialize splice junction database (dictionary object) + sjDB = {} + + # Create dictionary object to store all annotated SJs in the GTF + annoSJ = build_anno_SJdb(gtfPath, chrList) + + # Open the BAM file as a SAM + samfile = pysam.AlignmentFile(bamPath, 'rb') + + # Open FASTA file + genome = pysam.Fastafile(fastaPath) + + # Iterate through the reads in the SAM file and update the + # sjDB object + for read in samfile.fetch(): + # Check if read satisfies the following: + # * is proper pair + # * is uniquely mapped by STAR + # * maps to chromosomes 1-22, X, Y + # * CIGAR string only contains M and N + if (read.is_proper_pair + and read.mapping_quality == 255 + and read.reference_name in chrList + and check_cigar(read.cigarstring)): + # Check if read has the specified read length + # if provided + if filterRL == -1: + # Update sjDB with given read + sjDB = update_SJdb(read, sjDB, annoSJ, genome, + minOverhang, minOverhangC, minOverhangNC) + else: + if read.query_length == filterRL: + # Query read has expected length + sjDB = update_SJdb(read, sjDB, annoSJ, genome, + minOverhang, minOverhangC, minOverhangNC) + + # Print out sjDB to outfile + write_output(sjDB, outfile) + +if __name__ == '__main__': + main() diff --git a/IRIS/IRIS_formatting.py b/IRIS/IRIS_formatting.py index 408378a..feebfe2 100644 --- a/IRIS/IRIS_formatting.py +++ b/IRIS/IRIS_formatting.py @@ -5,7 +5,7 @@ def loadSamplelist(fin_samples, sample_fin_list, sample_header, sample_name_fiel ls=l.strip() sample_fin_list.append(ls) for r in open(ls): - rs=map(lambda x:x.split('/')[-sample_name_field].split('.bam')[0],r.strip().strip(',').split(',')) + rs=map(lambda x:x.split('/')[-sample_name_field].split('.bam')[0].split('.aln')[0],r.strip().strip(',').split(',')) #rs=map(lambda x:x.split('/')[-2],r.strip().strip(',').split(',')) if sample_name_field==2: sn_list=r.strip().strip(',').split(',') @@ -16,30 +16,96 @@ def loadSamplelist(fin_samples, sample_fin_list, sample_header, sample_name_fiel sample_size[ls]=len(r.split(',')) return sample_fin_list, sample_header, sample_size -def mergeEvents(events_fin_list): +def parseEventRowSE(line_split): + return line_split[1].strip('"')+'\t'+line_split[2].strip('"')+'\t'+'\t'.join(line_split[3:7]+line_split[8:10]) + +def parseEventRow(line_split): + return line_split[1].strip('"')+'\t'+line_split[2].strip('"')+'\t'+'\t'.join(line_split[3:11]) + +def loadGTF(gtf): + exon_start_dict={} + exon_end_dict={} + for l in open(gtf): + if l.startswith('#'): + continue + ls=l.strip().split('\t') + if ls[2]=='exon': + chrom=ls[0] + if chrom.startswith('chr')==False: + chrom='chr'+chrom + exon_start_dict[ls[6]+':'+chrom+':'+ls[3]]='' + exon_end_dict[ls[6]+':'+chrom+':'+ls[4]]='' + return exon_start_dict, exon_end_dict + +def checkNovelSS(head, ls, splicing_event_type, exon_start_dict, exon_end_dict):# This is a conservative def of novelSS than rMATS4.1 (0.4% events less- complex cases ) + ld=dict(zip(head, ls)) + strand=ld['strand'] + chrom=ld['chr'] + if splicing_event_type=='SE': + check1=strand+':'+chrom+':'+str(int(ld['exonStart_0base'])+1) not in exon_start_dict + check2=strand+':'+chrom+':'+str(int(ld['downstreamES'])+1) not in exon_start_dict + check3=strand+':'+chrom+':'+ld['exonEnd'] not in exon_end_dict + check4=strand+':'+chrom+':'+ld['upstreamEE'] not in exon_end_dict + elif splicing_event_type=='A5SS': + check1=strand+':'+chrom+':'+str(int(ld['shortES'])+1) not in exon_start_dict + check2=strand+':'+chrom+':'+str(int(ld['flankingES'])+1) not in exon_start_dict + check3=strand+':'+chrom+':'+ld['longExonEnd'] not in exon_end_dict + check4=False + #check4=strand+':'+chrom+':'+str(int(ld['shortES'])+1) not in exon_end_dict + elif splicing_event_type=='A3SS': + check1=strand+':'+chrom+':'+str(int(ld['longExonStart_0base'])+1) not in exon_start_dict + check2=strand+':'+chrom+':'+str(int(ld['shortES'])+1) not in exon_start_dict + check3=strand+':'+chrom+':'+ld['flankingEE'] not in exon_end_dict + check4=False + + elif splicing_event_type=='RI': + check1,check2,check3,check4=[False,False,False,False] + else: + exit('choose AS type.') + return check1, check2, check3, check4 + + +def mergeEvents(events_fin_list, splicing_event_type, novelSS, exon_start_dict, exon_end_dict): + parseRow=parseEventRow + if splicing_event_type=='SE': + parseRow=parseEventRowSE total_event_dict={} - for events_fin in events_fin_list: + for i, events_fin in enumerate(events_fin_list): + head=[] for index,event_l in enumerate(open(events_fin)): if index==0: + head=event_l.strip().split('\t') continue event_ls=event_l.strip().split('\t') - events_cord=event_ls[1].strip('"')+'\t'+event_ls[2].strip('"')+'\t'+'\t'.join(event_ls[3:7]+event_ls[8:10]) + if novelSS: + check1, check2, check3, check4= checkNovelSS(head, event_ls, splicing_event_type, exon_start_dict, exon_end_dict) + novel=True if check1 or check2 or check3 or check4 else False + if novel==False: # if no novel, will not parse the row and save + continue + events_cord=parseRow(event_ls) if events_cord in total_event_dict: continue total_event_dict[events_cord]='' return total_event_dict -def writeMergedEvents(events_fin_list, splicing_event_type, cov_cutoff, data_name, fout_path): - total_event_dict=mergeEvents(events_fin_list) - print len(total_event_dict) +def writeMergedEvents(events_fin_list, splicing_event_type, cov_cutoff, data_name, fout_path, novelSS, exon_start_dict, exon_end_dict): + total_event_dict=mergeEvents(events_fin_list, splicing_event_type, novelSS, exon_start_dict, exon_end_dict) + novelss_tag='' + if novelSS: + novelss_tag='.novelSS' total_event_list=sorted(total_event_dict.keys()) - fout=open(fout_path+'/prefilter_events.splicing_matrix.'+splicing_event_type+'.cov'+str(cov_cutoff)+'.'+data_name+'.txt','w') + fout=open(fout_path+'/prefilter_events.splicing_matrix.'+splicing_event_type+novelss_tag+'.cov'+str(cov_cutoff)+'.'+data_name+'.txt','w') for e in total_event_list: fout.write(e.strip()+'\n') fout.close() + return total_event_list -def mergeMatrixInBatch(fin_list, events_fin_list, sample_fin_list, cov_cutoff, data_name, splicing_event_type, sample_header, sample_size, total_event_list, file_batch_list, batch, fout_path): +def mergeMatrixInBatch(fin_list, events_fin_list, sample_fin_list, cov_cutoff, data_name, splicing_event_type, sample_header, sample_size, total_event_list, file_batch_list, batch, fout_path, individual_filter, novelSS): + parseRow=parseEventRow + if splicing_event_type=='SE': + parseRow=parseEventRowSE + for b in range(0,len(total_event_list),batch): Intercep_Matrix={} print '[INFO] Merging in progress. Working on batch ',b @@ -51,7 +117,7 @@ def mergeMatrixInBatch(fin_list, events_fin_list, sample_fin_list, cov_cutoff, d if index==0: continue event_ls=event_l.strip().split('\t') - event_cord=event_ls[1].strip('"')+'\t'+event_ls[2].strip('"')+'\t'+'\t'.join(event_ls[3:7]+event_ls[8:10]) + event_cord=parseRow(event_ls) if event_cord in batch_event_dict: eventID[event_ls[0]]=event_cord print '[INFO] Merging file: ', fin, len(eventID) @@ -66,10 +132,16 @@ def mergeMatrixInBatch(fin_list, events_fin_list, sample_fin_list, cov_cutoff, d Cov=[num+Skip[o] for o,num in enumerate(Incl)] psi_values=[] for i,I in enumerate(Incl): - if int(I)+int(Skip[i])==0: - psi_values.append('NaN') + if individual_filter: # individual_filter. Use Cov[i] for each individual sample + if Cov[i]< cov_cutoff: + psi_values.append('NaN') + else: + psi_values.append(str(round(I/int(rs[5])/(I/int(rs[5])+Skip[i]/int(rs[6])),4))) else: - psi_values.append(str(round(I/int(rs[5])/(I/int(rs[5])+Skip[i]/int(rs[6])),4))) + if int(I)+int(Skip[i])==0: + psi_values.append('NaN') + else: + psi_values.append(str(round(I/int(rs[5])/(I/int(rs[5])+Skip[i]/int(rs[6])),4))) if eventID[rs[0]] not in Intercep_Matrix: Intercep_Matrix[eventID[rs[0]]]={} @@ -78,9 +150,20 @@ def mergeMatrixInBatch(fin_list, events_fin_list, sample_fin_list, cov_cutoff, d if len(psi_values)!=sample_size[sample_fin_list[n]]: exit('[Abort] Sample number does not match observations in JC file.') - file_batch_list.append(fout_path+'/splicing_matrix/splicing_matrix.'+splicing_event_type+'.cov'+str(cov_cutoff)+'.'+data_name+'.txt.batch_'+str(b)+'.txt') - fout=open(fout_path+'/splicing_matrix/splicing_matrix.'+splicing_event_type+'.cov'+str(cov_cutoff)+'.'+data_name+'.txt.batch_'+str(b)+'.txt','w') - fout.write('AC\tGeneName\tchr\tstrand\texonStart\texonEnd\tupstreamEE\tdownstreamES\t'+'\t'.join(sample_header)+'\n') + novelss_tag='' + if novelSS: + novelss_tag='.novelSS' + file_path_name=fout_path+'/splicing_matrix/splicing_matrix.'+splicing_event_type+novelss_tag+'.cov'+str(cov_cutoff)+'.'+data_name+'.txt.batch_'+str(b)+'.txt' + file_batch_list.append(file_path_name) + fout=open(file_path_name,'w') + header_line='AC\tGeneName\tchr\tstrand\texonStart\texonEnd\tupstreamEE\tdownstreamES\t'+'\t'.join(sample_header) + if splicing_event_type=='A5SS': + header_line='AC\tGeneName\tchr\tstrand\tlongExonStart\tlongExonEnd\tshortES\tshortEE\tflankingES\tflankingEE\t'+'\t'.join(sample_header) + if splicing_event_type=='A3SS': + header_line='AC\tGeneName\tchr\tstrand\tlongExonStart\tlongExonEnd\tshortES\tshortEE\tflankingES\tflankingEE\t'+'\t'.join(sample_header) + if splicing_event_type=='RI': + header_line='AC\tGeneName\tchr\tstrand\triExonStart\triExonEnd\tupstreamES\tupstreamEE\tdownstreamES\tdownstreamEE\t'+'\t'.join(sample_header) + fout.write(header_line+'\n') for k in sorted(Intercep_Matrix.keys()): psi_value_all=[] cov_all=[] @@ -90,15 +173,22 @@ def mergeMatrixInBatch(fin_list, events_fin_list, sample_fin_list, cov_cutoff, d cov_all+=Intercep_Matrix[k][sample][1] else: psi_value_all+=['NaN']*sample_size[sample] + if individual_filter==False: #if filter by group and cov < cov_cutoff, skip this event K. Otherwise, psi_vallue_all is being wrote to output. + mean=numpy.mean(cov_all) + if mean < cov_cutoff: + continue + if set(psi_value_all)==set(['NaN']): #remove full NaN events 2020 + continue + fout.write(k+'\t'+'\t'.join(psi_value_all)+'\n') - mean=numpy.mean(cov_all) - if mean>=cov_cutoff: - fout.write(k+'\t'+'\t'.join(psi_value_all)+'\n') fout.close() return file_batch_list -def mergeMatrixInOne(file_batch_list, cov_cutoff, data_name, splicing_event_type, fout_path): - fout_merge=open(fout_path+'/splicing_matrix/splicing_matrix.'+splicing_event_type+'.cov'+str(cov_cutoff)+'.'+data_name+'.txt','w') +def mergeMatrixInOne(file_batch_list, cov_cutoff, data_name, splicing_event_type, fout_path, novelSS): + novelss_tag='' + if novelSS: + novelss_tag='.novelSS' + fout_merge=open(fout_path+'/splicing_matrix/splicing_matrix.'+splicing_event_type+novelss_tag+'.cov'+str(cov_cutoff)+'.'+data_name+'.txt','w') header=0 for file_batch in file_batch_list: for j,l in enumerate(open(file_batch)): @@ -109,19 +199,22 @@ def mergeMatrixInOne(file_batch_list, cov_cutoff, data_name, splicing_event_type continue fout_merge.write(l) fout_merge.close() - os.system('rm '+fout_path+'/splicing_matrix/splicing_matrix.'+splicing_event_type+'.cov'+str(cov_cutoff)+'.'+data_name+'.txt.batch_*.txt') - return 'splicing_matrix.'+splicing_event_type+'.cov'+str(cov_cutoff)+'.'+data_name+'.txt' + os.system('rm '+fout_path+'/splicing_matrix/splicing_matrix.'+splicing_event_type+novelss_tag+'.cov'+str(cov_cutoff)+'.'+data_name+'.txt.batch_*.txt') + return 'splicing_matrix.'+splicing_event_type+novelss_tag+'.cov'+str(cov_cutoff)+'.'+data_name+'.txt' -def index_PsiMatrix(fn,outdir,delim): +def index_PsiMatrix(fn, outdir, delim, splicing_event_type): out_fp = outdir+'/'+fn.split('/')[-1]+'.idx' line_formatter = "{id}\t{offset}\n" - offset = 0 + offset = 0 + col_index=10 + if splicing_event_type=='SE':#handle SE and other types of AS events + col_index=8 with open(fn, 'r') as fin: with open(out_fp, 'w') as fout: offset += len(fin.readline()) for line in fin: ele = line.strip().split(delim) - eid = ':'.join([ele[0].split('_')[0].split('.')[0]]+ele[1:8]) + eid = ':'.join([ele[0].split('_')[0].split('.')[0]]+ele[1:col_index]) fout.write( line_formatter.format(id=eid, offset=offset) ) offset += len(line) return @@ -131,6 +224,14 @@ def main(args): data_name=args.data_name sample_name_field=args.sample_name_field splicing_event_type=args.splicing_event_type + individual_filter= args.sample_based_filter + novelSS= args.novelSS + exon_start_dict={} + exon_end_dict={} + if novelSS: + gtf=args.gtf + exon_start_dict, exon_end_dict= loadGTF(gtf) + if sample_name_field==1: print '[INFO] Sample name parsed from bam file. (alternatively can be parsed from up level folder)' if sample_name_field==2: @@ -154,21 +255,21 @@ def main(args): sample_fin_list, sample_header, sample_size= loadSamplelist(args.rmats_sample_order,sample_fin_list, sample_header,sample_name_field, sample_size) #MAKING MERGED EVENTS LIST - total_event_list= writeMergedEvents(events_fin_list, splicing_event_type, cov_cutoff, data_name, fout_path) + total_event_list= writeMergedEvents(events_fin_list, splicing_event_type, cov_cutoff, data_name, fout_path, novelSS, exon_start_dict, exon_end_dict) if args.merge_events_only: exit('[INFO] Done merging events only.') - print '[INFO] Done loading file dir', len(total_event_list) + print '[INFO] Done loading file dir. Total events:', len(total_event_list) #START MERGING MATRICES IN BATCH MODE FOLLOWING EVENTS LIST GENERATED. batch=20000 - file_batch_list=mergeMatrixInBatch(fin_list, events_fin_list, sample_fin_list, cov_cutoff, data_name, splicing_event_type, sample_header, sample_size, total_event_list, file_batch_list, batch, fout_path) + file_batch_list=mergeMatrixInBatch(fin_list, events_fin_list, sample_fin_list, cov_cutoff, data_name, splicing_event_type, sample_header, sample_size, total_event_list, file_batch_list, batch, fout_path, individual_filter, novelSS) print '[INFO] Done merging matrices by batch.' - merged_file_name=mergeMatrixInOne(file_batch_list, cov_cutoff, data_name, splicing_event_type, fout_path) + merged_file_name=mergeMatrixInOne(file_batch_list, cov_cutoff, data_name, splicing_event_type, fout_path, novelSS) print '[INFO] Done merging matrices: '+merged_file_name #create index in IRIS db directory - index_PsiMatrix(fout_path+'/splicing_matrix/'+merged_file_name,fout_path+'/splicing_matrix','\t') + index_PsiMatrix(fout_path+'/splicing_matrix/'+merged_file_name,fout_path+'/splicing_matrix','\t', splicing_event_type) print '[INFO] Finished. Created matrix: '+fout_path if __name__ == '__main__': diff --git a/IRIS/IRIS_indexing.py b/IRIS/IRIS_indexing.py index a2a64bc..cdfc879 100644 --- a/IRIS/IRIS_indexing.py +++ b/IRIS/IRIS_indexing.py @@ -3,32 +3,32 @@ from scipy import stats import statsmodels.stats.weightstats as smw -def index_PsiMatrix(fn,outdir,delim): - out_fp = outdir+'/'+fn.split('/')[-1]+'.idx' +def index_PsiMatrix(fn, outdir, delim, splicing_event_type, out_fp): line_formatter = "{id}\t{offset}\n" offset = 0 + col_index=10 + if splicing_event_type=='SE':#handle SE and other types of AS events + col_index=8 with open(fn, 'r') as fin: with open(out_fp, 'w') as fout: offset += len(fin.readline()) for line in fin: ele = line.strip().split(delim) - eid = ':'.join([ele[0].split('_')[0].split('.')[0]]+ele[1:8]) + eid = ':'.join([ele[0].split('_')[0].split('.')[0]]+ele[1:col_index]) fout.write( line_formatter.format(id=eid, offset=offset) ) offset += len(line) return def main(args): fin=args.splicing_matrix + splicing_event_type=args.splicing_event_type + cov_cutoff=args.cov_cutoff data_name=args.data_name - db_dir=args.db_dir.rstrip('/') - - #prepare files/folders in IRIS db directory - os.system('mkdir -p '+db_dir+'/'+data_name+' '+db_dir+'/'+data_name+'/splicing_matrix') - new_dir_fin=db_dir+'/'+data_name+'/splicing_matrix/splicing_matrix.SE.cov10.'+data_name+'.txt' - os.system('mv '+fin+' '+new_dir_fin) - #create index in IRIS db directory - index_PsiMatrix(new_dir_fin,db_dir+'/'+data_name+'/splicing_matrix','\t') - print '[INFO] Finished. Created matrix: '+new_dir_fin + outdir=args.outdir.rstrip('/') + out_fp = outdir+'/'+fin.split('/')[-1]+'.idx' + #create index in the current directory + index_PsiMatrix(fin,outdir,'\t',splicing_event_type, out_fp) + print '[INFO] Finished. Created matrix: '+out_fp if __name__ == '__main__': main() diff --git a/IRIS/IRIS_makeqsub_mapping.py b/IRIS/IRIS_makeqsub_mapping.py deleted file mode 100644 index e740129..0000000 --- a/IRIS/IRIS_makeqsub_mapping.py +++ /dev/null @@ -1,69 +0,0 @@ -import sys,glob,os -from . import config -#python make.shellsubmiter.py Out_dir Fastq_dir -# two inputs: 1)output folder/sh name/prefix 2)dir of all RNAseq files (if more than 2, recognize and sort by R1 R2 for star input) -def makeSubmit(out_prefix, fin, label_string, starGenomeDir, gtf): - fq_dir=fin - fqs=glob.glob(fq_dir+'/*') - #abs_path=os.path.abspath(sys.argv[2]) - r1=[] - r2=[] - for fq in fqs: - if fq.find('1'+label_string+'f')!=-1: - r1.append(os.path.abspath(fq)) - elif fq.find('2'+label_string+'f')!=-1: - r2.append(os.path.abspath(fq)) - if len(r1)!=len(r2) or len(r1)==0 or len(r2)==0: - print 'file name can not be recognize' - return '','' - out_dir=out_prefix.rstrip('/')+'.aln' - sample_name=out_dir.split('/')[-1].split('.')[0] - fout1=open('submit.STARmap.'+sample_name+'.sh','w') - fout2=open('submit.Cuffquant.'+sample_name+'.sh','w') - fq_path=','.join(sorted(r1)+sorted(r2)) - cmd1='IRIS process_rnaseq --starGenomeDir '+starGenomeDir+' --gtf '+gtf+' --mapping --sort -p '+out_dir+' '+fq_path - fout1.write(cmd1+'\n') - cmd2='IRIS process_rnaseq --starGenomeDir '+starGenomeDir+' --gtf '+gtf+' --quant -p '+out_dir+' '+fq_path - fout2.write(cmd2+'\n') - return 'submit.STARmap.'+sample_name+'.sh','submit.Cuffquant.'+sample_name+'.sh' - -def main(args): - starGenomeDir=args.starGenomeDir - gtf=args.gtf - fastq_folder_dir=args.fastq_folder_dir - fastq_folder_list=glob.glob(fastq_folder_dir+'') - out_dir=args.out_dir.rstrip('/') - task_name=args.data_name - label_string=args.label_string - os.system('mkdir -p '+out_dir) - - fout1=open('cmdlist.STAR.'+task_name,'w') - fout2=open('cmdlist.Cufflinks.'+task_name,'w') - i=0 - for folder in fastq_folder_list: - print folder - fn1,fn2=makeSubmit(out_dir+'/'+folder.split('/')[-1], folder, label_string, starGenomeDir) - if fn1=='': - continue - i+=1 - fout1.write(fn1+'\n') - fout2.write(fn2+'\n') - fout1.close() - fout2.close() - - fout_qsub1=open('qsub.STARmapping.'+task_name+'.sh','w') - cmd='qsub -t 1-'+str(i)+':1 qsub.STARmapping.'+task_name+'.sh' - fout_qsub1.write('#!/bin/bash\n#$ -N STARmapping\n#$ -S /bin/bash\n#$ -R y\n#$ -l '+config.QSUB_ALIGNMENT_CONFIG+'\n#$ -V\n#$ -cwd\n#$ -j y\n#$ -m bea\n') - fout_qsub1.write('export s=`sed -n ${SGE_TASK_ID}p '+'cmdlist.STAR.'+task_name+'`\necho $s\nbash $s') - fout_qsub1.close() - print cmd - - fout_qsub2=open('qsub.Cufflinks.'+task_name+'.sh','w') - cmd='qsub -t 1-'+str(i)+':1 qsub.Cufflinks.'+task_name+'.sh' - fout_qsub2.write('#!/bin/bash\n#$ -N Cufflinks\n#$ -S /bin/bash\n#$ -R y\n#$ -l '+config.QSUB_EXPRESSION_CONFIG+'\n#$ -V\n#$ -cwd\n#$ -j y\n#$ -m bea\n') - fout_qsub2.write('export s=`sed -n ${SGE_TASK_ID}p '+'cmdlist.Cufflinks.'+task_name+'`\necho $s\nnbash $s') - fout_qsub2.close() - print cmd - -if __name__ == '__main__': - main() \ No newline at end of file diff --git a/IRIS/IRIS_makeqsub_rmats.py b/IRIS/IRIS_makeqsub_rmats.py deleted file mode 100644 index 76f91c8..0000000 --- a/IRIS/IRIS_makeqsub_rmats.py +++ /dev/null @@ -1,32 +0,0 @@ -import sys, csv, glob, os -from . import config -def main(args): - - bam_dir= args.bam_dir - file_list=glob.glob(bam_dir.rstrip('/')+'/*/*.bam') - length=str(args.read_length) - gtf=args.gtf - task_name=args.data_name - rMATS_path=args.rMATS_path#/u/home/s/shiehshi/rMATS-2017-3-15/rmats.py - list_name='cmdlist.rMATS_prep_'+task_name - - i=0 - fout=open(list_name,'w') - for fin in file_list: - fin_name='/'.join(fin.split('/')[:-1]).rstrip('/') - i+=1 - fout_local=open(fin_name+'/bam_list.txt','w') - fout_local.write(fin) - fout_local.close() - fout.write('python '+args.rMATS_path+' --b1 '+fin_name+'/bam_list.txt --od '+fin_name+' --tmp '+fin_name+'.tmp --anchorLength 1 --readLength '+length+' --gtf '+gtf+' -t paired --task prep --nthread 8 --statoff\n') - fout.close() - - fout_qsub=open('qsub.rMATSturboPrep.'+task_name+'.sh','w') - cmd='qsub -t 1-'+str(i)+':1 qsub.rMATSturboPrep.'+task_name+'.sh' - fout_qsub.write('#!/bin/bash\n#$ -N rmats_prep\n#$ -S /bin/bash\n#$ -R y\n#$ -l '+config.QSUB_RMATS_PREP_CONFIG+'\n#$ -V\n#$ -cwd\n#$ -j y\n#$ -m bea\n') - fout_qsub.write('export s=`sed -n ${SGE_TASK_ID}p '+list_name+'`\necho $s\n$s') - fout_qsub.close() - print cmd - -if __name__ == '__main__': - main() diff --git a/IRIS/IRIS_makesubsh_extractsj.py b/IRIS/IRIS_makesubsh_extractsj.py new file mode 100644 index 0000000..84851de --- /dev/null +++ b/IRIS/IRIS_makesubsh_extractsj.py @@ -0,0 +1,55 @@ +import sys, csv, glob, os, argparse + +def parseMappingLog(log_fin): + read_length='' + for l in open(log_fin): + if l.find('Average input read length')!=-1: + ls=l.strip().split('|') + read_length=ls[1].strip().strip(' ') + if read_length=='': + exit('can not find log file for read length') + return str(int(round(int(read_length)/2))) + +def main(args): + #extractSJ_path=args.extractSJ_path + BAM_prefix=args.BAM_prefix + bam_folder_list=args.bam_folder_list + rl=args.rmats_used_read_length + parserl=False + if rl=='': + parserl=True + print('checking read legnth') + else: + print('use user specified read length') + gtf=args.gtf + task_name= args.task_name + genome_fasta= args.genome_fasta + task_dir = args.task_dir + if not task_dir: + task_dir = os.getcwd() + + list_name='cmdlist.extract_sjc.'+task_name + list_name = os.path.join(task_dir, list_name) + + n=0 + fout=open(list_name,'w') + for bam_folder in open(bam_folder_list): + n+=1 + bam_folder=bam_folder.strip() + if parserl: + rl=parseMappingLog(bam_folder+'/Log.final.out') + fout.write('IRIS extract_sjc -i '+bam_folder+'/'+BAM_prefix+'.bam -g '+gtf+' -r '+rl+' -f '+genome_fasta+' -o '+bam_folder+'/SJcount.txt \n') + fout.close() + + sh_file_name = 'subsh.extract_sjc.{}.sh'.format(task_name) + sh_file_name = os.path.join(task_dir, sh_file_name) + fout_qsub=open(sh_file_name,'w') + + cmd='sbatch --array=1-{} {}'.format(str(n), sh_file_name) + + fout_qsub.write('#!/bin/bash\n#SBATCH --job-name=extract_sjc\n#SBATCH --mem=5G\n#SBATCH -t 15:00:00\n') + fout_qsub.write('export s=`sed -n ${SLURM_ARRAY_TASK_ID}p '+list_name+'`\necho $s\n$s') + fout_qsub.close() + print(cmd) +if __name__ == '__main__': + main() diff --git a/IRIS/IRIS_makesubsh_hla.py b/IRIS/IRIS_makesubsh_hla.py new file mode 100644 index 0000000..d8b0aee --- /dev/null +++ b/IRIS/IRIS_makesubsh_hla.py @@ -0,0 +1,58 @@ +import sys,glob,os +from . import config + + +def write_task_script(out_prefix, fin, label_string, task_dir): + fq_dir=fin + fqs=glob.glob(fq_dir+'/*') + r1=[] + r2=[] + for fq in fqs: + if fq.find('1'+label_string+'f')!=-1: + r1.append(os.path.abspath(fq)) + elif fq.find('2'+label_string+'f')!=-1: + r2.append(os.path.abspath(fq)) + if len(r1)!=len(r2) or len(r1)==0 or len(r2)==0: + print '[Error] File name can not be recognized' + return + + out_dir=out_prefix.rstrip('/') + os.system('mkdir -p '+out_dir) + sample_name=out_dir.split('/')[-1].split('.')[0] + task_script_base = 'seq2hla.{}.sh'.format(sample_name) + task_script = os.path.join(task_dir, task_script_base) + fout=open(task_script,'w') + suffix=r1[0].split(label_string)[1] + path='/'.join(r1[0].split('/')[:-1]) + fq_path1=' '.join(sorted(r1)) + cmd_fq1='cat '+fq_path1+' > '+path+'/fq1.'+suffix + fout.write('#!/bin/bash\n'+cmd_fq1+'\n') + + fq_path2=' '.join(sorted(r2)) + cmd_fq2='cat '+fq_path2+' > '+path+'/fq2.'+suffix + fout.write(cmd_fq2+'\n') + + cmd1='seq2HLA -1 '+path+'/fq1.'+suffix+' -2 '+path+'/fq2.'+suffix+' -r '+out_dir+'/'+sample_name+' > '+out_dir+'/seq2hla.log 2>&1' + fout.write(cmd1+'\n') + fout.write('rm '+path+'/fq1.'+suffix+'\nrm '+path+'/fq2.'+suffix+'\n') + fout.close() + + +def main(args): + fastq_folder_dir=args.fastq_folder_dir.rstrip('/') + fastq_folder_list=glob.glob(fastq_folder_dir+'/*') + out_dir=args.outdir.rstrip('/') + task_name=args.data_name + label_string=args.label_string + os.system('mkdir -p '+out_dir) + task_dir=args.task_dir + if not os.path.exists(task_dir): + os.makedirs(task_dir) + + for folder in fastq_folder_list: + print folder + write_task_script(out_dir+'/'+folder.split('/')[-1], folder, label_string, task_dir) + + +if __name__ == '__main__': + main() diff --git a/IRIS/IRIS_makesubsh_mapping.py b/IRIS/IRIS_makesubsh_mapping.py new file mode 100644 index 0000000..1345fac --- /dev/null +++ b/IRIS/IRIS_makesubsh_mapping.py @@ -0,0 +1,55 @@ +import sys,glob,os +from . import config + +def write_task_script(out_prefix, fin, label_string, starGenomeDir, gtf, task_dir): + fq_dir=fin + fqs=glob.glob(fq_dir+'/*') + r1=[] + r2=[] + for fq in fqs: + if fq.find('1'+label_string+'f')!=-1: + r1.append(os.path.abspath(fq)) + elif fq.find('2'+label_string+'f')!=-1: + r2.append(os.path.abspath(fq)) + if len(r1)!=len(r2) or len(r1)==0 or len(r2)==0: + print 'file name can not be recognize' + return '','' + + out_dir=out_prefix.rstrip('/')+'.aln' + sample_name=out_dir.split('/')[-1].split('.')[0] + task_script_base1 = 'STARmap.{}.sh'.format(sample_name) + task_script1 = os.path.join(task_dir, task_script_base1) + task_script_base2 = 'Cuffquant.{}.sh'.format(sample_name) + task_script2 = os.path.join(task_dir, task_script_base2) + fout1=open(task_script1,'w') + fout2=open(task_script2,'w') + fq_path=','.join(sorted(r1)+sorted(r2)) + + cmd1='IRIS process_rnaseq --starGenomeDir '+starGenomeDir+' --gtf '+gtf+' --mapping --sort -p '+out_dir+' '+fq_path + fout1.write('#!/bin/bash\n'+cmd1+'\n') + cmd2='IRIS process_rnaseq --starGenomeDir '+starGenomeDir+' --gtf '+gtf+' --quant -p '+out_dir+' '+fq_path + fout2.write('#!/bin/bash\n'+cmd2+'\n') + return task_script1,task_script2 + +def main(args): + starGenomeDir=args.starGenomeDir + gtf=args.gtf + fastq_folder_dir=args.fastq_folder_dir.rstrip('/') + fastq_folder_list=glob.glob(fastq_folder_dir+'/*') + out_dir=args.outdir.rstrip('/') + task_name=args.data_name + label_string=args.label_string + os.system('mkdir -p '+out_dir) + task_dir=args.task_dir + if not os.path.exists(task_dir): + os.makedirs(task_dir) + + for folder in fastq_folder_list: + print folder + fn1,fn2=write_task_script(out_dir+'/'+folder.split('/')[-1], folder, label_string, starGenomeDir, gtf, task_dir) + if fn1=='': + continue + + +if __name__ == '__main__': + main() diff --git a/IRIS/IRIS_makesubsh_rmats.py b/IRIS/IRIS_makesubsh_rmats.py new file mode 100644 index 0000000..f4c9e4a --- /dev/null +++ b/IRIS/IRIS_makesubsh_rmats.py @@ -0,0 +1,66 @@ +import sys, csv, glob, os +from . import config + +def writeShell(rMATS_path, fin_name, folder_name, bam_dir, read_length_argument, gtf, novelSS, task_name, task_dir): + fout_local=open(folder_name+'/bam_list.txt','w') + fout_local.write(fin_name) + fout_local.close() + + sample_name=folder_name.split('/')[-1].split('.')[0] + task_script_base = 'rMATS_prep.{}.sh'.format(sample_name) + task_script = os.path.join(task_dir, task_script_base) + fout=open(task_script,'w') + fout.write('#!/bin/bash\n') + novelSS_str='' + if novelSS: + novelSS_str='--novelSS ' + # TODO the '|| true' at the end of this command ignores a + # failure return code from the python command. + # rMATS produces the desired output file despite the error return. + # A future version of rMATS may fix this behavior. + fout.write('python {} --b1 {}/bam_list.txt --od {} --tmp {}/{}.RL{}/{}.tmp --anchorLength 1 --readLength {} --gtf {} -t paired --task prep --nthread 8 --statoff {}|| true\n'.format(rMATS_path, folder_name, folder_name, bam_dir, task_name, read_length_argument, sample_name, read_length_argument, gtf, novelSS_str)) + fout.close() + +def organizeReadLength(rMATS_path, file_list_mapping, gtf, novelSS, bam_prefix, task_name, task_dir): + rl_dict={} + folder_names={} + for fin_name in file_list_mapping: + for l in open(fin_name): + if l.find('Average input read length |')!=-1: + map_rl=int(round(float(l.split('Average input read length |')[-1].strip())/2,0)) + rl_dict['/'.join(fin_name.split('/')[:-1])]=map_rl + folder_names[map_rl]='' + break + bam_dir='/'.join(file_list_mapping[0].split('/')[:-2]) + for folder_name in folder_names: + os.system('mkdir -p '+bam_dir+'/'+task_name+'.RL'+str(folder_name)) + for folder_name in rl_dict: + writeShell(rMATS_path, folder_name+'/'+bam_prefix+'.bam', folder_name, bam_dir, str(rl_dict[folder_name]), gtf, novelSS, task_name, task_dir) + +def main(args): + gtf=args.gtf + task_name=args.data_name + rMATS_path=args.rMATS_path + bam_dir= args.bam_dir.rstrip('/') + bam_prefix=args.bam_prefix + novelSS=args.novelSS + task_dir=args.task_dir + if not os.path.exists(task_dir): + os.makedirs(task_dir) + if args.read_length: + read_length= int(args.read_length) + + print 'preparing rMATS-turbo prep directories' + if args.read_length==False: + mapping_log_file_list=glob.glob(bam_dir+'/*/Log.final.out') + organizeReadLength(rMATS_path, mapping_log_file_list, gtf, novelSS, bam_prefix, task_name, task_dir) #relocated based on the read length + else: + mapping_bam_list=glob.glob(bam_dir+'/*/'+bam_prefix+'.bam') + os.system('mkdir -p '+bam_dir+'/'+task_name+'.RL'+str(read_length)) + for fin_name in mapping_bam_list: + folder_name= '/'.join(fin_name.split('/')[:-1]) + writeShell(rMATS_path, fin_name, folder_name, bam_dir, str(read_length), gtf, novelSS, task_name, task_dir) + + +if __name__ == '__main__': + main() diff --git a/IRIS/IRIS_makesubsh_rmatspost.py b/IRIS/IRIS_makesubsh_rmatspost.py new file mode 100644 index 0000000..aec5195 --- /dev/null +++ b/IRIS/IRIS_makesubsh_rmatspost.py @@ -0,0 +1,48 @@ +import sys,glob,os,argparse +from . import config + +def write_task_script(rMATS_path, bam_dir, task_name, gtf, novelSS, task_dir): + read_length=int(bam_dir.split('/')[-1].split('.')[-1][2:]) + dir_name=task_name+'_RL'+str(read_length) + graphlist=glob.glob(bam_dir+'/*.tmp/*') + os.system('mkdir -p '+bam_dir+'/'+dir_name+'.graph') + os.system('cp '+bam_dir+'/*.tmp/* '+bam_dir+'/'+dir_name+'.graph/.') + print '[INFO] Done copy' + cmd='head -n1 -q '+bam_dir+'/'+dir_name+'.graph/*.rmats |paste -d, -s >'+bam_dir+'/'+dir_name+'_rmatspost_list.txt' + print cmd + os.system(cmd) + + task_script_base = 'rMATS_post.{}.sh'.format(dir_name) + task_script = os.path.join(task_dir, task_script_base) + fout=open(task_script,'w') + + novelSS_str='' + if novelSS: + novelSS_str='--novelSS ' + # TODO the '|| true' at the end of this command ignores a + # failure return code from the python command. + # rMATS produces the desired output files despite the error return. + # A future version of rMATS may fix this behavior. + fout.write('#!/bin/bash\npython '+rMATS_path+' --b1 '+bam_dir+'/'+dir_name+'_rmatspost_list.txt --od '+bam_dir+'/'+dir_name+'.matrix --tmp '+bam_dir+'/'+dir_name+'.graph/ --anchorLength 1 --readLength '+str(read_length)+' --gtf '+gtf+' -t paired --nthread 8 --task post --statoff '+novelSS_str+'|| true\n') + fout.close() + return + + +def main(args): + rMATS_path=args.rMATS_path + prep_dir=args.bam_dir.rstrip('/') + gtf=args.gtf + task_name=args.data_name + novelSS=args.novelSS + task_dir=args.task_dir + if not os.path.exists(task_dir): + os.makedirs(task_dir) + + rl_bam_folders=glob.glob(prep_dir+'/'+task_name+'.RL*') + for bam_folder in rl_bam_folders: + print bam_folder + write_task_script(rMATS_path, bam_folder,task_name, gtf, novelSS, task_dir) + + +if __name__ == '__main__': + main() diff --git a/IRIS/IRIS_ms_makedb.py b/IRIS/IRIS_ms_makedb.py index 65eb75b..effe056 100644 --- a/IRIS/IRIS_ms_makedb.py +++ b/IRIS/IRIS_ms_makedb.py @@ -67,6 +67,9 @@ def main(args): outdir=args.outdir.rstrip('/') exp_fin_list=args.exp_fin_list uniprot_fasta=args.uniprot_fasta + java_path = args.java_path + MSGF_path = args.MSGF_path + print '##Creating ProteoTransicritomic Ref' makeProteoTranscriptomeRef(exp_fin_list, uniprot_fasta, outdir) @@ -91,7 +94,7 @@ def main(args): print cmd1 os.system(cmd1) - cmd2='/u/local/apps/java/jdk1.8.0_111/bin/java -Xmx8g -cp ~/MSGFPlus/MSGFPlus.jar edu.ucsd.msjava.msdbsearch.BuildSA -d '+outdir+'/tmp/proteome_ref_combined.fa' + cmd2=java_path+' -Xmx8g -cp '+MSGF_path+' edu.ucsd.msjava.msdbsearch.BuildSA -d '+outdir+'/tmp/proteome_ref_combined.fa' print '##Indexing the proteotranscriptomic db' print cmd2 os.system(cmd2) diff --git a/IRIS/IRIS_parse_hla.py b/IRIS/IRIS_parse_hla.py index 691026a..f2dc2ea 100644 --- a/IRIS/IRIS_parse_hla.py +++ b/IRIS/IRIS_parse_hla.py @@ -5,15 +5,14 @@ def main(args): outdir=args.outdir.rstrip('/') - fin_list=glob.glob(outdir+'/*/hla_types/hla_types-ClassI.HLAgenotype4digits') + fin_list=glob.glob(outdir+'/*/*-ClassI.HLAgenotype4digits') HLA2patients={} HLA_list=set() for fin in fin_list: - name=fin.split('/')[-3] - print name + name=fin.split('/')[-2] n=0 if name in HLA2patients: - print 'dup' + print 'Duplicated name '+name+'. Exit!' exit() HLA2patients[name]=[] for l in open(fin): @@ -21,12 +20,15 @@ def main(args): n+=1 continue ls=l.strip().split('\t') - if float(ls[2])<=0.05: - HLA2patients[name].append('HLA-'+ls[1].rstrip("'")) - HLA_list.add('HLA-'+ls[1].rstrip("'")) - if float(ls[4])<=0.05: - HLA2patients[name].append('HLA-'+ls[3].rstrip("'")) - HLA_list.add('HLA-'+ls[3].rstrip("'")) + if ls[2]!='NA': + if float(ls[2])<=0.05: + HLA2patients[name].append('HLA-'+ls[1].rstrip("'")) + HLA_list.add('HLA-'+ls[1].rstrip("'")) + if ls[4]!='NA': + if float(ls[4])<=0.05: + HLA2patients[name].append('HLA-'+ls[3].rstrip("'")) + HLA_list.add('HLA-'+ls[3].rstrip("'")) + fout1=open(outdir+'/hla_patient.tsv','w') for k in HLA2patients: fout1.write('\t'.join([k]+HLA2patients[k])+'\n') @@ -38,13 +40,12 @@ def main(args): fout2.write(h+'\n') fout2.close() - fin_list2=glob.glob(outdir+'/*/hla_types/hla_types-ClassI.expression') + fin_list2=glob.glob(outdir+'/*/*-ClassI.expression') HLAexp2patients={} for fin2 in fin_list2: - name=fin2.split('/')[-3] - print name + name=fin2.split('/')[-2] if name in HLAexp2patients: - print 'dup' + print 'Duplicated name '+name+'. Exit!' exit() HLAexp2patients[name]=[] for l in open(fin2): diff --git a/IRIS/IRIS_pep2epitope.py b/IRIS/IRIS_pep2epitope.py index d756dc3..442a2fe 100644 --- a/IRIS/IRIS_pep2epitope.py +++ b/IRIS/IRIS_pep2epitope.py @@ -3,22 +3,11 @@ #import multiprocessing as mp # from Bio.Blast import NCBIXML from subprocess import Popen, PIPE -#now = datetime.datetime.now() + ID = str(uuid.uuid4()).split('-')[0] def loadSampleMHC(f_s_in): - # if f_s_in.find(',')!=-1: hla_allele_list=f_s_in.split(',') - # else: - # n=0 - # hla_allele_list=[] - # for l in open(f_s_in): - # if n==0: - # n+=1 - # continue - # ls=l.strip().split('\t') - # hla_allele_list.append('HLA-'+ls[1].strip("'")) - # hla_allele_list.append('HLA-'+ls[3].strip("'")) return hla_allele_list def parsePred(stdout): @@ -40,7 +29,7 @@ def localIEDBCommand(iedb_path, hla_allele, epitope_len, outdir, JC_pep_fasta, e response = Popen(['python',iedb_path+'/predict_binding.py',IEDB_model,hla_allele,epitope_len,outdir+'/tmp/prot.compared/'+form+'/'+form+'.'+JC_pep_fasta], stdout=fout,shell = False) #response = Popen(['/u/home/p/panyang/local/bin/python',iedb_path+'/predict_binding.py',IEDB_model,hla_allele,epitope_len,outdir+'/tmp/prot.compared/'+form+'/'+form+'.'+JC_pep_fasta], stdout=fout,shell = False) fout.close() - + return response def pep2antigen(JC_pep_fasta, enst_id, hla_allele_list,epitope_len_list, iedb_path, outdir): predicting=[] @@ -58,14 +47,13 @@ def pep2antigen(JC_pep_fasta, enst_id, hla_allele_list,epitope_len_list, iedb_pa #return parsed_dict def pep2antigen_single(JC_pep_fasta, enst_id, form, hla_allele_list,epitope_len_list, iedb_path, outdir): predicting=[] - n=0 - #file_list=[] for hla_allele in hla_allele_list: for epitope_len in epitope_len_list: - #file_list.append(outdir+'/tmp/'+JC_pep_fasta+' '+hla_allele.replace('*','_').replace(':','_')+'.'+epitope_len+'_iedb.txt') - localIEDBCommand(iedb_path, hla_allele, epitope_len, outdir, JC_pep_fasta, enst_id, form) - #parsed_dict=[parsePred(open(tmp_file)) for tmp_file in file_list] - #return parsed_dict + response = localIEDBCommand(iedb_path, hla_allele, epitope_len, outdir, JC_pep_fasta, enst_id, form) + predicting.append(response) + + for response in predicting: + response.wait() def main(args): @@ -79,9 +67,10 @@ def main(args): hla_allele_list=loadSampleMHC(args.hla_allele_list) if hla_allele_list==[]: sys.exit("# No HLA Alleles. Exit.") - epitope_len_list=args.epitope_len_list.split(',') + epitope_len_list=map(int,args.epitope_len_list.split(',')) if min(epitope_len_list)<8: sys.exit("# The request epitope length is too small. Exit.") + epitope_len_list=map(str,epitope_len_list) fs=fin.split('/')[-1].split('.') JC_pep_fasta='.'.join(fs[1:]) diff --git a/IRIS/IRIS_prediction.py b/IRIS/IRIS_prediction.py index 5daecb6..16571ef 100644 --- a/IRIS/IRIS_prediction.py +++ b/IRIS/IRIS_prediction.py @@ -1,5 +1,6 @@ import sys, argparse, os ,datetime,logging, uuid, glob from . import config +import numpy as np ID = str(uuid.uuid4()).split('-')[0] @@ -18,16 +19,40 @@ def loadFeatures(fin): return extracelllularDict -def selectJC(AS_coord,deltaPSI_c2n,cut_off, select_all): - if select_all: - return [(AS_coord[2], AS_coord[3]),(AS_coord[2],AS_coord[0]),(AS_coord[1],AS_coord[3])] - if float(deltaPSI_c2n)'): + continue + junction_peptide.append(l.strip()) + if os.path.exists(peptide_file_name_full_inc): + for l in open(peptide_file_name_full_inc): + if l.startswith('>'): + continue + junction_peptide.append(l.strip()) + junction_peptide=';'.join(junction_peptide) + junction_peptide='-' if junction_peptide=='' else junction_peptide + return junction_peptide + +def loadGeneExp(gene_exp_matrix_fin): + Exp={} + i=0 + for l in open(gene_exp_matrix_fin): + if i==0: + i+=1 + continue + ls=l.strip().split('\t') + name=ls[0].split('_')[0].split('.')[0] + exp_list=map(float, ls[1:]) + Q1=round(np.nanpercentile(exp_list,25),2) + Q3=round(np.nanpercentile(exp_list,75),2) + mean=round(np.nanmean(exp_list),2) + Exp[name]=map(str,[mean, Q1, Q3]) + return Exp,['meanGeneExp','Q1GeneExp','Q3GeneExp'] + +def extracellularAnnotation(screening_result_fin, splicing_event_type, extracelllularDict, deltaPSI_cut_off, select_all, gene_exp_path, pep_dir_prefix): if select_all: deltaPSI_column=0 if select_all==False: @@ -71,7 +134,7 @@ def extracellularAnnotation(screening_result_fin, outdir, extracelllularDict, de continue ls=l.strip().split('\t') des=ls[0].split(':') - JC_pep_fasta=AS2FT(des[2],des[4:8],des[3],ls[deltaPSI_column],deltaPSI_cut_off,select_all,extracelllularDict,ls[0], fout_dict) + JC_coord_list=AS2FT(des[2],des[4:],des[3],ls[deltaPSI_column],deltaPSI_cut_off,select_all,extracelllularDict,ls[0], fout_dict, splicing_event_type) for k in fout_dict.keys(): fout.write(k+'\n') fout.close() @@ -84,8 +147,15 @@ def extracellularAnnotation(screening_result_fin, outdir, extracelllularDict, de if ls[3]+':'+ls[5] not in extracellular_AS[ls[0]]: extracellular_AS[ls[0]][ls[3]+':'+ls[5]]=[] extracellular_AS[ls[0]][ls[3]+':'+ls[5]].append(ls[1]+':'+ls[2]+':'+ls[4]) + + if gene_exp_path: + gene_exp_dict, gene_exp_header=loadGeneExp(gene_exp_path) + fout2=open(screening_result_fin+'.ExtraCellularAS.txt','w') - fout2.write('{}\t{}\t{}\t{}\n'.format('as_event','protein_domain_loc','protein_domain_loc_by_as_exon','\t'.join(screening_result_dict['header']))) + if gene_exp_path: + fout2.write('{}\t{}\t{}\t{}\t{}\t{}\n'.format('as_event','protein_domain_loc','protein_domain_loc_by_as_exon','\t'.join(screening_result_dict['header']),'\t'.join(gene_exp_header), 'junction_peptide')) + else: + fout2.write('{}\t{}\t{}\t{}\t{}\n'.format('as_event','protein_domain_loc','protein_domain_loc_by_as_exon','\t'.join(screening_result_dict['header']),'junction_peptide')) for k in sorted(extracellular_AS): line='' line=';'.join(extracellular_AS[k].keys())+'\t' @@ -97,15 +167,20 @@ def extracellularAnnotation(screening_result_fin, outdir, extracelllularDict, de line+=anno_line+';' screen_print='NA' if k in screening_result_dict: + junction_peptide=retriveJunctionPeptide(k, screening_result_fin, splicing_event_type, pep_dir_prefix) screen_print='\t'.join(screening_result_dict[k]) - fout2.write(k+'\t'+line.rstrip(';')+'\t'+screen_print+'\n') + optional_annotations='' + if gene_exp_path: + gene_exp_list=gene_exp_dict[k.split(':')[0]] + optional_annotations='\t'.join(gene_exp_list) + fout2.write('\t'.join([k,line.rstrip(';'), screen_print, optional_annotations, junction_peptide])+'\n') fout2.close() os.system('rm '+screening_result_fin+'.CellSurfAnno.tmp') -def epitopePredictionPrep(outdir, hla_list_fin, analysis_name, iedb_path): - fin_list=glob.glob(outdir+'/tmp/prot.compared/skp/*.fa') - fin_list=fin_list+glob.glob(outdir+'/tmp/prot.compared/inc/*.fa') +def epitopePredictionPrep(outdir, hla_list_fin, analysis_name, iedb_path, epitope_len_list, task_dir, pep_dir_prefix): + fin_list=glob.glob(outdir+'/tmp/'+pep_dir_prefix+'.compared/skp/*.fa') + fin_list=fin_list+glob.glob(outdir+'/tmp/'+pep_dir_prefix+'.compared/inc/*.fa') hla_list=[] num=90 for l in open(hla_list_fin): @@ -114,67 +189,76 @@ def epitopePredictionPrep(outdir, hla_list_fin, analysis_name, iedb_path): print "[INFO] Total HLA types loaded:", len(hla_list), ". Total peptide splice junctions loaded:",len(fin_list) analysis_name=outdir.split('/')[-1] - list_name='cmdlist.pep2epitope_'+analysis_name - fout_list=open(list_name,'w') - m=0 + script_count = 0 + def write_task_script(script_contents): + task_script_base = 'pep2epitope_{}.{}.sh'.format(analysis_name, script_count) + task_script = os.path.join(task_dir, task_script_base) + + with open(task_script, 'w') as f_h: + f_h.write('#!/bin/bash\n') + f_h.write(script_contents) + for i in xrange(0,len(hla_list),3): hla_types=','.join(hla_list[i:i+3]) n=0 - line='' + script_contents = '' for fin in fin_list: if os.stat(fin).st_size != 0: n+=1 - #line+='echo run '+fin+'\npython '+IRIS_PACKAGE_PATH+'/IRIS/IRIS.pep2epitope.py '+fin+' --hla-allele-list '+hla_types+' -o '+outdir+'\n' - line+='echo run '+fin+'\nIRIS pep2epitope '+fin+' --hla-allele-list '+hla_types+' -o '+outdir+' --iedb-local '+iedb_path+'\n' - line+='sleep 70\n' + script_contents += 'echo run '+fin+'\nIRIS pep2epitope '+fin+' --hla-allele-list '+hla_types+' -o '+outdir+' --iedb-local '+iedb_path+' -e '+epitope_len_list+'\n' if n%num==0: - submission_file_name=outdir+'/tmp/submit.IRIS_pep2epitope.py.'+str(n)+'.'+hla_types+'.sh' - fout_list.write(submission_file_name+'\n') - m+=1 - fout=open(submission_file_name,'w') - fout.write(line) - fout.close() - line='' + write_task_script(script_contents) + script_count += 1 + script_contents = '' if n%num!=0: - # print line - submission_file_name=outdir+'/tmp/submit.IRIS_pep2epitope.py.'+str(n)+'.'+hla_types+'.sh' - fout_list.write(submission_file_name+'\n') - m+=1 - fout=open(submission_file_name,'w') - fout=open(outdir+'/tmp/submit.IRIS_pep2epitope.py.'+str(n)+'.'+hla_types+'.sh','w') - fout.write(line) - fout.close() - line='' - - fout_list.close() - fout_qsub=open('qsub.IRIS_pep2epitope.py.'+analysis_name+'.sh','w') - - cmd='qsub -t 1-'+str(m)+':1 qsub.IRIS_pep2epitope.py.'+analysis_name+'.sh' - - fout_qsub.write('#!/bin/bash\n#$ -N IRIS_pep2epitope\n#$ -S /bin/bash\n#$ -R y\n#$ -l '+config.QSUB_PREDICTION_CONFIG+'\n#$ -V\n#$ -cwd\n#$ -j y\n#$ -m bea\n') - fout_qsub.write('export s=`sed -n ${SGE_TASK_ID}p '+list_name+'`\necho $s\nbash $s') - fout_qsub.close() - print cmd + write_task_script(script_contents) + script_count += 1 + script_contents = '' + def main(args): #Define parameters - IRIS_screening_result=args.IRIS_screening_result_path + IRIS_screening_result=args.IRIS_screening_result_path.rstrip('/') #Modified 2021 deltaPSI_cut_off=float(args.deltaPSI_cut_off) + splicing_event_type=args.splicing_event_type select_all= True if args.extracellular_anno_by_junction==False else False analysis_name=[l.strip() for l in open(args.parameter_fin)][0] - hla_list_fin=args.mhc_list - iedb_path=args.iedb_local - extracelllularDict=loadFeatures(config.EXTRACELLULAR_FEATURES_UNIPROT2GTF_MAP_PATH) #IRIS_package_dir.rstrip('/')+'/IRIS/data/features.uniprot2gtf.ExtraCell.txt' - print "[INFO] Total extracellular annotated loaded:",len(extracelllularDict) - - extracellularAnnotation(IRIS_screening_result+'/'+analysis_name+'.primary.txt', IRIS_screening_result, extracelllularDict, deltaPSI_cut_off, select_all) - epitopePredictionPrep(IRIS_screening_result+'/primary',hla_list_fin, analysis_name, iedb_path) + analysis_name=analysis_name+'.'+splicing_event_type #group_name.type + prioritized_only=args.tier3_only + extracellular_only=args.extracellular_only + gene_exp_matrix=args.gene_exp_matrix + task_dir=args.task_dir + all_orf=args.all_orf + pep_dir_prefix='prot' + if all_orf: + pep_dir_prefix='prot_allorf' + if not os.path.exists(task_dir): + os.makedirs(task_dir) + if extracellular_only==False: + hla_list_fin=args.mhc_list + iedb_path=args.iedb_local + epitope_len_list=args.epitope_len_list.rstrip(',') - extracellularAnnotation(IRIS_screening_result+'/'+analysis_name+'.prioritized.txt', IRIS_screening_result, extracelllularDict, deltaPSI_cut_off, select_all) - epitopePredictionPrep(IRIS_screening_result+'/prioritized',hla_list_fin, analysis_name, iedb_path) + extracelllularDict=loadFeatures(config.EXTRACELLULAR_FEATURES_UNIPROT2GTF_MAP_PATH) + print "[INFO] Total extracellular annotation loaded:",len(extracelllularDict) + + if config.file_len(IRIS_screening_result+'/'+analysis_name+'.tier1.txt')==1 and prioritized_only==False: + prioritized_only=True + print "[INFO] No tier1 comparisons (tissue-matched normal) found. Use tier2&tier3 only mode. " + if prioritized_only: + extracellularAnnotation(IRIS_screening_result+'/'+analysis_name+'.tier2tier3.txt', splicing_event_type, extracelllularDict, deltaPSI_cut_off, select_all, gene_exp_matrix, pep_dir_prefix) + if extracellular_only==False: + epitopePredictionPrep(IRIS_screening_result+'/'+splicing_event_type+'.tier2tier3',hla_list_fin, analysis_name, iedb_path, epitope_len_list, task_dir, pep_dir_prefix) + else: + extracellularAnnotation(IRIS_screening_result+'/'+analysis_name+'.tier1.txt', splicing_event_type, extracelllularDict, deltaPSI_cut_off, select_all, gene_exp_matrix, pep_dir_prefix) + extracellularAnnotation(IRIS_screening_result+'/'+analysis_name+'.tier2tier3.txt', splicing_event_type, extracelllularDict, deltaPSI_cut_off, select_all, gene_exp_matrix, pep_dir_prefix) + if extracellular_only==False: + epitopePredictionPrep(IRIS_screening_result+'/'+splicing_event_type+'.tier1',hla_list_fin, analysis_name, iedb_path, epitope_len_list, task_dir, pep_dir_prefix) + #epitopePredictionPrep(IRIS_screening_result+'/'+splicing_event_type+'.prioritized',hla_list_fin, analysis_name, iedb_path, epitope_len_list, task_dir) #Only primary is needed as priortized can be parsed from it + if __name__ == '__main__': main() diff --git a/IRIS/IRIS_screening.py b/IRIS/IRIS_screening.py index e33b302..9214977 100644 --- a/IRIS/IRIS_screening.py +++ b/IRIS/IRIS_screening.py @@ -1,4 +1,3 @@ - import numpy as np import os, glob, pyBigWig, argparse from scipy import stats @@ -57,28 +56,29 @@ def fetch_PsiMatrix(eid, fn, outdir, delim, index=None): data = np.asarray(f.readline().strip().split(delim)) return (header, data) -def openTestingFout(outdir, out_prefix, summary_file, ref_list, test_mode): +def openTestingFout(outdir, out_prefix, splicing_event_type, summary_file, panel_list, test_mode, fin_name=''): header=['as_event','meanPSI','Q1PSI','Q3PSI'] if test_mode=='group': header_prefix=['_pVal','_deltaPSI','_tumorFC'] if test_mode=='personalized': header_prefix=['_modifiedPctl','_deltaPSI','_tumorFC'] - fout=open(outdir+'/'+out_prefix+'.test.all.txt','w') + fout_name=outdir+'/'+out_prefix+'.'+splicing_event_type+'.test.all_'+fin_name+'.txt' + fout=open(fout_name,'w') if summary_file==False: - header+=['\t'.join(map(lambda x:ref+x ,header_prefix)) for ref in ref_list if ref!=out_prefix] + header+=['\t'.join(map(lambda x:ref+x ,header_prefix)) for ref in panel_list if ref!=out_prefix] fout.write('\t'.join(header)+'\n') - return fout + return fout, fout_name -def openScreeningFout(outdir, out_prefix, fout_name): - fout=open(outdir+'/'+out_prefix+'.'+fout_name+'.txt','w') +def openScreeningFout(outdir, out_prefix, splicing_event_type, fout_name): + fout=open(outdir+'/'+out_prefix+'.'+splicing_event_type+'.'+fout_name+'.txt','w') fout.write('{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\n'.format('as_event','meanPSI','Q1PSI','Q3PSI','deltaPSI','fc_of_tumor_isoform','tissue_matched_normal_panel','tumor_panel','normal_panel','tag','mappability','mappability_tag')) return fout -def writeSummaryFile(out_prefix, db_dir, index, fout): #no screening, preparing for MS search - fin_name=db_dir+'/'+out_prefix+'/splicing_matrix/splicing_matrix.SE.cov10.'+group_name+'.txt' +def writeSummaryFile(out_prefix, splicing_event_type, db_dir, index, fout, fetching_data_col): #no screening, preparing for MS search + fin_name=db_dir+'/'+out_prefix+'/splicing_matrix/splicing_matrix.'+splicing_event_type+'.cov10.'+group_name+'.txt' index[out_prefix]=read_PsiMatrix_index(fin_name,'/'.join(fin_name.split('/')[:-1])) for j,k in enumerate(index[out_prefix]): - psi_event=map(float,fetch_PsiMatrix(k,fin_name,'.','\t',index[out_prefix])[1][8:]) + psi_event=map(float,fetch_PsiMatrix(k,fin_name,'.','\t',index[out_prefix])[1][fetching_data_col:]) query_mean=[np.nanmean(psi_event),np.nanpercentile(psi_event,25),np.nanpercentile(psi_event,75)] result=[k]+query_mean fout.write('\t'.join(map(str,result))+'\n') @@ -107,6 +107,21 @@ def getMappability(splicing_event,bw_map,d): mappability=[str(up_mean),str(target_mean),str(down_mean)] return mappability +def getDirection(filter1_panel_list, psi, test, non_parametric, screening_type): + psi_primary=[] + deltaPSI_primary=[] + for primary_group in filter1_panel_list: + if test[primary_group]!=['-']*3: + psi_primary+=psi[primary_group] + deltaPSI_primary.append(test[primary_group][1]) + if psi_primary==[]: #in case tissue-matched normal doesn't have data + return [],[] + else: + direction='greater' if non_parametric else 'larger' + if np.median(deltaPSI_primary)<=0: + direction='less' if non_parametric else 'smaller' + return psi_primary, direction + def calcTumorFormFoc(delta_psi, mean_psi): if delta_psi>0: return mean_psi/(mean_psi - delta_psi+10**-8) @@ -137,91 +152,127 @@ def one2N(p1, g1, test_type): else: return [np.nan,delta_psi,tumor_foc] -#def groupTest(g1,g2, test_type, threshold_tost=0.05): -def groupTest(g1,g2, test_type, direction='two-sided'): +def statTest(g1,g2, direction, non_parametric): + if direction != 'equivalence': + if non_parametric: + pvalue=stats.mannwhitneyu(g1,g2, alternative=direction)[1] + else: + pvalue=smw.ttest_ind(g1,g2, alternative=direction, usevar='unequal')[1] + #pvalue=smw.ttest_ind(g1,g2, alternative=direction)[1] + else: + threshold_tost = 0.05 + pvalue=smw.ttost_ind(g1,g2,-threshold_tost,threshold_tost,usevar='unequal')[0] #equivalence test + return pvalue + +def statTest_minSampleCount(g1,g2, direction, non_parametric):#Only enabled when filters out by min_sample_count. With enough sample, using the default setting assume equal var for both groups is ok. + if direction != 'equivalence': + if non_parametric: + pvalue=stats.mannwhitneyu(g1,g2, alternative=direction)[1] + else: + pvalue=smw.ttest_ind(g1,g2, alternative=direction)[1] + #pvalue=smw.ttest_ind(g1,g2, alternative=direction)[1] + else: + threshold_tost = 0.05 + pvalue=smw.ttost_ind(g1,g2,-threshold_tost,threshold_tost)[0] #equivalence test + return pvalue + +def groupTest(g1,g2, non_parametric=False, direction='two-sided', min_sample_count=False): g1=np.array(g1) g2=np.array(g2) g1=g1[~np.isnan(g1)] g2=g2[~np.isnan(g2)] delta_psi=np.nanmean(g1)-np.nanmean(g2) tumor_foc=calcTumorFormFoc(delta_psi,np.nanmean(g1)) - if test_type=='sig': - t1=stats.ttest_ind(g1,g2)[1] - elif test_type=='equ': - t1=smw.ttest_ind(g1,g2,alternative='two-sided')[1] - # t1=smw.ttost_ind(g1,g2,-threshold_tost,threshold_tost,usevar='unequal')[0] #equalvalence test - return [t1, delta_psi, tumor_foc] - -def getDirection(filter1, psi, test): - psi_primary=[] - deltaPSI_primary=[] - for primary_group in filter1: - if test[primary_group]!=['-']*3: - psi_primary+=psi[primary_group] - deltaPSI_primary+=test[primary_group] - if psi_primary==[]: #in case tissue-matched normal doesn't have data - return [],[] - else: - direction='larger' - if np.median(deltaPSI_primary)<=0: - direction='smaller' - return psi_primary, direction - -def summarizeTestResult(filter1_cutoff_pval, filter1_cutoff_dpsi, filter1_cutoff_foc,filter2_cutoff_pval, filter2_cutoff_dpsi, filter2_cutoff_foc, filter3_cutoff_pval, filter3_cutoff_dpsi, filter3_cutoff_foc, primary, tumor_rec, pval, deltaPSI, foc, testing_type_index): - differential,equal,positive,negative=[0,0,0,0] - testable=0 + if min_sample_count: + pvalue = statTest_minSampleCount(g1, g2, direction, non_parametric) + else: + pvalue = statTest(g1, g2, direction, non_parametric) + return [pvalue, delta_psi, tumor_foc] + +def performTest(set_matched_tumor, has, j, group, screening_type_list, psi, out_prefix, non_parametric, test, filter1_panel_list, psi_primary, direction, min_sample_count): #PSI value-based screen allow two-sided or one-sided tests. Different from SJ count or CPM-based screen, where only one-sided test is needed. + screening_type = screening_type_list[j] + redirect_output = False + test_result = ['-']*3 #For missing in non-eesential tests/comparisons + if screening_type == 'association': + if has[group]: + test_result = groupTest(psi[out_prefix],psi[group], non_parametric,"two-sided", min_sample_count) + return test_result + + has_matched_tumor = False if psi_primary == [] else True #for clarity + + if screening_type == 'recurrence': + if set_matched_tumor and has_matched_tumor:#set_matched_tumor is redundent here. kept for future implemtation of additional output type. + if has[group]: + test_result = groupTest(psi[group],psi_primary, non_parametric, direction, min_sample_count) + else: + if has[group]: + test_result = groupTest(psi[out_prefix],psi[group], non_parametric, "equivalence", min_sample_count)#No or equivalent testing -John's use case + return test_result + + if screening_type == 'association_high': + if set_matched_tumor and has_matched_tumor: + if has[group]: + test_result = groupTest(psi[out_prefix],psi[group], non_parametric, direction, min_sample_count) + else: + if has[group]: + test_result = groupTest(psi[out_prefix],psi[group], non_parametric, "two-sided", min_sample_count) #Two-sided testing + return test_result + +def summarizeTestResult(filter1_cutoff_pval, filter1_cutoff_dpsi, filter1_cutoff_foc,filter2_cutoff_pval, filter2_cutoff_dpsi, filter2_cutoff_foc, filter3_cutoff_pval, filter3_cutoff_dpsi, filter3_cutoff_foc, pval, deltaPSI, foc, screening_type_list): + association_passed,recurrence_passed,specificity_positive,specificity_negative, specificity_testable=[0,0,0,0,0] primary_result, primary_result_foc=[[],[]] #take care of multiple tiisue matched norm - deltapsi_list,foc_list=[[],[]] #if no tissue-matched norm, use median - for i,group_type in enumerate(testing_type_index): - if pval[i]=='-': + deltapsi_list_voted,foc_list_voted=[[],[]] #if no tissue-matched norm, use median + for i,group_type in enumerate(screening_type_list): + if pval[i]=='-': #This is important - skip all missing, which is not useful for summarizing but not affecting consistancy. continue - if i=filter1_cutoff_dpsi and float(foc[i])>=filter1_cutoff_foc: - differential+=1 + association_passed+=1 continue - if testing_type_index[i]=='equ': + if screening_type_list[i]=='recurrence': if float(pval[i])<=filter2_cutoff_pval and abs(float(deltaPSI[i]))>=filter2_cutoff_dpsi: - equal+=1 + recurrence_passed+=1 continue - if testing_type_index[i]=='sig' and i>=(primary+tumor_rec): - testable+=1 + if screening_type_list[i]=='association_high':# TODO: judge set/has, then run + specificity_testable+=1 if float(pval[i])<=filter3_cutoff_pval and float(foc[i])>=filter3_cutoff_foc: - deltapsi_list.append(float(deltaPSI[i])) - foc_list.append(float(foc[i])) + deltapsi_list_voted.append(float(deltaPSI[i])) + foc_list_voted.append(float(foc[i])) if float(deltaPSI[i])>=filter3_cutoff_dpsi: - positive+=1 + specificity_positive+=1 continue if float(deltaPSI[i])<=-filter3_cutoff_dpsi: - negative+=1 + specificity_negative+=1 continue - return differential,equal,positive,negative,testable, np.median(primary_result), np.median(primary_result_foc), deltapsi_list,foc_list + return association_passed,recurrence_passed,specificity_positive,specificity_negative,specificity_testable, np.median(primary_result), np.median(primary_result_foc), deltapsi_list_voted,foc_list_voted -def defineTumorEvents(filter1_group_cutoff,filter2_group_cutoff,filter3_group_cutoff, primary, norm_tissue, differential, equal, positive, negative, testable, primary_result, primary_result_foc, deltapsi_list,foc_list, use_ratio): +def defineTumorEvents(filter1_group_cutoff,filter2_group_cutoff,filter3_group_cutoff, set_matched_tumor, specificity_panel_len, association_passed, recurrence_passed, specificity_positive, specificity_negative, specificity_testable, primary_result, primary_result_foc, deltapsi_list_voted, foc_list_voted, use_ratio): tag=[] - if differential>=filter1_group_cutoff: + if association_passed>=filter1_group_cutoff:#Improvement? current: 0>='' is false tag.append('associated') - if equal>=filter2_group_cutoff: + if recurrence_passed>=filter2_group_cutoff: tag.append('recurrent') - if primary==0: - tissue_specificity=max(positive,negative) - ratio=False if use_ratio==False else tissue_specificity/(testable+10**-8)>=filter3_group_cutoff/(norm_tissue+0.0) - if tissue_specificity>=filter3_group_cutoff or ratio: - tag.append('specific') - else: + if set_matched_tumor:#TODO-FUTURE: take care of set-yes has-no redirected events if primary_result>0: - tissue_specificity=positive - ratio=False if use_ratio==False else positive/(testable+10**-8)>=filter3_group_cutoff/(norm_tissue+0.0) - if positive>=filter3_group_cutoff or ratio: - tag.append('specific') + tissue_specificity=specificity_positive + ratio=False if use_ratio==False else specificity_positive/(specificity_testable+10**-8)>=filter3_group_cutoff/(specificity_panel_len+0.0) + if specificity_positive>=filter3_group_cutoff or ratio: + tag.append('high_assoc') else: - tissue_specificity=negative - ratio=False if use_ratio==False else negative/(testable+10**-8)>=filter3_group_cutoff/(norm_tissue+0.0) - if negative>=filter3_group_cutoff or ratio: - tag.append('specific') - primary_deltapsi=primary_result if primary_result!=0 else np.median(deltapsi_list) - primary_foc=primary_result_foc if primary_result!=0 else np.median(foc_list) + tissue_specificity=specificity_negative + ratio=False if use_ratio==False else specificity_negative/(specificity_testable+10**-8)>=filter3_group_cutoff/(specificity_panel_len+0.0) + if specificity_negative>=filter3_group_cutoff or ratio: + tag.append('high_assoc') + else: + tissue_specificity=max(specificity_positive,specificity_negative) + ratio=False if use_ratio==False else tissue_specificity/(specificity_testable+10**-8)>=filter3_group_cutoff/(specificity_panel_len+0.0) + if tissue_specificity>=filter3_group_cutoff or ratio: + tag.append('high_assoc') + primary_deltapsi=primary_result if (primary_result!=0 and set_matched_tumor) else np.median(deltapsi_list_voted)#TODO + primary_foc=primary_result_foc if (primary_result!=0 and set_matched_tumor) else np.median(foc_list_voted) + return primary_deltapsi, primary_foc, tissue_specificity, tag def mappability_write(k, bw_map, calc_length): @@ -244,25 +295,28 @@ def loadBlacklistEvents(fin): BlacklistEvents[des]='' return BlacklistEvents -def translationCMD(ref_genome, outdir, out_prefix, fout_name): +def translationCMD(ref_genome, gtf, outdir, out_prefix, splicing_event_type, all_orf, ignore_annotation, remove_early_stop, fout_name): #uSE name space - #Namespace(outdir='test1', parameter_fin='/u/home/p/panyang/bigdata-nobackup/Glioma_test/GBM.prioritze.par', subcommand='screening', translating=False) - cmd_translation='IRIS translation '+outdir+'/'+out_prefix+'.'+fout_name+'.txt '+' -o '+outdir+'/'+fout_name+' -g '+ref_genome + argument_line='' + if all_orf: + argument_line+=' --all-orf' + if ignore_annotation: + argument_line+=' --ignore-annotation' + if remove_early_stop: + argument_line+=' --remove-early-stop' + cmd_translation='IRIS translate '+outdir+'/'+out_prefix+'.'+splicing_event_type+'.'+fout_name+'.txt '+' -o '+outdir+'/'+splicing_event_type+'.'+fout_name+' -g '+ref_genome+' -t '+splicing_event_type+argument_line+' --gtf '+gtf print '[INFO] Working on translating: '+fout_name os.system(cmd_translation) -def loadParametersRow(filter_para, ref_list): - if len(filter_para.split(' '))==6: - filter_cutoff_pval, filter_cutoff_dpsi, filter_cutoff_foc, filter_group_cutoff, filter_list =filter_para.split(' ')[1:] - filter_cutoff_pval=float(filter_cutoff_pval) - filter_cutoff_dpsi=float(filter_cutoff_dpsi) - filter_cutoff_foc=float(filter_cutoff_foc) - filter_group_cutoff=int(filter_group_cutoff) - filter_list=filter_list.split(',') - ref_list+=filter_list +def loadParametersRow(filter_para, panel_list): + filter_cutoffs='' + if filter_para.strip()!='': + filter_cutoffs = map(float,filter_para.strip().split(' ')[0].split(',')) + filter_panel_list = filter_para.strip().split(' ')[1].split(',') + panel_list+=filter_panel_list else: - filter_cutoff_pval, filter_cutoff_dpsi, filter_cutoff_foc, filter_group_cutoff, filter_list =['','','','',[]] - return filter_cutoff_pval, filter_cutoff_dpsi, filter_cutoff_foc, filter_group_cutoff, filter_list, ref_list + filter_panel_list =[] + return filter_cutoffs, filter_panel_list, panel_list def main(args): @@ -270,162 +324,238 @@ def main(args): index={} fin_list={} para_fin=args.parameter_fin + splicing_event_type=args.splicing_event_type + fetching_data_col=8 if splicing_event_type == 'SE' else 10 out_prefix,db_dir,filter1_para,filter2_para,filter3_para,test_mode,use_ratio,blacklist_path,mappability_path,ref_genome=[l.strip() for l in open(para_fin)] - ref_list=[out_prefix] + panel_list=[out_prefix] + test_mode=test_mode.split(' ') use_ratio=True if use_ratio=='True' else False - if blacklist_path=='': - blacklist_path=config.BRAIN_BLACKLIST_PATH - blacklist_events=loadBlacklistEvents(blacklist_path) + blacklist_events={} + min_sample_count=args.min_sample_count + if min_sample_count: + min_sample_count=int(min_sample_count) + if blacklist_path!='': + #if blacklist_path=='BRAIN_BLACKLIST_PATH': + blacklist_events=loadBlacklistEvents(blacklist_path) bw_map,calc_length=loadMappability(mappability_path) - filter1_cutoff_pval, filter1_cutoff_dpsi, filter1_cutoff_foc, filter1_group_cutoff, filter1, ref_list =loadParametersRow(filter1_para, ref_list) - filter2_cutoff_pval, filter2_cutoff_dpsi, filter2_cutoff_foc, filter2_group_cutoff, filter2, ref_list =loadParametersRow(filter2_para, ref_list) - filter3_cutoff_pval, filter3_cutoff_dpsi, filter3_cutoff_foc, filter3_group_cutoff, filter3, ref_list =loadParametersRow(filter3_para, ref_list) - if filter1==[] and filter2==[] and filter3==[] and test_mode!='summary': + all_orf=args.all_orf + ignore_annotation=args.ignore_annotation + remove_early_stop=args.remove_early_stop + use_existing_test_result=args.use_existing_test_result + + filter1_cutoffs, filter1_panel_list, panel_list = loadParametersRow(filter1_para, panel_list) + filter2_cutoffs, filter2_panel_list, panel_list = loadParametersRow(filter2_para, panel_list) + filter3_cutoffs, filter3_panel_list, panel_list = loadParametersRow(filter3_para, panel_list) + + if filter1_cutoffs!='': + filter1_cutoff_pval, filter1_cutoff_dpsi, filter1_cutoff_foc, filter1_group_cutoff=filter1_cutoffs[:3]+[filter1_cutoffs[4]] + else: + filter1_cutoff_pval, filter1_cutoff_dpsi, filter1_cutoff_foc, filter1_group_cutoff=['','','',''] + if filter2_cutoffs!='': + filter2_cutoff_pval, filter2_cutoff_dpsi, filter2_cutoff_foc, filter2_group_cutoff=filter2_cutoffs[:3]+[filter2_cutoffs[4]] + else: + filter2_cutoff_pval, filter2_cutoff_dpsi, filter2_cutoff_foc, filter2_group_cutoff=['','','',''] + if filter3_cutoffs!='': + filter3_cutoff_pval, filter3_cutoff_dpsi, filter3_cutoff_foc, filter3_group_cutoff=filter3_cutoffs[:3]+[filter3_cutoffs[4]] + else: + filter3_cutoff_pval, filter3_cutoff_dpsi, filter3_cutoff_foc, filter3_group_cutoff=['','','',''] + if filter1_panel_list==[] and filter2_panel_list==[] and filter3_panel_list==[] and test_mode[0]!='summary': exit("[Error] No filtering required in parameteres file. exit!") - group_test=False if test_mode!='group' else True - individual_test=False if test_mode!='personalized' else True - summary_file=False if test_mode!='summary' else True + + non_parametric=False + if len(test_mode)>1: + non_parametric=True if test_mode[1]=='nonparametric' else False + + group_test=False if test_mode[0]!='group' else True + individual_test=False if test_mode[0]!='personalized' else True + summary_file=False if test_mode[0]!='summary' else True if [group_test,individual_test,summary_file]==[False,False,False]: exit('[Error] Need to choose one mode.exit!') - primary=len(filter1) - tumor_rec=len(filter2) - norm_tissue=len(filter3) - filter_count=sum(1 for i in [primary, tumor_rec, norm_tissue] if i!=0) - testing_type_index=['sig']*primary+['equ']*tumor_rec+['sig']*norm_tissue - - db_dir=db_dir.rstrip('/') + association_panel_len=len(filter1_panel_list) + recurrence_panel_len=len(filter2_panel_list) + specificity_panel_len=len(filter3_panel_list) + panel_count=sum(1 for i in [association_panel_len, recurrence_panel_len, specificity_panel_len] if i!=0) + screening_type_list=['association']*association_panel_len+['recurrence']*recurrence_panel_len+['association_high']*specificity_panel_len + set_matched_tumor= True if screening_type_list[0] == 'association' else False + + if args.translating: + gtf=args.gtf + if os.path.exists(gtf)==False: + exit('[Error] No gtf file provided for translation. exit!') + + ###Create Folders/Output#### outdir=args.outdir.rstrip('/') os.system('mkdir -p '+outdir) + db_dir=db_dir.rstrip('/') - ###Create Folders/Output#### - fout= openTestingFout(outdir, out_prefix, summary_file, ref_list, test_mode) - if summary_file: - writeSummaryFile(out_prefix, db_dir, index, fout) - exit() - - fout_filtered=open(outdir+'/'+out_prefix+'.notest.txt','w') - fout_primary=openScreeningFout(outdir, out_prefix, 'primary') - fout_prioritized=openScreeningFout(outdir,out_prefix, 'prioritized') + if use_existing_test_result==False: + fout_direct, fout_direct_name= openTestingFout(outdir, out_prefix, splicing_event_type, summary_file, panel_list, test_mode[0], 'guided') + fout_redirect, fout_redirect_name=openTestingFout(outdir, out_prefix, splicing_event_type, summary_file, panel_list, test_mode[0], 'voted') + if summary_file: + writeSummaryFile(out_prefix, splicing_event_type, db_dir, index, fout_direct, fetching_data_col) + exit() + fout_filtered=open(outdir+'/'+out_prefix+'.'+splicing_event_type+'.notest.txt','w') + else: + fout_direct_name=outdir+'/'+out_prefix+'.'+splicing_event_type+'.test.all_guided.txt' + fout_redirect_name=outdir+'/'+out_prefix+'.'+splicing_event_type+'.test.all_voted.txt' - for group_name in ref_list:##Load IRIS reference panels - fin_list[group_name]=db_dir+'/'+group_name+'/splicing_matrix/splicing_matrix.SE.cov10.'+group_name+'.txt' + fout_primary=openScreeningFout(outdir, out_prefix, splicing_event_type, 'tier1') + fout_prioritized=openScreeningFout(outdir,out_prefix, splicing_event_type, 'tier2tier3') + ##Load IRIS reference panels + for group_name in panel_list: + fin_list[group_name]=db_dir+'/'+group_name+'/splicing_matrix/splicing_matrix.'+splicing_event_type+'.cov10.'+group_name+'.txt' for group in fin_list.keys(): if not os.path.isfile(fin_list[group]+'.idx'): exit('[Error] Need to index '+fin_list[group]) index[group]=read_PsiMatrix_index(fin_list[group],'/'.join(fin_list[group].split('/')[:-1])) - has={} - #for j,k in enumerate(['ENSG00000083520:DIS3:chr13:-:73345041:73345126:73343050:73345218','ENSG00000110075:PPP6R3:chr11:+:68350510:68350597:68343511:68355265']): - tot=len(index[out_prefix])-1 - print '[INFO] IRIS screening started. Total input events:', tot+1 - for event_idx,k in enumerate(index[out_prefix]): - config.update_progress(event_idx/(0.0+tot)) - - for group in ref_list:#Initiate - if group!=out_prefix: - has[group]=True - psi={} - has_count=0 - for group in ref_list: - if k in index[group]: - psi[group]=map(float,fetch_PsiMatrix(k,fin_list[group],'.','\t',index[group])[1][8:]) - has_count+=1 - else: - has[group]=False - cat_psi=[] - for i in psi: - cat_psi+=psi[i] - if abs(max(cat_psi)-min(cat_psi))<0.05:#if change less than 5% skipped and no comparison available - fout_filtered.write('[LowVar]{}\t{}\t{}\n'.format(k,str(abs(max(cat_psi)-min(cat_psi))),str(has_count))) - continue - if k in blacklist_events: - fout_filtered.write('[Blacklisted]{}\t{}\t{}\n'.format(k,'-',str(has_count))) - continue - if has_count<=1: - fout_filtered.write('[NoTest]{}\t{}\t{}\n'.format(k,'-',str(has_count))) - continue + ## Load and perform test by row/event + if use_existing_test_result==False: + has={} + tot=len(index[out_prefix])-1 + print '[INFO] IRIS screen - started. Total input events:', tot+1 + + for event_idx,k in enumerate(index[out_prefix]): + config.update_progress(event_idx/(0.0+tot)) - if group_test: - test={} - query_mean=[np.nanmean(psi[out_prefix]),np.nanpercentile(psi[out_prefix],25),np.nanpercentile(psi[out_prefix],75)] - for j,group in enumerate(ref_list[1:]): - test[group]=['-']*3 - if has[group]: - if testing_type_index[j]=='equ': - psi_primary, direction= getDirection(filter1, psi, test) - if psi_primary==[]: #in case tissue-matched normal doesn't have data - continue - test[group]=groupTest(psi[group],psi_primary, testing_type_index[j], direction) - else: - test[group]=groupTest(psi[out_prefix],psi[group],testing_type_index[j]) - - result=[k]+query_mean+['\t'.join(map(str,test[t])) for t in ref_list if t!=out_prefix] - fout.write('\t'.join(map(str,result))+'\n') - - ## summarize test result and prioritze - pval=[test[t][0] for t in ref_list if t!=out_prefix] - deltaPSI=[test[t][1] for t in ref_list if t!=out_prefix]# select deltapsi col of screening result - foc=[test[t][2] for t in ref_list if t!=out_prefix] - - differential,equal,positive,negative,testable,primary_result, primary_result_foc, deltapsi_list,foc_list=summarizeTestResult(filter1_cutoff_pval, filter1_cutoff_dpsi, filter1_cutoff_foc,filter2_cutoff_pval, filter2_cutoff_dpsi, filter2_cutoff_foc, filter3_cutoff_pval, filter3_cutoff_dpsi, filter3_cutoff_foc, primary, tumor_rec, pval, deltaPSI, foc, testing_type_index) + #Initiate + for group in panel_list: + if group!=out_prefix: + has[group]=True + psi={} + has_count=0 + for group in panel_list: + if k in index[group]: + psi[group]=map(float,fetch_PsiMatrix(k,fin_list[group],'.','\t',index[group])[1][fetching_data_col:]) + has_count+=1 + else: + has[group]=False + #Filtering + cat_psi=[] + for i in psi: + cat_psi+=psi[i] + if abs(max(cat_psi)-min(cat_psi))<0.05:#if change less than 5% skipped and no comparison available + fout_filtered.write('[Low Range]{}\t{}\t{}\n'.format(k,str(abs(max(cat_psi)-min(cat_psi))),str(has_count))) + continue + if k in blacklist_events: + fout_filtered.write('[Blacklisted]{}\t{}\t{}\n'.format(k,'-',str(has_count))) + continue + if has_count<=1: + fout_filtered.write('[Unique in Input]{}\t{}\t{}\n'.format(k,'-',str(has_count))) + continue + if min_sample_count: + sample_count=np.count_nonzero(~np.isnan(psi[out_prefix])) + if sample_count=filter1_group_cutoff or filter1_group_cutoff=='') and (significant_tumor>=filter2_group_cutoff or filter2_group_cutoff=='') and (significant_normal>=filter3_group_cutoff or filter3_group_cutoff==''): + fout_cpm_sig.write(k+'\t'+'\t'.join(map(str,write_sj_list))+'\n') + sig_junction[k]='|'.join(map(str, [write_sj_list[0],significant_normal_match,significant_tumor,significant_normal])) + fout_cpm_count.write(k+'\t'+'\t'.join(map(str,write_sj_list))+'\n') + fout_cpm_count.close() + fout_cpm_sig.close() + + else: + print 'Use existing testing result.' + fout_cpm_count_name=outdir+'/CPM.'+out_prefix+'.'+splicing_event_type+'.test_all.txt' + for i, l in enumerate(open(fout_cpm_count_name)): + if i==0: + header=l.strip().split('\t') + group_list=map(lambda x: x.split('_CPM')[0], header[2::3]) + else: + ls=l.strip().split('\t') + if ls[1]=='NA': + continue + cpm_value= map(float,ls[2::3])# don't do map because '-' + change_value = map(float,ls[3::3]) + p_value= map(float,ls[4::3]) + significant_normal_match=0 + significant_normal=0 + significant_tumor=0 + #determine difference of a junction + for j,group in enumerate(group_list): + if group in filter1_panel_list: + if p_value[j]<=pvalue_cutoff_normal: + significant_normal_match+=1 + elif group in filter2_panel_list: + if p_value[j]<=pvalue_cutoff_tumor: + significant_tumor+=1 + else: + if p_value[j]<=pvalue_cutoff_normal: + significant_normal+=1 + if (significant_normal_match>=filter1_group_cutoff or filter1_group_cutoff=='') and (significant_tumor>=filter2_group_cutoff or filter2_group_cutoff=='') and (significant_normal>=filter3_group_cutoff or filter3_group_cutoff==''): + fout_cpm_sig.write(l.strip()+'\n') + sig_junction[ls[0]]=':'.join(map(str, [round(float(ls[1]),2),significant_normal_match,significant_tumor,significant_normal])) + fout_cpm_sig.close() + + #sig_junction=loadSigJunction(fout_cpm_sig_fname) + fout_summary_fname=summarizeSJ2ASevent(event_list_fin, splicing_event_type, sig_junction, outdir, out_prefix) + + +if __name__ == '__main__': + main() diff --git a/IRIS/IRIS_screening_novelss.py b/IRIS/IRIS_screening_novelss.py new file mode 100644 index 0000000..1224e55 --- /dev/null +++ b/IRIS/IRIS_screening_novelss.py @@ -0,0 +1,538 @@ +import numpy as np +import sys +import os, glob, pyBigWig, argparse +from scipy import stats +import statsmodels.stats.weightstats as smw +from . import config +import warnings +warnings.filterwarnings("ignore") + +def read_PsiMatrix_index(fn,outdir): + index = {} + for line in open(outdir+'/'+fn.split('/')[-1]+'.idx', 'r'): + ele = line.strip().split() + index[ele[0]] = int(ele[1]) + return index + +def fetch_PsiMatrix(eid, fn, delim, index=None): + with open(fn, 'r') as f: + ele = f.readline().strip().split(delim) + header = np.asarray([ x.split('.aln')[0] for x in ele ]) + f.seek(index[eid], 0) + data = np.asarray(f.readline().strip().split(delim)) + return (header, data) + +def read_SJMatrix_index(fn,outdir): + index = {} + for line in open(outdir+'/'+fn.split('/')[-1]+'.idx', 'r'): + ele = line.strip().split() + index[ele[0]] = int(ele[1]) + return index + +def fetch_SJMatrix(eid, fn, delim, index, head_only): + with open(fn, 'r') as f: + if head_only: + ele = f.readline().strip().split(delim) + retrieved_text = np.asarray([ x.split('.aln')[0] for x in ele ]) + else: + f.seek(index[eid], 0) + retrieved_text = np.asarray(f.readline().strip().split(delim)) + return retrieved_text + +def loadParametersRow(filter_para, panel_list): + if filter_para.strip()!='': + para, filter_panel_list=filter_para.split(' ') + filter_cutoff_pval, filter_cutoff_dpsi, filter_cutoff_foc, filter_cutoff_pval_PT, filter_group_cutoff =para.split(',') + filter_cutoff_pval=float(filter_cutoff_pval) + filter_cutoff_dpsi=float(filter_cutoff_dpsi) + filter_cutoff_foc=float(filter_cutoff_foc) + filter_group_cutoff=int(filter_group_cutoff) + filter_panel_list=filter_panel_list.split(',') + panel_list+=filter_panel_list + else: + filter_cutoff_pval, filter_cutoff_dpsi, filter_cutoff_foc, filter_group_cutoff, filter_panel_list =['','','','',[]] + return filter_cutoff_pval, filter_cutoff_dpsi, filter_cutoff_foc, filter_group_cutoff, filter_panel_list, panel_list + +def openTestingFout(outdir, out_prefix, splicing_event_type, summary_file, panel_list, fin_name=''): + header=['as_event','meanPSI','Q1PSI','Q3PSI'] + header_prefix=['_pVal','_deltaPSI','_tumorFC'] + fout_name=outdir+'/'+out_prefix+'.'+splicing_event_type+'.test_JCRA.all_'+fin_name+'.txt' + fout=open(fout_name,'w') + if summary_file==False: + header+=['\t'.join(map(lambda x:ref+x ,header_prefix)) for ref in panel_list if ref!=out_prefix] + fout.write('\t'.join(header)+'\n') + return fout, fout_name + +def openScreeningFout(outdir, out_prefix, splicing_event_type, fout_name): + fout=open(outdir+'/'+out_prefix+'.'+splicing_event_type+'.'+fout_name+'.txt','w') + fout.write('{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\n'.format('as_event','meanPSI','Q1PSI','Q3PSI','deltaPSI','fc_of_tumor_isoform','tissue_matched_normal_panel','tumor_panel','normal_panel','tag','mappability','mappability_tag','novel_ss_info')) + return fout + +def loadMappability(bigwig_fin): + bw_map=pyBigWig.open(bigwig_fin) + d=45 + return bw_map,d + +def getMappability(splicing_event,bw_map,d): + arr=splicing_event.split(':') + chrom,strand,start,end,up,down=arr[2],arr[3],int(arr[4]),int(arr[5]),int(arr[6]),int(arr[7]) + up_mean=bw_map.stats("%s"%chrom,up-d,up,type="mean")[0] + down_mean=bw_map.stats("%s"%chrom,down,down+d,type="mean")[0] + if abs(start-end)<2*d: + target_mean=bw_map.stats("%s"%chrom,start,end,type="mean")[0] + else: + target_left=bw_map.stats("%s"%chrom,start,start+d,type="mean")[0] + target_right=bw_map.stats("%s"%chrom,end-d,end,type="mean")[0] + target_mean=(target_right+target_left)/2 + if strand=='-': #switch the order + li=[up_mean,down_mean] + up_mean,down_mean=li[1],li[0] + + mappability=[str(up_mean),str(target_mean),str(down_mean)] + return mappability + +def loadGTF(gtf): + exon_start_dict={} + exon_end_dict={} + for l in open(gtf): + if l.startswith('#'): + continue + ls=l.strip().split('\t') + if ls[2]=='exon': + chrom=ls[0] + if chrom.startswith('chr')==False: + chrom='chr'+chrom + exon_start_dict[ls[6]+':'+chrom+':'+ls[3]]='' + exon_end_dict[ls[6]+':'+chrom+':'+ls[4]]='' + return exon_start_dict, exon_end_dict + +def selectJunction_forGTF(AS_coord, deltaPSI_c2n, cut_off, if_select_all, splicing_event_type): #For A3 and A5, inc1/2 are not considered as current sj db doesn't capture that(and it will not always generate novel sequence). May update later. + if splicing_event_type == 'SE': + # end_pos, start_pos; because rMATS is 0-based for start exon, it should +1 when compare to GTF + skp = (AS_coord[2], str(int(AS_coord[3])+1),'skp') + inc1 = (AS_coord[2], str(int(AS_coord[0])+1),'inc1') + inc2 = (AS_coord[1], str(int(AS_coord[3])+1),'inc2') + elif splicing_event_type == 'A3SS': + skp = (AS_coord[5], str(int(AS_coord[2])+1),'skp') + inc1 = (AS_coord[5],str(int(AS_coord[0])+1),'inc1') + inc2 = ''#(AS_coord[2],AS_coord[2],'inc2') + elif splicing_event_type == 'A5SS': + skp = (AS_coord[3], str(int(AS_coord[4])+1),'skp') + inc1 = ''#(AS_coord[3],AS_coord[3],'inc1') + inc2 = (AS_coord[1],str(int(AS_coord[4])+1),'inc2') + elif splicing_event_type == 'RI': + skp = ''#(AS_coord[3], AS_coord[4],'skp')#???? + inc1 = ''#(AS_coord[3],AS_coord[3],'inc1') + inc2 = ''#(AS_coord[4],AS_coord[4],'inc2') + if if_select_all: + return [skp, inc1, inc2] + if float(deltaPSI_c2n) < cut_off: # tumor skipping + return [skp] + else: + return [inc1, inc2] + +def findNovelSS(event_name, deltaPSI_c2n, cut_off, splicing_event_type, exon_start_dict, exon_end_dict):# This is a conservative def of novelSS than rMATS4.1 (0.4% events less- complex cases ) + es=event_name.split(':') + chrom=es[2] + strand=es[3] + AS_coord=es[4:] + selected=selectJunction_forGTF(AS_coord,deltaPSI_c2n, cut_off, False, splicing_event_type) + info=[] + for j in selected: + if j!='': + check1=strand+':'+chrom+':'+j[0] not in exon_end_dict + check2=strand+':'+chrom+':'+j[1] not in exon_start_dict + if check1 or check2: + info.append(j[2])## + if info==[]: + info.append('none') + return ';'.join(info) + +def readEventRow(row, header_line): + if header_line=='' or header_line==False: + rs=row.strip().split('\t') + return rs + else: + rs=row.strip().split('\t') + return dict(zip(header_line, rs)) + +def convert2SJevent(line_dict, splicing_event_type):#match to SJ db (different from gtf and rMATS - 0-based start; +1 for end exon) + if splicing_event_type=='SE': + event_row_list=[line_dict['chr']+':'+str(int(line_dict['upstreamEE'])+1)+':'+line_dict['exonStart'],line_dict['chr']+':'+str(int(line_dict['exonEnd'])+1)+':'+line_dict['downstreamES'], line_dict['chr']+':'+str(int(line_dict['upstreamEE'])+1)+':'+line_dict['downstreamES']] + as_event=line_dict['AC'].strip('"').split('.')[0]+':'+line_dict['GeneName'].strip('"')+':'+line_dict['chr']+':'+line_dict['strand']+':'+line_dict['exonStart']+':'+line_dict['exonEnd']+':'+line_dict['upstreamEE']+':'+line_dict['downstreamES'] + elif splicing_event_type=='A5SS':# Only use one junction for inc. Need to improve by updating db later + event_row_list=[line_dict['chr']+':'+str(int(line_dict['longExonEnd'])+1)+':'+line_dict['flankingES'],line_dict['chr']+':'+str(int(line_dict['shortEE'])+1)+':'+line_dict['flankingES']] + as_event=line_dict['AC'].strip('"').split('.')[0]+':'+line_dict['GeneName'].strip('"')+':'+line_dict['chr']+':'+line_dict['strand']+':'+line_dict['longExonStart']+':'+line_dict['longExonEnd']+':'+line_dict['shortES']+':'+line_dict['shortEE']+':'+line_dict['flankingES']+':'+line_dict['flankingEE'] + elif splicing_event_type=='A3SS': # Only use one junction for inc. Need to improve by updating db later + event_row_list=[line_dict['chr']+':'+str(int(line_dict['flankingEE'])+1)+':'+line_dict['longExonStart'],line_dict['chr']+':'+str(int(line_dict['flankingEE'])+1)+':'+line_dict['shortES']] + as_event=line_dict['AC'].strip('"').split('.')[0]+':'+line_dict['GeneName'].strip('"')+':'+line_dict['chr']+':'+line_dict['strand']+':'+line_dict['longExonStart']+':'+line_dict['longExonEnd']+':'+line_dict['shortES']+':'+line_dict['shortEE']+':'+line_dict['flankingES']+':'+line_dict['flankingEE'] + else: + exit('splicine event type not supported. Exiting.') + return event_row_list, as_event + +def getDirection(panel_dict, psi, test, non_parametric): + psi_primary=[] + deltaPSI_primary=[] + for primary_group in panel_dict: #check if matching normal + if test[primary_group]!=['-']*3: + psi_primary+=psi[primary_group] + deltaPSI_primary.append(test[primary_group][1]) + if psi_primary==[]: #in case tissue-matched normal doesn't have data + return [],[''] + else: + direction='greater' if non_parametric else 'larger' + if np.median(deltaPSI_primary)<=0: + direction='less' if non_parametric else 'smaller' + return psi_primary, direction + + +def calcTumorFormFoc(delta_psi, mean_psi): + if delta_psi>0: + return mean_psi/(mean_psi - delta_psi+10**-8) + elif delta_psi<0: + return (1- mean_psi)/ (1- mean_psi+ delta_psi+10**-8) + else: + return 1 + +def statTest(g1,g2, direction, non_parametric): + if direction != 'equivalence': + if non_parametric: + pvalue=stats.mannwhitneyu(g1,g2, alternative=direction)[1] + else: + pvalue=smw.ttest_ind(g1,g2, alternative=direction, usevar='unequal')[1] + #pvalue=smw.ttest_ind(g1,g2, alternative=direction)[1] + else: + threshold_tost = 0.05 + pvalue=smw.ttost_ind(g1,g2,-threshold_tost,threshold_tost,usevar='unequal')[0] #equivalence test + return pvalue + +def statTest_minSampleCount(g1,g2, direction, non_parametric):#Only enabled when filters out by min_sample_count. With enough sample, using the default setting assume equal var for both groups is ok. + if direction != 'equivalence': + if non_parametric: + pvalue=stats.mannwhitneyu(g1,g2, alternative=direction)[1] + else: + pvalue=smw.ttest_ind(g1,g2, alternative=direction)[1] + #pvalue=smw.ttest_ind(g1,g2, alternative=direction)[1] + else: + threshold_tost = 0.05 + pvalue=smw.ttost_ind(g1,g2,-threshold_tost,threshold_tost)[0] #equivalence test + return pvalue + +def groupTest(g1,g2, non_parametric=False, direction='two-sided', min_sample_count=False): + g1=np.array(g1) + g2=np.array(g2) + g1=g1[~np.isnan(g1)] + g2=g2[~np.isnan(g2)] + delta_psi=np.nanmean(g1)-np.nanmean(g2) + tumor_foc=calcTumorFormFoc(delta_psi,np.nanmean(g1)) + if min_sample_count: + pvalue = statTest_minSampleCount(g1, g2, direction, non_parametric) + else: + pvalue = statTest(g1, g2, direction, non_parametric) + return [pvalue, delta_psi, tumor_foc] + +def performTest(group, matching_norm_dict, tumor_dict, normal_dict, psi, out_prefix, non_parametric, psi_primary, direction): + redirect_output = False + test_result = ['-']*3 #For missing in non-eesential tests/comparisons + has_matched_tumor = False if psi_primary == [] else True #for clarity + if group in matching_norm_dict: + if psi[group]!=[]: + test_result = groupTest(psi[out_prefix],psi[group], non_parametric,"two-sided") + return test_result + + elif group in tumor_dict: + if has_matched_tumor:#set_matched_tumor is redundent here. kept for future implemtation of additional output type. + if psi[group]!=[]: + test_result = groupTest(psi[group],psi_primary, non_parametric, direction) + else: + if psi[group]!=[]: + test_result = groupTest(psi[out_prefix],psi[group], non_parametric, "equivalence")#No or equivalent testing + return test_result + + elif group in normal_dict: + if has_matched_tumor: + if psi[group]!=[]: + test_result = groupTest(psi[out_prefix],psi[group], non_parametric, direction) + else: + if psi[group]!=[]: + test_result = groupTest(psi[out_prefix],psi[group], non_parametric, "two-sided") #Two-sided testing + return test_result + else: + exit('error in group.') + + +def summarizeTestResult(filter1_cutoff_pval, filter1_cutoff_dpsi, filter1_cutoff_foc,filter2_cutoff_pval, filter2_cutoff_dpsi, filter2_cutoff_foc, filter3_cutoff_pval, filter3_cutoff_dpsi, filter3_cutoff_foc, pval, deltaPSI, foc, panel_list, matching_norm_dict, tumor_dict, normal_dict): + association_passed,recurrence_passed,specificity_positive,specificity_negative, specificity_testable=[0,0,0,0,0] + primary_result, primary_result_foc=[[],[]] #take care of multiple tiisue matched norm + deltapsi_list_voted,foc_list_voted=[[],[]] #if no tissue-matched norm, use median + for i,group in enumerate(panel_list[1:]): + if pval[i]=='-': #This is important - skip all missing, which is not useful for summarizing but not affecting consistancy. + continue + if group in matching_norm_dict: + primary_result.append(float(deltaPSI[i])) + primary_result_foc.append(float(foc[i])) + if float(pval[i])<=filter1_cutoff_pval and abs(float(deltaPSI[i]))>=filter1_cutoff_dpsi and float(foc[i])>=filter1_cutoff_foc: + association_passed+=1 + continue + elif group in tumor_dict: + if float(pval[i])<=filter2_cutoff_pval and abs(float(deltaPSI[i]))>=filter2_cutoff_dpsi: + recurrence_passed+=1 + continue + elif group in normal_dict:# TODO: judge set/has, then run + specificity_testable+=1 + if float(pval[i])<=filter3_cutoff_pval and float(foc[i])>=filter3_cutoff_foc: + deltapsi_list_voted.append(float(deltaPSI[i])) + foc_list_voted.append(float(foc[i])) + if float(deltaPSI[i])>=filter3_cutoff_dpsi: + specificity_positive+=1 + continue + if float(deltaPSI[i])<=-filter3_cutoff_dpsi: + specificity_negative+=1 + continue + return association_passed,recurrence_passed,specificity_positive,specificity_negative,specificity_testable, np.median(primary_result), np.median(primary_result_foc), deltapsi_list_voted,foc_list_voted + +def defineTumorEvents(filter1_group_cutoff,filter2_group_cutoff,filter3_group_cutoff, matching_norm_dict, specificity_panel_len, association_passed, recurrence_passed, specificity_positive, specificity_negative, specificity_testable, primary_result, primary_result_foc, deltapsi_list_voted, foc_list_voted, use_ratio): + tag=[] + if filter1_group_cutoff!='': + if association_passed>=filter1_group_cutoff:#Improvement? current: 0>='' is false + tag.append('associated') + if filter2_group_cutoff!='': + if recurrence_passed>=filter2_group_cutoff: + tag.append('recurrent') + if matching_norm_dict!={}:#TODO-FUTURE: take care of set-yes has-no redirected events + if primary_result>0: + tissue_specificity=specificity_positive + ratio=False if use_ratio==False else specificity_positive/(specificity_testable+10**-8)>=filter3_group_cutoff/(specificity_panel_len+0.0) + if specificity_positive>=filter3_group_cutoff or ratio: + tag.append('high_assoc') + else: + tissue_specificity=specificity_negative + ratio=False if use_ratio==False else specificity_negative/(specificity_testable+10**-8)>=filter3_group_cutoff/(specificity_panel_len+0.0) + if specificity_negative>=filter3_group_cutoff or ratio: + tag.append('high_assoc') + else: + tissue_specificity=max(specificity_positive,specificity_negative) + ratio=False if use_ratio==False else tissue_specificity/(specificity_testable+10**-8)>=filter3_group_cutoff/(specificity_panel_len+0.0) + if tissue_specificity>=filter3_group_cutoff or ratio: + tag.append('high_assoc') + primary_deltapsi=primary_result if primary_result!=0 else np.median(deltapsi_list_voted)#TODO + primary_foc=primary_result_foc if primary_result!=0 else np.median(foc_list_voted) + return primary_deltapsi, primary_foc, tissue_specificity, tag + +def mappability_write(k, bw_map, calc_length): + mappability_list=getMappability(k, bw_map, calc_length) + mappability_tag='PASS' + min_map_score=min(map(float, mappability_list)) + if min_map_score<0.8: + mappability_tag='MID' + if min_map_score<0.6:#from 0.7 + mappability_tag='LOW' + return mappability_tag, mappability_list + +def loadBlacklistEvents(fin): + BlacklistEvents={} + for l in open(fin): + ls=l.strip().split('\t') + des=ls[0].split(':') + ensg=des[0].split('_')[0].split('.')[0] + des=':'.join([ensg]+des[1:]).strip(':') + BlacklistEvents[des]='' + return BlacklistEvents + +def translationCMD(ref_genome, gtf, outdir, out_prefix, splicing_event_type, all_orf, ignore_annotation, remove_early_stop, fout_name, find_novel_ss): + #uSE name space + argument_line='' + if all_orf: + argument_line+=' --all-orf' + if ignore_annotation: + argument_line+=' --ignore-annotation' + if remove_early_stop: + argument_line+=' --remove-early-stop' + if find_novel_ss: + argument_line+=' --check-novel' + cmd_translation='IRIS translate '+outdir+'/'+out_prefix+'.'+splicing_event_type+'.'+fout_name+'.txt '+' -o '+outdir+'/'+splicing_event_type+'.'+fout_name+' -g '+ref_genome+' -t '+splicing_event_type+argument_line+' --gtf '+gtf + print '[INFO] Working on translating: '+fout_name + os.system(cmd_translation) + +def main(args): + ###Loading Parameters#### + para_fin=args.parameter_fin + splicing_event_type=args.splicing_event_type + event_list_fin=args.event_list_fin + fetching_sj_col=1 + out_prefix,db_dir,filter1_para,filter2_para,filter3_para,test_mode,use_ratio,blacklist_path,mappability_path,ref_genome=[l.strip() for l in open(para_fin)] + panel_list=[out_prefix] + test_mode=test_mode.split(' ') + use_ratio=True if use_ratio=='True' else False + blacklist_events={} + if blacklist_path!='': + blacklist_events=loadBlacklistEvents(blacklist_path) + bw_map,calc_length=loadMappability(mappability_path) + deltaPSI_cutoff=args.deltaPSI_cut_off + find_novel_ss=True if args.report_known_and_novelss_tumor_junction==False else False + if find_novel_ss: + gtf=args.gtf + exon_start_dict, exon_end_dict= loadGTF(gtf) + + all_orf=args.all_orf + ignore_annotation=args.ignore_annotation + remove_early_stop=args.remove_early_stop + use_existing_test_result=args.use_existing_test_result + + filter1_cutoff_pval, filter1_cutoff_dpsi, filter1_cutoff_foc, filter1_group_cutoff, filter1_panel_list, panel_list = loadParametersRow(filter1_para, panel_list) + filter2_cutoff_pval, filter2_cutoff_dpsi, filter2_cutoff_foc, filter2_group_cutoff, filter2_panel_list, panel_list = loadParametersRow(filter2_para, panel_list) + filter3_cutoff_pval, filter3_cutoff_dpsi, filter3_cutoff_foc, filter3_group_cutoff, filter3_panel_list, panel_list = loadParametersRow(filter3_para, panel_list) + if filter1_panel_list==[] and filter2_panel_list==[] and filter3_panel_list==[]: + # if filter1_panel_list==[] and filter2_panel_list==[] and filter3_panel_list==[] and test_mode[0]!='summary': + exit("[Error] No filtering required in parameteres file. exit!") + summary_file=False + + non_parametric=False + if len(test_mode)>1: + non_parametric=True if test_mode[1]=='nonparametric' else False + + ##this block is 2020 code; different and improved from screening.py + matching_norm_dict=dict.fromkeys(filter1_panel_list,'') + tumor_dict=dict.fromkeys(filter2_panel_list,'') + normal_dict=dict.fromkeys(filter3_panel_list,'') + tumor_dict[out_prefix]='' + association_panel_len=len(matching_norm_dict) + recurrence_panel_len=len(tumor_dict)-1 + specificity_panel_len=len(normal_dict) + + panel_count=sum(1 for i in [association_panel_len, recurrence_panel_len, specificity_panel_len] if i!=0) + + if args.translating: + gtf=args.gtf + if os.path.exists(gtf)==False: + exit('[Error] No gtf file provided for translation. exit!') + ###Create Folders/Output#### + outdir=args.outdir.rstrip('/') + os.system('mkdir -p '+outdir) + db_dir=db_dir.rstrip('/') + + if use_existing_test_result==False: + fout_direct, fout_direct_name= openTestingFout(outdir, out_prefix, splicing_event_type, summary_file, panel_list, 'guided') + fout_redirect, fout_redirect_name=openTestingFout(outdir, out_prefix, splicing_event_type, summary_file, panel_list, 'voted') + else: + fout_direct_name=outdir+'/'+out_prefix+'.'+splicing_event_type+'.test_JCRA.all_guided.txt' + fout_redirect_name=outdir+'/'+out_prefix+'.'+splicing_event_type+'.test_JCRA.all_voted.txt' + + fout_primary=openScreeningFout(outdir, out_prefix, splicing_event_type, 'tier1_JCRA') + fout_prioritized=openScreeningFout(outdir,out_prefix, splicing_event_type, 'tier2tier3_JCRA') + + ##Load IRIS reference panels to 'fin_list', 'index' + index={} + fin_list={} + for group_name in panel_list: + fin_list[group_name]=db_dir+'/'+group_name.split('_')[0]+'/SJ_count.'+group_name+'.txt' + for group in fin_list.keys(): + if not os.path.isfile(fin_list[group]+'.idx'): + exit('[Error] Need to index '+fin_list[group]) + index[group]=read_SJMatrix_index(fin_list[group],'/'.join(fin_list[group].split('/')[:-1])) + + tot=config.file_len(event_list_fin)-1 + print '[INFO] IRIS screen_novelss - started. Total input events:', tot+1 + ## Load and perform test by row/event + sample_size={} + for group in panel_list: + random_key=index[group].keys()[0] + sample_names=map(str,fetch_SJMatrix(random_key,fin_list[group],'\t',index[group],True)[fetching_sj_col:]) + sample_size[group]=len(sample_names) + + if use_existing_test_result==False: + header_list=[] + junction_dict={} + for event_idx, event_row in enumerate(open(event_list_fin)): + if event_idx==0: + header_list=readEventRow(event_row,'') + continue + line_dict=readEventRow(event_row, header_list) + event_row_list, as_event=convert2SJevent(line_dict, splicing_event_type) + config.update_progress(event_idx/(0.0+tot)) + + sj={} + for eid, k in enumerate(event_row_list): + sj[eid]={} + # if k not in junction_dict: + # junction_dict[k]='' + # else: + # continue + + #Initiate psi matrix by each row to 'sj' + for group in panel_list: + if k in index[group]: + sj[eid][group]=map(int,fetch_SJMatrix(k,fin_list[group],'\t',index[group], False)[fetching_sj_col:]) + else: + sj[eid][group]=[0]*sample_size[group] + psi={} + for group in panel_list: + psi[group]=[] + for sid in xrange(0,sample_size[group]): + inc1=sj[0][group][sid] + inc2=sj[1][group][sid] + skp=sj[2][group][sid] + if inc1+ inc2+ skp>=10: + psi[group].append(((inc1+inc2)/2.0)/(((inc1+inc2)/2.0)+skp)) + + test={} + redirect=False + psi_primary='' + direction='' + query_mean=map(lambda x:round(x,2),[np.nanmean(psi[out_prefix]),np.nanpercentile(psi[out_prefix],25),np.nanpercentile(psi[out_prefix],75)]) + for group in panel_list[1:]: + if group not in matching_norm_dict and direction=='':#redirect or find one-sided test direction + psi_primary, direction= getDirection(matching_norm_dict, psi, test, non_parametric) + redirect = True if psi_primary == [] else False #redirect if tumor-matched normal is missing + test[group]=performTest(group, matching_norm_dict, tumor_dict, normal_dict, psi, out_prefix, non_parametric, psi_primary, direction) + result=[as_event]+query_mean+['\t'.join(map(str,test[t])) for t in panel_list if t!=out_prefix] + + if redirect: + fout_redirect.write('\t'.join(map(str,result))+'\n') #to redicted; summarize direction and calculate FDR differently. + else: + fout_direct.write('\t'.join(map(str,result))+'\n') + + fout_redirect.close() + fout_direct.close() + + testing_intermediate_file = fout_direct_name if matching_norm_dict!={} else fout_redirect_name + + tot=config.file_len(testing_intermediate_file)-1 + print '[INFO] IRIS screen_novelss - summarizing. Total events from last step:', tot + for event_idx,l in enumerate(open(testing_intermediate_file)): + config.update_progress(event_idx/(0.0+tot)) + if event_idx==0: + continue + ls=l.strip().split('\t') + pval= ls[4::3]# don't do map because '-' + deltaPSI = ls[5::3] + foc= ls[6::3] + + association_passed,recurrence_passed,specificity_positive,specificity_negative,specificity_testable,primary_result, primary_result_foc, deltapsi_list_voted,foc_list_voted=summarizeTestResult(filter1_cutoff_pval, filter1_cutoff_dpsi, filter1_cutoff_foc,filter2_cutoff_pval, filter2_cutoff_dpsi, filter2_cutoff_foc, filter3_cutoff_pval, filter3_cutoff_dpsi, filter3_cutoff_foc, pval, deltaPSI, foc, panel_list, matching_norm_dict, tumor_dict, normal_dict) + + primary_deltapsi, primary_foc, tissue_specificity, tag = defineTumorEvents(filter1_group_cutoff,filter2_group_cutoff,filter3_group_cutoff, matching_norm_dict, specificity_panel_len, association_passed, recurrence_passed, specificity_positive, specificity_negative, specificity_testable, primary_result, primary_result_foc, deltapsi_list_voted, foc_list_voted, use_ratio) + novel_ss_status='disabled' + if find_novel_ss: + novel_ss_status=findNovelSS(ls[0], primary_deltapsi, deltaPSI_cutoff, splicing_event_type, exon_start_dict, exon_end_dict) + if novel_ss_status=='none': + continue + if tag!=[]: + if tag[0]=='associated': + mappability_tag, mappability_list=mappability_write(ls[0], bw_map, calc_length) + fout_primary.write('{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\n'.format(ls[0],ls[1],ls[2],ls[3],primary_deltapsi, primary_foc, str(association_passed)+'/'+str(association_panel_len),str(recurrence_passed)+'/'+str(recurrence_panel_len),str(tissue_specificity)+'/'+str(specificity_panel_len),';'.join(tag), ';'.join(mappability_list),mappability_tag, novel_ss_status)) + if panel_count==len(tag):#Modified 2021 + mappability_tag, mappability_list=mappability_write(ls[0], bw_map, calc_length) #Modified 2021 + fout_prioritized.write('{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\t{}\n'.format(ls[0],ls[1],ls[2],ls[3],primary_deltapsi, primary_foc, str(association_passed)+'/'+str(association_panel_len),str(recurrence_passed)+'/'+str(recurrence_panel_len),str(tissue_specificity)+'/'+str(specificity_panel_len),';'.join(tag),';'.join(mappability_list),mappability_tag, novel_ss_status)) + mappability_tag, mappability_list=['',''] #clear + + fout_primary.close() + fout_prioritized.close() + + ##Translation + if args.translating: + translationCMD(ref_genome, gtf, outdir, out_prefix, splicing_event_type, all_orf, ignore_annotation, remove_early_stop, 'tier1_JCRA', find_novel_ss) + translationCMD(ref_genome, gtf, outdir, out_prefix, splicing_event_type, all_orf, ignore_annotation, remove_early_stop, 'tier2tier3_JCRA', find_novel_ss) + + +if __name__ == '__main__': + main() diff --git a/IRIS/IRIS_screening_plot.py b/IRIS/IRIS_screening_plot.py index 663f1e2..f18ce92 100644 --- a/IRIS/IRIS_screening_plot.py +++ b/IRIS/IRIS_screening_plot.py @@ -71,50 +71,54 @@ def fileLength(fin): i=sum(1 for l in open(fin)) return i -def indiviualPlot(psi_df, event_id, ref_list): +def indiviualPlot(psi_df, event_id, panel_list, outdir): plt.figure(figsize=(15,9)) sns_plot=sns.violinplot(data=psi_df, inner="box",cut=0) #sns_plot.set_yticks(np.arange(0,1,0.2)) sns_plot.set(ylim=(0, 1)) sns.despine(offset=10, trim=True) sns_plot.set_ylabel('Percent-Spliced-In',fontweight='bold',fontsize=14) - sns_plot.set_xticklabels(ref_list,rotation=20,ha='right',fontsize=12) + sns_plot.set_xticklabels(panel_list,rotation=20,ha='right',fontsize=12) sns_plot.set_title(event_id,fontsize=15,fontweight='bold') - sns_plot.figure.savefig(event_id+".pdf") + sns_plot.figure.savefig(outdir+'/'+event_id+".pdf") -def loadParametersRow(filter_para, ref_list): - if len(filter_para.split(' '))==6: - filter_cutoff_pval, filter_cutoff_dpsi, filter_cutoff_foc, filter_group_cutoff, filter_list =filter_para.split(' ')[1:] +def loadParametersRow(filter_para, panel_list): + if filter_para.strip()!='': + para, filter_panel_list=filter_para.split(' ') + filter_cutoff_pval, filter_cutoff_dpsi, filter_cutoff_foc, filter_cutoff_pval_PT, filter_group_cutoff =para.split(',') filter_cutoff_pval=float(filter_cutoff_pval) filter_cutoff_dpsi=float(filter_cutoff_dpsi) filter_cutoff_foc=float(filter_cutoff_foc) filter_group_cutoff=int(filter_group_cutoff) - filter_list=filter_list.split(',') - ref_list+=filter_list + filter_panel_list=filter_panel_list.split(',') + panel_list+=filter_panel_list else: - filter_cutoff_pval, filter_cutoff_dpsi, filter_cutoff_foc, filter_group_cutoff, filter_list =['','','','',[]] - return filter_cutoff_pval, filter_cutoff_dpsi, filter_cutoff_foc, filter_group_cutoff, filter_list, ref_list + filter_cutoff_pval, filter_cutoff_dpsi, filter_cutoff_foc, filter_group_cutoff, filter_panel_list =['','','','',[]] + return filter_cutoff_pval, filter_cutoff_dpsi, filter_cutoff_foc, filter_group_cutoff, filter_panel_list, panel_list ## screening def main(args): index={} fin_list={} - para_fin=args.parameter_fin - fin_plot_query=args.event_list #to the txt file - step=int(args.step) - has_header=args.header - + para_fin=args.parameter_file + splicing_event_type=args.splicing_event_type + fetching_data_col=8 if splicing_event_type == 'SE' else 10 + fin_plot_query=args.event_list#for plot + step=int(args.step)#for plot + has_header=args.header#for plot out_prefix,db_dir,filter1_para,filter2_para,filter3_para,test_mode,use_ratio,blacklist_path,mappability_path,ref_genome=[l.strip() for l in open(para_fin)] - ref_list=[out_prefix] - filter1_cutoff_pval, filter1_cutoff_dpsi, filter1_cutoff_foc, filter1_group_cutoff, filter1, ref_list =loadParametersRow(filter1_para, ref_list) - filter2_cutoff_pval, filter2_cutoff_dpsi, filter2_cutoff_foc, filter2_group_cutoff, filter2, ref_list =loadParametersRow(filter2_para, ref_list) - filter3_cutoff_pval, filter3_cutoff_dpsi, filter3_cutoff_foc, filter3_group_cutoff, filter3, ref_list =loadParametersRow(filter3_para, ref_list) - if filter1==[] and filter2==[] and filter3==[] and test_mode!='summary': + panel_list=[out_prefix] + test_mode=test_mode.split(' ') + filter1_cutoff_pval, filter1_cutoff_dpsi, filter1_cutoff_foc, filter1_group_cutoff, filter1, panel_list =loadParametersRow(filter1_para, panel_list) + filter2_cutoff_pval, filter2_cutoff_dpsi, filter2_cutoff_foc, filter2_group_cutoff, filter2, panel_list =loadParametersRow(filter2_para, panel_list) + filter3_cutoff_pval, filter3_cutoff_dpsi, filter3_cutoff_foc, filter3_group_cutoff, filter3, panel_list =loadParametersRow(filter3_para, panel_list) + if filter1==[] and filter2==[] and filter3==[] and test_mode[0]!='summary': exit("no filtering required in para file. exit!") - group_test=False if test_mode!='group' else True - individual_test=False if test_mode!='personalized' else True - summary_file=False if test_mode!='summary' else True + + group_test=False if test_mode[0]!='group' else True + individual_test=False if test_mode[0]!='personalized' else True + summary_file=False if test_mode[0]!='summary' else True if [group_test,individual_test,summary_file]==[False,False,False]: exit('Need to choose one mode.exit!') single_plot=False @@ -122,11 +126,15 @@ def main(args): if group_test==individual_test: exit('can only choose one mode') - db_dir=db_dir - ref_list=list([out_prefix]+filter1+filter2+filter3) + db_dir=db_dir.rstrip('/') + outdir=args.outdir.rstrip('/') + os.system('mkdir -p '+outdir) + + panel_list=list([out_prefix]+filter1+filter2+filter3) - for group_name in ref_list:##Load IRIS reference panels - fin_list[group_name]=db_dir+'/'+group_name+'/splicing_matrix/splicing_matrix.SE.cov10.'+group_name+'.txt' + ##Load IRIS reference panels + for group_name in panel_list: + fin_list[group_name]=db_dir+'/'+group_name+'/splicing_matrix/splicing_matrix.'+splicing_event_type+'.cov10.'+group_name+'.txt' for group in fin_list.keys(): if not os.path.isfile(fin_list[group]+'.idx'): @@ -136,7 +144,7 @@ def main(args): filered=0 has={} - for group in ref_list: + for group in panel_list: if group!=out_prefix: has[group]=True @@ -149,7 +157,7 @@ def main(args): start_point=1 for r_start in xrange(start_point,fin_plot_query_len,step): if group_plot: - pdf = PdfPages(fin_plot_query+'.'+str(r_start)+'.pdf') + pdf = PdfPages(outdir+'/'+fin_plot_query.split('/')[-1]+'.'+str(r_start)+'.psiPlot.pdf') fig = plt.figure(figsize=(8.8,12)) fig.suptitle('', fontsize=14,fontweight='bold') sns.set(style="white", color_codes=True) @@ -161,11 +169,11 @@ def main(args): continue if j > r_start+step-1: break - psi[out_prefix]=map(float,fetch_PsiMatrix(event_id,fin_list[out_prefix],'.','\t',index[out_prefix])[1][8:]) + psi[out_prefix]=map(float,fetch_PsiMatrix(event_id,fin_list[out_prefix],'.','\t',index[out_prefix])[1][fetching_data_col:]) - for group in ref_list: + for group in panel_list: try: - psi[group]=map(float,fetch_PsiMatrix(event_id,fin_list[group],'.','\t',index[group])[1][8:]) + psi[group]=map(float,fetch_PsiMatrix(event_id,fin_list[group],'.','\t',index[group])[1][fetching_data_col:]) except: has[group]=False psi[group]=[] @@ -176,9 +184,9 @@ def main(args): filered+=1 continue - psi_df = pd.DataFrame.from_dict(psi, orient='index').transpose()[ref_list] + psi_df = pd.DataFrame.from_dict(psi, orient='index').transpose()[panel_list] if single_plot: - indiviualPlot(psi_df, event_id, ref_list) + indiviualPlot(psi_df, event_id, panel_list) if group_plot: print step, j ax_i = plt.subplot2grid((step,11), (j-r_start*1,0), colspan=10, rowspan=1) @@ -187,11 +195,15 @@ def main(args): sns_plot.set_yticks(np.arange(0,1.1,0.5)) sns_plot.set(xticklabels=[]) sns.despine(offset=0, trim=False) - sns_plot.text(15.7, 0.4, event_id.split(':')[1], horizontalalignment='left', size='large', color='black',fontweight='bold') + ax_j = plt.subplot2grid((step,11), (j-r_start*1,10), colspan=1, rowspan=1) + ax_j.text(0.5, 0.5, event_id.split(':')[1], horizontalalignment='center', size='large', color='black',fontweight='bold') + #ax_j.get_xaxis().set_visible(False) + #ax_j.get_yaxis().set_visible(False) + ax_j.axis('off') if group_plot: pdf.savefig(fig) pdf.close() - plt.savefig('{}.png'.format(fin_plot_query)) + #plt.savefig('{}.png'.format(fin_plot_query)) if __name__ == '__main__': diff --git a/IRIS/IRIS_screening_sjc.py b/IRIS/IRIS_screening_sjc.py new file mode 100644 index 0000000..5bfc771 --- /dev/null +++ b/IRIS/IRIS_screening_sjc.py @@ -0,0 +1,276 @@ +import numpy as np +import sys +import os, glob, pyBigWig, argparse +from scipy import stats +import statsmodels.stats.weightstats as smw +from . import config +import warnings +warnings.filterwarnings("ignore") + +def read_SJMatrix_index(fn,outdir): + index = {} + for line in open(outdir+'/'+fn.split('/')[-1]+'.idx', 'r'): + ele = line.strip().split() + index[ele[0]] = int(ele[1]) + return index + +def fetch_SJMatrix(eid, fn, delim, index, head_only): + with open(fn, 'r') as f: + if head_only: + ele = f.readline().strip().split(delim) + retrieved_text = np.asarray([ x.split('.aln')[0] for x in ele ]) + else: + f.seek(index[eid], 0) + retrieved_text = np.asarray(f.readline().strip().split(delim)) + return retrieved_text + +def loadParametersRow(filter_para, panel_list): + filter_cutoffs='' + if filter_para.strip()!='': + filter_cutoffs = map(float,filter_para.strip().split(' ')[0].split(',')) + filter_panel_list = filter_para.strip().split(' ')[1].split(',') + panel_list+=filter_panel_list + else: + filter_panel_list =[] + return filter_cutoffs, filter_panel_list, panel_list + +def readEventRow(row, header_line): + if header_line=='' or header_line==False: + rs=row.strip().split('\t') + return rs + else: + rs=row.strip().split('\t') + return dict(zip(header_line, rs)) + +def convert2SJevent(line_dict, splicing_event_type): + if splicing_event_type=='SE': + event_row_list=[line_dict['chr']+':'+str(int(line_dict['upstreamEE'])+1)+':'+line_dict['exonStart'],line_dict['chr']+':'+str(int(line_dict['exonEnd'])+1)+':'+line_dict['downstreamES'], line_dict['chr']+':'+str(int(line_dict['upstreamEE'])+1)+':'+line_dict['downstreamES']] + elif splicing_event_type=='A5SS':# Only use one junction for inc. Need to improve by updating db later + event_row_list=[line_dict['chr']+':'+str(int(line_dict['longExonEnd'])+1)+':'+line_dict['flankingES'],line_dict['chr']+':'+str(int(line_dict['shortEE'])+1)+':'+line_dict['flankingES']] + elif splicing_event_type=='A3SS': # Only use one junction for inc. Need to improve by updating db later + event_row_list=[line_dict['chr']+':'+str(int(line_dict['flankingEE'])+1)+':'+line_dict['longExonStart'],line_dict['chr']+':'+str(int(line_dict['flankingEE'])+1)+':'+line_dict['shortES']] + else: + exit('splicine event type not supported. Exiting.') + return event_row_list + +def convert2SJASevent(line_dict, splicing_event_type): + if splicing_event_type=='SE': + event_row_list=[line_dict['chr']+':'+str(int(line_dict['upstreamEE'])+1)+':'+line_dict['exonStart'],line_dict['chr']+':'+str(int(line_dict['exonEnd'])+1)+':'+line_dict['downstreamES'], line_dict['chr']+':'+str(int(line_dict['upstreamEE'])+1)+':'+line_dict['downstreamES']] + as_event=line_dict['AC'].strip('"').split('.')[0]+':'+line_dict['GeneName'].strip('"')+':'+line_dict['chr']+':'+line_dict['strand']+':'+line_dict['exonStart']+':'+line_dict['exonEnd']+':'+line_dict['upstreamEE']+':'+line_dict['downstreamES'] + elif splicing_event_type=='A5SS':# Only use one junction for inc. Need to improve by updating db later + event_row_list=[line_dict['chr']+':'+str(int(line_dict['longExonEnd'])+1)+':'+line_dict['flankingES'],line_dict['chr']+':'+str(int(line_dict['shortEE'])+1)+':'+line_dict['flankingES']] + as_event=line_dict['AC'].strip('"').split('.')[0]+':'+line_dict['GeneName'].strip('"')+':'+line_dict['chr']+':'+line_dict['strand']+':'+line_dict['longExonStart']+':'+line_dict['longExonEnd']+':'+line_dict['shortES']+':'+line_dict['shortEE']+':'+line_dict['flankingES']+':'+line_dict['flankingEE'] + elif splicing_event_type=='A3SS': # Only use one junction for inc. Need to improve by updating db later + event_row_list=[line_dict['chr']+':'+str(int(line_dict['flankingEE'])+1)+':'+line_dict['longExonStart'],line_dict['chr']+':'+str(int(line_dict['flankingEE'])+1)+':'+line_dict['shortES']] + as_event=line_dict['AC'].strip('"').split('.')[0]+':'+line_dict['GeneName'].strip('"')+':'+line_dict['chr']+':'+line_dict['strand']+':'+line_dict['longExonStart']+':'+line_dict['longExonEnd']+':'+line_dict['shortES']+':'+line_dict['shortEE']+':'+line_dict['flankingES']+':'+line_dict['flankingEE'] + else: + exit('splicine event type not supported. Exiting.') + return event_row_list, as_event + +def loadSigJunction(fin): + sig_junction={} + for i,l in enumerate(open(fin)): + if i==0: + continue + sig_junction[l.strip().split('\t')[0]]='' + return sig_junction + +def summarizeSJ2ASevent(event_list_fin, splicing_event_type, sig_junction, outdir, out_prefix): + fout_summary_fname=outdir+'/SJ.'+out_prefix+'.'+splicing_event_type+'.summary_by_sig_event.txt' + fout_summary=open(fout_summary_fname,'w') + for event_idx, event_row in enumerate(open(event_list_fin)): + if event_idx==0: + header_list=readEventRow(event_row,'') + continue + line_dict=readEventRow(event_row, header_list) + event_row_list, as_event=convert2SJASevent(line_dict, splicing_event_type) + as_event_result=[] + as_event_result_list=[] + for k in event_row_list: + if k not in sig_junction: + as_event_result.append(False) + else: + as_event_result.append(True) + as_event_result_list.append(k) + if as_event_result[0]==as_event_result[1]==as_event_result[2]==True: + fout_summary.write(as_event+'\tAll junctions\t'+';'.join(as_event_result_list)+'\n') + elif as_event_result[0]==as_event_result[1]==as_event_result[2]==False: + continue + else: + if as_event_result[0]==as_event_result[1]!=as_event_result[2]: + fout_summary.write(as_event+'\tOnly alternative junctions\t'+';'.join(as_event_result_list)+'\n') + else: + fout_summary.write(as_event+'\tOther combination\t'+';'.join(as_event_result_list)+'\n') + fout_summary.close() + return fout_summary_fname + +def main(args): + ###Loading Parameters#### + para_fin=args.parameter_file + splicing_event_type=args.splicing_event_type + event_list_fin=args.event_list_file + use_existing_test_result=args.use_existing_test_result + + outdir=args.outdir.rstrip('/') + os.system('mkdir -p '+outdir) + fetching_sj_col=1 + out_prefix,db_dir,filter1_para,filter2_para,filter3_para=[l.strip() for l in open(para_fin)][:5] + db_dir=db_dir.rstrip('/') + if os.path.isdir(db_dir+'_sjc'): #automatically use db_sjc if in the same dir. Otherwise, use the user input db_dir + db_dir=db_dir+'_sjc' + panel_list=[out_prefix] + + filter1_cutoffs, filter1_panel_list, panel_list = loadParametersRow(filter1_para, panel_list) + filter2_cutoffs, filter2_panel_list, panel_list = loadParametersRow(filter2_para, panel_list) + filter3_cutoffs, filter3_panel_list, panel_list = loadParametersRow(filter3_para, panel_list) + tumor_dict=dict.fromkeys(filter2_panel_list,'') + tumor_dict[out_prefix]='' + pvalue_cutoff_normal=''; pvalue_cutoff_tumor='' + filter1_group_cutoff=''; filter2_group_cutoff=''; filter3_group_cutoff=''; + if filter1_cutoffs!='': + pvalue_cutoff_normal,filter1_group_cutoff=filter1_cutoffs[3:] + if filter2_cutoffs!='': + pvalue_cutoff_tumor,filter2_group_cutoff=filter2_cutoffs[3:] + if filter3_cutoffs!='': + pvalue_cutoff_normal,filter3_group_cutoff=filter3_cutoffs[3:] + + tumor_read_cov_cutoff=int(args.tumor_read_cov_cutoff)#5 + normal_read_cov_cutoff=int(args.normal_read_cov_cutoff)#2 + # if filter1_panel_list==[] and filter2_panel_list==[] and filter3_panel_list==[] and test_mode[0]!='summary': + # exit("[Error] No filtering required in parameteres file. exit!") + + ##Load IRIS reference panels to 'fin_list', 'index' + index={} + fin_list={} + for group_name in panel_list: + fin_list[group_name]=db_dir+'/'+group_name+'/sjc_matrix/SJ_count.'+group_name+'.txt' + for group in fin_list.keys(): + if not os.path.isfile(fin_list[group]+'.idx'): + exit('[Error] Need to index '+fin_list[group]) + index[group]=read_SJMatrix_index(fin_list[group],'/'.join(fin_list[group].split('/')[:-1])) + + + tot=config.file_len(event_list_fin)-1 + if tot==0: + exit('[Ended] no test performed because no testable events. Check input or filtering parameteres.') #Modified 2021 + print '[INFO] IRIS screening - started. Total input events:', tot+1 + if use_existing_test_result==False: + fout_sj_count=open(outdir+'/SJ.'+out_prefix+'.'+splicing_event_type+'.test_all.txt','w') + header_line=[] + sample_size={} + for group in panel_list: + random_key=index[group].keys()[0] + sample_names=map(str,fetch_SJMatrix(random_key,fin_list[group],'\t',index[group],True)[fetching_sj_col:]) + sample_size[group]=len(sample_names) + if group==out_prefix: + header_line+=[out_prefix+'_carrier_number', out_prefix+'_fraction'] + continue + header_line+=[group+'_carrier_number', group+'_fraction', group+'_pvalue'] + fout_sj_count.write('Junction\t'+'\t'.join(header_line)+'\n') + + header_line=[] + fout_sj_sig_fname=outdir+'/SJ.'+out_prefix+'.'+splicing_event_type+'.test_sig.txt' + fout_sj_sig=open(fout_sj_sig_fname,'w') + for group in panel_list: + if group==out_prefix: + header_line+=[out_prefix+'_carrier_number', out_prefix+'_fraction'] + continue + header_line+=[group+'_carrier_number', group+'_fraction', group+'_pvalue'] + fout_sj_sig.write('Junction\t'+'\t'.join(header_line)+'\n') + + if use_existing_test_result==False: + header_list=[] + junction_dict={} + for event_idx, event_row in enumerate(open(event_list_fin)): + if event_idx==0: + header_list=readEventRow(event_row,'') + continue + line_dict=readEventRow(event_row, header_list) + event_row_list=convert2SJevent(line_dict, splicing_event_type) + for k in event_row_list: + if k not in junction_dict: + junction_dict[k]='' + else: + continue + config.update_progress(event_idx/(0.0+tot)) + + #Initiate psi matrix by each row to 'sj' + sj={} + for group in panel_list: + if k in index[group]: + sj[group]=map(int,fetch_SJMatrix(k,fin_list[group],'\t',index[group], False)[fetching_sj_col:]) + else: + sj[group]=[0]*sample_size[group] + write_sj_list=[] + significant_normal_match=0 + significant_normal=0 + significant_tumor=0 + prevalence={} + prevalence_test='' + for group in panel_list: + if group in tumor_dict: + prevalence[group]=sum(v>=tumor_read_cov_cutoff for v in sj[group]) + else: + prevalence[group]=sum(v>=normal_read_cov_cutoff for v in sj[group]) + + if group==out_prefix: + write_sj_list=[prevalence[group],prevalence[group]/(0.0+sample_size[group])] + continue + else: + if group in filter2_panel_list: # filter2 always require filer1!!! if no filter1, all references should be defined in filter3 + prevalence_test=stats.fisher_exact([[prevalence[group],sample_size[group]-prevalence[group]],[prevalence[filter1_panel_list[0]], sample_size[filter1_panel_list[0]]-prevalence[filter1_panel_list[0]]]], alternative='greater') + else: + prevalence_test=stats.fisher_exact([[prevalence[out_prefix],sample_size[out_prefix]-prevalence[out_prefix]],[prevalence[group], sample_size[group]-prevalence[group]]], alternative='greater') + write_sj_list+=[prevalence[group], prevalence[group]/(0.0+sample_size[group]), prevalence_test[1]] + #determine difference of a junction + if group in filter1_panel_list: + if prevalence_test[1]<=pvalue_cutoff_normal: + significant_normal_match+=1 + elif group in filter2_panel_list: + if prevalence_test[1]<=pvalue_cutoff_tumor: + significant_tumor+=1 + else: + if prevalence_test[1]<=pvalue_cutoff_normal: + significant_normal+=1 + if (significant_normal_match>=filter1_group_cutoff or filter1_group_cutoff=='') and (significant_tumor>=filter2_group_cutoff or filter2_group_cutoff=='') and (significant_normal>=filter3_group_cutoff or filter3_group_cutoff==''): + fout_sj_sig.write(k+'\t'+'\t'.join(map(str,write_sj_list))+'\n') + fout_sj_count.write(k+'\t'+'\t'.join(map(str,write_sj_list))+'\n') + fout_sj_count.close() + fout_sj_sig.close() + + else: + print 'Use existing testing result.' + fout_sj_count_name=outdir+'/SJ.'+out_prefix+'.'+splicing_event_type+'.test_all.txt' + for i, l in enumerate(open(fout_sj_count_name)): + if i==0: + header=l.strip().split('\t') + group_list=map(lambda x: x.split('_carrier_number')[0], header[3::3]) + else: + ls=l.strip().split('\t') + prevalence_value= map(int,ls[3::3])# don't do map because '-' + percent_value = map(float,ls[4::3]) + p_value= map(float,ls[5::3]) + significant_normal_match=0 + significant_normal=0 + significant_tumor=0 + #determine difference of a junction + for j,group in enumerate(group_list): + if group in filter1_panel_list: + if p_value[j]<=pvalue_cutoff_normal: + significant_normal_match+=1 + elif group in filter2_panel_list: + if p_value[j]<=pvalue_cutoff_tumor: + significant_tumor+=1 + else: + if p_value[j]<=pvalue_cutoff_normal: + significant_normal+=1 + if (significant_normal_match>=filter1_group_cutoff or filter1_group_cutoff=='') and (significant_tumor>=filter2_group_cutoff or filter2_group_cutoff=='') and (significant_normal>=filter3_group_cutoff or filter3_group_cutoff==''): + fout_sj_sig.write(l.strip()+'\n') + fout_sj_sig.close() + + sig_junction=loadSigJunction(fout_sj_sig_fname) + fout_summary_fname=summarizeSJ2ASevent(event_list_fin, splicing_event_type, sig_junction, outdir, out_prefix) + + +if __name__ == '__main__': + main() diff --git a/IRIS/IRIS_screening_sjcplot.py b/IRIS/IRIS_screening_sjcplot.py new file mode 100644 index 0000000..0797b8c --- /dev/null +++ b/IRIS/IRIS_screening_sjcplot.py @@ -0,0 +1,193 @@ +import numpy as np +import os,sys,glob,argparse +from scipy import stats +import statsmodels.stats.weightstats as smw +import matplotlib +matplotlib.use('agg') +import pandas as pd +import matplotlib.pyplot as plt +import seaborn as sns +from . import config + +def fileLength(fin): + i=sum(1 for l in open(fin)) + return i + +# def indiviualPlot(psi_df, event_id, panel_list): +# plt.figure(figsize=(15,9)) +# sns_plot=sns.violinplot(data=psi_df, inner="box",cut=0) +# #sns_plot.set_yticks(np.arange(0,1,0.2)) +# sns_plot.set(ylim=(0, 1)) +# sns.despine(offset=10, trim=True) +# sns_plot.set_ylabel('Percent-Spliced-In',fontweight='bold',fontsize=14) +# sns_plot.set_xticklabels(panel_list,rotation=20,ha='right',fontsize=12) +# sns_plot.set_title(event_id,fontsize=15,fontweight='bold') +# sns_plot.figure.savefig(event_id+".pdf") + +def selectJC(event_id, tumor_form, tumor_form_cutoff, splicing_event_type): + coord=event_id.split(':') + if splicing_event_type=='SE': + inc1=coord[2]+':'+str(int(coord[6])+1)+':'+coord[4] + inc2=coord[2]+':'+str(int(coord[5])+1)+':'+coord[7] + skp=coord[2]+':'+str(int(coord[6])+1)+':'+coord[7] + if tumor_form>tumor_form_cutoff:#inc + return [inc1, inc2] + else: + return [skp] + else: + exit('[Error] The splicine event type is not supported currently. Exiting.') + +def loadJCvalue4events(fin_plot_query, deltaPSI_col, tumor_form_cutoff, jc_result, splicing_event_type, has_header): + JC_dict={} + for i,l in enumerate(open(fin_plot_query)): + if i==0 and has_header: + continue + ls=l.strip().split('\t') + event_id=ls[0] + tumor_form=float(ls[deltaPSI_col]) # negative means skipping, pos means inc + JC_select=selectJC(event_id, tumor_form, tumor_form_cutoff, splicing_event_type) + for k in JC_select: + JC_dict[k]='' + for i,l in enumerate(open(jc_result)): + if i==0: + continue + ls=l.strip().split('\t') + if ls[0] in JC_dict: + JC_dict[ls[0]]=l + return JC_dict + +def loadParametersRow(filter_para, panel_list): + if filter_para.strip()!='': + para, filter_panel_list=filter_para.split(' ') + filter_cutoff_pval, filter_cutoff_dpsi, filter_cutoff_foc, filter_cutoff_pval_PT, filter_group_cutoff =para.split(',') + filter_cutoff_pval_PT=float(filter_cutoff_pval_PT) + filter_cutoff_dpsi=float(filter_cutoff_dpsi) + filter_cutoff_foc=float(filter_cutoff_foc) + filter_group_cutoff=int(filter_group_cutoff) + filter_panel_list=filter_panel_list.split(',') + panel_list+=filter_panel_list + else: + filter_cutoff_pval_PT, filter_cutoff_dpsi, filter_cutoff_foc, filter_group_cutoff, filter_panel_list =['','','','',[]] + return filter_cutoff_pval_PT, filter_cutoff_dpsi, filter_cutoff_foc, filter_group_cutoff, filter_panel_list, panel_list + +## screening +def main(args): + index={} + fin_list={} + para_fin=args.parameter_fin + splicing_event_type=args.splicing_event_type + fin_plot_query=args.event_list #IRIS screening output OR any file contain event ID and deltaPSI/indicators + jc_result=args.jc_full_result #IRIS prev screening output + tumor_form_col=int(args.deltaPSI_column)-1 + tumor_form_cutoff=float(args.deltaPSI_cut_off) + step=int(args.step) + has_header=args.header + outdir=args.outdir.rstrip('/') + + out_prefix,db_dir,filter1_para,filter2_para,filter3_para=[l.strip() for l in open(para_fin)][:5] + db_dir=db_dir.rstrip('/') + if os.path.isdir(db_dir+'_sjc'): #automatically use db_sjc if in the same dir. Otherwise, use the user input db_dir + db_dir=db_dir+'_sjc' + panel_list=[out_prefix] + + filter1_cutoff_pval, filter1_cutoff_dpsi, filter1_cutoff_foc, filter1_group_cutoff, filter1_panel_list, panel_list = loadParametersRow(filter1_para, panel_list) + filter2_cutoff_pval, filter2_cutoff_dpsi, filter2_cutoff_foc, filter2_group_cutoff, filter2_panel_list, panel_list = loadParametersRow(filter2_para, panel_list) + filter3_cutoff_pval, filter3_cutoff_dpsi, filter3_cutoff_foc, filter3_group_cutoff, filter3_panel_list, panel_list = loadParametersRow(filter3_para, panel_list) + if filter1_panel_list==[] and filter2_panel_list==[] and filter3_panel_list==[]: + # if filter1_panel_list==[] and filter2_panel_list==[] and filter3_panel_list==[] and test_mode[0]!='summary': + exit("[Error] No filtering required in parameteres file. exit!") + + single_plot=False + group_plot=True + + #Load JC values into dict based on event file + JC_dict=loadJCvalue4events(fin_plot_query, tumor_form_col, tumor_form_cutoff, jc_result, splicing_event_type, has_header) + #Make tmp file for JC plot based events + query_fin_name=outdir+'/'+fin_plot_query.split('/')[-1]+'.tmpdata.txt' + fout_tmp=open(query_fin_name,'w') + for i,l in enumerate(open(fin_plot_query)): + if i==0 and has_header: + continue + ls=l.strip().split('\t') + event_id=ls[0] + tumor_form=float(ls[tumor_form_col]) # negative means skipping, pos means inc + JC_select=selectJC(event_id, tumor_form, tumor_form_cutoff, splicing_event_type) + if tumor_form r_start+step-1: + break + ks=k.strip().split('\t') + as_event_gene=ks[0].split(':')[1] + if ks[1]=='skp': + sj_id=ks[2] + prev={} + #percentage[out_prefix]=map(float,fetch_PsiMatrix(event_id,fin_list[out_prefix],'.','\t',index[out_prefix])[1][8:]) + prev[out_prefix]=[float(ks[4])] + for n,group in enumerate(panel_list[1:]): + prev[group]=[float(ks[6+n*3])] + + prev_df = pd.DataFrame.from_dict(prev, orient='index').transpose()[panel_list] + if single_plot: + indiviualPlot(prev_df, event_id, panel_list) + if group_plot: + print step, j + ax_i = plt.subplot2grid((step,11), (j-r_start*1,0), colspan=10, rowspan=1) + ##SHOULD SET TO area. use width just for prelim + sns_plot=sns.barplot(data=prev_df, ax=ax_i,linewidth=1.5) + sns_plot.set_yticks(np.arange(0,1.1,0.5)) + sns_plot.set(xticklabels=[]) + sns.despine(offset=0, trim=False) + sns_plot.text(15.3, 0.4, as_event_gene+'\n'+sj_id, horizontalalignment='left', size='small', color='black',fontweight='bold') + else: + sj_id=ks[2]#TODO + prev={} + inc1_list=[float(ks[4])]+[float(ks[6+n*3]) for n in range(len(panel_list[1:]))] + inc2_list=[float(ks[5+(len(panel_list)-1)*3+2])]+[float(ks[5+(len(panel_list)-1)*3+4+n*3]) for n in range(len(panel_list[1:]))] + prev_df = pd.DataFrame({'Groups': panel_list,'Inc1': inc1_list , 'Inc2': inc2_list }) + prev_df_tidy = prev_df.melt(id_vars='Groups').rename(columns=str.title) + + if single_plot: + indiviualPlot(prev_df, event_id, panel_list) + if group_plot: + print step, j + ax_i = plt.subplot2grid((step,11), (j-r_start*1,0), colspan=10, rowspan=1) + ##SHOULD SET TO area. use width just for prelim + sns_plot=sns.barplot(data=prev_df_tidy, x='Groups', y='Value',hue='Variable',ax=ax_i, linewidth=1, ci=None) + my_pal=sns.husl_palette(len(panel_list)*2, s=.75, l=0.7)#color_palette("husl", len(panel_list))#(n_colors=len(panel_list)) + # for i, bar in enumerate(sns_plot.patches): + # bar.set_color(my_pal[i%len(panel_list)])#(i-i%2)/2 + for i, bar in enumerate(sns_plot.patches): + if i<=len(panel_list)-1: + bar.set_color(my_pal[i*2]) + else: + bar.set_color(my_pal[1+(i-len(panel_list))*2]) + sns_plot.legend_.remove() + sns_plot.set_yticks(np.arange(0,1.1,0.5)) + sns_plot.set(xticklabels=[]) + sns.despine(offset=0, trim=False) + sns_plot.text(15.3, 0.4, as_event_gene+'\n'+sj_id, horizontalalignment='left', size='small', color='black',fontweight='bold') + + if group_plot: + pdf.savefig(fig) + pdf.close() + +if __name__ == '__main__': + main() + diff --git a/IRIS/IRIS_seq2hla.py b/IRIS/IRIS_seq2hla.py deleted file mode 100644 index 8c1b4ea..0000000 --- a/IRIS/IRIS_seq2hla.py +++ /dev/null @@ -1,61 +0,0 @@ -import sys, os, csv, argparse, logging, datetime -from . import config -#from utilities.seq2hla import seq2HLA - -def run_seq2HLA(readsFilesCaseRNA,runname,bindir): - if os.path.exists(runname+'/hla_types-ClassI.HLAgenotype4digits')==False: - readsFiles_split=readsFilesCaseRNA.split(',') - os.system('mkdir -p '+runname) - #seq2HLA.main(runname,readsFiles_split[0],readsFiles_split[1]) - cmd = 'python '+bindir+'/seq2HLA.py -1 '+readsFiles_split[0]+' -2 '+readsFiles_split[1]+' -r '+runname+'/hla_types' - os.system (cmd) - print cmd - if os.path.exists(runname+'/hla_types-ClassI.HLAgenotype4digits')==False: - sys.exit('[seq2hla] # An Error Has Occured. seq2hla Incomplete. Exit!') - else: - print '[seq2hla] # Skipped seq2HLA.' - - HLA_type=[] - for n,l in enumerate(open(runname+'/hla_types-ClassI.HLAgenotype4digits')): - if n==0: - continue - ls=l.strip().split('\t') - #print ls - if ls[2]!='NA': - if float(ls[2])<=0.05: - HLA_type.append('HLA-'+ls[1].strip('\'')) - if ls[4]!='NA': - if float(ls[4])<=0.05: - HLA_type.append('HLA-'+ls[3].strip('\'')) - continue - # for n,l in enumerate(open(runname+'-ClassII.HLAgenotype4digits')): - # if n==0: - # continue - # ls=l.strip().split('\t') - # if ls[2]!='NA': - # if float(l[2])<=0.05: - # HLA_type.append('HLA-'+l[1].strip('\'')) - # if ls[4]!='NA': - # if float(ls[4])<=0.05: - # HLA_type.append('HLA-'+l[3].strip('\'')) - # continue - - if len(HLA_type)==0: - sys.exit('# [INFO] No HLA type predicted. Exit.') - - HLA_type_str=','.join(list(set(HLA_type))) - return HLA_type_str - -def main(args): - os.system('mkdir -p '+args.sampleID_outdir) - sampleID = args.sampleID_outdir.rstrip('/') - runname = sampleID+'/hla_types' - bindir = args.seq2hla_path.rstrip('/') - print '[INFO] # Start HLA typing.' - - hla=run_seq2HLA(args.readsFilesCaseRNA, runname, bindir) - - print '[INFO] # Completed. HLA types: '+hla - -if __name__ == '__main__': - main() diff --git a/IRIS/IRIS_sjc_matrix.py b/IRIS/IRIS_sjc_matrix.py new file mode 100644 index 0000000..3587a2b --- /dev/null +++ b/IRIS/IRIS_sjc_matrix.py @@ -0,0 +1,126 @@ +import sys, os + +def loadFinlist(fin_list_input): + fin_list=[] + for l in open(fin_list_input): + fin_list.append(l.strip()) + return fin_list +def readSJfile_STAR(fin, SJ_dict): + for l in open(fin): + ls=l.strip().split('\t') + sj=':'.join(ls[0:3]) + SJ_dict[sj]=ls[5] #add strand info + return SJ_dict + +def readSJfile(fin, SJ_dict): + for l in open(fin): + ls=l.strip().split('\t') + sj=ls[0] + SJ_dict[sj]='' #add strand info + return SJ_dict + +def index_SJMatrix(fn, outdir, delim): + out_fp = outdir+'/'+fn.split('/')[-1]+'.idx' + line_formatter = "{id}\t{offset}\n" + offset = 0 + with open(outdir+'/'+fn, 'r') as fin: + with open(out_fp, 'w') as fout: + offset += len(fin.readline()) + for line in fin: + ele = line.strip().split(delim) + eid = ele[0] + fout.write( line_formatter.format(id=eid, offset=offset) ) + offset += len(line) + return + +def main(args): + + fname_pos=args.sample_name_field#2 + fin_list_input=args.file_list_input + fin_list=loadFinlist(fin_list_input) + data_name=args.data_name + db_dir=args.iris_db_path.rstrip('/') + os.system('mkdir -p '+db_dir+'/'+data_name+' '+db_dir+'/'+data_name+'/sjc_matrix') + + + if os.path.exists(db_dir+'/'+data_name+'/sjc_matrix/SJ_count.'+data_name+'.txt'): + print '[INFO] Output '+db_dir+'/'+data_name+'/sjc_matrix/SJ_count.'+data_name+'.txt'+' exists. Only perform indexing.' + index_SJMatrix('SJ_count.'+data_name+'.txt', db_dir+'/'+data_name+'/sjc_matrix/', '\t') + exit('[INFO] Index finished.') + + fname_dict={} + for fn in fin_list: + name=fn.split('/')[-fname_pos].split('.aln')[0] + if name in fname_dict: + print name + exit('dup name'+fn+' '+name) + fname_dict[name]='' + fname_list=fname_dict.keys() + print '[INFO] Done checking file names.' + + SJ_dict={} + for i,fin in enumerate(fin_list): + if i%100==0: + print i + SJ_dict=readSJfile(fin, SJ_dict) + + SJ_list=sorted(SJ_dict.keys()) + + fout_SJ=open(db_dir+'/'+data_name+'/sjc_matrix/SJ_coordiate.'+data_name+'.txt','w') + for s in SJ_list: + fout_SJ.write(s+'\t'+SJ_dict[s]+'\n') + fout_SJ.close() + + print '[INFO] Done summarizing SJ coordinates.' + + batch=100000 + for b in range(0, len(SJ_list), batch): + print b + batch_SJ_list=SJ_list[b:min(b+batch,len(SJ_list))] + batch_SJ_dict=dict.fromkeys(batch_SJ_list,0) # can't use this to store values. will only store the last input + batch_SJ_count={} + for fin in fin_list: + fname=fin.split('/')[-fname_pos].split('.aln')[0] + for l in open(fin): + ls=l.strip().split('\t') + # sj=':'.join(ls[:3])# STAR + # count=ls[6]# STAR + sj=ls[0] + count=ls[1] + if sj in batch_SJ_dict: + if sj not in batch_SJ_count: + batch_SJ_count[sj]={} + batch_SJ_count[sj][fname]=count + continue + fout_intermediate=open(db_dir+'/'+data_name+'/sjc_matrix/SJ_count.'+data_name+'.batch_'+str(b)+'.txt','w') + for k in sorted(batch_SJ_count.keys()): + sj_line=[k] + for sample in fname_list: + if sample in batch_SJ_count[k]: + sj_line.append(batch_SJ_count[k][sample]) + else: + sj_line.append('0') ##It's okay to change to 0 later + fout_intermediate.write('\t'.join(sj_line)+'\n') + fout_intermediate.close() + fout_head=open(db_dir+'/'+data_name+'/sjc_matrix/SJ_count.'+data_name+'.header.txt','w') + fout_head.write('\t'.join(['SJ']+fname_list)+'\n') + fout_head.close() + cmd='cat '+db_dir+'/'+data_name+'/sjc_matrix/SJ_count.'+data_name+'.batch_*.txt > '+db_dir+'/'+data_name+'/sjc_matrix/SJ_count.'+data_name+'.txt_tmp' + cmd_merge='cat '+db_dir+'/'+data_name+'/sjc_matrix/SJ_count.'+data_name+'.header.txt '+db_dir+'/'+data_name+'/sjc_matrix/SJ_count.'+data_name+'.txt_tmp > '+db_dir+'/'+data_name+'/sjc_matrix/SJ_count.'+data_name+'.txt' + cmd_rm='rm '+db_dir+'/'+data_name+'/sjc_matrix/SJ_count.'+data_name+'.batch_*.txt' + cmd_rm2='rm '+db_dir+'/'+data_name+'/sjc_matrix/SJ_count.'+data_name+'.header.txt' + cmd_rm3='rm '+db_dir+'/'+data_name+'/sjc_matrix/SJ_count.'+data_name+'.txt_tmp' + print cmd + os.system(cmd) + os.system(cmd_merge) + print cmd_rm + os.system(cmd_rm) + os.system(cmd_rm2) + os.system(cmd_rm3) + + print '[INFO] Matrix finished.' + index_SJMatrix('SJ_count.'+data_name+'.txt', db_dir+'/'+data_name+'/sjc_matrix/', '\t') + print '[INFO] Index finished.' + +if __name__ == '__main__': + main() diff --git a/IRIS/IRIS_translation.py b/IRIS/IRIS_translation.py index fbb777d..3e13197 100644 --- a/IRIS/IRIS_translation.py +++ b/IRIS/IRIS_translation.py @@ -10,8 +10,8 @@ def loadFrame(fin): exonFrameDict={} for l in open(fin): ls=l.strip().split('|') - if len(ls)>5: #no hits - if (float(ls[1])/3)*0.4>float(ls[11]): #identities too less + if len(ls)>5: #skip no hits + if (float(ls[1])/3)*0.4>float(ls[11]): #skip identities too less mis+=1 continue exon_start_end=ls[0].split('-') ##TODO: name two variables, exonFrameDict key can be "'5end|'+exon_start_coord", the same for end. much clearner variable name in downstream. @@ -25,69 +25,136 @@ def loadFrame(fin): nomap+=1 return exonFrameDict, nomap, mis -def findJCRange(AS_chrom,JC_coord,exonFrameDict): +def loadGTFMicroexonInfo(gtf):#laod gtf and exon start and end and legnth, used to check microexon + micro_exon={} + for l in open(gtf): + if l.startswith('#'): + continue + ls=l.strip().split('\t') + if ls[2]=='exon': + exon_name=ls[0]+':'+ls[3]+'-'+ls[4] + if exon_name in micro_exon: + continue + length=int(ls[4])-int(ls[3]) + if length<=30: + micro_exon[exon_name]=int(ls[4])-int(ls[3]) + microexon_start={}#by junction + microexon_end={} + for e in micro_exon: + chrom=e.split(':')[0] + es, ee=e.split(':')[1].split('-') + es=chrom+':'+es + ee=chrom+':'+ee + if es not in microexon_start: + microexon_start[es]=micro_exon[e] + else: + if micro_exon[e]0: # if the left coord is the start, then find the ideal starting poistion + JC_region_d='+' + JC_region_len=min(int(align_e)-int(align_s),30) + JC_region_s=(int(JC_coord[0])-(int(exon_len)-int(align_e))-JC_region_len,JC_coord[0]) + evidence_level[0]=True + break ### Limitation: allow multiple starting + else: # if left coord is the end coord, frame doesn't matter, keep proper length(30-33) + JC_region_d='-' + if JC_region_e=='': + JC_region_e=(max(int(JC_coord[0])-33,int(JC_coord[0])-int(exon_len)), JC_coord[0])# max means min here + evidence_level[1]=True + break ### allowing loop will give more restrained junc region. but will be slower. disable for now. + else: + continue + if AS_chrom+':'+JC_coord[1] in exonFrameDict:# check the right coord of the juction & check if it is the right pair + for exon_info in exonFrameDict[AS_chrom+':'+JC_coord[1]]: exon_len,frame,align_s,align_e=exon_info[1] - if int(frame)>0: # if the left coord is the start, then find the ideal starting poistion - JC_region_d='+' - JC_region_len=min(int(align_e)-int(align_s),30) - JC_region_s=(int(JC_coord[0])-(int(exon_len)-int(align_e))-JC_region_len,JC_coord[0]) + if int(frame)>0:# if right coord is the end coord, frame doesn't matter, keep proper length(30-33) + if JC_region_d=='-': + print 'frame conflict.', JC_coord;JC_region_e='';evidence_level[1]=False;break + if JC_region_d=='':# when left coord has no hit + JC_region_d='+' + JC_region_e=(JC_coord[1],min(int(JC_coord[1])+33,int(exon_info[0]))) + evidence_level[1]=True + break ### allowing loop will give more restrained junc region. but will be slower. disable for now. + else:# right with be the start. then find the ideal starting poistion + if JC_region_d=='+': + print 'frame conflict.', JC_coord;JC_region_s='';evidence_level[0]=False;break + if JC_region_d=='':# when left coord has no hit + JC_region_d='-' + if int(align_e)<=32: + JC_region_s=(JC_coord[1],int(JC_coord[1])+int(align_e)) + elif int(align_e)>32: + shift=int(align_e)%3 + JC_region_len=30+shift + JC_region_s=(JC_coord[1],int(JC_coord[1])+JC_region_len) evidence_level[0]=True - break ### TODO: allow multiple starting - else: # if left coord is the end coord, frame doesn't matter, keep proper length(30-33) - JC_region_d='-' - if JC_region_e=='': - JC_region_e=(max(int(JC_coord[0])-33,int(JC_coord[0])-int(exon_len)), JC_coord[0])# max means min here - evidence_level[1]=True - break ### allowing loop will give more restrained junc region. but will be slower. disable for now. - # else: - # JC_region_e=(max(JC_region_e[0],int(JC_coord[0])-int(exon_len)),JC_coord[0]) + break ### Limitation: allow multiple starting + if JC_region_s!='' and JC_region_e=='':#check if JC region end is beyond exon end. handle microexons + if JC_region_d=='+': + range_bp=30#default + if AS_chrom+':'+str(int(JC_coord[1])+1) in microexon_start:#check if downstream exon start pos is a microexon start + range_bp=min(microexon_start[AS_chrom+':'+str(int(JC_coord[1])+1)],30) + JC_region_e=(JC_coord[1],int(JC_coord[1])+range_bp+1)#+/-1 from gtf else: - continue - if AS_chrom+':'+JC_coord[1] in exonFrameDict:# check the right coord of the juction & check if it is the right pair - for exon_info in exonFrameDict[AS_chrom+':'+JC_coord[1]]: - exon_len,frame,align_s,align_e=exon_info[1] - if int(frame)>0:# if right coord is the end coord, frame doesn't matter, keep proper length(30-33) - if JC_region_d=='-': - print 'frame conflict.', JC_coord;JC_region_e='';evidence_level[1]=False;break - if JC_region_d=='':# when left coord has no hit - JC_region_d='+' - JC_region_e=(JC_coord[1],min(int(JC_coord[1])+33,int(exon_info[0]))) - evidence_level[1]=True - break ### allowing loop will give more restrained junc region. but will be slower. disable for now. - else:# right with be the start. then find the ideal starting poistion - if JC_region_d=='+': - print 'frame conflict.', JC_coord;JC_region_s='';evidence_level[0]=False;break - if JC_region_d=='':# when left coord has no hit - JC_region_d='-' - if int(align_e)<=32: - JC_region_s=(JC_coord[1],int(JC_coord[1])+int(align_e)) - elif int(align_e)>32: - shift=int(align_e)%3 - JC_region_len=30+shift - JC_region_s=(JC_coord[1],int(JC_coord[1])+JC_region_len) - evidence_level[0]=True - break ### TO-DO: allow multiple starting - if JC_region_s!='' and JC_region_e=='': + range_bp=30#default + if AS_chrom+':'+JC_coord[0] in microexon_end:#check if upstream exon(downstream in translation) start pos is microexon end + range_bp=min(microexon_end[AS_chrom+':'+JC_coord[0]],30) + JC_region_e=(int(JC_coord[0])-range_bp-1,JC_coord[0]) # For events partially in the ORF annotation: Extend the end based on the known start frame #Control Microexon!! + ##Step 2: assign 30bp +/-. This is for 3 ORF search. Can be used when need/force 3 ORF search. TODO: check up/down exon length + if (all_orf and evidence_level[0]==False) or ignore_annotation: # For events not in the ORF annotation: use all 3 orf. Note: use 'evidence_level' instead of 'JC_region_s' to avoid mini exon/wrong annotation. + JC_region_d = AS_direction if JC_region_d=='+': - JC_region_e=(JC_coord[1],int(JC_coord[1])+30) + JC_region_s, JC_region_e= [(int(JC_coord[0])-30,JC_coord[0]),(JC_coord[1],int(JC_coord[1])+30)] else: - JC_region_e=(int(JC_coord[0])-30,JC_coord[0]) - return JC_region_s,JC_region_e, JC_region_d, evidence_level + JC_region_s, JC_region_e= [(JC_coord[1],int(JC_coord[1])+30),(int(JC_coord[0])-30,JC_coord[0])] + return JC_region_s, JC_region_e, JC_region_d, evidence_level + +#handle multiple AS types by taking 4 or 6 coordinate and select antigen-deriving junctions +def selectJunction(AS_coord, deltaPSI_c2n, cut_off, if_select_all, splicing_event_type): + if splicing_event_type == 'SE': + skp = (AS_coord[2], AS_coord[3],'skp') + inc1 = (AS_coord[2],AS_coord[0],'inc1') + inc2 = (AS_coord[1],AS_coord[3],'inc2') + elif splicing_event_type == 'A3SS': + skp = (AS_coord[5], AS_coord[2],'skp') + inc1 = (AS_coord[5],AS_coord[0],'inc1') + inc2 = (AS_coord[2],AS_coord[2],'inc2') + elif splicing_event_type == 'A5SS': + skp = (AS_coord[3], AS_coord[4],'skp') + inc1 = (AS_coord[3],AS_coord[3],'inc1') + inc2 = (AS_coord[1],AS_coord[4],'inc2') + elif splicing_event_type == 'RI': + skp = (AS_coord[3], AS_coord[4],'skp') + inc1 = (AS_coord[3],AS_coord[3],'inc1') + inc2 = (AS_coord[4],AS_coord[4],'inc2') -def selectJC(AS_coord,deltaPSI_c2n,cut_off, select_all): - if select_all: - return [(AS_coord[2], AS_coord[3],'skp'),(AS_coord[2],AS_coord[0],'inc1'),(AS_coord[1],AS_coord[3],'inc2')] - if float(deltaPSI_c2n)'): if JC_nuc_seq_ID!='': @@ -158,25 +230,32 @@ def JC2pep(JC_region_bed,ref_genome,outdir,info):#getfasta of the coord and tran if JC_nuc_seq!='': first_half_len=len(JC_nuc_seq) JC_nuc_seq+=row.strip().upper() - JC_prot_seq=translateNuc(JC_nuc_seq,1)[0][0] - if len(JC_prot_seq)<8 or len(JC_prot_seq)*3<=first_half_len:# skip seq finds PTC before junction site - JC_nuc_seq_ID='' - JC_nuc_seq='' - continue - junction_pos=first_half_len/3+1 - JC_prot_seq_up=JC_prot_seq[:junction_pos-1]+JC_prot_seq[junction_pos-1].lower()+JC_prot_seq[junction_pos:] - if junction_pos>11:#long upstream junction peptides, often due to half-mapped but quality BLAST aligment(known exon but not/partially known in protein annotation) - diff=junction_pos-11 - JC_nuc_seq_ID=JC_nuc_seq_ID+'|trim_'+str(diff) - JC_prot_seq_up=JC_prot_seq_up[diff:] - #print JC_nuc_seq, JC_prot_seq_up, JC_prot_seq, first_half_len, junction_pos - fout_JC_pep.write('>{}\n{}\n'.format(JC_nuc_seq_ID,JC_prot_seq_up)) + JC_prot_seq_list=translateNuc(JC_nuc_seq, orf_range, remove_early_stop) + for JC_prot_seq_pair in JC_prot_seq_list: + JC_prot_seq,JC_prot_orf_start=JC_prot_seq_pair + if len(JC_prot_seq)<8 or len(JC_prot_seq)*3<=first_half_len:# skip seq finds PTC before junction site + continue + junction_pos=first_half_len/3+1 + JC_prot_seq_up=JC_prot_seq[:junction_pos-1]+JC_prot_seq[junction_pos-1].lower()+JC_prot_seq[junction_pos:] + if junction_pos>11:#long upstream junction peptides, often due to half-mapped but quality BLAST aligment(known exon but not/partially known in protein annotation) + diff=junction_pos-11 + JC_nuc_seq_ID=JC_nuc_seq_ID+'|trim_'+str(diff) + JC_prot_seq_up=JC_prot_seq_up[diff:] + if JC_prot_orf_start==0: + fout_JC_pep.write('>{}\n{}\n'.format(JC_nuc_seq_ID,JC_prot_seq_up)) + elif JC_prot_orf_start==1: + fout_JC_pep_1.write('>{}\n{}\n'.format('FrameAdd1'.join(JC_nuc_seq_ID.split('Frame')),JC_prot_seq_up)) + elif JC_prot_orf_start==2: + fout_JC_pep_2.write('>{}\n{}\n'.format('FrameAdd2'.join(JC_nuc_seq_ID.split('Frame')),JC_prot_seq_up)) JC_nuc_seq_ID='' JC_nuc_seq='' continue JC_nuc_seq=row.strip().upper() fout_JC_pep.close() - return JC_pep_fasta + if all_orf: + fout_JC_pep_1.close() + fout_JC_pep_2.close() + return JC_pep_fasta_name def loadSeq(fin): dic={} @@ -186,7 +265,7 @@ def loadSeq(fin): for l in open(fin): if l.startswith('>'): name=l.strip()[1:] - form_field=name.split(':')[5]#May be different for different Bedtools#Feb19 + form_field=name.split(':')[5]#May different for different Bedtools. Feb19 if form_field.startswith('inc1'): form='inc1' elif form_field.startswith('inc2'): @@ -247,7 +326,7 @@ def compSeq(seq1,seq2,seq3): pass_js2=True if j+1==len(seq1): short_seq1=True - true_junction2=i+1 + true_junction2=j+1 break continue else: @@ -278,9 +357,9 @@ def compSeq(seq1,seq2,seq3): seq3==seq3.upper() return seq1, seq2, seq3 -def compPepFile(fin,outdir,select_form): +def compPepFile(fin,outdir,pep_dir_prefix,select_form, novel_info): comp_result_dic={} - dic,direction=loadSeq(outdir+'/tmp/prot/'+fin) + dic,direction=loadSeq(outdir+'/tmp/'+pep_dir_prefix+'/'+fin) if len(dic)>1: if 'skp' in dic: #where the comparison is needed seq2='' @@ -307,65 +386,83 @@ def compPepFile(fin,outdir,select_form): dic['inc1'][1]=comp_result[2] if 'inc2' in dic: dic['inc2'][1]=comp_result[1] + if novel_info==[]: + novel_info=dic if select_form!=2: - fout_skp=open(outdir+'/tmp/prot.compared/skp/skp.'+fin.split('/')[-1],'w') - if 'skp' in dic: + fout_skp=open(outdir+'/tmp/'+pep_dir_prefix+'.compared/skp/skp.'+fin.split('/')[-1],'w') + if 'skp' in dic and 'skp' in novel_info: fout_skp.write('>{}\n{}\n'.format(dic['skp'][0],dic['skp'][1])) fout_skp.close() if select_form!=1: - fout_inc=open(outdir+'/tmp/prot.compared/inc/inc.'+fin.split('/')[-1],'w') - if 'inc1' in dic: + fout_inc=open(outdir+'/tmp/'+pep_dir_prefix+'.compared/inc/inc.'+fin.split('/')[-1],'w') + if 'inc1' in dic and 'inc1' in novel_info: fout_inc.write('>{}\n{}\n'.format(dic['inc1'][0],dic['inc1'][1])) - if 'inc2' in dic: + if 'inc2' in dic and 'inc2' in novel_info: fout_inc.write('>{}\n{}\n'.format(dic['inc2'][0],dic['inc2'][1])) fout_inc.close() -def AS2pep(AS_chrom, AS_coord,AS_direction, deltaPSI_c2n,cuf_off, select_all, exonFrameDict,ref_genome,outdir,info): - JC_coord_list=selectJC(AS_coord,deltaPSI_c2n,cuf_off, True) #under select_all=True - select_form=len(selectJC(AS_coord,deltaPSI_c2n,cuf_off, select_all)) - JC_region_bed='_'.join([AS_chrom,AS_direction]+AS_coord)+'.JCregion.bed' - fout_JC_region=open(outdir+'/tmp/junction/'+JC_region_bed,'w') #The JC file created. with all forms included regardless of selecting setting. Required for comparison. +def AS2pep(AS_chrom, AS_coord, AS_direction, deltaPSI_c2n, cuf_off, if_select_all, all_orf, pep_dir_prefix, microexon_start, microexon_end, ignore_annotation, remove_early_stop, splicing_event_type, exonFrameDict, ref_genome, outdir, info, novel_info): + JC_coord_list = selectJunction(AS_coord, deltaPSI_c2n, cuf_off, True, splicing_event_type) #under if_select_all=True + select_form = len(selectJunction(AS_coord, deltaPSI_c2n, cuf_off, if_select_all, splicing_event_type)) + JC_region_bed = '_'.join([AS_chrom,AS_direction]+AS_coord)+'.JCregion.bed' + fout_JC_region = open(outdir+'/tmp/junction/'+JC_region_bed,'w') #The JC file created. with all forms included regardless of selecting setting. Required for comparison. for JC_coord in JC_coord_list: - JC_region=findJCRange(AS_chrom,JC_coord[:2], exonFrameDict) - if JC_region[3][0]: - fout_JC_region.write('{}\t{}\t{}\t{}\t{}\t{}\n'.format(AS_chrom,JC_region[0][0],JC_region[0][1],AS_chrom+':'+JC_coord[0]+'|'+JC_coord[1]+':uniprotFrame:'+AS_direction+':form:'+JC_coord[2],'.',JC_region[2])) - fout_JC_region.write('{}\t{}\t{}\t{}\t{}\t{}\n'.format(AS_chrom,JC_region[1][0],JC_region[1][1],AS_chrom+':'+JC_coord[0]+'|'+JC_coord[1]+':uniprotFrame:'+AS_direction+':form:'+JC_coord[2],'.',JC_region[2])) - #else: #no known frame; implement 3 orf search latter - #fout_JC_region.write('{}\t{}\t{}\t{}\t{}\t{}\n'.format(AS_chrom,JC_region[0][0],JC_region[0][1],AS_chrom+':'+JC_coord[0]+'|'+JC_coord[1]+':unknownFrame:'+AS_direction,'.',JC_region[2])) - #fout_JC_region.write('{}\t{}\t{}\t{}\t{}\t{}\n'.format(AS_chrom,JC_region[1][0],JC_region[1][1],AS_chrom+':'+JC_coord[0]+'|'+JC_coord[1]+':unknownFrame:'+AS_direction,'.',JC_region[2])) + JC_region=findJCRange(AS_chrom, JC_coord[:2], AS_direction, exonFrameDict, all_orf, microexon_start, microexon_end, ignore_annotation) + if JC_region[3][0]:#check if the starting end orf is known in annotation + fout_JC_region.write('{}\t{}\t{}\t{}\t{}\t{}\n'.format(AS_chrom,JC_region[0][0],JC_region[0][1],AS_chrom+':'+JC_coord[0]+'|'+JC_coord[1]+':uniprotFrame:'+JC_region[2]+':form:'+JC_coord[2],'.',JC_region[2])) + fout_JC_region.write('{}\t{}\t{}\t{}\t{}\t{}\n'.format(AS_chrom,JC_region[1][0],JC_region[1][1],AS_chrom+':'+JC_coord[0]+'|'+JC_coord[1]+':uniprotFrame:'+JC_region[2]+':form:'+JC_coord[2],'.',JC_region[2])) + elif (all_orf or ignore_annotation): #3 orf frame; + fout_JC_region.write('{}\t{}\t{}\t{}\t{}\t{}\n'.format(AS_chrom,JC_region[0][0],JC_region[0][1],AS_chrom+':'+JC_coord[0]+'|'+JC_coord[1]+':unknownFrame:'+AS_direction+':form:'+JC_coord[2],'.',JC_region[2])) + fout_JC_region.write('{}\t{}\t{}\t{}\t{}\t{}\n'.format(AS_chrom,JC_region[1][0],JC_region[1][1],AS_chrom+':'+JC_coord[0]+'|'+JC_coord[1]+':unknownFrame:'+AS_direction+':form:'+JC_coord[2],'.',JC_region[2])) fout_JC_region.close() - JC_pep_fasta=JC2pep(JC_region_bed,ref_genome,outdir,info)# Junction peptides file created. With all forms included regardless of form selection. - compPepFile(JC_pep_fasta, outdir, select_form) # Compared Junction peptides file created. Form selection STARTS HERE. + JC_pep_fasta_name=JC2pep(JC_region_bed, ref_genome, outdir, info, remove_early_stop, all_orf, pep_dir_prefix)# Junction peptides file created. With all forms included regardless of form selection. + compPepFile(JC_pep_fasta_name+'.prot.fa', outdir, pep_dir_prefix, select_form, novel_info) # Compared Junction peptides file created. Form selection STARTS HERE. + if all_orf: + compPepFile(JC_pep_fasta_name+'_orf1.prot.fa', outdir, pep_dir_prefix, select_form, novel_info) + compPepFile(JC_pep_fasta_name+'_orf2.prot.fa', outdir, pep_dir_prefix, select_form, novel_info) def main(args): orf_mapping_file=config.ORF_MAP_PATH - ref_genome=args.ref_genome #ref_genome='~/nobackup-yxing/references.annotations/37.chr/ucsc.hg19.fasta' + ref_genome=args.ref_genome fin=args.as_input + splicing_event_type=args.splicing_event_type + all_orf=args.all_orf + gtf=args.gtf + microexon_start, microexon_end=loadGTFMicroexonInfo(gtf) + ignore_annotation=args.ignore_annotation + remove_early_stop=args.remove_early_stop outdir=args.outdir.rstrip('/') - select_all=args.no_tumor_form_selection + if_select_all=args.no_tumor_form_selection deltaPSI_cut_off=float(args.deltaPSI_cut_off) deltaPSI_column=int(args.deltaPSI_column)-1 - + check_novel=args.check_novel + pep_dir_prefix='prot' + if all_orf: + pep_dir_prefix='prot_allorf' os.system('mkdir -p '+outdir+'/tmp') - os.system('mkdir -p '+outdir+'/tmp/junction '+outdir+'/tmp/rna '+outdir+'/tmp/prot '+outdir+'/tmp/pred') - os.system('mkdir -p '+outdir+'/tmp/prot.compared '+outdir+'/tmp/prot.compared/skp '+outdir+'/tmp/prot.compared/inc') + os.system('mkdir -p '+outdir+'/tmp/junction '+outdir+'/tmp/rna '+outdir+'/tmp/'+pep_dir_prefix+' '+outdir+'/tmp/pred') + os.system('mkdir -p '+outdir+'/tmp/'+pep_dir_prefix+'.compared '+outdir+'/tmp/'+pep_dir_prefix+'.compared/skp '+outdir+'/tmp/'+pep_dir_prefix+'.compared/inc') exonFrameDict,nomap, mis =loadFrame(orf_mapping_file) print '[INFO] Total exon-orf loaded',len(exonFrameDict), nomap, mis #Select AS version, region and frame, translate to peptides. events_processed=0 tot=config.file_len(fin)-1 + header=[] for i,l in enumerate(open(fin)): if l.startswith('ENSG')==False: + header=l.strip().split('\t') continue config.update_progress(i/(0.0+tot)) - # if i%500==0: - # print i - ls=l.strip('\n').split('\t') + ls=l.strip().split('\t') + ld=dict(zip(header,ls)) events_processed+=1 des=ls[0].split(':') info='_'.join(des[:2]).strip('_').replace('/','+') - AS2pep(des[2],des[4:8],des[3],ls[deltaPSI_column],deltaPSI_cut_off,False,exonFrameDict,ref_genome,outdir,info) + novel_info=[] + if check_novel: + novel_info=ld['novel_ss_info'].split(';') + AS2pep(des[2], des[4:], des[3], ls[deltaPSI_column], deltaPSI_cut_off, False, all_orf, pep_dir_prefix, microexon_start, microexon_end, ignore_annotation, remove_early_stop, splicing_event_type, exonFrameDict, ref_genome, outdir, info, novel_info) print '[INFO] Total processed',events_processed if __name__ == '__main__': diff --git a/IRIS/IRIS_visual_summary.py b/IRIS/IRIS_visual_summary.py new file mode 100644 index 0000000..e8b4160 --- /dev/null +++ b/IRIS/IRIS_visual_summary.py @@ -0,0 +1,1154 @@ +from __future__ import print_function + +import collections +import os +import sys + +import matplotlib +matplotlib.use('agg') # sets the plotting mode to non-interactive + +import matplotlib.gridspec as gridspec +import matplotlib.patches +import matplotlib.pyplot as plt +import numpy as np +import pandas as pd +import seaborn as sns + +COLOR_BLACK = '#000000' +COLOR_BURNT_ORANGE = '#c04e01' +COLOR_CREAM = '#ffffc2' +COLOR_GREEN = '#15b01a' +COLOR_LIGHT_VIOLET = '#d6b4fc' +COLOR_OCHRE = '#bf9005' + +COLOR_BY_PANEL_TYPE = { + 'output': COLOR_BLACK, + 'tissue_matched_normal': COLOR_GREEN, + 'tumor': COLOR_BURNT_ORANGE, + 'normal': COLOR_GREEN +} + +Z_LOWEST = 0 + +# size in points +FONT_SIZE_MEDIUM = 11 +FONT_SIZE_LARGE = 13 + + +def points_to_pixels(points): + # 1 point = (1/72.0) inches + # 1 inch = 96.0 pixels + inches = points / 72.0 + return inches * 96.0 + + +def get_matrix_index(event_id, index_f_name): + with open(index_f_name, 'rt') as f_h: + for line in f_h: + event, offset_str = line.strip().split('\t') + if event == event_id: + offset_int, error = parse_int(offset_str) + if error: + return None, error + + return offset_int, None + + return None, '{} not in {}'.format(event_id, index_f_name) + + +def read_event_row(event_id, matrix_f_name, index_f_name): + row = collections.OrderedDict() + offset, error = get_matrix_index(event_id, index_f_name) + fill_nan = False + if error: + print('{}; filling with NaN values'.format(error), file=sys.stderr) + fill_nan = True + + with open(matrix_f_name, 'rt') as f_h: + raw_headers = f_h.readline().strip().split('\t') + header = np.asarray([ + x.split('.')[0] if x.startswith('SRR') else x for x in raw_headers + ]) + + if fill_nan: + data = np.asarray(['NaN' for _ in header]) + else: + f_h.seek(offset, 0) + data = np.asarray(f_h.readline().strip().split('\t')) + + row = collections.OrderedDict(zip(header, data)) + + return row, None + + +def read_panel_parameters(line): + parameters = dict() + if not line.strip(): + return parameters, None + + splits = line.split(' ') + if len(splits) != 2: + return parameters, 'expected 2 fields in {} but found {}'.format( + line, len(splits)) + + cutoffs_str, groups_str = splits + cutoffs = cutoffs_str.split(',') + groups = groups_str.split(',') + floats = list() + for cutoff in cutoffs: + parsed_float, error = parse_float(cutoff) + if error: + return parameters, error + + floats.append(parsed_float) + + parameters['psi_pval_cutoff'] = floats[0] + parameters['delta_psi_cutoff'] = floats[1] + parameters['foc_cutoff'] = floats[2] + parameters['sjc_pval_cutoff'] = floats[3] + parameters['group_cutoff'] = floats[4] + parameters['groups'] = groups + return parameters, None + + +def read_parameters(parameter_f_name): + parameters = dict() + + with open(parameter_f_name, 'rt') as f_h: + lines = [line.strip() for line in f_h] + + if len(lines) != 10: + return parameters, 'expected 10 lines in {} but found {}'.format( + parameter_f_name, len(lines)) + + parameters['out_prefix'] = lines[0] + parameters['db_dir'] = lines[1] + panel_params = list() + for line in lines[2:5]: + this_panel_params, error = read_panel_parameters(line) + if error: + return parameters, error + + panel_params.append(this_panel_params) + + parameters['tissue_matched_normal_panel'] = panel_params[0] + parameters['tumor_panel'] = panel_params[1] + parameters['normal_panel'] = panel_params[2] + parameters['test_mode'] = lines[5] + parameters['use_ratio'] = lines[6] == 'True' + parameters['blacklist'] = lines[7] + parameters['mappability_path'] = lines[8] + parameters['ref_genome_path'] = lines[9] + + return parameters, None + + +def get_group_to_panel_type(parameters): + panel_tmn = parameters['tissue_matched_normal_panel'].get('groups', list()) + panel_t = parameters['tumor_panel'].get('groups', list()) + panel_n = parameters['normal_panel'].get('groups', list()) + output_group = parameters['out_prefix'] + + group_to_panel_type = collections.OrderedDict() + group_to_panel_type[output_group] = 'output' + + for group in panel_tmn: + group_to_panel_type[group] = 'tissue_matched_normal' + + for group in panel_t: + group_to_panel_type[group] = 'tumor' + + for group in panel_n: + group_to_panel_type[group] = 'normal' + + return group_to_panel_type + + +def get_matrix_file_names(groups, parameters, event_type): + group_to_matrix_f_name = collections.OrderedDict() + group_to_matrix_index_f_name = collections.OrderedDict() + + if not groups: + return group_to_matrix_f_name, group_to_matrix_index_f_name, 'no groups specified' + + db_dir = parameters['db_dir'] + for group_name in groups: + splicing_f_name = 'splicing_matrix.{}.cov10.{}.txt'.format( + event_type, group_name) + group_to_matrix_f_name[group_name] = os.path.join( + db_dir, group_name, 'splicing_matrix', splicing_f_name) + + for group, matrix_f_name in group_to_matrix_f_name.items(): + if not os.path.isfile(matrix_f_name): + error = 'no matrix file found for {}. expected {}'.format( + group, matrix_f_name) + return group_to_matrix_f_name, group_to_matrix_index_f_name, error + + index_f_name = '{}.idx'.format(matrix_f_name) + if not os.path.isfile(index_f_name): + error = 'no index file found for {}. expected {}'.format( + group, matrix_f_name) + return group_to_matrix_f_name, group_to_matrix_index_f_name, error + + group_to_matrix_index_f_name[group] = index_f_name + + return group_to_matrix_f_name, group_to_matrix_index_f_name, None + + +def get_events(parameters, screening_out_dir, event_type): + events = dict() + + for variant in ['tier1', 'tier2tier3']: + variant_events = list() + events[variant] = variant_events + file_name = '{}.{}.{}.txt'.format(parameters['out_prefix'], event_type, + variant) + file_path = os.path.join(screening_out_dir, file_name) + if not os.path.isfile(file_path): + return events, 'missing required file: {}'.format(file_path) + + event_id_index = None + with open(file_path, 'rt') as f_h: + for i, line in enumerate(f_h): + splits = line.strip().split('\t') + if i == 0: # header + event_id_header = 'as_event' + if event_id_header not in splits: + return events, 'required header {} not found in {}'.format( + event_id_header, file_path) + + event_id_index = splits.index(event_id_header) + continue + + if len(splits) <= event_id_index: + return events, 'no {} for line {} of {}'.format( + event_id_header, i, file_path) + + event_id = splits[event_id_index] + variant_events.append(event_id) + + return events, None + + +def get_psi_data_by_event(group_to_matrix_f_name, group_to_matrix_index_f_name, + events): + psi_data_by_event = collections.OrderedDict() + tier2tier3_events = events['tier2tier3'] + if not tier2tier3_events: + return psi_data_by_event, 'no tier2tier3 events' + + groups = group_to_matrix_f_name.keys() + for event_id in tier2tier3_events: + psi_by_group = collections.OrderedDict() + all_psi = list() + for group in groups: + row, error = read_event_row(event_id, + group_to_matrix_f_name[group], + group_to_matrix_index_f_name[group]) + if error: + return psi_data_by_event, error + + psi_strings = list(row.values())[8:] + psi_floats = list() + for psi_s in psi_strings: + psi_f, error = parse_float(psi_s) + if error: + return psi_data_by_event, error + + psi_floats.append(psi_f) + + psi_by_group[group] = psi_floats + all_psi.extend(psi_floats) + + abs_change = max(all_psi) - min(all_psi) + if abs_change < 0.05: + continue + + psi_df = pd.DataFrame.from_dict(psi_by_group, + orient='index').transpose()[groups] + psi_data_by_event[event_id] = psi_df + + return psi_data_by_event, None + + +def add_or_verify_match(key, source_dict, dest_dict, parser): + if key not in source_dict: + return 'missing {}'.format(key) + + source_v = source_dict[key] + parsed_source_v, error = parser(source_v) + if error: + return error + + if key in dest_dict: + existing_v = dest_dict[key] + if parsed_source_v == existing_v: + return None + + return 'differing values for {}: {}, {}'.format( + key, parsed_source_v, existing_v) + + dest_dict[key] = parsed_source_v + return None + + +def read_tsv(f_name): + rows = list() + header = list() + with open(f_name, 'rt') as f_h: + for i, line in enumerate(f_h): + tokens = line.strip().split('\t') + if i == 0: + header = tokens + continue + + if len(tokens) != len(header): + return header, rows, 'expected {} columns at line {} of {} but found {}'.format( + len(header), i, f_name, len(tokens)) + + rows.append(dict(zip(header, tokens))) + + return header, rows, None + + +def get_hlas_with_binding_affinity(hla_headers, row, row_num, f_name): + hlas_with_binding_affinity = list() + for header in hla_headers: + full_str = row[header] + if full_str == '-': + continue + + col_error_prefix = 'row {} in {} with column {}={}: '.format( + row_num, f_name, header, full_str) + semi_splits = full_str.split(';') + for semi_split in semi_splits: + pipe_splits = semi_split.split('|') + if len(pipe_splits) != 2: + return hlas_with_binding_affinity, '{}expected exactly 1 "|" in {}'.format( + col_error_prefix, semi_split) + + hla_str, binding_str = pipe_splits + hla_prefix = 'HLA-' + if not hla_str.startswith(hla_prefix): + return hlas_with_binding_affinity, '{}expected a value starting with "{}"'.format( + col_error_prefix, hla_prefix) + + hla_sub_str = hla_str[len(hla_prefix):] + binding_float, error = parse_float(binding_str) + if error: + return hlas_with_binding_affinity, '{}{}'.format( + col_error_prefix, error) + + hlas_with_binding_affinity.append((hla_sub_str, binding_float)) + + return hlas_with_binding_affinity, None + + +def parse_int_ratio(s): + splits = s.split('/') + if len(splits) != 2: + return None, 'expected exactly one "/" in {}'.format(s) + + ints = list() + for split in splits: + as_int, error = parse_int(split) + if error: + return None, error + + ints.append(as_int) + + return ints, None + + +def parse_float(s): + try: + as_float = float(s) + except ValueError as e: + return None, 'could not parse {} as float: {}'.format(s, e) + + return as_float, None + + +def parse_int(s): + try: + as_int = int(s) + except ValueError as e: + return None, 'could not parse {} as int: {}'.format(s, e) + + return as_int, None + + +def process_epitope_summary_row(row, row_num, summary, hla_headers, f_name, + has_prediction): + event_id = row['as_event'] + event_summary = summary.get(event_id) + if not event_summary: + event_summary = dict() + summary[event_id] = event_summary + + event_epitopes_hla_affinity_patients = event_summary.get('epitopes') + if not event_epitopes_hla_affinity_patients: + event_epitopes_hla_affinity_patients = list() + event_summary['epitopes'] = event_epitopes_hla_affinity_patients + + row_error_prefix = 'row {} in {}: '.format(row_num, f_name) + int_ratio_keys = [ + 'tissue_matched_normal_panel', 'tumor_panel', 'normal_panel' + ] + for key in int_ratio_keys: + error = add_or_verify_match(key, row, event_summary, parse_int_ratio) + if error: + return '{}{}'.format(row_error_prefix, error) + + error = add_or_verify_match('fc_of_tumor_isoform', row, event_summary, + parse_float) + if error: + return '{}{}'.format(row_error_prefix, error) + + if not has_prediction: + return None + + hlas_with_binding_affinity, error = get_hlas_with_binding_affinity( + hla_headers, row, row_num, f_name) + if error: + return error + + if not hlas_with_binding_affinity: + return '{}no HLAs found'.format(row_error_prefix) + + hlas_with_binding_affinity.sort(key=lambda p: p[1]) + hla, binding_affinity = hlas_with_binding_affinity[0] + event_epitopes_hla_affinity_patients.append({ + 'epitope': + row['epitope'], + 'hla': + hla, + 'binding_affinity': + binding_affinity, + 'num_patients': + row['num_sample'] + }) + + return None + + +def get_epitope_summary(screening_out_dir, has_prediction, parameters, + event_type): + summary = dict() + + out_prefix = parameters['out_prefix'] + if has_prediction: + summary_f_name = os.path.join(screening_out_dir, + '{}.tier2tier3'.format(event_type), + 'epitope_summary.peptide-based.txt') + else: + summary_f_name = os.path.join( + screening_out_dir, + '{}.{}.tier2tier3.txt'.format(out_prefix, event_type)) + + header, rows, error = read_tsv(summary_f_name) + if error: + return summary, error + + expected_columns = { + 'as_event', 'meanPSI', 'Q1PSI', 'Q3PSI', 'deltaPSI', + 'fc_of_tumor_isoform', 'tissue_matched_normal_panel', 'tumor_panel', + 'normal_panel', 'tag', 'mappability', 'mappability_tag' + } + if has_prediction: + expected_columns = expected_columns.union({ + 'epitope', 'junction_peptide_form', 'inclusion_form', 'num_hla', + 'num_sample', 'hla_types', 'canonical_match', 'uniqueness' + }) + + ignored_optional_columns = {'meanGeneExp', 'Q1GeneExp', 'Q3GeneExp'} + header_set = set(header) + missing_headers = expected_columns.difference(header_set) + if missing_headers: + return summary, 'missing headers in {}: {}'.format( + summary_f_name, ', '.join(sorted(missing_headers))) + + ignored_columns = expected_columns.union(ignored_optional_columns) + extra_headers = header_set.difference(ignored_columns) + for i, row in enumerate(rows): + error = process_epitope_summary_row(row, i, summary, extra_headers, + summary_f_name, has_prediction) + if error: + return summary, error + + if has_prediction: + for event_summary in summary.values(): + event_summary['epitopes'].sort(key=lambda p: p['binding_affinity']) + + return summary, None + + +def remove_spines(ax): + for spine in ax.spines.values(): + spine.set_visible(False) + + +def remove_ticks_and_labels(ax): + ax.set_xticks(list()) + ax.set_yticks(list()) + + +def remove_spines_ticks_and_labels(ax): + remove_spines(ax) + remove_ticks_and_labels(ax) + + +def hide_ax(ax): + ax.set_visible(False) + + +def hide_ax_except_color(ax, color): + ax.clear() + ax.set_zorder(Z_LOWEST) + remove_spines_ticks_and_labels(ax) + ax.set_facecolor(color) + + +def make_violin_plots(events, psi_data_by_event, groups, group_to_panel_type, + axes_by_name): + gene_header_ax, gene_axes = axes_by_name['gene_name'] + violins_header_ax, violins_axes = axes_by_name['violins'] + violins_y_ticks_header_ax, violins_y_ticks_axes = axes_by_name[ + 'violins_y_ticks'] + + hide_ax_except_color(violins_y_ticks_header_ax, COLOR_CREAM) + for y_tick_ax in violins_y_ticks_axes: + hide_ax(y_tick_ax) + + make_gene_header(gene_header_ax) + + if len(events) != len(psi_data_by_event) or len(events) != len( + gene_axes) or len(events) != len(violins_axes): + return 'all inputs must have the same length' + + sns.set(style="white", color_codes=True) + for i, event in enumerate(events): + event = events[i] + psi_data = psi_data_by_event.get(event) + if psi_data is None: + return 'no psi data for {}'.format(event) + + gene_ax = gene_axes[i] + violins_ax = violins_axes[i] + + sns.violinplot(data=psi_data, + ax=violins_ax, + inner="box", + cut=0, + scale='width', + linewidth=1.5) + + violins_ax.set_yticks(np.arange(0, 1.1, 0.5)) + violins_ax.set_xticklabels(list()) + violins_ax.xaxis.set_tick_params(which='both', length=0) + sns.despine(ax=violins_ax, offset=0, trim=False) + + remove_spines_ticks_and_labels(gene_ax) + gene_name = event.split(':')[1] + gene_ax.text(0.25, + 0.5, + gene_name, + horizontalalignment='center', + verticalalignment='center', + transform=gene_ax.transAxes, + style='italic', + size=FONT_SIZE_LARGE, + color=COLOR_BLACK, + fontweight='bold') + + top_violin_ax = violins_axes[0] + make_violins_header(violins_header_ax, top_violin_ax, groups, + group_to_panel_type) + return None + + +def make_gene_header(gene_header_ax): + gene_header_ax.text(0.25, + 0, + 'Gene of\nAS event', + horizontalalignment='center', + verticalalignment='bottom', + transform=gene_header_ax.transAxes, + size=FONT_SIZE_LARGE, + color=COLOR_BLACK, + fontweight='bold') + + remove_spines_ticks_and_labels(gene_header_ax) + + +def make_violins_header(violin_header_ax, top_violin_ax, groups, + group_to_panel_type): + violin_xticks_data_coords = [(x, 0) for x in top_violin_ax.get_xticks()] + violin_xticks_display_coords = top_violin_ax.transData.transform( + violin_xticks_data_coords) + violin_tick_label_y_display_coord = violin_header_ax.transAxes.transform_point( + (0, 0))[1] + violin_tick_label_display_coords = [(p[0], + violin_tick_label_y_display_coord) + for p in violin_xticks_display_coords] + trans_display_to_axes = violin_header_ax.transAxes.inverted( + ).transform_point + for i, coord in enumerate(violin_tick_label_display_coords): + x, y = trans_display_to_axes(coord) + group = groups[i] + color = COLOR_BY_PANEL_TYPE[group_to_panel_type[group]] + violin_header_ax.text(x, + y, + group, + horizontalalignment='center', + verticalalignment='bottom', + rotation=90, + transform=violin_header_ax.transAxes, + size=FONT_SIZE_MEDIUM, + color=color, + fontweight='bold') + + violin_header_ax.text(0.5, + 1, + 'PSI by tissue or tumor', + horizontalalignment='center', + verticalalignment='top', + transform=violin_header_ax.transAxes, + size=FONT_SIZE_LARGE, + color=COLOR_BLACK, + fontweight='bold') + text_height = points_to_pixels(FONT_SIZE_LARGE) + top_text_y_max_display = violin_header_ax.transAxes.transform_point( + (0, 1))[1] + top_text_y_min_display = top_text_y_max_display - text_height + top_text_y_min_axes = trans_display_to_axes((0, top_text_y_min_display))[1] + underline_x_min_display = violin_xticks_display_coords[0][0] + underline_x_max_display = violin_xticks_display_coords[-1][0] + underline_x_min_axes = trans_display_to_axes( + (underline_x_min_display, 0))[0] + underline_x_max_axes = trans_display_to_axes( + (underline_x_max_display, 0))[0] + violin_header_ax.plot([underline_x_min_axes, underline_x_max_axes], + [top_text_y_min_axes] * 2, + color=COLOR_BLACK, + linestyle='solid', + transform=violin_header_ax.transAxes) + + remove_spines_ticks_and_labels(violin_header_ax) + violin_header_ax.set_facecolor(COLOR_CREAM) + + +def create_grid_of_axes(fig, num_events, has_prediction): + content_rows = num_events + 1 + spacer_rows = content_rows - 1 + content_rows_to_spacer = 9 + extra_header_room = content_rows_to_spacer + grid_rows = (content_rows_to_spacer * content_rows) + spacer_rows + grid_rows += extra_header_room + + grid_col_names_and_widths = [('gene_name', 3), ('violins_y_ticks', 2), + ('violins', 13), ('tissue_matched', 1), + ('fold_change', 2), ('tumor', 1), + ('normal', 1)] + if has_prediction: + grid_col_names_and_widths.extend([('epitopes', 5), ('hlas', 3), + ('num_patients', 2), + ('affinity', 3)]) + + grid_col_intervals_by_name = dict() + grid_cols = 0 + for name, width in grid_col_names_and_widths: + start_cols = grid_cols + grid_cols += width + grid_col_intervals_by_name[name] = (start_cols, grid_cols) + + grid = gridspec.GridSpec(grid_rows, grid_cols, wspace=0, hspace=0) + + filler_rows = list() + axes_by_name = dict() + for name, col_interval in grid_col_intervals_by_name.items(): + start_col, end_col = col_interval + axes = list() + + row = 0 + for _ in range(0, content_rows): + is_header = row == 0 + if not is_header: + filler_rows.append( + fig.add_subplot(grid[row, start_col:end_col])) + row += 1 + + start_row = row + row += content_rows_to_spacer + if is_header: + row += extra_header_room + + axes.append(fig.add_subplot( + grid[start_row:row, start_col:end_col])) + + axes_by_name[name] = (axes[0], axes[1:]) + + return axes_by_name, filler_rows + + +def make_shaded_dots_column(header_ax, dots_axes, header_text, events, + epitope_summary, summary_ratio_key, color): + header_ax.text(0.5, + 0, + 'vs. ', + horizontalalignment='center', + verticalalignment='bottom', + rotation=90, + transform=header_ax.transAxes, + size=FONT_SIZE_MEDIUM, + color=COLOR_BLACK, + fontweight='bold') + + y_min_display = header_ax.transAxes.transform_point((0, 0))[1] + pixels_per_large_char = points_to_pixels(FONT_SIZE_MEDIUM) + # 'vs. ' is about two large characters in the default variable width font + pixels_of_text = pixels_per_large_char * 2 + vs_space_y_max_display = y_min_display + pixels_of_text + + trans_display_to_axes = header_ax.transAxes.inverted().transform_point + vs_space_y_max_axes = trans_display_to_axes((0, vs_space_y_max_display))[1] + # put the header_text after 'vs. ' in a (possibly) different color + header_ax.text(0.5, + vs_space_y_max_axes, + header_text, + horizontalalignment='center', + verticalalignment='bottom', + rotation=90, + transform=header_ax.transAxes, + size=FONT_SIZE_MEDIUM, + color=color, + fontweight='bold') + + remove_spines_ticks_and_labels(header_ax) + header_ax.set_facecolor(COLOR_CREAM) + + for i, event in enumerate(events): + event_summary = epitope_summary.get(event) + if not event_summary: + return 'no event {} in epitope_summary'.format(event) + + hits, total = event_summary[summary_ratio_key] + if total == 0: + percent = 0 + else: + percent = hits / float(total) + + dots_ax = dots_axes[i] + xmin_display, ymin_display = dots_ax.transAxes.transform_point((0, 0)) + xmax_display, ymax_display = dots_ax.transAxes.transform_point((1, 1)) + x_spread_display = xmax_display - xmin_display + y_spread_display = ymax_display - ymin_display + # dots_ax is a rectangle. + # Normalize the axes scale so a circle is not distorted + x_mid_data = 0.5 + y_mid_data = 0.5 + if x_spread_display > y_spread_display: + new_scale = x_spread_display / float(y_spread_display) + half_new_scale = new_scale / 2.0 + dots_ax.set_xlim((0, new_scale)) + x_mid_data = half_new_scale + else: + new_scale = y_spread_display / float(x_spread_display) + half_new_scale = new_scale / 2.0 + dots_ax.set_ylim((0, new_scale)) + y_mid_data = half_new_scale + + circle = matplotlib.patches.Circle((x_mid_data, y_mid_data), + radius=0.25, + alpha=percent, + color=COLOR_OCHRE, + transform=dots_ax.transData) + dots_ax.add_patch(circle) + + remove_spines_ticks_and_labels(dots_ax) + + return None + + +def show_fold_change_values(header_ax, fc_axes, events, epitope_summary): + header_ax.text(0.5, + 0, + 'FC of tumor\nisoform', + horizontalalignment='center', + verticalalignment='bottom', + rotation=90, + transform=header_ax.transAxes, + size=FONT_SIZE_MEDIUM, + color=COLOR_BLACK, + fontweight='bold') + remove_spines_ticks_and_labels(header_ax) + header_ax.set_facecolor(COLOR_CREAM) + + for i, event in enumerate(events): + event_summary = epitope_summary.get(event) + if not event_summary: + return 'no event {} in epitope_summary'.format(event) + + fc = event_summary['fc_of_tumor_isoform'] + fc_ax = fc_axes[i] + fc_ax.text(0.5, + 0.5, + '{:.1f}'.format(fc), + horizontalalignment='center', + verticalalignment='center', + transform=fc_ax.transAxes, + size=FONT_SIZE_MEDIUM) + remove_spines_ticks_and_labels(fc_ax) + + return None + + +def make_top_underline_header(left_header_ax, right_header_ax, other_axes, + header_text): + # need to plot on all axes using display coordinates to avoid clipping + all_axes = [left_header_ax, right_header_ax] + other_axes + + x_min_display = left_header_ax.transAxes.transform_point((0, 0))[0] + x_max_display = right_header_ax.transAxes.transform_point((1, 0))[0] + x_mid_display = (x_min_display + x_max_display) / 2.0 + y_max_display = left_header_ax.transAxes.transform_point((0, 1))[1] + x_range_display = x_max_display - x_min_display + underline_margin_display = 0.02 * x_range_display + underline_x_min_display = x_min_display + underline_margin_display + underline_x_max_display = x_max_display - underline_margin_display + for ax in all_axes: + trans_display_to_axes = ax.transAxes.inverted().transform_point + x_mid_axes, y_max_axes = trans_display_to_axes( + (x_mid_display, y_max_display)) + ax.text(x_mid_axes, + y_max_axes, + header_text, + horizontalalignment='center', + verticalalignment='top', + transform=ax.transAxes, + size=FONT_SIZE_LARGE, + color=COLOR_BLACK, + fontweight='bold', + clip_on=True) + + text_y_min_display = y_max_display - points_to_pixels(FONT_SIZE_LARGE) + + for ax in all_axes: + trans_display_to_axes = ax.transAxes.inverted().transform_point + underline_x_min_axes, text_y_min_axes = trans_display_to_axes( + (underline_x_min_display, text_y_min_display)) + underline_x_max_axes = trans_display_to_axes( + (underline_x_max_display, 0))[0] + ax.plot([underline_x_min_axes, underline_x_max_axes], + [text_y_min_axes] * 2, + color=COLOR_BLACK, + linestyle='solid', + transform=ax.transAxes, + clip_on=True) + + +def make_shaded_dots_and_show_fold_change(events, epitope_summary, + axes_by_name): + tmn_header_ax, tmn_axes = axes_by_name['tissue_matched'] + t_header_ax, t_axes = axes_by_name['tumor'] + n_header_ax, n_axes = axes_by_name['normal'] + fc_header_ax, fc_axes = axes_by_name['fold_change'] + + error = make_shaded_dots_column( + tmn_header_ax, tmn_axes, 'Tissue Matched', events, epitope_summary, + 'tissue_matched_normal_panel', + COLOR_BY_PANEL_TYPE['tissue_matched_normal']) + if error: + return error + + error = make_shaded_dots_column(t_header_ax, t_axes, 'Tumor', events, + epitope_summary, 'tumor_panel', + COLOR_BY_PANEL_TYPE['tumor']) + if error: + return error + + error = make_shaded_dots_column(n_header_ax, n_axes, 'Normal', events, + epitope_summary, 'normal_panel', + COLOR_BY_PANEL_TYPE['normal']) + if error: + return error + + show_fold_change_values(fc_header_ax, fc_axes, events, epitope_summary) + + make_top_underline_header(tmn_header_ax, n_header_ax, + [t_header_ax, fc_header_ax], 'Summary') + + return None + + +def show_epitope_names(header_ax, epitope_axes, events, epitope_summary): + header_ax.text(0.5, + 0, + 'Junction\nepitopes', + horizontalalignment='center', + verticalalignment='bottom', + transform=header_ax.transAxes, + size=FONT_SIZE_LARGE, + color=COLOR_BLACK, + fontweight='bold') + remove_spines_ticks_and_labels(header_ax) + header_ax.set_facecolor(COLOR_LIGHT_VIOLET) + + for i, event in enumerate(events): + event_summary = epitope_summary.get(event) + if not event_summary: + return 'no event {} in epitope_summary'.format(event) + + epitopes = event_summary['epitopes'] + epitope_names = [e['epitope'] for e in epitopes[:2]] + name_text = '\n'.join(epitope_names) + epitope_ax = epitope_axes[i] + epitope_ax.text(0.5, + 0.5, + name_text, + horizontalalignment='center', + verticalalignment='center', + transform=epitope_ax.transAxes, + size=FONT_SIZE_MEDIUM) + remove_spines_ticks_and_labels(epitope_ax) + + return None + + +def show_hla_names(header_ax, hla_axes, events, epitope_summary): + header_ax.text(0.5, + 0, + 'Best\nHLA', + horizontalalignment='center', + verticalalignment='bottom', + transform=header_ax.transAxes, + size=FONT_SIZE_LARGE, + color=COLOR_BLACK, + fontweight='bold') + remove_spines_ticks_and_labels(header_ax) + header_ax.set_facecolor(COLOR_LIGHT_VIOLET) + + for i, event in enumerate(events): + event_summary = epitope_summary.get(event) + if not event_summary: + return 'no event {} in epitope_summary'.format(event) + + epitopes = event_summary['epitopes'] + epitope_hlas = [e['hla'] for e in epitopes[:2]] + hla_text = '\n'.join(epitope_hlas) + hla_ax = hla_axes[i] + hla_ax.text(0.5, + 0.5, + hla_text, + horizontalalignment='center', + verticalalignment='center', + transform=hla_ax.transAxes, + size=FONT_SIZE_MEDIUM) + remove_spines_ticks_and_labels(hla_ax) + + return None + + +def show_num_patients(header_ax, patient_axes, events, epitope_summary): + header_ax.text(0.5, + 0, + '# Pt.\nw/HLA', + horizontalalignment='center', + verticalalignment='bottom', + transform=header_ax.transAxes, + size=FONT_SIZE_LARGE, + color=COLOR_BLACK, + fontweight='bold') + remove_spines_ticks_and_labels(header_ax) + header_ax.set_facecolor(COLOR_LIGHT_VIOLET) + + for i, event in enumerate(events): + event_summary = epitope_summary.get(event) + if not event_summary: + return 'no event {} in epitope_summary'.format(event) + + epitopes = event_summary['epitopes'] + epitope_patients = [e['num_patients'] for e in epitopes[:2]] + patient_text = '\n'.join(epitope_patients) + patient_ax = patient_axes[i] + patient_ax.text(0.5, + 0.5, + patient_text, + horizontalalignment='center', + verticalalignment='center', + transform=patient_ax.transAxes, + size=FONT_SIZE_MEDIUM) + remove_spines_ticks_and_labels(patient_ax) + + return None + + +def show_binding_affinity(header_ax, affinity_axes, events, epitope_summary): + header_ax.text(0.5, + 0, + 'IC$_{50}$\n(nM)', + horizontalalignment='center', + verticalalignment='bottom', + transform=header_ax.transAxes, + size=FONT_SIZE_LARGE, + color=COLOR_BLACK, + fontweight='bold') + remove_spines_ticks_and_labels(header_ax) + header_ax.set_facecolor(COLOR_LIGHT_VIOLET) + + for i, event in enumerate(events): + event_summary = epitope_summary.get(event) + if not event_summary: + return 'no event {} in epitope_summary'.format(event) + + epitopes = event_summary['epitopes'] + affinities = [ + str(int(round(e['binding_affinity']))) for e in epitopes[:2] + ] + affinity_text = '\n'.join(affinities) + affinity_ax = affinity_axes[i] + affinity_ax.text(0.5, + 0.5, + affinity_text, + horizontalalignment='center', + verticalalignment='center', + transform=affinity_ax.transAxes, + size=FONT_SIZE_MEDIUM) + remove_spines_ticks_and_labels(affinity_ax) + + return None + + +def show_epitope_bindings(events, epitope_summary, axes_by_name): + epitope_header_ax, epitope_axes = axes_by_name['epitopes'] + hla_header_ax, hla_axes = axes_by_name['hlas'] + patient_header_ax, patient_axes = axes_by_name['num_patients'] + affinity_header_ax, affinity_axes = axes_by_name['affinity'] + + error = show_epitope_names(epitope_header_ax, epitope_axes, events, + epitope_summary) + if error: + return error + + error = show_hla_names(hla_header_ax, hla_axes, events, epitope_summary) + if error: + return error + + error = show_num_patients(patient_header_ax, patient_axes, events, + epitope_summary) + if error: + return error + + error = show_binding_affinity(affinity_header_ax, affinity_axes, events, + epitope_summary) + if error: + return error + + make_top_underline_header(epitope_header_ax, affinity_header_ax, + [hla_header_ax, patient_header_ax], + 'Predicted HLA-epitope binding') + + return None + + +def make_plots(psi_data_by_event, epitope_summary, group_to_panel_type, + out_file_name, has_prediction): + num_events = len(psi_data_by_event) + if num_events == 0: + return 'no events to plot' + + fig_width = 8 + if has_prediction: + fig_width = 12 + + # 1.5 inches per row. + # The header is given 2 rows worth of height when creating the grid. + fig_height = 1.5 * (num_events + 2) + fig = plt.figure(figsize=(fig_width, fig_height), constrained_layout=False) + + events = list(psi_data_by_event.keys()) + groups = psi_data_by_event[events[0]].columns + + axes_by_name, filler_rows = create_grid_of_axes(fig, num_events, + has_prediction) + for filler_row in filler_rows: + hide_ax(filler_row) + + error = make_violin_plots(events, psi_data_by_event, groups, + group_to_panel_type, axes_by_name) + if error: + return error + + error = make_shaded_dots_and_show_fold_change(events, epitope_summary, + axes_by_name) + if error: + return error + + if has_prediction: + error = show_epitope_bindings(events, epitope_summary, axes_by_name) + if error: + return error + + plt.savefig(out_file_name) + plt.close(fig) + return None + + +def filter_events(events, epitope_summary): + variants = list(events.keys()) + for variant in variants: + event_list = events[variant] + filtered = list() + for event in event_list: + # filter to events that are in epitope summary + if event in epitope_summary: + filtered.append(event) + # only keep up to 10 events + if len(filtered) == 10: + break + + events[variant] = filtered + + +def exit_with_error(error): + print(error, file=sys.stderr) + sys.exit(1) + + +def main(args): + event_type = args.splicing_event_type + has_prediction = not args.no_prediction + + parameters, error = read_parameters(args.parameter_fin) + if error: + exit_with_error(error) + + group_to_panel_type = get_group_to_panel_type(parameters) + groups = list(group_to_panel_type.keys()) + + group_to_matrix_f_name, group_to_matrix_index_f_name, error = get_matrix_file_names( + groups, parameters, event_type) + if error: + exit_with_error(error) + + events, error = get_events(parameters, args.screening_out_dir, event_type) + if error: + exit_with_error(error) + + epitope_summary, error = get_epitope_summary(args.screening_out_dir, + has_prediction, parameters, + event_type) + if error: + exit_with_error(error) + + filter_events(events, epitope_summary) + + psi_data_by_event, error = get_psi_data_by_event( + group_to_matrix_f_name, group_to_matrix_index_f_name, events) + if error: + exit_with_error(error) + + error = make_plots(psi_data_by_event, epitope_summary, group_to_panel_type, + args.out_file_name, has_prediction) + if error: + exit_with_error(error) diff --git a/IRIS/config.py b/IRIS/config.py index ee35241..945bf89 100644 --- a/IRIS/config.py +++ b/IRIS/config.py @@ -10,7 +10,7 @@ #import yaml -CURRENT_VERSION = "v1.0" +CURRENT_VERSION = "v2.0.0" def update_progress(progress): @@ -40,22 +40,12 @@ def file_len(fin): # For screening and translation -BRAIN_BLACKLIST_PATH = resource_filename('IRIS.data','brain_blacklistMay.txt') +BRAIN_BLACKLIST_PATH = resource_filename('IRIS.data','blacklist.brain_2020.txt') ORF_MAP_PATH = resource_filename('IRIS.data','uniprot2gtf.blastout.uniprotAll.txt') ## For TCR mapping EXTRACELLULAR_FEATURES_UNIPROT2GTF_MAP_PATH = resource_filename('IRIS.data','features.uniprot2gtf.ExtraCell.txt') -# ## For HLA typing -# SEQ2HLA_PATH = resource_filename('IRIS.utilities.seq2hla', 'seq2HLA.py') -# FOURDIGITS_PATH = resource_filename('IRIS.utilities.seq2hla', 'fourdigits.py') - -## For qsub -QSUB_PREDICTION_CONFIG='h_data=15G,h_rt=5:00:00' -QSUB_ALIGNMENT_CONFIG='h_data=38G,h_rt=4:30:00' -QSUB_EXPRESSION_CONFIG='h_data=8G,h_rt=14:00:00' -QSUB_RMATS_PREP_CONFIG='h_data=4G,h_rt=5:00:00' - ## For proteogenomics UNIPROT_ENSG_ID_MAP_PATH = resource_filename('IRIS.data','UniprotENSGmap.txt') diff --git a/IRIS/data/blacklist.brain_2020.txt b/IRIS/data/blacklist.brain_2020.txt new file mode 100644 index 0000000..0441015 --- /dev/null +++ b/IRIS/data/blacklist.brain_2020.txt @@ -0,0 +1,369 @@ +ENSG00000005810:MYCBP2:chr13:-:77692474:77692654:77673148:77695507 +ENSG00000006282:SPATA20:chr17:+:48625080:48625315:48624646:48625643 +ENSG00000008394:MGST1:chr12:+:16507164:16507204:16500644:16510538 +ENSG00000008869:HEATR5B:chr2:-:37217790:37217942:37216002:37227728 +ENSG00000011465:DCN:chr12:-:91573138:91573463:91572362:91576431 +ENSG00000013561:RNF14:chr5:+:141350268:141350442:141348732:141353147 +ENSG00000031823:RANBP3:chr19:-:5957928:5957979:5941846:5978071 +ENSG00000047579:DTNBP1:chr6:-:15651543:15651639:15638035:15652317 +ENSG00000050130:JKAMP:chr14:+:59953398:59953522:59951311:59954391 +ENSG00000050130:JKAMP:chr14:+:59953448:59953522:59951311:59954391 +ENSG00000050426:LETMD1:chr12:+:51447594:51447643:51442968:51450132 +ENSG00000055917:PUM2:chr2:-:20478343:20478580:20463221:20482707 +ENSG00000055950:MRPL43:chr10:-:102743704:102743831:102743574:102746505 +ENSG00000055950:MRPL43:chr10:-:102746846:102746953:102743574:102747069 +ENSG00000059588:TARBP1:chr1:-:234536598:234536694:234534299:234536926 +ENSG00000061676:NCKAP1:chr2:-:183889705:183889723:183888644:183902719 +ENSG00000062598:ELMO2:chr20:-:45027335:45027410:45023171:45035186 +ENSG00000063854:HAGH:chr16:-:1876507:1876603:1873038:1876712 +ENSG00000064115:TM7SF3:chr12:-:27156168:27156323:27152609:27167010 +ENSG00000064115:TM7SF3:chr12:-:27156168:27156338:27152609:27167010 +ENSG00000065029:ZNF76:chr6:+:35261527:35261692:35260821:35262232 +ENSG00000067064:IDI1:chr10:-:1089938:1090111:1089333:1094803 +ENSG00000067836:ROGDI:chr16:-:4851050:4851322:4850579:4851503 +ENSG00000071564:TCF3:chr19:-:1612205:1612432:1611848:1615283 +ENSG00000073350:LLGL2:chr17:+:73570533:73570576:73570341:73570690 +ENSG00000074266:EED:chr11:+:85963189:85963282:85961490:85966263 +ENSG00000075234:TTC38:chr22:+:46688099:46688225:46685796:46688687 +ENSG00000075711:DLG1:chr3:-:196802707:196802741:196796131:196803456 +ENSG00000078549:ADCYAP1R1:chr7:+:31120227:31120248:31117713:31123755 +ENSG00000078668:VDAC3:chr8:+:42254195:42254198:42252651:42256229 +ENSG00000079819:EPB41L2:chr6:-:131188598:131188721:131184858:131206235 +ENSG00000080822:CLDND1:chr3:-:98240496:98240547:98240281:98241692 +ENSG00000080823:MOK:chr14:-:102729882:102729953:102718332:102749814 +ENSG00000080845:DLGAP4:chr20:+:35127990:35128079:35127724:35128601 +ENSG00000085274:MYNN:chr3:+:169501264:169501348:169500431:169502409 +ENSG00000085449:WDFY1:chr2:-:224746658:224746789:224744949:224749364 +ENSG00000087206:UIMC1:chr5:-:176395555:176396292:176385155:176396601 +ENSG00000088543:C3orf18:chr3:-:50602896:50603292:50599178:50604893 +ENSG00000088833:NSFL1C:chr20:-:1436358:1436515:1435777:1438844 +ENSG00000089639:GMIP:chr19:-:19745315:19745512:19744999:19745600 +ENSG00000090006:LTBP4:chr19:+:41122794:41122926:41120352:41123005 +ENSG00000092020:PPP2R3C:chr14:-:35579024:35579053:35577442:35579730 +ENSG00000092199:HNRNPC:chr14:-:21731469:21731495:21702388:21737456 +ENSG00000092199:HNRNPC:chr14:-:21731469:21731741:21702388:21737456 +ENSG00000092330:TINF2:chr14:-:24711095:24711127:24710982:24711346 +ENSG00000092421:SEMA6A:chr5:-:115803278:115803443:115783507:115808768 +ENSG00000092841:MYL6:chr12:+:56553281:56553406:56552495:56553758 +ENSG00000099204:ABLIM1:chr10:-:116207638:116207779:116205162:116211382 +ENSG00000099330:OCEL1:chr19:+:17339611:17339724:17339118:17339817 +ENSG00000099622:CIRBP:chr19:+:1273599:1273714:1272050:1274305 +ENSG00000099785:MARCH2:chr19:+:8483420:8483691:8478304:8486672 +ENSG00000099785:MARCH2:chr19:+:8483625:8483691:8478304:8486672 +ENSG00000099875:MKNK2:chr19:-:2039629:2039855:2037828:2040132 +ENSG00000099889:ARVCF:chr22:-:19958738:19958858:19958266:19959408 +ENSG00000099957:P2RX6:chr22:+:21376964:21377040:21372349:21377230 +ENSG00000100138:SNU13:chr22:-:42078359:42078591:42076368:42084797 +ENSG00000100209:HSCB:chr22:+:29139869:29139911:29138319:29147228 +ENSG00000100209:HSCB:chr22:+:29140602:29140697:29139966:29141851 +ENSG00000100227:POLDIP3:chr22:-:42997975:42998113:42995799:42998775 +ENSG00000100288:CHKB:chr22:-:51018485:51018511:51018231:51018618 +ENSG00000100288:CHKB:chr22:-:51020457:51020524:51020291:51020677 +ENSG00000100379:KCTD17:chr22:+:37456862:37456962:37455478:37457578 +ENSG00000100505:TRIM9:chr14:-:51449659:51449683:51448821:51464767 +ENSG00000101150:TPD52L2:chr20:+:62518916:62518958:62514173:62520542 +ENSG00000101187:SLCO4A1:chr20:+:61299506:61299536:61299262:61299828 +ENSG00000101187:SLCO4A1:chr20:+:61299509:61299536:61299262:61299828 +ENSG00000101363:MANBAL:chr20:+:35927165:35927282:35918089:35929610 +ENSG00000102878:HSF4:chr16:+:67201205:67201305:67201125:67201406 +ENSG00000102977:ACD:chr16:-:67693131:67693166:67692982:67693439 +ENSG00000103121:CMC2:chr16:-:81014374:81014484:81010076:81015410 +ENSG00000103148:NPRL3:chr16:-:167299:167374:162774:180520 +ENSG00000104231:ZFAND1:chr8:-:82629483:82629523:82627349:82630416 +ENSG00000104325:DECR1:chr8:+:91029529:91029554:91013789:91031136 +ENSG00000104723:TUSC3:chr8:+:15615299:15615364:15605974:15621711 +ENSG00000105127:AKAP8:chr19:-:15479877:15480035:15479133:15480956 +ENSG00000105223:PLD3:chr19:+:40871459:40871492:40854631:40872325 +ENSG00000105223:PLD3:chr19:+:40871459:40871492:40854675:40872290 +ENSG00000105223:PLD3:chr19:+:40871459:40871837:40854675:40872325 +ENSG00000105223:PLD3:chr19:+:40871568:40871837:40854675:40872325 +ENSG00000105223:PLD3:chr19:+:40871624:40871837:40854675:40872325 +ENSG00000105278:ZFR2:chr19:-:3855399:3855557:3852612:3868962 +ENSG00000105552:BCAT2:chr19:-:49311012:49311493:49310331:49314240 +ENSG00000106125:MINDY4:chr7:+:30921795:30921976:30915271:30922535 +ENSG00000106133:NSUN5P2:chr7:-:72422690:72422834:72420735:72425163 +ENSG00000106772:PRUNE2:chr9:-:79234255:79234303:79229516:79239938 +ENSG00000107863:ARHGAP21:chr10:-:24879134:24879408:24878231:24880153 +ENSG00000108219:TSPAN14:chr10:+:82228302:82228443:82214127:82248972 +ENSG00000108669:CYTH1:chr17:-:76692088:76692091:76688575:76694350 +ENSG00000108848:LUC7L3:chr17:+:48826584:48826705:48824063:48828107 +ENSG00000109083:IFT20:chr17:-:26659171:26659207:26659013:26662365 +ENSG00000109381:ELF2:chr4:-:139988837:139988924:139983211:139993019 +ENSG00000110074:FOXRED1:chr11:+:126141114:126141552:126139186:126142863 +ENSG00000111144:LTA4H:chr12:-:96397615:96397759:96396842:96400091 +ENSG00000111237:VPS29:chr12:-:110937261:110937351:110934008:110939853 +ENSG00000111596:CNOT2:chr12:+:70687850:70688074:70672054:70704674 +ENSG00000111907:TPD52L1:chr6:+:125578243:125578326:125574901:125583979 +ENSG00000113456:RAD1:chr5:-:34913574:34913683:34911917:34914799 +ENSG00000114698:PLSCR4:chr3:-:145918821:145918864:145914580:145924312 +ENSG00000114993:RTKN:chr2:-:74666677:74667104:74659793:74667479 +ENSG00000115325:DOK1:chr2:+:74783020:74783205:74782795:74783434 +ENSG00000115414:FN1:chr2:-:216257653:216257926:216256537:216259250 +ENSG00000115524:SF3B1:chr2:-:198283619:198283675:198283312:198285151 +ENSG00000116473:RAP1A:chr1:+:112170091:112170185:112162556:112233955 +ENSG00000117523:PRRC2C:chr1:+:171560290:171560339:171557644:171560725 +ENSG00000117616:RSRP1:chr1:-:25570944:25570989:25570715:25571640 +ENSG00000119328:FAM206A:chr9:+:111698587:111698717:111697969:111701475 +ENSG00000119522:DENND1A:chr9:-:126150008:126150137:126146192:126165680 +ENSG00000119979:FAM45A:chr10:+:120864275:120864534:120863709:120867459 +ENSG00000121716:PILRB:chr7:+:99951517:99951635:99950893:99952765 +ENSG00000121964:GTDC1:chr2:-:144764748:144765102:144728329:144899448 +ENSG00000122085:MTERF4:chr2:-:242035807:242035853:242033847:242036657 +ENSG00000122085:MTERF4:chr2:-:242038810:242039309:242036842:242041667 +ENSG00000122203:KIAA1191:chr5:-:175786483:175786570:175782752:175788604 +ENSG00000122203:KIAA1191:chr5:-:175786813:175786921:175782752:175788604 +ENSG00000122257:RBBP6:chr16:+:24579112:24579214:24578826:24580065 +ENSG00000123159:GIPC1:chr19:-:14603668:14603724:14602555:14606848 +ENSG00000123562:MORF4L2:chrX:-:102939608:102939657:102933579:102940098 +ENSG00000123595:RAB9A:chrX:+:13721932:13722021:13707407:13726839 +ENSG00000124074:ENKD1:chr16:-:67698898:67699071:67697965:67699973 +ENSG00000124140:SLC12A5:chr20:+:44651481:44651710:44650525:44652007 +ENSG00000124140:SLC12A5:chr20:+:44651568:44651710:44650525:44652007 +ENSG00000124356:STAMBP:chr2:+:74056531:74056637:74056123:74057971 +ENSG00000125386:FAM193A:chr4:+:2691232:2691431:2674099:2692424 +ENSG00000125388:GRK4:chr4:+:3039100:3039238:3037250:3042298 +ENSG00000125462:C1orf61:chr1:-:156377633:156377767:156374393:156399167 +ENSG00000125462:C1orf61:chr1:-:156386560:156386656:156384545:156399167 +ENSG00000126214:KLC1:chr14:+:104151322:104151373:104145882:104153417 +ENSG00000126261:UBA2:chr19:+:34920997:34921091:34919475:34921480 +ENSG00000126858:RHOT1:chr17:+:30538134:30538257:30535328:30551634 +ENSG00000127419:TMEM175:chr4:+:941903:941942:926328:944208 +ENSG00000128463:EMC4:chr15:+:34520637:34520790:34520047:34521953 +ENSG00000129103:SUMF2:chr7:+:56141806:56141911:56140804:56142278 +ENSG00000129103:SUMF2:chr7:+:56141866:56141911:56140804:56144526 +ENSG00000129538:RNASE1:chr14:-:21270407:21270478:21270252:21270955 +ENSG00000129538:RNASE1:chr14:-:21270407:21270490:21270252:21270955 +ENSG00000129646:QRICH2:chr17:-:74286072:74286158:74283978:74287095 +ENSG00000129993:CBFA2T3:chr16:-:88964485:88964560:88958893:88967911 +ENSG00000130396:AFDN:chr6:+:168355140:168355173:168352867:168363112 +ENSG00000131051:RBM39:chr20:-:34328446:34328519:34326939:34328745 +ENSG00000131095:GFAP:chr17:-:42989039:42989165:42988689:42991095 +ENSG00000131473:ACLY:chr17:-:40052872:40052902:40049427:40054001 +ENSG00000132199:ENOSF1:chr18:-:677344:677444:675402:678695 +ENSG00000132199:ENOSF1:chr18:-:678695:678737:675402:683245 +ENSG00000132341:RAN:chr12:+:131357128:131357162:131356671:131357380 +ENSG00000132613:MTSS1L:chr16:-:70713217:70713226:70712312:70713532 +ENSG00000133612:AGAP3:chr7:+:150817606:150817654:150817232:150819811 +ENSG00000134042:MRO:chr18:-:48327718:48327874:48326513:48331523 +ENSG00000134574:DDB2:chr11:+:47256307:47256485:47256223:47256820 +ENSG00000134717:BTF3L4:chr1:+:52549011:52549213:52530610:52551783 +ENSG00000134769:DTNA:chr18:+:32446074:32446095:32444026:32455202 +ENSG00000134779:TPGS2:chr18:-:34385336:34385360:34380274:34387809 +ENSG00000134851:TMEM165:chr4:+:56277780:56278006:56262563:56290704 +ENSG00000134851:TMEM165:chr4:+:56283969:56284015:56283414:56284214 +ENSG00000135502:SLC26A10:chr12:+:58017785:58017881:58017696:58018648 +ENSG00000136114:THSD1:chr13:-:52960162:52960321:52952924:52971366 +ENSG00000136270:TBRG4:chr7:-:45143697:45143855:45143042:45145039 +ENSG00000136754:ABI1:chr10:-:27047990:27048167:27040712:27052808 +ENSG00000136878:USP20:chr9:+:132627581:132627660:132625578:132630283 +ENSG00000137210:TMEM14B:chr6:+:10755374:10755465:10749931:10770309 +ENSG00000137501:SYTL2:chr11:-:85428525:85428769:85425550:85429832 +ENSG00000137776:SLTM:chr15:-:59191667:59192082:59191051:59193458 +ENSG00000137776:SLTM:chr15:-:59204761:59204809:59193486:59209133 +ENSG00000138443:ABI2:chr2:+:204259422:204259569:204255866:204260378 +ENSG00000138593:SECISBP2L:chr15:-:49311614:49311749:49309825:49319561 +ENSG00000139154:AEBP2:chr12:+:19667604:19667718:19665399:19671020 +ENSG00000139197:PEX5:chr12:+:7354836:7354947:7354437:7355207 +ENSG00000139631:CSAD:chr12:-:53566129:53566220:53565772:53567128 +ENSG00000140416:TPM1:chr15:+:63336225:63336351:63336030:63349183 +ENSG00000140464:PML:chr15:+:74324912:74325056:74317268:74325496 +ENSG00000140538:NTRK3:chr15:-:88671941:88671965:88670457:88678331 +ENSG00000140750:ARHGAP17:chr16:-:24950684:24950918:24946960:24953307 +ENSG00000141127:PRPSAP2:chr17:+:18770569:18770647:18769265:18775895 +ENSG00000141646:SMAD4:chr18:+:48584494:48584614:48581363:48586235 +ENSG00000141646:SMAD4:chr18:+:48584709:48584826:48581363:48586235 +ENSG00000142208:AKT1:chr14:-:105259463:105259547:105259059:105261820 +ENSG00000142208:AKT1:chr14:-:105259463:105259641:105259059:105261820 +ENSG00000142252:GEMIN7:chr19:+:45583164:45583287:45582537:45593364 +ENSG00000143303:RRNAD1:chr1:+:156703800:156703844:156703312:156705516 +ENSG00000143303:RRNAD1:chr1:+:156703800:156704285:156703312:156705516 +ENSG00000143353:LYPLAL1:chr1:+:219366471:219366593:219352588:219383873 +ENSG00000143537:ADAM15:chr1:+:155034379:155034451:155033308:155034720 +ENSG00000143727:ACP1:chr2:+:272036:272065:271939:272191 +ENSG00000143742:SRP9:chr1:+:225974563:225974687:225971070:225976941 +ENSG00000143774:GUK1:chr1:+:228328824:228328989:228328064:228333211 +ENSG00000143776:CDC42BPA:chr1:-:227300371:227300614:227300123:227307504 +ENSG00000144199:FAHD2B:chr2:-:97757198:97757449:97756062:97760437 +ENSG00000144741:SLC25A26:chr3:+:66396787:66396832:66313803:66419901 +ENSG00000145016:RUBCN:chr3:-:197417944:197418019:197411088:197420585 +ENSG00000145349:CAMK2D:chr4:-:114424091:114424133:114421667:114430793 +ENSG00000145740:SLC30A5:chr5:+:68423830:68423959:68419252:68425273 +ENSG00000145782:ATG12:chr5:-:115176193:115176309:115173461:115177086 +ENSG00000146267:FAXC:chr6:-:99781226:99781423:99739696:99790773 +ENSG00000148053:NTRK2:chr9:+:87284594:87284803:87284338:87285291 +ENSG00000148057:IDNK:chr9:+:86242921:86243126:86238136:86243787 +ENSG00000148180:GSN:chr9:+:124062333:124062404:124045670:124064240 +ENSG00000148341:SH3GLB2:chr9:-:131771731:131771746:131771070:131772049 +ENSG00000148399:DPH7:chr9:-:140472028:140472055:140470619:140473076 +ENSG00000148481:MINDY3:chr10:-:15828559:15828647:15824225:15831245 +ENSG00000148660:CAMK2G:chr10:-:75585078:75585105:75583842:75597225 +ENSG00000148840:PPRC1:chr10:+:103906428:103907149:103904064:103908128 +ENSG00000149294:NCAM1:chr11:+:113117089:113117092:113105886:113126598 +ENSG00000149531:FRG1BP:chr20:+:29625872:29625984:29623254:29628226 +ENSG00000150967:ABCB9:chr12:-:123425353:123425542:123424831:123428937 +ENSG00000151276:MAGI1:chr3:-:65361419:65361623:65350621:65364935 +ENSG00000152465:NMT2:chr10:-:15182985:15183028:15177417:15183420 +ENSG00000153391:INO80C:chr18:-:33069295:33069349:33060527:33077682 +ENSG00000154134:ROBO3:chr11:+:124747056:124747205:124746828:124747414 +ENSG00000154134:ROBO3:chr11:+:124747162:124747205:124746828:124747414 +ENSG00000154845:PPP4R1:chr18:-:9562919:9563044:9562073:9563375 +ENSG00000155897:ADCY8:chr8:-:131859669:131859759:131848695:131861847 +ENSG00000156219:ART3:chr4:+:77025749:77025782:77025122:77033538 +ENSG00000156345:CDK20:chr9:-:90584710:90584834:90584264:90585690 +ENSG00000157538:DSCR3:chr21:-:38605662:38605743:38604752:38610760 +ENSG00000158856:DMTN:chr8:+:21938315:21938381:21938136:21938623 +ENSG00000159214:CCDC24:chr1:+:44457883:44458059:44457676:44458194 +ENSG00000159899:NPR2:chr9:+:35802728:35802800:35802604:35805507 +ENSG00000160072:ATAD3B:chr1:+:1425071:1425191:1424654:1425636 +ENSG00000160323:ADAMTS13:chr9:+:136303365:136303486:136303017:136304486 +ENSG00000160408:ST6GALNAC6:chr9:-:130660104:130660289:130658611:130661781 +ENSG00000160584:SIK3:chr11:-:116738661:116738805:116734534:116741046 +ENSG00000160767:FAM189B:chr1:-:155223649:155223769:155223523:155224443 +ENSG00000161203:AP2M1:chr3:+:183898432:183898529:183898039:183898636 +ENSG00000161249:DMKN:chr19:-:35996840:35996888:35996667:36001085 +ENSG00000161692:DBF4B:chr17:+:42818703:42818820:42815792:42824450 +ENSG00000161955:TNFSF13:chr17:+:7462940:7463019:7462614:7463162 +ENSG00000162065:TBC1D24:chr16:+:2547710:2547728:2547114:2548238 +ENSG00000162066:AMDHD2:chr16:+:2577561:2577616:2571124:2578060 +ENSG00000162430:SELENON:chr1:+:26128506:26128608:26127651:26131632 +ENSG00000162910:MRPL55:chr1:-:228296137:228296175:228296019:228296849 +ENSG00000162910:MRPL55:chr1:-:228296137:228296722:228296019:228296849 +ENSG00000162910:MRPL55:chr1:-:228296137:228296722:228296022:228296849 +ENSG00000162910:MRPL55:chr1:-:228296141:228296209:228296019:228296849 +ENSG00000162910:MRPL55:chr1:-:228296655:228296722:228295570:228296849 +ENSG00000162910:MRPL55:chr1:-:228296655:228296722:228296175:228296849 +ENSG00000162961:DPY30:chr2:-:32108454:32108531:32095021:32142994 +ENSG00000163170:BOLA3:chr2:-:74369398:74369487:74362785:74372315 +ENSG00000163875:MEAF6:chr1:-:37962307:37962337:37962205:37967404 +ENSG00000164615:CAMLG:chr5:+:134076752:134077213:134074482:134086448 +ENSG00000164830:OXR1:chr8:+:107749747:107749828:107738537:107751685 +ENSG00000165416:SUGT1:chr13:+:53235609:53235705:53233384:53236783 +ENSG00000165644:COMTD1:chr10:-:76995373:76995501:76995130:76995592 +ENSG00000165669:FAM204A:chr10:-:120101238:120101439:120095935:120101781 +ENSG00000165795:NDRG2:chr14:-:21492188:21492255:21491480:21493187 +ENSG00000165795:NDRG2:chr14:-:21492188:21492298:21491480:21493187 +ENSG00000165949:IFI27:chr14:+:94577970:94578119:94577143:94581196 +ENSG00000165949:IFI27:chr14:+:94578004:94578119:94577143:94581196 +ENSG00000166140:ZFYVE19:chr15:+:41104896:41105100:41102955:41105535 +ENSG00000166295:ANAPC16:chr10:+:73983645:73983814:73975830:73990123 +ENSG00000166352:C11orf74:chr11:+:36657600:36657667:36631789:36669565 +ENSG00000167508:MVD:chr16:-:88726927:88727045:88725128:88729418 +ENSG00000167515:TRAPPC2L:chr16:+:88925752:88925851:88925197:88925980 +ENSG00000167985:SDHAF2:chr11:+:61205475:61205585:61197654:61213412 +ENSG00000168000:BSCL2:chr11:-:62472772:62473030:62462183:62474580 +ENSG00000168137:SETD5:chr3:+:9478533:9478572:9477590:9482139 +ENSG00000168591:TMUB2:chr17:+:42265043:42265111:42264477:42266389 +ENSG00000168591:TMUB2:chr17:+:42265274:42265377:42265111:42266302 +ENSG00000168765:GSTM4:chr1:+:110202411:110202581:110201732:110203786 +ENSG00000168781:PPIP5K1:chr15:-:43856300:43856363:43851130:43857064 +ENSG00000168958:MFF:chr2:+:228207460:228207535:228205096:228220392 +ENSG00000169045:HNRNPH1:chr5:-:179046269:179046361:179045324:179047892 +ENSG00000169180:XPO6:chr16:-:28164250:28164285:28164106:28167394 +ENSG00000169231:THBS3:chr1:-:155170241:155170401:155169904:155170687 +ENSG00000169764:UGP2:chr2:+:64069672:64069733:64069338:64083439 +ENSG00000170632:ARMC10:chr7:+:102732923:102733100:102727211:102737723 +ENSG00000170919:TPT1-AS1:chr13:+:45964848:45965037:45963955:45965166 +ENSG00000170919:TPT1-AS1:chr13:+:45964892:45965037:45963955:45965166 +ENSG00000170954:ZNF415:chr19:-:53618462:53618560:53613161:53618956 +ENSG00000171202:TMEM126A:chr11:+:85361355:85361385:85359133:85365106 +ENSG00000171792:RHNO1:chr12:+:2994327:2994700:2986448:2997076 +ENSG00000171792:RHNO1:chr12:+:2994428:2994700:2986448:2997076 +ENSG00000172046:USP19:chr3:-:49154491:49154791:49154376:49154869 +ENSG00000172508:CARNS1:chr11:+:67183646:67185148:67183234:67185901 +ENSG00000172508:CARNS1:chr11:+:67184877:67185148:67183234:67185901 +ENSG00000172663:TMEM134:chr11:-:67232526:67232738:67232327:67234782 +ENSG00000172785:CBWD1:chr9:-:163977:164037:162469:172080 +ENSG00000172890:NADSYN1:chr11:+:71191264:71191320:71189515:71191800 +ENSG00000173599:PC:chr11:-:66721719:66721907:66719939:66725792 +ENSG00000173660:UQCRH:chr1:+:46775568:46775716:46774799:46775826 +ENSG00000173744:AGFG1:chr2:+:228395806:228395926:228389631:228398264 +ENSG00000174446:SNAPC5:chr15:-:66787240:66787757:66786890:66789979 +ENSG00000175203:DCTN2:chr12:-:57932265:57932307:57929628:57939810 +ENSG00000175309:PHYKPL:chr5:-:177651642:177651730:177651565:177652355 +ENSG00000176261:ZBTB8OS:chr1:-:33100302:33100393:33099673:33116033 +ENSG00000176261:ZBTB8OS:chr1:-:33100368:33100393:33093145:33116033 +ENSG00000176261:ZBTB8OS:chr1:-:33100368:33100393:33099711:33116033 +ENSG00000177225:PDDC1:chr11:-:775065:775142:774113:777398 +ENSG00000177410:ZFAS1:chr20:+:47897021:47897107:47895745:47905581 +ENSG00000177479:ARIH2:chr3:+:48962150:48962404:48960244:48964894 +ENSG00000177697:CD151:chr11:+:834457:834591:833026:836062 +ENSG00000177697:CD151:chr11:+:834529:834591:833022:836062 +ENSG00000177697:CD151:chr11:+:834529:834591:833026:836062 +ENSG00000177830:CHID1:chr11:-:908541:908645:904859:910774 +ENSG00000178104:PDE4DIP:chr1:-:144859758:144859998:144857042:144863317 +ENSG00000178498:DTX3:chr12:+:57999126:57999514:57998641:57999972 +ENSG00000178498:DTX3:chr12:+:57999353:57999514:57998641:57999972 +ENSG00000178761:FAM219B:chr15:-:75198664:75198706:75197572:75198927 +ENSG00000178927:C17orf62:chr17:-:80405455:80405522:80404572:80407045 +ENSG00000179818:PCBP1-AS1:chr2:-:70310669:70310819:70278437:70312678 +ENSG00000181038:METTL23:chr17:+:74725771:74725876:74723260:74729059 +ENSG00000182179:UBA7:chr3:-:49846822:49846897:49846592:49846974 +ENSG00000182534:MXRA7:chr17:-:74679928:74680009:74676961:74681153 +ENSG00000182796:TMEM198B:chr12:+:56228634:56229399:56227325:56229909 +ENSG00000182872:RBM10:chrX:+:47034417:47034491:47032596:47035898 +ENSG00000182985:CADM1:chr11:-:115069125:115069158:115049495:115085327 +ENSG00000183780:SLC35F3:chr1:+:234444846:234445066:234367487:234452347 +ENSG00000185046:ANKS1B:chr12:-:99201621:99201693:99194903:99222951 +ENSG00000185324:CDK10:chr16:+:89755659:89755732:89753167:89756960 +ENSG00000185485:SDHAP1:chr3:-:195695089:195695256:195694861:195698193 +ENSG00000185565:LSAMP:chr3:-:115535460:115535493:115529261:115560691 +ENSG00000185565:LSAMP:chr3:-:115553408:115553444:115529261:115560691 +ENSG00000185596:WASH3P:chr15:+:102512798:102512897:102501844:102513103 +ENSG00000186575:NF2:chr22:+:30079008:30079068:30077590:30090740 +ENSG00000186998:EMID1:chr22:+:29622478:29622540:29611619:29627008 +ENSG00000187735:TCEA1:chr8:-:54915451:54915652:54912610:54922989 +ENSG00000188338:SLC38A3:chr3:+:50244768:50244910:50242781:50251581 +ENSG00000188343:FAM92A:chr8:+:94714747:94714771:94713686:94715847 +ENSG00000189171:S100A13:chr1:-:153599958:153600074:153599009:153600596 +ENSG00000189171:S100A13:chr1:-:153602925:153603132:153600713:153603486 +ENSG00000189403:HMGB1:chr13:-:31035775:31035825:31035670:31036674 +ENSG00000196586:MYO6:chr6:+:76608089:76608128:76602407:76617321 +ENSG00000196839:ADA:chr20:-:43251228:43251293:43249788:43251469 +ENSG00000196923:PDLIM7:chr5:-:176918404:176918421:176918147:176918807 +ENSG00000196923:PDLIM7:chr5:-:176918807:176918977:176918421:176919405 +ENSG00000196923:PDLIM7:chr5:-:176918807:176918996:176918421:176919405 +ENSG00000197451:HNRNPAB:chr5:+:177637132:177637273:177636448:177637553 +ENSG00000197798:FAM118B:chr11:+:126099119:126099231:126081725:126104889 +ENSG00000197971:MBP:chr18:-:74700255:74700483:74696850:74700832 +ENSG00000197971:MBP:chr18:-:74728787:74728802:74702016:74728869 +ENSG00000198276:UCKL1:chr20:-:62573822:62573845:62572561:62575005 +ENSG00000198794:SCAMP5:chr15:+:75299928:75300069:75288014:75304132 +ENSG00000204580:DDR1:chr6:+:30863180:30863291:30862448:30864397 +ENSG00000204843:DCTN1:chr2:-:74600054:74600075:74598855:74604558 +ENSG00000205581:HMGN1:chr21:-:40717755:40717884:40717200:40720217 +ENSG00000205981:DNAJC19:chr3:-:180704785:180704810:180703784:180705810 +ENSG00000214078:CPNE1:chr20:-:34243123:34243266:34220845:34246851 +ENSG00000214078:CPNE1:chr20:-:34243123:34243266:34220845:34252681 +ENSG00000215788:TNFRSF25:chr1:-:6523131:6523187:6523016:6524434 +ENSG00000215908:CROCCP2:chr1:-:16969261:16969345:16961663:16969522 +ENSG00000217555:CKLF:chr16:+:66592092:66592251:66586696:66597024 +ENSG00000221978:CCNL2:chr1:-:1328169:1328183:1326245:1328775 +ENSG00000223482:NUTM2A-AS1:chr10:-:89067713:89067817:89048252:89086386 +ENSG00000225177:RP11-390P2.4:chr6:+:139015529:139015684:139013754:139017733 +ENSG00000228315:GUSBP11:chr22:-:24029017:24029182:24026054:24032422 +ENSG00000228315:GUSBP11:chr22:-:24042912:24043032:24037704:24047615 +ENSG00000228439:TSTD3:chr6:+:99979238:99979412:99973985:99979507 +ENSG00000231312:AC007246.3:chr2:+:39680979:39681072:39664676:39743554 +ENSG00000231312:AC007246.3:chr2:+:39681437:39681506:39664676:39743554 +ENSG00000232527:RP11-14N7.2:chr1:+:148933290:148933368:148932920:148951244 +ENSG00000234171:RNASEH1-AS1:chr2:+:3607037:3607319:3606588:3608906 +ENSG00000234608:MAPKAPK5-AS1:chr12:-:112279130:112279274:112278281:112279487 +ENSG00000237441:RGL2:chr6:-:33266231:33266428:33264892:33266646 +ENSG00000237651:C2orf74:chr2:+:61384996:61385147:61372331:61386452 +ENSG00000239382:ALKBH6:chr19:-:36504415:36504490:36504324:36505076 +ENSG00000239779:WBP1:chr2:+:74686123:74686225:74685798:74686769 +ENSG00000239779:WBP1:chr2:+:74686564:74686679:74685798:74686769 +ENSG00000239779:WBP1:chr2:+:74686604:74686689:74685798:74686769 +ENSG00000241231:RP11-275H4.1:chr3:-:181157764:181157822:181156487:181160161 +ENSG00000243147:MRPL33:chr2:+:27997290:27997397:27995559:28002299 +ENSG00000245958:RP11-33B1.1:chr4:+:120418965:120419058:120415678:120433505 +ENSG00000247828:TMEM161B-AS1:chr5:+:87577858:87577923:87566402:87732089 +ENSG00000260807:RP11-161M6.2:chr16:-:1026326:1026400:1025995:1026777 +ENSG00000269313:MAGIX:chrX:+:49021200:49021428:49021127:49021527 +ENSG00000269313:MAGIX:chrX:+:49021245:49021428:49021127:49021527 +ENSG00000269893:SNHG8:chr4:+:119200098:119200292:119199947:119200543 +ENSG00000270249:RP11-514P8.7:chr7:-:102207028:102207183:102182109:102207443 +ENSG00000275052:PPP4R3B:chr2:-:55825551:55826175:55816092:55831113 +ENSG00000276087:RP11-507M3.1:chr2:+:24362239:24362320:24358057:24369617 +ENSG00000277363:SRCIN1:chr17:-:36724458:36724482:36720549:36728914 +ENSG00000278535:DHRS11:chr17:+:34954591:34954686:34951610:34955349 diff --git a/IRIS_functions.md b/IRIS_functions.md new file mode 100644 index 0000000..6d4181a --- /dev/null +++ b/IRIS_functions.md @@ -0,0 +1,484 @@ +# IRIS modules + +[back to IRIS quick guide](README.md) + +For questions about input file format, see example folder. + +For test runs with files in example folder, users will need to modify directories of `fin_matrices`, etc. + +## `format` + +When starting from standard output of [rMATS](https://github.com/Xinglab/rmats-turbo), users should use this step to 1) reformat splice junction counts into a PSI (percent-spliced-in) value matrix, and 2) index and 3) move the PSI matrix for IRIS screening (when -d is enabled). +``` +usage: IRIS format [-h] -t {SE,RI,A3SS,A5SS} -n DATA_NAME -s {1,2} + [-c COV_CUTOFF] [-i] [-e] [-d IRIS_DB_PATH] [--novelSS] + [--gtf GTF] + rmats_mat_path_manifest rmats_sample_order + +required arguments: + rmats_mat_path_manifest + txt manifest of path(s) to rMATS output folder(s) + rmats_sample_order TXT file manifest of corresponding rMATS input sample + order file(s). Required input for rMATS + -t {SE,RI,A3SS,A5SS}, --splicing-event-type {SE,RI,A3SS,A5SS} + String of splicing event types based on rMATS + definition (SE,RI,A3SS,A5SS).Used to name output file + -n DATA_NAME, --data-name DATA_NAME + Defines dataset name (disease state, study name, group + name etc.). Used during IRIS screening + -s {1,2}, --sample-name-field {1,2} + Specifies sample name field (1- SJ count file name, 2- + SJ count folder name), for each sample the name should + match their name in "rmats_sample_order" + +optional arguments: + -h, --help show this help message and exit + -c COV_CUTOFF, --cov-cutoff COV_CUTOFF + Average coverage filter for merged matrix (Default is + 10) + -i, --sample-based-filter + Coverage filter by individual sample not by entire + input group. (Default is disabled) + -e, --merge-events-only + Do not perform matrix merge, only merge events list + -d IRIS_DB_PATH, --iris-db-path IRIS_DB_PATH + Path to store the formatted/indexed AS matrix. + Strongly recommend to store the AS matrix to the IRIS + db by setting the path to the directory containing + folders of pre-index AS reference + ("full_path/IRIS_data.vX/db"). Default is current + location. + --novelSS Enable formatting events with splice junctions + containing novelSS. (Different and a subset of rMATS + novelSS definition. Default is False) + --gtf GTF Path to the Genome annotation GTF file. Required input + when novelSS is enabled. +``` + +## `screen` + +This step takes a user-defined screening parameter file ([example/NEPC_test.para](example/NEPC_test.para)), and performs comparisons against reference databases, and returns tumor-associated, tumor-recurrent, and tumor-specific AS events based on user-defined criteria. + +When the -t option is enabled, the screening step translates identified tumor AS events into peptide sequences that can be used in the prediction step. +``` +usage: IRIS screen [-h] -p PARAMETER_FIN + [--splicing-event-type {SE,RI,A3SS,A5SS}] -o OUTDIR [-t] + [-g GTF] [--all-orf] [--ignore-annotation] + [--remove-early-stop] [--min-sample-count MIN_SAMPLE_COUNT] + [--use-existing-test-result] + +required arguments: + -p PARAMETER_FIN, --parameter-fin PARAMETER_FIN + File of 'IRIS screen' parameters + --splicing-event-type {SE,RI,A3SS,A5SS} + String of splicing event types based on rMATS + definition (SE,RI,A3SS,A5SS).Used to name output file. + (Default is SE event) + -o OUTDIR, --outdir OUTDIR + Directory of IRIS screening results + +optional arguments: + -h, --help show this help message and exit + -t, --translating Translates IRIS-screened tumor splice junctions into + peptides + -g GTF, --gtf GTF The Genome annotation GTF file. Required by IRIS + translate option. + --all-orf Perform the 3 ORF translation. ORF known in the + UniProtKB will be labeled as uniprotFrame in the bed + file (Default is to use the known ORF ONLY) + --ignore-annotation Perform 3 ORF translation without annotating known ORF + from the UniProtKB (Default is disabled) + --remove-early-stop Discard the peptide if containing early stop codon + (Default is keep the truncated peptide) + --min-sample-count MIN_SAMPLE_COUNT + The minimum number of non-missing sample in the input + group for an event to be considered for testiing. Once + specified, removed events will be written to "notest" + file. (Default is no minimum) + --use-existing-test-result + Skip testing and use existing testing result (Default + is run full testing steps) +``` +Additionally, `screen_sjc` can be performed as part of the 'tumor-specificity' screen. +``` +usage: IRIS screen_sjc [-h] -p PARAMETER_FIN + --splicing-event-type {SE,RI,A3SS,A5SS} + -e EVENT_LIST_FILE -o OUTDIR + [--use-existing-test-result] + [--tumor-read-cov-cutoff TUMOR_READ_COV_CUTOFF] + [--normal-read-cov-cutoff NORMAL_READ_COV_CUTOFF] +``` +Optionally, `screen_cpm` can be performed to as a higher stringent 'tumor-association' screen or less stringent 'tumor-specificity' by using normalized splice junction counts (in CPM). +``` +usage: IRIS screen_cpm [-h] -p PARAMETER_FIN + --splicing-event-type {SE,RI,A3SS,A5SS} + -e EVENT_LIST_FILE -o OUTDIR + [--use-existing-test-result] +``` + +## `predict` + +This step takes the screening result and performs annotation of extracellular and HLA-binding epitope predictions to discover immunotherapy targets. + +IRIS prediction of HLA-binding epitopes is a massive prediction job that can utilize a compute cluster. The `prediction` step will create scripts to perform subtasks. If properly configured, those subtask scripts can be executed concurrently by snakemake. +``` +usage: IRIS predict [-h] --task-dir TASK_DIR -p PARAMETER_FIN + [-t {SE,RI,A3SS,A5SS}] [--iedb-local IEDB_LOCAL] + [-m MHC_LIST] [--extracellular-only] [--tier3-only] + [--gene-exp-matrix GENE_EXP_MATRIX] [-c DELTAPSI_COLUMN] + [-d DELTAPSI_CUT_OFF] [-e EPITOPE_LEN_LIST] [--all-orf] + [--extracellular-anno-by-junction] + IRIS_screening_result_path + +required arguments: + IRIS_screening_result_path + Directory of IRIS screening results + --task-dir TASK_DIR Directory to write individual task scripts + -p PARAMETER_FIN, --parameter-fin PARAMETER_FIN + File of parameters used in 'IRIS screen' + -t {SE,RI,A3SS,A5SS}, --splicing-event-type {SE,RI,A3SS,A5SS} + String of splicing event types based on rMATS + definition (SE,RI,A3SS,A5SS).Used to name output file. + (Default is SE event) + +optional arguments: + -h, --help show this help message and exit + --iedb-local IEDB_LOCAL + Specify local IEDB location (if installed) + -m MHC_LIST, --mhc-list MHC_LIST + List of HLA/MHC types among samples. HLA type follows + seq2HLA format + --extracellular-only Only predict CAR-T Targets. Will not predict HLA + binding. + --tier3-only To only run predict on events passing all screen + tiers, which is the tier3 output. Will be much faster + when both the tier1 and tier3 were used. + --gene-exp-matrix GENE_EXP_MATRIX + Tab-delimited matrix of gene expression vs. samples + -c DELTAPSI_COLUMN, --deltaPSI-column DELTAPSI_COLUMN + Column of deltaPSI value in matrix, 1-based (Default + is 5th column) + -d DELTAPSI_CUT_OFF, --deltaPSI-cut-off DELTAPSI_CUT_OFF + Defines cutoff of deltaPSI (or other metric) to select + tumor-enriched splice form (Default is 0) + -e EPITOPE_LEN_LIST, --epitope-len-list EPITOPE_LEN_LIST + Epitope length for prediction (Default is 9,10,11) + --all-orf Perform prediction based on 3 ORF translation + peptides. Enable this if translation/screening used + this option (Default is False) + --extracellular-anno-by-junction + By default, CAR-T targets are annotated by association + of event with extracellular domain. This option + annotates target based on a junction (not recommended) +``` + +## `epitope_post` + +``` +usage: IRIS epitope_post [-h] -p PARAMETER_FIN -o OUTDIR + [-t {SE,RI,A3SS,A5SS}] -m MHC_BY_SAMPLE + [-e GENE_EXP_MATRIX] [--tier3-only] [--keep-exist] + [--epitope-len-list EPITOPE_LEN_LIST] + [--no-match-to-canonical-proteome] + [--no-uniqueness-annotation] + [--ic50-cut-off IC50_CUT_OFF] + +required arguments: + -p PARAMETER_FIN, --parameter-fin PARAMETER_FIN + File of parameters used in IRIS screen + -o OUTDIR, --outdir OUTDIR + Directory of IRIS screening results + -t {SE,RI,A3SS,A5SS}, --splicing-event-type {SE,RI,A3SS,A5SS} + String of splicing event types based on rMATS + definition (SE,RI,A3SS,A5SS).Used to name output file + (Default is SE event) + -m MHC_BY_SAMPLE, --mhc-by-sample MHC_BY_SAMPLE + Tab-delimited matrix of HLA/MHC type vs. samples. HLA + type follows seq2HLA format + -e GENE_EXP_MATRIX, --gene-exp-matrix GENE_EXP_MATRIX + Tab-delimited matrix of gene expression vs. samples + +optional arguments: + -h, --help show this help message and exit + --tier3-only Only predict tier3 events. Will be much faster. + --keep-exist Do not rewrite a new postive prediction file when the + file existed. Default is False + --epitope-len-list EPITOPE_LEN_LIST + Epitope length for prediction (Default is 9,10,11) + --no-match-to-canonical-proteome + Matches epitopes to UniProt canonical protein + sequences as an annotation. + --no-uniqueness-annotation + Matches epitopes to all IRIS translated junction + peptides in the same analysis as an annotation. + --ic50-cut-off IC50_CUT_OFF + Specifies IC50 cut-off to define HLA-binding epitopes + (Default is 500) +``` + +## `process_rnaseq` + +When starting from a fastq file, users should use this step to perform RNA-Seq alignment and quantification. This module uses STAR and cufflinks. This module only takes one sample (can be multiple fastq files) for each run. Users are recommended to run this module in parallel (use `makesubsh_mapping` for snakemake). +``` +usage: IRIS process_rnaseq [-h] --starGenomeDir STARGENOMEDIR --gtf GTF -p + SAMPLEID_OUTDIR [--db-length DB_LENGTH] [--mapping] + [--quant] [--sort] + readsFilesRNA + +required arguments: + --starGenomeDir STARGENOMEDIR + The path to the STAR indexed reference genome. Pass to + the "genomeDir" parameter in STAR + --gtf GTF Path to the Genome annotation GTF file + -p SAMPLEID_OUTDIR, --sampleID-outdir SAMPLEID_OUTDIR + Output directory where sample ID will be used as the + output folder name + --db-length DB_LENGTH + Pass to the "sjdbOverhang" parameter in STAR. Default + is 100 + readsFilesRNA Specify the path to the paired-end FASTQ files for the + sample. Files are seperated eperated by ",". + +optional arguments: + -h, --help show this help message and exit + --mapping Only perform reads mapping + --quant Only perform gene expression and AS quantification + --sort Only perform BAM file sorting +``` + +## `makesubsh_mapping` + +Run `process_rnaseq` jobs in parallel on HPC or cloud based on snakemake. +``` +usage: IRIS makesubsh_mapping [-h] [--fastq-folder-dir FASTQ_FOLDER_DIR] + --starGenomeDir STARGENOMEDIR --gtf GTF + --data-name DATA_NAME --outdir OUTDIR + --label-string LABEL_STRING --task-dir TASK_DIR + +required arguments: + --fastq-folder-dir FASTQ_FOLDER_DIR + Specify the path to the higher level of all folders + containing FASTQ files + --starGenomeDir STARGENOMEDIR + The path to the STAR indexed reference genome. Pass to + the "genomeDir" parameter in STAR + --gtf GTF Path to the Genome annotation GTF file + --data-name DATA_NAME + Data set name used to name submission shell scripts + files. + --outdir OUTDIR Output directory for folders of aligned BAM files + --label-string LABEL_STRING + String in the fastq file name between the reads pair + number and "fastq/fq". This is used to recognize + paired-end reads. e.g. For FASTQ_file_L1_R2.fastq.gz, + the label string is the "." between "2" and "fastq". + --task-dir TASK_DIR Directory to write individual task scripts + +optional arguments: + -h, --help show this help message and exit +``` + +## `makesubsh_rmats` + +After running `process_rnaseq`, this step can be used to prepare files to run rMATS-turbo in parallel. +``` +usage: IRIS makesubsh_rmats [-h] --rMATS-path RMATS_PATH --bam-dir BAM_DIR + [--bam-prefix BAM_PREFIX] --gtf GTF --data-name + DATA_NAME --task-dir TASK_DIR [--novelSS] + [--read-length READ_LENGTH] + +required arguments: + --rMATS-path RMATS_PATH + Path to the rMATS-turbo script. + --bam-dir BAM_DIR The path one level higher to folders containing BAM + file generated by "process_rnaseq". + --bam-prefix BAM_PREFIX + BAM file prefix (Default is + "Aligned.sortedByCoord.out") + --gtf GTF Path to the Genome annotation GTF file + --data-name DATA_NAME + Data set name used to name submission shell scripts + --task-dir TASK_DIR Directory to write individual task scripts + +optional arguments: + -h, --help show this help message and exit + --novelSS Enable rMATS novelSS option to include novel splice + site detected from the RNA-seq data (Default is False) + --read-length READ_LENGTH + User defined read length instead of using STAR maaping + log file to define automatically. +``` + +## `makesubsh_rmatspost` + +After running `makesubsh_rmats`, this step can be used to merge files to generate final rMATS-turbo results. +``` +usage: IRIS makesubsh_rmatspost [-h] --rMATS-path RMATS_PATH --bam-dir BAM_DIR + --gtf GTF --data-name DATA_NAME [--novelSS] + --task-dir TASK_DIR + +required arguments: + --rMATS-path RMATS_PATH + Path to the rMATS-turbo scripte + --bam-dir BAM_DIR The path one level higher to folders containing BAM + file generated by "process_rnaseq". + --gtf GTF Path to the Genome annotation GTF file + --data-name DATA_NAME + Data set name used to name submission shell scripts + --task-dir TASK_DIR Directory to write individual task scripts + +optional arguments: + -h, --help show this help message and exit + --novelSS Enable rMATS novelSS option to include novel splice + site detected from the RNA-seq data (Default is False) +``` + +## `exp_matrix` + +After running `process_rnaseq`, if samples of interest are all processed, users can use this script to generate a gene expression matrix, which will be used as annotations in downstream IRIS prediction and/or proteomics reports. +``` +usage: IRIS exp_matrix [-h] [--exp-cutoff EXP_CUTOFF] -o OUTDIR -n DATA_NAME + gene_exp_file_list + +required arguments: + gene_exp_file_list A txt manifest of path(s) of cufflinks gene expression + output(s). + -n DATA_NAME, --data-name DATA_NAME + Name of the dataset (disease state, study name, group + name etc.). + +optional arguments: + -h, --help show this help message and exit + --exp-cutoff EXP_CUTOFF + Gene expression cut-off based on FPKM (Default is 1) + -o OUTDIR, --outdir OUTDIR + Output directory for IRIS exp_matrix +``` + +## `index` + +This step is incorporated by formatting. For users who already have a matrix of AS PSI values (generated by rMATS or another tool), this command could finish the indexing and other steps to prepare for IRIS screening. +``` +usage: IRIS index [-h] -t {SE,RI,A3SS,A5SS} -n DATA_NAME [-c COV_CUTOFF] + [-o OUTDIR] + splicing_matrix + +required arguments: + splicing_matrix Tab-delimited matrix of splicing events (row) vs. + sample IDs (col) + -t {SE,RI,A3SS,A5SS}, --splicing-event-type {SE,RI,A3SS,A5SS} + String of splicing event types based on rMATS + definition (SE,RI,A3SS,A5SS).Used to name output file + -n DATA_NAME, --data-name DATA_NAME + Name of data matrix (disease state, study name, group + name, etc.) being indexed. Used by IRIS during + screening + +optional arguments: + -h, --help show this help message and exit + -c COV_CUTOFF, --cov-cutoff COV_CUTOFF + For the naming purpose, Input average coverage cutoff + used when generating the PSI matrix (Default is 10) + -o OUTDIR, --outdir OUTDIR + Output directory for IRIS database +``` + +## `translate` + +``` +usage: IRIS translate [-h] -g REF_GENOME -t {SE,RI,A3SS,A5SS} --gtf GTF -o + OUTDIR [--all-orf] [--ignore-annotation] + [--remove-early-stop] [-c DELTAPSI_COLUMN] + [-d DELTAPSI_CUT_OFF] [--no-tumor-form-selection] + [--check-novel] + as_input + +required arguments: + as_input Inputs AS event coordinates and delta PSI values + -g REF_GENOME, --ref-genome REF_GENOME + Specifies reference genome (FASTA format) location + -t {SE,RI,A3SS,A5SS}, --splicing-event-type {SE,RI,A3SS,A5SS} + String of splicing event types based on rMATS + definition (SE,RI,A3SS,A5SS).Used to name output file + --gtf GTF Path to the Genome annotation GTF file. Used to define + exon ends for microexons + -o OUTDIR, --outdir OUTDIR + Defines IRIS translation output directory + +optional arguments: + -h, --help show this help message and exit + --all-orf Perform the 3 ORF translation. ORF known in the + UniProtKB will be labeled as uniprotFrame in the bed + file (Default is to use the known ORF ONLY) + --ignore-annotation Perform 3 ORF translation without annotating known ORF + from the UniProtKB (Default is disabled) + --remove-early-stop Discard the peptide if containing early stop codon + (Default is keep the truncated peptide) + -c DELTAPSI_COLUMN, --deltaPSI-column DELTAPSI_COLUMN + Column of deltaPSI value in matrix, 1-based (Default + is 5th column) + -d DELTAPSI_CUT_OFF, --deltaPSI-cut-off DELTAPSI_CUT_OFF + Defines cutoff of deltaPSI (or other metric) used to + select tumor-enriched splice form (Default is 0) + --no-tumor-form-selection + Translates splicing junctions derived from both + skipping and inclusion forms (Default is False) + --check-novel Translates splicing junctions derived from novel + splice sites only using information passed from + screen_novelss (Default is False) +``` + +## `makesubsh_hla` + +This step uses the RNA-Seq FASTQ file to infer the HLA type of a sample. +``` +usage: IRIS makesubsh_hla [-h] [--fastq-folder-dir FASTQ_FOLDER_DIR] + --data-name DATA_NAME -o OUTDIR --label-string + LABEL_STRING --task-dir TASK_DIR + +required arguments: + --fastq-folder-dir FASTQ_FOLDER_DIR + Specify the path to the higher level of all folders + containing FASTQ files + --data-name DATA_NAME + Data set name used to name submission shell scripts. + -o OUTDIR, --outdir OUTDIR + Output directory for folders of seq2hla result + --label-string LABEL_STRING + String in the fastq file name between the reads pair + number and "fastq/fq". This is used to recognize + paired-end reads. e.g. For FASTQ_file_L1_R2.fastq.gz, + the label string is the "." between "2" and "fastq". + --task-dir TASK_DIR Directory to write individual task scripts + +optional arguments: + -h, --help show this help message and exit +``` + +## `pep2epitope` + +This module is a wrapper of prediction tools (IEDB) for predicting peptide-HLA binding. The `prediction` and `epitope_post` modules can generate scripts to run this module in parallel and summarize the result into one TCR target report. +``` +usage: IRIS pep2epitope [-h] [-e EPITOPE_LEN_LIST] [-a HLA_ALLELE_LIST] -o + OUTDIR [--iedb-local IEDB_LOCAL] + [--ic50-cut-off IC50_CUT_OFF] + junction_pep_input + +required arguments: + junction_pep_input Inputs junction peptides + -e EPITOPE_LEN_LIST, --epitope-len-list EPITOPE_LEN_LIST + Epitope length for prediction (Default is 9,10,11) + -a HLA_ALLELE_LIST, --hla-allele-list HLA_ALLELE_LIST + List of HLA types (Default is HLA-A*01:01, + HLA-B*08:01, HLA-C*07:01) + -o OUTDIR, --outdir OUTDIR + Define output directory of pep2epitope + --iedb-local IEDB_LOCAL + Specify local IEDB location (if installed) + --ic50-cut-off IC50_CUT_OFF + Cut-off based on median value of consensus-predicted + IC50 values (Default is 500) + +optional arguments: + -h, --help show this help message and exit +``` diff --git a/IRIS_modules.md b/IRIS_modules.md deleted file mode 100644 index 7f41f7e..0000000 --- a/IRIS_modules.md +++ /dev/null @@ -1,271 +0,0 @@ -[back to IRIS quick guide](README.md) - -For questions about input file format, see example folder. - -For test runs with files in example folder, users will need to modify directories of 'fin_matrices', etc. - -##### formatting -When starting from standard output of [rMATS](http://rnaseq-mats.sourceforge.net), users should use this step to 1) reformat splice junction counts into a PSI (percent-spliced-in) value matrix, and 2) index and 3) move the PSI matrix for IRIS screening (when -d is enabled). -```bash -IRIS formatting -h -usage: IRIS formatting [-h] -t {SE,RI,A3,A5} -n DATA_NAME -s {1,2} - [-c COV_CUTOFF] [-e] [-d IRIS_DB_PATH] - rmats_mat_path_manifest rmats_sample_order - -required arguments: - rmats_mat_path_manifest - txt manifest of path(s) to rMATS output folder(s) - rmats_sample_order txt manifest of corresponding rMATS input sample order file(s) - Required input for rMATS - -t {SE,RI,A3,A5}, --splicing_event_type {SE,RI,A3,A5} - String of splicing event types based on rMATS definition (SE,RI,A3,A5) - Used to name output file - -n DATA_NAME, --data-name DATA_NAME - Defines dataset name (disease state, study name, group name etc.) - Used during IRIS screening - -s {1,2}, --sample-name-field {1,2} - Specifies sample name field for each sample in sample order file(s) - listed by "rmats_sample_order" (1- BAM file name, 2- BAM folder name) - -optional arguments: - -h, --help Shows help message and exits - -c COV_CUTOFF, --cov-cutoff COV_CUTOFF - Average coverage filter for merged matrix (default is 10) - -e, --merge-events-only - Do not perform matrix merge, only merge events list - -d IRIS_DB_PATH, --iris-db-path IRIS_DB_PATH - Path to IRIS database - Formatted/indexed AS matrices are stored here and used during IRIS screening -``` - -##### screening -This step takes a user-defined screening parameter file ([example](example/Test.para)), which performs comparisons against reference databases, and returns tumor-associated, tumor-recurrent, and tumor-specific AS events based on user-defined criteria. - -When the -t option is enabled, the screening step translates identified tumor AS events into peptide sequences that can be used in the prediction step. -```bash -IRIS screening -h -usage: IRIS screening [-h] [-o OUTDIR] [-t] parameter_fin - -required arguments: - parameter_fin File of IRIS screening parameters - -o OUTDIR, --outdir OUTDIR - Directory of IRIS screening results - -optional arguments: - -h, --help Shows help message and exits - -t, --translating Translates IRIS-screened tumor splice junctions into peptides -``` - -##### prediction -This step takes the screening result and performs annotation of extracellular and HLA-binding epitope predictions to discover immunotherapy targets. - -IRIS prediction of HLA-binding epitopes is a massive prediction job that requires access to computing clusters with the SGE system for completion. The 'prediction' step will create qsub scripts for job array submission. - -###### Perform extracellular (CAR-T) target annotation & prepare epitope (TCR) target prediction - ``` -IRIS prediction -h -usage: IRIS prediction [-h] [-p PARAMETER_FIN] [--iedb-local IEDB_LOCAL] - [-c DELTAPSI_COLUMN] [-d DELTAPSI_CUT_OFF] -m MHC_LIST - [--extracellular-anno-by-junction] - IRIS_screening_result_path - -required arguments: - IRIS_screening_result_path - Input AS event coordinates and PSI values - -p PARAMETER_FIN, --parameter-fin PARAMETER_FIN - File of parameters used in IRIS screening - --iedb-local IEDB_LOCAL - Specify local IEDB location (if installed) - -m MHC_LIST, --mhc-list MHC_LIST - List of HLA/MHC types among samples - HLA type follows seq2HLA format - -optional arguments: - -h, --help Shows help message and exits - -c DELTAPSI_COLUMN, --deltaPSI-column DELTAPSI_COLUMN - Column of deltaPSI value in matrix, 1-based (default is 5th column) - -d DELTAPSI_CUT_OFF, --deltaPSI-cut-off DELTAPSI_CUT_OFF - Defines cutoff of deltaPSI (or other metric) to select tumor-enriched - splice form (default is 0) - --extracellular-anno-by-junction - By default, CAR-T targets are annotated by association of event - with extracellular domain - This option annotates target based on a junction (not recommended) - ``` - -###### Epitope (TCR) target prediction (requires SGE system) - ``` -IRIS epitope_post -h -usage: IRIS epitope_post [-h] -p PARAMETER_FIN -o OUTDIR -m MHC_BY_SAMPLE - [-e GENE_EXP_MATRIX] [--ic50-cut-off IC50_CUT_OFF] - -required arguments: - -p PARAMETER_FIN, --parameter_fin PARAMETER_FIN - File of parameters used in IRIS screening - -o OUTDIR, --outdir OUTDIR - Directory of IRIS screening results - -m MHC_BY_SAMPLE, --mhc-by-sample MHC_BY_SAMPLE - Tab-delimited matrix of HLA/MHC type vs. samples - HLA type follows seq2HLA format - -e GENE_EXP_MATRIX, --gene-exp-matrix GENE_EXP_MATRIX - Tab-delimited matrix of gene expression vs. samples - -optional arguments: - -h, --help Shows help message and exits - --ic50-cut-off IC50_CUT_OFF - Specifies IC50 cut-off to define HLA-binding epitopes (default is 500) - ``` - -##### process_rnaseq -When starting from a FASTQ file, users should use this step to perform RNA-Seq alignment and quantification. This module uses STAR and cufflinks. This module only takes one sample (can be multiple FASTQ files) for each run. Users are recommended to run this module in parallel in the SGE system. -``` -IRIS process_rnaseq -h -usage: IRIS process_rnaseq [-h] --starGenomeDir STARGENOMEDIR --gtf GTF -p - SAMPLEID_OUTDIR [--db-length DB_LENGTH] [--mapping] - [--quant] [--sort] - readsFilesRNA - -required arguments: - --starGenomeDir STARGENOMEDIR - Path to STAR-indexed reference genome - Passes to "genomeDir" parameter in STAR - --gtf GTF Genome annotation file. - -p SAMPLEID_OUTDIR, --sampleID-outdir SAMPLEID_OUTDIR - Output directory, where sample ID will be used as output folder name - --db-length DB_LENGTH - Passes to "sjdbOverhang" parameter in STAR (default is 100) - readsFilesRNA Specifies path to paired-end FASTQ files for sample - Files separated by "," - -optional arguments: - -h, --help Shows help message and exits - --mapping Only perform reads mapping - --quant Only perform gene expression and AS quantification - --sort Only perform BAM file sorting - ``` - -##### makeqsub_rmats (requires SGE system) -After running 'process_rnaseq', this step can be used to prepare files to run rMATS-turbo in parallel in the SGE system. -``` -IRIS makeqsub_rmats -h -usage: IRIS makeqsub_rmats [-h] --rMATS-path RMATS_PATH --bam-dir BAM_DIR - --gtf GTF --read-length READ_LENGTH - -required arguments: - --rMATS-path RMATS_PATH - Path to rMATS-turbo script - --bam-dir BAM_DIR Path one level higher to folders containing BAM file generated by 'process_rnaseq' - --gtf GTF Genome annotation file - --read-length READ_LENGTH - Passes to "readLength" parameter in rMATS-turbo - -optional arguments: - -h, --help Shows help message and exits - ``` - -##### exp_matrix -After running 'process_rnaseq', if samples of interest are all processed, users can use this script to generate a gene expression matrix, which will be used as annotations in downstream IRIS prediction and/or proteomics reports. -``` -IRIS exp_matrix -h -usage: IRIS exp_matrix [-h] [--exp-cutoff EXP_CUTOFF] -o OUTDIR -n DATA_NAME - gene_exp_file_list - -required arguments: - gene_exp_file_list txt manifest of path(s) of cufflinks gene expression output(s) - -n DATA_NAME, --data-name DATA_NAME - Name of dataset (disease state, study name, group name, etc.) - -optional arguments: - -h, --help Shows help message and exits - --exp-cutoff EXP_CUTOFF - Gene expression cut-off based on FPKM (default is 1) - -o OUTDIR, --outdir OUTDIR - Output directory for IRIS exp_matrix -``` - -##### indexing -This step is incorporated by formatting. For users who already have a matrix of AS PSI values (generated by rMATS or another tool), this command could finish the indexing and other steps to prepare for IRIS screening. -```bash -IRIS indexing -h -usage: IRIS indexing [-h] -n DATA_NAME [-d DB_DIR] splicing_matrix - -required arguments: - splicing_matrix Tab-delimited matrix of splicing events (row) vs. sample IDs (col) - -n DATA_NAME, --data-name DATA_NAME - Name of data matrix (disease state, study name, group name, etc.) being - formatted & indexed - Used by IRIS during screening - -optional arguments: - -h, --help Shows help message and exits - -d DB_DIR, --db-dir DB_DIR - Directory of IRIS database - Program creates a folder in this directory for IRIS to recognize -``` - -##### translation -```bash -IRIS translation -h -usage: IRIS translation [-h] -g REF_GENOME -o OUTDIR [-c DELTAPSI_COLUMN] - [-d DELTAPSI_CUT_OFF] [--no-tumor-form-selection] - as_input - -required arguments: - as_input Inputs AS event coordinates and PSI values - -g REF_GENOME, --ref-genome REF_GENOME - Specifies reference genome (FASTA format) location - -o OUTDIR, --outdir OUTDIR - Defines IRIS translation output directory - -optional arguments: - -h, --help Show help message and exits - -c DELTAPSI_COLUMN, --deltaPSI-column DELTAPSI_COLUMN - Column of deltaPSI value in matrix, 1-based (default is 5th column) - -d DELTAPSI_CUT_OFF, --deltaPSI-cut-off DELTAPSI_CUT_OFF - Defines cutoff of deltaPSI (or other metric) used to select tumor-enriched - splice form (default is 0) - --no-tumor-form-selection - Translates splicing junctions derived from both skipping and inclusion forms - ``` - -##### seq2hla -This step uses the RNA-Seq FASTQ file to infer the HLA type of a sample. -``` -IRIS seq2hla -h -usage: IRIS seq2hla [-h] -b SEQ2HLA_PATH -p SAMPLEID_OUTDIR readsFilesCaseRNA - -required arguments: - -b SEQ2HLA_PATH, --seq2hla-path SEQ2HLA_PATH - Path to seq2hla folder - -p SAMPLEID_OUTDIR, --sampleID-outdir SAMPLEID_OUTDIR - Output directory, where sample ID will be used as output folder name - readsFilesCaseRNA Tumor sample paired-end fastq files, separated by "," - -optional arguments: - -h, --help Shows help message and exits - ``` - -##### pep2epitope -This module is a wrapper of prediction tools (IEDB) for predicting peptide-HLA binding. The 'prediction' and 'epitope_post' modules can make qsub submissions to run this module in parallel and summarize the result into one TCR target report. -```IRIS pep2epitope -h -usage: IRIS pep2epitope [-h] [-e EPITOPE_LEN_LIST] [-a HLA_ALLELE_LIST] -o - OUTDIR [--iedb-local IEDB_LOCAL] - [--ic50-cut-off IC50_CUT_OFF] - junction_pep_input - -required arguments: - junction_pep_input Inputs AS event coordinates and PSI values - -e EPITOPE_LEN_LIST, --epitope-len-list EPITOPE_LEN_LIST - Epitope length for prediction (default is 9,10,11) - -a HLA_ALLELE_LIST, --hla-allele-list HLA_ALLELE_LIST - List of HLA types (default is HLA-A*01:01, HLA-B*08:01, HLA-C*07:01) - -o OUTDIR, --outdir OUTDIR - Define output directory of pep2epitope - --iedb-local IEDB_LOCAL - Specify local IEDB location (if installed) - --ic50-cut-off IC50_CUT_OFF - Cut-off based on median value of consensus-predicted IC50 values (default is 500) - ``` - -optional arguments: - -h, --help Shows help message and exits diff --git a/README.md b/README.md index e01c3ef..89d00a4 100644 --- a/README.md +++ b/README.md @@ -1,94 +1,201 @@ # IRIS: Isoform peptides from RNA splicing for Immunotherapy target Screening +## Quick guide -### Quick guide -- [Dependencies](#dependencies) -- [Installation](#installation) -- [Usage](#usage) - - [Usage - individual modules (for customized pipelines)](#individual-modules) - - [Usage - streamlined major modules (for common use)](#streamlined-major-modules) -- [Example](#example) -- [Output](#example-output) -- [Contact](#contact) -- [Publication](#publication) +* [Dependencies](#dependencies) +* [Installation](#installation) +* [Usage](#usage) + + [Usage - individual functions (for customized pipelines)](#individual-functions) + + [Usage - streamlined major functions (for common use)](#streamlined-major-functions) + + [Snakemake](#snakemake) +* [Example](#example) +* [Output](#example-output) +* [Contact](#contact) +* [Publication](#publication) +## Dependencies +### Core dependencies (required for major IRIS functions/steps - format, screen, and predict) -### Dependencies +* python 2.7.x (numpy, scipy, seaborn, pyBigWig, statsmodels, pysam) +* [IEDB stand-alone 20130222 2.15.5](http://tools.iedb.org/main/download/) + + IEDB additionally depends on: + - [tcsh](http://www.tcsh.org) + - [gawk](http://www.gnu.org/software/gawk/) +* [bedtools 2.29.0](https://bedtools.readthedocs.io/en/latest/) -#### Core dependencies (required for major IRIS modules/steps - formatting, screening, and prediction): -- python 2.7.x (numpy, scipy, seaborn, pyBigWig, etc.) -- [IEDB stand-alone 20130222 2.15.5 (2.22.1 is not fully tested)](http://tools.iedb.org/main/download/) -- [bedtools 2.29.0](https://bedtools.readthedocs.io/en/latest/) +### Other dependencies (required for processing raw RNA-Seq and MS data) -#### Other dependencies (required for processing raw RNA-Seq and MS data) -- [STAR 2.5.3](https://github.com/alexdobin/STAR/releases/tag/2.5.3a): required for IRIS RNA-seq processing -- [samtools 1.3](https://sourceforge.net/projects/samtools/files/samtools/): required for IRIS RNA-seq processing -- [rMATS-turbo](http://rnaseq-mats.sourceforge.net): required for IRIS RNA-seq processing -- [Cufflinks 2.2.1](http://cole-trapnell-lab.github.io/cufflinks/install/): required for IRIS RNA-seq processing -- [seq2HLA](https://bitbucket.org/sebastian_boegel/seq2hla/src/default/): required for HLA typing; requires [bowtie](http://bowtie-bio.sourceforge.net/index.shtml) -- [MS GF+ (v2018.07.17)](https://github.com/MSGFPlus/msgfplus): required for MS search; requiring [Java](https://www.java.com/en/download/) +* [STAR 2.5.3](https://github.com/alexdobin/STAR/releases/tag/2.5.3a): required for IRIS RNA-seq processing +* [samtools 1.3](https://sourceforge.net/projects/samtools/files/samtools/): required for IRIS RNA-seq processing +* [rMATS-turbo](https://github.com/Xinglab/rmats-turbo): required for IRIS RNA-seq processing +* [Cufflinks 2.2.1](http://cole-trapnell-lab.github.io/cufflinks/install/): required for IRIS RNA-seq processing +* [seq2HLA](https://bitbucket.org/sebastian_boegel/seq2hla/src/default/): required for HLA typing; requires [bowtie](http://bowtie-bio.sourceforge.net/index.shtml) +* [MS GF+ (v2018.07.17)](https://github.com/MSGFPlus/msgfplus): required for MS search; requiring [Java](https://www.java.com/en/download/) +* [R](https://www.r-project.org/): used by seq2HLA +## Installation +### 1. Download + +#### 1.1 Download IRIS program -### Installation -Two steps to set up IRIS: -#### 1. Download -##### 1.1 Download IRIS program The IRIS program can be downloaded directly from the repository, as shown below: ``` git clone https://github.com/Xinglab/IRIS.git cd IRIS ``` -__For full functionality, IRIS requires use of the SGE system. For users who want to use functions involving SGE (see [Usage](#usage) for details), please check IRIS/config.py to ensure qsub parameters are correct before moving to the next step.__ -##### 1.2 Download IRIS db -IRIS loads a big-data reference database of splicing events and other genomic annotations. \ -These data are included in [IRIS_data.tgz](https://drive.google.com/file/d/1TaswpWPnEd4TXst46jsa9XSMzLsbzjOQ/view?usp=sharing) (a Google Drive link; size ~10 GB). Users need to move this file to the IRIS folder for streamlined installation. -##### 1.3 Download IEDB MHC I prediction tools -Download IEDB_MHC_I-X.XX.X.tar.gz from IEDB website (see [Dependencies](#dependencies)). Create a folder named 'IEDB' in the IRIS folder, then move the downloaded gz file to the 'IEDB' folder. -#### 2. Install and configure -Under the IRIS folder, do: +__IRIS is designed to make use of a computing cluster to improve performance. For users who want to enable cluster execution for functions that support it (see [Configure](#3-configure-for-compute-cluster) for details), please update the contents of [snakemake_profile/](snakemake_profile/) to ensure compatibility with the available compute environment.__ + +#### 1.2 Download IRIS db + +IRIS loads a big-data reference database of splicing events and other genomic annotations. These data are included in [IRIS_data.v2.0.0](https://drive.google.com/drive/folders/1zhmXoajD5RyjxVTYbGZ-ebic1VPfEYKz?usp=sharing) (a Google Drive link; size of entire folder is ~400 GB; users can select reference groups to download). The files need to be placed under `./IRIS_data/` + +The files can be automatically downloaded with [google_drive_download.py](google_drive_download.py). Downloading a large amount of data with the API requires authentication: +* https://cloud.google.com/docs/authentication/production +* https://cloud.google.com/bigquery/docs/authentication/service-account-file + +To use the script, first create a service account: +* Go to google cloud console -> IAM & Admin -> Service Accounts -> create service account +* Give the new account: role=owner +* Click the new service account email on the service account page +* Download a .json key by clicking: keys -> add key -> create new key -> json + +That .json key is passed to [google_drive_download.py](google_drive_download.py) + +#### 1.3 Download IEDB MHC I prediction tools + +Download `IEDB_MHC_I-2.15.5.tar.gz` from the IEDB website (see [Dependencies](#dependencies)). Create a folder named `IEDB/` in the IRIS folder, then move the downloaded gz file to `IEDB/`. From http://tools.iedb.org/main/download/ +* click "MHC Class I" +* click "previous version" +* find and download version 2.15.5 + +The manual download is needed because there is a license that must be accepted. + +### 2. Install + +[./install](./install) can automatically install most dependencies to conda environments: +* conda must already be installed for the script to work + + https://docs.conda.io/en/latest/miniconda.html +* The install script will check if `IRIS_data/` has been downloaded + + To download see [1.2 Download IRIS db](#12-download-iris-db) +* The install script will check if IEDB tools has been downloaded + + To download see [1.3 Download IEDB MHC I prediction tools](#13-download-iedb-mhc-i-prediction-tools) + +Under the IRIS folder, to install IRIS [core dependencies](#core-dependencies-required-for-major-iris-functionssteps---format-screen-and-predict), do: ``` ./install core ``` -Follow instructions to finish the installation of conda, python and its dependencies, bedtools, the downloaded IEDB package, and the IRIS data and packages. To install optional dependencies not needed for the most common IRIS usage: + +To install [optional dependencies](#other-dependencies-required-for-processing-raw-rna-seq-and-ms-data) not needed for the most common IRIS usage: ``` ./install all ``` -### Usage -- For streamlined AS-derived target discovery, please follow [major modules](#streamlined-major-modules) and run the corresponding toy example. -- For customized pipeline development, please check [all modules](#individual-modules) of IRIS. +### 3. Configure for compute cluster -#### Individual modules -IRIS provides individual modules/steps, allowing users to build pipelines for their customized needs.\ -For a description of each [module/step](IRIS_modules.md), including RNA-seq preprocessing, HLA typing, proteo-transcriptomic MS searching, visualization, etc., please click [here](IRIS_modules.md) or the subheader above. -``` -usage: IRIS [-h] [--version] - {formatting,screening,prediction,epitope_post,process_rnaseq,makeqsub_rmats,exp_matrix,indexing,translation,pep2epitope,screening_plot,seq2hla,parse_hla,ms_makedb,ms_search,ms_parse} - ... +[Snakefile](Snakefile) describes the IRIS pipeline. The configuration for running jobs can be set by editing [snakemake_profile/](snakemake_profile/). The provided configuration adapts IRIS to use Slurm. Other compute environments can be supported by updating this directory +* [snakemake_profile/config.yaml](snakemake_profile/config.yaml): Sets various Snakemake parameters including whether to submit jobs to a cluster. +* [snakemake_profile/cluster_submit.py](snakemake_profile/cluster_submit.py): Script to submit jobs. +* [snakemake_profile/cluster_status.py](snakemake_profile/cluster_status.py): Script to check job status. +* [snakemake_profile/cluster_commands.py](snakemake_profile/cluster_commands.py): Commands specific to the cluster management system being used. The default implementation is for Slurm. Other cluster environments can be used by changing this file. For example, [snakemake_profile/cluster_commands_sge.py](snakemake_profile/cluster_commands_sge.py) can be used to overwrite `cluster_commands.py` to support an SGE cluster. +* To force Snakemake to execute on the local machine modify [snakemake_profile/config.yaml](snakemake_profile/config.yaml): + + comment out `cluster` + + set `jobs: {local cores to use}` + + uncomment the `resources` section and set `mem_mb: {MB of RAM to use}` + +### 4. Known issues + +* The conda install of Python 2 may give an error like `ImportError: No module named _sysconfigdata_x86_64_conda_linux_gnu` + + Check for the error by activating `conda_env_2` and running `python` + + Resolve with commands similar to + - `cd conda_env_2/lib/python2.7/` + - `cp _sysconfigdata_x86_64_conda_cos6_linux_gnu.py _sysconfigdata_x86_64_conda_linux_gnu.py` +* The installed version of R may depend on old version of libreadline that is not available in conda + + Check for the error by activating `conda_env_2` and running `R` + + Resolve by activating `conda_env_2` and manually following the steps in the `install_readline()` function of [./install](./install) +* IRIS uses `--label-string` to determine which fastq files are for read 1 and read 2 + + To avoid any issues name your fastq files so that they end with `1.fastq` and `2.fastq` to indicate which file represents which pair of the read + +## Usage + +* For streamlined AS-derived target discovery, please follow [major functions](#streamlined-major-functions) and run the corresponding toy example. +* For customized pipeline development, please check [all functions](#individual-functions) of IRIS. + +This flowchart shows how the IRIS functions are organized +![iris_diagram](docs/iris_diagram.png) + +### Individual functions + +IRIS provides individual functions/steps, allowing users to build pipelines for their customized needs. [IRIS_functions.md](IRIS_functions.md) describes each model/step, including RNA-seq preprocessing, HLA typing, proteo-transcriptomic MS searching, visualization, etc. +``` IRIS -- IRIS positional arguments: - {formatting,screening,prediction,epitope_post,process_rnaseq,makeqsub_rmats,exp_matrix,indexing,translation,pep2epitope,screening_plot,seq2hla,parse_hla,ms_makedb,ms_search,ms_parse} - formatting Formats AS matrices from rMATS, followed by indexing for IRIS - screening Screens AS-derived tumor antigens using big-data reference - prediction Predicts and annotates AS-derived TCR (pre-prediction) and CAR-T targets - epitope_post Post-prediction step to summarize predicted TCR targets - process_rnaseq Processes RNA-Seq FASTQ files to quantify gene expression and AS - makeqsub_rmats Makes qsub files for running rMATS-turbo 'prep' step - exp_matrix Makes a merged gene expression matrix from multiple cufflinks results - indexing Indexes AS matrices for IRIS - translation Translates AS junctions into junction peptides + format Formats AS matrices from rMATS, followed by indexing + for IRIS + screen Screens AS-derived tumor antigens using big-data + reference + predict Predicts and annotates AS-derived TCR (pre-prediction) + and CAR-T targets + epitope_post Post-prediction step to summarize predicted TCR + targets + process_rnaseq Processes RNA-Seq FASTQ files to quantify gene + expression and AS + makesubsh_mapping Makes submission shell scripts for running + 'process_rnaseq' + makesubsh_rmats Makes submission shell scripts for running rMATS-turbo + 'prep' step + makesubsh_rmatspost + Makes submission shell scripts for running rMATS-turbo + 'post' step + exp_matrix Makes a merged gene expression matrix from multiple + cufflinks results + makesubsh_extract_sjc + Makes submission shell scripts for running + 'extract_sjc' + extract_sjc Extracts SJ counts from STAR-aligned BAM file and + annotates SJs with number of uniquely mapped reads + that support the splice junction. + sjc_matrix Makes SJ count matrix by merging SJ count files from a + specified list of samples. Performs indexing of the + merged file. + index Indexes AS matrices for IRIS + translate Translates AS junctions into junction peptides pep2epitope Wrapper to run IEDB for peptide-HLA binding prediction - screening_plot Makes stacked/individual violin plots for list of AS events - seq2hla Wrapper to run seq2HLA for HLA typing using RNA-Seq - parse_hla Summarizes seq2HLA results of all input samples into matrices for IRIS use + screen_plot Makes stacked/individual violin plots for list of AS + events + screen_sjc Screens AS-derived tumor antigens by comparing number + of samples expressing a splice junction using big-data + reference of SJ counts + append_sjc Appends SJC result as an annotation to PSI-based + screening results and epitope prediction results in a + specified screening output folder. + annotate_ijc Annotates inclusion junction count info to PSI-based + screening results or epitope prediction results in a + specified screening output folder. Can be called from + append sjc to save time. + screen_cpm Screens AS-derived tumor antigens by comparing splice + junction CPM using big-data reference of SJ counts + append_cpm Appends CPM result as an annotation to PSI-based + screening results and epitope prediction results in a + specified screening output folder. + screen_novelss Screens AS-derived tumor antigens for unannotated + events using big-data reference of SJ counts + screen_sjc_plot Makes stacked/individual barplots of percentage of + samples expressing a splice junction for list of AS + events + makesubsh_hla Makes submission shell scripts for running seq2HLA for + HLA typing using RNA-Seq + parse_hla Summarizes seq2HLA results of all input samples into + matrices for IRIS use ms_makedb Generates proteo-transcriptomic database for MS search ms_search Wrapper to run MSGF+ for MS search - ms_parse Parses MS search results to generate tables of identified peptides + ms_parse Parses MS search results to generate tables of + identified peptides + visual_summary Makes a graphic summary of IRIS results optional arguments: -h, --help show this help message and exit @@ -97,98 +204,218 @@ optional arguments: For command line options of each sub-command, type: IRIS COMMAND -h ``` +### Streamlined major functions + +The common use of IRIS immunotherapy target discovery comprises three major steps. For a quick test, see [Example](#example) which uses the snakemake to run a small data set. +* Step 1. generate PSI-based AS matrix from rMATS output (& index) + + IRIS format option -d should be used to save the generated PSI-based AS matrix to the downloaded IRIS db. + + Example files for `rmats_mat_path_manifest` and `rmats_sample_order` can be found under the 'example' folder for the test run. + + IRIS index will create an index for the IRIS format generated PSI-based AS matrix, and -o should be the path to the folder containing the AS matrix. +``` +usage: IRIS format [-h] -t {SE,RI,A3SS,A5SS} -n DATA_NAME -s {1,2} + [-c COV_CUTOFF] [-i] [-e] [-d IRIS_DB_PATH] [--novelSS] + [--gtf GTF] + rmats_mat_path_manifest rmats_sample_order + +usage: IRIS index [-h] -t {SE,RI,A3SS,A5SS} -n DATA_NAME + [-c COV_CUTOFF] [-o OUTDIR] + splicing_matrix +``` -#### Streamlined major modules -The common use of IRIS immunotherapy target discovery comprises three major steps. For a quick test, see [Example](#example), in which a shell script is provided for a streamlined example run: -- Step 1. IRIS formatting (& indexing) +* Step 2. IRIS screen (& translation) ('tumor-association' screen) + + [example/parameter_file_description.txt](example/parameter_file_description.txt) describes `PARAMETER_FIN` and [example/Test.para](example/Test.para) is an example. + + Option -t is required for TCR target prediction ``` -usage: IRIS formatting [-h] -t {SE,RI,A3,A5} -n DATA_NAME -s {1,2} - [-c COV_CUTOFF] [-e] [-d IRIS_DB_PATH] - rmats_mat_path_manifest rmats_sample_order +usage: IRIS screen [-h] -p PARAMETER_FIN + --splicing-event-type {SE,RI,A3SS,A5SS} -o OUTDIR [-t] + [-g GTF] [--all-orf] [--ignore-annotation] + [--remove-early-stop] [--min-sample-count MIN_SAMPLE_COUNT] + [--use-existing-test-result] ``` -- Step 2. IRIS screening (& translation) -Here is a [description of the parameter file](example/parameter_file_description.txt) and an [example file](example/Test.para). +* Step 3. IRIS predict (predicts both extracellular targets and epitopes; __designed for cluster execution__) + + IRIS predict can generate CAR-T annotation results and prepare a job array submission for TCR epitope prediction. TCR prediction preparation is optional and can be disabled by using --extraceullular-only. + + `IRIS epitope_post` will summarize TCR epitope prediction results after TCR epitope prediction jobs from IRIS predict are submitted and finished (job array submission step can be done manually or using snakemake) + + `MHC_LIST` and `MHC_BY_SAMPLE` can be generated by running `HLA_typing` (within or outside IRIS). Note that it is not necessary to restrict HLA types detected from input RNA samples. It is recommended for users to specify dummy files only containing HLA types of interest or common HLA types as long as HLA types in the dummy `hla_types.list` and `hla_patient.tsv` are consistent. Example files for `hla_types.list` and `hla_patient.tsv` can be found under [example/HLA_types/](example/HLA_types/). ``` -usage: IRIS screening [-h] [-o OUTDIR] [-t] parameter_fin +usage: IRIS predict [-h] --task-dir TASK_DIR -p PARAMETER_FIN + -t {SE,RI,A3SS,A5SS} [--iedb-local IEDB_LOCAL] + [-m MHC_LIST] [--extracellular-only] [--tier3-only] + [--gene-exp-matrix GENE_EXP_MATRIX] [-c DELTAPSI_COLUMN] + [-d DELTAPSI_CUT_OFF] [-e EPITOPE_LEN_LIST] [--all-orf] + [--extracellular-anno-by-junction] + IRIS_screening_result_path + +usage: IRIS epitope_post [-h] -p PARAMETER_FIN -o OUTDIR + -t {SE,RI,A3SS,A5SS} -m MHC_BY_SAMPLE + -e GENE_EXP_MATRIX [--tier3-only] [--keep-exist] + [--epitope-len-list EPITOPE_LEN_LIST] + [--no-match-to-canonical-proteome] + [--no-uniqueness-annotation] + [--ic50-cut-off IC50_CUT_OFF] ``` -- Step 3. IRIS prediction (predicts both extracellular targets and epitopes; __requires SGE system__) +* Step 4. IRIS screen of the presence-absence of splice junctions (required for the 'tumor-specificity' screen) + + IRIS append_sjc combines `screen` and `screen_sjc` results (by appending `screen_sjc` outputs to `screen` outputs). The 'integrated' output contains annotations for tumor-specific targets. + + IRIS append_sjc -i option can be used to execute both IRIS append_sjc and IRIS annotate_ijc functions. If -i option is used, -p and -e arguments are required. ``` -usage: IRIS prediction [-h] [-p PARAMETER_FIN] [--iedb-local IEDB_LOCAL] - [-c DELTAPSI_COLUMN] [-d DELTAPSI_CUT_OFF] -m MHC_LIST - [--extracellular-anno-by-junction] - IRIS_screening_result_path - -usage: IRIS epitope_post [-h] -p PARAMETER_FIN -o OUTDIR -m MHC_BY_SAMPLE - [-e GENE_EXP_MATRIX] [--ic50-cut-off IC50_CUT_OFF] +usage: IRIS screen_sjc [-h] -p PARAMETER_FIN + --splicing-event-type {SE,RI,A3SS,A5SS} + -e EVENT_LIST_FILE -o OUTDIR + [--use-existing-test-result] + [--tumor-read-cov-cutoff TUMOR_READ_COV_CUTOFF] + [--normal-read-cov-cutoff NORMAL_READ_COV_CUTOFF] + +usage: IRIS append_sjc [-h] --sjc-summary SJC_SUMMARY + --splicing-event-type {SE,RI,A3SS,A5SS} -o OUTDIR + [-i] [-u] [-p PARAMETER_FILE] + [-e SCREENING_RESULT_EVENT_LIST] + [--inc-read-cov-cutoff INC_READ_COV_CUTOFF] + [--event-read-cov-cutoff EVENT_READ_COV_CUTOFF] + +usage: IRIS annotate_ijc [-h] -p PARAMETER_FILE + --splicing-event-type {SE,RI,A3SS,A5SS} + -e SCREENING_RESULT_EVENT_LIST -o OUTDIR + [--inc-read-cov-cutoff INC_READ_COV_CUTOFF] + [--event-read-cov-cutoff EVENT_READ_COV_CUTOFF] + ``` -### Example -We provide a wrapper ([run_example](run_example)) to run the above [IRIS streamlined major modules](#streamlined-major-modules) using [example files](example), included in the IRIS package. For customized pipeline development, we recommend that users use this script and [run_iris](run_iris) as a reference. Under the IRIS folder, do: +### Snakemake + +The Snakemake workflow can be run with [./run](./run). First set the configuration values in [snakemake_config.yaml](snakemake_config.yaml) +* Set the resources to allocate for each job: + + `{job_name}_{threads}` + + `{job_name}_{mem_gb}` + + `{job_name}_{time_hr}` +* Set the reference files + + Provide the file names as `gtf_name:` and `fasta_name:` + + Either place the files in `./references/` + + Or provide a url under `reference_files:` to download the (potentially gzipped) files: +``` +gtf_name: 'some_filename.gtf' +fasta_name: other_filename.fasta' +reference_files: + some_filename.gtf.gz: + url: 'protocol://url/for/some_filename.gtf.gz' + other_filename.fasta.gz: + url: 'protocol://url/for/other_filename.fasta.gz' +``` +* Set the input files + + `sample_fastqs:` Set the read 1 and read 2 fastq files for each sample. For example: ``` -./run_example +sample_fastqs: + sample_name_1: + - '/path/to/sample_1_read_1.fq' + - '/path/to/sample_1_read_2.fq' + sample_name_2: + - '/path/to/sample_2_read_1.fq' + - '/path/to/sample_2_read_2.fq' ``` -__As mentioned in [Usage](#usage), this example run will involve submitting the job array to the SGE system.__ It will take < 5 min for the formatting and screening steps and usually < 15 min for the prediction step (SGE job arrays).\ -A successful test run will generate the following result files in ./results/example/Glioma_test/screening (row numbers are displayed before each file name): + + `blacklist`: an optional black list of AS events similar to [IRIS/data/blacklist.brain_2020.txt](IRIS/data/blacklist.brain_2020.txt) + + `mapability_bigwig`: an optional file for evaluating splice region mappability similar to `IRIS_data/resources/mappability/wgEncodeCrgMapabilityAlign24mer.bigWig` + + `mhc_list`: required if not starting with fastq files. similar to [example/HLA_types/hla_types.list](example/HLA_types/hla_types.list) + + `mhc_by_sample`: required if not starting with fastq files. similar to [example/HLA_types/hla_patient.tsv](example/HLA_types/hla_patient.tsv) + + `gene_exp_matrix`: optional tsv file with geneName as the first column and the expression for each sample in the remaining columns + + `splice_matrix_txt`: optional output file from IRIS index that can be used as a starting point + + `splice_matrix_idx`: the index file for `splice_matrix_txt` + + `sjc_count_txt`: optional output file from IRIS sjc_matrix that can be used as a starting point. Only relevant if `should_run_sjc_steps` + + `sjc_count_idx`: the index file for `sjc_count_txt` +* Set other options + + `run_core_modules`: set to `true` to start with existing `IRIS format` output and HLA lists + + `run_all_modules`: set to `true` to start with fastq files + + `should_run_sjc_steps`: set to `true` to enable splice junction based evaluation steps + + `star_sjdb_overhang`: used by STAR alignment. Ideally it should be `read_length -1`, but the STAR manual says that 100 should work well as a default + + `run_name`: used to name output files that will be written to `IRIS_data/` + + `splice_event_type`: one of `[SE, RI,A3SS, A5SS]` + + `comparison_mode`:one of `[group, individual]` + + `stat_test_type`: one of `[parametric, nonparametric]` + + `use_ratio`: set to `true` to require a ratio of reference groups to pass the checks rather than a fixed count + + `tissue_matched_normal_..._{cutoff}`: set the cutoffs for the tissue matched normal reference group (tier 1) + + `tissue_matched_normal_reference_group_names`: a comma separate list of directory names under `IRIS_data/db` + + `tumor_..._{cutoff}`: set the cutoffs for the tumor reference group (tier 2) + + `tumor_reference_group_names`: a comma separate list of directory names under `IRIS_data/db` + + `normal_..._{cutoff}`: set the cutoffs for the normal reference group (tier 3) + + `normal_reference_group_names`: a comma separate list of directory names under `IRIS_data/db` + +## Example + +The snakemake is configured to run the above [IRIS streamlined major functions](#streamlined-major-functions) using [example/](example/). For customized pipeline development, we recommend that users refer to the [Snakefile](Snakefile) as a reference. [Snakefile](Snakefile) defines the steps of the pipeline. Update the `/path/to/` values with full paths in [snakemake_config.yaml](snakemake_config.yaml) and make any adjustments to [snakemake_profile/](snakemake_profile/). Then ``` - 0 _example_Glioma_test.notest.txt - 13 _example_Glioma_test.primary.txt - 3 _example_Glioma_test.primary.txt.ExtraCellularAS.txt - 11 _example_Glioma_test.prioritized.txt - 3 _example_Glioma_test.prioritized.txt.ExtraCellularAS.txt - 13 _example_Glioma_test.test.all.txt - 13 primary/epitope_summary.junction-based.txt - 74 primary/epitope_summary.peptide-based.txt - 148 primary/pred_filtered.score500.txt - 11 prioritized/epitope_summary.junction-based.txt - 45 prioritized/epitope_summary.peptide-based.txt - 84 prioritized/pred_filtered.score500.txt +./run ``` -__Users can refer to relative paths in the parameter file Test.para, the file manifest matrice.txt, and the file samples.txt. These relative paths were made for the example run. Users will need to change the path for their own analyses.__ The run_iris script takes as input a [simplified parameter file](example/Test_simplified.para) and a .tar.gz of the [SJ_matrices](example/SJ_matrices.tar.gz) which are preprocessed before calling the IRIS modules. The preprocessing adds absolute paths based on the input relative paths. +__As mentioned in [Usage](#usage), the full example is designed to be run with a compute cluster.__ It will take < 5 min for the formatting and screening steps and usually < 15 min for the prediction step (depending on available cluster resources). + + +A successful test run will generate the following result files in `./results/NEPC_test/screen/` (row numbers are displayed before each file name): +``` + 0 NEPC_test.SE.notest.txt + 1 NEPC_test.SE.test.all_guided.txt + 1 NEPC_test.SE.tier1.txt + 1 NEPC_test.SE.tier1.txt.integratedSJC.txt + 4 NEPC_test.SE.tier2tier3.txt.ExtraCellularAS.txt + 4 NEPC_test.SE.tier2tier3.txt.ExtraCellularAS.txt.integratedSJC.txt + 6 NEPC_test.SE.tier2tier3.txt + 6 NEPC_test.SE.tier2tier3.txt.ijc_info.txt + 6 NEPC_test.SE.tier2tier3.txt.integratedSJC.txt + 11 NEPC_test.SE.test.all_voted.txt + 4 SE.tier2tier3/epitope_summary.junction-based.txt + 4 SE.tier2tier3/epitope_summary.junction-based.txt.integratedSJC.txt + 9 SE.tier2tier3/epitope_summary.peptide-based.txt + 9 SE.tier2tier3/epitope_summary.peptide-based.txt.integratedSJC.txt + 11 SE.tier2tier3/pred_filtered.score500.txt +``` +A summary graphic is generated to `./results/NEPC_test/visualization/summary.png` + +## Example output -### Example output Final reports are shown in __bold__ font. -#### Screening results -[TASK/DATA_NAME].test.all.txt: All AS events tested by IRIS screening +### Screening results -[TASK/DATA_NAME].notest.txt: During screening, AS events skipped due to no variance or no available comparisons +`[TASK/DATA_NAME].[AS_TYPE].test.all_guided.txt`: All AS events tested by IRIS screening with tissue-matched normal tissue reference panel available. One-sided test will be used to generate p-value. -[TASK/DATA_NAME].primary.txt: Tumor AS events after comparison to tissue-matched normal panel ('primary' events) +`[TASK/DATA_NAME].[AS_TYPE].test.all_voted.txt`: All AS events tested by IRIS screening without tissue-matched normal tissue reference panel. Two-sided test will be used to generate p-value for comparisons to normal panels. -[TASK/DATA_NAME].prioritized.txt: Tumor AS events after comparison to tissue-matched normal panel, tumor panel, and normal tissue panel ('prioritized' AS events) +`[TASK/DATA_NAME].[AS_TYPE].notest.txt`: During screening, AS events skipped due to no variance or no available comparisons -#### CAR-T annotation reports -__[TASK/DATA_NAME].primary.txt.ExtraCellularAS.txt__: Tumor AS events in 'primary' set that are associated with protein extracellular annotation and may be used for CAR-T targets +`[TASK/DATA_NAME].[AS_TYPE].tier1.txt`: Tumor AS events after comparison to tissue-matched normal panel ('tier1' events) -__[TASK/DATA_NAME].prioritized.txt.ExtraCellularAS.txt__: Tumor AS events in 'prioritized' set that are associated with protein extracellular annotation and may be used for CAR-T targets +`[TASK/DATA_NAME].[AS_TYPE].tier2tier3.txt`: Tumor AS events after comparison to tissue-matched normal panel, tumor panel, and normal tissue panel ('tier3' AS events) -#### TCR prediction reports -primary/pred_filtered.score500.txt: IEDB prediction outputs for AS junction peptides from 'primary' set with HLA-peptide binding IC50 values passing user-defined cut-off +### CAR-T annotation reports -__primary/epitope_summary.peptide-based.txt__: AS-derived epitopes from 'primary' set that are predicted to bind user-defined HLA type +__`[TASK/DATA_NAME].[AS_TYPE].tier1.txt.ExtraCellularAS.txt`__: Tumor AS events in 'tier1' set that are associated with protein extracellular annotation and may be used for CAR-T targets -__primary/epitope_summary.junction-based.txt__: Epitope-producing AS junctions from 'primary' set that are predicted to bind user-defined HLA type +__`[TASK/DATA_NAME].[AS_TYPE].tier2tier3.txt.ExtraCellularAS.txt`__: Tumor AS events in 'tier3' set that are associated with protein extracellular annotation and may be used for CAR-T targets -prioritized/pred_filtered.score500.txt: IEDB prediction outputs for AS junction peptides from 'prioritized' set with HLA-peptide binding IC50 value passing user-defined cut-off +### TCR prediction reports -__prioritized/epitope_summary.peptide-based.txt__: AS-derived epitopes from 'prioritized' set that are predicted to bind user-defined HLA type +`[AS_TYPE].tier1/pred_filtered.score500.txt`: IEDB prediction outputs for AS junction peptides from 'tier1' set with HLA-peptide binding IC50 values passing user-defined cut-off -__prioritized/epitope_summary.junction-based.txt__: Epitope-producing AS junctions from 'prioritized' set that are predicted to bind user-defined HLA type +__`[AS_TYPE].tier1/epitope_summary.peptide-based.txt`__: AS-derived epitopes from 'tier1' set that are predicted to bind user-defined HLA type +__`[AS_TYPE].tier1/epitope_summary.junction-based.txt`__: Epitope-producing AS junctions from 'tier1' set that are predicted to bind user-defined HLA type +`[AS_TYPE].tier2tier3/pred_filtered.score500.txt`: IEDB prediction outputs for AS junction peptides from 'tier3' set with HLA-peptide binding IC50 value passing user-defined cut-off +__`[AS_TYPE].tier2tier3/epitope_summary.peptide-based.txt`__: AS-derived epitopes from 'tier3' set that are predicted to bind user-defined HLA type +__`[AS_TYPE].tier2tier3/epitope_summary.junction-based.txt`__: Epitope-producing AS junctions from 'tier3' set that are predicted to bind user-defined HLA type + +### Tumor-specific screen reports + +Screening or prediction outputs that integrated `screen` and `screen_sjc` results contain annotation for tumor-specific targets. These output files are indicated by `.integratedSJC.txt`, such as `[TASK/DATA_NAME].[AS_TYPE]tier2tier3.txt.integratedSJC.txt` and __`[AS_TYPE].tier2tier3/epitope_summary.peptide-based.txt.integratedSJC.txt`__, etc. + +## Contact - ### Contact Yang Pan +Eric Kutschera + Yi Xing - +## Publication -### Publication Manuscript in submission - diff --git a/Snakefile b/Snakefile new file mode 100644 index 0000000..af3cf8f --- /dev/null +++ b/Snakefile @@ -0,0 +1,1766 @@ +import snakemake.utils + +snakemake.utils.min_version('6.5.0') + +configfile: 'snakemake_config.yaml' + +onsuccess: + print('workflow success') + +onerror: + print('workflow error') + +DEFAULT_MEM_MB=4 * 1024 # 4 GB +DEFAULT_TIME_HOURS=12 + +# Specifying this as an input to a rule will disable that rule. +# This can be used in combination with "ruleorder:" to determine what +# rule should be used to create a particular output file. +UNSATISFIABLE_INPUT='unsatisfiable_input_file_path' + + +def all_input(wildcards): + inputs = dict() + run_all_modules = bool(config.get('run_all_modules')) + run_core_modules = bool(config.get('run_core_modules')) + should_run_sjc = bool(config.get('should_run_sjc_steps')) + if run_core_modules or run_all_modules: + # core modules + inputs.update(iris_epitope_post_out_files()) + inputs['visualization'] = os.path.join(result_dir(), 'visualization', + 'summary.png') + + if should_run_sjc: + if has_tier_1(): + inputs['sjc_tier1'] = iris_append_sjc_out_file_name_for_tier('tier1') + if has_tier_3(): + inputs['sjc_tier2tier3'] = iris_append_sjc_out_file_name_for_tier('tier2tier3') + + return inputs + + +localrules: all +rule all: + input: + unpack(all_input), + + +def result_dir(): + return os.path.join('results', config['run_name']) + + +def iris_db_path(): + return os.path.join(config['iris_data'], 'db') + + +def iris_db_sjc_path(): + return os.path.join(config['iris_data'], 'db_sjc') + + +def iris_exp_matrix_out_matrix(): + run_name = config['run_name'] + basename = 'exp.merged_matrix.{}.txt'.format(run_name) + return os.path.join(result_dir(), 'exp_matrix', basename) + + +def gene_exp_matrix_path_for_run(): + from_config = config.get('gene_exp_matrix') + if from_config: + return from_config + + if config.get('run_all_modules'): + return iris_exp_matrix_out_matrix() + + return None + + +def hla_types_list_for_run(): + from_config = config.get('mhc_list') + if from_config: + return from_config + + return os.path.join(result_dir(), 'hla_typing', 'hla_types.list') + + +def hla_from_patients_for_run(): + from_config = config.get('mhc_by_sample') + if from_config: + return from_config + + return os.path.join(result_dir(), 'hla_typing', 'hla_patient.tsv') + +def splicing_matrix_path_for_run(): + db_path = iris_db_path() + + return os.path.join(db_path, config['run_name'], 'splicing_matrix') + + +def sjc_count_path_for_run(): + db_path = iris_db_sjc_path() + + return os.path.join(db_path, config['run_name'], 'sjc_matrix') + + +def splicing_matrix_txt_path_for_run(): + matrix_path = splicing_matrix_path_for_run() + file_name = ('splicing_matrix.{}.cov10.{}.txt' + .format(config['splice_event_type'], config['run_name'])) + return os.path.join(matrix_path, file_name) + + +def splicing_matrix_idx_path_for_run(): + return '{}.idx'.format(splicing_matrix_txt_path_for_run()) + + +def sjc_count_txt_path_for_run(): + matrix_path = sjc_count_path_for_run() + file_name = 'SJ_count.{}.txt'.format(config['run_name']) + return os.path.join(matrix_path, file_name) + + +def sjc_count_idx_path_for_run(): + return '{}.idx'.format(sjc_count_txt_path_for_run()) + + +def format_ref_names(config_key): + configured = config.get(config_key, '') + # if no ref names -> provide a quoted empty string on the command line + if not configured.strip(): + return "''" + + return configured + + +# must have either tier 1 or tier 3 +def has_tier_1(): + return len(tier_1_group_names()) > 0 + + +def has_tier_3(): + return len(tier_3_group_names()) > 0 + + +def tier_1_group_names(): + return group_names_from_config_key('tissue_matched_normal_reference_group_names') + + +def tier_3_group_names(): + return group_names_from_config_key('normal_reference_group_names') + + +def group_names_from_config_key(key): + names_str = config.get(key) + split = names_str.split(',') + return [x.strip() for x in split if x] + + +def reference_file_wildcard_constraints(): + reference_files = config.get('reference_files') + if reference_files: + file_names = '|'.join([re.escape(file_name) + for file_name in reference_files]) + without_gz = '|'.join([re.escape(file_name[:-3]) + for file_name in reference_files + if file_name.endswith('.gz')]) + else: + no_match = '^$' # only matches empty string + file_names = no_match + without_gz = no_match + + return {'file_names': file_names, 'without_gz': without_gz} + + +def get_url_for_download_reference_file(wildcards): + file_name = wildcards.file_name + return config['reference_files'][file_name]['url'] + + +rule download_reference_file: + output: + ref_file=os.path.join('references', '{file_name}'), + log: + out=os.path.join('references', + 'download_reference_file_{file_name}_log.out'), + err=os.path.join('references', + 'download_reference_file_{file_name}_log.err'), + wildcard_constraints: + file_name=reference_file_wildcard_constraints()['file_names'] + params: + url=get_url_for_download_reference_file, + resources: + mem_mb=DEFAULT_MEM_MB, + time_hours=DEFAULT_TIME_HOURS, + shell: + 'curl -L \'{params.url}\'' + ' -o {output.ref_file}' + ' 1> {log.out}' + ' 2> {log.err}' + +rule unzip_reference_file: + input: + gz=os.path.join('references', '{file_name}.gz'), + output: + un_gz=os.path.join('references', '{file_name}'), + log: + out=os.path.join('references', + 'unzip_reference_file_{file_name}_log.out'), + err=os.path.join('references', + 'unzip_reference_file_{file_name}_log.err'), + wildcard_constraints: + file_name=reference_file_wildcard_constraints()['without_gz'] + resources: + mem_mb=DEFAULT_MEM_MB, + time_hours=DEFAULT_TIME_HOURS, + shell: + ' gunzip -c {input.gz}' + ' 1> {output.un_gz}' + ' 2> {log.err}' + + +def write_param_file_blacklist_param(): + value = config.get('blacklist') + if value: + return '--blacklist-file {}'.format(value) + + return '' + + +def write_param_file_bigwig_param(): + value = config.get('mapability_bigwig') + if value: + return '--mapability-bigwig {}'.format(value) + + return '' + + +def write_param_file_genome_param(): + value = config.get('fasta_name') + if value: + reference_path = os.path.join('references', value) + return '--reference-genome {}'.format(reference_path) + + return '' + + +def write_param_file_input(wildcards): + inputs = dict() + fasta = config.get('fasta_name') + if fasta: + inputs['fasta'] = os.path.join('references', fasta) + + return inputs + + +rule write_param_file: + input: + unpack(write_param_file_input), + output: + param_file=os.path.join(result_dir(), 'screen.para'), + log: + out=os.path.join(result_dir(), 'write_param_file_log.out'), + err=os.path.join(result_dir(), 'write_param_file_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_3=config['conda_env_3'], + script=os.path.join('scripts', 'write_param_file.py'), + group_name=config['run_name'], + iris_db=iris_db_path(), + matched_psi_cut=config.get('tissue_matched_normal_psi_p_value_cutoff', ''), + matched_sjc_cut=config.get('tissue_matched_normal_sjc_p_value_cutoff', ''), + matched_delta_psi_cut=config.get('tissue_matched_normal_delta_psi_p_value_cutoff', ''), + matched_fc_cut=config.get('tissue_matched_normal_fold_change_cutoff', ''), + matched_group_cut=config.get('tissue_matched_normal_group_count_cutoff', ''), + matched_ref_names=format_ref_names('tissue_matched_normal_reference_group_names'), + tumor_psi_cut=config.get('tumor_psi_p_value_cutoff', ''), + tumor_sjc_cut=config.get('tumor_sjc_p_value_cutoff', ''), + tumor_delta_psi_cut=config.get('tumor_delta_psi_p_value_cutoff', ''), + tumor_fc_cut=config.get('tumor_fold_change_cutoff', ''), + tumor_group_cut=config.get('tumor_group_count_cutoff', ''), + tumor_ref_names=format_ref_names('tumor_reference_group_names'), + normal_psi_cut=config.get('normal_psi_p_value_cutoff', ''), + normal_sjc_cut=config.get('normal_sjc_p_value_cutoff', ''), + normal_delta_psi_cut=config.get('normal_delta_psi_p_value_cutoff', ''), + normal_fc_cut=config.get('normal_fold_change_cutoff', ''), + normal_group_cut=config.get('normal_group_count_cutoff', ''), + normal_ref_names=format_ref_names('normal_reference_group_names'), + comparison_mode=config['comparison_mode'], + stat_test_type=config['stat_test_type'], + use_ratio='--use-ratio' if config.get('use_ratio') else '', + blacklist=write_param_file_blacklist_param(), + bigwig=write_param_file_bigwig_param(), + genome=write_param_file_genome_param(), + resources: + mem_mb=DEFAULT_MEM_MB, + time_hours=DEFAULT_TIME_HOURS, + shell: + '{params.conda_wrapper} {params.conda_env_3} python {params.script}' + ' --out-path {output.param_file}' + ' --group-name {params.group_name}' + ' --iris-db {params.iris_db}' + ' --psi-p-value-cutoffs' + ' {params.matched_psi_cut},{params.tumor_psi_cut},{params.normal_psi_cut}' + ' --sjc-p-value-cutoffs' + ' {params.matched_sjc_cut},{params.tumor_sjc_cut},{params.normal_sjc_cut}' + ' --delta-psi-cutoffs' + ' {params.matched_delta_psi_cut},{params.tumor_delta_psi_cut},{params.normal_delta_psi_cut}' + ' --fold-change-cutoffs' + ' {params.matched_fc_cut},{params.tumor_fc_cut},{params.normal_fc_cut}' + ' --group-count-cutoffs' + ' {params.matched_group_cut},{params.tumor_group_cut},{params.normal_group_cut}' + ' --reference-names-tissue-matched-normal {params.matched_ref_names}' + ' --reference-names-tumor {params.tumor_ref_names}' + ' --reference-names-normal {params.normal_ref_names}' + ' --comparison-mode {params.comparison_mode}' + ' --statistical-test-type {params.stat_test_type}' + ' {params.use_ratio}' + ' {params.blacklist}' + ' {params.bigwig}' + ' {params.genome}' + ' 1> {log.out}' + ' 2> {log.err}' + + +# if the necessary files are specified in the config, then +# use them rather than run IRIS format +def copy_splice_matrix_files_input(wildcards): + inputs = dict() + inputs['splice_txt'] = config.get('splice_matrix_txt', UNSATISFIABLE_INPUT) + inputs['splice_idx'] = config.get('splice_matrix_idx', UNSATISFIABLE_INPUT) + if config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + +ruleorder: copy_splice_matrix_files > iris_format +localrules: copy_splice_matrix_files +rule copy_splice_matrix_files: + input: + unpack(copy_splice_matrix_files_input), + output: + splice_txt=splicing_matrix_txt_path_for_run(), + splice_idx=splicing_matrix_idx_path_for_run(), + shell: + 'cp {input.splice_txt} {output.splice_txt}' + ' && cp {input.splice_idx} {output.splice_idx}' + + +def copy_sjc_count_files_input(wildcards): + inputs = dict() + inputs['count_txt'] = config.get('sjc_count_txt', UNSATISFIABLE_INPUT) + inputs['count_idx'] = config.get('sjc_count_idx', UNSATISFIABLE_INPUT) + if config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + +ruleorder: copy_sjc_count_files > iris_sjc_matrix +localrules: copy_sjc_count_files +rule copy_sjc_count_files: + input: + unpack(copy_sjc_count_files_input), + output: + count_txt=sjc_count_txt_path_for_run(), + count_idx=sjc_count_idx_path_for_run(), + shell: + 'cp {input.count_txt} {output.count_txt}' + ' && cp {input.count_idx} {output.count_idx}' + + +def create_star_index_out_dir_param(wildcards, output): + return os.path.dirname(output.index) + + +def create_star_index_input(wildcards): + inputs = dict() + inputs['gtf'] = os.path.join('references', config['gtf_name']) + inputs['fasta'] = os.path.join('references', config['fasta_name']) + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule create_star_index: + input: + unpack(create_star_index_input), + output: + index=os.path.join('references', 'star_index', 'SA'), + log: + out=os.path.join('references', 'create_star_index_log.out'), + err=os.path.join('references', 'create_star_index_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + out_dir=create_star_index_out_dir_param, + overhang=config['star_sjdb_overhang'], + threads: config['create_star_index_threads'] + resources: + mem_mb=config['create_star_index_mem_gb'] * 1024, + time_hours=config['create_star_index_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} STAR' + ' --runMode genomeGenerate' + ' --runThreadN {threads}' + ' --genomeDir {params.out_dir}' + ' --genomeFastaFiles {input.fasta}' + ' --sjdbGTFfile {input.gtf}' + ' --sjdbOverhang {params.overhang}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def organize_fastqs_sample_details(): + details = dict() + fastq_dict = config.get('sample_fastqs') + if not fastq_dict: + return details + + sample_names = list() + all_fastqs = list() + for name, fastqs in fastq_dict.items(): + for fastq in fastqs: + sample_names.append(name) + all_fastqs.append(fastq) + + details['sample_names'] = sample_names + details['fastqs'] = all_fastqs + return details + + +def unique_sample_names(): + fastq_dict = config.get('sample_fastqs') + if not fastq_dict: + return list() + + return list(fastq_dict.keys()) + + +def organize_fastqs_input(wildcards): + details = organize_fastqs_sample_details() + if not details: + return {'unsatisfiable': UNSATISFIABLE_INPUT} + + return {'fastqs': details['fastqs']} + + +def organize_fastqs_sample_names_param(): + sample_names = organize_fastqs_sample_details().get('sample_names', list()) + return sample_names + + +localrules: organize_fastqs +rule organize_fastqs: + input: + unpack(organize_fastqs_input), + output: + done=touch(os.path.join(result_dir(), 'fastq_dir', 'organize_fastqs.done')), + params: + sample_names=organize_fastqs_sample_names_param(), + out_dir=os.path.join(result_dir(), 'fastq_dir'), + run: + import os + import os.path + + out_dir = params.out_dir + if os.path.isdir(out_dir): + files = os.listdir(out_dir) + if files: + raise Exception('organize_fastqs: {} already contains files' + .format(out_dir)) + + for i, sample_name in enumerate(params.sample_names): + sample_dir = os.path.join(out_dir, sample_name) + orig_fastq_path = input.fastqs[i] + fastq_basename = os.path.basename(orig_fastq_path) + new_fastq_path = os.path.join(sample_dir, fastq_basename) + os.makedirs(sample_dir, exist_ok=True) + os.symlink(orig_fastq_path, new_fastq_path) + + +def iris_makesubsh_mapping_task_out_file_names(): + task_dir = os.path.join(result_dir(), 'mapping_tasks') + sample_names = unique_sample_names() + star_tasks = list() + cuff_tasks = list() + for sample_name in sample_names: + star_name = 'STARmap.{}.sh'.format(sample_name) + cuff_name = 'Cuffquant.{}.sh'.format(sample_name) + star_tasks.append(os.path.join(task_dir, star_name)) + cuff_tasks.append(os.path.join(task_dir, cuff_name)) + + return {'star_tasks': star_tasks, 'cuff_tasks': cuff_tasks} + + +def iris_makesubsh_mapping_star_done_file_names(): + out_dir = os.path.join(result_dir(), 'process_rnaseq') + sample_names = unique_sample_names() + final_bams = list() + for sample in sample_names: + align_dir = os.path.join(out_dir, '{}.aln'.format(sample)) + final_bam = os.path.join(align_dir, 'Aligned.sortedByCoord.out.bam') + final_bams.append(final_bam) + + return final_bams + + +def iris_makesubsh_mapping_cuff_done_file_names(): + cuff_tasks = iris_makesubsh_mapping_task_out_file_names()['cuff_tasks'] + done_names = list() + for task in cuff_tasks: + done_names.append('{}.done'.format(task)) + + return done_names + + +def iris_makesubsh_mapping_star_dir_param(wildcards): + input = iris_makesubsh_mapping_input(wildcards) + return os.path.dirname(input['index']) + + +def iris_makesubsh_mapping_task_dir_param(wildcards, output): + return os.path.dirname(output.star_tasks[0]) + + +def label_string_param(): + # IRIS uses this value to tell which files are for read 1 or read 2. + # Specifically it looks for '1{label_string}f' and '1{label_string}f' + return '.' + + +def iris_makesubsh_mapping_input(wildcards): + inputs = dict() + inputs['organize_fastqs_done'] = os.path.join(result_dir(), 'fastq_dir', + 'organize_fastqs.done') + inputs['index'] = os.path.join('references', 'star_index', 'SA') + inputs['gtf'] = os.path.join('references', config['gtf_name']) + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_makesubsh_mapping: + input: + unpack(iris_makesubsh_mapping_input), + output: + **iris_makesubsh_mapping_task_out_file_names() + log: + out=os.path.join(result_dir(), 'iris_makesubsh_mapping_log.out'), + err=os.path.join(result_dir(), 'iris_makesubsh_mapping_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + fastq_dir=os.path.join(result_dir(), 'fastq_dir'), + star_dir=iris_makesubsh_mapping_star_dir_param, + run_name=config['run_name'], + out_dir=os.path.join(result_dir(), 'process_rnaseq'), + label_string=label_string_param(), + task_dir=iris_makesubsh_mapping_task_dir_param, + resources: + mem_mb=DEFAULT_MEM_MB, + time_hours=DEFAULT_TIME_HOURS, + shell: + '{params.conda_wrapper} {params.conda_env_2} IRIS makesubsh_mapping' + ' --fastq-folder-dir {params.fastq_dir}' + ' --starGenomeDir {params.star_dir}' + ' --gtf {input.gtf}' + ' --data-name {params.run_name}' + ' --outdir {params.out_dir}' + ' --label-string {params.label_string}' + ' --task-dir {params.task_dir}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_star_task_input(wildcards): + inputs = dict() + inputs['star_task'] = os.path.join(result_dir(), 'mapping_tasks', + 'STARmap.{sample}.sh') + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_star_task: + input: + unpack(iris_star_task_input), + output: + unsorted_bam=os.path.join(result_dir(), 'process_rnaseq', + '{sample}.aln', 'Aligned.out.bam'), + sorted_bam=os.path.join(result_dir(), 'process_rnaseq', '{sample}.aln', + 'Aligned.sortedByCoord.out.bam'), + log: + out=os.path.join(result_dir(), 'mapping_tasks', + 'iris_star_task_{sample}_log.out'), + err=os.path.join(result_dir(), 'mapping_tasks', + 'iris_star_task_{sample}_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + threads: config['iris_star_task_threads'] + resources: + mem_mb=config['iris_star_task_mem_gb'] * 1024, + time_hours=config['iris_star_task_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} bash' + ' {input.star_task}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_cuff_task_input(wildcards): + inputs = dict() + inputs['cuff_task'] = os.path.join(result_dir(), 'mapping_tasks', + 'Cuffquant.{sample}.sh') + inputs['star_task_done'] = os.path.join(result_dir(), 'process_rnaseq', + '{sample}.aln', + 'Aligned.sortedByCoord.out.bam') + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_cuff_task: + input: + unpack(iris_cuff_task_input), + output: + cuff_task_done=touch(os.path.join(result_dir(), 'process_rnaseq', + '{sample}.aln', 'cufflinks', + 'genes.fpkm_tracking')), + log: + out=os.path.join(result_dir(), 'mapping_tasks', + 'iris_cuff_task_{sample}_log.out'), + err=os.path.join(result_dir(), 'mapping_tasks', + 'iris_cuff_task_{sample}_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + threads: config['iris_cuff_task_threads'] + resources: + mem_mb=config['iris_cuff_task_mem_gb'] * 1024, + time_hours=config['iris_cuff_task_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} bash' + ' {input.cuff_task}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_makesubsh_hla_task_out_file_names(): + task_dir = os.path.join(result_dir(), 'hla_tasks') + sample_names = unique_sample_names() + hla_tasks = list() + for sample_name in sample_names: + hla_name = 'seq2hla.{}.sh'.format(sample_name) + hla_tasks.append(os.path.join(task_dir, hla_name)) + + return {'hla_tasks': hla_tasks} + + +def iris_hla_task_done_file_names(): + sample_names = unique_sample_names() + done_file_names = list() + hla_dir = os.path.join(result_dir(), 'hla_typing') + for sample in sample_names: + out_dir = os.path.join(hla_dir, sample) + expression = os.path.join(out_dir, + '{}-ClassI.expression'.format(sample)) + genotype = os.path.join(out_dir, + '{}-ClassI.HLAgenotype4digits'.format(sample)) + done_file_names.append(expression) + done_file_names.append(genotype) + + return done_file_names + + +def iris_makesubsh_hla_task_dir_param(wildcards, output): + return os.path.dirname(output.hla_tasks[0]) + + +def iris_makesubsh_hla_input(wildcards): + inputs = dict() + inputs['organize_fastqs_done'] = os.path.join(result_dir(), 'fastq_dir', + 'organize_fastqs.done') + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_makesubsh_hla: + input: + unpack(iris_makesubsh_hla_input), + + output: + **iris_makesubsh_hla_task_out_file_names() + log: + out=os.path.join(result_dir(), 'iris_makesubsh_hla_log.out'), + err=os.path.join(result_dir(), 'iris_makesubsh_hla_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + fastq_dir=os.path.join(result_dir(), 'fastq_dir'), + run_name=config['run_name'], + out_dir=os.path.join(result_dir(), 'hla_typing'), + label_string=label_string_param(), + task_dir=iris_makesubsh_hla_task_dir_param, + resources: + mem_mb=DEFAULT_MEM_MB, + time_hours=DEFAULT_TIME_HOURS, + shell: + '{params.conda_wrapper} {params.conda_env_2} IRIS makesubsh_hla' + ' --fastq-folder-dir {params.fastq_dir}' + ' --data-name {params.run_name}' + ' --outdir {params.out_dir}' + ' --label-string {params.label_string}' + ' --task-dir {params.task_dir}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_hla_task_input(wildcards): + inputs = dict() + inputs['hla_task'] = os.path.join(result_dir(), 'hla_tasks', + 'seq2hla.{sample}.sh') + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_hla_task: + input: + unpack(iris_hla_task_input), + output: + expression=os.path.join(result_dir(), 'hla_typing', '{sample}', + '{sample}-ClassI.expression'), + genotype=os.path.join(result_dir(), 'hla_typing', '{sample}', + '{sample}-ClassI.HLAgenotype4digits'), + log: + out=os.path.join(result_dir(), 'hla_tasks', + 'iris_hla_task_{sample}_log.out'), + err=os.path.join(result_dir(), 'hla_tasks', + 'iris_hla_task_{sample}_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + threads: config['iris_hla_task_threads'] + resources: + mem_mb=config['iris_hla_task_mem_gb'] * 1024, + time_hours=config['iris_hla_task_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} bash' + ' {input.hla_task}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_parse_hla_input(wildcards): + inputs = dict() + inputs['hla_tasks_done'] = iris_hla_task_done_file_names() + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_parse_hla: + input: + unpack(iris_parse_hla_input), + output: + patient=os.path.join(result_dir(), 'hla_typing', 'hla_patient.tsv'), + types=os.path.join(result_dir(), 'hla_typing', 'hla_types.list'), + exp=os.path.join(result_dir(), 'hla_typing', 'hla_exp.list'), + log: + out=os.path.join(result_dir(), 'iris_parse_hla_log.out'), + err=os.path.join(result_dir(), 'iris_parse_hla_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + out_dir=os.path.join(result_dir(), 'hla_typing'), + resources: + mem_mb=config['iris_parse_hla_mem_gb'] * 1024, + time_hours=config['iris_parse_hla_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} IRIS parse_hla' + ' --outdir {params.out_dir}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_makesubsh_rmats_task_out_file_names(): + task_dir = os.path.join(result_dir(), 'rmats_tasks') + sample_names = unique_sample_names() + rmats_tasks = list() + for sample_name in sample_names: + rmats_name = 'rMATS_prep.{}.sh'.format(sample_name) + rmats_tasks.append(os.path.join(task_dir, rmats_name)) + + return {'rmats_tasks': rmats_tasks} + + +def iris_makesubsh_rmats_done_file_names(): + rmats_tasks = iris_makesubsh_rmats_task_out_file_names()['rmats_tasks'] + done_names = list() + for task in rmats_tasks: + done_names.append('{}.done'.format(task)) + + return done_names + + +def iris_makesubsh_rmats_task_dir_param(wildcards, output): + return os.path.dirname(output.rmats_tasks[0]) + + +def iris_makesubsh_rmats_input(wildcards): + inputs = dict() + inputs['star_done'] = iris_makesubsh_mapping_star_done_file_names() + inputs['gtf'] = os.path.join('references', config['gtf_name']) + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_makesubsh_rmats: + input: + unpack(iris_makesubsh_rmats_input), + output: + **iris_makesubsh_rmats_task_out_file_names() + log: + out=os.path.join(result_dir(), 'iris_makesubsh_rmats_log.out'), + err=os.path.join(result_dir(), 'iris_makesubsh_rmats_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + rmats_path=config['rmats_path'], + bam_dir=os.path.join(result_dir(), 'process_rnaseq'), + run_name=config['run_name'], + task_dir=iris_makesubsh_rmats_task_dir_param, + resources: + mem_mb=DEFAULT_MEM_MB, + time_hours=DEFAULT_TIME_HOURS, + shell: + '{params.conda_wrapper} {params.conda_env_2} IRIS makesubsh_rmats' + ' --rMATS-path {params.rmats_path}' + ' --bam-dir {params.bam_dir}' + ' --gtf {input.gtf}' + ' --data-name {params.run_name}' + ' --task-dir {params.task_dir}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_rmats_task_input(wildcards): + inputs = dict() + inputs['rmats_task'] = os.path.join(result_dir(), 'rmats_tasks', + 'rMATS_prep.{sample}.sh') + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_rmats_task: + input: + unpack(iris_rmats_task_input), + output: + # The output files have the format: + # result_dir()/process_rnaseq/{run}.RL{readLength}/{sample}.tmp/{datetime}_{n}.rmats + # Just using a .done file instead. + rmats_task_done=touch(os.path.join(result_dir(), 'rmats_tasks', + 'rMATS_prep.{sample}.sh.done')), + log: + out=os.path.join(result_dir(), 'rmats_tasks', + 'iris_rmats_task_{sample}_log.out'), + err=os.path.join(result_dir(), 'rmats_tasks', + 'iris_rmats_task_{sample}_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + threads: config['iris_rmats_task_threads'] + resources: + mem_mb=config['iris_rmats_task_mem_gb'] * 1024, + time_hours=config['iris_rmats_task_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} bash' + ' {input.rmats_task}' + ' 1> {log.out}' + ' 2> {log.err}' + +checkpoint check_read_lengths: + input: + rmats_done=iris_makesubsh_rmats_done_file_names(), + output: + read_lengths=os.path.join(result_dir(), 'process_rnaseq', + 'read_lengths.txt'), + log: + out=os.path.join(result_dir(), 'process_rnaseq', 'check_read_lengths_log.out'), + err=os.path.join(result_dir(), 'process_rnaseq', 'check_read_lengths_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_3=config['conda_env_3'], + script=os.path.join('scripts', 'check_read_lengths.py'), + parent_dir=os.path.join(result_dir(), 'process_rnaseq'), + run_name=config['run_name'], + resources: + mem_mb=DEFAULT_MEM_MB, + time_hours=DEFAULT_TIME_HOURS, + shell: + '{params.conda_wrapper} {params.conda_env_3} python {params.script}' + ' --parent-dir {params.parent_dir}' + ' --run-name {params.run_name}' + ' --out {output.read_lengths}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_makesubsh_rmatspost_input(wildcards): + inputs = dict() + inputs['rmats_done'] = iris_makesubsh_rmats_done_file_names() + inputs['gtf'] = os.path.join('references', config['gtf_name']) + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_makesubsh_rmatspost: + input: + rmats_done=iris_makesubsh_rmats_done_file_names(), + gtf=os.path.join('references', config['gtf_name']), + output: + makesubsh_done=touch(os.path.join(result_dir(), 'iris_makesubsh_rmats_post.done')), + log: + out=os.path.join(result_dir(), 'iris_makesubsh_rmatspost_log.out'), + err=os.path.join(result_dir(), 'iris_makesubsh_rmatspost_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + rmats_path=config['rmats_path'], + bam_dir=os.path.join(result_dir(), 'process_rnaseq'), + run_name=config['run_name'], + task_dir=os.path.join(result_dir(), 'rmats_post_tasks'), + resources: + mem_mb=DEFAULT_MEM_MB, + time_hours=DEFAULT_TIME_HOURS, + shell: + '{params.conda_wrapper} {params.conda_env_2} IRIS makesubsh_rmatspost' + ' --rMATS-path {params.rmats_path}' + ' --bam-dir {params.bam_dir}' + ' --gtf {input.gtf}' + ' --data-name {params.run_name}' + ' --task-dir {params.task_dir}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_rmatspost_task_input(wildcards): + inputs = dict() + inputs['makesubsh_done'] = os.path.join( + result_dir(), 'iris_makesubsh_rmats_post.done') + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_rmatspost_task: + input: + unpack(iris_rmatspost_task_input), + output: + summary=os.path.join(result_dir(), 'process_rnaseq', + '{run_name}.RL{read_length}', + '{run_name}_RL{read_length}.matrix', + 'summary.txt'), + log: + out=os.path.join(result_dir(), 'rmats_post_tasks', + 'iris_rmatspost_task_{run_name}_{read_length}_log.out'), + err=os.path.join(result_dir(), 'rmats_post_tasks', + 'iris_rmatspost_task_{run_name}_{read_length}_log.err'), + wildcard_constraints: + run_name=config['run_name'], + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + # post_task is generated by iris_makesubsh_rmatspost, but since the number of + # read_lengths is not known until the check_read_lengths checkpoint, + # that .sh file is not used in the "input" or "output" sections of the snakemake + post_task=os.path.join(result_dir(), 'rmats_post_tasks', + 'rMATS_post.{run_name}_RL{read_length}.sh'), + threads: config['iris_rmatspost_task_threads'] + resources: + mem_mb=config['iris_rmatspost_task_mem_gb'] * 1024, + time_hours=config['iris_rmatspost_task_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} bash' + ' {params.post_task}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def prepare_iris_format_input(wildcards): + read_lengths_file = checkpoints.check_read_lengths.get().output[0] + summaries = list() + run_name = config['run_name'] + input_prefix = os.path.join(result_dir(), 'process_rnaseq') + with open(read_lengths_file, 'rt') as in_handle: + for line in in_handle: + read_length = line.strip() + run_with_read_length_dot = '{}.RL{}'.format(run_name, read_length) + run_with_read_length_underscore = '{}_RL{}'.format(run_name, read_length) + summary = os.path.join( + input_prefix, run_with_read_length_dot, + '{}.matrix'.format(run_with_read_length_underscore), + 'summary.txt') + summaries.append(summary) + + return {'summaries': summaries} + + +rule prepare_iris_format: + input: + unpack(prepare_iris_format_input), + output: + matrix=os.path.join(result_dir(), 'process_rnaseq', 'matrix_list.txt'), + sample=os.path.join(result_dir(), 'process_rnaseq', 'sample_list.txt'), + log: + out=os.path.join(result_dir(), 'process_rnaseq', + 'prepare_iris_format_log.out'), + err=os.path.join(result_dir(), 'process_rnaseq', + 'prepare_iris_format_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_3=config['conda_env_3'], + script=os.path.join('scripts', 'prepare_iris_format.py'), + resources: + mem_mb=DEFAULT_MEM_MB, + time_hours=DEFAULT_TIME_HOURS, + shell: + '{params.conda_wrapper} {params.conda_env_3} python {params.script}' + ' --matrix-out {output.matrix}' + ' --sample-out {output.sample}' + ' --summaries {input.summaries}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_format_input(wildcards): + inputs = dict() + inputs['matrix_list'] = os.path.join(result_dir(), 'process_rnaseq', + 'matrix_list.txt') + inputs['sample_list'] = os.path.join(result_dir(), 'process_rnaseq', + 'sample_list.txt') + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +# after IRIS format, IRIS index does not need to be run +rule iris_format: + input: + unpack(iris_format_input), + output: + matrix=splicing_matrix_txt_path_for_run(), + idx=splicing_matrix_idx_path_for_run(), + log: + out=os.path.join(result_dir(), 'iris_format_log.out'), + err=os.path.join(result_dir(), 'iris_format_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + splice_type=config['splice_event_type'], + run_name=config['run_name'], + iris_db=iris_db_path(), + sample_name_field='2', + resources: + mem_mb=config['iris_format_mem_gb'] * 1024, + time_hours=config['iris_format_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} IRIS format' + ' {input.matrix_list}' + ' {input.sample_list}' + ' --splicing-event-type {params.splice_type}' + ' --data-name {params.run_name}' + ' --sample-name-field {params.sample_name_field}' + ' --sample-based-filter' + ' --iris-db-path {params.iris_db}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def prepare_iris_exp_matrix_input(wildcards): + fpkm_files = list() + sample_names = unique_sample_names() + for name in sample_names: + fpkm_path = os.path.join( + result_dir(), 'process_rnaseq', '{}.aln'.format(name), 'cufflinks', + 'genes.fpkm_tracking') + fpkm_files.append(fpkm_path) + + return {'fpkm_files': fpkm_files} + + +rule prepare_iris_exp_matrix: + input: + unpack(prepare_iris_exp_matrix_input), + output: + manifest=os.path.join(result_dir(), 'process_rnaseq', + 'cufflinks_manifest.txt'), + log: + out=os.path.join(result_dir(), 'process_rnaseq', + 'prepare_iris_exp_matrix_log.out'), + err=os.path.join(result_dir(), 'process_rnaseq', + 'prepare_iris_exp_matrix_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_3=config['conda_env_3'], + script=os.path.join('scripts', 'prepare_iris_exp_matrix.py'), + resources: + mem_mb=DEFAULT_MEM_MB, + time_hours=DEFAULT_TIME_HOURS, + shell: + '{params.conda_wrapper} {params.conda_env_3} python {params.script}' + ' --out-manifest {output.manifest}' + ' --fpkm-files {input.fpkm_files}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_exp_matrix_out_dir_param(wildcards, output): + return os.path.dirname(output.matrix) + + +def iris_exp_matrix_input(wildcards): + inputs = dict() + inputs['manifest'] = os.path.join(result_dir(), 'process_rnaseq', + 'cufflinks_manifest.txt') + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_exp_matrix: + input: + unpack(iris_exp_matrix_input), + output: + matrix=iris_exp_matrix_out_matrix(), + log: + out=os.path.join(result_dir(), 'iris_exp_matrix_log.out'), + err=os.path.join(result_dir(), 'iris_exp_matrix_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + run_name=config['run_name'], + out_dir=iris_exp_matrix_out_dir_param, + resources: + mem_mb=config['iris_exp_matrix_mem_gb'] * 1024, + time_hours=config['iris_exp_matrix_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} IRIS exp_matrix' + ' --outdir {params.out_dir}' + ' --data-name {params.run_name}' + ' {input.manifest}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_makesubsh_extract_sjc_bam_dir_param(wildcards): + input = iris_makesubsh_extract_sjc_input(wildcards) + return os.path.dirname(input['bam']) + + +def iris_makesubsh_extract_sjc_task_dir_param(wildcards, output): + return os.path.dirname(output.extract_task) + + +def iris_makesubsh_extract_sjc_input(wildcards): + inputs = dict() + inputs['bam'] = os.path.join(result_dir(), 'process_rnaseq', '{sample}.aln', + 'Aligned.sortedByCoord.out.bam') + inputs['gtf'] = os.path.join('references', config['gtf_name']) + inputs['fasta'] = os.path.join('references', config['fasta_name']) + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_makesubsh_extract_sjc: + input: + unpack(iris_makesubsh_extract_sjc_input), + output: + extract_task=os.path.join(result_dir(), 'extract_sjc_tasks', + 'cmdlist.extract_sjc.{sample}'), + bam_list=os.path.join(result_dir(), 'extract_sjc_tasks', + 'bam_folder_list_{sample}.txt'), + log: + out=os.path.join(result_dir(), 'extract_sjc_tasks', + 'iris_makesubsh_extract_sjc_{sample}_log.out'), + err=os.path.join(result_dir(), 'extract_sjc_tasks', + 'iris_makesubsh_extract_sjc_{sample}_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + bam_dir=iris_makesubsh_extract_sjc_bam_dir_param, + task_name='{sample}', + bam_prefix='Aligned.sortedByCoord.out', + task_dir=iris_makesubsh_extract_sjc_task_dir_param, + resources: + shell: + 'echo {params.bam_dir} > {output.bam_list}' + ' && {params.conda_wrapper} {params.conda_env_2} IRIS' + ' makesubsh_extract_sjc' + ' --bam-folder-list {output.bam_list}' + ' --task-name {params.task_name}' + ' --gtf {input.gtf}' + ' --genome-fasta {input.fasta}' + ' --BAM-prefix {params.bam_prefix}' + ' --task-dir {params.task_dir}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_extract_sjc_task_input(wildcards): + inputs = dict() + inputs['extract_task'] = os.path.join(result_dir(), 'extract_sjc_tasks', + 'cmdlist.extract_sjc.{sample}') + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_extract_sjc_task: + input: + unpack(iris_extract_sjc_task_input), + output: + sj_count=os.path.join(result_dir(), 'process_rnaseq', '{sample}.aln', + 'SJcount.txt'), + log: + out=os.path.join(result_dir(), 'extract_sjc_tasks', + 'iris_extract_sjc_task_{sample}_log.out'), + err=os.path.join(result_dir(), 'extract_sjc_tasks', + 'iris_extract_sjc_task_{sample}_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + resources: + mem_mb=config['iris_extract_sjc_task_mem_gb'] * 1024, + time_hours=config['iris_extract_sjc_task_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} bash' + ' {input.extract_task}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def prepare_iris_sjc_matrix_input(wildcards): + sample_names = unique_sample_names() + sj_files = list() + for name in sample_names: + sample_dir = '{}.aln'.format(name) + sj_file = os.path.join(result_dir(), 'process_rnaseq', sample_dir, + 'SJcount.txt') + sj_files.append(sj_file) + + return {'sj_files': sj_files} + + +rule prepare_iris_sjc_matrix: + input: + unpack(prepare_iris_sjc_matrix_input), + output: + sj_list=os.path.join(result_dir(), 'process_rnaseq', 'sjc_file_list.txt'), + log: + out=os.path.join(result_dir(), 'process_rnaseq', + 'prepare_iris_sjc_matrix_log.out'), + err=os.path.join(result_dir(), 'process_rnaseq', + 'prepare_iris_sjc_matrix_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_3=config['conda_env_3'], + script=os.path.join('scripts', 'prepare_iris_sjc_matrix.py'), + resources: + mem_mb=DEFAULT_MEM_MB, + time_hours=DEFAULT_TIME_HOURS, + shell: + '{params.conda_wrapper} {params.conda_env_3} python {params.script}' + ' --sj-out {output.sj_list}' + ' --sj-files {input.sj_files}' + ' 1> {log.out}' + ' 2> {log.err}' + +def iris_sjc_matrix_input(wildcards): + inputs = dict() + inputs['sj_list'] = os.path.join(result_dir(), 'process_rnaseq', + 'sjc_file_list.txt') + if not config['run_all_modules']: + inputs['run_all_modules'] = UNSATISFIABLE_INPUT + + return inputs + + +rule iris_sjc_matrix: + input: + unpack(iris_sjc_matrix_input), + output: + count_txt=sjc_count_txt_path_for_run(), + count_idx=sjc_count_idx_path_for_run(), + log: + out=os.path.join(result_dir(), 'iris_sjc_matrix_log.out'), + err=os.path.join(result_dir(), 'iris_sjc_matrix_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + run_name=config['run_name'], + sample_name_field='2', + db_sjc=iris_db_sjc_path(), + resources: + mem_mb=config['iris_sjc_matrix_mem_gb'] * 1024, + time_hours=config['iris_sjc_matrix_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} IRIS sjc_matrix' + ' --file-list-input {input.sj_list}' + ' --data-name {params.run_name}' + ' --sample-name-field {params.sample_name_field}' + ' --iris-db-path {params.db_sjc}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_screen_out_dir_param(wildcards, output): + return os.path.dirname(output.guided) + + +def iris_screen_out_files(): + out_files = dict() + out_dir = os.path.join(result_dir(), 'screen') + out_prefix = '{}.{}'.format(config['run_name'], config['splice_event_type']) + out_files['guided'] = os.path.join( + out_dir, '{}.test.all_guided.txt'.format(out_prefix)) + out_files['voted'] = os.path.join( + out_dir, '{}.test.all_voted.txt'.format(out_prefix)) + out_files['notest'] = os.path.join( + out_dir, '{}.notest.txt'.format(out_prefix)) + out_files['tier1'] = os.path.join( + out_dir, '{}.tier1.txt'.format(out_prefix)) + out_files['tier2tier3'] = os.path.join( + out_dir, '{}.tier2tier3.txt'.format(out_prefix)) + + return out_files + + +rule iris_screen: + input: + parameter_file=os.path.join(result_dir(), 'screen.para'), + gtf=os.path.join('references', config['gtf_name']), + splice_txt=splicing_matrix_txt_path_for_run(), + splice_idx=splicing_matrix_idx_path_for_run(), + output: + **iris_screen_out_files() + log: + out=os.path.join(result_dir(), 'iris_screen_log.out'), + err=os.path.join(result_dir(), 'iris_screen_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + splice_event_type=config['splice_event_type'], + out_dir=iris_screen_out_dir_param, + resources: + mem_mb=config['iris_screen_mem_gb'] * 1024, + time_hours=config['iris_screen_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} IRIS screen' + ' --parameter-fin {input.parameter_file}' + ' --splicing-event-type {params.splice_event_type}' + ' --outdir {params.out_dir}' + ' --translating' # runs IRIS translate within IRIS screen + ' --gtf {input.gtf}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_predict_out_dir_param(wildcards, output): + return os.path.dirname(output.predict_out[0]) + + +def iris_predict_out_file_names(): + tier_names = list() + if has_tier_1(): + tier_names.append('tier1') + if has_tier_3(): + tier_names.append('tier2tier3') + + names = list() + for tier_name in tier_names: + basename = '{}.{}.{}.txt.ExtraCellularAS.txt'.format( + config['run_name'], config['splice_event_type'], tier_name) + name = os.path.join(result_dir(), 'screen', basename) + names.append(name) + + return names + + +def iris_predict_input(wildcards): + inputs = dict() + inputs['parameter_file'] = os.path.join(result_dir(), 'screen.para') + inputs['screen_out'] = iris_screen_out_files()['guided'] + inputs['mhc_list'] = hla_types_list_for_run() + gene_path = gene_exp_matrix_path_for_run() + if gene_path: + inputs['gene_exp_matrix'] = gene_path + + return inputs + + +def iris_predict_gene_exp_param(): + gene_exp_path = gene_exp_matrix_path_for_run() + if not gene_exp_path: + return '' + + return ' --gene-exp-matrix {}'.format(gene_exp_path) + + +rule iris_predict: + input: + unpack(iris_predict_input), + output: + predict_out=iris_predict_out_file_names(), + log: + out=os.path.join(result_dir(), 'iris_predict_log.out'), + err=os.path.join(result_dir(), 'iris_predict_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + out_dir=iris_predict_out_dir_param, + task_dir=os.path.join(result_dir(), 'predict_tasks'), + splice_event_type=config['splice_event_type'], + iedb_path=config['iedb_path'], + task_wildcard_string=os.path.join(result_dir(), 'predict_tasks', + 'pep2epitope_{}.tier*.*.sh'.format( + config['splice_event_type'])), + gene_exp=iris_predict_gene_exp_param(), + resources: + mem_mb=config['iris_predict_mem_gb'] * 1024, + time_hours=config['iris_predict_time_hr'], + shell: + # Remove any existing task scripts. + # Usually snakemake automatically removes output before running a job, but + # in this case the number of output files is not known in advance. + 'if [[ -n "$(ls {params.task_wildcard_string})" ]];' + ' then rm {params.task_wildcard_string};' + ' fi;' + ' {params.conda_wrapper} {params.conda_env_2} IRIS predict' + ' {params.out_dir}' + ' --task-dir {params.task_dir}' + ' --parameter-fin {input.parameter_file}' + ' --splicing-event-type {params.splice_event_type}' + ' --iedb-local {params.iedb_path}' + ' --mhc-list {input.mhc_list}' + ' {params.gene_exp}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def count_iris_predict_tasks_task_dir_param(wildcards, output): + return os.path.dirname(output.predict_task_list) + + +checkpoint count_iris_predict_tasks: + input: + predict_out=iris_predict_out_file_names(), + output: + predict_task_list=os.path.join(result_dir(), 'predict_tasks', + 'predict_tasks_list.txt'), + log: + out=os.path.join(result_dir(), 'predict_tasks', + 'count_iris_predict_tasks_log.out'), + err=os.path.join(result_dir(), 'predict_tasks', + 'count_iris_predict_tasks_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_3=config['conda_env_3'], + script=os.path.join('scripts', 'count_iris_predict_tasks.py'), + task_dir=count_iris_predict_tasks_task_dir_param, + splice_type=config['splice_event_type'], + resources: + mem_mb=DEFAULT_MEM_MB, + time_hours=DEFAULT_TIME_HOURS, + shell: + '{params.conda_wrapper} {params.conda_env_3} python {params.script}' + ' --out-list {output.predict_task_list}' + ' --task-dir {params.task_dir}' + ' --splice-type {params.splice_type}' + ' 1> {log.out}' + ' 2> {log.err}' + +rule iris_predict_task: + input: + predict_task=os.path.join(result_dir(), 'predict_tasks', '{task_name}.sh'), + output: + predict_task_done=touch(os.path.join(result_dir(), 'predict_tasks', + '{task_name}.sh.done')), + log: + out=os.path.join(result_dir(), 'predict_tasks', '{task_name}_log.out'), + err=os.path.join(result_dir(), 'predict_tasks', '{task_name}_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + resources: + mem_mb=config['iris_predict_task_mem_gb'] * 1024, + time_hours=config['iris_predict_task_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} bash' + ' {input.predict_task}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_epitope_post_out_dir_param(): + return os.path.join(result_dir(), 'screen') + + +def iris_epitope_post_out_files(): + files = dict() + splice_type = config['splice_event_type'] + tier1_dir = os.path.join(result_dir(), 'screen', '{}.tier1'.format(splice_type)) + if has_tier_1(): + files['tier1_junction'] = os.path.join(tier1_dir, 'epitope_summary.junction-based.txt') + files['tier1_peptide'] = os.path.join(tier1_dir, 'epitope_summary.peptide-based.txt') + files['tier1_filtered'] = os.path.join(tier1_dir, 'pred_filtered.score500.txt') + + tier2tier3_dir = os.path.join(result_dir(), 'screen', + '{}.tier2tier3'.format(splice_type)) + if has_tier_3(): + files['tier2tier3_junction'] = os.path.join( + tier2tier3_dir, 'epitope_summary.junction-based.txt') + files['tier2tier3_peptide'] = os.path.join( + tier2tier3_dir, 'epitope_summary.peptide-based.txt') + files['tier2tier3_filtered'] = os.path.join( + tier2tier3_dir, 'pred_filtered.score500.txt') + + return files + + +def iris_predict_task_done_file_names(): + tasks_list_file = checkpoints.count_iris_predict_tasks.get().output[0] + predict_tasks_done = list() + with open(tasks_list_file, 'rt') as handle: + for line in handle: + task_file = line.strip() + task_done_file = '{}.done'.format(task_file) + predict_tasks_done.append(task_done_file) + + return predict_tasks_done + + +def iris_epitope_post_input(wildcards): + inputs = dict() + predict_tasks_done = iris_predict_task_done_file_names() + inputs['predict_tasks_done'] = predict_tasks_done + inputs['parameter_file'] = os.path.join(result_dir(), 'screen.para') + inputs['mhc_by_sample'] = hla_from_patients_for_run() + gene_exp = gene_exp_matrix_path_for_run() + if gene_exp: + inputs['gene_exp_matrix'] = gene_exp + + return inputs + + +def iris_epitope_post_gene_exp_param(): + gene_exp_path = gene_exp_matrix_path_for_run() + if not gene_exp_path: + return '' + + return ' --gene-exp-matrix {}'.format(gene_exp_path) + + +rule iris_epitope_post: + input: + unpack(iris_epitope_post_input), + output: + **iris_epitope_post_out_files() + log: + out=os.path.join(result_dir(), 'iris_epitope_post_log.out'), + err=os.path.join(result_dir(), 'iris_epitope_post_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + out_dir=iris_epitope_post_out_dir_param(), + splice_event_type=config['splice_event_type'], + gene_exp=iris_epitope_post_gene_exp_param(), + resources: + mem_mb=config['iris_epitope_post_mem_gb'] * 1024, + time_hours=config['iris_epitope_post_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} IRIS epitope_post' + ' --parameter-fin {input.parameter_file}' + ' --outdir {params.out_dir}' + ' --splicing-event-type {params.splice_event_type}' + ' --mhc-by-sample {input.mhc_by_sample}' + ' {params.gene_exp}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_screen_sjc_out_dir_param(wildcards, output): + return os.path.dirname(output.screen_sjc_out) + + +def iris_screen_sjc_out_file_name(): + run_name = config['run_name'] + splice_type = config['splice_event_type'] + name = 'SJ.{}.{}.summary_by_sig_event.txt'.format(run_name, splice_type) + return os.path.join(result_dir(), 'screen_sjc', name) + + +rule iris_screen_sjc: + input: + parameter_file=os.path.join(result_dir(), 'screen.para'), + splice_txt=splicing_matrix_txt_path_for_run(), + splice_idx=splicing_matrix_idx_path_for_run(), + sjc_count_txt=sjc_count_txt_path_for_run(), + sjc_count_idx=sjc_count_idx_path_for_run(), + output: + screen_sjc_out=iris_screen_sjc_out_file_name(), + log: + out=os.path.join(result_dir(), 'iris_screen_sjc_log.out'), + err=os.path.join(result_dir(), 'iris_screen_sjc_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + splice_event_type=config['splice_event_type'], + out_dir=iris_screen_sjc_out_dir_param, + resources: + mem_mb=config['iris_screen_sjc_mem_gb'] * 1024, + time_hours=config['iris_screen_sjc_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} IRIS screen_sjc' + ' --parameter-file {input.parameter_file}' + ' --splicing-event-type {params.splice_event_type}' + ' --event-list-file {input.splice_txt}' + ' --outdir {params.out_dir}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_append_sjc_out_dir_param(wildcards): + input = iris_append_sjc_input(wildcards) + return os.path.dirname(input['event_list']) + + +def iris_append_sjc_event_list_for_tier(tier): + screen_out_files = iris_screen_out_files() + if tier not in ['tier1', 'tier2tier3']: + raise Exception('iris_append_sjc_event_list_for_tier({}): unexpected tier' + .format(tier)) + + return screen_out_files.get(tier) + + +def iris_append_sjc_out_file_name_for_tier(tier): + run_name = config['run_name'] + splice_type = config['splice_event_type'] + name = '{}.{}.{}.txt.ijc_info.txt'.format(run_name, splice_type, tier) + return os.path.join(result_dir(), 'screen', name) + + +def iris_append_sjc_out_file_name_with_tier_wildcard(): + run_name = config['run_name'] + splice_type = config['splice_event_type'] + name = '{}.{}.{{tier}}.txt.ijc_info.txt'.format(run_name, splice_type) + return os.path.join(result_dir(), 'screen', name) + + +def iris_append_sjc_input(wildcards): + inputs = dict() + predict_tasks_done = iris_predict_task_done_file_names() + inputs['predict_tasks_done'] = predict_tasks_done + inputs['parameter_file'] = os.path.join(result_dir(), 'screen.para') + inputs['screen_sjc_out'] = iris_screen_sjc_out_file_name() + inputs['event_list'] = iris_append_sjc_event_list_for_tier(wildcards.tier) + + epitope_post_out_files = iris_epitope_post_out_files() + junction_key = '{}_junction'.format(wildcards.tier) + inputs['epitope_post_junction'] = epitope_post_out_files[junction_key] + return inputs + + +rule iris_append_sjc: + input: + unpack(iris_append_sjc_input), + output: + append_sjc_out=iris_append_sjc_out_file_name_with_tier_wildcard(), + log: + out=os.path.join(result_dir(), 'iris_append_sjc_{tier}_log.out'), + err=os.path.join(result_dir(), 'iris_append_sjc_{tier}_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + splice_event_type=config['splice_event_type'], + out_dir=iris_append_sjc_out_dir_param, + resources: + mem_mb=config['iris_append_sjc_mem_gb'] * 1024, + time_hours=config['iris_append_sjc_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} IRIS append_sjc' + ' --sjc-summary {input.screen_sjc_out}' + ' --splicing-event-type {params.splice_event_type}' + ' --outdir {params.out_dir}' + ' --add-ijc-info' # runs IRIS annotate_ijc within IRIS append_sjc + ' --parameter-file {input.parameter_file}' + ' --screening-result-event-list {input.event_list}' + ' 1> {log.out}' + ' 2> {log.err}' + + +def iris_visual_summary_input(wildcards): + inputs = dict() + inputs['parameter_file'] = os.path.join(result_dir(), 'screen.para') + splice_type = config['splice_event_type'] + tier1_dir = os.path.join(result_dir(), 'screen', + '{}.tier1'.format(splice_type)) + if has_tier_1(): + inputs['tier1_peptide'] = os.path.join( + tier1_dir, 'epitope_summary.peptide-based.txt') + + tier2tier3_dir = os.path.join(result_dir(), 'screen', + '{}.tier2tier3'.format(splice_type)) + if has_tier_3(): + inputs['tier2tier3_peptide'] = os.path.join( + tier2tier3_dir, 'epitope_summary.peptide-based.txt') + + return inputs + + +rule iris_visual_summary: + input: + unpack(iris_visual_summary_input), + output: + summary=os.path.join(result_dir(), 'visualization', 'summary.png'), + log: + out=os.path.join(result_dir(), 'visualization', + 'iris_visual_summary_log.out'), + err=os.path.join(result_dir(), 'visualization', + 'iris_visual_summary_log.err'), + params: + conda_wrapper=config['conda_wrapper'], + conda_env_2=config['conda_env_2'], + splice_event_type=config['splice_event_type'], + screen_dir=os.path.join(result_dir(), 'screen'), + resources: + mem_mb=config['iris_visual_summary_mem_gb'] * 1024, + time_hours=config['iris_visual_summary_time_hr'], + shell: + '{params.conda_wrapper} {params.conda_env_2} IRIS visual_summary' + ' --parameter-fin {input.parameter_file}' + ' --screening-out-dir {params.screen_dir}' + ' --out-file-name {output.summary}' + ' --splicing-event-type {params.splice_event_type}' + ' 1> {log.out}' + ' 2> {log.err}' diff --git a/bin/IRIS b/bin/IRIS index 799cc64..af2c1d5 100644 --- a/bin/IRIS +++ b/bin/IRIS @@ -19,15 +19,15 @@ def main(): subcommand = args.subcommand - if subcommand == 'formatting': + if subcommand == 'format': from IRIS import IRIS_formatting IRIS_formatting.main(args) - elif subcommand == 'screening': + elif subcommand == 'screen': from IRIS import IRIS_screening IRIS_screening.main(args) - elif subcommand == 'prediction': + elif subcommand == 'predict': from IRIS import IRIS_prediction IRIS_prediction.main( args ) @@ -39,19 +39,39 @@ def main(): from IRIS import IRIS_process_rnaseq IRIS_process_rnaseq.main(args) - elif subcommand == 'makeqsub_rmats': - from IRIS import IRIS_makeqsub_rmats - IRIS_makeqsub_rmats.main(args) + elif subcommand == 'makesubsh_mapping': + from IRIS import IRIS_makesubsh_mapping + IRIS_makesubsh_mapping.main(args) + + elif subcommand == 'makesubsh_rmats': + from IRIS import IRIS_makesubsh_rmats + IRIS_makesubsh_rmats.main(args) + + elif subcommand == 'makesubsh_rmatspost': + from IRIS import IRIS_makesubsh_rmatspost + IRIS_makesubsh_rmatspost.main(args) elif subcommand == 'exp_matrix': from IRIS import IRIS_exp_matrix IRIS_exp_matrix.main(args) + + elif subcommand == 'makesubsh_extract_sjc': + from IRIS import IRIS_makesubsh_extractsj + IRIS_makesubsh_extractsj.main(args) + + elif subcommand == 'extract_sjc': + from IRIS import IRIS_extract_sjc + IRIS_extract_sjc.main(args) + + elif subcommand == 'sjc_matrix': + from IRIS import IRIS_sjc_matrix + IRIS_sjc_matrix.main(args) - elif subcommand == 'indexing': + elif subcommand == 'index': from IRIS import IRIS_indexing IRIS_indexing.main(args) - elif subcommand == 'translation': + elif subcommand == 'translate': from IRIS import IRIS_translation IRIS_translation.main(args) @@ -59,13 +79,41 @@ def main(): from IRIS import IRIS_pep2epitope IRIS_pep2epitope.main(args) - elif subcommand == 'screening_plot': + elif subcommand == 'screen_plot': from IRIS import IRIS_screening_plot IRIS_screening_plot.main(args) - elif subcommand == 'seq2hla': - from IRIS import IRIS_seq2hla - IRIS_seq2hla.main(args) + elif subcommand == 'screen_sjc': + from IRIS import IRIS_screening_sjc + IRIS_screening_sjc.main(args) + + elif subcommand == 'append_sjc': + from IRIS import IRIS_append_sjc + IRIS_append_sjc.main(args) + + elif subcommand == 'annotate_ijc': + from IRIS import IRIS_annotate_ijc + IRIS_annotate_ijc.main(args) + + elif subcommand == 'screen_cpm': + from IRIS import IRIS_screening_cpm + IRIS_screening_cpm.main(args) + + elif subcommand == 'append_cpm': + from IRIS import IRIS_append_cpm + IRIS_append_cpm.main(args) + + elif subcommand == 'screen_novelss': + from IRIS import IRIS_screening_novelss + IRIS_screening_novelss.main(args) + + elif subcommand == 'screen_sjc_plot': + from IRIS import IRIS_screening_sjcplot + IRIS_screening_sjcplot.main(args) + + elif subcommand == 'makesubsh_hla': + from IRIS import IRIS_makesubsh_hla + IRIS_makesubsh_hla.main(args) elif subcommand == 'parse_hla': from IRIS import IRIS_parse_hla @@ -83,6 +131,10 @@ def main(): from IRIS import IRIS_ms_parse IRIS_ms_parse.main(args) + elif subcommand == 'visual_summary': + from IRIS import IRIS_visual_summary + IRIS_visual_summary.main(args) + def get_arg_parser(): """DOCSTRING Args @@ -102,56 +154,86 @@ def get_arg_parser(): add_prediction_parser(subparsers) add_epitope_post_parser(subparsers) add_process_rnaseq_parser(subparsers) - add_rmats_prep_parser(subparsers) + add_makesubsh_mapping_parser(subparsers) + add_makesubsh_rmats_parser(subparsers) + add_makesubsh_rmatspost_parser(subparsers) add_exp_matrix_parser(subparsers) + add_makesubsh_extractsj(subparsers) + add_extract_sjc(subparsers) + add_sjc_matrix(subparsers) add_indexing_parser(subparsers) add_translation_parser(subparsers) add_pep2epitope_parser(subparsers) add_screening_plot_parser(subparsers) - add_seq2hla_parser(subparsers) + add_screening_sjc_parser(subparsers) + add_append_sjc_parser(subparsers) + add_annotate_ijc_parser(subparsers) + add_screening_cpm_parser(subparsers) + add_append_cpm_parser(subparsers) + add_screening_novelss_parser(subparsers) + add_screening_sjcplot_parser(subparsers) + add_makesubsh_hla_parser(subparsers) add_parse_hla_parser(subparsers) add_ms_makedb_parser(subparsers) add_ms_search_parser(subparsers) add_ms_parse_parser(subparsers) + add_visual_summary_parser(subparsers) return argparser def add_formatting_parser( subparsers ): - arg_formatting = subparsers.add_parser("formatting", help="Formats AS matrices from rMATS, followed by indexing for IRIS") + arg_formatting = subparsers.add_parser("format", help="Formats AS matrices from rMATS, followed by indexing for IRIS") optional_args = arg_formatting._action_groups.pop() required_args = arg_formatting.add_argument_group('required arguments') - required_args.add_argument('rmats_mat_path_manifest', help='A txt manifest of path(s) to rMATS output folder(s).') - required_args.add_argument('rmats_sample_order', help='A txt manifest of corresponding rMATS input sample order file(s), which is an required input for rMATS.') - required_args.add_argument('-t','--splicing_event_type', choices=['SE','RI','A3','A5'],help='A string of splicing event type based on rMATS defination (SE,RI,A3,A5). Will be used to name the output file name.', required=True) - required_args.add_argument('-n', '--data-name', help='Name of the dataset (disease state, study name, group name etc.). This will be also used during IRIS screening.', required=True) - required_args.add_argument('-s', '--sample-name-field',type=int, choices=[1, 2], help='Specify a field as the sample name field for each sample in the sample order file(s) listed by "rmats_sample_order". 1- use the BAM file name,2- use the BAM folder name. ', required=True) - optional_args.add_argument('-c', '--cov-cutoff', default=10, type=float, help='Average coverage filter for the merged matrix. Defualt is 10.') - optional_args.add_argument('-e', '--merge-events-only', default=False, action="store_true" ,help='Will not perform the matrix merge, only merge the events list.') - optional_args.add_argument('-d', '--iris-db-path', default='.', help='The path to the IRIS database. The formatted/indexed AS matrices will be added to db and used for IRIS screening. Output to "." when the path is not specified.') + required_args.add_argument('rmats_mat_path_manifest', help='txt manifest of path(s) to rMATS output folder(s)') + required_args.add_argument('rmats_sample_order', help='TXT file manifest of corresponding rMATS input sample order file(s). Required input for rMATS') + required_args.add_argument('-t','--splicing-event-type', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file', required=True) + required_args.add_argument('-n', '--data-name', help='Defines dataset name (disease state, study name, group name etc.). Used during IRIS screening ', required=True) + required_args.add_argument('-s', '--sample-name-field',type=int, choices=[1, 2], help='Specifies sample name field (1- SJ count file name, 2- SJ count folder name), for each sample the name should match their name in "rmats_sample_order"', required=True) + optional_args.add_argument('-c', '--cov-cutoff', default=10, type=float, help='Average coverage filter for merged matrix (Default is 10)') + optional_args.add_argument('-i', '--sample-based-filter', default=False, action="store_true" ,help='Coverage filter by individual sample not by entire input group. (Default is disabled)') + optional_args.add_argument('-e', '--merge-events-only', default=False, action="store_true" ,help='Do not perform matrix merge, only merge events list') + optional_args.add_argument('-d', '--iris-db-path', default='.', help='Path to store the formatted/indexed AS matrix. Strongly recommend to store the AS matrix to the IRIS db by setting the path to the directory containing folders of pre-index AS reference ("full_path/IRIS_data.vX/db"). Default is current location.') + optional_args.add_argument('--novelSS', default=False, action="store_true", help='Enable formatting events with splice junctions containing novelSS. (Different and a subset of rMATS novelSS definition. Default is False)') + optional_args.add_argument('--gtf', help='Path to the Genome annotation GTF file. Required input when novelSS is enabled.') arg_formatting._action_groups.append(optional_args) return def add_screening_parser( subparsers ): - arg_screening = subparsers.add_parser("screening", help="Screens AS-derived tumor antigens using big-data reference") + arg_screening = subparsers.add_parser("screen", help="Screens AS-derived tumor antigens using big-data reference") optional_args = arg_screening._action_groups.pop() required_args=arg_screening.add_argument_group('required arguments') - required_args.add_argument('parameter_fin', help='A file of IRIS screening parameters.') - required_args.add_argument('-o', '--outdir', help='Output directory for IRIS screening.') - optional_args.add_argument('-t', '--translating', action= "store_true", help='Translating IRIS screened tumor splice junction into peptides.') + required_args.add_argument('-p', '--parameter-fin', help="File of 'IRIS screen' parameters",required=True) + required_args.add_argument('--splicing-event-type', default='SE', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file. (Default is SE event)') + required_args.add_argument('-o', '--outdir', help='Directory of IRIS screening results', required= True) + optional_args.add_argument('-t', '--translating', action= "store_true", help='Translates IRIS-screened tumor splice junctions into peptides') + optional_args.add_argument('-g', '--gtf', help='The Genome annotation GTF file. Required by IRIS translate option.') + optional_args.add_argument('--all-orf', default=False, action= "store_true", help='Perform the 3 ORF translation. ORF known in the UniProtKB will be labeled as uniprotFrame in the bed file (Default is to use the known ORF ONLY)') + optional_args.add_argument('--ignore-annotation', default=False, action= "store_true", help='Perform 3 ORF translation without annotating known ORF from the UniProtKB (Default is disabled)') + optional_args.add_argument('--remove-early-stop', default=False, action= "store_true", help='Discard the peptide if containing early stop codon (Default is keep the truncated peptide)') + optional_args.add_argument('--min-sample-count', default=False, help='The minimum number of non-missing sample in the input group for an event to be considered for testing. Once specified, removed events will be written to "notest" file. (Default is no minimum)') + optional_args.add_argument('--use-existing-test-result', default=False, action= "store_true", help='Skip testing and use existing testing result (Default is run full testing steps)') arg_screening._action_groups.append(optional_args) return def add_prediction_parser( subparsers ): - arg_prediction = subparsers.add_parser("prediction", help="Predicts and annotates AS-derived TCR (pre-prediction) and CAR-T targets") + arg_prediction = subparsers.add_parser("predict", help="Predicts and annotates AS-derived TCR (pre-prediction) and CAR-T targets") optional_args = arg_prediction._action_groups.pop() required_args = arg_prediction.add_argument_group('required arguments') - required_args.add_argument('IRIS_screening_result_path', help='The same output directory of IRIS screening.') - required_args.add_argument('-p','--parameter-fin', help='The parameter file used in IRIS screening.') - required_args.add_argument('--iedb-local', help='Specify local IEDB location (Needs to be installed).') - optional_args.add_argument('-c','--deltaPSI-column', default=5, help='Column of deltaPSI value in the matrix, 1-based. Default is the 5th column.') - optional_args.add_argument('-d','--deltaPSI-cut-off', default=0, help='Define the cutoff of deltaPSI (or other metric) to be used to select tumor-enriched splice form. Default is 0.') - required_args.add_argument('-m','--mhc-list', help='A list of HLA/MHC types among samples. HLA type follows seq2HLA format.',required=True) - optional_args.add_argument('--extracellular-anno-by-junction', action="store_true", help='The default is to annotate CAR-T target based on if an event is associated with extracellular domain. This option is to annotate target based on a junction (Not recommanded).' ) + required_args.add_argument('IRIS_screening_result_path', help='Directory of IRIS screening results') + required_args.add_argument('--task-dir', help='Directory to write individual task scripts', required=True) + required_args.add_argument('-p','--parameter-fin', help="File of parameters used in 'IRIS screen'",required=True) + required_args.add_argument('-t','--splicing-event-type', default='SE', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file. (Default is SE event)') + optional_args.add_argument('--iedb-local', help='Specify local IEDB location (if installed)') + optional_args.add_argument('-m','--mhc-list', help='List of HLA/MHC types among samples. HLA type follows seq2HLA format') + optional_args.add_argument('--extracellular-only', default=False, action="store_true", help='Only predict CAR-T Targets. Will not predict HLA binding.') + optional_args.add_argument('--tier3-only', default=False, action="store_true", help='To only run predict on events passing all screen tiers, which is the tier3 output. Will be much faster when both the tier1 and tier3 were used.') + optional_args.add_argument('--gene-exp-matrix', default=False, help='Tab-delimited matrix of gene expression vs. samples') + optional_args.add_argument('-c','--deltaPSI-column', default=5, help='Column of deltaPSI value in matrix, 1-based (Default is 5th column)') + optional_args.add_argument('-d','--deltaPSI-cut-off', default=0, help='Defines cutoff of deltaPSI (or other metric) to select tumor-enriched splice form (Default is 0)') + optional_args.add_argument('-e', '--epitope-len-list', default='9,10,11', help='Epitope length for prediction (Default is 9,10,11)') + optional_args.add_argument('--all-orf', default=False, action= "store_true", help='Perform prediction based on 3 ORF translation peptides. Enable this if translation/screening used this option (Default is False)') + optional_args.add_argument('--extracellular-anno-by-junction', action="store_true", help='By default, CAR-T targets are annotated by association of event with extracellular domain. This option annotates target based on a junction (not recommended)' ) arg_prediction._action_groups.append(optional_args) return @@ -159,11 +241,17 @@ def add_epitope_post_parser( subparsers ): arg_epitope_post = subparsers.add_parser("epitope_post", help="Post-prediction step to summarize predicted TCR targets") optional_args = arg_epitope_post._action_groups.pop() required_args = arg_epitope_post.add_argument_group('required arguments') - required_args.add_argument('-p','--parameter_fin', help='The parameter file used in IRIS screening.', required=True) - required_args.add_argument('-o','--outdir', help='The same output directory of IRIS screening.', required=True) - required_args.add_argument('-m','--mhc-by-sample', help='A tsv file of HLA/MHC type vs. samples. HLA type follows seq2HLA format.', required=True) - required_args.add_argument('-e','--gene-exp-matrix', default=False, help='A tsv file of gene expression vs. samples.') - optional_args.add_argument('--ic50-cut-off', default=500, type=float, help='The IC50 cut-off to define HLA-binding epitopes. default is 500.') + required_args.add_argument('-p','--parameter-fin', help='File of parameters used in IRIS screen', required=True) + required_args.add_argument('-o','--outdir', help='Directory of IRIS screening results', required=True) + required_args.add_argument('-t','--splicing-event-type', default='SE', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file (Default is SE event)') + required_args.add_argument('-m','--mhc-by-sample', help=' Tab-delimited matrix of HLA/MHC type vs. samples. HLA type follows seq2HLA format', required=True) + optional_args.add_argument('-e','--gene-exp-matrix', default=False, help='Tab-delimited matrix of gene expression vs. samples') + optional_args.add_argument('--tier3-only', default=False, action="store_true", help='Only predict tier3 events. Will be much faster.') + optional_args.add_argument('--keep-exist', default=False, action="store_true", help='Do not rewrite a new postive prediction file when the file existed. Default is False') + optional_args.add_argument('--epitope-len-list', default='9,10,11', help='Epitope length for prediction (Default is 9,10,11)') + optional_args.add_argument('--no-match-to-canonical-proteome', default=False, action="store_true", help='Matches epitopes to UniProt canonical protein sequences as an annotation.') + optional_args.add_argument('--no-uniqueness-annotation', default=False, action="store_true", help='Matches epitopes to all IRIS translated junction peptides in the same analysis as an annotation.') + optional_args.add_argument('--ic50-cut-off', default=500, type=float, help='Specifies IC50 cut-off to define HLA-binding epitopes (Default is 500)') arg_epitope_post._action_groups.append(optional_args) return @@ -172,58 +260,143 @@ def add_process_rnaseq_parser( subparsers ): optional_args = arg_process_rnaseq._action_groups.pop() required_args = arg_process_rnaseq.add_argument_group('required arguments') required_args.add_argument('--starGenomeDir',help='The path to the STAR indexed reference genome. Pass to the "genomeDir" parameter in STAR', required=True) - required_args.add_argument('--gtf',help='Genome annotation file.', required=True) - required_args.add_argument('-p','--sampleID-outdir', help='Output directory where sample ID will be used as the output folder name.', required=True) - required_args.add_argument('--db-length',default=100, help='Pass to the "sjdbOverhang" parameter in STAR. Default is 100.') + required_args.add_argument('--gtf',help='Path to the Genome annotation GTF file', required=True) + required_args.add_argument('-p','--sampleID-outdir', help='Output directory where sample ID will be used as the output folder name', required=True) + required_args.add_argument('--db-length',default=100, help='Pass to the "sjdbOverhang" parameter in STAR. Default is 100') required_args.add_argument('readsFilesRNA',help='Specify the path to the paired-end FASTQ files for the sample. Files are seperated eperated by ",".') - optional_args.add_argument('--mapping',help= 'Only perform reads mapping.', action='store_true') + optional_args.add_argument('--mapping',help= 'Only perform reads mapping', action='store_true') optional_args.add_argument('--quant',help='Only perform gene expression and AS quantification', action='store_true') - optional_args.add_argument('--sort',help='Only perform BAM file sorting.',action='store_true') + optional_args.add_argument('--sort',help='Only perform BAM file sorting',action='store_true') arg_process_rnaseq._action_groups.append(optional_args) return -def add_rmats_prep_parser(subparsers): - arg_rmats_prep = subparsers.add_parser("makeqsub_rmats", help="Makes qsub files for running rMATS-turbo 'prep' step") - optional_args = arg_rmats_prep._action_groups.pop() - required_args = arg_rmats_prep.add_argument_group('required arguments') - required_args.add_argument('--rMATS-path',help= 'Path to rMATS-turbo script.', required=True) +def add_makesubsh_mapping_parser(subparsers): + arg_makesubsh_mapping = subparsers.add_parser("makesubsh_mapping", help="Makes submission shell scripts for running 'process_rnaseq'") + optional_args = arg_makesubsh_mapping._action_groups.pop() + required_args = arg_makesubsh_mapping.add_argument_group('required arguments') + required_args.add_argument('--fastq-folder-dir',help='Specify the path to the higher level of all folders containing FASTQ files') + required_args.add_argument('--starGenomeDir',help='The path to the STAR indexed reference genome. Pass to the "genomeDir" parameter in STAR', required=True) + required_args.add_argument('--gtf',help='Path to the Genome annotation GTF file', required=True) + required_args.add_argument('--data-name',help='Data set name used to name submission shell scripts files.', required=True) + required_args.add_argument('--outdir',help='Output directory for folders of aligned BAM files', required=True) + required_args.add_argument('--label-string', help='String in the fastq file name between the reads pair number and "fastq/fq". This is used to recognize paired-end reads. e.g. For FASTQ_file_L1_R2.fastq.gz, the label string is the "." between "2" and "fastq".', required=True) + required_args.add_argument('--task-dir', help='Directory to write individual task scripts', required=True) + arg_makesubsh_mapping._action_groups.append(optional_args) + +def add_makesubsh_rmats_parser(subparsers): + arg_makesubsh_rmats = subparsers.add_parser("makesubsh_rmats", help="Makes submission shell scripts for running rMATS-turbo 'prep' step") + optional_args = arg_makesubsh_rmats._action_groups.pop() + required_args = arg_makesubsh_rmats.add_argument_group('required arguments') + required_args.add_argument('--rMATS-path',help= 'Path to the rMATS-turbo script.', required=True) + required_args.add_argument('--bam-dir',help='The path one level higher to folders containing BAM file generated by "process_rnaseq".', required=True) + required_args.add_argument('--bam-prefix', default='Aligned.sortedByCoord.out', help='BAM file prefix (Default is "Aligned.sortedByCoord.out")') + required_args.add_argument('--gtf',help='Path to the Genome annotation GTF file', required=True) + required_args.add_argument('--data-name',help='Data set name used to name submission shell scripts', required=True) + required_args.add_argument('--task-dir', help='Directory to write individual task scripts', required=True) + optional_args.add_argument('--novelSS',default=False, action= "store_true", help='Enable rMATS novelSS option to include novel splice site detected from the RNA-seq data (Default is False)') + optional_args.add_argument('--read-length',default=False, help='User defined read length instead of using STAR maaping log file to define automatically.') + arg_makesubsh_rmats._action_groups.append(optional_args) + return + +def add_makesubsh_rmatspost_parser(subparsers): + arg_makesubsh_rmatspost = subparsers.add_parser("makesubsh_rmatspost", help="Makes submission shell scripts for running rMATS-turbo 'post' step") + optional_args = arg_makesubsh_rmatspost._action_groups.pop() + required_args = arg_makesubsh_rmatspost.add_argument_group('required arguments') + required_args.add_argument('--rMATS-path',help= 'Path to the rMATS-turbo scripte', required=True) required_args.add_argument('--bam-dir',help='The path one level higher to folders containing BAM file generated by "process_rnaseq".', required=True) - required_args.add_argument('--gtf',help='Genome annotation file.', required=True) - required_args.add_argument('--read-length',help='Pass to the "readLength" parameter in rMATS-turbo.', required=True) - arg_rmats_prep._action_groups.append(optional_args) + required_args.add_argument('--gtf',help='Path to the Genome annotation GTF file', required=True) + required_args.add_argument('--data-name',help='Data set name used to name submission shell scripts', required=True) + optional_args.add_argument('--novelSS',default=False, action= "store_true", help='Enable rMATS novelSS option to include novel splice site detected from the RNA-seq data (Default is False)') + required_args.add_argument('--task-dir', help='Directory to write individual task scripts', required=True) + arg_makesubsh_rmatspost._action_groups.append(optional_args) return -def add_exp_matrix_parser(subparsers): +def add_exp_matrix_parser(subparsers): arg_exp_matrix = subparsers.add_parser("exp_matrix", help="Makes a merged gene expression matrix from multiple cufflinks results") optional_args = arg_exp_matrix._action_groups.pop() required_args = arg_exp_matrix.add_argument_group('required arguments') required_args.add_argument('gene_exp_file_list', help='A txt manifest of path(s) of cufflinks gene expression output(s).') - optional_args.add_argument('--exp-cutoff', default=1, help='Gene expression cut-off based on FPKM. Default is 1.') + optional_args.add_argument('--exp-cutoff', default=1, help='Gene expression cut-off based on FPKM (Default is 1)') optional_args.add_argument('-o','--outdir', default='.',help='Output directory for IRIS exp_matrix', required=True) required_args.add_argument('-n', '--data-name', help='Name of the dataset (disease state, study name, group name etc.).', required=True) arg_exp_matrix._action_groups.append(optional_args) return + +def add_makesubsh_extractsj( subparsers ): + arg_makesubsh_extractsj = subparsers.add_parser('makesubsh_extract_sjc',help="Makes submission shell scripts for running 'extract_sjc'") + optional_args = arg_makesubsh_extractsj._action_groups.pop() + required_args = arg_makesubsh_extractsj.add_argument_group('required arguments') + required_args.add_argument('-b','--bam-folder-list', help='Path to a file listing all paths to BAM folders', required=True) + required_args.add_argument('-g', '--gtf', help='Path to the Genome annotation GTF file', required=True) + required_args.add_argument('-f', '--genome-fasta', help='Path to the reference genome FASTA file', required=True) + required_args.add_argument('-n', '--task-name', help='The task name. Used to name the command file and the bash file', required=True) + required_args.add_argument('--BAM-prefix', default='Aligned.sortedByCoord.out', help='BAM file prefix', required=True) + optional_args.add_argument( + '-r','--rmats-used-read-length', default='', + help=('if not given, the read length will be parsed from' + ' Log.final.out in the bam folder')) + optional_args.add_argument( + '--task-dir', + help=('the directory to write the command and bash file.' + ' Defaults to the working directory')) + arg_makesubsh_extractsj._action_groups.append(optional_args) + return + +def add_extract_sjc( subparsers ): + arg_extract_sjc = subparsers.add_parser('extract_sjc',help='Extracts SJ counts from STAR-aligned BAM file and annotates SJs with number of uniquely mapped reads that support the splice junction.') + optional_args = arg_extract_sjc._action_groups.pop() + required_args = arg_extract_sjc.add_argument_group('required arguments') + required_args.add_argument('-i','--bam-path', help='Path to BAM files', required=True) + required_args.add_argument('-f', '--genome-fasta', help='Path to the reference genome FASTA file', required=True) + required_args.add_argument('-g', '--gtf', help='Path to the Genome annotation GTF file', required=True) + required_args.add_argument('-a','--minimum-overhang-length-annotated', default=1) + required_args.add_argument('-c','--minimum-overhang-length-unannotated-canonical', default=8) + required_args.add_argument('-u','--minimum-overhang-length-unannotated-noncanonical', default=10) + required_args.add_argument('-o', '--outdir', help='', required=True) + optional_args.add_argument('-r','--read-length', help='length of reads to keep when counting junction reads') + arg_extract_sjc._action_groups.append(optional_args) + return + +def add_sjc_matrix ( subparsers ): + arg_sjc_matrix = subparsers.add_parser('sjc_matrix',help='Makes SJ count matrix by merging SJ count files from a specified list of samples. Performs indexing of the merged file.') + optional_args = arg_sjc_matrix._action_groups.pop() + required_args = arg_sjc_matrix.add_argument_group('required arguments') + required_args.add_argument('-i','--file-list-input', help='Path to the file contains a list of SJ count files.', required=True) + required_args.add_argument('-n', '--data-name', help='Defines dataset name (disease state, study name, group name etc.). Used during IRIS screening ', required=True) + required_args.add_argument('-s', '--sample-name-field',type=int, choices=[1, 2], help='Specifies sample name field (1- SJ count file name, 2- SJ count folder name), for each sample the name should match their name in "rmats_sample_order"', required=True) + optional_args.add_argument('-d', '--iris-db-path', default='.', help='Path to IRIS database. Formatted/indexed AS matrices are stored here and used during IRIS screening') + arg_sjc_matrix._action_groups.append(optional_args) + return + def add_indexing_parser( subparsers ): - arg_indexing = subparsers.add_parser("indexing", help="Indexes AS matrices for IRIS") + arg_indexing = subparsers.add_parser("index", help="Indexes AS matrices for IRIS") optional_args = arg_indexing._action_groups.pop() required_args = arg_indexing.add_argument_group('required arguments') - required_args.add_argument('splicing_matrix', help='A tab-delimited matrix of splicing events (row) vs. sample IDs (col).') - required_args.add_argument('-n', '--data-name', help='Name of the dataset (disease state, study name, group name etc.). This will be also used during IRIS screening.', required=True) - optional_args.add_argument('-d', '--db-dir', default='.', help='The directory of the IRIS database. The program will create folder inside this directory in order to make IRIS recognize.') + required_args.add_argument('splicing_matrix', help='Tab-delimited matrix of splicing events (row) vs. sample IDs (col)') + required_args.add_argument('-t','--splicing-event-type', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file', required=True) + required_args.add_argument('-n', '--data-name', help='Name of data matrix (disease state, study name, group name, etc.) being indexed. Used by IRIS during screening', required=True) + optional_args.add_argument('-c', '--cov-cutoff', default=10, type=float, help='For the naming purpose, Input average coverage cutoff used when generating the PSI matrix (Default is 10)') + optional_args.add_argument('-o', '--outdir', default='.', help='Output directory for IRIS database') arg_indexing._action_groups.append(optional_args) return def add_translation_parser( subparsers ): - arg_translation = subparsers.add_parser("translation", help="Translates AS junctions into junction peptides") + arg_translation = subparsers.add_parser("translate", help="Translates AS junctions into junction peptides") optional_args = arg_translation._action_groups.pop() required_args = arg_translation.add_argument_group('required arguments') - required_args.add_argument('as_input', help='A tsv file generated by IRIS screening, containing AS events and deltaPSI value.') - required_args.add_argument('-g','--ref-genome', help='The path to the reference genome file (FASTA).', required=True) - required_args.add_argument('-o','--outdir', help='Output directory for IRIS translation.',required=True ) - optional_args.add_argument('-c','--deltaPSI-column', default=5, help='Column of deltaPSI value in the matrix, 1-based. Default is the 5th column.') - optional_args.add_argument('-d','--deltaPSI-cut-off', default=0, help='Cutoff of deltaPSI (or other metric) to be used to select tumor-enriched splice form. Default is 0.') - optional_args.add_argument('--no-tumor-form-selection', action= "store_true", help='Splicing junctions derived from both skipping and inclusion forms are translated.') + required_args.add_argument('as_input', help='Inputs AS event coordinates and delta PSI values') + required_args.add_argument('-g','--ref-genome', help='Specifies reference genome (FASTA format) location', required=True) + required_args.add_argument('-t','--splicing-event-type', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file', required=True) + required_args.add_argument('--gtf',help='Path to the Genome annotation GTF file. Used to define exon ends for microexons', required=True) + required_args.add_argument('-o','--outdir', help='Defines IRIS translation output directory',required=True ) + optional_args.add_argument('--all-orf', default=False, action= "store_true", help='Perform the 3 ORF translation. ORF known in the UniProtKB will be labeled as uniprotFrame in the bed file (Default is to use the known ORF ONLY)') + optional_args.add_argument('--ignore-annotation', default=False, action= "store_true", help='Perform 3 ORF translation without annotating known ORF from the UniProtKB (Default is disabled)') + optional_args.add_argument('--remove-early-stop', default=False, action= "store_true", help='Discard the peptide if containing early stop codon (Default is keep the truncated peptide)') + optional_args.add_argument('-c','--deltaPSI-column', default=5, help='Column of deltaPSI value in matrix, 1-based (Default is 5th column)') + optional_args.add_argument('-d','--deltaPSI-cut-off', default=0, help='Defines cutoff of deltaPSI (or other metric) used to select tumor-enriched splice form (Default is 0)') + optional_args.add_argument('--no-tumor-form-selection', action= "store_true", help='Translates splicing junctions derived from both skipping and inclusion forms (Default is False)') + optional_args.add_argument('--check-novel', action= "store_true", help='Translates splicing junctions derived from novel splice sites only using information passed from screen_novelss (Default is False)', default=False) arg_translation._action_groups.append(optional_args) return @@ -231,23 +404,25 @@ def add_pep2epitope_parser( subparsers ): arg_pep2epitope = subparsers.add_parser("pep2epitope", help="Wrapper to run IEDB for peptide-HLA binding prediction") optional_args = arg_pep2epitope._action_groups.pop() required_args = arg_pep2epitope.add_argument_group('required arguments') - required_args.add_argument('junction_pep_input', help='input alternative splicing events coordinates and PSI value.') - required_args.add_argument('-e', '--epitope-len-list', default='9,10,11', help='epitope length for prediction. Default is 9,10,11.') - required_args.add_argument('-a', '--hla-allele-list', default='HLA-A*01:01,HLA-B*08:01,HLA-C*07:01', help='a list of HLA types. Default is HLA-A*01:01,HLA-B*08:01,HLA-C*07:01.') - required_args.add_argument('-o', '--outdir', help='Define the output directory of pep2epitope.', required=True) - required_args.add_argument('--iedb-local', help='Specify local IEDB location if it is installed.') - required_args.add_argument('--ic50-cut-off', default=500, help='Cut-off based on median value of concensus predicted IC50 values. Default is 500.') + required_args.add_argument('junction_pep_input', help='Inputs junction peptides') + required_args.add_argument('-e', '--epitope-len-list', default='9,10,11', help='Epitope length for prediction (Default is 9,10,11)') + required_args.add_argument('-a', '--hla-allele-list', default='HLA-A*01:01,HLA-B*08:01,HLA-C*07:01', help='List of HLA types (Default is HLA-A*01:01, HLA-B*08:01, HLA-C*07:01)') + required_args.add_argument('-o', '--outdir', help='Define output directory of pep2epitope', required=True) + required_args.add_argument('--iedb-local', help='Specify local IEDB location (if installed)') + required_args.add_argument('--ic50-cut-off', default=500, help='Cut-off based on median value of consensus-predicted IC50 values (Default is 500)') arg_pep2epitope._action_groups.append(optional_args) return -def add_seq2hla_parser(subparsers): - arg_seq2hla = subparsers.add_parser("seq2hla",help='Wrapper to run seq2HLA for HLA typing using RNA-Seq') - optional_args = arg_seq2hla._action_groups.pop() - required_args = arg_seq2hla.add_argument_group('required arguments') - required_args.add_argument('-b','--seq2hla-path', help='Path to seq2hla folder.', required=True) - required_args.add_argument('-p','--sampleID-outdir', help='Output directory where sample ID will be used as the output folder name.', required=True) - required_args.add_argument('readsFilesCaseRNA',help='Tumor sample paired-end fastq files seperated by ",". ') - arg_seq2hla._action_groups.append(optional_args) +def add_makesubsh_hla_parser(subparsers): + arg_makesubsh_hla = subparsers.add_parser("makesubsh_hla",help='Makes submission shell scripts for running seq2HLA for HLA typing using RNA-Seq') + optional_args = arg_makesubsh_hla._action_groups.pop() + required_args = arg_makesubsh_hla.add_argument_group('required arguments') + required_args.add_argument('--fastq-folder-dir',help='Specify the path to the higher level of all folders containing FASTQ files') + required_args.add_argument('--data-name',help='Data set name used to name submission shell scripts.', required=True) + required_args.add_argument('-o','--outdir',help='Output directory for folders of seq2hla result', required=True) + required_args.add_argument('--label-string', help='String in the fastq file name between the reads pair number and "fastq/fq". This is used to recognize paired-end reads. e.g. For FASTQ_file_L1_R2.fastq.gz, the label string is the "." between "2" and "fastq".', required=True) + required_args.add_argument('--task-dir', help='Directory to write individual task scripts', required=True) + arg_makesubsh_hla._action_groups.append(optional_args) return def add_parse_hla_parser(subparsers): @@ -259,20 +434,124 @@ def add_parse_hla_parser(subparsers): return def add_screening_plot_parser(subparsers): - arg_screening_plot = subparsers.add_parser("screening_plot",help='Makes stacked/individual violin plots for list of AS events') + arg_screening_plot = subparsers.add_parser("screen_plot",help='Makes stacked/individual violin plots for list of AS events') optional_args = arg_screening_plot._action_groups.pop() required_args = arg_screening_plot.add_argument_group('required arguments') - required_args.add_argument('event_list', help='input alternative splicing events coordinates for visualization.') - required_args.add_argument('-p','--parameter-fin', help='The file of parameters used in IRIS screening.', required=True) - required_args.add_argument('--step','-s', default=10, help='number of events in each plot.') - optional_args.add_argument('--header', action="store_true", help='Skipping the header line in the input event list.') + required_args.add_argument('event_list', help='Inputs AS event coordinates for visualization') + required_args.add_argument('-p','--parameter-file', help="Parameter file for 'IRIS screen'", required=True) + required_args.add_argument('-t', '--splicing-event-type', default='SE', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file (Default is SE event)') + required_args.add_argument('--step','-s', default=10, help='Number of events in each plot (Default is 10)') + required_args.add_argument('-o', '--outdir', help='Define the output directory of the plot.', required=True) + optional_args.add_argument('--header', action="store_true", default=False, help='Skipping the header line of the input (Default is False)') arg_screening_plot._action_groups.append(optional_args) return +def add_screening_sjc_parser( subparsers ): + arg_screening_sjc = subparsers.add_parser("screen_sjc", help="Screens AS-derived tumor antigens by comparing number of samples expressing a splice junction using big-data reference of SJ counts") + optional_args = arg_screening_sjc._action_groups.pop() + required_args= arg_screening_sjc.add_argument_group('required arguments') + required_args.add_argument('-p','--parameter-file', help='Parameter file containing SJ db directory, selected data sets, etc.', required=True) + required_args.add_argument('--splicing-event-type', default='SE', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file. (Default is SE event)') + required_args.add_argument('-e','--event-list-file', help='AS event list in the format of PSI value matrices (see the output format of IRIS format module or IRIS_db PSI matrices)') + required_args.add_argument('-o', '--outdir', help='Directory of IRIS screening results', required= True) + optional_args.add_argument('--use-existing-test-result', default=False, action= "store_true", help='Skip testing and use existing testing result (Default is run full testing steps)') + optional_args.add_argument('--tumor-read-cov-cutoff', default=5, type=int, help='Minimum read coverage for a tumor sample to be considered as expressing the junction (Default is 5)') + optional_args.add_argument('--normal-read-cov-cutoff', default=2, type=int, help='Minimum read coverage for a normal sample to be considered as expressing the junction (Default is 2)') + arg_screening_sjc._action_groups.append(optional_args) + return + +def add_append_sjc_parser( subparsers ): + arg_append_sjc = subparsers.add_parser("append_sjc", help="Appends SJC result as an annotation to PSI-based screening results and epitope prediction results in a specified screening output folder.") + optional_args = arg_append_sjc._action_groups.pop() + required_args= arg_append_sjc.add_argument_group('required arguments') + required_args.add_argument('--sjc-summary',help='Full path to the \"summary\" file from the SJC screening output',required=True) + required_args.add_argument('--splicing-event-type', default='SE', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file. (Default is SE event)') + required_args.add_argument('-o', '--outdir', help='Directory of IRIS screening results', required= True) + optional_args.add_argument('-i', '--add-ijc-info', action='store_true', default=False, help='Add inclusion junction related annotation to the PSI-based screening results and epitope prediction results. This can be slow when \'--screening-result-event-list\' is large. (Default is False)') + optional_args.add_argument('-u','--use-existing-result', default=False, action= "store_true", help='Skip retrieving and use existing ijc result (Default is False)') + optional_args.add_argument('-p','--parameter-file', help='Parameter file. This is required when \'--add-ijc-info\' is enabled', default='') + optional_args.add_argument('-e','--screening-result-event-list', help='A list of AS events of interest in the same format as the \'as_event\' column in the IRIS screen output. This is required when \'--add-ijc-info\' is enabled', default='') + optional_args.add_argument('--inc-read-cov-cutoff', default=2, type=int, help='Minimum read coverage for the two inclusion junctions combined to be considered as expressing. This is a parameter for annotate_ijc (Default is 2)') + optional_args.add_argument('--event-read-cov-cutoff', default=10, type=int, help='Minimum read coverage for an event to be considered in the analysis. This is a parameter for annotate_ijc (Default is 10)') + arg_append_sjc._action_groups.append(optional_args) + return + +def add_annotate_ijc_parser( subparsers ): + arg_annotate_ijc = subparsers.add_parser("annotate_ijc", help="Annotates inclusion junction count info to PSI-based screening results or epitope prediction results in a specified screening output folder. Can be called from append sjc to save time.") + optional_args = arg_annotate_ijc._action_groups.pop() + required_args= arg_annotate_ijc.add_argument_group('required arguments') + required_args.add_argument('-p','--parameter-file', help='Parameter file containing SJ db directory, selected data sets, etc.', required=True) + required_args.add_argument('--splicing-event-type', default='SE', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file. (Default is SE event)') + required_args.add_argument('-e','--screening-result-event-list', help='A list of AS events of interest in the same format as the \'as_event\' column in the IRIS screen output', required=True) + optional_args.add_argument('--inc-read-cov-cutoff', default=2, type=int, help='Minimum read coverage for the two inclusion junctions combined to be considered as expressing (Default is 2)') + optional_args.add_argument('--event-read-cov-cutoff', default=10, type=int, help='Minimum read coverage for an event to be considered in the analysis (Default is 10)') + required_args.add_argument('-o', '--outdir', help='Directory of IRIS screening results', required= True) + arg_annotate_ijc._action_groups.append(optional_args) + return + +def add_screening_cpm_parser( subparsers ): + arg_screening_cpm = subparsers.add_parser("screen_cpm", help="Screens AS-derived tumor antigens by comparing splice junction CPM using big-data reference of SJ counts") + optional_args = arg_screening_cpm._action_groups.pop() + required_args= arg_screening_cpm.add_argument_group('required arguments') + required_args.add_argument('-p','--parameter-file', help='Parameter file containing SJ db directory, selected data sets, etc.', required=True) + required_args.add_argument('--splicing-event-type', default='SE', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file. (Default is SE event)') + required_args.add_argument('-e','--event-list-file', help='AS event list in the format of PSI value matrices (see the output format of IRIS format module or IRIS_db PSI matrices)') + required_args.add_argument('-o', '--outdir', help='Directory of IRIS screening results', required= True) + optional_args.add_argument('--use-existing-test-result', default=False, action= "store_true", help='Skip testing and use existing testing result (Default is run full testing steps)') + arg_screening_cpm._action_groups.append(optional_args) + return + +def add_append_cpm_parser( subparsers ): + arg_append_cpm = subparsers.add_parser("append_cpm", help="Appends CPM result as an annotation to PSI-based screening results and epitope prediction results in a specified screening output folder.") + optional_args = arg_append_cpm._action_groups.pop() + required_args= arg_append_cpm.add_argument_group('required arguments') + required_args.add_argument('--cpm-summary',help='Full path to the \"summary\" file from the CPM screening output',required=True) + required_args.add_argument('--splicing-event-type', default='SE', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file. (Default is SE event)') + required_args.add_argument('-o', '--outdir', help='Directory of IRIS screening results', required= True) + arg_append_cpm._action_groups.append(optional_args) + return + +def add_screening_novelss_parser( subparsers ): + arg_screening_novelss = subparsers.add_parser("screen_novelss", help="Screens AS-derived tumor antigens for unannotated events using big-data reference of SJ counts") + optional_args = arg_screening_novelss._action_groups.pop() + required_args= arg_screening_novelss.add_argument_group('required arguments') + required_args.add_argument('-p','--parameter-fin', help='Parameter file containing SJ db directory, selected data sets, etc.', required=True) + required_args.add_argument('--splicing-event-type', default='SE', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file. (Default is SE event)') + required_args.add_argument('-e','--event-list-fin', help='AS Event list in the format of PSI value matrices (modifed rMATS format)') + required_args.add_argument('-o', '--outdir', help='Directory of IRIS screening results', required= True) + optional_args.add_argument('--gtf', help='Path to the Genome annotation GTF file. Required input when checking novelSS for tumor junctions and IRIS translate option.') + optional_args.add_argument('-d','--deltaPSI-cut-off', default=0, help='Defines cutoff of deltaPSI (or other metric) used to select tumor-enriched splice form (Default is 0)') + optional_args.add_argument('--use-existing-test-result', default=False, action= "store_true", help='Skip testing and use existing testing result (Default is run full testing steps)') + optional_args.add_argument('-t', '--translating', action= "store_true", help='Translates IRIS-screened tumor splice junctions into peptides') + optional_args.add_argument('--report-known-and-novelss-tumor-junction', default=False, action= "store_true", help='Report both known and novel splice site-derived tumor junctions. Despite the input is AS events from rMATS with novel splice sites detected, not every junctions in an AS event will contain novel splice site(s). By default, an event will be reported when tumor-form junction derived from novel splice site(s). (Default is False)') + optional_args.add_argument('--all-orf', default=False, action= "store_true", help='Perform the 3 ORF translation. ORF known in the UniProtKB will be labeled as uniprotFrame in the bed file (Default is to use the known ORF ONLY)') + optional_args.add_argument('--ignore-annotation', default=False, action= "store_true", help='Perform 3 ORF translation without annotating known ORF from the UniProtKB (Default is disabled)') + optional_args.add_argument('--remove-early-stop', default=False, action= "store_true", help='Discard the peptide if containing early stop codon (Default is keep the truncated peptide)') + arg_screening_novelss._action_groups.append(optional_args) + return + +def add_screening_sjcplot_parser( subparsers ): + arg_screening_sjcplot = subparsers.add_parser("screen_sjc_plot",help='Makes stacked/individual barplots of percentage of samples expressing a splice junction for list of AS events') + optional_args = arg_screening_sjcplot._action_groups.pop() + required_args = arg_screening_sjcplot.add_argument_group('required arguments') + required_args.add_argument('event_list', help='Inputs a list of AS event and direction for visualization. IRIS screening result format is preferred (Default deltaPSI column and cutoff are based on IRIS screening format)') + required_args.add_argument('-j','--jc-full-result', help='File contains information about percentage of samples expressing a SJ from the output of IRIS SJC screening.') + required_args.add_argument('-p','--parameter-fin', help="Parameter file used in 'IRIS screen_sjc' (using SJ db)", required=True) + required_args.add_argument('-t', '--splicing-event-type', default='SE', choices=['SE','RI','A3SS','A5SS'],help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file (Default is SE event)') + required_args.add_argument('--step','-s', default=10, help='Number of events in each plot (Default is 10)') + required_args.add_argument('-o', '--outdir', help='Define the output directory of the plot.', required=True) + optional_args.add_argument('-c','--deltaPSI-column', default=5, help='Column of deltaPSI value in matrix, 1-based (Default is 5th column)') + optional_args.add_argument('-d','--deltaPSI-cut-off', default=0, help='Defines cutoff of deltaPSI (or other metric) to select tumor-enriched splice form (Default is 0)') + optional_args.add_argument('--header', action="store_true", default=False, help='Skipping the header line of the input (Default is False)') + arg_screening_sjcplot._action_groups.append(optional_args) + return + def add_ms_makedb_parser(subparsers): arg_ms_makedb = subparsers.add_parser("ms_makedb",help='Generates proteo-transcriptomic database for MS search') optional_args = arg_ms_makedb._action_groups.pop() required_args = arg_ms_makedb.add_argument_group('required arguments') + required_args.add_argument('--java-path', help='The path of Java.') + required_args.add_argument('--MSGF-path', help='The path of MSGF+.') required_args.add_argument('-o', '--outdir', help='The path to IRIS traslation output directory.', required=True) required_args.add_argument('--uniprot-fasta',help='Specify the path of the UniProt proteome FASTA file.', required= True) required_args.add_argument('--exp-fin-list',help='Specify a file contains paths of gene expression files (by rows) that should be considered to form the proteogenomic db.', required= True) @@ -301,11 +580,23 @@ def add_ms_parse_parser(subparsers): required_args.add_argument('-o', '--outdir', help='Specify a directory to output parsed MS result.', required=True) optional_args.add_argument('--dump-all', action= "store_true", help='') arg_ms_parse._action_groups.append(optional_args) - return + return + +def add_visual_summary_parser(subparsers): + arg_visual_summary = subparsers.add_parser("visual_summary",help='Makes a graphic summary of IRIS results') + optional_args = arg_visual_summary._action_groups.pop() + required_args = arg_visual_summary.add_argument_group('required arguments') + required_args.add_argument('-p','--parameter-fin', help="Parameter file used in 'IRIS screen'", required=True) + required_args.add_argument('-s', '--screening-out-dir', help='The directory where IRIS screening output was written', required=True) + required_args.add_argument('-o', '--out-file-name', help='The .png file name to write', required=True) + required_args.add_argument('-t', '--splicing-event-type', default='SE', choices=['SE','RI','A3SS','A5SS'], help='String of splicing event types based on rMATS definition (SE,RI,A3SS,A5SS).Used to name output file (Default is SE event)') + optional_args.add_argument('--no-prediction', action='store_true', required=False) + arg_visual_summary._action_groups.append(optional_args) + return if __name__ == '__main__': try: main() except KeyboardInterrupt: sys.stderr.write("[INFO] User interrupted; program terminated.") - sys.exit(0) \ No newline at end of file + sys.exit(0) diff --git a/conda.sh b/conda.sh deleted file mode 100644 index 1bae2fa..0000000 --- a/conda.sh +++ /dev/null @@ -1,57 +0,0 @@ -#!/bin/bash -# -# Provide functions for working with conda -# Usage: -# * `source conda.sh` -# * `conda::create_env_with_name_and_python_version {env_name} {python_version}` -# + example: `conda::create_env_with_name_and_python_version my-conda-env 3.6` -# * `conda::activate_env {env_name}` -# + example: `conda::activate_env my-conda-env` -# * `conda::deactivate_env` -# -# Assumes that conda is installed -# https://docs.conda.io/en/latest/miniconda.html -# https://repo.anaconda.com/miniconda/Miniconda2-latest-Linux-x86_64.sh - -function conda::create_env_with_name_and_python_version() { - local ERROR_PREFIX="error in conda::create_env_with_name_and_python_version()" - - local ENV_NAME="$1" - local PYTHON_VERSION="$2" - - local FOUND_COUNT=$(conda info --envs | grep "^${ENV_NAME} .*/${ENV_NAME}$" | wc -l) - if [[ "$?" -ne 0 ]]; then - echo "${ERROR_PREFIX}: checking conda envs" >&2 - return 1 - fi - - if [[ "${FOUND_COUNT}" -eq 1 ]]; then - echo "using existing ${ENV_NAME} conda environment" - return 0 - fi - - echo "creating new conda environment: ${ENV_NAME} python=${PYTHON_VERSION}" - conda create --name "${ENV_NAME}" python="${PYTHON_VERSION}" - if [[ "$?" -ne 0 ]]; then - echo "${ERROR_PREFIX}: creating env" >&2 - return 1 - fi -} -export -f conda::create_env_with_name_and_python_version - -function conda::activate_env() { - conda activate "$1" -} -export -f conda::activate_env - -function conda::deactivate_env() { - conda deactivate -} -export -f conda::deactivate_env - -function main() { - # need to use the setup that conda init writes to .bashrc - source ${HOME}/.bashrc || return 1 -} - -main "$@" diff --git a/conda_requirements_py2.txt b/conda_requirements_py2.txt new file mode 100644 index 0000000..5d650c9 --- /dev/null +++ b/conda_requirements_py2.txt @@ -0,0 +1,7 @@ +bedtools=2.29.0 +numpy=1.16.5 +pybigwig=0.3.13 +python=2.7.* +scipy=1.2.0 +seaborn=0.9.0 +statsmodels=0.10.2 diff --git a/conda_requirements_py2_optional.txt b/conda_requirements_py2_optional.txt new file mode 100644 index 0000000..e9afb14 --- /dev/null +++ b/conda_requirements_py2_optional.txt @@ -0,0 +1,10 @@ +cufflinks=2.2.1 +# openssl 1.0 needs to be explicitly installed because pysam depends on it, +# but openssl 1.1 would get installed otherwise +openssl=1.0.2u +pysam=0.14.1 +r-base=3.2.2 +rmats=4.1.2 +samtools=1.3 +seq2hla=2.2 +star=2.5.3a diff --git a/conda_requirements_py3.txt b/conda_requirements_py3.txt new file mode 100644 index 0000000..eab4d43 --- /dev/null +++ b/conda_requirements_py3.txt @@ -0,0 +1,5 @@ +beautifulsoup4=4.11.* +google-api-python-client=2.* +python=3.9.* +snakemake=7.17.1 +tqdm=4.64.* diff --git a/conda_wrapper b/conda_wrapper new file mode 100755 index 0000000..e64f439 --- /dev/null +++ b/conda_wrapper @@ -0,0 +1,30 @@ +#!/bin/bash +# conda_wrapper activates the conda environment and then +# executes its arguments in that environment. +# +function set_script_dir() { + local ORIG_DIR="$(pwd)" || return 1 + + local REL_SCRIPT_DIR="$(dirname ${BASH_SOURCE[0]})" || return 1 + cd "${REL_SCRIPT_DIR}" || return 1 + SCRIPT_DIR="$(pwd)" || return 1 + cd "${ORIG_DIR}" || return 1 +} + +function main() { + local CONDA_ENV_PREFIX="$1" + shift || return 1 + + set_script_dir || return 1 + source "${SCRIPT_DIR}/set_env_vars.sh" || return 1 + + conda activate "${CONDA_ENV_PREFIX}" || return 1 + + "$@" + local RETURN_VALUE="$?" + + conda deactivate || return 1 + return "${RETURN_VALUE}" +} + +main "$@" diff --git a/docs/iris_diagram.png b/docs/iris_diagram.png new file mode 100644 index 0000000000000000000000000000000000000000..8148bf8149f8266b16afda8ded263703260c484c GIT binary patch literal 120132 zcma&OWmp_t6D>*vLhulRdk7vNxLa^%aCdii2@>2bFi5Zf8Qfiiy9_qCyStv|{qFsH zp7XC~rg!(2+O=1$S~Xz`a^k2+_(*VYaHx_JA|N=pH>z-OuT>CV1K)ra≠HUU6G| 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 Iunq+N12_TC5C8xG diff --git a/example/SJ_matrices/RL_100.matrix/JC.raw.input.SE.txt b/example/SJ_matrices/RL_100.matrix/JC.raw.input.SE.txt deleted file mode 100644 index c008df7..0000000 --- a/example/SJ_matrices/RL_100.matrix/JC.raw.input.SE.txt +++ /dev/null @@ -1,13 +0,0 @@ -ID IJC_SAMPLE_1 SJC_SAMPLE_1 IJC_SAMPLE_2 SJC_SAMPLE_2 IncFormLen SkipFormLen -15225 20,22,22,32,27,50 42,24,44,39,47,52 187 99 -34517 54,84,63,84,88,57 1,11,1,2,0,0 198 99 -51809 157,56,239,139,132,105 4,4,2,6,1,6 188 99 -58989 205,202,213,230,423,75 13,11,19,13,9,17 198 99 -64939 36,51,21,51,73,8 1,2,3,0,16,0 183 99 -64969 128,48,109,60,37,139 19,5,22,18,6,29 131 99 -68704 65,31,98,48,37,68 14,5,10,18,8,10 198 99 -75710 27,60,84,42,46,40 0,3,1,2,1,3 198 99 -76672 21,16,43,11,27,34 42,25,71,28,16,59 156 99 -77359 2,1,0,1,0,4 90,68,71,70,153,47 198 99 -86641 189,623,470,565,684,386 110,143,153,249,428,189 146 99 -91362 154,44,114,79,45,135 27,11,17,12,38,18 198 99 diff --git a/example/SJ_matrices/RL_100.matrix/fromGTF.SE.txt b/example/SJ_matrices/RL_100.matrix/fromGTF.SE.txt deleted file mode 100644 index 5f19a0b..0000000 --- a/example/SJ_matrices/RL_100.matrix/fromGTF.SE.txt +++ /dev/null @@ -1,13 +0,0 @@ -ID GeneID geneSymbol chr strand exonStart_0base exonEnd upstreamES upstreamEE downstreamES downstreamEE -34517 "ENSG00000137842.6_3" "TMEM62" chr15 + 43470804 43470909 43461790 43461875 43473378 43473497 -51809 "ENSG00000163681.14_3" "SLMAP" chr3 + 57911571 57911661 57908615 57908750 57913022 57913213 -64939 "ENSG00000090554.12_3" "FLT3LG" chr19 + 49982219 49982304 49979679 49979823 49983554 49983733 -64969 "ENSG00000116001.15_3" "TIA1" chr2 - 70456190 70456223 70454866 70454954 70456395 70456450 -68704 "ENSG00000184381.18_3" "PLA2G6" chr22 - 38524275 38524437 38522377 38522456 38525460 38525569 -75710 "ENSG00000154370.15_3" "TRIM11" chr1 - 228588664 228588895 228584843 228584866 228589766 228589862 -76672 "ENSG00000077458.12_2" "FAM76B" chr11 - 95512241 95512299 95511985 95512121 95512770 95512851 -77359 "ENSG00000089159.16_3" "PXN" chr12 - 120657009 120657894 120652905 120653076 120659425 120659561 -86641 "ENSG00000125970.11_3" "RALY" chr20 + 32661624 32661672 32661368 32661441 32663679 32663845 -91362 "ENSG00000117625.13_3" "RCOR3" chr1 + 211486061 211486303 211474802 211477482 211486765 211487181 -15225 "ENSG00000137814.10_2" "HAUS2" chr15 + 42852979 42853068 42851536 42851606 42853467 42853497 -58989 "ENSG00000169919.16_2" "GUSB" chr7 - 65444713 65444898 65444385 65444528 65445210 65445396 diff --git a/example/SJ_matrices/RL_100_rmatspost_list.txt b/example/SJ_matrices/RL_100_rmatspost_list.txt deleted file mode 100644 index 0a23d56..0000000 --- a/example/SJ_matrices/RL_100_rmatspost_list.txt +++ /dev/null @@ -1 +0,0 @@ -BAMs/RL_100/1325.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_100/2158.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_100/2675.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_100/2870.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_100/2867.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_100/803.aln/Aligned.sortedByCoord.out.bam diff --git a/example/SJ_matrices/RL_150.matrix/JC.raw.input.SE.txt b/example/SJ_matrices/RL_150.matrix/JC.raw.input.SE.txt deleted file mode 100644 index fe3c47b..0000000 --- a/example/SJ_matrices/RL_150.matrix/JC.raw.input.SE.txt +++ /dev/null @@ -1,13 +0,0 @@ -ID IJC_SAMPLE_1 SJC_SAMPLE_1 IJC_SAMPLE_2 SJC_SAMPLE_2 IncFormLen SkipFormLen -8580 243,637,161,347,833,665,1184,587,248,1326,995,988,339,355,2083,361 31,73,17,34,70,59,85,91,63,213,55,129,59,8,78,55 298 149 -13714 293,330,119,259,308,320,379,262,198,68,271,244,28,110,529,207 135,3,1,13,5,13,5,8,3,9,2,0,5,12,12,2 238 149 -43344 34,64,93,88,159,64,73,66,47,162,54,132,209,92,81,42 12,11,10,18,5,17,5,10,7,45,13,16,12,18,12,11 233 149 -48104 61,65,15,117,82,130,117,75,115,168,125,82,10,36,182,133 108,87,28,179,234,131,198,101,71,160,97,91,6,38,153,126 237 149 -66005 358,133,63,205,158,159,251,185,397,318,225,229,44,118,390,379 66,40,6,42,64,90,52,39,23,24,50,15,8,22,51,52 298 149 -97372 42,98,18,48,111,43,83,53,39,76,60,53,17,17,65,30 58,70,17,87,116,167,145,119,280,89,100,77,30,25,118,89 206 149 -119182 202,49,20,123,61,271,85,98,341,23,276,134,25,210,233,290 160,19,3,39,10,42,24,10,33,2,37,33,0,14,15,48 298 149 -120773 236,126,48,158,115,155,167,161,277,317,176,260,82,83,185,315 5,5,1,4,6,10,9,16,33,16,10,13,3,5,3,30 298 149 -131604 2,0,0,3,0,4,4,0,0,0,1,0,0,0,2,0 152,435,67,216,463,453,580,177,98,19,648,74,9,43,1609,219 298 149 -194597 139,100,37,300,232,350,230,198,478,170,167,273,41,61,339,588 60,21,7,10,33,80,16,24,113,77,43,39,9,5,72,94 181 149 -195109 1104,1173,527,1369,1902,1401,1194,1128,1848,4572,1535,1524,410,1767,1576,1852 429,561,209,303,560,760,543,543,1427,1191,602,790,158,603,732,1259 196 149 -224828 225,306,37,75,275,158,246,150,207,135,95,223,163,103,352,184 29,3,2,12,8,38,6,7,26,9,2,6,3,1,8,14 253 149 diff --git a/example/SJ_matrices/RL_150.matrix/fromGTF.SE.txt b/example/SJ_matrices/RL_150.matrix/fromGTF.SE.txt deleted file mode 100644 index 3577123..0000000 --- a/example/SJ_matrices/RL_150.matrix/fromGTF.SE.txt +++ /dev/null @@ -1,13 +0,0 @@ -ID GeneID geneSymbol chr strand exonStart_0base exonEnd upstreamES upstreamEE downstreamES downstreamEE -8580 "ENSG00000169919.16_2" "GUSB" chr7 - 65444713 65444898 65444385 65444528 65445210 65445396 -13714 "ENSG00000163681.14_3" "SLMAP" chr3 + 57911571 57911661 57908615 57908750 57913022 57913213 -43344 "ENSG00000090554.12_3" "FLT3LG" chr19 + 49982219 49982304 49979679 49979823 49983554 49983733 -48104 "ENSG00000137814.10_2" "HAUS2" chr15 + 42852979 42853068 42851536 42851606 42853467 42853497 -66005 "ENSG00000117625.13_3" "RCOR3" chr1 + 211486061 211486303 211474802 211477482 211486765 211487181 -97372 "ENSG00000077458.12_2" "FAM76B" chr11 - 95512241 95512299 95511985 95512121 95512770 95512851 -119182 "ENSG00000184381.18_3" "PLA2G6" chr22 - 38524275 38524437 38522377 38522456 38525460 38525569 -120773 "ENSG00000154370.15_3" "TRIM11" chr1 - 228588664 228588895 228584843 228584866 228589766 228589862 -131604 "ENSG00000089159.16_3" "PXN" chr12 - 120657009 120657894 120652905 120653076 120659425 120659561 -194597 "ENSG00000116001.15_3" "TIA1" chr2 - 70456190 70456223 70454866 70454954 70456395 70456450 -195109 "ENSG00000125970.11_3" "RALY" chr20 + 32661624 32661672 32661368 32661441 32663679 32663845 -224828 "ENSG00000137842.6_3" "TMEM62" chr15 + 43470804 43470909 43461790 43461875 43473378 43473497 diff --git a/example/SJ_matrices/RL_150_rmatspost_list.txt b/example/SJ_matrices/RL_150_rmatspost_list.txt deleted file mode 100644 index 4157290..0000000 --- a/example/SJ_matrices/RL_150_rmatspost_list.txt +++ /dev/null @@ -1 +0,0 @@ -BAMs/RL_150/1142.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/1989.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/2899.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/2907.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/LB2924.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/LB2938.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/LB2964.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/LB3001.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/LB3054.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/LB3336.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/LB3244.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/LB3404.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/LB3374.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/LB3372.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/LB3367.aln/Aligned.sortedByCoord.out.bam,BAMs/RL_150/LB3120.aln/Aligned.sortedByCoord.out.bam diff --git a/example/SJ_matrices/matrices.txt b/example/SJ_matrices/matrices.txt deleted file mode 100644 index 18c40ba..0000000 --- a/example/SJ_matrices/matrices.txt +++ /dev/null @@ -1,2 +0,0 @@ -RL_100.matrix -RL_150.matrix diff --git a/example/SJ_matrices/samples.txt b/example/SJ_matrices/samples.txt deleted file mode 100644 index 35fcbb1..0000000 --- a/example/SJ_matrices/samples.txt +++ /dev/null @@ -1,2 +0,0 @@ -RL_100_rmatspost_list.txt -RL_150_rmatspost_list.txt diff --git a/example/Test.para b/example/Test.para deleted file mode 100644 index cca8f48..0000000 --- a/example/Test.para +++ /dev/null @@ -1,10 +0,0 @@ -Glioma_test -IRIS_data/db/ -filter1 0.01 0.05 1 1 GTEx_Brain -filter2 0.000001593371574 0.05 1 1 TCGA_GBM,TCGA_LGG -filter3 0.01 0.05 1 2 GTEx_Heart,GTEx_Skin,GTEx_Blood,GTEx_Lung,GTEx_Liver,GTEx_Nerve,GTEx_Muscle,GTEx_Spleen,GTEx_Thyroid,GTEx_Kidney,GTEx_Stomach -group -False - -IRIS_data/resources/mappability/wgEncodeCrgMapabilityAlign24mer.bigWig -IRIS_data/resources/reference/ucsc.hg19.fasta diff --git a/example/Test_simplified.para b/example/Test_simplified.para deleted file mode 100644 index db02cc4..0000000 --- a/example/Test_simplified.para +++ /dev/null @@ -1,6 +0,0 @@ -Glioma_test -filter1 0.01 0.05 1 1 GTEx_Brain -filter2 0.000001593371574 0.05 1 1 TCGA_GBM,TCGA_LGG -filter3 0.01 0.05 1 2 GTEx_Heart,GTEx_Skin,GTEx_Blood,GTEx_Lung,GTEx_Liver,GTEx_Nerve,GTEx_Muscle,GTEx_Spleen,GTEx_Thyroid,GTEx_Kidney,GTEx_Stomach -group -False diff --git a/example/exp_matrix_test.txt b/example/exp_matrix_test.txt new file mode 100644 index 0000000..fd818e3 --- /dev/null +++ b/example/exp_matrix_test.txt @@ -0,0 +1,11 @@ +geneName Sample1 Sample2 Sample3 Sample4 Sample5 Sample6 Sample7 Sample8 Sample9 Sample10 +ENSG00000008282.8_2 13.8136 30.6688 28.7282 26.4562 32.2042 3.24332 29.1389 35.7019 31.3576 13.7008 +ENSG00000008710.19_3 25.6837 200.984 77.3921 92.3767 116.713 15.8125 117.737 128.881 36.1041 7.51057 +ENSG00000090674.15_3 15.863 24.4872 9.84709 18.8131 12.9838 32.5762 3.32612 4.70605 12.2316 37.9599 +ENSG00000105939.12_3 5.60172 5.12532 6.07503 6.76975 11.3333 7.30481 6.73759 15.3272 14.5119 6.05959 +ENSG00000110367.11_2 16.4841 17.6602 6.62134 21.7654 17.4255 12.3803 30.7404 18.8044 36.8079 23.5613 +ENSG00000125814.17_2 8.50579 10.2384 7.186 5.78771 6.18153 3.52837 3.78008 3.41479 9.14832 15.5857 +ENSG00000142192.20_3 176.655 556.907 251.233 347.058 312.269 94.2619 278.205 310.817 348.916 24.6239 +ENSG00000171603.16_2 36.7182 83.7313 6.87977 6.28606 120.779 51.4822 59.4516 111.927 91.1514 7.91309 +ENSG00000183773.15_3 0.846555 2.37515 0.302885 5.09117 5.33169 7.62877 0.69174 2.74661 0.834716 2.1072 +ENSG00000184220.11_3 35.252 32.0691 31.1302 6.09923 6.49715 7.98065 6.13434 6.71738 52.5053 11.1754 diff --git a/example/gene_exp_file_list.txt b/example/gene_exp_file_list.txt deleted file mode 100644 index e719ed4..0000000 --- a/example/gene_exp_file_list.txt +++ /dev/null @@ -1,22 +0,0 @@ -glioma/BAMs/RL_100/1325.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_100/2158.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_100/2675.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_100/2867.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_100/2870.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_100/803.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/1142.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/1989.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/2899.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/2907.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/LB2924.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/LB2938.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/LB2964.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/LB3001.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/LB3054.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/LB3120.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/LB3244.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/LB3336.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/LB3367.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/LB3372.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/LB3374.aln/cufflinks/genes.fpkm_tracking -glioma/BAMs/RL_150/LB3404.aln/cufflinks/genes.fpkm_tracking \ No newline at end of file diff --git a/example/hla_patient_test.tsv b/example/hla_patient_test.tsv new file mode 100644 index 0000000..e037797 --- /dev/null +++ b/example/hla_patient_test.tsv @@ -0,0 +1,10 @@ +Sample1 HLA-A*23:01 HLA-A*02:01 HLA-B*13:02 HLA-B*44:02 HLA-C*06:02 +Sample2 HLA-A*03:02 HLA-A*24:02 HLA-B*55:01 HLA-B*35:02 HLA-C*03:03 +Sample3 HLA-A*01:01 HLA-B*38:01 HLA-B*57:01 HLA-C*06:02 +Sample4 HLA-A*01:01 HLA-A*32:03 HLA-B*40:01 HLA-B*07:02 HLA-C*03:04 HLA-C*07:02 +Sample5 HLA-A*26:01 HLA-A*01:01 HLA-B*38:01 HLA-B*35:02 HLA-C*12:03 HLA-C*04:01 +Sample6 HLA-A*24:02 HLA-B*13:01 HLA-C*07:02 +Sample7 HLA-A*02:01 HLA-B*18:01 HLA-B*15:01 HLA-C*12:03 HLA-C*03:04 +Sample8 HLA-A*01:01 HLA-A*02:01 HLA-B*57:01 HLA-B*15:01 HLA-C*06:02 HLA-C*03:04 +Sample9 HLA-A*30:02 HLA-A*29:01 HLA-B*07:05 HLA-B*53:01 HLA-C*15:05 HLA-C*04:01 +Sample10 HLA-A*03:01 HLA-A*03:01 HLA-B*07:02 HLA-B*35:03 HLA-C*04:01 HLA-C*07:02 \ No newline at end of file diff --git a/example/HLA_types/hla_types.list b/example/hla_types_test.list similarity index 63% rename from example/HLA_types/hla_types.list rename to example/hla_types_test.list index 55a60ca..8095ed6 100644 --- a/example/HLA_types/hla_types.list +++ b/example/hla_types_test.list @@ -1,49 +1,31 @@ -HLA-A*01:01 -HLA-A*02:01 -HLA-A*02:05 -HLA-A*03:01 -HLA-A*03:02 -HLA-A*11:01 -HLA-A*23:01 -HLA-A*24:02 -HLA-A*26:01 -HLA-A*29:01 -HLA-A*30:02 -HLA-A*32:01 -HLA-A*32:03 -HLA-A*68:02 -HLA-A*74:03 -HLA-B*07:02 -HLA-B*07:05 -HLA-B*08:01 -HLA-B*13:01 -HLA-B*13:02 -HLA-B*14:02 -HLA-B*15:01 -HLA-B*18:01 -HLA-B*35:02 -HLA-B*35:03 -HLA-B*35:08 -HLA-B*37:01 -HLA-B*38:01 -HLA-B*40:01 -HLA-B*44:02 -HLA-B*44:03 -HLA-B*46:01 -HLA-B*48:01 -HLA-B*50:01 -HLA-B*51:01 -HLA-B*53:01 -HLA-B*55:01 -HLA-B*57:01 -HLA-C*03:03 -HLA-C*03:04 -HLA-C*04:01 -HLA-C*06:02 -HLA-C*07:01 -HLA-C*07:02 -HLA-C*08:01 -HLA-C*12:03 -HLA-C*14:02 -HLA-C*15:02 HLA-C*15:05 +HLA-C*12:03 +HLA-C*07:02 +HLA-C*06:02 +HLA-C*04:01 +HLA-C*03:04 +HLA-C*03:03 +HLA-B*57:01 +HLA-B*55:01 +HLA-B*53:01 +HLA-B*44:02 +HLA-B*40:01 +HLA-B*38:01 +HLA-B*35:03 +HLA-B*35:02 +HLA-B*18:01 +HLA-B*15:01 +HLA-B*13:02 +HLA-B*13:01 +HLA-B*07:05 +HLA-B*07:02 +HLA-A*32:03 +HLA-A*30:02 +HLA-A*29:01 +HLA-A*26:01 +HLA-A*24:02 +HLA-A*23:01 +HLA-A*03:02 +HLA-A*03:01 +HLA-A*02:01 +HLA-A*01:01 diff --git a/example/parameter_file_description.txt b/example/parameter_file_description.txt index a2d79ce..e9a348c 100644 --- a/example/parameter_file_description.txt +++ b/example/parameter_file_description.txt @@ -1,34 +1,45 @@ ------------------------------------------------------------------------------------------------------------------------- -Parameter file format (take the test file for example): ------------------------------------------------------------------------------------------------------------------------- -Glioma_test -IRIS_data/db/ -filter1 0.01 0.05 1 1 GTEx_Brain -filter2 0.000001593371574 0.05 1 1 TCGA_GBM,TCGA_LGG -filter3 0.01 0.05 1 2 GTEx_Heart,GTEx_Skin,GTEx_Blood,GTEx_Lung,GTEx_Liver,GTEx_Nerve,GTEx_Muscle,GTEx_Spleen,GTEx_Thyroid,GTEx_Kidney,GTEx_Stomach -group -False - -IRIS_data/resources/mappability/wgEncodeCrgMapabilityAlign24mer.bigWig -IRIS_data/resources/reference/ucsc.hg19.fasta ------------------------------------------------------------------------------------------------------------------------- -Row1: Input group's name in the IRIS db (see 'formatting' step) -Row2: Directory of IRIS db -Row3: Parameters for 'Tissue-matched normal panel'; screening tumor-associated events. -Fields are separated by ' ': -filter_name p-value_cutoff deltaPSI_cutoff FC_cutoff filter1_group_cutoff filter1_reference_list - # filter_name: Row name (No space; Required) - # p-value_cutoff: Cutoff of p-value for statistical tests being used (Optional) - # deltaPSI_cutoff: Difference of PSI values between input sample and normal control should be larger than this threshold, which ensures the effect size of splicing change (Optional) - # FC_cutoff: Fold Change cutoff of PSI value of input sample compared to normal control (Optional) - # filter1_group_cutoff: Minimum number of tumor/tissue reference panels satisfying above requirements (Optional) - # filter1_reference_list: A list of selected reference panels (separated by ',';Optional) - -Row4: Parameters for 'Tumor panel'; screening for tumor-recurrent events (See Row3) -Row5: Parameters for other 'Normal panel'; screening tumor-specific events (See Row3) -Row3-5: As described in Row3, all fields are optional (except Row name). At least one row has to have values to perform screening. Note that 'Tumor panel' along will not function if 'Tissue-matched normal panel' is missing. -Row6: Comparison mode, 'group' mode (number of input samples >=2) and 'individual mode' (number of input sample =1) are provided -Row7: Use ratio or not for group_cutoff (described in Row3) -Row8: Blacklist file, which can remove the AS events that are error-prone due to mappability or sequencing limitations (See methods for detail) -Row9: Mappability file, which can be used to evaluate the AS events ------------------------------------------------------------------------------------------------------------------------- +------------------------------------------------------------------------------------------------------------------------ +Screening parameter file format (take a screening task for 'Glioma_test' as an example): +------------------------------------------------------------------------------------------------------------------------ +Glioma_test +IRIS_data/db/ +0.01,0.05,1,0.000001,1 GTEx_Brain +0.000001,0.05,1,0.000001,1 TCGA_GBM,TCGA_LGG +0.01,0.05,1,0.000001,5 GTEx_Heart,GTEx_Skin,GTEx_Blood,GTEx_Lung,GTEx_Liver,GTEx_Nerve,GTEx_Muscle,GTEx_Spleen,GTEx_Thyroid,GTEx_Kidney,GTEx_Stomach +group parametric +False + +IRIS_data/resources/mappability/wgEncodeCrgMapabilityAlign24mer.bigWig +IRIS_data/resources/reference/ucsc.hg19.fasta +------------------------------------------------------------------------------------------------------------------------ +Row1 (required): Input group name. This should be identical to the folder name in the IRIS db (see 'format' step) + +Row2 (required): Path to the directory of IRIS db + +Row3 (see below): Parameters for the tier 1 screening (i.e. 'tissue-matched normal reference') +Fields are separated by ' ': fitler1_cutoffs and filter1_reference_list +within each field, parameters are separated by ',': +- tier1_cutoffs: PSI_p-value_cutoff,deltaPSI_cutoff,FC_cutoff,SJC_p-value_cutoff,filter1_group_cutoff + # PSI_p-value_cutoff: Cutoff of p-value for statistical tests in PSI-based tests. + # deltaPSI_cutoff: Minimum difference of PSI values (i.e deltaPSI) between tumor and normal groups + # FC_cutoff: Fold Change cutoff of PSI value of input sample compared to normal control + # SJC_p-value_cutoff: Cutoff of p-value for statistical tests in SJC-based tests. + # tier1_group_cutoff: Minimum number of tumor/tissue reference groups satisfying above requirements. This number should be no larger than the number of reference normal groups selected in the next field ('tier1_reference list') +- tier1_reference_list: A list of selected reference panels (separated by ','). Names should be identical to the folder name in the IRIS db (see 'format' step) + +Row4 (see below): Parameters for the tier 2 screening (i.e. 'tumor reference'). Same format as the row3 + +Row5 (see below): Parameters for the tier 3 screening (i.e. 'normal tissue reference'). Same format as the row3 + +Row3-5 (at least one NORMAL reference row is required) For example, tier 1 alone, or tier 3 alone, or tier 1 + tier 2 + tier 3 are valid settings for screening. Note that tier 2 alone will not function if tier 1 is missing (tumor recurrence is estabilshed by comparing to a user-specified tissue-matched normal reference in tier 1) + +Row6 (required for PSI-based tests): Comparison mode & statistical test type: 'group' mode (number of input samples >=2) and 'individual mode' (number of input sample =1) are provided. 'group' mode is default and recommended; for PSI-based tests, 'parametric' and 'nonparametric' tests are supported. 'parametric' is default + +Row7 (required for PSI-based tests): Use ratio instead of number of groups for the tierX_group_cutoff. Default is False + +Row8 (optional): Blacklist file. Removes the AS events that are error-prone due to artifacts. Optional + +Row9 (optional): Mappability annotation bigWig file. Required for evaluating splice region mappability. + +Row10 (optional): Reference genome file. Required for IRIS translate. +------------------------------------------------------------------------------------------------------------------------ diff --git a/example/sjc_matrix/SJ_count.NEPC_example.txt b/example/sjc_matrix/SJ_count.NEPC_example.txt new file mode 100644 index 0000000..78305c9 --- /dev/null +++ b/example/sjc_matrix/SJ_count.NEPC_example.txt @@ -0,0 +1,31 @@ +SJ Sample1 Sample2 Sample3 Sample4 Sample5 Sample6 Sample7 Sample8 Sample9 Sample10 +chr19:7593857:7593986 213 285 11 149 139 103 185 220 3 119 +chr19:7594089:7594475 203 231 13 110 115 142 184 149 7 97 +chr19:7593857:7594475 0 0 0 0 0 0 0 0 0 0 +chr22:21331228:21331320 40 30 36 0 3 0 23 8 2 10 +chr22:21331385:21331988 40 12 32 1 5 0 20 3 1 5 +chr22:21331228:21331988 0 0 0 0 0 0 0 0 0 0 +chr7:105733584:105736738 19 11 3 0 28 0 116 22 132 53 +chr7:105736749:105738135 11 8 1 0 21 0 115 13 118 45 +chr7:105733584:105738135 11 36 182 958 41 38 70 413 42 58 +chr3:99536888:99770075 0 0 1 0 0 0 0 0 0 0 +chr3:99770151:99865816 0 0 0 0 0 1 0 0 0 0 +chr3:99536888:99865816 9 131 193 340 18 140 29 0 7 23 +chr20:23375823:23377708 8 13 18 63 52 49 88 35 36 10 +chr20:23377826:23383629 3 4 26 46 38 46 67 18 42 3 +chr20:23375823:23383629 7 9 9 16 16 17 9 3 3 15 +chr7:138763400:138763849 2 2 3 20 2 17 0 1 5 20 +chr7:138764990:138768525 17 8 8 135 25 80 0 10 32 114 +chr7:138763400:138768525 0 0 0 0 0 0 0 0 0 1 +chr1:9796101:9797555 0 10 13 14 11 39 9 50 22 27 +chr1:9797613:9801151 0 7 8 12 5 23 5 36 15 19 +chr1:9796101:9801151 70 960 78 452 552 881 323 1308 1050 1373 +chr21:27354791:27369674 237 813 173 140 164 16 161 151 1867 458 +chr21:27369732:27394155 1 1 2 0 5 1 3 7 12 9 +chr21:27354791:27394155 61 43 367 5 181 7 34 1193 266 417 +chr16:2162965:2163041 135 109 76 29 154 114 117 63 24 144 +chr16:2163061:2163161 8 6 7 2 5 7 1 1 1 4 +chr16:2162965:2163161 11 11 8 6 8 15 18 19 7 12 +chr11:118651894:118656760 34 140 196 100 63 105 58 320 26 74 +chr11:118657228:118661806 13 53 44 20 21 31 24 47 10 27 +chr11:118651894:118661806 0 0 0 0 0 0 0 0 0 0 diff --git a/example/sjc_matrix/SJ_count.NEPC_example.txt.idx b/example/sjc_matrix/SJ_count.NEPC_example.txt.idx new file mode 100644 index 0000000..f9710e7 --- /dev/null +++ b/example/sjc_matrix/SJ_count.NEPC_example.txt.idx @@ -0,0 +1,30 @@ +chr19:7593857:7593986 84 +chr19:7594089:7594475 143 +chr19:7593857:7594475 201 +chr22:21331228:21331320 243 +chr22:21331385:21331988 292 +chr22:21331228:21331988 340 +chr7:105733584:105736738 384 +chr7:105736749:105738135 438 +chr7:105733584:105738135 491 +chr3:99536888:99770075 549 +chr3:99770151:99865816 592 +chr3:99536888:99865816 635 +chr20:23375823:23377708 689 +chr20:23377826:23383629 742 +chr20:23375823:23383629 793 +chr7:138763400:138763849 841 +chr7:138764990:138768525 889 +chr7:138763400:138768525 943 +chr1:9796101:9797555 988 +chr1:9797613:9801151 1037 +chr1:9796101:9801151 1083 +chr21:27354791:27369674 1145 +chr21:27369732:27394155 1209 +chr21:27354791:27394155 1254 +chr16:2162965:2163041 1312 +chr16:2163061:2163161 1370 +chr16:2162965:2163161 1412 +chr11:118651894:118656760 1460 +chr11:118657228:118661806 1521 +chr11:118651894:118661806 1577 diff --git a/example/splicing_matrix/splicing_matrix.SE.cov10.NEPC_example.txt b/example/splicing_matrix/splicing_matrix.SE.cov10.NEPC_example.txt new file mode 100644 index 0000000..fcec6b6 --- /dev/null +++ b/example/splicing_matrix/splicing_matrix.SE.cov10.NEPC_example.txt @@ -0,0 +1,11 @@ +AC GeneName chr strand exonStart exonEnd upstreamEE downstreamES Sample1 Sample2 Sample3 Sample4 Sample5 Sample6 Sample7 Sample8 Sample9 Sample10 +ENSG00000090674 MCOLN1 chr19 + 7593986 7594088 7593856 7594475 1.0 NaN NaN NaN NaN NaN 1.0 NaN NaN 1.0 +ENSG00000183773 AIFM3 chr22 + 21331320 21331384 21331227 21331988 1.0 NaN NaN NaN NaN NaN NaN NaN NaN NaN +ENSG00000008282 SYPL1 chr7 - 105736738 105736748 105733583 105738135 0.5792 0.1807 0.0099 0.0 0.381 0.0 0.5996 0.0401 0.7376 0.4413 +ENSG00000184220 CMSS1 chr3 + 99770075 99770150 99536887 99865816 NaN 0.0 0.003 0.0 0.0 0.0044 0.0 NaN NaN 0.0 +ENSG00000125814 NAPB chr20 - 23377708 23377825 23375822 23383629 0.44 0.4857 0.7097 0.773 0.7377 0.742 0.896 0.8983 0.9286 0.3023 +ENSG00000105939 ZC3HAV1 chr7 - 138763849 138764989 138763399 138768525 NaN NaN NaN 1.0 NaN 1.0 NaN NaN NaN 0.9853 +ENSG00000171603 CLSTN1 chr1 - 9797555 9797612 9796100 9801151 0.0 0.0082 0.1272 0.028 0.0131 0.0341 0.0206 0.0354 0.0167 0.0172 +ENSG00000142192 APP chr21 - 27369674 27369731 27354790 27394155 0.582 0.8617 0.1689 0.9306 0.3113 0.549 0.7069 0.0516 0.7034 0.3077 +ENSG00000008710 PKD1 chr16 - 2163041 2163060 2162964 2163161 0.728 0.7432 0.7465 0.594 0.8069 0.685 0.5441 0.3865 0.5327 0.7358 +ENSG00000110367 DDX6 chr11 - 118656760 118657227 118651893 118661806 NaN 1.0 1.0 NaN NaN NaN NaN 1.0 NaN NaN diff --git a/example/splicing_matrix/splicing_matrix.SE.cov10.NEPC_example.txt.idx b/example/splicing_matrix/splicing_matrix.SE.cov10.NEPC_example.txt.idx new file mode 100644 index 0000000..7349326 --- /dev/null +++ b/example/splicing_matrix/splicing_matrix.SE.cov10.NEPC_example.txt.idx @@ -0,0 +1,10 @@ +ENSG00000090674:MCOLN1:chr19:+:7593986:7594088:7593856:7594475 146 +ENSG00000183773:AIFM3:chr22:+:21331320:21331384:21331227:21331988 249 +ENSG00000008282:SYPL1:chr7:-:105736738:105736748:105733583:105738135 355 +ENSG00000184220:CMSS1:chr3:+:99770075:99770150:99536887:99865816 487 +ENSG00000125814:NAPB:chr20:-:23377708:23377825:23375822:23383629 597 +ENSG00000105939:ZC3HAV1:chr7:-:138763849:138764989:138763399:138768525 727 +ENSG00000171603:CLSTN1:chr1:-:9797555:9797612:9796100:9801151 841 +ENSG00000142192:APP:chr21:-:27369674:27369731:27354790:27394155 969 +ENSG00000008710:PKD1:chr16:-:2163041:2163060:2162964:2163161 1101 +ENSG00000110367:DDX6:chr11:-:118656760:118657227:118651893:118661806 1229 diff --git a/google_drive_download.py b/google_drive_download.py new file mode 100644 index 0000000..0e4033f --- /dev/null +++ b/google_drive_download.py @@ -0,0 +1,187 @@ +import argparse +import os +import os.path +import tempfile + +import tqdm + +from apiclient.http import MediaIoBaseDownload +from google.oauth2 import service_account +from googleapiclient.discovery import build + +TOP_DIR_NAME = 'IRIS_data' +CHUNK_SIZE = 1024 * 1024 * 8 # 8 MB +SCOPES = ['https://www.googleapis.com/auth/drive'] + + +def parse_args(): + parser = argparse.ArgumentParser( + description=('download IRIS files from google drive')) + parser.add_argument('--iris-folder-id', + help=('ID of IRIS_data folder on google drive' + ' (can be found in download url)')) + parser.add_argument( + '--dest-dir', + help='path to directory where IRIS_data/ will be written') + parser.add_argument('--download-all', + action='store_true', + help='download all data') + parser.add_argument('--list-files', + action='store_true', + help='write available files to --selected-tsv') + parser.add_argument('--selected-tsv', + help='path to a tsv with selected files to download') + parser.add_argument( + '--api-key-json-path', + required=True, + help='path to the .json file that is the service account key') + + args = parser.parse_args() + if args.download_all: + if not (args.iris_folder_id and args.dest_dir): + parser.error( + '--download-all requires --iris-folder-id and --dest-dir') + elif args.list_files: + if not (args.iris_folder_id and args.selected_tsv): + parser.error( + '--list-files requires --iris-folder-id and --selected-tsv') + elif not (args.dest_dir and args.selected_tsv): + parser.error( + 'download specific files with --dest-dir and --selected-tsv.' + ' Otherwise use --download-all or --list-files') + + return args + + +def main(): + args = parse_args() + + credentials = service_account.Credentials.from_service_account_file( + args.api_key_json_path, scopes=SCOPES) + with build('drive', 'v3', credentials=credentials) as drive_service: + if args.download_all: + download_all_files(args.iris_folder_id, args.dest_dir, + drive_service) + return + + if args.list_files: + list_all_files(args.iris_folder_id, args.selected_tsv, + drive_service) + return + + download_selected_files(args.dest_dir, args.selected_tsv, + drive_service) + + +def list_files_recursive(parent_id, drive_service): + results = list() + files_c = drive_service.files() + request = files_c.list(q="'{}' in parents".format(parent_id)) + + response = request.execute() + found = response.get('files', list()) + for file_dict in found: + file_id = file_dict['id'] + file_name = file_dict['name'] + is_folder = 'folder' in file_dict['mimeType'] + if is_folder: + sub_results = list_files_recursive(file_id, drive_service) + results.append({ + 'folder': file_name, + 'id': file_id, + 'files': sub_results + }) + else: + results.append({'name': file_name, 'id': file_id}) + + return results + + +def write_tsv_line(columns, tsv_handle): + tsv_handle.write('{}\n'.format('\t'.join(columns))) + + +def write_file_tsv(all_files, parent_dir_path, tsv_handle): + files = list() + folders = list() + for file_dict in all_files: + if 'folder' in file_dict: + folders.append(file_dict) + else: + files.append(file_dict) + + files.sort(key=lambda d: d['name']) + folders.sort(key=lambda d: d['folder']) + for file_dict in files: + full_path = os.path.join(parent_dir_path, file_dict['name']) + write_tsv_line([full_path, file_dict['id']], tsv_handle) + + for folder_dict in folders: + full_path = os.path.join(parent_dir_path, folder_dict['folder']) + write_file_tsv(folder_dict['files'], full_path, tsv_handle) + + +def download_all_files(iris_folder_id, dest_dir, drive_service): + all_files = list_files_recursive(iris_folder_id, drive_service) + + temp_name = None + try: + with tempfile.NamedTemporaryFile(delete=False) as temp_handle: + temp_name = temp_handle.name + + write_file_tsv(all_files, TOP_DIR_NAME, temp_handle) + + download_selected_files(dest_dir, temp_name, drive_service) + finally: + if temp_name: + os.remove(temp_name) + + +def list_all_files(iris_folder_id, selected_tsv, drive_service): + all_files = list_files_recursive(iris_folder_id, drive_service) + with open(selected_tsv, 'wt') as tsv_handle: + write_file_tsv(all_files, TOP_DIR_NAME, tsv_handle) + + +def download_file(file_id, dest_path, drive_service): + files_c = drive_service.files() + request = files_c.get_media(fileId=file_id) + dir_path = os.path.dirname(dest_path) + if not os.path.exists(dir_path): + os.makedirs(dir_path) + + with open(dest_path, 'wb') as out_handle: + downloader = MediaIoBaseDownload(out_handle, + request, + chunksize=CHUNK_SIZE) + progress = tqdm.tqdm(desc=dest_path, total=1.0, unit='file') + progress_so_far = 0 + done = False + while done is False: + status, done = downloader.next_chunk() + if status: + new_progress = status.progress() + additional = new_progress - progress_so_far + progress_so_far = new_progress + progress.update(additional) + + progress.close() + + +def download_selected_files(dest_dir, selected_tsv, drive_service): + with open(selected_tsv, 'rt') as tsv_handle: + for line in tsv_handle: + columns = line.strip().split('\t') + if len(columns) != 2: + raise Exception( + 'expected 2 columns in {}'.format(selected_tsv)) + + drive_file_path = columns[0] + file_id = columns[1] + + local_file_path = os.path.join(dest_dir, drive_file_path) + download_file(file_id, local_file_path, drive_service) + + +if __name__ == '__main__': + main() diff --git a/install b/install index bd316b9..dfdf5c0 100755 --- a/install +++ b/install @@ -2,24 +2,22 @@ # # Install dependencies # -# ** Must manually install conda first. ** -# This script will create two conda environments (Python 2 and 3) -# and install dependencies to them. -# https://docs.conda.io/en/latest/miniconda.html -# https://repo.anaconda.com/miniconda/Miniconda2-latest-Linux-x86_64.sh -# -# ** Must manually download the IRIS_data set ** -# https://drive.google.com/file/d/1TaswpWPnEd4TXst46jsa9XSMzLsbzjOQ/view?usp=sharing -# -# ** Must manually download IEDB tools version 2.15.5 to the IEDB directory. ** -# download file to IEDB/IEDB_MHC_I-2.15.5.tar.gz -# This script will unpack and install the IEDB tools. -# http://tools.iedb.org/main/download/ -> MHC Class I -> previous version -> 2.15.5 - function install_iedb() { echo echo "checking IEDB dependency" + # From IEDB/mhc_i/README: tcsh and gawk are required + which tcsh + if [[ "$?" -ne 0 ]]; then + echo "IEDB requires tcsh to be installed" >&2 + return 1 + fi + which gawk + if [[ "$?" -ne 0 ]]; then + echo "IEDB requires gawk to be installed" >&2 + return 1 + fi + cd "${SCRIPT_DIR}/IEDB" || return 1 if [[ ! -d mhc_i ]] @@ -29,156 +27,17 @@ function install_iedb() { cd mhc_i || return 1 - ./configure || return 1 -} - -# This script will automatically download and build bedtools -# https://github.com/arq5x/bedtools2/releases -function install_bedtools() { - echo - echo "checking bedtools dependency" - - cd "${SCRIPT_DIR}" || return 1 - - if [[ ! -d bedtools ]] - then - mkdir bedtools || return 1 - fi - - cd bedtools || return 1 - - if [[ ! -f bedtools-2.29.0.tar.gz ]] - then - local BEDTOOLS_URI="https://github.com/arq5x/bedtools2/releases/download/v2.29.0/bedtools-2.29.0.tar.gz" - curl -L "${BEDTOOLS_URI}" -o bedtools-2.29.0.tar.gz || return 1 - fi - - if [[ ! -d bedtools2 ]] - then - tar -xvf bedtools-2.29.0.tar.gz || return 1 - fi - - cd bedtools2 || return 1 - make || return 1 -} - -function install_star() { - echo - echo "checking STAR dependency" - - cd "${SCRIPT_DIR}" || return 1 - - if [[ ! -d STAR ]] - then - mkdir STAR || return 1 - fi - - cd STAR || return 1 - - if [[ ! -f 2.5.3a.tar.gz ]] - then - local STAR_URI="https://github.com/alexdobin/STAR/archive/2.5.3a.tar.gz" - curl -L "${STAR_URI}" -o 2.5.3a.tar.gz || return 1 - fi - - if [[ ! -d STAR-2.5.3a ]] - then - tar -xvf 2.5.3a.tar.gz || return 1 - fi - - cd STAR-2.5.3a/source || return 1 - make || return 1 -} - -function install_samtools() { - echo - echo "checking SAM tools dependency" - - cd "${SCRIPT_DIR}" || return 1 - - if [[ ! -d samtools ]] - then - mkdir samtools || return 1 - fi - - cd samtools || return 1 - - if [[ ! -f samtools-1.3.tar.bz2 ]] - then - local SAMTOOLS_URI="https://sourceforge.net/projects/samtools/files/samtools/1.3/samtools-1.3.tar.bz2/download" - curl -L "${SAMTOOLS_URI}" -o samtools-1.3.tar.bz2 || return 1 - fi - - if [[ ! -d samtools-1.3 ]] - then - tar -xvf samtools-1.3.tar.bz2 || return 1 - fi - - cd samtools-1.3 || return 1 - ./configure --enable-plugins --enable-libcurl --with-plugin-path="$(pwd)"/htslib-1.3 || return 1 - make all plugins-htslib || return 1 -} - -function install_rmats() { - echo - echo "checking rMATS dependency" - - cd "${SCRIPT_DIR}" || return 1 - - if [[ ! -d rMATS ]] - then - mkdir rMATS || return 1 - fi - - cd rMATS || return 1 - - if [[ ! -f rMATS.4.0.2.tgz ]] - then - local RMATS_URI="https://sourceforge.net/projects/rnaseq-mats/files/MATS/rMATS.4.0.2.tgz/download" - curl -L "${RMATS_URI}" -o rMATS.4.0.2.tgz || return 1 - fi - - if [[ ! -d rMATS.4.0.2 ]] - then - tar -xvf rMATS.4.0.2.tgz || return 1 - fi -} - -function install_cufflinks() { - echo - echo "checking Cufflinks dependency" - - cd "${SCRIPT_DIR}" || return 1 - - if [[ ! -d cufflinks ]] - then - mkdir cufflinks || return 1 - fi - - cd cufflinks || return 1 - - if [[ ! -f cufflinks-2.2.1.Linux_x86_64.tar.gz ]] - then - local CUFFLINKS_URI="http://cole-trapnell-lab.github.io/cufflinks/assets/downloads/cufflinks-2.2.1.Linux_x86_64.tar.gz" - curl "${CUFFLINKS_URI}" -o cufflinks-2.2.1.Linux_x86_64.tar.gz || return 1 - fi - - if [[ ! -d cufflinks-2.2.1.Linux_x86_64 ]] - then - tar -xvf cufflinks-2.2.1.Linux_x86_64.tar.gz || return 1 - fi -} - -function install_seq2hla() { - echo - echo "checking seq2HLA dependency" - - cd "${SCRIPT_DIR}" || return 1 - - if [[ ! -d seq2hla ]] - then - local SEQ_2_HLA_URI="https://bitbucket.org/sebastian_boegel/seq2hla" - hg clone "${SEQ_2_HLA_URI}" || return 1 + # The IEDB ./configure script unpacks .tar files which can take a long time. + # Create a .done file to indicate that ./configure has already been run. + # Only run ./configure if there's no .done file + local DONE_FILE='configure.done' + if [[ ! -f "${DONE_FILE}" ]]; then + # Need to activate Python 2 environment so that '/usr/bin/env python' used in + # ./configure finds python 2. Otherwise there is a syntax error. + conda activate "${CONDA_ENV_PREFIX_2}" || return 1 + ./configure || return 1 + conda deactivate || return 1 + touch "${DONE_FILE}" || return 1 fi } @@ -208,24 +67,74 @@ function install_ms_gf() { } function install_python_packages() { + local INSTALL_OPTIONAL="$1" echo echo "checking python dependencies" cd "${SCRIPT_DIR}" || return 1 + # rmats is an optional dependency. + # Record the path if it is installed. + local RMATS_PATH='' + # Python 2 - conda::activate_env "${CONDA_ENV_NAME_2}" || return 1 - pip install -r requirements.txt || return 1 - conda install -c bioconda bowtie # has an error return even when successful - conda::deactivate_env || return 1 + conda activate "${CONDA_ENV_PREFIX_2}" || return 1 + if [[ "${INSTALL_OPTIONAL}" -ne 0 ]]; then + conda install -c conda-forge -c bioconda --file conda_requirements_py2.txt \ + --file conda_requirements_py2_optional.txt || return 1 + # r-base is an optional dependency which requires an old version of libreadline. + # The version of libreadline is not available on conda, but it can be installed manually. + install_readline || return 1 + # Find the rmats path needed to pass as --rMATS-path to IRIS + RMATS_PATH="$(which rmats.py)" + if [[ "$?" -ne 0 ]]; then + echo "could not find path to rmats.py" >&2 + exit 1 + fi + else + conda install -c conda-forge -c bioconda --file conda_requirements_py2.txt || return 1 + fi + + # rmats_path will be '' if optional dependencies were not installed + echo "rmats_path: '${RMATS_PATH}'" \ + >> "${SCRIPT_DIR}/snakemake_config.yaml" || return 1 + + conda deactivate || return 1 # Python 3 - conda::activate_env "${CONDA_ENV_NAME_3}" || return 1 - pip install -r qsub/requirements.txt || return 1 - conda::deactivate_env || return 1 + conda activate "${CONDA_ENV_PREFIX_3}" || return 1 + conda install -c conda-forge -c bioconda --file conda_requirements_py3.txt || return 1 + conda deactivate || return 1 } -function install_iris_data() { +function install_readline() { + local ORIG_DIR="$(pwd)" || return 1 + cd "${SCRIPT_DIR}" || return 1 + mkdir -p readline || return 1 + cd readline || return 1 + local TAR_PATH='readline-6.3.tar.gz' + local TAR_URL='ftp://ftp.gnu.org/gnu/readline/readline-6.3.tar.gz' + if [[ ! -f "${TAR_PATH}" ]]; then + curl "${TAR_URL}" -o "${TAR_PATH}" || return 1 + fi + + if [[ ! -f 'readline-6.3/configure' ]]; then + tar -xvf "${TAR_PATH}" || return 1 + fi + + cd readline-6.3 || return 1 + local SO_PATH='shlib/libreadline.so.6.3' + if [[ ! -f "${SO_PATH}" ]]; then + ./configure || return 1 + make || return 1 + fi + + local DEST="${CONDA_ENV_PREFIX_2}/lib/libreadline.so.6" + cp "${SO_PATH}" "${DEST}" || return 1 + cd "${ORIG_DIR}" +} + +function check_iris_data() { echo echo "checking IRIS data dependency" @@ -233,13 +142,7 @@ function install_iris_data() { if [[ ! -d IRIS_data ]] then - if [[ ! -f IRIS_data.tgz ]]; then - echo "Need to download IRIS_data.tgz from:" - echo "https://drive.google.com/file/d/1TaswpWPnEd4TXst46jsa9XSMzLsbzjOQ/view?usp=sharing" - return 1 - fi - - tar -xvf IRIS_data.tgz || return 1 + echo "Need to download IRIS_data/" >&2 fi } @@ -249,48 +152,20 @@ function install_iris_package() { cd "${SCRIPT_DIR}" || return 1 - conda::activate_env "${CONDA_ENV_NAME_2}" || return 1 - + conda activate "${CONDA_ENV_PREFIX_2}" || return 1 python setup.py install || return 1 - - cd "${SCRIPT_DIR}" || return 1 - - conda::deactivate_env || return 1 + conda deactivate || return 1 } - function ensure_conda_envs() { echo echo "checking conda" - conda::create_env_with_name_and_python_version "${CONDA_ENV_NAME_2}"\ - "${CONDA_PYTHON_VERSION_2}" || return 1 - - conda::create_env_with_name_and_python_version "${CONDA_ENV_NAME_3}"\ - "${CONDA_PYTHON_VERSION_3}" || return 1 + conda create --prefix "${CONDA_ENV_PREFIX_2}" || return 1 + conda create --prefix "${CONDA_ENV_PREFIX_3}" || return 1 } function install_optional() { - install_star - if [[ "$?" -ne 0 ]]; then - echo "Error installing optional dependency: star" >&2 - fi - install_samtools - if [[ "$?" -ne 0 ]]; then - echo "Error installing optional dependency: samtools" >&2 - fi - install_rmats - if [[ "$?" -ne 0 ]]; then - echo "Error installing optional dependency: rmats" >&2 - fi - install_cufflinks - if [[ "$?" -ne 0 ]]; then - echo "Error installing optional dependency: cufflinks" >&2 - fi - install_seq2hla - if [[ "$?" -ne 0 ]]; then - echo "Error installing optional dependency: seq2hla" >&2 - fi install_ms_gf if [[ "$?" -ne 0 ]]; then echo "Error installing optional dependency: ms gf" >&2 @@ -301,17 +176,26 @@ function install() { local INSTALL_OPTIONAL="$1" ensure_conda_envs || return 1 - + install_python_packages "${INSTALL_OPTIONAL}" || return 1 install_iedb || return 1 - install_bedtools || return 1 if [[ "${INSTALL_OPTIONAL}" -ne 0 ]]; then install_optional || return 1 fi - install_python_packages || return 1 - install_iris_data || return 1 install_iris_package || return 1 + check_iris_data || return 1 + + echo "conda_wrapper: '${SCRIPT_DIR}/conda_wrapper'" \ + >> "${SCRIPT_DIR}/snakemake_config.yaml" || return 1 + echo "conda_env_2: '${SCRIPT_DIR}/conda_env_2'" \ + >> "${SCRIPT_DIR}/snakemake_config.yaml" || return 1 + echo "conda_env_3: '${SCRIPT_DIR}/conda_env_3'" \ + >> "${SCRIPT_DIR}/snakemake_config.yaml" || return 1 + echo "iris_data: '${SCRIPT_DIR}/IRIS_data'" \ + >> "${SCRIPT_DIR}/snakemake_config.yaml" || return 1 + echo "iedb_path: '${SCRIPT_DIR}/IEDB/mhc_i/src'" \ + >> "${SCRIPT_DIR}/snakemake_config.yaml" || return 1 } function display_usage() { @@ -341,10 +225,6 @@ function main() { fi source set_env_vars.sh || return 1 - source conda.sh || return 1 - - SCRIPT_DIR="$(pwd)" || return 1 - install "${INSTALL_OPTIONAL}" || return 1 } diff --git a/qsub/qsub.py b/qsub/qsub.py deleted file mode 100644 index b806f4f..0000000 --- a/qsub/qsub.py +++ /dev/null @@ -1,163 +0,0 @@ -import os - -import bs4 - - -class CommandResult(object): - def __init__(self): - self.status_code = None - self.out = None - self.err = None - - def from_completed_process(self, completed_process): - self.status_code = completed_process.returncode - self.out = completed_process.stdout - self.err = completed_process.stderr - - -class QsubJob(object): - """ - cmd_execute_func(cmd_tokens, exec_func_ref_data) -> CommandResult - - cmd_execute_func and log_error_func must be defined at the top level of a module so that - QsubJob can be pickled. - - The execute function allows either running a local qsub or qsub on a remote host. - """ - def __init__(self, - command_result, - job_name, - cmd_execute_func, - exec_func_ref_data=None, - out_dir=None, - log_error_func=print): - self.job_name = job_name - self._cmd_execute_func = cmd_execute_func - self._exec_func_ref_data = exec_func_ref_data - self._log_error_func = log_error_func - - self._status = 'running' - - self.j_id = _extract_qsub_j_id(command_result.out.decode()) - self._set_output_file_names(out_dir) - - def _set_output_file_names(self, out_dir): - if out_dir is None: - self.qsub_out = None - self.qsub_err = None - return - - out_f_name = '{}.o{}'.format(self.job_name, self.j_id) - err_f_name = '{}.e{}'.format(self.job_name, self.j_id) - self.qsub_out = os.path.join(out_dir, out_f_name) - self.qsub_err = os.path.join(out_dir, err_f_name) - - def _execute_cmd(self, cmd_tokens): - return self._cmd_execute_func(cmd_tokens, self._exec_func_ref_data) - - def get_status(self): - if self._status == 'finished': - return self._status - - if self.is_finished(): - self._status = 'finished' - return self._status - - return self._status - - def is_finished(self): - if not self.j_id: - self._log_error_func('no j_id when checking if qsub job is finished') - return False - - qstat_command_tokens = ['qstat', '-j', self.j_id, '-xml'] - qstat_process = self._execute_cmd(qstat_command_tokens) - if not qstat_process: - return None - - qstat_soup = _make_soup(qstat_process.out.decode()) - if _qstat_output_has_job_details(qstat_soup, self.j_id): - return False - - if _qstat_output_lists_job_as_unknown(qstat_soup, self.j_id): - return True - - self._log_error_func('unexpected output from: {}\n{}'.format(qstat_command_tokens, - qstat_soup)) - return True - - def get_stdout(self): - if self.qsub_out is None: - return None - - return self._cat_file(self.qsub_out) - - def get_stderr(self): - if self.qsub_err is None: - return None - - return self._cat_file(self.qsub_err) - - def _cat_file(self, f_name): - cat_command_tokens = ['cat', f_name] - process = self._execute_cmd(cat_command_tokens) - if not process: - return None - - return process.out - - -def _make_soup(output): - return bs4.BeautifulSoup(output, 'html.parser') - - -def _extract_qsub_j_id(out): - """ - Find j_id in out that looks like: - Your job {j_id}.{array_details} ("{job_name}") has been submitted - """ - tokens = out.split(' ') - for token in tokens: - if token and token[0].isdigit(): - return token.split('.')[0] - - return None - - -def _qstat_output_has_job_details(soup, j_id): - """ - Return True if j_id is a JB_job_number in soup that looks like: - - - - 11693229 - """ - djobs = soup.find_all('djob_info') - if len(djobs) != 1: - return False - - numbers = djobs[0].find_all('jb_job_number') - for number in numbers: - if number.string == j_id: - return True - - return False - - -def _qstat_output_lists_job_as_unknown(soup, j_id): - """ - Return True if j_id is an st_name in soup that looks like: - - - 123 - """ - unks = soup.find_all('unknown_jobs') - if len(unks) != 1: - return False - - names = unks[0].find_all('st_name') - for name in names: - if name.string == j_id: - return True - - return False diff --git a/qsub/requirements.txt b/qsub/requirements.txt deleted file mode 100644 index c1f5f71..0000000 --- a/qsub/requirements.txt +++ /dev/null @@ -1 +0,0 @@ -beautifulsoup4 diff --git a/qsub/submit_qsub_and_wait.py b/qsub/submit_qsub_and_wait.py deleted file mode 100644 index fe88c20..0000000 --- a/qsub/submit_qsub_and_wait.py +++ /dev/null @@ -1,63 +0,0 @@ -import argparse -import subprocess -import time - -import qsub - - -def local_subprocess_cmd_exec_func(tokens, _): - process = subprocess.run(tokens, stdout=subprocess.PIPE, stderr=subprocess.PIPE, check=False) - if process.returncode != 0: - print('returncode: {}, command: {}, stdout: {}, stderr: {}'.format( - process.returncode, tokens, process.stdout, process.stderr)) - return None - - command_result = qsub.CommandResult() - command_result.from_completed_process(process) - return command_result - - -def submit_job(cmd): - cmd = cmd.rstrip() - print('executing: {}'.format(cmd)) - - tokens = cmd.split(' ') - command_result = local_subprocess_cmd_exec_func(tokens, None) - if not command_result: - print('error submitting job') - return None - - job_name = '' - return qsub.QsubJob(command_result, job_name, local_subprocess_cmd_exec_func) - - -def main(): - parser = argparse.ArgumentParser( - description='execute qsub commands and wait for those jobs to complete') - parser.add_argument('command_file', type=str, help='a file with 1 qsub command per line') - parser.add_argument( - '--poll-interval-seconds', type=int, default=30, help='how frequently to check job status') - args = parser.parse_args() - - jobs = list() - with open(args.command_file, 'rt') as f_handle: - for cmd in f_handle: - job = submit_job(cmd) - if job: - jobs.append(job) - - while jobs: - time.sleep(args.poll_interval_seconds) - print('checking {} job(s)'.format(len(jobs))) - new_jobs = list() - for job in jobs: - if not job.is_finished(): - new_jobs.append(job) - else: - print('finished j_id: {}'.format(job.j_id)) - - jobs = new_jobs - - -if __name__ == '__main__': - main() diff --git a/qsub/test b/qsub/test deleted file mode 100755 index 83b0653..0000000 --- a/qsub/test +++ /dev/null @@ -1,21 +0,0 @@ -#!/bin/bash -# -# run tests - -function main() { - source ../set_env_vars.sh || return 1 - source ../conda.sh || return 1 - - conda::activate_env "${CONDA_ENV_NAME_3}" || return 1 - - if [[ "$#" -ne 0 ]]; then - echo "arguments given, but none expected" >&2 - return 1 - fi - - python test_submit_qsub_and_wait.py || return 1 - - conda::deactivate_env || return 1 -} - -main "$@" diff --git a/qsub/test_submit_qsub_and_wait.py b/qsub/test_submit_qsub_and_wait.py deleted file mode 100644 index 95d7b5d..0000000 --- a/qsub/test_submit_qsub_and_wait.py +++ /dev/null @@ -1,60 +0,0 @@ -import os -import sys -import tempfile -import time -import unittest - -import submit_qsub_and_wait - - -def _write_lines(f_name, lines): - with open(f_name, 'wt') as f_h: - for line in lines: - f_h.write('{}\n'.format(line)) - - -class TestSubmitQsubAndWait(unittest.TestCase): - def test(self): - temp_f_name_1 = None - temp_f_name_2 = None - try: - with tempfile.NamedTemporaryFile(delete=False) as temp_f_handle: - temp_f_name_1 = temp_f_handle.name - - with tempfile.NamedTemporaryFile(delete=False) as temp_f_handle: - temp_f_name_2 = temp_f_handle.name - - self._test(temp_f_name_1, temp_f_name_2, 1, 60) - self._test(temp_f_name_1, temp_f_name_2, 30, 90) - finally: - if temp_f_name_1 is not None: - os.remove(temp_f_name_1) - - if temp_f_name_2 is not None: - os.remove(temp_f_name_2) - - def _test(self, f_name_1, f_name_2, sleep_seconds, max_seconds): - lines_1 = [ - '#!/bin/bash', - 'sleep {}'.format(sleep_seconds), - ] - _write_lines(f_name_1, lines_1) - lines_2 = [ - 'qsub {}'.format(f_name_1), - 'qsub {}'.format(f_name_1), - ] - _write_lines(f_name_2, lines_2) - - sys.argv = ['submit_qsub_and_wait.py', f_name_2, '--poll-interval-seconds', '5'] - - begin = time.time() - submit_qsub_and_wait.main() - end = time.time() - - elapsed_seconds = end - begin - self.assertTrue(elapsed_seconds >= sleep_seconds) - self.assertTrue(elapsed_seconds <= max_seconds) - - -if __name__ == '__main__': - unittest.main(verbosity=2) diff --git a/requirements.txt b/requirements.txt deleted file mode 100644 index 0d7a59c..0000000 --- a/requirements.txt +++ /dev/null @@ -1,5 +0,0 @@ -numpy -scipy -seaborn -pyBigWig -statsmodels diff --git a/run b/run new file mode 100755 index 0000000..9496d76 --- /dev/null +++ b/run @@ -0,0 +1,3 @@ +#!/bin/bash + +./conda_wrapper ./conda_env_3 snakemake --profile ./snakemake_profile diff --git a/run_example b/run_example deleted file mode 100755 index 5a930a1..0000000 --- a/run_example +++ /dev/null @@ -1,23 +0,0 @@ -#!/bin/bash -# -# Run the IRIS pipeline for the inputs in example/ - -function main() { - if [[ "$#" -ne 0 ]]; then - echo "arguments given, but none expected" >&2 - return 1 - fi - - SCRIPT_DIR="$(pwd)" || return 1 - - USERNAME="example" - RMATS_MATRICES_TAR="${SCRIPT_DIR}/example/SJ_matrices.tar.gz" - RAW_SCREENING_PARAMS="${SCRIPT_DIR}/example/Test_simplified.para" - MHC_BY_SAMPLE="${SCRIPT_DIR}/example/HLA_types/hla_patient.tsv" - MHC_LIST="${SCRIPT_DIR}/example/HLA_types/hla_types.list" - - ./run_iris "${USERNAME}" "${RMATS_MATRICES_TAR}" "${RAW_SCREENING_PARAMS}"\ - "${MHC_BY_SAMPLE}" "${MHC_LIST}" || return 1 -} - -main "$@" diff --git a/run_iris b/run_iris deleted file mode 100755 index 139dbd3..0000000 --- a/run_iris +++ /dev/null @@ -1,238 +0,0 @@ -#!/bin/bash -# -# run the iris pipeline - -function parse_arguments() { - if [[ "$#" -ne 5 ]]; then - echo " -usage: - ./run_iris USERNAME RMATS_MATRICES_TAR SCREENING_PARAMS MHC_BY_SAMPLE MHC_LIST - -example: - ./run_iris user matrices.tar.gz job.para hla_patient.tsv hla_types.list -" - return 1 - fi - - USERNAME="$1" - RMATS_MATRICES_TAR="$2" - RAW_SCREENING_PARAMS="$3" - MHC_BY_SAMPLE="$4" - MHC_LIST="$5" -} - -function get_line_of_file() { - local LINE_NUM="$1" - local FILE_NAME="$2" - local LINE="$(head -n ${LINE_NUM} ${FILE_NAME} | tail -n 1)" || return 1 - echo "${LINE}" -} - -function pre_process_inputs() { - local RAW_DATA_SET_NAME="$(get_line_of_file 1 ${RAW_SCREENING_PARAMS})" || return 1 - local FILTER_1="$(get_line_of_file 2 ${RAW_SCREENING_PARAMS})" || return 1 - local FILTER_2="$(get_line_of_file 3 ${RAW_SCREENING_PARAMS})" || return 1 - local FILTER_3="$(get_line_of_file 4 ${RAW_SCREENING_PARAMS})" || return 1 - local TEST_MODE="$(get_line_of_file 5 ${RAW_SCREENING_PARAMS})" || return 1 - local USE_RATIO="$(get_line_of_file 6 ${RAW_SCREENING_PARAMS})" || return 1 - - # The leading underscore is used to distinguish user data sets from reference data sets - DATA_SET_NAME="_${USERNAME}_${RAW_DATA_SET_NAME}" - - RESULT_DIR="${SCRIPT_DIR}/results/${USERNAME}/${RAW_DATA_SET_NAME}" - mkdir -p "${RESULT_DIR}" || return 1 - SCREENING_OUT_DIR="${RESULT_DIR}/screening" - mkdir -p "${SCREENING_OUT_DIR}" || return 1 - RUN_DIR="${RESULT_DIR}/temp" - mkdir -p "${RUN_DIR}" || return 1 - - local IRIS_DATA="${SCRIPT_DIR}/IRIS_data" - IRIS_DB="${IRIS_DATA}/db/" - local IRIS_RESOURCES="${IRIS_DATA}/resources" - MAPPABILITY_PATH="${IRIS_RESOURCES}/mappability/wgEncodeCrgMapabilityAlign24mer.bigWig" - REF_GENOME="${IRIS_RESOURCES}/reference/ucsc.hg19.fasta" - - # create new param file - SCREENING_PARAMS="${RESULT_DIR}/job.para" - echo "${DATA_SET_NAME}" > "${SCREENING_PARAMS}" || return 1 - echo "${IRIS_DB}" >> "${SCREENING_PARAMS}" || return 1 - echo "${FILTER_1}" >> "${SCREENING_PARAMS}" || return 1 - echo "${FILTER_2}" >> "${SCREENING_PARAMS}" || return 1 - echo "${FILTER_3}" >> "${SCREENING_PARAMS}" || return 1 - echo "${TEST_MODE}" >> "${SCREENING_PARAMS}" || return 1 - echo "${USE_RATIO}" >> "${SCREENING_PARAMS}" || return 1 - local BLACKLIST_PATH="" - echo "${BLACKLIST_PATH}" >> "${SCREENING_PARAMS}" || return 1 - echo "${MAPPABILITY_PATH}" >> "${SCREENING_PARAMS}" || return 1 - echo "${REF_GENOME}" >> "${SCREENING_PARAMS}" || return 1 - - # update matrices files to use absolute paths - cd "${RESULT_DIR}" || return 1 - tar -xf "${RMATS_MATRICES_TAR}" || return 1 - - local MATRICES_PATH="${RESULT_DIR}/SJ_matrices" - cd "${MATRICES_PATH}" || return 1 - - local TEMP_F_NAME="$(mktemp)" || return 1 - local FILE_NAME - for FILE_NAME in matrices.txt samples.txt; do - mv "${FILE_NAME}" "${TEMP_F_NAME}" || return 1 - local LINE - while read LINE; do - echo "${MATRICES_PATH}/${LINE}" >> "${FILE_NAME}" || return 1 - done < "${TEMP_F_NAME}" - done - rm "${TEMP_F_NAME}" || return 1 - - cd "${SCRIPT_DIR}" || return 1 - - RMATS_MAT_PATH_MANIFEST="${MATRICES_PATH}/matrices.txt" - RMATS_SAMPLE_ORDER="${MATRICES_PATH}/samples.txt" -} - -function formatting_step() { - echo - echo "formatting" - - cd "${RUN_DIR}" || return 1 - - local SAMPLE_NAME_FIELD='2' - local SPLICING_EVENT_TYPE='SE' - - IRIS formatting "${RMATS_MAT_PATH_MANIFEST}" "${RMATS_SAMPLE_ORDER}" -s "${SAMPLE_NAME_FIELD}"\ - -d "${IRIS_DB}" -t "${SPLICING_EVENT_TYPE}" -n "${DATA_SET_NAME}" || return 1 -} - -function screening_step() { - echo - echo "screening" - - cd "${RUN_DIR}" || return 1 - - IRIS screening "${SCREENING_PARAMS}" -t -o "${SCREENING_OUT_DIR}" || return 1 -} - -function find_and_execute_qsub_commands_in_file() { - local IN_FILE="$1" - - local QSUB_CMDS=() - local LINE - while read LINE; do - local GREP_RES="$(echo ${LINE} | grep '^qsub.*\.sh$')" - if [[ -n "${GREP_RES}" ]]; then - QSUB_CMDS+=("${LINE}") - fi - done < "${IN_FILE}" - - if [[ "${#QSUB_CMDS[@]}" == 0 ]]; then - echo "could not find any qsub commands" - return 1 - fi - - echo - echo "executing qsub commands" - - local SUBMIT_AND_WAIT_PY="${SCRIPT_DIR}/qsub/submit_qsub_and_wait.py" - local TEMP_F_NAME="$(mktemp)" || return 1 - - for QSUB_CMD in "${QSUB_CMDS[@]}"; do - echo "${QSUB_CMD}" >> "${TEMP_F_NAME}" - done - - echo "execute: ${PYTHON_3_EXECUTABLE} ${SUBMIT_AND_WAIT_PY}" - echo "with qsub commands:" - cat "${TEMP_F_NAME}" - - "${PYTHON_3_EXECUTABLE}" "${SUBMIT_AND_WAIT_PY}" "${TEMP_F_NAME}" || return 1 - - rm "${TEMP_F_NAME}" || return 1 -} - -function prediction_step() { - echo - echo "prediction" - - cd "${RUN_DIR}" || return 1 - - local TEMP_F_NAME="$(mktemp)" || return 1 - - local DELTA_PSI_COLUMN='5' - local IEDB_DIR="${SCRIPT_DIR}/IEDB/mhc_i/src" - - # TODO --iedb-local should be required=True - IRIS prediction "${SCREENING_OUT_DIR}" -c "${DELTA_PSI_COLUMN}" -m "${MHC_LIST}"\ - -p "${SCREENING_PARAMS}" --iedb-local "${IEDB_DIR}"\ - > "${TEMP_F_NAME}" || return 1 - - cat "${TEMP_F_NAME}" || return 1 - - find_and_execute_qsub_commands_in_file "${TEMP_F_NAME}" || return 1 - rm "${TEMP_F_NAME}" || return 1 -} - -function epitope_post_step() { - echo - echo "epitope_post" - - cd "${RUN_DIR}" || return 1 - - # TODO -e is actually not required? - IRIS epitope_post -p "${SCREENING_PARAMS}" -o "${SCREENING_OUT_DIR}"\ - -m "${MHC_BY_SAMPLE}" || return 1 -} - -function screening_plot_step() { - echo - echo "screening_plot" - - cd "${RUN_DIR}" || return 1 - - local IN_PREFIX="${SCREENING_OUT_DIR}/${DATA_SET_NAME}" - local PRIMARY_IN="${IN_PREFIX}.primary.txt" - local PRIORITIZED_IN="${IN_PREFIX}.prioritized.txt" - - local OUT_PREFIX="${RESULT_DIR}/violin" - local PRIMARY_OUT="${OUT_PREFIX}_primary" - local PRIORITIZED_OUT="${OUT_PREFIX}_prioritized" - - cut -f 1 "${PRIMARY_IN}" | tail -n +2 > "${PRIMARY_OUT}" || return 1 - cut -f 1 "${PRIORITIZED_IN}" | tail -n +2 > "${PRIORITIZED_OUT}" || return 1 - - IRIS screening_plot "${PRIMARY_OUT}" -p "${SCREENING_PARAMS}" || return 1 - IRIS screening_plot "${PRIORITIZED_OUT}" -p "${SCREENING_PARAMS}" || return 1 -} - -function set_python3_executable() { - # Need to use Python3 for submit_qsub_and_wait.py. - # Also need to have the Python2 conda environment to run IRIS. - # Get the Python3 path and then go back to the Python2 environment - conda::activate_env "${CONDA_ENV_NAME_3}" || return 1 - - PYTHON_3_EXECUTABLE="$(which python)" || return 1 - - conda::deactivate_env || return 1 -} - -function main() { - source set_env_vars.sh || return 1 - source conda.sh || return 1 - - SCRIPT_DIR="$(pwd)" - export PATH="${PATH}:${SCRIPT_DIR}/bedtools/bedtools2/bin" - - set_python3_executable || return 1 - parse_arguments "$@" || return 1 - pre_process_inputs || return 1 - - conda::activate_env "${CONDA_ENV_NAME_2}" || return 1 - - formatting_step || return 1 - screening_step || return 1 - prediction_step || return 1 - epitope_post_step || return 1 - screening_plot_step || return 1 - - conda::deactivate_env || return 1 -} - -main "$@" diff --git a/scripts/check_read_lengths.py b/scripts/check_read_lengths.py new file mode 100644 index 0000000..254c168 --- /dev/null +++ b/scripts/check_read_lengths.py @@ -0,0 +1,57 @@ +import argparse +import os +import os.path + + +def parse_args(): + parser = argparse.ArgumentParser( + description=('determine the set of read lengths based on' + ' the rmats output file names')) + parser.add_argument( + '--parent-dir', + required=True, + help='path of directory which contains 1 directory per read length') + parser.add_argument('--run-name', + required=True, + help='prefix used to name output files') + parser.add_argument('--out', + required=True, + help='path to write read lengths') + + args = parser.parse_args() + return args + + +def check_read_lengths(parent_dir, run_name, out): + file_names = os.listdir(parent_dir) + prefix = '{}.RL'.format(run_name) + read_lengths = list() + for file_name in file_names: + file_path = os.path.join(parent_dir, file_name) + if not (os.path.isdir(file_path) and file_name.startswith(prefix)): + continue + + suffix = file_name[len(prefix):] + try: + read_length = int(suffix) + except ValueError: + print('ignoring: {}'.format(file_path)) + continue + + read_lengths.append(suffix) + + if not read_lengths: + raise Exception('no read lengths found in {}'.format(parent_dir)) + + with open(out, 'wt') as out_handle: + for read_length in read_lengths: + out_handle.write('{}\n'.format(read_length)) + + +def main(): + args = parse_args() + check_read_lengths(args.parent_dir, args.run_name, args.out) + + +if __name__ == '__main__': + main() diff --git a/scripts/count_iris_predict_tasks.py b/scripts/count_iris_predict_tasks.py new file mode 100644 index 0000000..caf34f5 --- /dev/null +++ b/scripts/count_iris_predict_tasks.py @@ -0,0 +1,62 @@ +import argparse +import os +import os.path + + +def parse_args(): + parser = argparse.ArgumentParser( + description=('find which tasks were created by IRIS predict')) + parser.add_argument('--out-list', + required=True, + help='path to write a file listing all created tasks') + parser.add_argument( + '--task-dir', + required=True, + help='directory where task files are expected to be found') + parser.add_argument( + '--splice-type', + required=True, + help='alternative splicing event type (expected in task file name)') + + args = parser.parse_args() + return args + + +def count_iris_predict_tasks(out_list, task_dir, splice_type): + file_names = os.listdir(task_dir) + task_paths = list() + base_prefix = 'pep2epitope_{}.'.format(splice_type) + tiers = ['tier1', 'tier2tier3'] + prefixes = ['{}{}.'.format(base_prefix, tier) for tier in tiers] + suffix = '.sh' + for name in file_names: + file_path = os.path.join(task_dir, name) + if not name.endswith(suffix): + continue + + for prefix in prefixes: + if name.startswith(prefix): + number_string = name[len(prefix):-len(suffix)] + try: + int(number_string) + except ValueError: + raise Exception('unexpected file: {}'.format(file_path)) + + task_paths.append(file_path) + continue + + if not task_paths: + raise Exception('could not find any predict tasks') + + with open(out_list, 'wt') as out_handle: + for task_path in task_paths: + out_handle.write('{}\n'.format(task_path)) + + +def main(): + args = parse_args() + count_iris_predict_tasks(args.out_list, args.task_dir, args.splice_type) + + +if __name__ == '__main__': + main() diff --git a/scripts/prepare_iris_exp_matrix.py b/scripts/prepare_iris_exp_matrix.py new file mode 100644 index 0000000..bfc0b93 --- /dev/null +++ b/scripts/prepare_iris_exp_matrix.py @@ -0,0 +1,31 @@ +import argparse + + +def parse_args(): + parser = argparse.ArgumentParser( + description=('write input file for IRIS exp_matrix')) + parser.add_argument('--out-manifest', + required=True, + help='path to write the list of gene expression files') + parser.add_argument('--fpkm-files', + required=True, + nargs='+', + help='the fpkm files from cufflinks') + + args = parser.parse_args() + return args + + +def prepare_iris_exp_matrix(out_manifest, fpkm_files): + with open(out_manifest, 'wt') as out_handle: + for file_name in fpkm_files: + out_handle.write('{}\n'.format(file_name)) + + +def main(): + args = parse_args() + prepare_iris_exp_matrix(args.out_manifest, args.fpkm_files) + + +if __name__ == '__main__': + main() diff --git a/scripts/prepare_iris_format.py b/scripts/prepare_iris_format.py new file mode 100644 index 0000000..5c2c4bc --- /dev/null +++ b/scripts/prepare_iris_format.py @@ -0,0 +1,57 @@ +import argparse +import os +import os.path + + +def parse_args(): + parser = argparse.ArgumentParser( + description=('write input files for IRIS format')) + parser.add_argument('--matrix-out', + required=True, + help='path to write the list of matrix directories') + parser.add_argument('--sample-out', + required=True, + help='path to write the list of BAM lists') + parser.add_argument('--summaries', + required=True, + nargs='+', + help='the summary files from the matrix directories') + + args = parser.parse_args() + return args + + +def prepare_iris_format(matrix_out, sample_out, summaries): + with open(matrix_out, 'wt') as matrix_out_handle: + with open(sample_out, 'wt') as sample_out_handle: + prepare_iris_format_with_handles(matrix_out_handle, + sample_out_handle, summaries) + + +def prepare_iris_format_with_handles(matrix_out_handle, sample_out_handle, + summaries): + for summary in summaries: + matrix_dir_path = os.path.dirname(summary) + matrix_dir_name = os.path.basename(matrix_dir_path) + matrix_dir_name_suffix = '.matrix' + if not matrix_dir_name.endswith(matrix_dir_name_suffix): + raise Exception('unexpected directory name for {}'.format(summary)) + + matrix_dir_name_prefix = matrix_dir_name[:-len(matrix_dir_name_suffix)] + matrix_dir_parent_dir_path = os.path.dirname(matrix_dir_path) + sample_list_name = '{}_rmatspost_list.txt'.format( + matrix_dir_name_prefix) + sample_path = os.path.join(matrix_dir_parent_dir_path, + sample_list_name) + + matrix_out_handle.write('{}\n'.format(matrix_dir_path)) + sample_out_handle.write('{}\n'.format(sample_path)) + + +def main(): + args = parse_args() + prepare_iris_format(args.matrix_out, args.sample_out, args.summaries) + + +if __name__ == '__main__': + main() diff --git a/scripts/prepare_iris_sjc_matrix.py b/scripts/prepare_iris_sjc_matrix.py new file mode 100644 index 0000000..73eec09 --- /dev/null +++ b/scripts/prepare_iris_sjc_matrix.py @@ -0,0 +1,31 @@ +import argparse + + +def parse_args(): + parser = argparse.ArgumentParser( + description=('write input file for IRIS sjc_matrix')) + parser.add_argument('--sj-out', + required=True, + help='path to write the list of SJ count files') + parser.add_argument('--sj-files', + required=True, + nargs='+', + help='the SJ count files from extract_sjc') + + args = parser.parse_args() + return args + + +def prepare_iris_sjc_matrix(sj_out, sj_files): + with open(sj_out, 'wt') as out_handle: + for file_name in sj_files: + out_handle.write('{}\n'.format(file_name)) + + +def main(): + args = parse_args() + prepare_iris_sjc_matrix(args.sj_out, args.sj_files) + + +if __name__ == '__main__': + main() diff --git a/scripts/write_param_file.py b/scripts/write_param_file.py new file mode 100644 index 0000000..9c237ba --- /dev/null +++ b/scripts/write_param_file.py @@ -0,0 +1,231 @@ +import argparse +import os +import os.path + + +def parse_args(): + parser = argparse.ArgumentParser( + description=('write the parameter file for IRIS screen')) + parser.add_argument('--out-path', + required=True, + help='path to write parameter file') + parser.add_argument('--group-name', + required=True, + help='name to use for sub directory in IRIS_data/db/') + parser.add_argument('--iris-db', + required=True, + help='/path/to/IRIS_data/db') + parser.add_argument( + '--psi-p-value-cutoffs', + required=True, + help=('comma separated p-value cutoffs for PSI-based statistical tests' + ' (tissue-matched normal, tumor, normal)')) + parser.add_argument( + '--sjc-p-value-cutoffs', + required=True, + help=('comma separated p-value cutoffs for SJC-based statistical tests' + ' (tissue-matched normal, tumor, normal)')) + parser.add_argument('--delta-psi-cutoffs', + required=True, + help=('comma separated minimum required delta PSIs' + ' (tissue-matched normal, tumor, normal)')) + parser.add_argument('--fold-change-cutoffs', + required=True, + help=('comma separated minimum required fold changes' + ' (tissue-matched normal, tumor, normal)')) + parser.add_argument( + '--group-count-cutoffs', + required=True, + help=('comma separated minimum counts of reference groups that' + ' need to meet other requirements' + ' (tissue-matched normal, tumor, normal)')) + parser.add_argument( + '--reference-names-tissue-matched-normal', + required=True, + help='comma separated reference groups for tissue-matched normal') + parser.add_argument('--reference-names-tumor', + required=True, + help='comma separated reference groups for tumor') + parser.add_argument('--reference-names-normal', + required=True, + help='comma separated reference groups for normal') + parser.add_argument('--comparison-mode', + required=True, + choices=['group', 'individual'], + help=('mode for statistical test' + ' (group requires at least 2 input samples)')) + parser.add_argument('--statistical-test-type', + required=True, + choices=['parametric', 'nonparametric'], + help='type of statistical test') + parser.add_argument('--use-ratio', + action='store_true', + help='use ratio instead of count for group cutoffs') + parser.add_argument('--blacklist-file', help='list of AS events to remove') + parser.add_argument('--mapability-bigwig', + help='allows evaluatio of splice region mapability') + parser.add_argument('--reference-genome', + help='required for IRIS translate') + + args = parser.parse_args() + check_file_exists(args.blacklist_file, parser) + check_file_exists(args.mapability_bigwig, parser) + check_file_exists(args.reference_genome, parser) + + args.psi_p_value_cutoffs = parse_floats(args.psi_p_value_cutoffs) + args.sjc_p_value_cutoffs = parse_floats(args.sjc_p_value_cutoffs) + args.delta_psi_cutoffs = parse_floats(args.delta_psi_cutoffs) + args.fold_change_cutoffs = parse_floats(args.fold_change_cutoffs) + args.group_count_cutoffs = parse_floats(args.group_count_cutoffs) + + if not (1 <= len(args.psi_p_value_cutoffs) <= 3): + parser.error('must give 1 to 3 cutoffs') + + expected_len = len(args.psi_p_value_cutoffs) + expected_non_none = [x is not None for x in args.psi_p_value_cutoffs] + if not (expected_non_none[0] or + ((expected_len == 3) and expected_non_none[2])): + parser.error('must provide values for at least one of' + ' tissue-matched-normal or normal') + + for name, values in [('sjc_p_value_cutoffs', args.sjc_p_value_cutoffs), + ('delta_psi_cutoffs', args.delta_psi_cutoffs), + ('fold_change_cutoffs', args.fold_change_cutoffs), + ('group_count_cutoffs', args.group_count_cutoffs)]: + if len(values) != expected_len: + parser.error('{} has len {}, but expected {}'.format( + name, len(values), expected_len)) + + for i, value in enumerate(values): + is_non_none = value is not None + if is_non_none != expected_non_none[i]: + expected = 'non-None' if expected_non_none[i] else 'None' + parser.error('{} value {} was {} when {} was expected'.format( + name, i, value, expected)) + + db_names = get_db_names(args.iris_db, parser) + args.reference_names_tissue_matched_normal = parse_reference_names( + args.reference_names_tissue_matched_normal, args.group_count_cutoffs, + 0, 'tissue-matched-normal', db_names, args.use_ratio, parser) + args.reference_names_tumor = parse_reference_names( + args.reference_names_tumor, args.group_count_cutoffs, 1, 'tumor', + db_names, args.use_ratio, parser) + args.reference_names_normal = parse_reference_names( + args.reference_names_normal, args.group_count_cutoffs, 2, 'normal', + db_names, args.use_ratio, parser) + + return args + + +def get_db_names(db_path, parser): + if not os.path.exists(db_path): + parser.error('{} does not exist'.format(db_path)) + if not os.path.isdir(db_path): + parser.error('{} is not a directory'.format(db_path)) + + db_names = list() + dir_entries = os.listdir(db_path) + for db_name in dir_entries: + full_path = os.path.join(db_path, db_name) + if os.path.isdir(full_path): + db_names.append(db_name) + + return db_names + + +def parse_floats(floats_str): + parts = floats_str.split(',') + stripped = [x.strip() for x in parts] + results = list() + for string in stripped: + if len(string) != 0: + float_value = float(string) + results.append(float_value) + else: + results.append(None) + + return results + + +def parse_reference_names(names_str, group_cutoffs, cutoff_index, group_name, + db_names, use_ratio, parser): + if len(group_cutoffs) <= cutoff_index: + parser.error( + 'missing list of reference groups for {}'.format(group_name)) + + cutoff = group_cutoffs[cutoff_index] + parts = names_str.split(',') + stripped = [x.strip() for x in parts] + if cutoff is None: + return list() # this group is skipped + + if use_ratio: + if not (0 <= cutoff <= 1): + parser.error('cutoff for {} was {} with use_ratio'.format( + group_name, cutoff)) + elif cutoff > len(stripped): + parser.error('{} cutoff is {}, but only {} references'.format( + group_name, cutoff, len(stripped))) + + for name in stripped: + if name not in db_names: + parser.error('reference {} in {} not found in db/'.format( + name, group_name)) + + return stripped + + +def check_file_exists(file_path, parser): + if file_path is None: + return + + if not os.path.isfile(file_path): + parser.error('{} does not exist'.format(file_path)) + + +def write_file_line_or_empty_line(out_handle, maybe_file): + if maybe_file: + out_handle.write('{}\n'.format(maybe_file)) + else: + out_handle.write('\n') + + +def write_param_file(args): + with open(args.out_path, 'wt') as out_handle: + out_handle.write('{}\n'.format(args.group_name)) + abs_db_path = os.path.abspath(args.iris_db) + out_handle.write('{}\n'.format(abs_db_path)) + references_by_i = [ + args.reference_names_tissue_matched_normal, + args.reference_names_tumor, args.reference_names_normal + ] + for i, psi_cutoff in enumerate(args.psi_p_value_cutoffs): + if psi_cutoff is None: + out_handle.write('\n') + continue + + cutoffs = [ + psi_cutoff, args.delta_psi_cutoffs[i], + args.fold_change_cutoffs[i], args.sjc_p_value_cutoffs[i], + args.group_count_cutoffs[i] + ] + cutoffs = [str(x) for x in cutoffs] + references = references_by_i[i] + out_handle.write('{} {}\n'.format(','.join(cutoffs), + ','.join(references))) + + out_handle.write('{} {}\n'.format(args.comparison_mode, + args.statistical_test_type)) + out_handle.write('{}\n'.format('True' if args.use_ratio else 'False')) + write_file_line_or_empty_line(out_handle, args.blacklist_file) + write_file_line_or_empty_line(out_handle, args.mapability_bigwig) + write_file_line_or_empty_line(out_handle, args.reference_genome) + + +def main(): + args = parse_args() + write_param_file(args) + + +if __name__ == '__main__': + main() diff --git a/set_env_vars.sh b/set_env_vars.sh index 197d0e4..097032e 100644 --- a/set_env_vars.sh +++ b/set_env_vars.sh @@ -1,9 +1,24 @@ #!/bin/bash # # Set environment variables used by other scripts +# +function set_conda_env_prefixes() { + local ORIG_DIR="$(pwd)" || return 1 + + local REL_SCRIPT_DIR="$(dirname ${BASH_SOURCE[0]})" || return 1 + cd "${REL_SCRIPT_DIR}" || return 1 + SCRIPT_DIR="$(pwd)" || return 1 + + cd "${ORIG_DIR}" || return 1 + + CONDA_ENV_PREFIX_2="${SCRIPT_DIR}/conda_env_2" + CONDA_ENV_PREFIX_3="${SCRIPT_DIR}/conda_env_3" +} -CONDA_ENV_NAME_2="iris-2" -CONDA_PYTHON_VERSION_2="2.7" +function main() { + # need to use the setup that conda init writes to .bashrc + source "${HOME}/.bashrc" || return 1 + set_conda_env_prefixes || return 1 +} -CONDA_ENV_NAME_3="iris-3" -CONDA_PYTHON_VERSION_3="3.6" +main "$@" diff --git a/setup.py b/setup.py index e08f81a..ab8f1c0 100644 --- a/setup.py +++ b/setup.py @@ -7,7 +7,7 @@ def main(): setup( name='IRIS', - version='1.0.0', + version='2.0.0', description='Isoform peptides from RNA splicing for Immunotherapy target Screening', @@ -24,7 +24,7 @@ def main(): include_package_data=True, package_data={'IRIS.data':[ - 'brain_blacklistMay.txt', + 'blacklist.brain_2020.txt', 'features.uniprot2gtf.ExtraCell.txt', 'UniprotENSGmap.txt', 'uniprot2gtf.blastout.uniprotAll.txt', diff --git a/snakemake_config.yaml b/snakemake_config.yaml new file mode 100644 index 0000000..faa1b2e --- /dev/null +++ b/snakemake_config.yaml @@ -0,0 +1,111 @@ +# Resource allocation +create_star_index_threads: 4 +create_star_index_mem_gb: 40 +create_star_index_time_hr: 12 +iris_append_sjc_mem_gb: 8 +iris_append_sjc_time_hr: 24 +# TODO 16 threads hardcoded in iris process_rnaseq +iris_cuff_task_threads: 8 +iris_cuff_task_mem_gb: 8 +iris_cuff_task_time_hr: 12 +iris_epitope_post_mem_gb: 8 +iris_epitope_post_time_hr: 12 +iris_exp_matrix_mem_gb: 8 +iris_exp_matrix_time_hr: 12 +iris_extract_sjc_task_mem_gb: 8 +iris_extract_sjc_task_time_hr: 12 +iris_format_mem_gb: 8 +iris_format_time_hr: 12 +# TODO seq2HLA defaults to 6 threads since IRIS does not supply the -p argument +iris_hla_task_threads: 6 +iris_hla_task_mem_gb: 8 +iris_hla_task_time_hr: 12 +iris_parse_hla_mem_gb: 8 +iris_parse_hla_time_hr: 12 +iris_predict_mem_gb: 8 +iris_predict_time_hr: 12 +iris_predict_task_mem_gb: 8 +iris_predict_task_time_hr: 12 +# TODO 8 hardcoded in makesubsh_rmats +iris_rmats_task_threads: 8 +iris_rmats_task_mem_gb: 8 +iris_rmats_task_time_hr: 12 +# TODO 8 hardcoded in makesubsh_rmatspost +iris_rmatspost_task_threads: 8 +iris_rmatspost_task_mem_gb: 8 +iris_rmatspost_task_time_hr: 12 +iris_screen_mem_gb: 8 +iris_screen_time_hr: 12 +iris_screen_sjc_mem_gb: 8 +iris_screen_sjc_time_hr: 12 +iris_sjc_matrix_mem_gb: 8 +iris_sjc_matrix_time_hr: 12 +# TODO 6 threads hardcoded in iris process_rnaseq +iris_star_task_threads: 6 +iris_star_task_mem_gb: 40 +iris_star_task_time_hr: 12 +iris_visual_summary_mem_gb: 8 +iris_visual_summary_time_hr: 12 +# Command options +run_core_modules: true +# run_all_modules toggles which rules can be run by +# conditionally adding UNSATISFIABLE_INPUT to certain rules. +run_all_modules: false +should_run_sjc_steps: true +star_sjdb_overhang: 100 +run_name: 'NEPC_test' # used to name output files +splice_event_type: 'SE' # one of [SE, RI,A3SS, A5SS] +comparison_mode: 'group' # group or individual +stat_test_type: 'parametric' # parametric or nonparametric +use_ratio: false +tissue_matched_normal_psi_p_value_cutoff: '' +tissue_matched_normal_sjc_p_value_cutoff: '' +tissue_matched_normal_delta_psi_p_value_cutoff: '' +tissue_matched_normal_fold_change_cutoff: '' +tissue_matched_normal_group_count_cutoff: '' +tissue_matched_normal_reference_group_names: '' +tumor_psi_p_value_cutoff: '' +tumor_sjc_p_value_cutoff: '' +tumor_delta_psi_p_value_cutoff: '' +tumor_fold_change_cutoff: '' +tumor_group_count_cutoff: '' +tumor_reference_group_names: '' +normal_psi_p_value_cutoff: '0.01' +normal_sjc_p_value_cutoff: '0.000001' +normal_delta_psi_p_value_cutoff: '0.05' +normal_fold_change_cutoff: '1' +normal_group_count_cutoff: '8' +normal_reference_group_names: 'GTEx_Heart,GTEx_Blood,GTEx_Lung,GTEx_Liver,GTEx_Brain,GTEx_Nerve,GTEx_Muscle,GTEx_Spleen,GTEx_Thyroid,GTEx_Skin,GTEx_Kidney' +# Input files +# sample_fastqs are not needed when just running the core modules +# sample_fastqs: +# sample_name_1: +# - '/path/to/sample_1_read_1.fq' +# - '/path/to/sample_1_read_2.fq' +# sample_name_2: +# - '/path/to/sample_2_read_1.fq' +# - '/path/to/sample_2_read_2.fq' +blacklist: '' +mapability_bigwig: '/path/to/IRIS_data/resources/mappability/wgEncodeCrgMapabilityAlign24mer.bigWig' +mhc_list: '/path/to/example/hla_types_test.list' +mhc_by_sample: '/path/to/example/hla_patient_test.tsv' +gene_exp_matrix: '' +splice_matrix_txt: '/path/to/example/splicing_matrix/splicing_matrix.SE.cov10.NEPC_example.txt' +splice_matrix_idx: '/path/to/example/splicing_matrix/splicing_matrix.SE.cov10.NEPC_example.txt.idx' +sjc_count_txt: '/path/to/example/sjc_matrix/SJ_count.NEPC_example.txt' +sjc_count_idx: '/path/to/example/sjc_matrix/SJ_count.NEPC_example.txt.idx' +# Reference files +gtf_name: 'gencode.v26lift37.annotation.gtf' +fasta_name: 'ucsc.hg19.fasta' +reference_files: + gencode.v26lift37.annotation.gtf.gz: + url: 'ftp://ftp.ebi.ac.uk/pub/databases/gencode/Gencode_human/release_26/GRCh37_mapping/gencode.v26lift37.annotation.gtf.gz' + ucsc.hg19.fasta.gz: + url: 'http://hgdownload.soe.ucsc.edu/goldenPath/hg19/bigZips/hg19.fa.gz' +# Additional configuration +# rmats_path: '/path/to/conda_env_2/bin/rmats.py' # should be written by ./install +# conda_wrapper: '/path/to/conda_wrapper' # should be written by ./install +# conda_env_2: '/path/to/conda_env_2' # should be written by ./install +# conda_env_3: '/path/to/conda_env_3' # should be written by ./install +# iris_data: '/path/to/IRIS_data # should be written by ./install +# iedb_path: '/path/to/IEDB/mhc_i/src' # should be written by ./install diff --git a/snakemake_profile/.gitignore b/snakemake_profile/.gitignore new file mode 100644 index 0000000..6fdf14e --- /dev/null +++ b/snakemake_profile/.gitignore @@ -0,0 +1 @@ +/job_resource_usage/ diff --git a/snakemake_profile/cluster_commands.py b/snakemake_profile/cluster_commands.py new file mode 100644 index 0000000..f89f34e --- /dev/null +++ b/snakemake_profile/cluster_commands.py @@ -0,0 +1,384 @@ +import os +import os.path + + +def submit_command(log_out, log_err, threads, time_hours, mem_mb, + mem_mb_per_thread, gpus, gpu_name, jobscript): + # sbatch requires that the directories for the log files already exist + for log_path in [log_out, log_err]: + log_dir = os.path.dirname(log_path) + if log_dir != '': + os.makedirs(log_dir, exist_ok=True) + + command = ['sbatch', '-o', log_out, '-e', log_err] + if threads: + command.extend(['-c', str(threads)]) + + if time_hours: + days, hours_float = divmod(time_hours, 24) + hours_whole, hours_part = divmod(hours_float, 1) + minutes = hours_part * 60 + time_str = '{}-{}:{}'.format(int(days), int(hours_whole), int(minutes)) + command.extend(['--time', time_str]) + + if mem_mb: + command.append('--mem={}M'.format(mem_mb)) + + if gpus: + gres_argument_base = '--gres=gpu' + if gpu_name: + gres_argument = '{}:{}:{}'.format(gres_argument_base, gpu_name, + gpus) + else: + gres_argument = '{}:{}'.format(gres_argument_base, gpus) + + command.extend(['-p', 'gpuq', gres_argument]) + + command.append(jobscript) + return command + + +def try_extract_job_id_from_submit_output(stdout): + tokens = stdout.split() + if len(tokens) < 4: + return None, 'expected at least 4 tokens' + + if (((tokens[0] != 'Submitted') or (tokens[1] != 'batch') + or (tokens[2] != 'job'))): + return None, 'expected output to look like "Submitted batch job ..."' + + try: + job_id = int(tokens[3]) + except ValueError as e: + return None, 'could not parse {} as an int: {}'.format(tokens[3], e) + + return job_id, None + + +def status_command(job_id): + status_fields = [ + 'ElapsedRaw', + 'End', + 'ExitCode', + 'JobIDRaw', + 'MaxDiskRead', + 'MaxDiskWrite', + 'MaxRss', + 'MaxVMSize', + 'Start', + 'State', + 'Submit', + 'TotalCPU', + ] + return [ + 'sacct', '--parsable', '-j', job_id, + '--format={}'.format(','.join(status_fields)) + ] + + +def try_extract_job_info_from_status_output(stdout, job_id): + rows, error = _parse_rows(stdout) + if error: + return None, error + + if not rows: + # the output may be empty if the job was submitted very recently + return {'status': 'running', 'resource_usage': None}, None + + parent_rows = list() + batch_rows = list() + other_rows = list() + for row in rows: + row_job_id, row_job_step = _get_job_id_and_step(row.get('JobIDRaw')) + if row_job_id != job_id: + continue + + if row_job_step is None: + parent_rows.append(row) + elif row_job_step == 'batch': + batch_rows.append(row) + else: + other_rows.append(row) + + usage = { + 'cpu': None, + 'end_time': None, + 'exit_code': None, + 'exit_signal': None, + 'max_disk_read': None, + 'max_disk_write': None, + 'max_rss': None, + 'max_vmem': None, + 'start_time': None, + 'state': None, + 'submit_time': None, + 'wallclock': None, + } + status, error = _update_from_parent_rows(parent_rows, usage) + if error: + return None, error + + status, error = _update_from_batch_rows(batch_rows, status, usage) + if error: + return None, error + + status, error = _update_from_other_rows(other_rows, status, usage) + if error: + return None, error + + if status is None: + return None, 'no status found' + + resource_usage = ('cpu: {cpu},' + ' end_time: {end_time},' + ' exit_code: {exit_code},' + ' exit_signal: {exit_signal},' + ' max_disk_read: {max_disk_read},' + ' max_disk_write: {max_disk_write},' + ' max_rss: {max_rss},' + ' max_vmem: {max_vmem},' + ' start_time: {start_time},' + ' state: {state},' + ' submit_time: {submit_time},' + ' wallclock: {wallclock}'.format(**usage)) + return {'status': status, 'resource_usage': resource_usage}, None + + +def _parse_rows(stdout): + lines = stdout.splitlines() + if not lines: + return list(), None + + header = lines[0] + header_cols = header.split('|') + rows = list() + for i, line in enumerate(lines[1:]): + row_cols = line.split('|') + if len(header_cols) != len(row_cols): + return None, 'row {} had {} columns but expected {}'.format( + i, len(row_cols), len(header_cols)) + + row = dict(zip(header_cols, row_cols)) + rows.append(row) + + return rows, None + + +def _get_job_id_and_step(job_id_raw): + job_id_sep_index = job_id_raw.find('.') + if job_id_sep_index <= 0: + return job_id_raw, None + + job_id_base = job_id_raw[:job_id_sep_index] + job_id_step = job_id_raw[job_id_sep_index + 1:] + return job_id_base, job_id_step + + +def _update_from_parent_rows(rows, usage): + if not rows: + return None, None + + if len(rows) > 1: + return None, 'expected at most 1 parent row' + + row = rows[0] + parsed_values = _parse_values(row) + # parent row is handled first. + # Add starting values which may be overwritten later + _add_if_not_none_keys(parsed_values, usage, [ + 'cpu', 'exit_code', 'exit_signal', 'wallclock', 'submit_time', + 'start_time', 'end_time', 'state', 'max_disk_read', 'max_disk_write', + 'max_rss', 'max_vmem' + ]) + + parsed_status = parsed_values.get('state_for_snakemake') + return parsed_status, None + + +def _update_from_batch_rows(rows, status, usage): + if not rows: + return status, None + + if len(rows) > 1: + return None, 'expected at most 1 batch row' + + row = rows[0] + parsed_values = _parse_values(row) + # the batch row seems to have more details for these fields + _overwrite_if_not_none_keys(parsed_values, usage, [ + 'cpu', 'exit_code', 'exit_signal', 'max_disk_read', 'max_disk_write', + 'max_rss', 'max_vmem' + ]) + # prefer the info from the parent row for these fields + _add_if_not_none_keys( + parsed_values, usage, + ['wallclock', 'submit_time', 'start_time', 'end_time', 'state']) + + # prefer the parent status + if status is None: + status = parsed_values.get('state_for_snakemake') + + return status, None + + +def _update_from_other_rows(rows, status, usage): + for row in rows: + parsed_values = _parse_values(row) + # use the "other" rows to fill in missing information + _add_if_not_none_keys(parsed_values, usage, [ + 'cpu', 'exit_code', 'exit_signal', 'wallclock', 'submit_time', + 'start_time', 'end_time', 'state', 'max_disk_read', + 'max_disk_write', 'max_rss', 'max_vmem' + ]) + + if status is None: + status = parsed_values.get('state_for_snakemake') + + return status, None + + +def _parse_values(row): + values = dict() + values['cpu'] = _parse_cpu_time(row) + exit_code, exit_signal = _parse_exit_code(row) + values['exit_code'] = exit_code + values['exit_signal'] = exit_signal + values['wallclock'] = _parse_wallclock(row) + values['submit_time'] = _parse_submit_time(row) + values['start_time'] = _parse_start_time(row) + values['end_time'] = _parse_end_time(row) + raw_state, state_for_snakemake = _parse_state(row) + values['state'] = raw_state + values['state_for_snakemake'] = state_for_snakemake + values['max_disk_read'] = _parse_max_disk_read(row) + values['max_disk_write'] = _parse_max_disk_write(row) + values['max_rss'] = _parse_max_rss(row) + values['max_vmem'] = _parse_max_vmem(row) + return values + + +def _parse_state(row): + raw = row.get('State') + if not raw: + return None, None + + # translate the slurm state into snakemake terms + for_snakemake = None + if ((raw.startswith('RUNNING') or raw.startswith('PENDING') + or raw.startswith('REQUEUED') or raw.startswith('RESIZING') + or raw.startswith('SUSPENDED'))): + for_snakemake = 'running' + elif raw.startswith('COMPLETED'): + for_snakemake = 'success' + else: + for_snakemake = 'failed' + + return raw, for_snakemake + + +def _parse_submit_time(row): + return _parse_datetime_col(row, 'Submit') + + +def _parse_start_time(row): + return _parse_datetime_col(row, 'Start') + + +def _parse_end_time(row): + return _parse_datetime_col(row, 'End') + + +def _parse_datetime_col(row, col): + # yyyy-mm-ddThh:mm:ss + raw = row.get(col) + if not raw: + return None + + return raw + + +def _parse_cpu_time(row): + # 'mm:ss.millis' + raw = row.get('TotalCPU') + if not raw: + return None + + return raw + + +def _parse_wallclock(row): + # 'num_seconds' + raw = row.get('ElapsedRaw') + if not raw: + return None + + return raw + + +def _parse_max_disk_read(row): + return _parse_disk(row, 'MaxDiskRead') + + +def _parse_max_disk_write(row): + return _parse_disk(row, 'MaxDiskWrite') + + +def _parse_disk(row, col): + # '{float}M' + raw = row.get(col) + if not raw: + return None + + return raw + + +def _parse_max_rss(row): + return _parse_mem(row, 'MaxRss') + + +def _parse_max_vmem(row): + return _parse_mem(row, 'MaxVMSize') + + +def _parse_mem(row, col): + # '{int}K' + raw = row.get(col) + if not raw: + return None + + return raw + + +def _parse_exit_code(row): + # 'exitcode:signal_num' + raw = row.get('ExitCode') + if not raw: + return None, None + + splits = raw.split(':') + if len(splits) != 2: + return raw, None + + return splits[0], splits[1] + + +def _overwrite_if_not_none_keys(source, dest, keys): + for key in keys: + value = source.get(key) + _overwrite_if_not_none(key, value, dest) + + +def _overwrite_if_not_none(key, value, dest): + if value is not None: + dest[key] = value + + +def _add_if_not_none_keys(source, dest, keys): + for key in keys: + value = source.get(key) + _add_if_not_none(key, value, dest) + + +def _add_if_not_none(key, value, dest): + if value is not None and dest.get(key) is None: + dest[key] = value diff --git a/snakemake_profile/cluster_commands_sge.py b/snakemake_profile/cluster_commands_sge.py new file mode 100644 index 0000000..5ce35ac --- /dev/null +++ b/snakemake_profile/cluster_commands_sge.py @@ -0,0 +1,132 @@ +import bs4 + + +def submit_command(log_out, log_err, threads, time_hours, mem_mb, + mem_mb_per_thread, gpus, gpu_name, jobscript): + command = ['qsub', '-o', log_out, '-e', log_err] + if threads: + command.extend(['-pe', 'smp', str(threads)]) + + if mem_mb_per_thread: + command.extend(['-l', 'h_vmem={}M'.format(mem_mb_per_thread)]) + + command.append(jobscript) + return command + + +def try_extract_job_id_from_submit_output(stdout): + tokens = stdout.split() + if len(tokens) < 3: + return None, 'expected at least 3 tokens' + + if (tokens[0] != 'Your') or (tokens[1] != 'job'): + return None, 'expected output to look like "Your job ..."' + + try: + job_id = int(tokens[2]) + except ValueError as e: + return None, 'could not parse {} as an int: {}'.format(tokens[2], e) + + return job_id, None + + +def status_command(job_id): + return ['qstat', '-j', job_id, '-xml'] + + +def try_extract_job_info_from_status_output(stdout, job_id): + info = {'status': None, 'resource_usage': None} + soup = bs4.BeautifulSoup(stdout, 'html.parser') + + unks = soup.find_all('unknown_jobs') + if unks: + names = unks[0].find_all('st_name') + if names: + if names[0].string == job_id: + info['status'] = 'success' + return info, None + + djobs = soup.find_all('djob_info') + if djobs: + numbers = djobs[0].find_all('jb_job_number') + if numbers: + if numbers[0].string == job_id: + info['status'] = 'running' + info['resource_usage'] = _extract_resource_usage(djobs[0]) + return info, None + + return None, 'unexpected output' + + +def _extract_resource_usage(soup): + resources = _extract_resource_usage_components(soup) + return ('wallclock: {wallclock}, cpu: {cpu}, io_wait: {io_wait},' + ' max_vmem: {max_vmem}, max_rss: {max_rss}, vmem: {vmem},' + ' rss: {rss}'.format(**resources)) + + +def _extract_resource_usage_components(soup): + resources = { + 'wallclock': None, + 'cpu': None, + 'io_wait': None, + 'vmem': None, + 'max_vmem': None, + 'rss': None, + 'max_rss': None, + } + usage_list = soup.find_all('jat_scaled_usage_list') + if not usage_list: + return resources + + wallclock_float = _extract_ua_by_name(usage_list[0], 'wallclock') + if wallclock_float: + resources['wallclock'] = '{:.2f}s'.format(wallclock_float) + + cpu_float = _extract_ua_by_name(usage_list[0], 'cpu') + if cpu_float: + resources['cpu'] = '{:.2f}s'.format(cpu_float) + + io_wait_float = _extract_ua_by_name(usage_list[0], 'iow') + if io_wait_float: + resources['io_wait'] = '{:.2f}s'.format(io_wait_float) + + bytes_per_gb = 1024**3 + vmem_float = _extract_ua_by_name(usage_list[0], 'vmem') + if vmem_float: + resources['vmem'] = '{:.2f}GB'.format(vmem_float / bytes_per_gb) + + max_vmem_float = _extract_ua_by_name(usage_list[0], 'maxvmem') + if max_vmem_float: + resources['max_vmem'] = '{:.2f}GB'.format(max_vmem_float / + bytes_per_gb) + + rss_float = _extract_ua_by_name(usage_list[0], 'rss') + if rss_float: + resources['rss'] = '{:.2f}GB'.format(rss_float / bytes_per_gb) + + max_rss_float = _extract_ua_by_name(usage_list[0], 'maxrss') + if max_rss_float: + resources['max_rss'] = '{:.2f}GB'.format(max_rss_float / bytes_per_gb) + + return resources + + +def _extract_ua_by_name(soup, ua_name): + name_node = soup.find_all('ua_name', string=ua_name) + if not name_node: + return None + + value_node = name_node[0].find_next_siblings('ua_value') + if not value_node: + return None + + value_str = value_node[0].string + return _try_parse_float(value_str) + + +def _try_parse_float(s): + try: + return float(s) + except ValueError: + return None diff --git a/snakemake_profile/cluster_status.py b/snakemake_profile/cluster_status.py new file mode 100644 index 0000000..1af63b5 --- /dev/null +++ b/snakemake_profile/cluster_status.py @@ -0,0 +1,127 @@ +import argparse +import datetime +import os.path +import sys +import time + +import cluster_commands +import try_command + + +def parse_args(): + parser = argparse.ArgumentParser() + parser.add_argument( + 'job_id', help='the cluster id of the job to check the status of') + parser.add_argument( + '--retry-status-interval-seconds', + default='', + help='a "," separated list of integers representing' + ' the number of seconds to wait after sequential failed' + ' job status commands before retrying') + parser.add_argument( + '--resource-usage-dir', + help='a directory for storing the file paths where the resource usage' + ' of each job should be logged') + parser.add_argument( + '--resource-usage-min-interval', + type=float, + default=120, + help='only log the resource usage if it has been at least this many' + ' seconds since the last log') + args = parser.parse_args() + + retry_status_interval_seconds = list() + for int_str in args.retry_status_interval_seconds.split(','): + retry_status_interval_seconds.append(int(int_str)) + + return { + 'job_id': args.job_id, + 'retry_status_interval_seconds': retry_status_interval_seconds, + 'resource_usage_dir': args.resource_usage_dir, + 'resource_usage_min_interval': args.resource_usage_min_interval, + } + + +def extract_job_info(stdout, job_id): + info, error = cluster_commands.try_extract_job_info_from_status_output( + stdout, job_id) + if error: + print('error: {}\n{}'.format(error, stdout), file=sys.stderr) + sys.exit(1) + + now = datetime.datetime.now() + formatted_now = now.isoformat() + info['resource_usage'] = 'current_time: {}, {}'.format( + formatted_now, info['resource_usage']) + + return info + + +def update_resource_log(status, resource_usage, resource_dir, + min_interval_seconds, job_id): + if not (resource_dir and os.path.isdir(resource_dir)): + return + + resource_dir_job_file = os.path.join(resource_dir, '{}.txt'.format(job_id)) + if not os.path.exists(resource_dir_job_file): + return + + with open(resource_dir_job_file, 'rt') as f_handle: + resource_log_file = f_handle.read().strip() + + is_final_update = status != 'running' + update_resource_log_with_file(resource_usage, min_interval_seconds, + resource_log_file, is_final_update) + + if is_final_update: + os.remove(resource_dir_job_file) + + +def update_resource_log_with_file(resource_usage, min_interval_seconds, + resource_log_file, is_final_update): + if not resource_usage: + return + + # resource_log_file is created and written to by this function. + # log_dir should have been created by snakemake when the job was submitted. + log_dir = os.path.dirname(resource_log_file) + if not (resource_log_file.endswith('.cluster.usage') + and os.path.isdir(log_dir)): + return + + # The first write creates the file and does not check min_interval_seconds + if ((min_interval_seconds and os.path.exists(resource_log_file) + and not is_final_update)): + mod_time_seconds = os.stat(resource_log_file).st_mtime + current_seconds = time.time() + diff_seconds = current_seconds - mod_time_seconds + if diff_seconds < min_interval_seconds: + return + + with open(resource_log_file, 'at') as f_handle: + f_handle.write('{}\n'.format(resource_usage)) + + +def run_status_command(command, retry_status_interval_seconds): + stdout, error = try_command.try_command(command, retry_status_interval_seconds) + if error: + sys.exit(error) + + return stdout + + +def main(): + args = parse_args() + job_id = args['job_id'] + command = cluster_commands.status_command(job_id) + stdout = run_status_command(command, args['retry_status_interval_seconds']) + job_info = extract_job_info(stdout, job_id) + status = job_info['status'] + update_resource_log(status, job_info['resource_usage'], + args['resource_usage_dir'], + args['resource_usage_min_interval'], job_id) + print(status) + + +if __name__ == '__main__': + main() diff --git a/snakemake_profile/cluster_submit.py b/snakemake_profile/cluster_submit.py new file mode 100644 index 0000000..8f0c1e5 --- /dev/null +++ b/snakemake_profile/cluster_submit.py @@ -0,0 +1,149 @@ +import argparse +import math +import os +import os.path +import sys + +from snakemake.utils import read_job_properties + +import cluster_commands +import try_command + + +def parse_args(): + parser = argparse.ArgumentParser() + parser.add_argument('jobscript', + help='the script to be executed on the cluster') + parser.add_argument( + '--retry-submit-interval-seconds', + default='', + help='a "," separated list of integers representing' + ' the number of seconds to wait after sequential failed' + ' job submission commands before retrying') + parser.add_argument( + '--resource-usage-dir', + help='a directory for storing the file paths where cluster_status.py' + ' should log the resource usage of each job') + args = parser.parse_args() + jobscript = args.jobscript + + job_properties = read_job_properties(jobscript) + + retry_submit_interval_seconds = list() + for int_str in args.retry_submit_interval_seconds.split(','): + retry_submit_interval_seconds.append(int(int_str)) + + resource_usage_dir = args.resource_usage_dir + if resource_usage_dir: + os.makedirs(resource_usage_dir, exist_ok=True) + + return { + 'jobscript': jobscript, + 'job_properties': job_properties, + 'retry_submit_interval_seconds': retry_submit_interval_seconds, + 'resource_usage_dir': resource_usage_dir, + } + + +def get_base_path_from_jobscript(jobscript): + with open(jobscript) as f_handle: + for line in f_handle: + tokens = line.split() + if len(tokens) == 4: + if ((tokens[0] == 'cd' and tokens[2] == '&&' + and tokens[3] == '\\')): + base_path = tokens[1] + if os.path.isdir(base_path): + return os.path.abspath(base_path) + + return None + + +def get_cluster_log_paths(jobscript, job_properties): + cluster_logs = {'out': os.devnull, 'err': os.devnull, 'usage': os.devnull} + base_path = get_base_path_from_jobscript(jobscript) + orig_logs = job_properties.get('log') + if (not base_path) or (not orig_logs): + return cluster_logs + + if len(orig_logs) == 1: + orig_log_out = orig_logs[0] + orig_log_err = orig_log_out + else: + orig_log_out = orig_logs[0] + orig_log_err = orig_logs[1] + + cluster_logs['out'] = os.path.join(base_path, + '{}.cluster.out'.format(orig_log_out)) + cluster_logs['err'] = os.path.join(base_path, + '{}.cluster.err'.format(orig_log_err)) + cluster_logs['usage'] = os.path.join( + base_path, '{}.cluster.usage'.format(orig_log_out)) + return cluster_logs + + +def build_submit_command(jobscript, job_properties, cluster_log_out, + cluster_log_err): + threads = job_properties.get('threads') + resources = job_properties.get('resources') + time_hours = None + mem_mb = None + mem_mb_per_thread = None + if resources: + time_hours = resources.get('time_hours') + gpus = resources.get('gpus') + gpu_name = resources.get('gpu_name') + mem_mb = resources.get('mem_mb') + mem_mb_per_thread = mem_mb + if mem_mb and threads: + mem_mb_per_thread /= float(threads) + mem_mb_per_thread = math.ceil(mem_mb_per_thread) + + return cluster_commands.submit_command(cluster_log_out, cluster_log_err, + threads, time_hours, mem_mb, + mem_mb_per_thread, gpus, gpu_name, + jobscript) + + +def run_submit_command(command, retry_submit_interval_seconds): + stdout, error = try_command.try_command(command, + retry_submit_interval_seconds) + if error: + sys.exit(error) + + return stdout + + +def extract_job_id(stdout): + job_id, error = cluster_commands.try_extract_job_id_from_submit_output( + stdout) + if error: + print('error: {}\n{}'.format(error, stdout), file=sys.stderr) + sys.exit(1) + + return job_id + + +def record_usage_file(job_id, cluster_log_usage, resource_usage_dir): + job_file_path = os.path.join(resource_usage_dir, '{}.txt'.format(job_id)) + with open(job_file_path, 'wt') as f_handle: + f_handle.write('{}\n'.format(cluster_log_usage)) + + +def main(): + parsed_args = parse_args() + jobscript = parsed_args['jobscript'] + job_properties = parsed_args['job_properties'] + cluster_logs = get_cluster_log_paths(jobscript, job_properties) + command = build_submit_command(jobscript, job_properties, + cluster_logs['out'], cluster_logs['err']) + stdout = run_submit_command(command, + parsed_args['retry_submit_interval_seconds']) + job_id = extract_job_id(stdout) + record_usage_file(job_id, cluster_logs['usage'], + parsed_args['resource_usage_dir']) + print(job_id) + + +if __name__ == '__main__': + main() diff --git a/snakemake_profile/config.yaml b/snakemake_profile/config.yaml new file mode 100644 index 0000000..a0ce6df --- /dev/null +++ b/snakemake_profile/config.yaml @@ -0,0 +1,36 @@ +# Commenting out 'cluster' will force jobs to be run locally. +# '>-' is yaml syntax for splitting a string over multiple lines +cluster: >- + python ./snakemake_profile/cluster_submit.py + --retry-submit-interval-seconds 30,120,300 + --resource-usage-dir ./snakemake_profile/job_resource_usage +cluster-status: >- + python ./snakemake_profile/cluster_status.py + --retry-status-interval-seconds 30,120,300 + --resource-usage-dir ./snakemake_profile/job_resource_usage + --resource-usage-min-interval 300 + +# 'jobs' has two interpretations: +# * if running with 'cluster' +# + max number of jobs concurrently submitted to the cluster +# * else +# + max number of local cores to use +jobs: 100 + +# 'resources' defines limits that apply to both local and 'cluster' jobs. +# 'resources' is commented out when using 'cluster' to allow the cluster +# scheduler to determine the available memory. +# resources: +# - 'mem_mb=16384' + +# wait up to a minute for result files to be visible through the shared filesystem +latency-wait: 60 + +# Allow a failed job to be re-started once +restart-times: 1 + +# output settings +verbose: false +printshellcmds: true +show-failed-logs: false +reason: true diff --git a/snakemake_profile/try_command.py b/snakemake_profile/try_command.py new file mode 100644 index 0000000..05d9ae9 --- /dev/null +++ b/snakemake_profile/try_command.py @@ -0,0 +1,40 @@ +import subprocess +import time + + +def try_command_once(command): + completed_process = subprocess.run(command, + check=False, + stdout=subprocess.PIPE, + stderr=subprocess.PIPE) + decoded_stdout = completed_process.stdout.decode() + decoded_stderr = completed_process.stderr.decode() + if completed_process.returncode != 0: + return None, 'command:{}\nstdout:\n{}\n\nstderr:\n{}'.format( + command, decoded_stdout, decoded_stderr) + + return decoded_stdout, None + + +def try_command(command, retry_interval_seconds): + errors = list() + + # The final (retry_seconds: None) allows running the command, but + # without the ability to wait and retry. + retry_interval_seconds = retry_interval_seconds + [None] + for retry_seconds in retry_interval_seconds: + stdout, error = try_command_once(command) + if not error: + return stdout, None + + errors.append(error) + if retry_seconds is None: + break + + time.sleep(retry_seconds) + + formatted_errors = list() + for i, error in enumerate(errors): + formatted_errors.append('attempt {}\n{}'.format(i, error)) + + return None, '\n'.join(formatted_errors)