genienlp/metrics.py

422 lines
15 KiB
Python
Raw Normal View History

2018-06-20 06:22:34 +00:00
from subprocess import Popen, PIPE, CalledProcessError
import json
from text.torchtext.datasets.generic import Query
import logging
import os
import re
import string
import collections
from multiprocessing import Pool, cpu_count
from contextlib import closing
from pyrouge import Rouge155
from sacrebleu import corpus_bleu
def to_lf(s, table):
aggs = [y.lower() for y in Query.agg_ops]
agg_to_idx = {x: i for i, x in enumerate(aggs)}
conditionals = [y.lower() for y in Query.cond_ops]
headers_unsorted = [(y.lower(), i) for i, y in enumerate(table['header'])]
headers = [(y.lower(), i) for i, y in enumerate(table['header'])]
headers.sort(reverse=True, key=lambda x: len(x[0]))
condition_s, conds = None, []
if 'where' in s:
s, condition_s = s.split('where', 1)
s = ' '.join(s.split()[1:-2])
sel, agg = None, 0
for col, idx in headers:
if col == s:
sel = idx
if sel is None:
s = s.split()
agg = agg_to_idx[s[0]]
s = ' '.join(s[1:])
for col, idx in headers:
if col == s:
sel = idx
full_conditions = []
if not condition_s is None:
condition_s = ' ' + condition_s + ' '
for idx, col in enumerate(headers):
condition_s = condition_s.replace(' ' + col[0] + ' ', ' Col{} '.format(col[1]))
condition_s = condition_s.strip()
for idx, col in enumerate(conditionals):
new_s = []
for t in condition_s.split():
if t == col:
new_s.append('Cond{}'.format(idx))
else:
new_s.append(t)
condition_s = ' '.join(new_s)
s = condition_s
conds = re.split('(Col\d+ Cond\d+)', s)
if len(conds) == 0:
conds = [s]
conds = [x for x in conds if len(x.strip()) > 0]
full_conditions = []
for i, x in enumerate(conds):
if i % 2 == 0:
x = x.split()
col_num = int(x[0].replace('Col', ''))
opp_num = int(x[1].replace('Cond', ''))
full_conditions.append([col_num, opp_num])
else:
x = x.split()
if x[-1] == 'and':
x = x[:-1]
x = ' '.join(x)
if 'Col' in x:
new_x = []
for t in x.split():
if 'Col' in t:
idx = int(t.replace('Col', ''))
t = headers_unsorted[idx][0]
new_x.append(t)
x = new_x
x = ' '.join(x)
if 'Cond' in x:
new_x = []
for t in x.split():
if 'Cond' in t:
idx = int(t.replace('Cond', ''))
t = conditionals[idx]
new_x.append(t)
x = new_x
x = ' '.join(x)
full_conditions[-1].append(x)
logical_form = {'sel': sel, 'conds': full_conditions, 'agg': agg}
return logical_form
def computeLFEM(greedy, answer, args):
answer = [x[0] for x in answer]
count = 0
correct = 0
text_answers = []
for idx, (g, ex) in enumerate(zip(greedy, answer)):
count += 1
text_answers.append([ex['answer'].lower()])
try:
lf = to_lf(g, ex['table'])
gt = ex['sql']
conds = gt['conds']
lower_conds = []
for c in conds:
lc = c
lc[2] = str(lc[2]).lower()
lower_conds.append(lc)
gt['conds'] = lower_conds
correct += lf == gt
except Exception as e:
continue
return correct / count * 100, text_answers
def computeCF1(greedy, answer):
def remove_and(text):
return re.sub(r'\b(and)\b', ' ', text)
greedy_counters = []
num_not_null_greedy = 0
for g in greedy:
clean_g = remove_and(g).split()
if (len(clean_g) == 1) and ('unanswerable' == clean_g[0]):
greedy_counters.append(None)
else:
greedy_counters.append(collections.Counter(clean_g))
num_not_null_greedy += 1
answer_counters = []
num_not_null_answer = 0
for aa in answer:
a_counters = []
num_not_null_a = 0
for a in aa:
clean_a = remove_and(a).split()
if (len(clean_g) == 1) and ('unanswerable' == clean_a[0]):
a_counters.append(None)
else:
a_counters.append(collections.Counter(clean_a))
num_not_null_a += 1
if num_not_null_a > 0:
num_not_null_answer += 1
answer_counters.append(a_counters)
num_true_positive = 0
for g, aa in zip(greedy_counters, answer_counters):
if g == None:
continue
for a in aa:
if a == None:
continue
elif a == g:
num_true_positive += 1
break
precision = num_true_positive / num_not_null_greedy
recall = num_true_positive / num_not_null_answer
return 2 * (precision * recall) / (precision + recall) * 100, precision * 100, recall * 100
def normalize_text(s):
"""Lower text and remove punctuation, articles and extra whitespace."""
def remove_articles(text):
return re.sub(r'\b(a|an|the)\b', ' ', text)
def white_space_fix(text):
return ' '.join(text.split())
def remove_punc(text):
exclude = set(string.punctuation)
return ''.join(ch for ch in text if ch not in exclude)
def lower(text):
return text.lower()
return white_space_fix(remove_articles(remove_punc(lower(s))))
def f1_score(prediction, ground_truth):
prediction_tokens = prediction.split()
ground_truth_tokens = ground_truth.split()
common = collections.Counter(prediction_tokens) & collections.Counter(ground_truth_tokens)
num_same = sum(common.values())
if num_same == 0:
return 0
precision = 1.0 * num_same / len(prediction_tokens)
recall = 1.0 * num_same / len(ground_truth_tokens)
f1 = (2 * precision * recall) / (precision + recall)
return f1
def exact_match(prediction, ground_truth):
return prediction == ground_truth
def metric_max_over_ground_truths(metric_fn, prediction, ground_truths):
scores_for_ground_truths = []
for idx, ground_truth in enumerate(ground_truths):
score = metric_fn(prediction, ground_truth)
scores_for_ground_truths.append(score)
return max(scores_for_ground_truths)
def computeF1(outputs, targets):
return sum([metric_max_over_ground_truths(f1_score, o, t) for o, t in zip(outputs, targets)])/len(outputs) * 100
def computeEM(outputs, targets):
outs = [metric_max_over_ground_truths(exact_match, o, t) for o, t in zip(outputs, targets)]
return sum(outs)/len(outputs) * 100
def computeBLEU(outputs, targets):
targets = [[t[i] for t in targets] for i in range(len(targets[0]))]
return corpus_bleu(outputs, targets, lowercase=True).score
class Rouge(Rouge155):
"""Rouge calculator class with custom command-line options."""
# See full list of options here:
# https://github.com/andersjo/pyrouge/blob/master/tools/ROUGE-1.5.5/README.txt#L82
DEFAULT_OPTIONS = [
'-a', # evaluate all systems
'-n', 4, # max-ngram
'-x', # do not calculate ROUGE-L
'-2', 4, # max-gap-length
'-u', # include unigram in skip-bigram
'-c', 95, # confidence interval
'-r', 1000, # number-of-samples (for resampling)
'-f', 'A', # scoring formula
'-p', 0.5, # 0 <= alpha <=1
'-t', 0, # count by token instead of sentence
'-d', # print per evaluation scores
]
def __init__(self, n_words=None,
keep_files=False, options=None):
if options is None:
self.options = self.DEFAULT_OPTIONS.copy()
else:
self.options = options
if n_words:
options.extend(["-l", n_words])
stem = "-m" in self.options
super(Rouge, self).__init__(
n_words=n_words, stem=stem,
keep_files=keep_files)
def _run_rouge(self):
# Get full options
options = (
['-e', self._rouge_data] +
list(map(str, self.options)) +
[os.path.join(self._config_dir, "settings.xml")])
logging.info("Running ROUGE with options {}".format(" ".join(options)))
# print([self._rouge_bin] + list(options))
pipes = Popen([self._rouge_bin] + options, stdout=PIPE, stderr=PIPE)
std_out, std_err = pipes.communicate()
div_by_zero_error = std_err.decode("utf-8").\
startswith("Illegal division by zero")
if pipes.returncode == 0 or div_by_zero_error:
# Still returns the correct output even with div by zero
return std_out
else:
raise ValueError(
std_out.decode("utf-8") + "\n" + std_err.decode("utf-8"))
def computeROUGE(greedy, answer):
rouges = compute_rouge_scores(greedy, answer)
if len(rouges) > 0:
avg_rouges = {}
for key in rouges[0].keys():
avg_rouges[key] = sum(
[r.get(key, 0.0) for r in rouges]) / len(rouges) * 100
else:
avg_rouges = None
return avg_rouges
def split_sentences(txt, splitchar=".", include_splitchar=False):
"""Split sentences of a text based on a given EOS char."""
out = [s.split() for s in txt.strip().split(splitchar) if len(s) > 0]
return out
def compute_rouge_scores(summs, refs, splitchar='.', options=None, parallel=True):
assert len(summs) == len(refs)
options = [
'-a', # evaluate all systems
'-c', 95, # confidence interval
'-m', # use Porter stemmer
'-n', 2, # max-ngram
'-w', 1.3, # weight (weighting factor for WLCS)
]
rr = Rouge(options=options)
rouge_args = []
for summ, ref in zip(summs, refs):
letter = "A"
ref_dict = {}
for r in ref:
ref_dict[letter] = [x for x in split_sentences(r, splitchar) if len(x) > 0]
letter = chr(ord(letter) + 1)
s = [x for x in split_sentences(summ, splitchar) if len(x) > 0]
rouge_args.append((s, ref_dict))
if parallel:
with closing(Pool(cpu_count()//2)) as pool:
rouge_scores = pool.starmap(rr.score_summary, rouge_args)
else:
rouge_scores = []
for s, a in rouge_args:
rouge_scores.append(rr.score_summary(s, ref_dict))
return rouge_scores
def to_delta_state(line):
delta_state = {'inform': {}, 'request': {}}
try:
if line == 'None' or line.strip() == '':
return delta_state
inform, request = [[y.strip() for y in x.strip().split(',')] for x in line.split(';')]
inform_pairs = {}
for i in inform:
try:
k, v = i.split(':')
inform_pairs[k.strip()] = v.strip()
except:
pass
delta_state = {'inform': inform_pairs, 'request': request}
except:
pass
finally:
return delta_state
def update_state(state, delta):
for act, slot in delta.items():
state[act] = slot
return state
def dict_cmp(d1, d2):
def cmp(a, b):
for k1, v1 in a.items():
if k1 not in b:
return False
else:
if v1 != b[k1]:
return False
return True
return cmp(d1, d2) and cmp(d2, d1)
def computeDialogue(greedy, answer):
examples = []
for idx, (g, a) in enumerate(zip(greedy, answer)):
examples.append((a[0][0], g, a[0][1], idx))
examples.sort()
turn_request_positives = 0
turn_goal_positives = 0
joint_goal_positives = 0
ldt = None
for ex in examples:
if ldt is None or ldt.split('_')[:-1] != ex[0].split('_')[:-1]:
state, answer_state = {}, {}
ldt = ex[0]
delta_state = to_delta_state(ex[1])
answer_delta_state = to_delta_state(ex[2])
state = update_state(state, delta_state['inform'])
answer_state = update_state(answer_state, answer_delta_state['inform'])
if dict_cmp(state, answer_state):
joint_goal_positives += 1
if delta_state['request'] == answer_delta_state['request']:
turn_request_positives += 1
if dict_cmp(delta_state['inform'], answer_delta_state['inform']):
turn_goal_positives += 1
joint_goal_em = joint_goal_positives / len(examples) * 100
turn_request_em = turn_request_positives / len(examples) * 100
turn_goal_em = turn_goal_positives / len(examples) * 100
answer = [(x[-1], x[-2]) for x in examples]
answer.sort()
answer = [[x[1]] for x in answer]
return joint_goal_em, turn_request_em, turn_goal_em, answer
def compute_metrics(greedy, answer, rouge=False, bleu=False, corpus_f1=False, logical_form=False, args=None, dialogue=False):
metric_keys = []
metric_values = []
if not isinstance(answer[0], list):
answer = [[a] for a in answer]
if logical_form:
lfem, answer = computeLFEM(greedy, answer, args)
metric_keys += ['lfem']
metric_values += [lfem]
if dialogue:
joint_goal_em, request_em, turn_goal_em, answer = computeDialogue(greedy, answer)
avg_dialogue = (joint_goal_em + request_em) / 2
metric_keys += ['joint_goal_em', 'turn_request_em', 'turn_goal_em', 'avg_dialogue']
metric_values += [joint_goal_em, request_em, turn_goal_em, avg_dialogue]
em = computeEM(greedy, answer)
metric_keys += ['em']
metric_values += [em]
if bleu:
bleu = computeBLEU(greedy, answer)
metric_keys.append('bleu')
metric_values.append(bleu)
if rouge:
rouge = computeROUGE(greedy, answer)
metric_keys += ['rouge1', 'rouge2', 'rougeL', 'avg_rouge']
avg_rouge = (rouge['rouge_1_f_score'] + rouge['rouge_2_f_score'] + rouge['rouge_l_f_score']) / 3
metric_values += [rouge['rouge_1_f_score'], rouge['rouge_2_f_score'], rouge['rouge_l_f_score'], avg_rouge]
norm_greedy = [normalize_text(g) for g in greedy]
norm_answer = [[normalize_text(a) for a in al] for al in answer]
nf1 = computeF1(norm_greedy, norm_answer)
nem = computeEM(norm_greedy, norm_answer)
metric_keys.extend(['nf1', 'nem'])
metric_values.extend([nf1, nem])
if corpus_f1:
corpus_f1, precision, recall = computeCF1(norm_greedy, norm_answer)
metric_keys += ['corpus_f1', 'precision', 'recall']
metric_values += [corpus_f1, precision, recall]
metric_dict = collections.OrderedDict(list(zip(metric_keys, metric_values)))
return metric_dict, answer