FastAPI 學習之路(四十六)WebSockets(二)
阿新 • • 發佈:2021-10-24
上一篇文章,我們分享了WebSockets一些入門的,我們這節課,在原來的基礎上,對於講解的進行一個演示。我們最後分享了依賴token等。首先我們對上次的程式碼進行調整。
我們之前分享FastAPI 學習之路(三十八)Static Files,我們分享了動態檔案的使用,我們按照之前分享的。將之前程式碼的靜態檔案抽離出來。
放在了templates下面的webchat.html。
<!DOCTYPE html> <html> <head> <title>Chat</title> </head> <body> <h1>WebSocket 聊天</h1> <form action=""onsubmit="sendMessage(event)"> <input type="text" id="messageText" autocomplete="off"/> <button>Send</button> </form> <ul id='messages'> </ul> <script> var ws = new WebSocket("ws://localhost:8000/items/ws"); ws.onmessage = function (event) { var messages= document.getElementById('messages') var message = document.createElement('li') var content = document.createTextNode(event.data) message.appendChild(content) messages.appendChild(message) }; function sendMessage(event) { var input = document.getElementById("messageText") ws.send(input.value) input.value = '' event.preventDefault() } </script> </body> </html>
我們對於main裡面的程式碼調整為
from typing import Optional from fastapi import Cookie, Depends, FastAPI,Request, Query, WebSocket, status from fastapi.templating import Jinja2Templates app = FastAPI() templates = Jinja2Templates(directory="./templates") @app.get("/") async def get(request: Request): return templates.TemplateResponse( "webchat.html", { "request": request } ) async def get_cookie_or_token( websocket: WebSocket, session: Optional[str] = Cookie(None), token: Optional[str] = Query(None), ): if session is None and token is None: await websocket.close(code=status.WS_1008_POLICY_VIOLATION) return session or token @app.websocket("/items/ws") async def websocket_endpoint( websocket: WebSocket, cookie_or_token: str = Depends(get_cookie_or_token), ): await websocket.accept() while True: data = await websocket.receive_text() await websocket.send_text(f"訊息是: {data}")
但是我們之前的html程式碼去除錯的時候,發現報錯,因為我們需要依靠session或者token。那麼我們需要對html進行調整。
<!DOCTYPE html> <html> <head> <title>Chat</title> </head> <body> <h1>WebSocket Chat</h1> <form action="" onsubmit="sendMessage(event)"> <label>Token: <input type="text" id="token" autocomplete="off" value="some-key-token"/></label> <button onclick="connect(event)">連結</button> <hr> <label>訊息: <input type="text" id="messageText" autocomplete="off"/></label> <button>傳送</button> </form> <ul id='messages'> </ul> <script> var ws = null; function connect(event) { var token = document.getElementById("token") ws = new WebSocket("ws://localhost:8000/items/ws?token=" + token.value); ws.onmessage = function(event) { var messages = document.getElementById('messages') var message = document.createElement('li') var content = document.createTextNode(event.data) message.appendChild(content) messages.appendChild(message) }; event.preventDefault() } function sendMessage(event) { var input = document.getElementById("messageText") ws.send(input.value) input.value = '' event.preventDefault() } </script> </body> </html>
其實我們就是增加了帶了token。
但是我們直接點擊發送。無法傳送訊息,但是我們增加了帶token之後就可以傳送成功了。
這樣我們的WebSockets就可以帶token來做登入了,但是我們的token呢,只是做了簡單的校驗。那麼我們是不是可以和登入退出放在一起呢。肯定是可以的,我們在下次分享的時候將登入退出分享出來。
文章首發在公眾號,歡迎關注。