-
Notifications
You must be signed in to change notification settings - Fork 110
/
Copy pathonto_to_spacy_json.py
163 lines (133 loc) · 4.83 KB
/
onto_to_spacy_json.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
import json # for tuple support
import plac
import os
import re
from spacy.gold import biluo_tags_from_offsets
import spacy
nlp = spacy.blank("xx")
from tqdm import tqdm
import random
def get_root_filename(onto_dir):
name_files = []
for dirpath, subdirs, files in os.walk(onto_dir):
for fname in files:
if bool(re.search(".name", fname)):
fn = os.path.join(dirpath, fname)
fn = re.sub("\.name", "", fn)
name_files.append(fn)
return name_files
def split_sentence(text):
text = text.strip().split("\n")[1:-1]
return text
def split_doc(text):
text_list = text.strip().split("</DOC>\s<DOC")
ids = [re.findall('<DOC DOCNO="(.+?)">', t)[0] for t in text_list]
text_list = [re.sub('<DOC DOCNO=".+?">', "", t).strip() for t in text_list]
return ids, text_list
def clean_ent(ent):
tag = re.findall('TYPE="(.+?)"', ent)[0]
text = re.findall(">(.+)", ent)[0]
text = re.sub("\$", "\$", text)
return (text, tag)
def raw_text(text):
"""Remove entity tags"""
text = re.sub("<ENAMEX .+?>", "", text)
text = re.sub("</ENAMEX>", "", text)
return text
def ent_position(ents, text):
search_point = 0
spacy_ents = []
for ent in ents:
remain_text = text[search_point:]
ma = re.search(ent[0], remain_text)
ent_tup = (ma.start() + search_point, ma.end() + search_point, ent[1])
spacy_ents.append(ent_tup)
# update search point to prevent same word in different entity,
# it will cause bug which hard to debug
search_point = search_point + ma.end()
return spacy_ents
def text_to_spacy(markup):
raw_ents = re.findall("<ENAMEX(.+?)</ENAMEX>", markup)
ents = [clean_ent(raw_ent) for raw_ent in raw_ents]
text = raw_text(markup)
spacy_ents = ent_position(ents, text)
final = (text, {"entities": spacy_ents})
return final
def onf_to_raw(onf_file):
"""
Take in a path to a .onf Ontonotes file. Return the raw text (as much as possible).
The quotes are usually quite messed up, so this is not going to look like real input text.
"""
with open(onf_file, "r") as f:
onf = f.read()
sentences = re.findall(
"Plain sentence\:\n\-+?\n(.+?)Treebanked sentence", onf, re.DOTALL
)
sentences = [re.sub("\n+?\s*", " ", i).strip() for i in sentences]
paragraph = " ".join(sentences)
return paragraph
def name_to_sentences(ner_filename):
"""
Take a .name file and return a sentence list of the kind described here:
https://github.com/explosion/spacy/blob/master/examples/training/training-data.json
"""
with open(ner_filename, "r") as f:
doc = f.read()
sentences = []
onto_sents = split_sentence(doc)
for sent in onto_sents:
offsets = text_to_spacy(sent)
doc = nlp(offsets[0])
tags = biluo_tags_from_offsets(doc, offsets[1]["entities"])
ner_info = list(zip(doc, tags))
tokens = []
for n, i in enumerate(ner_info):
token = {
"head": 0,
"dep": "",
"tag": "",
"orth": i[0].string,
"ner": i[1],
"id": n,
}
tokens.append(token)
sentences.append({"tokens": tokens})
return sentences
def dir_to_annotation(onto_dir):
fns = get_root_filename(onto_dir)
all_annotations = []
for fn in tqdm(fns):
ner_filename = fn + ".name"
onf_filename = fn + ".onf"
try:
raw = onf_to_raw(onf_filename)
sentences = name_to_sentences(ner_filename)
final = {"id": "fake", "paragraphs": [{"raw": raw, "sentences": sentences}]}
all_annotations.append(final)
except Exception as e:
print("Error formatting ", fn, e)
return all_annotations
@plac.annotations(
onto_dir=("Directory of OntoNotes data to traverse", "option", "i", str),
train_file=("File to write training spaCy JSON out to", "option", "t", str),
val_file=("File to write validation spaCy JSON out to", "option", "e", str),
val_split=("Percentage to use for evaluation", "option", "v", float),
)
def main(onto_dir, train_file, val_file, val_split=0.75):
print("Reading and formatting annotations")
all_annotations = dir_to_annotation(onto_dir)
random.shuffle(all_annotations)
cutpoint = round(val_split * len(all_annotations))
val = all_annotations[:cutpoint]
train = all_annotations[cutpoint:]
print(
"Saving {0} training examples and {1} validation examples".format(
len(train), len(val)
)
)
with open(train_file, "w") as f:
json.dump(train, f, ensure_ascii=False, indent=4)
with open(val_file, "w") as f:
json.dump(val, f, ensure_ascii=False, indent=4)
if __name__ == "__main__":
plac.call(main)