-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy path1_get_bacteria_sister_clades.py
295 lines (234 loc) · 11.7 KB
/
1_get_bacteria_sister_clades.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
'''
'''
import os, glob, argparse, multiprocessing, time, itertools
from functools import partial
import pandas as pd
import numpy as np
from ete3 import Tree, RectFace
def parse_args():
parser = argparse.ArgumentParser(description='')
requiredArgs = parser.add_argument_group("Required Arguments")
requiredArgs.add_argument('-td', '--trees_dir',
dest='trees_dir',
required=True,
help='directory with the .nw trees'
)
# requiredArgs.add_argument('-f', '--faa_dir',
# dest='faa_dir',
# required=True,
# help='directory with the .faa files obtained with '
# '"extract_seqs_NR.py"'
# )
requiredArgs.add_argument('-s', '--summary_dir',
dest='summary_dir',
required=True,
help='directory to save the .summary files'
)
requiredArgs.add_argument('-n', '--run_name',
dest='run_name',
required=True,
help='name of the analysis (round1, round2...)'
)
requiredArgs.add_argument('-o', '--outdir',
dest='outdir',
required=True,
help='directory to save the .faa files'
)
return parser.parse_args()
def get_crassvirales_taxonomy_df(taxa_file):
'''
Parses the taxonomy and returns a dataframe, index=genomes, 3 columns (family, subfamily, genus)
'''
#"/home/danielc/software/github/old_june/tree_functions/files/crassphages_taxonomy_terL_and_new.txt"
df = pd.read_csv(taxa_file, sep="\t", index_col=0, names=["family","subfamily","genus"])
df['superkingdom'] = np.where(df['family'].isin(["new", "crass_env"]), "new", "crassvirales")
return df
def create_target_taxa_groups():
taxa_groups_raw = ["crassvirales","NA", "new", "Viruses", "bact_crass_polytomy", "prophage"]
taxa_groups = list()
# create the target taxa_groups to look for in the tree
for i in range(2, len(taxa_groups_raw)+1):
combinations = itertools.combinations(taxa_groups_raw, i)
# iterate the combinations and retain those with "crassvirales" in them
for combination in combinations:
if "crassvirales" in combination:
taxa_groups.append(list(combination))
return taxa_groups
def resolve_crass_bacteria_polytomies(t):
'''
Looks for clades with Crassvirales and Bacteria, all of them with 0 dist
'''
checked_leaves = list()
crass_polytom_leaves = t.search_nodes(dist=0.0, superkingdom="crassvirales")
for leaf in crass_polytom_leaves:
if leaf.name not in checked_leaves:
checked_leaves.append(leaf.name)
# go one node up, check superkingdom of descendants
node_up = leaf.up
for node in node_up.iter_descendants():
if node.superkingdom == "Bacteria":
node.superkingdom = "bact_crass_polytomy"
print(f"\t- polytomy bacteria-crass, {node.name}")
return t
def read_and_annotate_tree(tree_file, ncbi_summary_df, crassvirales_taxa_df):
'''
Reads the tree of a CL with NCBI and Crassvirales sequences and assigns
source and taxonomy to them
'''
# read tree
t = Tree(tree_file, format=1)
# iterate the leaves, assi
for leaf in t.iter_leaves():
# check if it is an NCBI hit:
if leaf.name in ncbi_summary_df.index:
leaf.add_features(source="ncbi",
crass=str(ncbi_summary_df.loc[leaf.name, "ncbi_crass"]),
superkingdom=ncbi_summary_df.loc[leaf.name, "superkingdom"],
phylum=ncbi_summary_df.loc[leaf.name, "phylum"],
#class=ncbi_summary_df.loc[leaf.name, "class"],
order=ncbi_summary_df.loc[leaf.name, "order"],
family=ncbi_summary_df.loc[leaf.name, "family"],
genus=ncbi_summary_df.loc[leaf.name, "genus"]
)
# if not, it is a Crassvirales sequence
else:
genome = leaf.name.split("|")[0]
leaf.add_features(source="crass",
crass=str(True),
superkingdom=crassvirales_taxa_df.loc[genome, "superkingdom"],
# I put the Crassvirales family here
phylum=crassvirales_taxa_df.loc[genome, "family"],
#class="crass",
order="crass",
family=crassvirales_taxa_df.loc[genome, "family"],
genus=crassvirales_taxa_df.loc[genome, "genus"]
)
return t
def identify_crassvirales_mrcas(t, taxa_groups, start_cont, faces=False):
'''
'''
monophyletic_clades_leaves = dict()
monophyletic_clades = dict()
mrcas_ids_to_clean = list()
cont = 0
for taxa_group in taxa_groups:
mrcas = t.get_monophyletic(target_attr="superkingdom", values=taxa_group)
if mrcas:
for mrca in mrcas:
cont += 1
mrca_id = f"mrca_{cont}"
# get names of the leaves
mrca_leaves_names = {leaf.name for leaf in mrca.iter_leaves()}
if monophyletic_clades_leaves:
for target_mrca, target_leaves_names in monophyletic_clades_leaves.items():
tmp = target_leaves_names.union(mrca_leaves_names)
if len(tmp) == len(mrca_leaves_names):
mrcas_ids_to_clean.append(target_mrca)
monophyletic_clades_leaves[mrca_id] = mrca_leaves_names
monophyletic_clades[mrca_id] = mrca
# remove MRCAs already contained in higher MRCAs
mrcas_ids_to_clean = list(set(mrcas_ids_to_clean))
for mrca_id in mrcas_ids_to_clean:
del monophyletic_clades[mrca_id]
del monophyletic_clades_leaves[mrca_id]
# change mrca_ids to lower numbers
final_monophyletic_clades = dict()
final_cont = start_cont
for mrca_id, mrca in monophyletic_clades.items():
final_monophyletic_clades[f"mrca_{final_cont}"] = mrca
if faces:
mrca.add_features(mrca_id=f"mrca_{final_cont}")
face = RectFace(5,5, "blue","blue", label=f"mrca_{final_cont}")
mrca.add_face(face, 0, "float")
final_cont +=1
return t, final_monophyletic_clades, final_cont
def root_farthest_bacteria_hits(t, monophyletic_clades):
# get the most distant node to the mrca
rooted_trees = dict()
for mrca_id, mrca in monophyletic_clades.items():
bacteria_leaves_dists = {bacteria_leaf:mrca.get_distance(bacteria_leaf) for bacteria_leaf in t.search_nodes(superkingdom="Bacteria")}
# check if there are still Bacteria after resolving polytomies
if bacteria_leaves_dists:
# sort by distance
bacteria_leaves_dists = {node: distance for node, distance in sorted(bacteria_leaves_dists.items(), key=lambda item: item[1], reverse=True)}
farthest_bact_leaf = list(bacteria_leaves_dists.keys())[0]
# root using the farthest Bacteria
t.set_outgroup(farthest_bact_leaf)
#print(bacteria_leaves_dists)
rooted_trees[farthest_bact_leaf.name] = t
return rooted_trees
def process_MRCA(mrca):
'''
'''
polytomy_seqs = ",".join([leaf.name for leaf in mrca.iter_leaves() if leaf.superkingdom == "bact_crass_polytomy"])
n_seqs_mrca = len(mrca.get_leaf_names())
sister = mrca.get_sisters()
if len(sister) > 1:
print("wuh?")
n_seqs_sister = len(sister[0].get_leaf_names())
bacteria_seqs_sister = ",".join([leaf.name for leaf in sister[0].iter_leaves() if leaf.superkingdom == "Bacteria"])
sister_support = mrca.up.name
return polytomy_seqs, n_seqs_mrca, n_seqs_sister, sister_support, bacteria_seqs_sister
def process_tree(summary_dir, crassvirales_taxa_df, taxa_groups, outdir, run_name, tree_file):
# get CL_id
cl_id = os.path.basename(tree_file).split("_ncbi")[0]
# read summary file of the CL to know wheter it needs to be processed for Bacteria
summary_file = f"{summary_dir}/{cl_id}.summary"
ncbi_summary_df = pd.read_csv(summary_file, sep="\t", header=0, index_col=0, low_memory=False)
ncbi_summary_df = ncbi_summary_df[ncbi_summary_df.included]
# check if there are Bacteria hits
if "Bacteria" in ncbi_summary_df.superkingdom.to_list():
# replace nan by "NA"
ncbi_summary_df = ncbi_summary_df.fillna("NA")
#print(ncbi_summary_df)
# read the CL tree
t = read_and_annotate_tree(tree_file, ncbi_summary_df, crassvirales_taxa_df)
# resolve Crassvirales-Bacteria polytomies
t = resolve_crass_bacteria_polytomies(t)
# identify Crassvirales MRCAs in the tree
start_cont = 1
t, mrcas, cont = identify_crassvirales_mrcas(t, taxa_groups, start_cont)
# for all the MRCAs identified, get the farthest Bacteria sequence
# and root with it. Return as many trees as different outgroups
rooted_trees = root_farthest_bacteria_hits(t, mrcas)
if rooted_trees:
# for each of the rooted trees, indentify MRCAs again
cont = 1
header = ["mrca_id", "cl_id", "outgroup", "polytomies", "n_seqs_mrca", "n_seqs_sister", "sister_support", "bacteria_seqs_sister"]
to_write = list()
for out_seq, rooted_tree in rooted_trees.items():
t, mrcas, cont = identify_crassvirales_mrcas(rooted_tree, taxa_groups, cont, faces=True)
#print(mrcas)
for mrca_id, mrca in mrcas.items():
polytomy_seqs, n_seqs_mrca, n_seqs_sister, sister_support, bacteria_seqs_sister = process_MRCA(mrca)
to_write.append([mrca_id,
cl_id,
out_seq,
polytomy_seqs,
n_seqs_mrca,
n_seqs_sister,
sister_support,
bacteria_seqs_sister])
# convert to df and write
df = pd.DataFrame(to_write, columns=header)
outfile = f"{outdir}/{run_name}/0_mrcas/{cl_id}.txt"
df.to_csv(outfile, header=True, index=False, sep="\t")
def main():
args = parse_args()
# create output dirs
os.makedirs(f"{args.outdir}/{args.run_name}/0_mrcas", exist_ok=True)
# get crassvirales taxonomy
crassvirales_taxa_file = "/home/danielc/projects/Bas_phages/crass_protein_families_scripts/tree_functions/crassphages_taxonomy_terL_and_new.txt"
crassvirales_taxa_df = get_crassvirales_taxonomy_df(crassvirales_taxa_file)
# target taxa clades to look for in the trees
taxa_groups = create_target_taxa_groups()
taxa_groups = [["crassvirales"]] + taxa_groups # add crassvirales at the beginning
# list trees
tree_files = glob.glob(f"{args.trees_dir}/*.nw")
part = partial(process_tree, args.summary_dir, crassvirales_taxa_df, taxa_groups, args.outdir, args.run_name)
pool = multiprocessing.Pool(processes=18)
pool.map(part, tree_files)
pool.close()
pool.join()
if __name__ == "__main__":
main()