-
Notifications
You must be signed in to change notification settings - Fork 3
/
main.py
416 lines (378 loc) · 18.4 KB
/
main.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
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
import re
from vllm import LLM, SamplingParams
from config import *
from build_react_prompt import (
build_input_text,
TOOL_DESC,
PROMPT_REACT,
parse_latest_plugin_call,
)
import os
from tools_call import function_call
import json
import json5
from transformers import AutoTokenizer
from tools_caption import tools
from cpm_utils import save_cpm3_data,switch_cpm_tool
# from agent_demo import *
tokenizer = AutoTokenizer.from_pretrained(model_path,trust_remote_code = True)
sampling_params = SamplingParams(**params_dict)
def split_react_data(react_str):
pattern = re.compile(
r"Thought:\s*(.*?)\nAction:\s*(.*?)\nAction Input:\s*(.*?)\nObservation:\s*(.*?)\nThought:\s*(.*?)\nFinal Answer:\s*(.*)",
re.DOTALL,
)
matches = pattern.findall(react_str)
try:
for match in matches:
Thought1 = match[0]
Action = match[1]
Action_Input = match[2]
Observation = match[3]
Thought2 = match[4]
Final_Answer = match[5]
return Thought1, Action, Action_Input, Observation, Thought2, Final_Answer
except:
return None, None, None, None, None, None
def get_answer_from_output(output):
pattern = r"「问题开始」(.*?)「问题结束」"
questions = re.findall(pattern, output, re.DOTALL)
questions = [q.strip() for q in questions]
return questions
def get_complex_question_from_output(output):
pattern = r"--复杂问题_start--(.*?)--复杂问题_end--"
questions = re.findall(pattern, output, re.DOTALL)
pattern2 = r"--任务规划_start--(.*?)--任务规划_end--"
task_plan = re.findall(pattern2, output, re.DOTALL)
if len(task_plan) == len(questions):
questions = [{q.strip():task_plan[i]} for i,q in enumerate(questions)]
else:
questions = []
return questions
def get_tool_description(tool):
tool_descp = "工具名称是{},作用是{},".format(
tool["name_for_model"], tool["description_for_model"]
)
for t in tool["parameters"]:
if t["required"]:
if t["scope"]:
tool_descp += "参数“{}”是必须输入的,作用是{},该参数的取值范围是{}。".format(
t["name"], t["description"], t["scope"]
)
else:
tool_descp += "参数“{}”是必须输入的,作用是{}。".format(t["name"], t["description"])
elif t["scope"]:
tool_descp += "参数“{}”是可选的,作用是{},该参数的取值范围是{}。".format(
t["name"], t["description"], t["scope"]
)
else:
tool_descp += "参数“{}”是可选的,作用是{}。".format(t["name"], t["description"])
return tool_descp
def get_question():
if 'llm' not in locals():
llm = LLM(
model=model_path,
tensor_parallel_size=8,
max_model_len=4096,
dtype="bfloat16",
trust_remote_code=True,
enforce_eager=True,
gpu_memory_utilization=0.8,
)
prompt_template = """你是一个智能助手,现在我请你为以下工具生成问题,要求生成的问题能够被这个工具解决。工具的详细介绍如下:\n{}\n我现在给你一个关于此工具问题的示例「问题开始」
{}「问题结束」,接下来请你根据此示例和工具描述再生成{}个能够使用该工具解决的问题,并且用「问题开始」和「问题结束」将其包裹,不要生成其他的无关文字。"""
all_questions = []
all_react_prompt = []
questinos_dict = {}
for tool in tools:
questions = []
while True:
tool_description = get_tool_description(tool)
input_prompt = prompt_template.format(
tool_description, tool["example"], gen_datas_per_tool
)
input_prompt = """<|im_start|> system\n you are a helpful assistant<|im_end|>\n<|im_start|> user\n {}<|im_end|>\n<|im_start|> assistant\n""".format(
input_prompt
)
outputs = llm.generate(input_prompt, sampling_params)
output = outputs[0].outputs[0].text
questions.extend(get_answer_from_output(output))
if len(questions) >= gen_datas_per_tool:
all_questions.extend(questions)
print(questions)
questinos_dict[tool["name_for_model"]] = questions
break
with open(save_question_json, "w", encoding="utf-8") as f:
json.dump(questinos_dict, f, ensure_ascii=False, indent=4)
print("{}条输入指令已经保存到{}".format(len(all_questions), save_question_json))
def get_complex_question():
if 'llm' not in locals():
llm = LLM(
model=model_path,
tensor_parallel_size=8,
max_model_len=4096,
dtype="bfloat16",
trust_remote_code=True,
enforce_eager=True,
gpu_memory_utilization=0.8,
)
if complex_example_json.endswith('json'):
with open(complex_example_json,'r',encoding='utf-8') as f:
examples = json.load(f)
else:
examples = []
all_questions = []
for example in examples:
questions = []
question,plan = list(example.items())[0]
example = '''--复杂问题_start--\n{}\n--复杂问题_end--\n--任务规划_start--\n{}\n--任务规划_end--'''.format(question,plan)
# example += """--复杂问题2_start--
# 计算一下,如果敌方有意向我方发射阵地1的300人使用化学武器,我方用一架直升机需要多久可以疏散所有人员到我方指挥所完成?
# --复杂问题2_end--
# --任务规划2_start--
# 1使用knowledge_graph获取我们直升机的承载人数A.
# 2根据我放发射阵地1的300人和直升机的承载人数B计算需要的疏散次数C.
# 3使用knowledge_graph获取直升机的速度D.
# 4使用map_search获取我方指挥所的坐标F和我方发射阵地1的位置G.
# 5使用distance_calculation计算F到G的距离H.
# 6计算直升机以速度D行驶距离H来回一次需要的时间I.
# 7计算完成C次疏散需要的总时间J,并判断是否在直升机续航E之内.
# --任务规划2_end--
# """
prompt_template = """\n所有工具的详细介绍如上所示,你是一个智能助手,现在我请你为以下工具生成复合问题,要求生成的问题是具体的问题,有具体的目标,实体,任务,且能够被这几个工具所解决,并且最少这个复杂问题最少要使用两个以上的工具才能完成。接下来请你根据每个工具的简单问题示例和工具描述再生成{}个能够使用以上最少两个工具解决的复杂问题及其解决方案,接下来我会给你一个示例:/n{},\n.并且严格按照示例的格式,用--复杂问题_start--和--复杂问题_end--以及--任务规划_start--和--任务规划_end--进行包裹,不要生成其他无关的文字."""
tool_prompt = ''
questinos_dict = {}
for index,tool in enumerate(tools):
questions = []
tool_description = get_tool_description(tool)
tool_prompt += '\n第{}个工具:'.format(index+1)+tool_description
input_prompt = tool_prompt + prompt_template.format(3,example)
messages = [
{"role": "system", "content":"You are a helpful assistant."},
{"role": "user", "content": input_prompt}
]
input_prompt= tokenizer.apply_chat_template(messages, tokenize=False, add_generation_prompt=True)
# input_prompt = """<|im_start|> system\n You are a helpful assistant<|im_end|>\n<|im_start|> user\n {}<|im_end|>\n<|im_start|> assistant\n""".format(
# input_prompt
# )
while True:
outputs = llm.generate(input_prompt, sampling_params)
output = outputs[0].outputs[0].text
print(output)
questions.extend(get_complex_question_from_output(output))
if len(questions) >= 2:
with open(save_complex_question_json, "a+", encoding="utf-8") as f:
json.dump(questions, f, ensure_ascii=False, indent=4)
print("{}条输入指令已经保存到{}".format(len(questions), save_complex_question_json))
break
def get_react_data():
if 'llm' not in locals():
llm = LLM(
model=model_path,
tensor_parallel_size=8,
max_model_len=4096,
dtype="bfloat16",
trust_remote_code=True,
enforce_eager=True,
gpu_memory_utilization=0.8,
)
with open(input_question_json, "r", encoding="utf-8") as file:
# 将json文件内容解析为Python对象
tool_questions = json.load(file)
all_questions = []
for i in tool_questions:
all_questions.extend(tool_questions[i])
react_question = [build_input_text([(q, "")], tools) for q in all_questions]
params_dict["top_k"] = 1
params_dict["stop"] = ["Observation:"]
sampling_params = SamplingParams(**params_dict)
for index in range(0, len(react_question), inference_batch_size):
react_qa = []
outputs = llm.generate(
react_question[index : index + inference_batch_size], sampling_params
)
for i in range(len(outputs)):
output = outputs[i].outputs[0].text
plugin_name, plugin_args, text = parse_latest_plugin_call(output)
excute_flag = True
for tool in tools:
print(tool)
if (
tool["name_for_model"] == plugin_name
and tool["excute_function"] == False
):
excute_flag = False
second_input = (
react_question[index + i] + output + "Observation: "
)
output2 = (
llm.generate(second_input, sampling_params)[0]
.outputs[0]
.text
)
if excute_flag:
observation = function_call(plugin_name, plugin_args,llm,tokenizer)
second_input = (
react_question[index + i]
+ output
+ "Observation: {}".format(observation)
)
output2 = llm.generate(second_input, sampling_params)[0].outputs[0].text
print(output2)
# react_qa.append({react_question[index+i]: second_input[len(react_question[index+i]):]+output2})
react_qa.append(
{
"instruction": "You are a helpful assistant.",
"input": react_question[index + i][75:-33],
"output": second_input[len(react_question[index + i]) :]
+ output2,
}
)
with open(save_react_qa_json, "a+", encoding="utf-8") as f:
json.dump(react_qa, f, ensure_ascii=False, indent=4)
print("{}条react qa数据已经保存到{}".format(len(react_qa), save_react_qa_json))
def get_complex_react_data():
if 'llm' not in locals():
llm = LLM(
model=model_path,
tensor_parallel_size=8,
max_model_len=4096,
dtype="bfloat16",
trust_remote_code=True,
enforce_eager=True,
gpu_memory_utilization=0.8,
)
with open(input_complex_question_json ,"r", encoding="utf-8") as file:
# 将json文件内容解析为Python对象
tool_questions = json.load(file)
all_questions = []
for i in tool_questions:
all_questions.append(''.join([f"{key}请按以下执行顺序完成以上问题,请注意A、B、C等等都是指代某些特殊值。:{value}" for key, value in i.items()]))
react_question = [build_input_text([(q, "")], tools) for q in all_questions]
params_dict["top_k"] = 1
params_dict["stop"] = ["Observation:"]
sampling_params = SamplingParams(**params_dict)
for index in range(0, len(react_question), inference_batch_size):
react_qa = []
outputs = llm.generate(
react_question[index : index + inference_batch_size], sampling_params
)
for i in range(len(outputs)):
output = outputs[i].outputs[0].text
all_output = react_question[index+i] + output
try:
while 'Final Answer' not in output:
plugin_name, plugin_args, text = parse_latest_plugin_call(output)
excute_flag = True
for tool in tools:
if (
tool["name_for_model"] == plugin_name
and tool["excute_function"] == False
):
excute_flag = False
second_input = (
all_output + "Observation: "
)
output = (
llm.generate(second_input, sampling_params)[0]
.outputs[0]
.text
)
all_output += "Observation: " + output
if excute_flag:
observation = function_call(plugin_name, plugin_args,llm,tokenizer)
second_input = (
all_output
+ "Observation: {}".format(observation)
)
output = llm.generate(second_input, sampling_params)[0].outputs[0].text
print(observation)
print(output)
all_output += "Observation: {}".format(observation)+output
# react_qa.append({react_question[index+i]: second_input[len(react_question[index+i]):]+output2})
react_qa.append(
{
"instruction": "You are a helpful assistant.",
"input": react_question[index + i][75:-33],
"output": all_output[len(react_question[index + i]):]
}
)
except:
print('#####{}#####'.format(all_questions[index+i]))
continue
with open(save_complex_react_qa_json, "a+", encoding="utf-8") as f:
json.dump(react_qa, f, ensure_ascii=False, indent=4)
print("{}条react qa数据已经保存到{}".format(len(react_qa), save_react_qa_json))
def get_complex_react_data_batch():
if 'llm' not in locals():
llm = LLM(
model=model_path,
tensor_parallel_size=8,
max_model_len=4096,
dtype="bfloat16",
trust_remote_code=True,
enforce_eager=True,
gpu_memory_utilization=0.8,
)
with open(input_complex_question_json ,"r", encoding="utf-8") as file:
# 将json文件内容解析为Python对象
tool_questions = json.load(file)
all_questions = []
for i in tool_questions:
all_questions.append(''.join([f"{key}请按以下执行顺序完成以上问题,请注意A、B、C等等都是指代某些特殊值。:{value}" for key, value in i.items()]))
react_question = [build_input_text([(q, "")], tools) for q in all_questions]
params_dict["top_k"] = 1
params_dict["stop"] = ["Observation:"]
sampling_params = SamplingParams(**params_dict)
point = inference_batch_size
batch_inputs = react_question[ : inference_batch_size]
exsit_num = len(react_question)
while exsit_num > 2:
outputs = llm.generate(
batch_inputs, sampling_params
)
batch_inputs = [batch_inputs[i] + outputs[i].outputs[0].text for i in range(len(batch_inputs))]
for i,line in enumerate(batch_inputs):
if 'Final Answer' in outputs[i].outputs[0].text:
data = {
"instruction": "You are a helpful assistant.",
"input": batch_inputs[i][75:].split('<|im_end|>\n<|im_start|>assistant\n')[0],
"output": batch_inputs[i].split('<|im_end|>\n<|im_start|>assistant\n')[1]
}
with open(save_complex_react_qa_json, "a+", encoding="utf-8") as f:
json.dump(data, f, ensure_ascii=False, indent=4)
print("{}条react qa数据已经保存到{}".format(1, save_complex_react_qa_json))
point += 1
batch_inputs[i] = react_question[point]
exsit_num -= 1
else:
plugin_name, plugin_args, text = parse_latest_plugin_call(outputs[i].outputs[0].text)
excute_flag = True
for tool in tools:
if (
tool["name_for_model"] == plugin_name
and tool["excute_function"] == False
):
excute_flag = False
batch_inputs[i] += "Observation: "
if excute_flag:
try:
observation = function_call(plugin_name, plugin_args,llm,tokenizer)
batch_inputs[i] += "Observation: {}".format(observation)
except:
data = {
"instruction": "You are a helpful assistant.",
"input": batch_inputs[i][75:].split('<|im_end|>\n<|im_start|>assistant\n')[0],
"output": batch_inputs[i].split('<|im_end|>\n<|im_start|>assistant\n')[1]
}
with open(except_comple_react_json, "a+", encoding="utf-8") as f:
json.dump(data, f, ensure_ascii=False, indent=4)
point += 1
batch_inputs[i] = react_question[point]
exsit_num -= 1
print("{}条react qa数据已经保存到{}".format(1, except_comple_react_json))
if __name__ == "__main__":
#get_question()# 获取简单Agent问题数据
#get_complex_question()
#get_react_data() # 获取agent执行过程数据
get_complex_react_data_batch() # 获取复杂agent的执行过程数据