# import gradio as gr
import gradio
# import lmdb
# import base64
# import io
# import random
# import time
import json
import copy
# import sqlite3
from urllib.parse import urljoin
import openai
from app_js import api_key__get_from_browser, api_key__save_to_browser, saved_prompts_refresh_btn__click_js, selected_saved_prompt_title__change_js, saved_prompts_delete_btn__click_js, saved_prompts_save_btn__click_js, copy_prompt__click_js, paste_prompt__click_js, chat_copy_history_btn__click_js, chat_copy_history_md_btn__click_js, api_key_refresh_btn__click_js, api_key_save_btn__click_js
from functions import sequential_chat_fn, make_history_file_fn, on_click_send_btn, clear_history, copy_history, update_saved_prompt_titles, save_prompt, load_saved_prompt
introduction = """
ChatGPT 批处理工具
您好。这是一个用于批量向 ChatGPT 发送消息的工具。
通过这个工具,您可以一次性计划好要给 ChatGPT 发送哪些消息,并依次发送。
请注意:
1. 为了使用本工具,您需要填写自己的 API Key ,并承担可能的费用。我们不会收集或存储您的 API Key 。您可访问 https://platform.openai.com/account/api-keys 来获取您的 API Key 。
2. 这个 demo 页面的 space 是公共的。出于研究和改善代码的需要,我们需要记录通过这个页面发送的聊天内容,因此,我们有能力在后台看到您与 ChatGPT 的聊天记录。 **如果您继续在此页面使用本工具,意味着您同意我们查看、使用、传播您的聊天数据。** 如果您希望避免这一情况,可以 [将此工具复制一份到自己专用的 space](https://huggingface.co/spaces/hugforziio/chat-gpt-batch?duplicate=true) ,同时也可以避免排队等候。
"""
css = """
.table-wrap .cell-wrap input {min-width:80%}
#api-key-textbox textarea {filter:blur(8px); transition: filter 0.25s}
#api-key-textbox textarea:focus {filter:none}
#chat-log-md hr {margin-top: 1rem; margin-bottom: 1rem;}
#chat-markdown-wrap-box {max-height:80vh; overflow: auto !important;}
"""
with gradio.Blocks(title="ChatGPT 批处理", css=css) as demo:
with gradio.Accordion("说明", open=True):
gradio.Markdown(introduction)
with gradio.Accordion("基本设置", open=True):
system_prompt_enabled = gradio.Checkbox(label='是否使用系统全局提示语', info='是否要以“系统”身份,给 ChatGPT 描述任务?', value=True)
# 系统提示
system_prompt = gradio.Textbox(label='系统级全局提示语', info='以“系统”身份,给 ChatGPT 描述任务', value='你是一个词性分类器,用户将给你发送一个单词,请判断该单词的词性,如名词、动词等。!!请注意!!⚠️最高优先级!!:你只能直接返回词性,而不要返回任何多余的内容,🈲不要解释为什么是这个词性等等,否则用户所使用的程序将会出错,给用户带来严重的损失😱!!!')
# 用户消息模板
user_message_template = gradio.Textbox(label='用户消息模板', info='要批量发送的消息的模板', value='单词:```___```')
with gradio.Row():
# 用户消息模板中的替换区
user_message_template_mask = gradio.Textbox(label='模板占位符', info='消息模板中需要被替换的部分,可以是正则表达式', value='___')
# 用户消息模板中的替换区是正则吗
user_message_template_mask_is_regex = gradio.Checkbox(label='模板占位符是正则吗', info='模板占位符是不是正则表达式?', value=False)
# 用户消息替换区清单文本
user_message_list_text = gradio.Textbox(label='用户消息列表', info='所有待发送的消息', value='动物 火车 介于 的 置于 你在做什么')
with gradio.Row():
# 用户消息替换区清单分隔符
user_message_list_text_splitter = gradio.Textbox(label='用户消息分隔符', info='用于分割用户消息列表的分隔符,如逗号(`,`)、换行符(`\\n`)等,也可以是正则表达式', value='\\s+')
# 用户消息替换区清单分隔符是正则吗
user_message_list_text_splitter_is_regex = gradio.Checkbox(label='分隔符是正则吗', info='用户消息分隔符是不是正则表达式?', value=True)
# 历史记录条数
history_prompt_num = gradio.Slider(label="发送历史记录条数", info='每次发生消息时,同时携带多少条先前的历史记录(以便 ChatGPT 了解上下文)', value=0, minimum=0, maximum=12000)
# load_config_from_browser = gradio.Button("🔄 从浏览器加载配置")
# save_config_to_browser = gradio.Button("💾 将配置保存到浏览器")
# export_config_to_file = gradio.Button("📤 将配置导出为文件")
# 更多参数
with gradio.Accordion("更多参数", open=False):
# 时间间隔
sleep_base = gradio.Number(label='时间间隔 ms', value=700)
# 时间间隔浮动
sleep_rand = gradio.Number(label='时间间隔浮动 ms', value=200)
# 那些参数
prop_stream = gradio.Checkbox(label="流式传输 stream", value=True)
prop_model = gradio.Textbox(label="模型 model", value="gpt-3.5-turbo")
prop_temperature = gradio.Slider(label="temperature", value=1, minimum=0, maximum=2)
prop_top_p = gradio.Slider(label="top_p", value=1, minimum=0, maximum=1)
prop_choices_num = gradio.Slider(label="choices num(n)", value=1, minimum=1, maximum=20)
prop_max_tokens = gradio.Slider(label="max_tokens", value=-1, minimum=-1, maximum=4096)
prop_presence_penalty = gradio.Slider(label="presence_penalty", value=0, minimum=-2, maximum=2)
prop_frequency_penalty = gradio.Slider(label="frequency_penalty", value=0, minimum=-2, maximum=2)
prop_logit_bias = gradio.Textbox(label="logit_bias", visible=False)
pass
# 欸丕艾科易
token_text = gradio.Textbox(visible=False)
with gradio.Row():
with gradio.Column(scale=10, min_width=100):
api_key_text = gradio.Textbox(label="你的 API key", placeholder="sk-...", elem_id="api-key-textbox")
with gradio.Column(scale=1, min_width=100):
api_key_load_btn = gradio.Button("🔄 从浏览器本地存储加载")
api_key_load_btn.click(
None,
inputs=[],
outputs=[api_key_text, token_text],
_js=api_key__get_from_browser,
)
with gradio.Column(scale=1, min_width=100):
api_key_save_btn = gradio.Button("💾 保存到浏览器本地存储")
api_key_save_btn.click(
None,
inputs=[api_key_text, token_text],
outputs=[api_key_text, token_text],
_js=api_key__save_to_browser,
)
pass
pass
# 开始执行按钮
start_btn = gradio.Button(value='开始!')
with gradio.Accordion(label="聊天记录", elem_id='chat-markdown-wrap-box'):
# 输出区域(隐藏状态)
history = gradio.State(value=[])
# 输出区域(md渲染)
history_md_stable = gradio.Markdown(value="🙂")
history_md_stream = gradio.Markdown(value="🤖")
with gradio.Accordion("状态"):
tips = gradio.Markdown(value="待命")
# 中止执行按钮
stop_btn = gradio.Button(value='中止!')
with gradio.Accordion("下载聊天记录", open=False):
# gradio.Markdown("(暂时无法下载,可能是 Hugging Face 的限制,之后更新)")
make_file_btn = gradio.Button(value='生成文件')
with gradio.Row(visible=False) as file_row:
# 下载区域(json文件)
history_file_json = gradio.File(label='Json 下载', interactive=False)
# 下载区域(md文件)
history_file_md = gradio.File(label='Markdown 下载', interactive=False)
pass
pass
make_file_btn.click(
fn=make_history_file_fn,
inputs=[history],
outputs=[history_file_json, history_file_md, file_row],
)
start_event = start_btn.click(
fn=sequential_chat_fn,
inputs=[
history,
system_prompt_enabled,
system_prompt,
user_message_template,
user_message_template_mask,
user_message_template_mask_is_regex,
user_message_list_text,
user_message_list_text_splitter,
user_message_list_text_splitter_is_regex,
history_prompt_num,
api_key_text, token_text,
sleep_base,
sleep_rand,
prop_stream,
prop_model,
prop_temperature,
prop_top_p,
prop_choices_num,
prop_max_tokens,
prop_presence_penalty,
prop_frequency_penalty,
prop_logit_bias,
],
outputs=[
history,
history_md_stable,
history_md_stream,
tips,
file_row,
],
)
stop_btn.click(
fn=None,
inputs=[],
outputs=[],
cancels=[start_event],
)
if __name__ == "__main__":
demo.queue(concurrency_count=200).launch()