-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmake_graphs_evaluation.py
203 lines (186 loc) · 9.56 KB
/
make_graphs_evaluation.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
import numpy as np
import matplotlib.pyplot as plt
import re
import statistics
import os
import matplotlib.ticker as mtick
import dAUC
def extract_data(summary_path):
"""
extracts the needed percentages from a text file generated during the evaluation
:param summary_path: path to text file
:return: percentage of morf leading to same prediction, percentage of random components leading to same predictions
"""
summary_file = open(summary_path, 'r')
lines = summary_file.readlines()
string_percentage_exp = lines[3]
string_percentage_rand = lines[4]
pattern = r"\d\.\d*"
percentage_rand = re.search(pattern, string_percentage_rand).group()
percentage_exp = re.search(pattern, string_percentage_exp).group()
return percentage_exp, percentage_rand
def make_graph(comps, values, save_path, plot_dev=False, dev_exp=None, dev_rand=None, dauc=False, decomp='ls'):
"""
function that makes a graph summarizing the evaluation
:param comps: list of components that were included -> for x axis
:param values: list of values -> for y axis of graph
:param save_path: path to save the generated figure
:param plot_dev: bool, whether to also plot the standard deviation as errorbar
:param dev_exp: if not None: list of std dev for morf
:param dev_rand: if not None: list of std dev for random
:param dauc: float or None: delta area under curve, if None: don't plot the delta AUC
:param: string: used decomposition
:return: nothing, saves and shows generated figure
"""
fig = plt.figure()
ax = fig.add_subplot(111)
plt.subplots_adjust(wspace=0.6, hspace=0.6, left=0.15, bottom=0.1, right=0.96, top=0.9)
for index, c in enumerate(comps):
if plot_dev:
if dev_exp is not None:
print(dev_exp[index])
plt.errorbar(c, values[0, index], dev_exp[index], marker='.', color='r')
if dev_rand is not None:
print(dev_rand[index])
plt.errorbar(c, values[1, index], dev_rand[index], marker='.', color='b')
plt.plot(c, values[1, index], 'b.')
label = "{:.2f}".format(values[1, index])
plt.annotate(label,
(c, values[1, index]),
textcoords="offset points",
xytext=(0, 5),
ha='center')
plt.plot(c, values[0, index], 'r.')
label = "{:.2f}".format(values[0, index])
plt.annotate(label,
(c, values[0, index]),
textcoords="offset points",
xytext=(0, 5),
ha='center')
plt.plot(comps, values[1, :], 'b', label='Random components')
plt.plot(comps, values[0, :], 'r', label='Most relevant components')
if dauc is not False:
# need to color the area in between the curves and add the dauc value to the plot
lbl = f"Delta-Area Under Curve = {round(dauc, 4)}"
plt.fill_between(comps, values[0, :], values[1, :], color='mediumpurple', alpha=0.5, label=lbl)
plt.title("Spectral: percentage of predictions leading\nto same class for k flipped components")
plt.xticks(comps)
ax.yaxis.set_major_formatter(mtick.PercentFormatter(1))
plt.xlabel("Number of flipped components")
plt.ylabel("Percentage of same predictions")
plt.legend()
plt.savefig(f'{save_path}/plot_{decomp}.png')
plt.show()
def calculate_std_var(exps, comp, path, mode='exp', decomp='ls'):
"""
calculate std deviation and variance and saves them to an overview text file
:param exps: 2d array of values over various runs
:param comp: list of number of removed components for which the results have been generated
:param path: path to save the overview text file
:param mode: 'exp' or 'rand', whether to calculate the statistics for morf or random components
:param decomp: string, chosen decomposition
:return: std deviations
"""
# calculate standard deviation and variance
deviations = np.zeros((len(comp)))
variances = np.zeros((len(comp)))
path_save_summary = f"{path}/statistics_{decomp}_{mode}.txt"
with open(path_save_summary, 'w') as summary:
for index, c in enumerate(comp):
current_dev = statistics.stdev(exps[index, :])
current_var = statistics.variance(exps[index, :])
deviations[index] = current_dev
variances[index] = current_var
summary.write(f'For {c} components, the standard deviation of the generated explanations is {current_dev} and the variance is {current_var}\n')
summary.write(f"Averaged standard deviation over all components: {np.mean(deviations)}\n")
summary.write(f"Averaged variance over all components: {np.mean(variances)}\n")
return deviations
def make_single_graph(components, path_text_files, dauc=True):
"""
wrapper function that calls the subordinate functions to make a graph
:param components: list of number of removed components for which the evaluation was conducted
:param path_text_files: path to the results of the evaluation
:param dauc: bool, whether to include the delta area under curve metric in the graph
"""
percentages = np.zeros((2, len(components)))
for i, comp in enumerate(components):
summary_path = f"{path_text_files}/{comp}_components.txt" # TODO:adapt
exp, rand = extract_data(summary_path)
percentages[0, i] = exp
percentages[1, i] = rand
if dauc:
dauc = dAUC.main(components, path_text_files)
make_graph(components, percentages, path_text_files, dauc=dauc)
else:
make_graph(components, percentages, path_text_files, dauc=False)
def get_significance_metrics(exp, rand, comps, path_to_save, dauc=True, decomp='ls'):
"""
wrapper function to make graph for an evaluation that includes various runs
:param exp: 2d list, explanation scores over various runs
:param rand: 2d list, random scores over various runs
:param comps: list of number of removed components for which the evaluation was conducted
:param path_to_save: path for saving the results
:param dauc: bool, whether to highlight the obtained delta area under the curve
:param decomp: string, chosen decomposition
"""
average_exp = np.mean(exp, axis=1)
average_rand = np.mean(rand, axis=1)
# make graph for average
average_values = np.stack((average_exp, average_rand))
stdev_exp = calculate_std_var(exp, comps, path_to_save, mode='exp', decomp=decomp)
stddev_rand = calculate_std_var(rand, comps, path_to_save, mode='rand', decomp=decomp)
if dauc:
area_morf = dAUC.calculate_area(average_exp, comps)
area_rand = dAUC.calculate_area(average_rand, comps)
dauc = abs(area_rand - area_morf)
print("Delta Area Under Curve:", dauc)
make_graph(comps, average_values, path_to_save, plot_dev=True, dev_exp=stdev_exp, dev_rand=stddev_rand,
dauc=dauc, decomp=decomp)
def best_performing(exps, rands, comps, path_to_save, dauc=False, decomp='ls'):
"""
wrapper function in case the best performing morf model should be selected and averaged performance for random
component selection
:param exps: 2d list, explanation scores over various runs
:param rands: 2d list, random scores over various runs
:param comps: list of number of removed components for which the evaluation was conducted
:param path_to_save: path for saving the results
:param dauc: bool, whether to include delta AUC in graph
"""
average_rand = np.mean(rands, axis=1)
sums_models = np.sum(exps, axis=0)
best_exp = exps[:, np.argmin(sums_models)] # argmin for pixel flipping, argmax for quantitative eval
# check dimensions, compare with previous versions
values = np.stack((best_exp, average_rand))
stdev = calculate_std_var(rands, comps, path_to_save, mode='rand', decomp=decomp)
if dauc:
area_morf = dAUC.calculate_area(best_exp, comps)
area_rand = dAUC.calculate_area(average_rand, comps)
dauc = abs(area_rand - area_morf)
print("Delta Area Under Curve:", dauc)
make_graph(comps, values, path_to_save, plot_dev=True, dev_exp=None, dev_rand=stdev, dauc=dauc,
decomp=decomp)
def significance_analysis(comps, path, runs, stddev='random', dauc=True, decomp='ls'):
"""
wrapper function for results over various runs
:param comps: list of number of removed components for which the evaluation was conducted
:param path: path to results of evaluation
:param stddev: can be 'random' or 'exp', if random: plot std dev for random prediction, if exp: plot std dev for morf
:param dauc: bool, whether to include delta AUC in graph
:param decomp: string, chosen decomposition
"""
explanations = np.zeros((len(comps), runs))
randoms = np.zeros((len(comps), runs))
for run in range(runs):
for comp_index, comp in enumerate(comps):
summary_path = f"{path}/output_run_{run}/{comp}_removed_components.txt"
explanations[comp_index, run], randoms[comp_index, run] = extract_data(summary_path)
if stddev == 'both':
get_significance_metrics(explanations, randoms, comps, path, dauc, decomp)
elif stddev == 'random':
best_performing(explanations, randoms, comps, path, dauc, decomp)
if __name__ == "__main__":
components = list(range(7))
path_text_files = '/Users/anne/PycharmProjects/LIME_cough/old_evals/Spectral /fold1'
# make_single_graph(components, path_text_files, dauc=True)
significance_analysis(components, path_text_files, runs=2, stddev='both', dauc=True, decomp='spectral')
print('All done :) ')