-
Notifications
You must be signed in to change notification settings - Fork 4
/
utils.py
164 lines (121 loc) · 4.75 KB
/
utils.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
import re
import json
REG_DIGITS_SINGLETON = re.compile(r"^\d+[./]?\d*%?$")
REG_DIGITS_BEGIN = re.compile(r"^(\d+[./]?\d*%?) ?(?=[\u4e00-\u9fa5,,。°℃$])")
REG_DIGITS_MIDDLE = re.compile(r"(?<=[\u4e00-\u9fa5$::,=≈=->{]) ?(\d+[./]?\d*%?) ?(?:[\u4e00-\u9fa5,,。°℃$}(]|\([^\d\s]|\.$)")
REG_DIGITS_END = re.compile(r"(?<=[\u4e00-\u9fa5$::=≈=\->]) ?(\d+[./]?\d*%?)$")
REG_NORM = re.compile(r"(?<=\d)[, ](?=\d)")
REG_LATEX_FRAC = re.compile(r'\\frac{([^}]+)}{([^}]+)}')
REG_CN_FRAC = re.compile(r'(\d+)分之(\d+)')
def read_jsonl_keys(file: str, keys: list):
"""读取jsonl文件中每一行的特定字段,以dict返回"""
output_dict = {}
for f in keys:
output_dict[f] = []
with open(file, 'r', encoding="UTF-8") as fin:
for line in fin:
content = json.loads(line.strip())
for f in keys:
if f in content:
output_dict[f].append(content[f])
else:
output_dict[f].append(None)
return output_dict
def has_exception(answer: str) -> bool:
"""用于判断模型生成的答案是否为异常"""
if answer is None:
return True
if len(answer.strip()) == 0:
return True
reg_timeout = re.compile("(请求.*超时)|(timeout)")
if bool(re.search(reg_timeout, answer)):
return True
reg_error = re.compile("error|异常|失败|content_filter")
# 请求出现错误或异常
if "{" in answer and "}" in answer and (bool(re.search(reg_error, answer))):
return True
return False
def extract_cn_fractal(line):
"""提取输入字符串中的中文汉字分数表达,例如`五分之三`,并将其转换为基于阿拉伯数字的分数表达`3/5`"""
res = re.findall(REG_CN_FRAC, line)
if len(res) != 0:
return ["{}/{}".format(b, a) for a, b in res]
else:
return res
def extract_digits_from_line(line):
"""提取输入字符串中的所有数字"""
res1 = re.findall(REG_DIGITS_BEGIN, line)
res2 = re.findall(REG_DIGITS_MIDDLE, line)
res3 = re.findall(REG_DIGITS_END, line)
res4 = re.findall(REG_DIGITS_SINGLETON, line)
res_cn_frac = extract_cn_fractal(line)
concat = res1 + res2 + res_cn_frac + res3 + res4
candidates = [s.strip() for s in concat]
return [s for s in candidates if not (s.startswith("/") or s.endswith("/"))]
def extract_digits_prediction(response, truncation="t", exception_regs: list = None):
"""从模型回复中提取答案候选"""
# 检测模型回复是否出现异常
if has_exception(response, exception_regs):
return ["ERROR"]
response = REG_LATEX_FRAC.sub(r'\1/\2', response)
response = re.sub(REG_NORM, "", response)
candidates = []
# 逐行提取数字
for line in response.splitlines():
candidates += extract_digits_from_line(line)
if truncation is None:
# 不对candidate做截断
res = candidates
elif truncation == "t":
# 只考虑最末尾的两个数字作为模型可能的回答
if len(candidates) <= 2:
res = candidates
else:
res = candidates[-2:]
elif truncation == "h":
res = candidates[:2]
elif truncation == "ht" or truncation == "th":
if len(candidates) < 4:
res = candidates
else:
res = candidates[:2] + candidates[-2:]
else:
raise ValueError("Illegal truncation argument... Only `h`, `t`, `ht` or None are supported.")
return list(set(res))
def string2num(string: str):
"""尝试把一个string转化成一个浮点数或整数"""
string = string.strip()
if string.endswith("%"):
string = string.replace("%", "")
return float(string) / 100
if "/" in string:
parts = string.split("/")
if len(parts) != 2 or float(parts[1]) == 0.0:
print("Warning: {} is illegal!".format(string))
return 0
else:
return float(parts[0]) / float(parts[1])
if "." in string:
return float(string)
return int(string)
def match_digits(a, b):
"""判断数字a和b是否近似相等"""
if isinstance(a, int) and isinstance(b, int):
return a == b
else:
relative_diff = abs(a - b) / (min(abs(a), abs(b)) + 1e-6)
return relative_diff < 1e-2
def match_digit_response(golden, responses: list) -> bool:
"""判断标准答案是与提取的答案候选中的某一个匹配"""
if "ERROR" in responses:
return False
if isinstance(golden, str):
golden = string2num(golden)
for r in responses:
try:
num = string2num(r)
if match_digits(golden, num):
return True
except ValueError:
pass
return False