import aiohttp import asyncio import httpx import json import pprint import urllib from conversation_creator import ConversationCreator from chathub_request_constructor import ChathubRequestConstructor from logger.logger import logger http_proxy = "http://localhost:11111" # Replace with yours def serialize_websocket_message(msg: dict) -> str: return json.dumps(msg, ensure_ascii=False) + "\x1e" class ConversationConnector: def __init__( self, sec_access_token=None, client_id=None, conversation_id=None, invocation_id=0, cookies={}, ): self.sec_access_token = sec_access_token self.client_id = client_id self.conversation_id = conversation_id self.invocation_id = invocation_id self.cookies = cookies self.ws_url = ( "wss://sydney.bing.com/sydney/ChatHub" + f"?sec_access_token={urllib.parse.quote(self.sec_access_token)}" ) async def _init_handshake(self): await self.wss.send_str( serialize_websocket_message({"protocol": "json", "version": 1}) ) await self.wss.receive_str() await self.wss.send_str(serialize_websocket_message({"type": 6})) async def stream_chat(self, prompt=""): self.aiohttp_session = aiohttp.ClientSession(cookies=self.cookies) request_headers = { "Accept-Encoding": " gzip, deflate, br", "Accept-Language": "en-US,en;q=0.9,zh-CN;q=0.8,zh;q=0.7", "Cache-Control": "no-cache", "Connection": "Upgrade", "Host": "sydney.bing.com", "Origin": "https://www.bing.com", "Pragma": "no-cache", "Sec-Websocket-Extensions": "permessage-deflate; client_max_window_bits", "Sec-Websocket-Version": "13", "Upgrade": "websocket", "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/118.0.0.0 Safari/537.36", } self.wss = await self.aiohttp_session.ws_connect( self.ws_url, headers=request_headers, proxy=http_proxy, ) await self._init_handshake() chathub_request_constructor = ChathubRequestConstructor( prompt=prompt, conversation_style="precise", client_id=self.client_id, conversation_id=self.conversation_id, invocation_id=self.invocation_id, ) chathub_request_constructor.construct() await self.wss.send_str( serialize_websocket_message(chathub_request_constructor.request_payload) ) delta_content_pointer = 0 while not self.wss.closed: response_lines_str = await self.wss.receive_str() if isinstance(response_lines_str, str): response_lines = response_lines_str.split("\x1e") else: continue for line in response_lines: if not line: continue data = json.loads(line) # Stream: Meaningful Messages if data.get("type") == 1: arguments = data["arguments"][0] if arguments.get("throttling"): throttling = arguments.get("throttling") # pprint.pprint(throttling) if arguments.get("messages"): for message in arguments.get("messages"): message_type = message.get("messageType") # Message: Displayed answer if message_type is None: content = message["adaptiveCards"][0]["body"][0]["text"] delta_content = content[delta_content_pointer:] logger.mesg(delta_content, end="") delta_content_pointer = len(content) # Message: Suggested Questions if message.get("suggestedResponses"): logger.note("\n\nSuggested Questions: ") for suggestion in message.get("suggestedResponses"): suggestion_text = suggestion.get("text") logger.file(f"- {suggestion_text}") # Message: Search Query elif message_type in ["InternalSearchQuery"]: message_hidden_text = message["hiddenText"] logger.note(f"\n[Searching: [{message_hidden_text}]]") # Message: Internal Search Results elif message_type in ["InternalSearchResult"]: logger.note("[Analyzing search results ...]") # Message: Loader status, such as "Generating Answers" elif message_type in ["InternalLoaderMessage"]: logger.note("[Generating answers ...]\n") # Message: Render Cards for Webpages elif message_type in ["RenderCardRequest"]: continue # Message: Not Implemented else: raise NotImplementedError( f"Not Supported Message Type: {message_type}" ) # Stream: List of whole conversation messages elif data.get("type") == 2: if data.get("item"): item = data.get("item") logger.note("\n[Saving chat messages ...]") # for message in item.get("messages"): # author = message["author"] # message_text = message["text"] # Stream: End of Conversation elif data.get("type") == 3: logger.success("[Finished]") self.invocation_id += 1 await self.wss.close() await self.aiohttp_session.close() break # Stream: Signal elif data.get("type") == 6: continue # Stream: Not Monitored else: # pprint.pprint(data) continue if __name__ == "__main__": creator = ConversationCreator() creator.create() connector = ConversationConnector( sec_access_token=creator.response_headers[ "x-sydney-encryptedconversationsignature" ], client_id=creator.response_content["clientId"], conversation_id=creator.response_content["conversationId"], ) prompt = "Today's weather of California" # prompt = "Tell me your name. Your output should be no more than 3 words." logger.success(f"\n[User]: ", end="") logger.mesg(f"{prompt}") logger.success(f"\n[Bing]:") loop = asyncio.get_event_loop() loop.run_until_complete(connector.stream_chat(prompt=prompt)) loop.close()