QChatGPT/pkg/openai/session.py

182 lines
5.7 KiB
Python
Raw Normal View History

2022-12-09 00:01:49 +08:00
import logging
2022-12-07 22:27:05 +08:00
import time
import pkg.openai.manager
2022-12-08 00:41:35 +08:00
import pkg.database.manager
2022-12-07 22:27:05 +08:00
2022-12-08 00:41:35 +08:00
sessions = {}
2022-12-07 22:27:05 +08:00
2022-12-08 00:41:35 +08:00
2022-12-08 13:22:54 +08:00
class SessionOfflineStatus:
ON_GOING = 'on_going'
EXPLICITLY_CLOSED = 'explicitly_closed'
2022-12-08 00:41:35 +08:00
def load_sessions():
global sessions
db_inst = pkg.database.manager.get_inst()
session_data = db_inst.load_valid_sessions()
for session_name in session_data:
2022-12-09 00:01:49 +08:00
logging.info('加载session: {}'.format(session_name))
2022-12-08 00:41:35 +08:00
temp_session = Session(session_name)
temp_session.name = session_name
temp_session.create_timestamp = session_data[session_name]['create_timestamp']
temp_session.last_interact_timestamp = session_data[session_name]['last_interact_timestamp']
temp_session.prompt = session_data[session_name]['prompt']
sessions[session_name] = temp_session
def get_session(session_name: str):
global sessions
if session_name not in sessions:
sessions[session_name] = Session(session_name)
return sessions[session_name]
def dump_session(session_name: str):
global sessions
if session_name in sessions:
assert isinstance(sessions[session_name], Session)
sessions[session_name].persistence()
del sessions[session_name]
2022-12-07 22:27:05 +08:00
# 通用的OpenAI API交互session
class Session:
name = ''
prompt = ''
user_name = 'You'
bot_name = 'Bot'
create_timestamp = 0
last_interact_timestamp = 0
2022-12-08 21:58:02 +08:00
just_switched_to_exist_session = False
2022-12-07 22:27:05 +08:00
def __init__(self, name: str):
self.name = name
self.create_timestamp = int(time.time())
2022-12-08 14:28:46 +08:00
self.last_interact_timestamp = int(time.time())
2022-12-07 22:27:05 +08:00
# 请求回复
# 这个函数是阻塞的
def append(self, text: str) -> str:
2022-12-08 00:41:35 +08:00
self.prompt += self.user_name + ':' + text + '\n' + self.bot_name + ':'
2022-12-07 22:27:05 +08:00
self.last_interact_timestamp = int(time.time())
# 向API请求补全
2022-12-09 16:17:04 +08:00
response = pkg.openai.manager.get_inst().request_completion(self.cut_out(self.prompt + self.user_name + ':' +
text + '\n' + self.bot_name + ':',
7, 1024), self.user_name + ':')
2022-12-07 22:27:05 +08:00
2022-12-09 16:17:04 +08:00
self.prompt += self.user_name + ':' + text + '\n' + self.bot_name + ':'
2022-12-07 22:50:49 +08:00
# print(response)
2022-12-07 22:27:05 +08:00
# 处理回复
res_test = response["choices"][0]["text"]
res_ans = res_test
# 去除开头可能的提示
res_ans_spt = res_test.split("\n\n")
if len(res_ans_spt) > 1:
del (res_ans_spt[0])
res_ans = '\n\n'.join(res_ans_spt)
2022-12-07 22:50:49 +08:00
self.prompt += "{}".format(res_ans) + '\n'
2022-12-08 21:58:02 +08:00
if self.just_switched_to_exist_session:
self.just_switched_to_exist_session = False
self.set_ongoing()
2022-12-07 22:27:05 +08:00
return res_ans
2022-12-09 16:17:04 +08:00
# 截取prompt里不多于max_rounds个回合长度为大于max_tokens的最小整数字符串
# 保证都是完整的对话
def cut_out(self, prompt: str, max_rounds: int, max_tokens: int) -> str:
# 分隔出每个回合
rounds_spt_by_user_name = prompt.split(self.user_name + ':')
result = ''
checked_rounds = 0
# 从后往前遍历加到result前面检查result是否符合要求
for i in range(len(rounds_spt_by_user_name) - 1, 0, -1):
result = self.user_name + ':' + rounds_spt_by_user_name[i] + result
checked_rounds += 1
if checked_rounds >= max_rounds:
break
if len(result) > max_tokens:
break
logging.debug('cut_out: {}'.format(result))
return result
2022-12-07 22:27:05 +08:00
def persistence(self):
2022-12-08 14:28:46 +08:00
if self.prompt == '':
return
2022-12-08 00:41:35 +08:00
db_inst = pkg.database.manager.get_inst()
name_spt = self.name.split('_')
subject_type = name_spt[0]
subject_number = int(name_spt[1])
db_inst.persistence_session(subject_type, subject_number, self.create_timestamp, self.last_interact_timestamp,
self.prompt)
2022-12-08 12:06:04 +08:00
2022-12-08 13:22:54 +08:00
def reset(self, explicit: bool = False):
2022-12-08 12:06:04 +08:00
if self.prompt != '':
self.persistence()
2022-12-08 13:22:54 +08:00
if explicit:
pkg.database.manager.get_inst().explicit_close_session(self.name, self.create_timestamp)
2022-12-08 12:06:04 +08:00
self.prompt = ''
self.create_timestamp = int(time.time())
2022-12-08 14:28:46 +08:00
self.last_interact_timestamp = int(time.time())
2022-12-08 21:58:02 +08:00
self.just_switched_to_exist_session = False
# 将本session的数据库状态设置为on_going
def set_ongoing(self):
pkg.database.manager.get_inst().set_session_ongoing(self.name, self.create_timestamp)
2022-12-08 12:06:04 +08:00
2022-12-08 14:28:46 +08:00
# 切换到上一个session
2022-12-08 12:06:04 +08:00
def last_session(self):
2022-12-08 14:28:46 +08:00
last_one = pkg.database.manager.get_inst().last_session(self.name, self.last_interact_timestamp)
if last_one is None:
return None
else:
self.persistence()
self.create_timestamp = last_one['create_timestamp']
self.last_interact_timestamp = last_one['last_interact_timestamp']
self.prompt = last_one['prompt']
2022-12-08 21:58:02 +08:00
just_switched = True
2022-12-08 14:28:46 +08:00
return self
2022-12-08 12:06:04 +08:00
def next_session(self):
2022-12-08 14:28:46 +08:00
next_one = pkg.database.manager.get_inst().next_session(self.name, self.last_interact_timestamp)
if next_one is None:
return None
else:
self.persistence()
self.create_timestamp = next_one['create_timestamp']
self.last_interact_timestamp = next_one['last_interact_timestamp']
self.prompt = next_one['prompt']
2022-12-08 21:58:02 +08:00
just_switched = True
2022-12-08 14:28:46 +08:00
return self
def list_history(self, capacity: int = 10, page: int = 0):
return pkg.database.manager.get_inst().list_history(self.name, capacity, page)