2023-04-09 23:09:29 +00:00
from quora . api import Client as PoeClient
from quora . mail import Mail
from requests import Session
from re import search , findall
from json import loads
2023-04-11 16:44:29 +00:00
from time import sleep
2023-04-09 23:09:29 +00:00
from pathlib import Path
2023-04-11 16:44:29 +00:00
from random import choice , choices , randint
from string import ascii_letters , digits
2023-04-09 23:09:29 +00:00
from urllib import parse
2023-04-16 16:10:37 +00:00
from os import urandom
from hashlib import md5
from json import dumps
def extract_formkey ( html ) :
script_regex = r ' <script>if \ (.+ \ )throw new Error;(.+)</script> '
script_text = search ( script_regex , html ) . group ( 1 )
key_regex = r ' var .= " ([0-9a-f]+) " , '
key_text = search ( key_regex , script_text ) . group ( 1 )
cipher_regex = r ' . \ [( \ d+) \ ]=. \ [( \ d+) \ ] '
cipher_pairs = findall ( cipher_regex , script_text )
formkey_list = [ " " ] * len ( cipher_pairs )
for pair in cipher_pairs :
formkey_index , key_index = map ( int , pair )
formkey_list [ formkey_index ] = key_text [ key_index ]
formkey = " " . join ( formkey_list )
return formkey
2023-04-11 16:44:29 +00:00
2023-03-29 20:10:42 +00:00
class PoeResponse :
class Completion :
class Choices :
def __init__ ( self , choice : dict ) - > None :
self . text = choice [ ' text ' ]
self . content = self . text . encode ( )
self . index = choice [ ' index ' ]
self . logprobs = choice [ ' logprobs ' ]
self . finish_reason = choice [ ' finish_reason ' ]
def __repr__ ( self ) - > str :
return f ''' <__main__.APIResponse.Completion.Choices( \n text = { self . text . encode ( ) } , \n index = { self . index } , \n logprobs = { self . logprobs } , \n finish_reason = { self . finish_reason } )object at 0x1337> '''
def __init__ ( self , choices : dict ) - > None :
self . choices = [ self . Choices ( choice ) for choice in choices ]
class Usage :
def __init__ ( self , usage_dict : dict ) - > None :
self . prompt_tokens = usage_dict [ ' prompt_tokens ' ]
self . completion_tokens = usage_dict [ ' completion_tokens ' ]
self . total_tokens = usage_dict [ ' total_tokens ' ]
def __repr__ ( self ) :
return f ''' <__main__.APIResponse.Usage( \n prompt_tokens = { self . prompt_tokens } , \n completion_tokens = { self . completion_tokens } , \n total_tokens = { self . total_tokens } )object at 0x1337> '''
def __init__ ( self , response_dict : dict ) - > None :
self . response_dict = response_dict
self . id = response_dict [ ' id ' ]
self . object = response_dict [ ' object ' ]
self . created = response_dict [ ' created ' ]
self . model = response_dict [ ' model ' ]
self . completion = self . Completion ( response_dict [ ' choices ' ] )
self . usage = self . Usage ( response_dict [ ' usage ' ] )
def json ( self ) - > dict :
return self . response_dict
2023-04-11 16:44:29 +00:00
class ModelResponse :
def __init__ ( self , json_response : dict ) - > None :
self . id = json_response [ ' data ' ] [ ' poeBotCreate ' ] [ ' bot ' ] [ ' id ' ]
self . name = json_response [ ' data ' ] [ ' poeBotCreate ' ] [ ' bot ' ] [ ' displayName ' ]
self . limit = json_response [ ' data ' ] [ ' poeBotCreate ' ] [ ' bot ' ] [ ' messageLimit ' ] [ ' dailyLimit ' ]
self . deleted = json_response [ ' data ' ] [ ' poeBotCreate ' ] [ ' bot ' ] [ ' deletionState ' ]
class Model :
def create (
token : str ,
model : str = ' gpt-3.5-turbo ' , # claude-instant
system_prompt : str = ' You are ChatGPT a large language model developed by Openai. Answer as consisely as possible ' ,
description : str = ' gpt-3.5 language model from openai, skidded by poe.com ' ,
handle : str = None ) - > ModelResponse :
models = {
' gpt-3.5-turbo ' : ' chinchilla ' ,
2023-04-13 15:08:04 +00:00
' claude-instant-v1.0 ' : ' a2 ' ,
' gpt-4 ' : ' beaver '
2023-04-11 16:44:29 +00:00
}
if not handle :
handle = f ' gptx { randint ( 1111111 , 9999999 ) } '
client = Session ( )
client . cookies [ ' p-b ' ] = token
2023-04-16 16:10:37 +00:00
formkey = extract_formkey ( client . get ( ' https://poe.com ' ) . text )
2023-04-11 16:44:29 +00:00
settings = client . get ( ' https://poe.com/api/settings ' ) . json ( )
client . headers = {
" host " : " poe.com " ,
" origin " : " https://poe.com " ,
" referer " : " https://poe.com/ " ,
" content-type " : " application/json " ,
2023-04-16 16:10:37 +00:00
" poe-formkey " : formkey ,
2023-04-11 16:44:29 +00:00
" poe-tchannel " : settings [ ' tchannelData ' ] [ ' channel ' ] ,
" user-agent " : " Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/112.0.0.0 Safari/537.36 " ,
" connection " : " keep-alive " ,
" sec-ch-ua " : " \" Chromium \" ;v= \" 112 \" , \" Google Chrome \" ;v= \" 112 \" , \" Not:A-Brand \" ;v= \" 99 \" " ,
" sec-ch-ua-mobile " : " ?0 " ,
" sec-ch-ua-platform " : " \" macOS \" " ,
2023-04-15 14:42:07 +00:00
" content-type " : " application/json " ,
2023-04-11 16:44:29 +00:00
" sec-fetch-site " : " same-origin " ,
" sec-fetch-mode " : " cors " ,
" sec-fetch-dest " : " empty " ,
" accept " : " */* " ,
" accept-encoding " : " gzip, deflate, br " ,
" accept-language " : " en-GB,en-US;q=0.9,en;q=0.8 " ,
}
2023-04-16 16:10:37 +00:00
payload = dumps ( separators = ( ' , ' , ' : ' ) , obj = {
2023-04-11 16:44:29 +00:00
' queryName ' : ' CreateBotMain_poeBotCreate_Mutation ' ,
' variables ' : {
' model ' : models [ model ] ,
' handle ' : handle ,
' prompt ' : system_prompt ,
' isPromptPublic ' : True ,
' introduction ' : ' ' ,
' description ' : description ,
' profilePictureUrl ' : ' https://qph.fs.quoracdn.net/main-qimg-24e0b480dcd946e1cc6728802c5128b6 ' ,
' apiUrl ' : None ,
' apiKey ' : ' ' . join ( choices ( ascii_letters + digits , k = 32 ) ) ,
' isApiBot ' : False ,
' hasLinkification ' : False ,
' hasMarkdownRendering ' : False ,
' hasSuggestedReplies ' : False ,
' isPrivateBot ' : False
} ,
' query ' : ' mutation CreateBotMain_poeBotCreate_Mutation( \n $model: String! \n $handle: String! \n $prompt: String! \n $isPromptPublic: Boolean! \n $introduction: String! \n $description: String! \n $profilePictureUrl: String \n $apiUrl: String \n $apiKey: String \n $isApiBot: Boolean \n $hasLinkification: Boolean \n $hasMarkdownRendering: Boolean \n $hasSuggestedReplies: Boolean \n $isPrivateBot: Boolean \n ) { \n poeBotCreate(model: $model, handle: $handle, promptPlaintext: $prompt, isPromptPublic: $isPromptPublic, introduction: $introduction, description: $description, profilePicture: $profilePictureUrl, apiUrl: $apiUrl, apiKey: $apiKey, isApiBot: $isApiBot, hasLinkification: $hasLinkification, hasMarkdownRendering: $hasMarkdownRendering, hasSuggestedReplies: $hasSuggestedReplies, isPrivateBot: $isPrivateBot) { \n status \n bot { \n id \n ...BotHeader_bot \n } \n } \n } \n \n fragment BotHeader_bot on Bot { \n displayName \n messageLimit { \n dailyLimit \n } \n ...BotImage_bot \n ...BotLink_bot \n ...IdAnnotation_node \n ...botHelpers_useViewerCanAccessPrivateBot \n ...botHelpers_useDeletion_bot \n } \n \n fragment BotImage_bot on Bot { \n displayName \n ...botHelpers_useDeletion_bot \n ...BotImage_useProfileImage_bot \n } \n \n fragment BotImage_useProfileImage_bot on Bot { \n image { \n __typename \n ... on LocalBotImage { \n localName \n } \n ... on UrlBotImage { \n url \n } \n } \n ...botHelpers_useDeletion_bot \n } \n \n fragment BotLink_bot on Bot { \n displayName \n } \n \n fragment IdAnnotation_node on Node { \n __isNode: __typename \n id \n } \n \n fragment botHelpers_useDeletion_bot on Bot { \n deletionState \n } \n \n fragment botHelpers_useViewerCanAccessPrivateBot on Bot { \n isPrivateBot \n viewerIsCreator \n } \n ' ,
} )
2023-04-16 16:10:37 +00:00
base_string = payload + client . headers [ " poe-formkey " ] + ' WpuLMiXEKKE98j56k '
client . headers [ " poe-tag-id " ] = md5 ( base_string . encode ( ) ) . hexdigest ( )
response = client . post ( " https://poe.com/api/gql_POST " , data = payload )
2023-04-11 16:44:29 +00:00
if not ' success ' in response . text :
raise Exception ( '''
Bot creation Failed
! ! Important ! !
Bot creation was not enabled on this account
please use : quora . Account . create with enable_bot_creation set to True
''' )
return ModelResponse ( response . json ( ) )
2023-03-29 20:10:42 +00:00
class Account :
2023-04-11 16:44:29 +00:00
def create ( proxy : None or str = None , logging : bool = False , enable_bot_creation : bool = False ) :
2023-04-06 19:24:04 +00:00
client = Session ( )
2023-03-29 20:10:42 +00:00
client . proxies = {
' http ' : f ' http:// { proxy } ' ,
' https ' : f ' http:// { proxy } ' } if proxy else None
2023-04-16 16:10:37 +00:00
2023-03-29 20:10:42 +00:00
mail = Mail ( client . proxies )
mail_token = None
2023-04-16 16:10:37 +00:00
_ , mail_address = mail . get_mail ( )
2023-04-16 23:07:26 +00:00
if mail_address is None :
raise Exception ( ' Error creating mail, please use proxies ' )
2023-03-29 20:10:42 +00:00
if logging : print ( ' email ' , mail_address )
client . headers = {
" host " : " poe.com " ,
" connection " : " keep-alive " ,
" cache-control " : " max-age=0 " ,
" sec-ch-ua " : " \" Microsoft Edge \" ;v= \" 111 \" , \" Not(A:Brand \" ;v= \" 8 \" , \" Chromium \" ;v= \" 111 \" " ,
" sec-ch-ua-mobile " : " ?0 " ,
" sec-ch-ua-platform " : " \" macOS \" " ,
" user-agent " : " Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/111.0.0.0 Safari/537.36 Edg/111.0.1661.54 " ,
" accept " : " text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7 " ,
" sec-fetch-site " : " same-origin " ,
" sec-fetch-mode " : " navigate " ,
2023-04-15 14:42:07 +00:00
" content-type " : " application/json " ,
2023-03-29 20:10:42 +00:00
" sec-fetch-user " : " ?1 " ,
" sec-fetch-dest " : " document " ,
" accept-encoding " : " gzip, deflate, br " ,
" accept-language " : " en-GB,en;q=0.9,en-US;q=0.8 " ,
" upgrade-insecure-requests " : " 1 " ,
}
2023-04-16 16:10:37 +00:00
client . headers [ " poe-formkey " ] = extract_formkey ( client . get ( ' https://poe.com/login ' ) . text )
client . headers [ " poe-tchannel " ] = client . get ( ' https://poe.com/api/settings ' ) . json ( ) [ ' tchannelData ' ] [ ' channel ' ]
2023-03-29 20:10:42 +00:00
2023-04-15 14:42:07 +00:00
payload = dumps ( separators = ( ' , ' , ' : ' ) , obj = {
' queryName ' : ' MainSignupLoginSection_sendVerificationCodeMutation_Mutation ' ,
' variables ' : {
' emailAddress ' : mail_address ,
' phoneNumber ' : None ,
' recaptchaToken ' : None ,
2023-03-29 20:10:42 +00:00
} ,
2023-04-15 14:42:07 +00:00
' query ' : ' mutation MainSignupLoginSection_sendVerificationCodeMutation_Mutation( \n $emailAddress: String \n $phoneNumber: String \n $recaptchaToken: String \n ) { \n sendVerificationCode(verificationReason: login, emailAddress: $emailAddress, phoneNumber: $phoneNumber, recaptchaToken: $recaptchaToken) { \n status \n errorMessage \n } \n } \n ' ,
} )
base_string = payload + client . headers [ " poe-formkey " ] + ' WpuLMiXEKKE98j56k '
client . headers [ " poe-tag-id " ] = md5 ( base_string . encode ( ) ) . hexdigest ( )
2023-03-29 20:10:42 +00:00
2023-04-15 14:42:07 +00:00
response = client . post ( ' https://poe.com/api/gql_POST ' , data = payload )
2023-03-29 20:10:42 +00:00
if ' Bad Request ' in response . text :
if logging : print ( ' bad request, retrying... ' , response . json ( ) )
2023-04-16 16:10:37 +00:00
quit ( )
2023-03-29 20:10:42 +00:00
if logging : print ( ' send_code ' , response . json ( ) )
while True :
sleep ( 1 )
2023-04-16 16:10:37 +00:00
messages = mail . fetch_inbox ( )
if len ( messages [ " messages " ] ) > 0 :
email_content = mail . get_message_content ( messages [ " messages " ] [ 0 ] [ " _id " ] )
mail_token = findall ( r ' ; " >( \ d { 6,7})</div> ' , email_content ) [ 0 ]
2023-03-29 20:10:42 +00:00
if mail_token :
break
if logging : print ( ' code ' , mail_token )
2023-04-15 14:42:07 +00:00
payload = dumps ( separators = ( ' , ' , ' : ' ) , obj = {
2023-03-29 20:10:42 +00:00
" queryName " : " SignupOrLoginWithCodeSection_signupWithVerificationCodeMutation_Mutation " ,
" variables " : {
" verificationCode " : mail_token ,
" emailAddress " : mail_address ,
" phoneNumber " : None
} ,
" query " : " mutation SignupOrLoginWithCodeSection_signupWithVerificationCodeMutation_Mutation( \n $verificationCode: String! \n $emailAddress: String \n $phoneNumber: String \n ) { \n signupWithVerificationCode(verificationCode: $verificationCode, emailAddress: $emailAddress, phoneNumber: $phoneNumber) { \n status \n errorMessage \n } \n } \n "
2023-04-15 14:42:07 +00:00
} )
2023-04-16 16:10:37 +00:00
2023-04-15 14:42:07 +00:00
base_string = payload + client . headers [ " poe-formkey " ] + ' WpuLMiXEKKE98j56k '
client . headers [ " poe-tag-id " ] = md5 ( base_string . encode ( ) ) . hexdigest ( )
2023-03-29 20:10:42 +00:00
2023-04-15 14:42:07 +00:00
response = client . post ( ' https://poe.com/api/gql_POST ' , data = payload )
2023-03-29 20:10:42 +00:00
if logging : print ( ' verify_code ' , response . json ( ) )
token = parse . unquote ( client . cookies . get_dict ( ) [ ' p-b ' ] )
2023-04-16 16:10:37 +00:00
2023-03-29 20:10:42 +00:00
with open ( Path ( __file__ ) . resolve ( ) . parent / ' cookies.txt ' , ' a ' ) as f :
f . write ( f ' { token } \n ' )
2023-04-11 16:44:29 +00:00
if enable_bot_creation :
2023-04-16 16:10:37 +00:00
payload = dumps ( separators = ( ' , ' , ' : ' ) , obj = {
2023-04-11 16:44:29 +00:00
" queryName " : " UserProfileConfigurePreviewModal_markMultiplayerNuxCompleted_Mutation " ,
" variables " : { } ,
" query " : " mutation UserProfileConfigurePreviewModal_markMultiplayerNuxCompleted_Mutation { \n markMultiplayerNuxCompleted { \n viewer { \n hasCompletedMultiplayerNux \n id \n } \n } \n } \n "
2023-04-16 16:10:37 +00:00
} )
2023-04-15 14:42:07 +00:00
2023-04-16 16:10:37 +00:00
base_string = payload + client . headers [ " poe-formkey " ] + ' WpuLMiXEKKE98j56k '
2023-04-15 14:42:07 +00:00
client . headers [ " poe-tag-id " ] = md5 ( base_string . encode ( ) ) . hexdigest ( )
2023-04-16 16:10:37 +00:00
resp = client . post ( " https://poe.com/api/gql_POST " , data = payload )
if logging : print ( resp . json ( ) )
2023-03-29 20:10:42 +00:00
return token
def get ( ) :
cookies = open ( Path ( __file__ ) . resolve ( ) . parent / ' cookies.txt ' , ' r ' ) . read ( ) . splitlines ( )
return choice ( cookies )
class StreamingCompletion :
def create (
model : str = ' gpt-4 ' ,
2023-04-13 15:08:04 +00:00
custom_model : bool = None ,
2023-03-29 20:10:42 +00:00
prompt : str = ' hello world ' ,
token : str = ' ' ) :
models = {
' sage ' : ' capybara ' ,
' gpt-4 ' : ' beaver ' ,
2023-04-11 16:44:29 +00:00
' claude-v1.2 ' : ' a2_2 ' ,
' claude-instant-v1.0 ' : ' a2 ' ,
' gpt-3.5-turbo ' : ' chinchilla '
2023-03-29 20:10:42 +00:00
}
2023-04-11 16:44:29 +00:00
_model = models [ model ] if not custom_model else custom_model
2023-03-29 20:10:42 +00:00
client = PoeClient ( token )
2023-04-13 15:08:04 +00:00
for chunk in client . send_message ( _model , prompt ) :
2023-03-29 20:10:42 +00:00
yield PoeResponse ( {
' id ' : chunk [ " messageId " ] ,
' object ' : ' text_completion ' ,
' created ' : chunk [ ' creationTime ' ] ,
2023-04-11 16:44:29 +00:00
' model ' : _model ,
2023-03-29 20:10:42 +00:00
' choices ' : [ {
' text ' : chunk [ " text_new " ] ,
' index ' : 0 ,
' logprobs ' : None ,
' finish_reason ' : ' stop '
} ] ,
' usage ' : {
' prompt_tokens ' : len ( prompt ) ,
' completion_tokens ' : len ( chunk [ " text_new " ] ) ,
' total_tokens ' : len ( prompt ) + len ( chunk [ " text_new " ] )
}
} )
class Completion :
def create (
model : str = ' gpt-4 ' ,
2023-04-11 16:44:29 +00:00
custom_model : str = None ,
2023-03-29 20:10:42 +00:00
prompt : str = ' hello world ' ,
token : str = ' ' ) :
models = {
' sage ' : ' capybara ' ,
' gpt-4 ' : ' beaver ' ,
2023-04-11 16:44:29 +00:00
' claude-v1.2 ' : ' a2_2 ' ,
' claude-instant-v1.0 ' : ' a2 ' ,
' gpt-3.5-turbo ' : ' chinchilla '
2023-03-29 20:10:42 +00:00
}
2023-04-11 16:44:29 +00:00
_model = models [ model ] if not custom_model else custom_model
2023-03-29 20:10:42 +00:00
client = PoeClient ( token )
2023-04-13 15:08:04 +00:00
for chunk in client . send_message ( _model , prompt ) :
2023-03-29 20:10:42 +00:00
pass
return PoeResponse ( {
' id ' : chunk [ " messageId " ] ,
' object ' : ' text_completion ' ,
' created ' : chunk [ ' creationTime ' ] ,
2023-04-11 16:44:29 +00:00
' model ' : _model ,
2023-03-29 20:10:42 +00:00
' choices ' : [ {
' text ' : chunk [ " text " ] ,
' index ' : 0 ,
' logprobs ' : None ,
' finish_reason ' : ' stop '
} ] ,
' usage ' : {
' prompt_tokens ' : len ( prompt ) ,
' completion_tokens ' : len ( chunk [ " text " ] ) ,
' total_tokens ' : len ( prompt ) + len ( chunk [ " text " ] )
}
2023-04-15 14:42:07 +00:00
} )