-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathbaseline-overlap.py
92 lines (70 loc) · 2.87 KB
/
baseline-overlap.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
# -*- coding: utf-8 -*-
from __future__ import division, unicode_literals
"""
Script implementing the overlap baseline for the ASSIN shared task.
It extracts two features from each pair: the amount of words exclusive
to the first sentence and to the second one. Both are fed to an SVM
(for entailment classification) or an SVR (for similarity regression).
It produces an XML file as the output, which can be evaluated with the
`assin-eval.py` script.
"""
import argparse
import numpy as np
from xml.etree import cElementTree as ET
from sklearn.linear_model import LinearRegression, LogisticRegression
from commons import read_xml, entailment_to_str, tokenize_sentence
def extract_features(pairs):
'''
Extract a vector of features from the given pairs and return
them as a numpy array.
'''
features = []
for pair in pairs:
t = pair.t.lower()
tokens1 = tokenize_sentence(t)
h = pair.h.lower()
tokens2 = tokenize_sentence(h)
value1, value2 = words_in_common(tokens1, tokens2)
features.append((value1, value2))
return np.array(features)
def words_in_common(sentence1, sentence2):
'''
Return the proportion of words in common in a pair.
Repeated words are ignored.
'''
tokenset1 = set(sentence1)
tokenset2 = set(sentence2)
num_common_tokens = len(tokenset2.intersection(tokenset1))
proportion1 = num_common_tokens / len(tokenset1)
proportion2 = num_common_tokens / len(tokenset2)
return (proportion1, proportion2)
if __name__ == '__main__':
parser = argparse.ArgumentParser(description=__doc__)
parser.add_argument('train', help='XML file with training data')
parser.add_argument('test', help='XML file with test data')
parser.add_argument('output', help='Output tagged XML file')
args = parser.parse_args()
# extract features and labels
train_pairs = read_xml(args.train, need_labels=True)
features = extract_features(train_pairs)
entailment_target = np.array([pair.entailment for pair in train_pairs])
similarity_target = np.array([pair.similarity for pair in train_pairs])
# train models
classifier = LogisticRegression(class_weight='balanced')
classifier.fit(features, entailment_target)
regressor = LinearRegression()
regressor.fit(features, similarity_target)
# run models
test_pairs = read_xml(args.test, need_labels=False)
features = extract_features(test_pairs)
predicted_entailment = classifier.predict(features)
predicted_similarity = regressor.predict(features)
# write output
tree = ET.parse(args.test)
root = tree.getroot()
for i in range(len(test_pairs)):
pair = root[i]
entailment_str = entailment_to_str[predicted_entailment[i]]
pair.set('entailment', entailment_str)
pair.set('similarity', str(predicted_similarity[i]))
tree.write(args.output, 'utf-8')