commit
cf178140d3
5 changed files with 258 additions and 0 deletions
@ -0,0 +1,6 @@ |
|||
FROM python:3.10 |
|||
RUN python -m pip install git+https://git.pblr-nyk.pro/gsd/Donation-Alerts-API-Python aiohttp websocket-client |
|||
ENV PYTHONUNBUFFERED 1 |
|||
WORKDIR /app |
|||
COPY ./ ./ |
|||
ENTRYPOINT ["python", "paybot.facti13.py"] |
@ -0,0 +1,77 @@ |
|||
import ssl |
|||
import aiohttp, os, sys |
|||
from colors import * |
|||
import traceback |
|||
|
|||
class BackendClient: |
|||
up = False |
|||
warn = False |
|||
|
|||
def __init__(self): |
|||
if not os.getenv("BACKEND_URL", ""): |
|||
error("BACKEND_URL not setted in env") |
|||
sys.exit(10) |
|||
if not os.getenv("SECRET_KEY", ""): |
|||
error("SECRET_KEY not setted in env") |
|||
sys.exit(11) |
|||
self.secret_key = os.getenv("SECRET_KEY") |
|||
self.pulse_url = f"{os.getenv('BACKEND_URL')}/api/pulse" |
|||
self.vip_url = f"{os.getenv('BACKEND_URL')}/api/external/vip" |
|||
self.detect = f"{os.getenv('BACKEND_URL')}/api/profile/steam" |
|||
|
|||
async def getSteam(self, any): |
|||
async with aiohttp.ClientSession(cookies={ |
|||
"secretkey":self.secret_key}) as session: |
|||
async with session.post(self.detect, ssl=False, json={"any":any}) as response: |
|||
return await response.json() |
|||
|
|||
async def pulse(self, exit_if_error = False): |
|||
async with aiohttp.ClientSession(cookies={ |
|||
"secretkey":self.secret_key}) as session: |
|||
try: |
|||
async with session.get(self.pulse_url, ssl=False) as response: |
|||
await response.text() |
|||
self.up = True |
|||
if not self.warn: |
|||
info("Backend connected!") |
|||
self.warn = True |
|||
except: |
|||
error("Backend not respond") |
|||
traceback.print_exc() |
|||
if exit_if_error: |
|||
sys.exit(200) |
|||
self.up = False |
|||
self.warn = False |
|||
return self.up |
|||
|
|||
async def vip(self, steamid, amount: int, extra: str, unique: str = ""): |
|||
async with aiohttp.ClientSession(cookies={ |
|||
"secretkey":self.secret_key}) as session: |
|||
async with session.post(self.vip_url + f"?steam={steamid}&amount={int(amount)}&service=qiwi&extra={extra}&unique={unique}", ssl=False) as response: |
|||
try: |
|||
result = int(await response.text()) |
|||
if result == 0: |
|||
warning(f"[S64:{steamid}] VIP as not be added, maybe permition already exists") |
|||
return 99 |
|||
elif result > 0: |
|||
info(f"[S64:{steamid}] VIP has be added!") |
|||
return 100 |
|||
elif result < 0: |
|||
info(f"[S64:{steamid}] VIP has be extends!") |
|||
return 101 |
|||
except: |
|||
error(f"[S64:{steamid}] Backend returned error") |
|||
traceback.print_exc() |
|||
return False |
|||
return False |
|||
|
|||
async def prices(self, exit_if_error = False): |
|||
async with aiohttp.ClientSession() as session: |
|||
try: |
|||
async with session.get(self.vip_url) as response: |
|||
return await response.json() |
|||
except: |
|||
traceback.print_exc() |
|||
if exit_if_error: |
|||
error("Cannot fetch prices") |
|||
sys.exit(200) |
@ -0,0 +1,29 @@ |
|||
from time import strftime |
|||
|
|||
class colors: |
|||
HEADER = u'\033[95m' |
|||
OKBLUE = u'\033[94m' |
|||
OKGREEN = u'\033[92m' |
|||
WARNING = u'\033[93m' |
|||
FAIL = u'\033[91m' |
|||
ENDC = u'\033[0m' |
|||
BOLD = u'\033[1m' |
|||
UNDERLINE = u'\033[4m' |
|||
|
|||
HEADER = u"[{:^7}] ({:^17}) {}" |
|||
|
|||
def ok(text): |
|||
HEAD = colors.OKGREEN + u"OK" + colors.ENDC |
|||
return print(HEADER.format(HEAD, strftime(u"%H:%M:%S %d.%m.%y"), text)) |
|||
|
|||
def error(text): |
|||
HEAD = colors.FAIL + u"FAIL" + colors.ENDC |
|||
return print(HEADER.format(HEAD, strftime(u"%H:%M:%S %d.%m.%y"), text)) |
|||
|
|||
def warning(text): |
|||
HEAD = colors.WARNING + u"WARNING" + colors.ENDC |
|||
return print(HEADER.format(HEAD, strftime(u"%H:%M:%S %d.%m.%y"), text)) |
|||
|
|||
def info(text): |
|||
HEAD = colors.HEADER + u"INFO" + colors.ENDC |
|||
return print(HEADER.format(HEAD, strftime(u"%H:%M:%S %d.%m.%y"), text)) |
@ -0,0 +1,85 @@ |
|||
from http import client |
|||
from turtle import back |
|||
from donationalerts_api.asyncio_api import Alert, Event |
|||
from backend_integration import BackendClient |
|||
from colors import * |
|||
import os, sys, asyncio |
|||
import ws |
|||
|
|||
class CustomAlerts(Alert): |
|||
backend: BackendClient |
|||
prices = {} |
|||
|
|||
convert_map = { |
|||
"month":30*24*60*60, |
|||
"week":7*24*60*60, |
|||
"day":1*24*60*60 |
|||
} |
|||
|
|||
def __init__(self, token, backend): |
|||
super().__init__(token) |
|||
|
|||
def run(self): |
|||
@self.event() |
|||
async def handler(event): |
|||
print(event) |
|||
self.calculate(event) |
|||
|
|||
async def calculate(self, event: Event): |
|||
if event.currency != "RUB": |
|||
warning(f"[{event.id}] donate is not RUB") |
|||
return None |
|||
|
|||
info("Update prices") |
|||
await self.update_prices_from_server() |
|||
|
|||
money = int(event.amount) |
|||
steam2 = self.backend.detect(event.message).steam2 |
|||
uid = event.id |
|||
|
|||
seconds = self.price_checker(money) |
|||
if seconds == 0: |
|||
warning(f"[{uid}] so smol donate {money} RUB") |
|||
|
|||
info(f"Add vip {steam2}, amount {money}") |
|||
await self.backend.vip(steam2, seconds, f"rub={int(money * 0.9)};", unique=f"da_{uid}") |
|||
|
|||
async def update_prices_from_server(self, exit_if_error = True): |
|||
#info("Fetch prices from server backend") |
|||
prices = await self.backend.prices(exit_if_error) |
|||
self.update_prices(prices) |
|||
|
|||
def update_prices(self, prices): |
|||
for price in prices: |
|||
if price["money_price"] == 0: |
|||
continue |
|||
self.prices = {} |
|||
self.prices.update({float(price["money_price"]) + float(price["money_price"] * price["da_percent"] / 10):self.convert_map[price['period']]}) |
|||
if not self.prices: |
|||
error("cannot get prices from server") |
|||
sys.exit(228) |
|||
|
|||
def price_checker(self, amount): |
|||
seconds2give = 0 |
|||
for rub, sec in self.prices.items(): |
|||
if amount >= rub and seconds2give <= sec: |
|||
seconds2give = sec |
|||
return seconds2give |
|||
|
|||
if __name__ == "__main__": |
|||
if not os.getenv("BACKEND_URL", "") or not os.getenv("SECRET_KEY", ""): |
|||
error("BACKEND_URL or SECRET_KEY is not setted!") |
|||
sys.exit(1) |
|||
if not os.getenv("DA_TOKEN", ""): |
|||
error("DA_TOKEN in not setted!") |
|||
sys.exit(2) |
|||
|
|||
print("Build date: "+os.getenv("BUILDDATE", "not setted")) |
|||
def run(): |
|||
client = CustomAlerts(os.getenv("DA_TOKEN"), BackendClient()) |
|||
client.run() |
|||
|
|||
ws.SERVICE_NAME = "dapay" |
|||
ws.START_AFTER_CONNECT = run |
|||
wsc = ws.WS(os.getenv("BACKEND_URL"), os.getenv("SECRET_KEY")) |
|||
wsc.run() |
@ -0,0 +1,61 @@ |
|||
import websocket, sys |
|||
from threading import Thread |
|||
from json import dumps |
|||
from time import sleep |
|||
import os |
|||
|
|||
START_AFTER_CONNECT = None |
|||
SERVICE_NAME = None |
|||
|
|||
def create_pulser(ws: websocket.WebSocketApp): |
|||
def run(ws): |
|||
builddate = int(os.getenv("BUILDDATE", "0")) |
|||
while 1: |
|||
ws.send_text(dumps({"name":SERVICE_NAME, "builddate":builddate})) |
|||
sleep(15) |
|||
thread = Thread(target=run, args=[ws], daemon=True) |
|||
thread.start() |
|||
|
|||
def start(): |
|||
print("substart") |
|||
|
|||
|
|||
class WS: |
|||
ws: websocket.WebSocketApp |
|||
def __init__(self, backend, auth): |
|||
if (START_AFTER_CONNECT is None or SERVICE_NAME is None): |
|||
print("Setup START_AFTER_CONNECT and SERVICE_NAME") |
|||
sys.exit(1) |
|||
else: |
|||
print(f"Run after setup {SERVICE_NAME}: {START_AFTER_CONNECT}") |
|||
|
|||
addr = backend |
|||
if not "wss://" in addr: |
|||
addr = addr.replace("https://", "wss://") |
|||
if not "/ws/services" in addr: |
|||
addr += "/ws/services" |
|||
|
|||
print(f"Connect to {addr}, secret len: {len(auth)}") |
|||
self.ws = websocket.WebSocketApp(addr, cookie="secretkey="+auth, on_close=self.on_close, on_open=self.on_open) |
|||
|
|||
@staticmethod |
|||
def on_close(ws_self, status_code, msg): |
|||
print(f"WS Close, code:{status_code}, msg: {msg}") |
|||
sys.exit(0) |
|||
|
|||
@staticmethod |
|||
def on_open(ws_self): |
|||
print("Connected") |
|||
create_pulser(ws_self) |
|||
if START_AFTER_CONNECT: |
|||
START_AFTER_CONNECT() |
|||
|
|||
def run(self): |
|||
if self.ws.run_forever(): |
|||
print("fuck") |
|||
|
|||
if __name__ == "__main__": |
|||
START_AFTER_CONNECT = start |
|||
SERVICE_NAME = "test" |
|||
wsc = WS("wss://tf2.pblr-nyk.pro/ws/services", "") |
|||
wsc.run() |
Loading…
Reference in new issue