-
Notifications
You must be signed in to change notification settings - Fork 384
/
Copy pathmain.py
207 lines (167 loc) · 6.99 KB
/
main.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
import asyncio
import ctypes
import os
import random
import sys
import traceback
import aiohttp
from art import text2art
from imap_tools import MailboxLoginError
from termcolor import colored, cprint
from better_proxy import Proxy
from core import Grass
from core.autoreger import AutoReger
from core.utils import logger, file_to_list
from core.utils.accounts_db import AccountsDB
from core.utils.exception import EmailApproveLinkNotFoundException, LoginException, RegistrationException
from core.utils.generate.person import Person
from data.config import ACCOUNTS_FILE_PATH, PROXIES_FILE_PATH, REGISTER_ACCOUNT_ONLY, THREADS, REGISTER_DELAY, \
CLAIM_REWARDS_ONLY, APPROVE_EMAIL, APPROVE_WALLET_ON_EMAIL, MINING_MODE, CONNECT_WALLET, \
WALLETS_FILE_PATH, SEND_WALLET_APPROVE_LINK_TO_EMAIL, SINGLE_IMAP_ACCOUNT, SEMI_AUTOMATIC_APPROVE_LINK, \
PROXY_DB_PATH, USE_CONSOLE_VERSION
def bot_info(name: str = ""):
cprint(text2art(name), 'green')
if sys.platform == 'win32':
ctypes.windll.kernel32.SetConsoleTitleW(f"{name}")
print(
f"{colored('EnJoYeR <crypto/> moves:', color='light_yellow')} "
f"{colored('https://t.me/+tdC-PXRzhnczNDli', color='light_green')}"
)
async def worker_task(_id, account: str, proxy: str = None, wallet: str = None, db: AccountsDB = None):
consumables = account.split(":")[:3]
imap_pass = None
if SINGLE_IMAP_ACCOUNT:
consumables.append(SINGLE_IMAP_ACCOUNT.split(":")[1])
if len(consumables) == 1:
email = consumables[0]
password = Person().random_string(8)
elif len(consumables) == 2:
email, password = consumables
else:
email, password, imap_pass = consumables
grass = None
try:
grass = Grass(_id, email, password, proxy, db)
if MINING_MODE:
await asyncio.sleep(random.uniform(1, 2) * _id)
logger.info(f"Starting №{_id} | {email} | {password} | {proxy}")
else:
await asyncio.sleep(random.uniform(*REGISTER_DELAY))
logger.info(f"Starting №{_id} | {email} | {password} | {proxy}")
if REGISTER_ACCOUNT_ONLY:
await grass.create_account()
elif APPROVE_EMAIL or CONNECT_WALLET or SEND_WALLET_APPROVE_LINK_TO_EMAIL or APPROVE_WALLET_ON_EMAIL:
await grass.enter_account()
user_info = await grass.retrieve_user()
if APPROVE_EMAIL:
if user_info['result']['data'].get("isVerified"):
logger.info(f"{grass.id} | {grass.email} email already verified!")
else:
if SEMI_AUTOMATIC_APPROVE_LINK:
imap_pass = "placeholder"
elif imap_pass is None:
raise TypeError("IMAP password is not provided")
await grass.confirm_email(imap_pass)
if CONNECT_WALLET:
if user_info['result']['data'].get("walletAddress"):
logger.info(f"{grass.id} | {grass.email} wallet already linked!")
else:
await grass.link_wallet(wallet)
if user_info['result']['data'].get("isWalletAddressVerified"):
logger.info(f"{grass.id} | {grass.email} wallet already verified!")
else:
if SEND_WALLET_APPROVE_LINK_TO_EMAIL:
await grass.send_approve_link(endpoint="sendWalletAddressEmailVerification")
if APPROVE_WALLET_ON_EMAIL:
if SEMI_AUTOMATIC_APPROVE_LINK:
imap_pass = "placeholder"
elif imap_pass is None:
raise TypeError("IMAP password is not provided")
await grass.confirm_wallet_by_email(imap_pass)
elif CLAIM_REWARDS_ONLY:
await grass.claim_rewards()
else:
await grass.start()
return True
except (LoginException, RegistrationException) as e:
logger.warning(f"{_id} | {e}")
except MailboxLoginError as e:
logger.error(f"{_id} | {e}")
# except NoProxiesException as e:
# logger.warning(e)
except EmailApproveLinkNotFoundException as e:
logger.warning(e)
except aiohttp.ClientError as e:
logger.warning(f"{_id} | Some connection error: {e}...")
except Exception as e:
logger.error(f"{_id} | not handled exception | error: {e} {traceback.format_exc()}")
finally:
if grass:
await grass.session.close()
# await grass.ws_session.close()
async def main():
accounts = file_to_list(ACCOUNTS_FILE_PATH)
if not accounts:
logger.warning("No accounts found!")
return
proxies = [Proxy.from_str(proxy).as_url for proxy in file_to_list(PROXIES_FILE_PATH)]
#### delete DB if it exists to clean up
try:
if os.path.exists(PROXY_DB_PATH):
os.remove(PROXY_DB_PATH)
except PermissionError:
logger.warning(f"Cannot remove {PROXY_DB_PATH}, file is in use")
db = AccountsDB(PROXY_DB_PATH)
await db.connect()
for i, account in enumerate(accounts):
account = account.split(":")[0]
proxy = proxies[i] if len(proxies) > i else None
if await db.proxies_exist(proxy) or not proxy:
continue
await db.add_account(account, proxy)
await db.delete_all_from_extra_proxies()
await db.push_extra_proxies(proxies[len(accounts):])
autoreger = AutoReger.get_accounts(
(ACCOUNTS_FILE_PATH, PROXIES_FILE_PATH, WALLETS_FILE_PATH),
with_id=True,
static_extra=(db,)
)
threads = THREADS
if REGISTER_ACCOUNT_ONLY:
msg = "__REGISTER__ MODE"
elif APPROVE_EMAIL or CONNECT_WALLET or SEND_WALLET_APPROVE_LINK_TO_EMAIL or APPROVE_WALLET_ON_EMAIL:
if CONNECT_WALLET:
wallets = file_to_list(WALLETS_FILE_PATH)
if len(wallets) == 0:
logger.error("Wallet file is empty")
return
elif len(wallets) != len(accounts):
logger.error("Wallets count != accounts count")
return
elif len(accounts[0].split(":")) != 3:
logger.error(
"For __APPROVE__ mode: Need to provide email, password and imap password - email:password:imap_password")
return
msg = "__APPROVE__ MODE"
elif CLAIM_REWARDS_ONLY:
msg = "__CLAIM__ MODE"
else:
msg = "__MINING__ MODE"
threads = len(autoreger.accounts)
logger.info(msg)
await autoreger.start(worker_task, threads)
await db.close_connection()
if __name__ == "__main__":
if sys.platform == 'win32':
asyncio.set_event_loop_policy(asyncio.WindowsSelectorEventLoopPolicy())
if not USE_CONSOLE_VERSION:
import interface
interface.start_ui()
else:
bot_info("GRASS_AUTO")
loop = asyncio.ProactorEventLoop()
asyncio.set_event_loop(loop)
loop.run_until_complete(main())
else:
bot_info("GRASS_AUTO")
asyncio.run(main())