gpt4free/g4f/Provider/Bing.py

325 lines
11 KiB
Python
Raw Normal View History

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
import browser_cookie3
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-07-28 13:07:17 +03:00
from .base_provider import BaseProvider
class Bing(BaseProvider):
url = "https://bing.com/chat"
2023-08-23 03:16:35 +03:00
working = True
2023-07-28 13:07:17 +03:00
supports_gpt_4 = True
2023-08-23 03:16:35 +03:00
@classmethod
2023-07-28 13:07:17 +03:00
def create_completion(
2023-08-23 03:16:35 +03:00
cls,
2023-07-28 13:07:17 +03:00
model: str,
messages: list[dict[str, str]],
stream: bool,
2023-08-21 23:39:57 +03:00
**kwargs: Any
2023-07-28 13:07:17 +03:00
) -> CreateResult:
2023-08-23 03:16:35 +03:00
if stream:
yield from run(cls.create_async_generator(model, messages, **kwargs))
else:
yield asyncio.run(cls.create_async(model, messages, **kwargs))
2023-08-21 23:39:57 +03:00
2023-08-23 03:16:35 +03:00
@classmethod
async def create_async(
cls,
model: str,
2023-08-21 23:39:57 +03:00
messages: list[dict[str, str]],
2023-08-23 03:16:35 +03:00
**kwargs: Any,
) -> str:
result = []
async for chunk in cls.create_async_generator(model, messages, **kwargs):
result.append(chunk)
if result:
return "".join(result)
@staticmethod
def create_async_generator(
model: str,
messages: list[dict[str, str]],
cookies: dict = {}
) -> AsyncGenerator:
if len(messages) < 2:
prompt = messages[0]["content"]
context = None
2023-07-28 13:07:17 +03:00
2023-08-23 03:16:35 +03:00
else:
prompt = messages[-1]["content"]
context = convert(messages[:-1])
2023-07-28 13:07:17 +03:00
2023-08-23 03:16:35 +03:00
if not cookies:
for cookie in browser_cookie3.load(domain_name='.bing.com'):
cookies[cookie.name] = cookie.value
2023-07-28 13:07:17 +03:00
2023-08-23 03:16:35 +03:00
return stream_generate(prompt, context, cookies)
2023-07-28 13:07:17 +03:00
def convert(messages: list[dict[str, str]]):
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,
}
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"
]
}
def format_message(msg: dict) -> str:
return json.dumps(msg, ensure_ascii=False) + Defaults.delimiter
def create_message(conversation: Conversation, prompt: str, context: str=None) -> str:
struct = {
'arguments': [
{
**Defaults.optionsSets,
'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 returned_text.endswith(' '):
final = True
break
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