-
Notifications
You must be signed in to change notification settings - Fork 107
/
Copy pathchunked_transfer.py
85 lines (73 loc) · 2.9 KB
/
chunked_transfer.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
from fasthtml.common import *
from claudette import *
from starlette.responses import StreamingResponse
import asyncio
# Set up the app, including daisyui and tailwind for the chat component
hdrs = (picolink,
Script(src="https://cdn.tailwindcss.com"),
Link(rel="stylesheet", href="https://cdn.jsdelivr.net/npm/[email protected]/dist/full.min.css"))
app = FastHTML(hdrs=hdrs, ct_hdr=True, cls="p-4 max-w-lg mx-auto", live=True, debug=True)
@app.route("/{fname:path}.{ext:static}")
async def get(fname:str, ext:str):
return FileResponse(f'{fname}.{ext}')
# Set up a chat model (https://claudette.answer.ai/)
cli = Client(models[-1])
sp = "You are a helpful and concise assistant."
def ChatMessage(msg, user: bool, id=None):
bubble_class = "chat-bubble-primary" if user else "chat-bubble-secondary"
chat_class = "chat-end" if user else "chat-start"
return Div(cls=f"chat {chat_class}", id=f"msg-{id}")(
Div("user" if user else "assistant", cls="chat-header"),
Div(
msg,
cls=f"chat-bubble {bubble_class}",
id=f"msg-{id}-content" if id else None,
),
Hidden(
msg,
name="messages",
id=f"msg-{id}-hidden" if id else None,
),
)
# The input field for the user message. Also used to clear the
# input field after sending a message via an OOB swap
def ChatInput():
return Input(name='msg', id='msg-input', placeholder="Type a message",
cls="input input-bordered w-full", hx_swap_oob='true')
# The main screen
@app.get
def index():
page = Form(hx_post=send, hx_target="#chatlist", hx_swap="beforeend", hx_ext="chunked-transfer", hx_disabled_elt="#msg-group")(
Div(id="chatlist", cls="chat-box h-[73vh] overflow-y-auto"),
Div(cls="flex space-x-2 mt-2")(
Group(ChatInput(), Button("Send", cls="btn btn-primary"), id="msg-group")
)
)
return Titled('Chatbot Demo', page)
async def stream_response(msg, messages):
yield to_xml(ChatMessage(msg, True, id=len(messages)-1))
yield to_xml(ChatMessage('', False, id=len(messages)))
r = (cli(messages, sp=sp, stream=True))
response_txt = ''
for chunk in r:
response_txt += chunk
yield to_xml(Div(
response_txt,
cls=f"chat-bubble chat-bubble-secondary",
id=f"msg-{len(messages)}-content",
hx_swap_oob="outerHTML",
))
await asyncio.sleep(0.2)
yield to_xml(Hidden(
response_txt,
name="messages",
id=f"msg-{len(messages)}-hidden",
hx_swap_oob="outerHTML",
))
yield to_xml(ChatInput())
@app.post
async def send(msg:str, messages:list[str]=None):
if not messages: messages = []
messages.append(msg.rstrip())
return StreamingResponse(stream_response(msg, messages), media_type="text/plain", headers={"X-Transfer-Encoding": "chunked"})
serve()