Spaces:
Sleeping
Sleeping
feat: 增加多用户显示与多用户历史文件夹 (#489)
Browse files* 功能添加: 对每个用户添加独立history文件夹
* 修改: 无用户名时的兼容性 Fix Username
* UI美化
---------
Co-authored-by: Keldos <[email protected]>
- ChuanhuChatbot.py +17 -6
- assets/custom.css +13 -0
- modules/presets.py +2 -1
- modules/utils.py +20 -19
ChuanhuChatbot.py
CHANGED
|
@@ -19,6 +19,7 @@ with open("assets/custom.css", "r", encoding="utf-8") as f:
|
|
| 19 |
customCSS = f.read()
|
| 20 |
|
| 21 |
with gr.Blocks(css=customCSS, theme=small_and_beautiful_theme) as demo:
|
|
|
|
| 22 |
history = gr.State([])
|
| 23 |
token_count = gr.State([])
|
| 24 |
promptTemplates = gr.State(load_template(get_template_names(plain=True)[0], mode=2))
|
|
@@ -29,8 +30,18 @@ with gr.Blocks(css=customCSS, theme=small_and_beautiful_theme) as demo:
|
|
| 29 |
|
| 30 |
with gr.Row():
|
| 31 |
gr.HTML(title)
|
|
|
|
| 32 |
status_display = gr.Markdown(get_geoip(), elem_id="status_display")
|
| 33 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 34 |
with gr.Row().style(equal_height=True):
|
| 35 |
with gr.Column(scale=5):
|
| 36 |
with gr.Row():
|
|
@@ -323,27 +334,27 @@ with gr.Blocks(css=customCSS, theme=small_and_beautiful_theme) as demo:
|
|
| 323 |
# S&L
|
| 324 |
saveHistoryBtn.click(
|
| 325 |
save_chat_history,
|
| 326 |
-
[saveFileName, systemPromptTxt, history, chatbot],
|
| 327 |
downloadFile,
|
| 328 |
show_progress=True,
|
| 329 |
)
|
| 330 |
-
saveHistoryBtn.click(get_history_names,
|
| 331 |
exportMarkdownBtn.click(
|
| 332 |
export_markdown,
|
| 333 |
-
[saveFileName, systemPromptTxt, history, chatbot],
|
| 334 |
downloadFile,
|
| 335 |
show_progress=True,
|
| 336 |
)
|
| 337 |
-
historyRefreshBtn.click(get_history_names,
|
| 338 |
historyFileSelectDropdown.change(
|
| 339 |
load_chat_history,
|
| 340 |
-
[historyFileSelectDropdown, systemPromptTxt, history, chatbot],
|
| 341 |
[saveFileName, systemPromptTxt, history, chatbot],
|
| 342 |
show_progress=True,
|
| 343 |
)
|
| 344 |
downloadFile.change(
|
| 345 |
load_chat_history,
|
| 346 |
-
[downloadFile, systemPromptTxt, history, chatbot],
|
| 347 |
[saveFileName, systemPromptTxt, history, chatbot],
|
| 348 |
)
|
| 349 |
|
|
|
|
| 19 |
customCSS = f.read()
|
| 20 |
|
| 21 |
with gr.Blocks(css=customCSS, theme=small_and_beautiful_theme) as demo:
|
| 22 |
+
user_name = gr.State("")
|
| 23 |
history = gr.State([])
|
| 24 |
token_count = gr.State([])
|
| 25 |
promptTemplates = gr.State(load_template(get_template_names(plain=True)[0], mode=2))
|
|
|
|
| 30 |
|
| 31 |
with gr.Row():
|
| 32 |
gr.HTML(title)
|
| 33 |
+
user_ip = gr.Markdown(value="", elem_id="user_ip")
|
| 34 |
status_display = gr.Markdown(get_geoip(), elem_id="status_display")
|
| 35 |
|
| 36 |
+
# https://github.com/gradio-app/gradio/pull/3296
|
| 37 |
+
def create_greeting(request: gr.Request):
|
| 38 |
+
if hasattr(request, "username") and request.username: # is not None or is not ""
|
| 39 |
+
logging.info(f"Get User Name: {request.username}")
|
| 40 |
+
return gr.Markdown.update(value=f"User: {request.username}"), request.username
|
| 41 |
+
else:
|
| 42 |
+
return gr.Markdown.update(value=f"User: 管理员"), ""
|
| 43 |
+
demo.load(create_greeting, inputs=None, outputs=[user_ip, user_name])
|
| 44 |
+
|
| 45 |
with gr.Row().style(equal_height=True):
|
| 46 |
with gr.Column(scale=5):
|
| 47 |
with gr.Row():
|
|
|
|
| 334 |
# S&L
|
| 335 |
saveHistoryBtn.click(
|
| 336 |
save_chat_history,
|
| 337 |
+
[saveFileName, systemPromptTxt, history, chatbot, user_name],
|
| 338 |
downloadFile,
|
| 339 |
show_progress=True,
|
| 340 |
)
|
| 341 |
+
saveHistoryBtn.click(get_history_names, [gr.State(False), user_name], [historyFileSelectDropdown])
|
| 342 |
exportMarkdownBtn.click(
|
| 343 |
export_markdown,
|
| 344 |
+
[saveFileName, systemPromptTxt, history, chatbot, user_name],
|
| 345 |
downloadFile,
|
| 346 |
show_progress=True,
|
| 347 |
)
|
| 348 |
+
historyRefreshBtn.click(get_history_names, [gr.State(False), user_name], [historyFileSelectDropdown])
|
| 349 |
historyFileSelectDropdown.change(
|
| 350 |
load_chat_history,
|
| 351 |
+
[historyFileSelectDropdown, systemPromptTxt, history, chatbot, user_name],
|
| 352 |
[saveFileName, systemPromptTxt, history, chatbot],
|
| 353 |
show_progress=True,
|
| 354 |
)
|
| 355 |
downloadFile.change(
|
| 356 |
load_chat_history,
|
| 357 |
+
[downloadFile, systemPromptTxt, history, chatbot, user_name],
|
| 358 |
[saveFileName, systemPromptTxt, history, chatbot],
|
| 359 |
)
|
| 360 |
|
assets/custom.css
CHANGED
|
@@ -18,6 +18,19 @@ footer {
|
|
| 18 |
opacity: 0.85;
|
| 19 |
}
|
| 20 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 21 |
/* status_display */
|
| 22 |
#status_display {
|
| 23 |
display: flex;
|
|
|
|
| 18 |
opacity: 0.85;
|
| 19 |
}
|
| 20 |
|
| 21 |
+
/* user_ip */
|
| 22 |
+
#user_ip {
|
| 23 |
+
display: flex;
|
| 24 |
+
min-height: 2.5em;
|
| 25 |
+
align-items: flex-end;
|
| 26 |
+
justify-content: flex-end;
|
| 27 |
+
}
|
| 28 |
+
#user_ip p {
|
| 29 |
+
font-size: .85em;
|
| 30 |
+
font-family: monospace;
|
| 31 |
+
color: var(--body-text-color-subdued);
|
| 32 |
+
}
|
| 33 |
+
|
| 34 |
/* status_display */
|
| 35 |
#status_display {
|
| 36 |
display: flex;
|
modules/presets.py
CHANGED
|
@@ -1,5 +1,6 @@
|
|
| 1 |
# -*- coding:utf-8 -*-
|
| 2 |
import gradio as gr
|
|
|
|
| 3 |
|
| 4 |
# ChatGPT 设置
|
| 5 |
initial_prompt = "You are a helpful assistant."
|
|
@@ -7,7 +8,7 @@ API_HOST = "api.openai.com"
|
|
| 7 |
COMPLETION_URL = "https://api.openai.com/v1/chat/completions"
|
| 8 |
BALANCE_API_URL="https://api.openai.com/dashboard/billing/credit_grants"
|
| 9 |
USAGE_API_URL="https://api.openai.com/dashboard/billing/usage"
|
| 10 |
-
HISTORY_DIR = "history"
|
| 11 |
TEMPLATES_DIR = "templates"
|
| 12 |
|
| 13 |
# 错误信息
|
|
|
|
| 1 |
# -*- coding:utf-8 -*-
|
| 2 |
import gradio as gr
|
| 3 |
+
from pathlib import Path
|
| 4 |
|
| 5 |
# ChatGPT 设置
|
| 6 |
initial_prompt = "You are a helpful assistant."
|
|
|
|
| 8 |
COMPLETION_URL = "https://api.openai.com/v1/chat/completions"
|
| 9 |
BALANCE_API_URL="https://api.openai.com/dashboard/billing/credit_grants"
|
| 10 |
USAGE_API_URL="https://api.openai.com/dashboard/billing/usage"
|
| 11 |
+
HISTORY_DIR = Path("history")
|
| 12 |
TEMPLATES_DIR = "templates"
|
| 13 |
|
| 14 |
# 错误信息
|
modules/utils.py
CHANGED
|
@@ -190,46 +190,46 @@ def delete_last_conversation(chatbot, history, previous_token_count):
|
|
| 190 |
)
|
| 191 |
|
| 192 |
|
| 193 |
-
def save_file(filename, system, history, chatbot):
|
| 194 |
-
logging.info("保存对话历史中……")
|
| 195 |
-
os.makedirs(HISTORY_DIR, exist_ok=True)
|
| 196 |
if filename.endswith(".json"):
|
| 197 |
json_s = {"system": system, "history": history, "chatbot": chatbot}
|
| 198 |
print(json_s)
|
| 199 |
-
with open(os.path.join(HISTORY_DIR, filename), "w") as f:
|
| 200 |
json.dump(json_s, f)
|
| 201 |
elif filename.endswith(".md"):
|
| 202 |
md_s = f"system: \n- {system} \n"
|
| 203 |
for data in history:
|
| 204 |
md_s += f"\n{data['role']}: \n- {data['content']} \n"
|
| 205 |
-
with open(os.path.join(HISTORY_DIR, filename), "w", encoding="utf8") as f:
|
| 206 |
f.write(md_s)
|
| 207 |
-
logging.info("保存对话历史完毕")
|
| 208 |
-
return os.path.join(HISTORY_DIR, filename)
|
| 209 |
|
| 210 |
|
| 211 |
-
def save_chat_history(filename, system, history, chatbot):
|
| 212 |
if filename == "":
|
| 213 |
return
|
| 214 |
if not filename.endswith(".json"):
|
| 215 |
filename += ".json"
|
| 216 |
-
return save_file(filename, system, history, chatbot)
|
| 217 |
|
| 218 |
|
| 219 |
-
def export_markdown(filename, system, history, chatbot):
|
| 220 |
if filename == "":
|
| 221 |
return
|
| 222 |
if not filename.endswith(".md"):
|
| 223 |
filename += ".md"
|
| 224 |
-
return save_file(filename, system, history, chatbot)
|
| 225 |
|
| 226 |
|
| 227 |
-
def load_chat_history(filename, system, history, chatbot):
|
| 228 |
-
logging.info("加载对话历史中……")
|
| 229 |
if type(filename) != str:
|
| 230 |
filename = filename.name
|
| 231 |
try:
|
| 232 |
-
with open(os.path.join(HISTORY_DIR, filename), "r") as f:
|
| 233 |
json_s = json.load(f)
|
| 234 |
try:
|
| 235 |
if type(json_s["history"][0]) == str:
|
|
@@ -245,10 +245,10 @@ def load_chat_history(filename, system, history, chatbot):
|
|
| 245 |
except:
|
| 246 |
# 没有对话历史
|
| 247 |
pass
|
| 248 |
-
logging.info("加载对话历史完毕")
|
| 249 |
return filename, json_s["system"], json_s["history"], json_s["chatbot"]
|
| 250 |
except FileNotFoundError:
|
| 251 |
-
logging.info("没有找到对话历史文件,不执行任何操作")
|
| 252 |
return filename, system, history, chatbot
|
| 253 |
|
| 254 |
|
|
@@ -267,15 +267,16 @@ def get_file_names(dir, plain=False, filetypes=[".json"]):
|
|
| 267 |
files = sorted_by_pinyin(files)
|
| 268 |
if files == []:
|
| 269 |
files = [""]
|
|
|
|
| 270 |
if plain:
|
| 271 |
return files
|
| 272 |
else:
|
| 273 |
return gr.Dropdown.update(choices=files)
|
| 274 |
|
| 275 |
|
| 276 |
-
def get_history_names(plain=False):
|
| 277 |
-
logging.info("
|
| 278 |
-
return get_file_names(HISTORY_DIR, plain)
|
| 279 |
|
| 280 |
|
| 281 |
def load_template(filename, mode=0):
|
|
|
|
| 190 |
)
|
| 191 |
|
| 192 |
|
| 193 |
+
def save_file(filename, system, history, chatbot, user_name):
|
| 194 |
+
logging.info(f"{user_name} 保存对话历史中……")
|
| 195 |
+
os.makedirs(HISTORY_DIR / user_name, exist_ok=True)
|
| 196 |
if filename.endswith(".json"):
|
| 197 |
json_s = {"system": system, "history": history, "chatbot": chatbot}
|
| 198 |
print(json_s)
|
| 199 |
+
with open(os.path.join(HISTORY_DIR / user_name, filename), "w") as f:
|
| 200 |
json.dump(json_s, f)
|
| 201 |
elif filename.endswith(".md"):
|
| 202 |
md_s = f"system: \n- {system} \n"
|
| 203 |
for data in history:
|
| 204 |
md_s += f"\n{data['role']}: \n- {data['content']} \n"
|
| 205 |
+
with open(os.path.join(HISTORY_DIR / user_name, filename), "w", encoding="utf8") as f:
|
| 206 |
f.write(md_s)
|
| 207 |
+
logging.info(f"{user_name} 保存对话历史完毕")
|
| 208 |
+
return os.path.join(HISTORY_DIR / user_name, filename)
|
| 209 |
|
| 210 |
|
| 211 |
+
def save_chat_history(filename, system, history, chatbot, user_name):
|
| 212 |
if filename == "":
|
| 213 |
return
|
| 214 |
if not filename.endswith(".json"):
|
| 215 |
filename += ".json"
|
| 216 |
+
return save_file(filename, system, history, chatbot, user_name)
|
| 217 |
|
| 218 |
|
| 219 |
+
def export_markdown(filename, system, history, chatbot, user_name):
|
| 220 |
if filename == "":
|
| 221 |
return
|
| 222 |
if not filename.endswith(".md"):
|
| 223 |
filename += ".md"
|
| 224 |
+
return save_file(filename, system, history, chatbot, user_name)
|
| 225 |
|
| 226 |
|
| 227 |
+
def load_chat_history(filename, system, history, chatbot, user_name):
|
| 228 |
+
logging.info(f"{user_name} 加载对话历史中……")
|
| 229 |
if type(filename) != str:
|
| 230 |
filename = filename.name
|
| 231 |
try:
|
| 232 |
+
with open(os.path.join(HISTORY_DIR / user_name, filename), "r") as f:
|
| 233 |
json_s = json.load(f)
|
| 234 |
try:
|
| 235 |
if type(json_s["history"][0]) == str:
|
|
|
|
| 245 |
except:
|
| 246 |
# 没有对话历史
|
| 247 |
pass
|
| 248 |
+
logging.info(f"{user_name} 加载对话历史完毕")
|
| 249 |
return filename, json_s["system"], json_s["history"], json_s["chatbot"]
|
| 250 |
except FileNotFoundError:
|
| 251 |
+
logging.info(f"{user_name} 没有找到对话历史文件,不执行任何操作")
|
| 252 |
return filename, system, history, chatbot
|
| 253 |
|
| 254 |
|
|
|
|
| 267 |
files = sorted_by_pinyin(files)
|
| 268 |
if files == []:
|
| 269 |
files = [""]
|
| 270 |
+
logging.debug(f"files are:{files}")
|
| 271 |
if plain:
|
| 272 |
return files
|
| 273 |
else:
|
| 274 |
return gr.Dropdown.update(choices=files)
|
| 275 |
|
| 276 |
|
| 277 |
+
def get_history_names(plain=False, user_name=""):
|
| 278 |
+
logging.info(f"从用户 {user_name} 中获取历史记录文件名列表")
|
| 279 |
+
return get_file_names(HISTORY_DIR / user_name, plain)
|
| 280 |
|
| 281 |
|
| 282 |
def load_template(filename, mode=0):
|