Skip to content

Commit

Permalink
add: asyncio_core
Browse files Browse the repository at this point in the history
  • Loading branch information
dsdanielpark committed May 29, 2023
1 parent e55267d commit c8b4194
Showing 1 changed file with 179 additions and 0 deletions.
179 changes: 179 additions & 0 deletions translate_to/asyncio_core.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,179 @@
import os
import string
import random
import json
import re
import aiohttp
import asyncio
from deep_translator import GoogleTranslator
from bardapi.constants import ALLOWED_LANGUAGES, SESSION_HEADERS


class Bard:
"""
Bard class for interacting with the Bard API.
"""

def __init__(
self,
token: str = None,
timeout: int = 20,
proxies: dict = None,
session: aiohttp.ClientSession = None,
language: str = None,
):
"""
Initialize the Bard instance.
Args:
token (str): Bard API token.
timeout (int): Request timeout in seconds.
proxies (dict): Proxy configuration for requests.
session (aiohttp.ClientSession): aiohttp session object.
language (str): Language code for translation (e.g., "en", "ko", "ja").
"""
self.token = token or os.getenv("_BARD_API_KEY")
self.proxies = proxies
self.timeout = timeout
self._reqid = int("".join(random.choices(string.digits, k=4)))
self.conversation_id = ""
self.response_id = ""
self.choice_id = ""
if session is None:
self.session = aiohttp.ClientSession(headers=SESSION_HEADERS, cookies={"__Secure-1PSID": self.token})
else:
self.session = session
self.SNlM0e = self._get_snim0e()
self.language = language or os.getenv("_BARD_API_LANG")

async def _get_snim0e(self) -> str:
"""
Get the SNlM0e value from the Bard API response.
Returns:
str: SNlM0e value.
Raises:
Exception: If the __Secure-1PSID value is invalid or SNlM0e value is not found in the response.
"""
if not self.token or self.token[-1] != ".":
raise Exception(
"__Secure-1PSID value must end with a single dot. Enter correct __Secure-1PSID value."
)
resp = await self.session.get("https://bard.google.com/", timeout=self.timeout, proxies=self.proxies)
if resp.status != 200:
raise Exception(f"Response code not 200. Response Status is {resp.status}")
resp_text = await resp.text()
snim0e = re.search(r"SNlM0e\":\"(.*?)\"", resp_text)
if not snim0e:
raise Exception("SNlM0e value not found in response. Check __Secure-1PSID value.")
return snim0e.group(1)

def _extract_links(self, data: list) -> list:
"""
Extract links from the given data.
Args:
data: Data to extract links from.
Returns:
list: Extracted links.
"""
links = []
if isinstance(data, list):
for item in data:
if isinstance(item, list):
links.extend(self._extract_links(item))
elif (
isinstance(item, str)
and item.startswith("http")
and "favicon" not in item
):
links.append(item)
return links

async def get_answer(self, input_text: str) -> dict:
"""
Get an answer from the Bard API for the given input text.
Example:
>>> token = 'xxxxxxxxxx'
>>> bard = Bard(token=token)
>>> response = await bard.get_answer("나와 내 동년배들이 좋아하는 뉴진스에 대해서 알려줘")
>>> print(response['content'])
Args:
input_text (str): Input text for the query.
Returns:
dict: Answer from the Bard API in the following format:
{
"content": str,
"conversation_id": str,
"response_id": str,
"factualityQueries": list,
"textQuery": str,
"choices": list,
"links": list
"imgaes": set
}
"""
params = {
"bl": "boq_assistant-bard-web-server_20230419.00_p1",
"_reqid": str(self._reqid),
"rt": "c",
}
if self.language is not None and self.language not in ALLOWED_LANGUAGES:
translator_to_eng = GoogleTranslator(source="auto", target="en")
input_text = await translator_to_eng.translate(input_text)
input_text_struct = [
[input_text],
None,
[self.conversation_id, self.response_id, self.choice_id],
]
data = {
"f.req": json.dumps([None, json.dumps(input_text_struct)]),
"at": self.SNlM0e,
}
resp = await self.session.post(
"https://bard.google.com/_/BardChatUi/data/assistant.lamda.BardFrontendService/StreamGenerate",
params=params,
data=data,
timeout=self.timeout,
proxies=self.proxies,
)
resp_text = await resp.text()
resp_dict = json.loads(resp_text.splitlines()[3])[0][2]

if not resp_dict:
return {"content": f"Response Error: {resp_text}."}
resp_json = json.loads(resp_dict)
images = set()
if len(resp_json) >= 3:
if len(resp_json[4][0]) >= 4 and resp_json[4][0][4] is not None:
for img in resp_json[4][0][4]:
images.add(img[0][0][0])
parsed_answer = json.loads(resp_dict)
if self.language is not None and self.language not in ALLOWED_LANGUAGES:
translator_to_lang = GoogleTranslator(source="auto", target=self.language)
parsed_answer[0][0] = await translator_to_lang.translate(parsed_answer[0][0])
parsed_answer[4] = [
(x[0], await translator_to_lang.translate(x[1][0])) for x in parsed_answer[4]
]
bard_answer = {
"content": parsed_answer[0][0],
"conversation_id": parsed_answer[1][0],
"response_id": parsed_answer[1][1],
"factualityQueries": parsed_answer[3],
"textQuery": parsed_answer[2][0] if parsed_answer[2] else "",
"choices": [{"id": x[0], "content": x[1]} for x in parsed_answer[4]],
"links": self._extract_links(parsed_answer[4]),
"images": images,
}
self.conversation_id, self.response_id, self.choice_id = (
bard_answer["conversation_id"],
bard_answer["response_id"],
bard_answer["choices"][0]["id"],
)
self._reqid += 100000

return bard_answer

0 comments on commit c8b4194

Please sign in to comment.