-
Notifications
You must be signed in to change notification settings - Fork 326
/
Copy pathannotate.py
executable file
·122 lines (107 loc) · 4.54 KB
/
annotate.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
#!/usr/bin/env python3
from argparse import ArgumentDefaultsHelpFormatter, ArgumentParser
import os
import records
import ujson as json
from stanza.nlp.corenlp import CoreNLPClient
from tqdm import tqdm
import copy
from lib.common import count_lines, detokenize
from lib.query import Query
client = None
def annotate(sentence, lower=True):
global client
if client is None:
client = CoreNLPClient(default_annotators='ssplit,tokenize'.split(','))
words, gloss, after = [], [], []
for s in client.annotate(sentence):
for t in s:
words.append(t.word)
gloss.append(t.originalText)
after.append(t.after)
if lower:
words = [w.lower() for w in words]
return {
'gloss': gloss,
'words': words,
'after': after,
}
def annotate_example(example, table):
ann = {'table_id': example['table_id']}
ann['question'] = annotate(example['question'])
ann['table'] = {
'header': [annotate(h) for h in table['header']],
}
ann['query'] = sql = copy.deepcopy(example['sql'])
for c in ann['query']['conds']:
c[-1] = annotate(str(c[-1]))
q1 = 'SYMSELECT SYMAGG {} SYMCOL {}'.format(Query.agg_ops[sql['agg']], table['header'][sql['sel']])
q2 = ['SYMCOL {} SYMOP {} SYMCOND {}'.format(table['header'][col], Query.cond_ops[op], detokenize(cond)) for col, op, cond in sql['conds']]
if q2:
q2 = 'SYMWHERE ' + ' SYMAND '.join(q2) + ' SYMEND'
else:
q2 = 'SYMEND'
inp = 'SYMSYMS {syms} SYMAGGOPS {aggops} SYMCONDOPS {condops} SYMTABLE {table} SYMQUESTION {question} SYMEND'.format(
syms=' '.join(['SYM' + s for s in Query.syms]),
table=' '.join(['SYMCOL ' + s for s in table['header']]),
question=example['question'],
aggops=' '.join([s for s in Query.agg_ops]),
condops=' '.join([s for s in Query.cond_ops]),
)
ann['seq_input'] = annotate(inp)
out = '{q1} {q2}'.format(q1=q1, q2=q2) if q2 else q1
ann['seq_output'] = annotate(out)
ann['where_output'] = annotate(q2)
assert 'symend' in ann['seq_output']['words']
assert 'symend' in ann['where_output']['words']
return ann
def is_valid_example(e):
if not all([h['words'] for h in e['table']['header']]):
return False
headers = [detokenize(h).lower() for h in e['table']['header']]
if len(headers) != len(set(headers)):
return False
input_vocab = set(e['seq_input']['words'])
for w in e['seq_output']['words']:
if w not in input_vocab:
print('query word "{}" is not in input vocabulary.\n{}'.format(w, e['seq_input']['words']))
return False
input_vocab = set(e['question']['words'])
for col, op, cond in e['query']['conds']:
for w in cond['words']:
if w not in input_vocab:
print('cond word "{}" is not in input vocabulary.\n{}'.format(w, e['question']['words']))
return False
return True
if __name__ == '__main__':
parser = ArgumentParser(formatter_class=ArgumentDefaultsHelpFormatter)
parser.add_argument('--din', default='data', help='data directory')
parser.add_argument('--dout', default='annotated', help='output directory')
args = parser.parse_args()
if not os.path.isdir(args.dout):
os.makedirs(args.dout)
for split in ['train', 'dev', 'test']:
fsplit = os.path.join(args.din, split) + '.jsonl'
ftable = os.path.join(args.din, split) + '.tables.jsonl'
fout = os.path.join(args.dout, split) + '.jsonl'
print('annotating {}'.format(fsplit))
with open(fsplit) as fs, open(ftable) as ft, open(fout, 'wt') as fo:
print('loading tables')
tables = {}
for line in tqdm(ft, total=count_lines(ftable)):
d = json.loads(line)
tables[d['id']] = d
print('loading examples')
n_written = 0
for line in tqdm(fs, total=count_lines(fsplit)):
d = json.loads(line)
a = annotate_example(d, tables[d['table_id']])
if not is_valid_example(a):
raise Exception(str(a))
gold = Query.from_tokenized_dict(a['query'])
reconstruct = Query.from_sequence(a['seq_output'], a['table'], lowercase=True)
if gold.lower() != reconstruct.lower():
raise Exception ('Expected:\n{}\nGot:\n{}'.format(gold, reconstruct))
fo.write(json.dumps(a) + '\n')
n_written += 1
print('wrote {} examples'.format(n_written))