-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathscoring.py
executable file
·299 lines (262 loc) · 13.4 KB
/
scoring.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
#!/usr/bin/env python
# This file is licensed under the Apache 2.0 License terms, see
# https://www.apache.org/licenses/LICENSE-2.0
# see https://github.com/codalab/codabench/wiki/Competition-Bundle-Structure#scoring-program
import sys
import os
import json
import csv
from collections import defaultdict
import numpy as np
import argparse
from sklearn.metrics import accuracy_score, f1_score, precision_score, recall_score
from scipy.spatial import distance
GLOBALS = dict(debug=False)
EPS=0.001 # small value to allow for rounding errors when checking for a valid distribution that sums to 1.0
MULT_LABELS = ["0-Kein", "1-Gering", "2-Vorhanden", "3-Stark", "4-Extrem"]
ST1_COLUMNS = ['id', 'bin_maj', 'bin_one', 'bin_all', 'multi_maj', 'disagree_bin']
ST2_COLUMNS = ['id', 'dist_bin_0', 'dist_bin_1', 'dist_multi_0', 'dist_multi_1', 'dist_multi_2', 'dist_multi_3', 'dist_multi_4']
def load_targets(targets_file):
with open(targets_file, "rt") as infp:
targets = json.load(infp)
return targets
def check_columns(data, columns):
"""Check if the expected columns and only the expected columns are present in the data,
if not, print an error message to
stderr and throw an exception. Otherwise return to the caller."""
for column in columns:
if column not in data:
raise ValueError(f"Column {column} not found in data, got {data.keys()}")
for column in data.keys():
if column not in columns:
raise ValueError(f"Column {column} not expected in data, expected {columns}")
def check_allowed(data, column, allowed=None):
"""Check if the predictions are in the allowed set, if not, print an error message to
stderr also showing the id and throw an exception. Otherwise return to the caller."""
if allowed is None:
allowed = ["0", "1"]
if column not in data:
raise ValueError(f"Column {column} not found in data")
for i, value in enumerate(data[column]):
if value not in allowed:
raise ValueError(f"Invalid value {value} not one of {allowed} in column {column} at index {i} with id {data['id'][i]}")
print(f"Column {column} is OK")
def check_dist(data, columns):
"""Check if the predictions are in the allowed range, if not, print an error message to
stderr also showing the id and throw an exception. Otherwise return to the caller."""
for column in columns:
if column not in data:
raise ValueError(f"Column {column} not found in data")
for i in range(len(data["id"])):
sum = 0.0
theid = data["id"][i]
for column in columns:
try:
value = float(data[column][i])
except ValueError:
raise ValueError(f"Invalid value {data[column][i]} not a float in column {column} at index {i} with id {theid}")
if value < 0.0 or value > 1.0:
raise ValueError(f"Invalid value {value} not in range [0.0, 1.0] in column {column} at index {i} with id {data['id'][i]}")
sum += value
if abs(sum - 1.0) > EPS:
raise ValueError(f"Values in columns {columns} do not sum to 1.0 at index {i} with id {theid}")
def load_tsv(submission_dir, expected_rows, expected_cols, file=None):
"""
Try to load a TSV file from the submission directory. This expects a single TSV file to be present in the submission directory.
If there is no TSV file or there are multiple files, it will log an error to stderr and return None.
"""
if file is not None:
tsv_file = file
else:
tsv_files = [f for f in os.listdir(submission_dir) if f.endswith('.tsv')]
if len(tsv_files) == 0:
print("No TSV file ending with '.tsv' found in submission directory", file=sys.stderr)
return None
if len(tsv_files) > 1:
print("Multiple TSV files found in submission directory", file=sys.stderr)
return None
tsv_file = tsv_files[0]
tsv_path = os.path.join(submission_dir, tsv_file)
print("Loading TSV file", tsv_path)
# Read the TSV file incrementally row by row and create a dictionary where the key is the column name and the value is a list of values for that column.
# Expect the column names in the first row of the TSV file.
# Abort reading and log an error to stderr if the file is not a valid TSV file, if it contains more than one row with the same id,
# if the column name is not known, or if there are more than N_MAX rows.
data = defaultdict(list)
with open(tsv_path, 'rt') as infp:
reader = csv.DictReader(infp, delimiter='\t')
for i, row in enumerate(reader):
if i == 0:
if set(reader.fieldnames) != set(expected_cols):
gotcols = ", ".join(list(set(reader.fieldnames)))
print(f"Invalid column names in TSV file, expected:\n {', '.join(expected_cols)}\ngot\n {gotcols}", file=sys.stderr)
return None
if i >= expected_rows:
print(f"Too many rows in TSV file, expected {expected_rows}", file=sys.stderr)
return None
for col_name in reader.fieldnames:
data[col_name].append(row[col_name])
if len(data['id']) != expected_rows:
print(f"Missing values in TSV file, expected {expected_rows} rows, got {len(data['id'])}", file=sys.stderr)
return None
return data
def score_st1(data, targets):
"""Calculate the score for subtask 1"""
# NOTE: targets are a dictionary with the same keys as data, and the values are lists of the target values
# for some columns where more than one prediction is allowed, the target values are lists of lists
# for those columns, where more than one prediction is allowed, we need to select either the one
# predicted by the model, or a random incorrect one if the model did not predict a correct one
check_columns(data, ST1_COLUMNS)
check_allowed(data, 'bin_maj', ["0", "1"])
check_allowed(data, 'bin_one', ["0", "1"])
check_allowed(data, 'bin_all', ["0", "1"])
check_allowed(data, 'multi_maj', MULT_LABELS)
check_allowed(data, 'disagree_bin', ["0", "1"])
target_bin_maj = []
for pred, target in zip(data['bin_maj'], targets['bin_maj']):
if isinstance(target, list):
target_bin_maj.append(pred)
else:
target_bin_maj.append(target)
targets['bin_maj'] = target_bin_maj
target_multi_maj = []
for pred, target in zip(data['multi_maj'], targets['multi_maj']):
if isinstance(target, list):
if pred not in target:
# select a random incorrect target: just pick the first one
target_multi_maj.append(target[0])
else:
# the prediction is correct
target_multi_maj.append(pred)
else:
target_multi_maj.append(target)
targets['multi_maj'] = target_multi_maj
scores = {}
used_scores = []
for col_name in data.keys():
if col_name == 'id':
continue
if GLOBALS['debug']:
print(f"Calculating scores for {col_name}")
scores[col_name+"_acc"] = accuracy_score(data[col_name], targets[col_name])
scores[col_name+"_f1"] = f1_score(data[col_name], targets[col_name], average='macro')
used_scores.append(scores[col_name+"_f1"])
# calculate average over all f1 scores
scores['score'] = np.mean(used_scores)
return scores
def score_st2(data, targets):
"""Calculate the score for subtask 2"""
check_dist(data, ['dist_bin_0', 'dist_bin_1'])
check_dist(data, ['dist_multi_0', 'dist_multi_1', 'dist_multi_2', 'dist_multi_3', 'dist_multi_4'])
scores = {}
sum_bin = 0.0
sum_multi = 0.0
for idx in range(len(data['id'])):
# calculate the vectors for the binary and multi-class predictions
dist_bin = [float(data['dist_bin_0'][idx]), float(data['dist_bin_1'][idx])]
dist_multi = [float(data[colname][idx]) for colname in ['dist_multi_0', 'dist_multi_1', 'dist_multi_2', 'dist_multi_3', 'dist_multi_4']]
# calculate the vectors for the binary and multi-class targets
target_bin = [targets['dist_bin_0'][idx], targets['dist_bin_1'][idx]]
target_multi = [targets['dist_multi_0'][idx], targets['dist_multi_1'][idx], targets['dist_multi_2'][idx], targets['dist_multi_3'][idx], targets['dist_multi_4'][idx]]
# calculate the distances
score_bin = distance.jensenshannon(dist_bin, target_bin, base=2)
score_multi = distance.jensenshannon(dist_multi, target_multi, base=2)
sum_bin += score_bin
sum_multi += score_multi
scores['js_dist_bin'] = sum_bin / len(data['id'])
scores['js_dist_multi'] = sum_multi / len(data['id'])
scores['score'] = np.mean([scores['js_dist_bin'], scores['js_dist_multi']])
return scores
def main():
parser = argparse.ArgumentParser(description='Scorer for the competition')
parser.add_argument('--submission-dir', help='Directory containing the submission (.)', default=".")
parser.add_argument('--submission-file', help='If submission directory contains more than one file, name of the file to use (None)', default=None)
parser.add_argument('--reference-dir', help='Directory containing the reference data (./dev_phase/reference_data/)', default="./dev_phase/reference_data/")
parser.add_argument('--score-dir', help='Directory to write the scores to (.)', default=".")
parser.add_argument('--codabench', help='Indicate we are running on codabench, not locally', action='store_true')
parser.add_argument("--st", required=True, choices=["1", "2"], help='Subtask to evaluate, one of 1, 2')
parser.add_argument("--debug", help='Print debug information', action='store_true')
args = parser.parse_args()
GLOBALS['debug'] = args.debug
print(f'Running scorer for subtask {args.st}')
if args.codabench:
run_locally = False
print("Running on codabench")
submission_dir = '/app/input/res'
reference_dir = '/app/input/ref'
score_dir = '/app/output/'
else:
run_locally = True
print("Running locally")
submission_dir = args.submission_dir
reference_dir = args.reference_dir
score_dir = args.score_dir
# if we are on codabench, list all files and directories under /app recursively
# if not run_locally:
# for root, dirs, files in os.walk('/app'):
# for file in files:
# print("FILE:", os.path.join(root, file))
# for dir in dirs:
# print("DIR:", os.path.join(root, dir))
# # also load and show the contents of the metadata file in /app/input/res/metadata as a text file, if it exists
# metadata_file = os.path.join(submission_dir, "metadata")
# if os.path.exists(metadata_file):
# with open(metadata_file, 'rt') as infp:
# metadata = infp.read()
# print("Metadata file contents:", metadata)
# else:
# print("No metadata file found")
targets_file = os.path.join(reference_dir, "targets.json")
print(f"Using targets file {targets_file}")
targets = load_targets(targets_file)
print(f"Loaded {len(targets)} targets")
# index the targets for easy lookup
targets_index = {t['id']: t for t in targets}
# load the submissing tsv file
if args.st == "1":
data = load_tsv(submission_dir, expected_rows=len(targets), expected_cols=ST1_COLUMNS, file=args.submission_file)
elif args.st == "2":
data = load_tsv(submission_dir, expected_rows=len(targets), expected_cols=ST2_COLUMNS, file=args.submission_file)
else:
print("Unknown subtask", file=sys.stderr)
sys.exit(1)
if data is None:
print("Problems loading the submission, aborting", file=sys.stderr)
sys.exit(1)
print(f"Loaded {len(data['id'])} rows from the submission")
# check if the ids in the submission match the ids in the targets exactly: the targets are a list of
# dictionaries with keys 'id' and the various prediction targets, the data is a dictionary where the keys are
# the column names (one of which is "id") and the values are lists of values for that column
if set(data['id']) != set(targets_index.keys()):
print("IDs in submission do not match IDs in targets", file=sys.stderr)
sys.exit(1)
# convert the targets to the same format as the submission, and in the same order by id as the submission
targets_dir = {}
for col_name in data.keys():
if col_name == 'id':
continue
col_values = []
for idx, id in enumerate(data['id']):
if id not in targets_index:
print(f"ID {id} not found in targets for id {id} in row {idx}", file=sys.stderr)
sys.exit(1)
if col_name not in targets_index[id]:
print(f"Column {col_name} not found in targets for id {id} in row {idx}", file=sys.stderr)
sys.exit(1)
col_values.append(targets_index[id][col_name])
targets_dir[col_name] = col_values
if args.st == "1":
scores = score_st1(data, targets_dir)
elif args.st == "2":
scores = score_st2(data, targets_dir)
else:
print("Unknown subtask", file=sys.stderr)
sys.exit(1)
print("Scores:", scores)
with open(os.path.join(score_dir, 'scores.json'), 'w') as score_file:
score_file.write(json.dumps(scores))
#with open(os.path.join(score_dir, 'details.html'), 'w') as html_file:
# html_file.write("<html>Some text</html>")
print("Ending scorer")
if __name__ == '__main__':
main()