2023-07-28 13:07:17 +03:00
|
|
|
import asyncio
|
|
|
|
import json
|
|
|
|
import os
|
|
|
|
import random
|
|
|
|
|
|
|
|
import aiohttp
|
2023-08-21 23:39:57 +03:00
|
|
|
import asyncio
|
|
|
|
from aiohttp import ClientSession
|
2023-07-28 13:07:17 +03:00
|
|
|
|
2023-08-21 23:39:57 +03:00
|
|
|
from ..typing import Any, AsyncGenerator, CreateResult, Union
|
2023-08-27 19:58:36 +03:00
|
|
|
from .base_provider import BaseProvider
|
2023-07-28 13:07:17 +03:00
|
|
|
|
2023-08-27 19:58:36 +03:00
|
|
|
class Bing(BaseProvider):
|
2023-07-28 13:07:17 +03:00
|
|
|
url = "https://bing.com/chat"
|
|
|
|
supports_gpt_4 = True
|
2023-08-27 19:58:36 +03:00
|
|
|
working=True
|
2023-08-25 07:41:32 +03:00
|
|
|
supports_stream=True
|
2023-08-27 19:58:36 +03:00
|
|
|
|
2023-08-23 03:16:35 +03:00
|
|
|
@staticmethod
|
2023-08-27 19:58:36 +03:00
|
|
|
def create_completion(
|
|
|
|
model: str,
|
|
|
|
messages: list[dict[str, str]],
|
|
|
|
stream: bool,
|
|
|
|
**kwargs: Any
|
|
|
|
) -> CreateResult:
|
|
|
|
yield from run(create(messages, **kwargs))
|
|
|
|
|
|
|
|
def create(
|
|
|
|
messages: list[dict[str, str]],
|
|
|
|
cookies: dict = {}
|
|
|
|
):
|
|
|
|
if len(messages) < 2:
|
|
|
|
prompt = messages[0]["content"]
|
|
|
|
context = None
|
|
|
|
|
|
|
|
else:
|
|
|
|
prompt = messages[-1]["content"]
|
|
|
|
context = convert(messages[:-1])
|
|
|
|
|
|
|
|
if not cookies:
|
|
|
|
cookies = {
|
|
|
|
'MUID': '',
|
|
|
|
'BCP': '',
|
|
|
|
'MUIDB': '',
|
|
|
|
'USRLOC': '',
|
|
|
|
'SRCHD': 'AF=hpcodx',
|
|
|
|
'MMCASM': '',
|
|
|
|
'_UR': '',
|
|
|
|
'ANON': '',
|
|
|
|
'NAP': '',
|
|
|
|
'ABDEF': '',
|
|
|
|
'PPLState': '1',
|
|
|
|
'KievRPSSecAuth': '',
|
|
|
|
'_U': '',
|
|
|
|
'SUID': '',
|
|
|
|
'_EDGE_S': '',
|
|
|
|
'WLS': '',
|
|
|
|
'_HPVN': '',
|
|
|
|
'_SS': '',
|
|
|
|
'_clck': '',
|
|
|
|
'SRCHUSR': '',
|
|
|
|
'_RwBf': '',
|
|
|
|
'SRCHHPGUSR': '',
|
|
|
|
'ipv6': '',
|
|
|
|
}
|
|
|
|
|
|
|
|
return stream_generate(prompt, context, cookies)
|
|
|
|
|
|
|
|
def convert(messages: list[dict[str, str]]):
|
2023-07-28 13:07:17 +03:00
|
|
|
context = ""
|
|
|
|
|
|
|
|
for message in messages:
|
|
|
|
context += "[%s](#message)\n%s\n\n" % (message["role"], message["content"])
|
|
|
|
|
|
|
|
return context
|
|
|
|
|
2023-08-21 23:39:57 +03:00
|
|
|
class Conversation():
|
|
|
|
def __init__(self, conversationId: str, clientId: str, conversationSignature: str) -> None:
|
|
|
|
self.conversationId = conversationId
|
|
|
|
self.clientId = clientId
|
|
|
|
self.conversationSignature = conversationSignature
|
|
|
|
|
|
|
|
async def create_conversation(session: ClientSession) -> Conversation:
|
|
|
|
url = 'https://www.bing.com/turing/conversation/create'
|
|
|
|
async with await session.get(url) as response:
|
|
|
|
response = await response.json()
|
|
|
|
conversationId = response.get('conversationId')
|
|
|
|
clientId = response.get('clientId')
|
|
|
|
conversationSignature = response.get('conversationSignature')
|
|
|
|
|
|
|
|
if not conversationId or not clientId or not conversationSignature:
|
|
|
|
raise Exception('Failed to create conversation.')
|
|
|
|
|
|
|
|
return Conversation(conversationId, clientId, conversationSignature)
|
|
|
|
|
|
|
|
async def list_conversations(session: ClientSession) -> list:
|
|
|
|
url = "https://www.bing.com/turing/conversation/chats"
|
|
|
|
async with session.get(url) as response:
|
|
|
|
response = await response.json()
|
|
|
|
return response["chats"]
|
|
|
|
|
|
|
|
async def delete_conversation(session: ClientSession, conversation: Conversation) -> list:
|
|
|
|
url = "https://sydney.bing.com/sydney/DeleteSingleConversation"
|
|
|
|
json = {
|
|
|
|
"conversationId": conversation.conversationId,
|
|
|
|
"conversationSignature": conversation.conversationSignature,
|
|
|
|
"participant": {"id": conversation.clientId},
|
2023-07-28 13:07:17 +03:00
|
|
|
"source": "cib",
|
2023-08-21 23:39:57 +03:00
|
|
|
"optionsSets": ["autosave"]
|
2023-07-28 13:07:17 +03:00
|
|
|
}
|
2023-08-21 23:39:57 +03:00
|
|
|
async with session.post(url, json=json) as response:
|
|
|
|
response = await response.json()
|
|
|
|
return response["result"]["value"] == "Success"
|
2023-07-28 13:07:17 +03:00
|
|
|
|
|
|
|
class Defaults:
|
|
|
|
delimiter = "\x1e"
|
|
|
|
ip_address = f"13.{random.randint(104, 107)}.{random.randint(0, 255)}.{random.randint(0, 255)}"
|
|
|
|
|
|
|
|
allowedMessageTypes = [
|
|
|
|
"Chat",
|
|
|
|
"Disengaged",
|
|
|
|
"AdsQuery",
|
|
|
|
"SemanticSerp",
|
|
|
|
"GenerateContentQuery",
|
|
|
|
"SearchQuery",
|
|
|
|
"ActionRequest",
|
|
|
|
"Context",
|
|
|
|
"Progress",
|
|
|
|
"AdsQuery",
|
|
|
|
"SemanticSerp",
|
|
|
|
]
|
|
|
|
|
|
|
|
sliceIds = [
|
|
|
|
"winmuid3tf",
|
|
|
|
"osbsdusgreccf",
|
|
|
|
"ttstmout",
|
|
|
|
"crchatrev",
|
|
|
|
"winlongmsgtf",
|
|
|
|
"ctrlworkpay",
|
|
|
|
"norespwtf",
|
|
|
|
"tempcacheread",
|
|
|
|
"temptacache",
|
|
|
|
"505scss0",
|
|
|
|
"508jbcars0",
|
|
|
|
"515enbotdets0",
|
|
|
|
"5082tsports",
|
|
|
|
"515vaoprvs",
|
|
|
|
"424dagslnv1s0",
|
|
|
|
"kcimgattcf",
|
|
|
|
"427startpms0",
|
|
|
|
]
|
|
|
|
|
|
|
|
location = {
|
|
|
|
"locale": "en-US",
|
|
|
|
"market": "en-US",
|
|
|
|
"region": "US",
|
|
|
|
"locationHints": [
|
|
|
|
{
|
|
|
|
"country": "United States",
|
|
|
|
"state": "California",
|
|
|
|
"city": "Los Angeles",
|
|
|
|
"timezoneoffset": 8,
|
|
|
|
"countryConfidence": 8,
|
|
|
|
"Center": {"Latitude": 34.0536909, "Longitude": -118.242766},
|
|
|
|
"RegionType": 2,
|
|
|
|
"SourceType": 1,
|
|
|
|
}
|
|
|
|
],
|
|
|
|
}
|
|
|
|
|
2023-08-21 23:39:57 +03:00
|
|
|
headers = {
|
|
|
|
'accept': '*/*',
|
|
|
|
'accept-language': 'en-US,en;q=0.9',
|
|
|
|
'cache-control': 'max-age=0',
|
|
|
|
'sec-ch-ua': '"Chromium";v="110", "Not A(Brand";v="24", "Microsoft Edge";v="110"',
|
|
|
|
'sec-ch-ua-arch': '"x86"',
|
|
|
|
'sec-ch-ua-bitness': '"64"',
|
|
|
|
'sec-ch-ua-full-version': '"110.0.1587.69"',
|
|
|
|
'sec-ch-ua-full-version-list': '"Chromium";v="110.0.5481.192", "Not A(Brand";v="24.0.0.0", "Microsoft Edge";v="110.0.1587.69"',
|
|
|
|
'sec-ch-ua-mobile': '?0',
|
|
|
|
'sec-ch-ua-model': '""',
|
|
|
|
'sec-ch-ua-platform': '"Windows"',
|
|
|
|
'sec-ch-ua-platform-version': '"15.0.0"',
|
|
|
|
'sec-fetch-dest': 'document',
|
|
|
|
'sec-fetch-mode': 'navigate',
|
|
|
|
'sec-fetch-site': 'none',
|
|
|
|
'sec-fetch-user': '?1',
|
|
|
|
'upgrade-insecure-requests': '1',
|
|
|
|
'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/110.0.0.0 Safari/537.36 Edg/110.0.1587.69',
|
|
|
|
'x-edge-shopping-flag': '1',
|
|
|
|
'x-forwarded-for': ip_address,
|
|
|
|
}
|
|
|
|
|
2023-08-27 19:58:36 +03:00
|
|
|
optionsSets = {
|
|
|
|
"optionsSets": [
|
|
|
|
'saharasugg',
|
|
|
|
'enablenewsfc',
|
|
|
|
'clgalileo',
|
|
|
|
'gencontentv3',
|
|
|
|
"nlu_direct_response_filter",
|
|
|
|
"deepleo",
|
|
|
|
"disable_emoji_spoken_text",
|
|
|
|
"responsible_ai_policy_235",
|
|
|
|
"enablemm",
|
|
|
|
"h3precise"
|
|
|
|
"dtappid",
|
|
|
|
"cricinfo",
|
|
|
|
"cricinfov2",
|
|
|
|
"dv3sugg",
|
|
|
|
"nojbfedge"
|
|
|
|
]
|
|
|
|
}
|
2023-08-21 23:39:57 +03:00
|
|
|
|
2023-08-27 19:58:36 +03:00
|
|
|
def format_message(msg: dict) -> str:
|
|
|
|
return json.dumps(msg, ensure_ascii=False) + Defaults.delimiter
|
2023-08-21 23:39:57 +03:00
|
|
|
|
|
|
|
def create_message(conversation: Conversation, prompt: str, context: str=None) -> str:
|
|
|
|
struct = {
|
|
|
|
'arguments': [
|
|
|
|
{
|
2023-08-27 19:58:36 +03:00
|
|
|
**Defaults.optionsSets,
|
2023-08-21 23:39:57 +03:00
|
|
|
'source': 'cib',
|
|
|
|
'allowedMessageTypes': Defaults.allowedMessageTypes,
|
|
|
|
'sliceIds': Defaults.sliceIds,
|
|
|
|
'traceId': os.urandom(16).hex(),
|
|
|
|
'isStartOfSession': True,
|
|
|
|
'message': Defaults.location | {
|
|
|
|
'author': 'user',
|
|
|
|
'inputMethod': 'Keyboard',
|
|
|
|
'text': prompt,
|
|
|
|
'messageType': 'Chat'
|
|
|
|
},
|
|
|
|
'conversationSignature': conversation.conversationSignature,
|
|
|
|
'participant': {
|
|
|
|
'id': conversation.clientId
|
|
|
|
},
|
|
|
|
'conversationId': conversation.conversationId
|
|
|
|
}
|
|
|
|
],
|
|
|
|
'invocationId': '0',
|
|
|
|
'target': 'chat',
|
|
|
|
'type': 4
|
|
|
|
}
|
|
|
|
|
|
|
|
if context:
|
|
|
|
struct['arguments'][0]['previousMessages'] = [{
|
|
|
|
"author": "user",
|
|
|
|
"description": context,
|
|
|
|
"contextType": "WebPage",
|
|
|
|
"messageType": "Context",
|
|
|
|
"messageId": "discover-web--page-ping-mriduna-----"
|
|
|
|
}]
|
|
|
|
return format_message(struct)
|
|
|
|
|
|
|
|
async def stream_generate(
|
|
|
|
prompt: str,
|
|
|
|
context: str=None,
|
|
|
|
cookies: dict=None
|
|
|
|
):
|
|
|
|
async with ClientSession(
|
|
|
|
timeout=aiohttp.ClientTimeout(total=900),
|
|
|
|
cookies=cookies,
|
|
|
|
headers=Defaults.headers,
|
|
|
|
) as session:
|
|
|
|
conversation = await create_conversation(session)
|
|
|
|
try:
|
|
|
|
async with session.ws_connect(
|
|
|
|
'wss://sydney.bing.com/sydney/ChatHub',
|
|
|
|
autoping=False,
|
|
|
|
) as wss:
|
|
|
|
|
|
|
|
await wss.send_str(format_message({'protocol': 'json', 'version': 1}))
|
|
|
|
msg = await wss.receive(timeout=900)
|
|
|
|
|
|
|
|
await wss.send_str(create_message(conversation, prompt, context))
|
|
|
|
|
|
|
|
response_txt = ''
|
|
|
|
result_text = ''
|
|
|
|
returned_text = ''
|
|
|
|
final = False
|
|
|
|
|
|
|
|
while not final:
|
|
|
|
msg = await wss.receive(timeout=900)
|
|
|
|
objects = msg.data.split(Defaults.delimiter)
|
|
|
|
for obj in objects:
|
|
|
|
if obj is None or not obj:
|
|
|
|
continue
|
|
|
|
|
|
|
|
response = json.loads(obj)
|
|
|
|
if response.get('type') == 1 and response['arguments'][0].get('messages'):
|
|
|
|
message = response['arguments'][0]['messages'][0]
|
|
|
|
if (message['contentOrigin'] != 'Apology'):
|
|
|
|
response_txt = result_text + \
|
|
|
|
message['adaptiveCards'][0]['body'][0].get('text', '')
|
|
|
|
|
|
|
|
if message.get('messageType'):
|
|
|
|
inline_txt = message['adaptiveCards'][0]['body'][0]['inlines'][0].get('text')
|
|
|
|
response_txt += inline_txt + '\n'
|
|
|
|
result_text += inline_txt + '\n'
|
|
|
|
|
|
|
|
if response_txt.startswith(returned_text):
|
|
|
|
new = response_txt[len(returned_text):]
|
|
|
|
if new != "\n":
|
|
|
|
yield new
|
|
|
|
returned_text = response_txt
|
|
|
|
elif response.get('type') == 2:
|
|
|
|
result = response['item']['result']
|
|
|
|
if result.get('error'):
|
|
|
|
raise Exception(f"{result['value']}: {result['message']}")
|
|
|
|
final = True
|
|
|
|
break
|
|
|
|
finally:
|
|
|
|
await delete_conversation(session, conversation)
|
2023-07-28 13:07:17 +03:00
|
|
|
|
2023-08-17 15:57:37 +03:00
|
|
|
def run(generator: AsyncGenerator[Union[Any, str], Any]):
|
2023-07-28 13:07:17 +03:00
|
|
|
loop = asyncio.get_event_loop()
|
|
|
|
gen = generator.__aiter__()
|
|
|
|
|
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
yield loop.run_until_complete(gen.__anext__())
|
|
|
|
|
|
|
|
except StopAsyncIteration:
|
|
|
|
break
|