import aiohttp import asyncio import httpx import json import pprint import urllib from chathub_request_constructor import ChathubRequestConstructor from logger.logger import logger http_proxy = "http://localhost:11111" # Replace with yours class ConversationCreator: conversation_create_url = "https://www.bing.com/turing/conversation/create" def __init__(self, cookies={}): self.cookies = cookies self.construct_cookies() def construct_cookies(self): self.httpx_cookies = httpx.Cookies() for key, val in self.cookies.items(): self.httpx_cookies.set(key, val) def create(self, proxy=None): self.response = httpx.get( self.conversation_create_url, proxies=http_proxy if proxy is None else proxy, cookies=self.httpx_cookies, ) self.response_content = json.loads(self.response.content.decode("utf-8")) self.response_headers = dict(self.response.headers) pprint.pprint(self.response_content) # pprint.pprint(self.response_headers) 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_message) ) 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) 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") if message_type is None: # Displayed message does not contain 'messageType' message_html = message["adaptiveCards"][0]["body"][0][ "text" ] delta_content = message_html[delta_content_pointer:] logger.mesg(delta_content, end="") delta_content_pointer = len(message_html) 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}") elif message_type in ["InternalSearchQuery"]: message_hidden_text = message["hiddenText"] logger.note(f"\n[Searching: [{message_hidden_text}]]") elif message_type in [ "InternalSearchResult", ]: logger.note("[Analyzing search results ...]") elif message_type in ["InternalLoaderMessage"]: logger.note("[Generating answers ...]\n") elif message_type in ["RenderCardRequest"]: continue else: raise NotImplementedError( f"Not Supported Message Type: {message_type}" ) 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"] elif data.get("type") == 3: logger.success("[Finished]") await self.wss.close() await self.aiohttp_session.close() break elif data.get("type") == 6: continue else: # pprint.pprint(data) continue if __name__ == "__main__": creator = ConversationCreator() creator.create() conversation_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(conversation_connector.stream_chat(prompt=prompt)) loop.close()