2020-10-03 02:45:45 +00:00
|
|
|
# SPDX-License-Identifier: BSD-3-Clause
|
2022-03-14 17:00:16 +00:00
|
|
|
# Copyright (c) 2022, The Monero Project.
|
|
|
|
# Copyright (c) 2022, dsc@xmr.pm
|
2020-10-03 02:45:45 +00:00
|
|
|
|
2021-04-06 23:24:46 +00:00
|
|
|
import re
|
2020-10-03 02:45:45 +00:00
|
|
|
import json
|
2020-12-22 18:03:48 +00:00
|
|
|
import asyncio
|
2020-10-03 02:45:45 +00:00
|
|
|
import os
|
|
|
|
import random
|
|
|
|
from datetime import datetime
|
2020-12-22 18:03:48 +00:00
|
|
|
from collections import Counter
|
|
|
|
from functools import wraps
|
|
|
|
from typing import List, Union
|
2021-04-05 19:03:35 +00:00
|
|
|
from io import BytesIO
|
2020-10-03 02:45:45 +00:00
|
|
|
|
2020-12-22 18:03:48 +00:00
|
|
|
import psutil
|
2020-10-03 02:45:45 +00:00
|
|
|
import aiohttp
|
2020-12-22 18:03:48 +00:00
|
|
|
from aiohttp_socks import ProxyConnector
|
2021-04-05 19:03:35 +00:00
|
|
|
from PIL import Image
|
2020-10-03 02:45:45 +00:00
|
|
|
|
2020-10-14 01:34:41 +00:00
|
|
|
import settings
|
|
|
|
|
2020-10-03 02:45:45 +00:00
|
|
|
|
2021-04-06 23:24:46 +00:00
|
|
|
RE_ADDRESS = r"^[a-zA-Z0-9]{97}$"
|
|
|
|
|
|
|
|
|
2020-12-22 18:03:48 +00:00
|
|
|
def print_banner():
|
|
|
|
print(f"""\033[91m
|
2021-04-05 17:53:11 +00:00
|
|
|
_______________ |*\_/*|________
|
|
|
|
| ___________ | .-. .-. ||_/-\_|______ |
|
|
|
|
| | | | .****. .****. | | | |
|
|
|
|
| | 0 0 | | .*****.*****. | | 0 0 | |
|
|
|
|
| | - | | .*********. | | - | |
|
|
|
|
| | \___/ | | .*******. | | \___/ | |
|
|
|
|
| |___ ___| | .*****. | |___________| |
|
|
|
|
|_____|\_/|_____| .***. |_______________|
|
|
|
|
_|__|/ \|_|_.............*.............._|________|_
|
|
|
|
/ ********** \ / ********** \\
|
|
|
|
/ ************ \ / ************ \\
|
|
|
|
-------------------- {settings.COIN_SYMBOL} --------------------\033[0m
|
2020-12-22 18:03:48 +00:00
|
|
|
""".strip())
|
2020-10-03 02:45:45 +00:00
|
|
|
|
|
|
|
|
2021-05-02 23:12:36 +00:00
|
|
|
async def httpget(url: str, json=True, timeout: int = 5, socks5: str = None, raise_for_status=True, verify_tls=True):
|
2020-10-03 02:45:45 +00:00
|
|
|
headers = {"User-Agent": random_agent()}
|
2020-12-22 18:03:48 +00:00
|
|
|
opts = {"timeout": aiohttp.ClientTimeout(total=timeout)}
|
|
|
|
if socks5:
|
|
|
|
opts['connector'] = ProxyConnector.from_url(socks5)
|
|
|
|
|
|
|
|
async with aiohttp.ClientSession(**opts) as session:
|
2021-05-02 23:12:36 +00:00
|
|
|
async with session.get(url, headers=headers, ssl=verify_tls) as response:
|
2020-12-22 18:03:48 +00:00
|
|
|
if raise_for_status:
|
|
|
|
response.raise_for_status()
|
|
|
|
|
|
|
|
result = await response.json() if json else await response.text()
|
|
|
|
if result is None or (isinstance(result, str) and result == ''):
|
|
|
|
raise Exception("empty response from request")
|
|
|
|
return result
|
2020-10-03 02:45:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
def random_agent():
|
2021-04-05 17:49:02 +00:00
|
|
|
from wowlet_backend.factory import user_agents
|
2020-10-03 02:45:45 +00:00
|
|
|
return random.choice(user_agents)
|
|
|
|
|
|
|
|
|
2022-03-14 17:00:16 +00:00
|
|
|
async def wowlet_data():
|
|
|
|
"""A collection of data collected by the various wowlet tasks"""
|
2021-04-05 17:49:02 +00:00
|
|
|
from wowlet_backend.factory import cache, now
|
2020-12-22 18:03:48 +00:00
|
|
|
data = await cache.get("data")
|
|
|
|
if data:
|
2020-10-03 02:45:45 +00:00
|
|
|
data = json.loads(data)
|
2020-12-22 18:03:48 +00:00
|
|
|
return data
|
2020-10-03 02:45:45 +00:00
|
|
|
|
2022-03-14 17:00:16 +00:00
|
|
|
keys = [
|
|
|
|
"blockheights",
|
|
|
|
"funding_proposals",
|
|
|
|
"crypto_rates",
|
|
|
|
"fiat_rates",
|
|
|
|
"reddit",
|
|
|
|
"rpc_nodes",
|
|
|
|
"xmrig",
|
|
|
|
"xmrto_rates",
|
|
|
|
"suchwow",
|
|
|
|
"forum",
|
|
|
|
"wowlet_releases",
|
2022-03-15 12:56:07 +00:00
|
|
|
"yellwow",
|
|
|
|
"wownerod_releases"
|
2022-03-14 17:00:16 +00:00
|
|
|
]
|
2020-12-22 18:03:48 +00:00
|
|
|
data = {keys[i]: json.loads(val) if val else None for i, val in enumerate(await cache.mget(*keys))}
|
2020-10-03 02:45:45 +00:00
|
|
|
|
2020-12-22 18:03:48 +00:00
|
|
|
# start caching when application lifetime is more than 20 seconds
|
|
|
|
if (datetime.now() - now).total_seconds() > 20:
|
|
|
|
await cache.setex("data", 30, json.dumps(data))
|
|
|
|
return data
|
2020-10-03 02:45:45 +00:00
|
|
|
|
|
|
|
|
2020-12-22 18:03:48 +00:00
|
|
|
def popularity_contest(lst: List[int]) -> Union[int, None]:
|
|
|
|
"""Return most common occurrences of List[int]. If
|
|
|
|
there are no duplicates, return max() instead.
|
|
|
|
"""
|
|
|
|
if not lst:
|
|
|
|
return
|
|
|
|
if len(set(lst)) == len(lst):
|
|
|
|
return max(lst)
|
|
|
|
return Counter(lst).most_common(1)[0][0]
|
2020-10-03 02:45:45 +00:00
|
|
|
|
|
|
|
|
2021-04-05 19:03:35 +00:00
|
|
|
async def image_resize(buffer: bytes, max_bounding_box: int = 512, quality: int = 70) -> bytes:
|
|
|
|
"""
|
|
|
|
- Resize if the image is too large
|
|
|
|
- PNG -> JPEG
|
|
|
|
- Removes EXIF
|
|
|
|
"""
|
|
|
|
buffer = BytesIO(buffer)
|
|
|
|
buffer.seek(0)
|
|
|
|
image = Image.open(buffer)
|
|
|
|
image = image.convert('RGB')
|
|
|
|
|
|
|
|
if max([image.height, image.width]) > max_bounding_box:
|
|
|
|
image.thumbnail((max_bounding_box, max_bounding_box), Image.BICUBIC)
|
|
|
|
|
|
|
|
data = list(image.getdata())
|
|
|
|
image_without_exif = Image.new(image.mode, image.size)
|
|
|
|
image_without_exif.putdata(data)
|
|
|
|
|
|
|
|
buffer = BytesIO()
|
|
|
|
image_without_exif.save(buffer, "JPEG", quality=quality)
|
|
|
|
buffer.seek(0)
|
|
|
|
|
|
|
|
return buffer.read()
|
2022-03-14 09:41:02 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def whaleornot(amount: float):
|
|
|
|
if amount <= 0:
|
|
|
|
fish_str = "amoeba"
|
|
|
|
elif amount < 100:
|
|
|
|
fish_str = "plankton"
|
|
|
|
elif amount >= 100 and amount < 200:
|
|
|
|
fish_str = "Paedocypris"
|
|
|
|
elif amount >= 200 and amount < 500:
|
|
|
|
fish_str = "Dwarf Goby"
|
|
|
|
elif amount >= 500 and amount < 1000:
|
|
|
|
fish_str = "European Pilchard"
|
|
|
|
elif amount >= 1000 and amount < 2000:
|
|
|
|
fish_str = "Goldfish"
|
|
|
|
elif amount >= 2000 and amount < 4000:
|
|
|
|
fish_str = "Herring"
|
|
|
|
elif amount >= 4000 and amount < 7000:
|
|
|
|
fish_str = "Atlantic Macerel"
|
|
|
|
elif amount >= 7000 and amount < 9000:
|
|
|
|
fish_str = "Gilt-head Bream"
|
|
|
|
elif amount >= 9000 and amount < 12000:
|
|
|
|
fish_str = "Salmonidae"
|
|
|
|
elif amount >= 12000 and amount < 20000:
|
|
|
|
fish_str = "Gadidae"
|
|
|
|
elif amount >= 20000 and amount < 40000:
|
|
|
|
fish_str = "Norwegian Delicious Salmon"
|
|
|
|
elif amount >= 40000 and amount < 60000:
|
|
|
|
fish_str = "Electric eel"
|
|
|
|
elif amount >= 60000 and amount < 80000:
|
|
|
|
fish_str = "Tuna"
|
|
|
|
elif amount >= 80000 and amount < 100000:
|
|
|
|
fish_str = "Wels catfish"
|
|
|
|
elif amount >= 100000 and amount < 120000:
|
|
|
|
fish_str = "Black marlin"
|
|
|
|
elif amount >= 120000 and amount < 160000:
|
|
|
|
fish_str = "Shark"
|
|
|
|
elif amount >= 160000 and amount < 220000:
|
|
|
|
fish_str = "Dolphin"
|
|
|
|
elif amount >= 220000 and amount < 320000:
|
|
|
|
fish_str = "Narwhal"
|
|
|
|
elif amount >= 320000 and amount < 500000:
|
|
|
|
fish_str = "Orca"
|
|
|
|
elif amount >= 500000 and amount < 700000:
|
|
|
|
fish_str = "Blue Whale"
|
|
|
|
elif amount >= 700000 and amount < 1000000:
|
|
|
|
fish_str = "Leviathan"
|
|
|
|
else:
|
|
|
|
fish_str = "Cthulu"
|
|
|
|
return fish_str
|
|
|
|
|