pluto/handler/agora.py

881 lines
35 KiB
Python
Raw Normal View History

2021-12-24 02:23:38 +00:00
# Twisted/Klein imports
from twisted.logger import Logger
2021-12-27 19:30:03 +00:00
from twisted.internet.task import LoopingCall
2022-01-12 19:55:50 +00:00
from twisted.internet.threads import deferToThread
2021-12-24 02:23:38 +00:00
# Other library imports
from json import loads
2021-12-24 02:23:38 +00:00
from forex_python.converter import CurrencyRates
2022-01-08 17:49:04 +00:00
from agoradesk_py import AgoraDesk
2022-01-25 08:43:53 +00:00
from httpx import ReadTimeout, ReadError
2022-01-09 14:34:37 +00:00
from pycoingecko import CoinGeckoAPI
2022-01-12 19:17:53 +00:00
from datetime import datetime
from time import sleep
2021-12-24 02:23:38 +00:00
# Project imports
from settings import settings
2022-01-27 12:08:26 +00:00
log = Logger("agora.global")
2021-12-24 02:23:38 +00:00
def handle_exceptions(func):
def inner_function(*args, **kwargs):
try:
rtrn = func(*args, **kwargs)
except (ReadTimeout, ReadError):
return False
2022-01-27 12:08:26 +00:00
if isinstance(rtrn, dict):
if "success" in rtrn:
if "message" in rtrn:
if not rtrn["success"] and rtrn["message"] == "API ERROR":
log.error("API error: {code}", code=rtrn["response"]["error"]["error_code"])
return False
return rtrn
return inner_function
2021-12-24 02:23:38 +00:00
class Agora(object):
"""
AgoraDesk API handler.
"""
2021-12-27 20:59:24 +00:00
def __init__(self):
2021-12-27 21:33:49 +00:00
"""
Initialise the AgoraDesk and CurrencyRates APIs.
Initialise the last_dash storage for detecting new trades.
"""
2021-12-24 02:23:38 +00:00
self.log = Logger("agora")
2021-12-24 17:26:31 +00:00
self.agora = AgoraDesk(settings.Agora.Token)
2021-12-24 02:23:38 +00:00
self.cr = CurrencyRates()
2022-01-09 14:34:37 +00:00
self.cg = CoinGeckoAPI()
# Cache for detecting new trades
2021-12-27 19:30:03 +00:00
self.last_dash = set()
2021-12-24 02:23:38 +00:00
# Cache for detecting new messages
self.last_messages = {}
2022-01-27 16:52:32 +00:00
# Assets that cheat has been run on
self.cheat_run_on = []
2021-12-27 20:59:24 +00:00
def set_irc(self, irc):
self.irc = irc
def set_tx(self, tx):
self.tx = tx
2022-01-27 19:12:15 +00:00
def set_notify(self, notify):
self.notify = notify
2021-12-27 19:30:03 +00:00
def setup_loop(self):
2021-12-27 21:33:49 +00:00
"""
Set up the LoopingCall to get all active trades and messages.
2021-12-27 21:33:49 +00:00
"""
2022-01-01 16:34:32 +00:00
self.lc_dash = LoopingCall(self.loop_check)
self.lc_dash.start(int(settings.Agora.RefreshSec))
2022-01-11 21:40:55 +00:00
if settings.Agora.Cheat == "1":
2022-01-27 16:52:32 +00:00
self.lc_cheat = LoopingCall(self._update_prices, None, None)
2022-01-11 21:40:55 +00:00
self.lc_cheat.start(int(settings.Agora.CheatSec))
2022-01-11 21:40:05 +00:00
@handle_exceptions
2022-01-09 22:11:39 +00:00
def wrap_dashboard(self):
dash = self.agora.dashboard_seller()
2022-01-27 16:52:32 +00:00
if dash is None:
return False
2022-01-27 12:08:26 +00:00
if dash is False:
2022-01-26 19:46:23 +00:00
return False
dash_tmp = {}
2022-01-27 16:52:32 +00:00
if not dash.items():
return False
2021-12-28 14:09:33 +00:00
if "data" not in dash["response"].keys():
self.log.error("Data not in dashboard response: {content}", content=dash)
2022-01-09 22:11:39 +00:00
return dash_tmp
2021-12-27 19:30:03 +00:00
if dash["response"]["data"]["contact_count"] > 0:
for contact in dash["response"]["data"]["contact_list"]:
contact_id = contact["data"]["contact_id"]
dash_tmp[contact_id] = contact
2022-01-09 22:11:39 +00:00
return dash_tmp
def loop_check(self):
"""
Calls hooks to parse dashboard info and get all contact messages.
"""
dash_tmp = self.wrap_dashboard()
2021-12-31 18:21:56 +00:00
2022-01-01 16:34:32 +00:00
# Call dashboard hooks
self.dashboard_hook(dash_tmp)
# Get recent messages
self.get_recent_messages()
2021-12-27 19:30:03 +00:00
return dash_tmp
2021-12-24 02:23:38 +00:00
2022-01-01 16:34:32 +00:00
def get_dashboard(self):
"""
Get dashboard helper for IRC only.
"""
2022-01-09 22:11:39 +00:00
dash = self.wrap_dashboard()
rtrn = []
2022-01-27 12:08:26 +00:00
if dash is False:
return False
2022-01-09 22:11:39 +00:00
for contact_id, contact in dash.items():
reference = self.tx.tx_to_ref(contact_id)
buyer = contact["data"]["buyer"]["username"]
amount = contact["data"]["amount"]
2022-01-21 13:54:47 +00:00
asset = contact["data"]["advertisement"]["asset"]
if asset == "XMR":
amount_crypto = contact["data"]["amount_xmr"]
2022-01-21 13:54:47 +00:00
elif asset == "BTC":
amount_crypto = contact["data"]["amount_btc"]
2022-01-09 22:11:39 +00:00
currency = contact["data"]["currency"]
if not contact["data"]["is_selling"]:
continue
2022-01-21 13:54:47 +00:00
rtrn.append(f"{reference}: {buyer} {amount}{currency} {amount_crypto}{asset}")
2022-01-09 22:11:39 +00:00
return rtrn
2022-01-01 16:34:32 +00:00
def dashboard_hook(self, dash):
2021-12-27 21:33:49 +00:00
"""
Get information about our open trades.
Post new trades to IRC and cache trades for the future.
2021-12-27 21:33:49 +00:00
"""
current_trades = []
2022-01-21 09:17:18 +00:00
if not dash:
return
if not dash.items():
return
for contact_id, contact in dash.items():
reference = self.tx.tx_to_ref(contact_id)
2021-12-29 18:40:44 +00:00
if reference:
current_trades.append(reference)
buyer = contact["data"]["buyer"]["username"]
amount = contact["data"]["amount"]
2022-01-21 13:54:47 +00:00
asset = contact["data"]["advertisement"]["asset"]
if asset == "XMR":
amount_crypto = contact["data"]["amount_xmr"]
2022-01-21 13:54:47 +00:00
elif asset == "BTC":
amount_crypto = contact["data"]["amount_btc"]
currency = contact["data"]["currency"]
if not contact["data"]["is_selling"]:
continue
if reference not in self.last_dash:
2022-01-21 13:54:47 +00:00
reference = self.tx.new_trade(asset, contact_id, buyer, currency, amount, amount_crypto)
2021-12-29 18:40:44 +00:00
if reference:
if reference not in current_trades:
current_trades.append(reference)
2022-01-01 16:34:32 +00:00
# Let us know there is a new trade
2022-01-21 13:54:47 +00:00
self.irc.sendmsg(f"AUTO {reference}: {buyer} {amount}{currency} {amount_crypto}{asset}")
2022-01-01 16:34:32 +00:00
# Note that we have seen this reference
self.last_dash.add(reference)
# Purge old trades from cache
for ref in list(self.last_dash): # We're removing from the list on the fly
if ref not in current_trades:
self.last_dash.remove(ref)
2021-12-29 18:40:44 +00:00
if reference and reference not in current_trades:
current_trades.append(reference)
2021-12-29 15:03:47 +00:00
self.tx.cleanup(current_trades)
2021-12-24 02:23:38 +00:00
@handle_exceptions
2021-12-28 13:42:31 +00:00
def dashboard_release_urls(self):
"""
Get information about our open trades.
Post new trades to IRC and cache trades for the future.
:return: human readable list of strings about our trades or False
:rtype: list or bool
"""
dash = self.agora.dashboard_seller()
2022-01-27 12:08:26 +00:00
if dash is False:
return False
2021-12-28 13:42:31 +00:00
dash_tmp = []
if "data" not in dash["response"]:
2021-12-28 14:09:33 +00:00
self.log.error("Data not in dashboard response: {content}", content=dash)
2021-12-28 13:42:31 +00:00
return False
if dash["response"]["data"]["contact_count"] > 0:
for contact in dash["response"]["data"]["contact_list"]:
contact_id = contact["data"]["contact_id"]
buyer = contact["data"]["buyer"]["username"]
amount = contact["data"]["amount"]
2022-01-21 13:54:47 +00:00
asset = contact["data"]["advertisement"]["asset"]
if asset == "XMR":
amount_crypto = contact["data"]["amount_xmr"]
2022-01-21 13:54:47 +00:00
elif asset == "BTC":
amount_crypto = contact["data"]["amount_btc"]
2021-12-28 13:42:31 +00:00
currency = contact["data"]["currency"]
release_url = contact["actions"]["release_url"]
if not contact["data"]["is_selling"]:
continue
reference = self.tx.tx_to_ref(contact_id)
if not reference:
reference = "not_set"
2022-01-21 13:54:47 +00:00
dash_tmp.append(f"{reference}: {buyer} {amount}{currency} {amount_crypto}{asset} {release_url}")
2021-12-28 13:42:31 +00:00
return dash_tmp
@handle_exceptions
def get_recent_messages(self, send_irc=True):
2021-12-27 21:33:49 +00:00
"""
Get recent messages.
2021-12-27 21:33:49 +00:00
"""
messages_tmp = {}
messages = self.agora.recent_messages()
2022-01-27 12:08:26 +00:00
if messages is False:
return False
if not messages["success"]:
2021-12-30 14:15:24 +00:00
return False
2022-01-12 19:17:53 +00:00
if "data" not in messages["response"]:
2022-01-11 21:16:49 +00:00
self.log.error("Data not in messages response: {content}", content=messages["response"])
return False
open_tx = self.tx.get_ref_map().keys()
for message in messages["response"]["data"]["message_list"]:
contact_id = message["contact_id"]
username = message["sender"]["username"]
msg = message["msg"]
if contact_id not in open_tx:
continue
reference = self.tx.tx_to_ref(contact_id)
if reference in messages_tmp:
messages_tmp[reference].append([username, msg])
else:
messages_tmp[reference] = [[username, msg]]
# Send new messages on IRC
if send_irc:
for user, message in messages_tmp[reference]:
if reference in self.last_messages:
if not [user, message] in self.last_messages[reference]:
self.irc.sendmsg(f"AUTO {reference}: ({user}) {message}")
# Append sent messages to last_messages so we don't send them again
self.last_messages[reference].append([user, message])
else:
self.last_messages[reference] = [[user, message]]
for x in messages_tmp[reference]:
self.irc.sendmsg(f"NEW {reference}: ({user}) {message}")
# Purge old trades from cache
for ref in list(self.last_messages): # We're removing from the list on the fly
if ref not in messages_tmp:
del self.last_messages[ref]
return messages_tmp
2021-12-24 02:23:38 +00:00
@handle_exceptions
2021-12-30 14:15:24 +00:00
def enum_ad_ids(self, page=0):
ads = self.agora._api_call(api_method="ads", query_values={"page": page})
2022-01-27 12:08:26 +00:00
if ads is False:
return False
2021-12-30 14:15:24 +00:00
ads_total = []
if not ads["success"]:
return False
for ad in ads["response"]["data"]["ad_list"]:
ads_total.append(ad["data"]["ad_id"])
2021-12-31 00:39:52 +00:00
if "pagination" in ads["response"]:
if "next" in ads["response"]["pagination"]:
page += 1
2022-01-27 12:08:26 +00:00
ads_iter = self.enum_ad_ids(page)
if ads_iter is None:
return False
if ads_iter is False:
return False
for ad in ads_iter:
2021-12-31 00:39:52 +00:00
ads_total.append(ad)
return ads_total
@handle_exceptions
2022-01-27 16:52:32 +00:00
def enum_ads(self, asset=None, page=0):
query_values = {"page": page}
if asset:
query_values["asset"] = asset
ads = self.agora._api_call(api_method="ads", query_values=query_values)
2022-01-27 12:08:26 +00:00
if ads is False:
return False
2021-12-31 00:39:52 +00:00
ads_total = []
if not ads["success"]:
return False
for ad in ads["response"]["data"]["ad_list"]:
ads_total.append([ad["data"]["asset"], ad["data"]["ad_id"], ad["data"]["countrycode"], ad["data"]["currency"]])
2021-12-31 00:39:52 +00:00
if "pagination" in ads["response"]:
if "next" in ads["response"]["pagination"]:
page += 1
2022-01-27 16:52:32 +00:00
ads_iter = self.enum_ads(asset, page)
2022-01-27 12:08:26 +00:00
if ads_iter is None:
return False
if ads_iter is False:
return False
for ad in ads_iter:
ads_total.append([ad[0], ad[1], ad[2], ad[3]])
2021-12-30 14:15:24 +00:00
return ads_total
2022-01-12 19:17:53 +00:00
def last_online_recent(self, date):
"""
Check if the last online date was recent.
:param date: date last online
:type date: string
:return: bool indicating whether the date was recent enough
:rtype: bool
"""
date_parsed = datetime.strptime(date, "%Y-%m-%dT%H:%M:%S.%fZ")
now = datetime.now()
sec_ago_date = (now - date_parsed).total_seconds()
self.log.debug("Seconds ago date for {date} ^ {now}: {x}", date=date, now=str(now), x=sec_ago_date)
return sec_ago_date < 172800
@handle_exceptions
def enum_public_ads(self, coin, currency, page=0):
# buy-monero-online, buy-bitcoin-online
# Work around Agora weirdness calling it bitcoins
if coin == "bitcoin":
coin = "bitcoins"
ads = self.agora._api_call(api_method=f"buy-{coin}-online/{currency}/REVOLUT", query_values={"page": page})
if ads is None:
return False
2022-01-27 12:08:26 +00:00
if ads is False:
2022-01-27 16:52:32 +00:00
return False
if "data" not in ads["response"]:
2022-01-09 14:34:37 +00:00
return False
for ad in ads["response"]["data"]["ad_list"]:
if ad["data"]["online_provider"] == "REVOLUT":
2022-01-12 19:17:53 +00:00
date_last_seen = ad["data"]["profile"]["last_online"]
# Check if this person was seen recently
if not self.last_online_recent(date_last_seen):
continue
2022-01-09 14:34:37 +00:00
yield [ad["data"]["ad_id"], ad["data"]["profile"]["username"], ad["data"]["temp_price"]]
if "pagination" in ads["response"]:
if "next" in ads["response"]["pagination"]:
page += 1
2022-01-27 12:08:26 +00:00
ads_iter = self.enum_public_ads(coin, currency, page)
if ads_iter is None:
return False
if ads_iter is False:
return False
for ad in ads_iter:
2022-01-09 14:34:37 +00:00
yield [ad[0], ad[1], ad[2]]
def wrap_public_ads(self, asset, currency, rates=None):
2022-01-09 14:34:37 +00:00
"""
Wrapper to sort public ads.
"""
if asset == "XMR":
coin = "monero"
elif asset == "BTC":
coin = "bitcoin"
2022-01-27 12:08:26 +00:00
ads_obj = self.enum_public_ads(coin, currency.upper())
if ads_obj is False:
return False
ads = list(ads_obj)
if ads_obj is False:
return False
2022-01-09 14:34:37 +00:00
if not rates:
# Set the price based on the asset
base_currency_price = self.cg.get_price(ids=coin, vs_currencies=currency)[coin][currency.lower()]
2022-01-09 14:34:37 +00:00
else:
base_currency_price = rates
2022-01-09 14:34:37 +00:00
for ad in ads:
price = float(ad[2])
rate = round(price / base_currency_price, 2)
2022-01-09 14:34:37 +00:00
ad.append(rate)
return ads
2022-01-27 16:52:32 +00:00
def _update_prices(self, xmr=None, btc=None):
2022-01-12 19:55:50 +00:00
"""
Update prices in another thread.
"""
2022-01-27 16:52:32 +00:00
if xmr is None and btc is None:
all_assets = loads(settings.Agora.AssetList)
assets_not_run = set(all_assets) ^ set(self.cheat_run_on)
if not assets_not_run:
print("TOP")
self.cheat_run_on = []
asset = list(all_assets).pop()
self.cheat_run_on.append(asset)
else:
print("BOTTOM")
asset = assets_not_run.pop()
self.cheat_run_on.append(asset)
print("end cheat", self.cheat_run_on)
if asset == "XMR":
deferToThread(self.update_prices, True, False) # XMR, BTC
print("Running cheat on XMR")
elif asset == "BTC":
deferToThread(self.update_prices, False, True) # XMR, BTC
print("Running cheat on BTC")
return asset
else:
print("Running cheat normally")
deferToThread(self.update_prices, xmr, btc)
2022-01-12 19:55:50 +00:00
@handle_exceptions
def update_prices(self, xmr=True, btc=True):
2022-01-27 16:52:32 +00:00
if xmr and btc:
our_ads = self.enum_ads()
elif xmr and not btc:
our_ads = self.enum_ads("XMR")
elif not xmr and btc:
our_ads = self.enum_ads("BTC")
self.log.debug("Our ads: {ads}", ads=our_ads)
to_update = []
2022-01-27 12:08:26 +00:00
if our_ads is None:
return False
if our_ads is False:
return False
currencies = [x[3].lower() for x in our_ads]
public_ad_dict_xmr = {}
public_ad_dict_btc = {}
rates_crypto = self.cg.get_price(ids=["monero", "bitcoin"], vs_currencies=currencies)
2022-01-09 17:52:26 +00:00
for currency in currencies:
try:
rates_xmr = rates_crypto["monero"][currency]
rates_btc = rates_crypto["bitcoin"][currency]
if xmr:
public_ad_dict_xmr[currency] = self.wrap_public_ads("XMR", currency, rates=rates_xmr)
2022-01-27 12:08:26 +00:00
if public_ad_dict_xmr[currency] is False:
return False
if btc:
public_ad_dict_btc[currency] = self.wrap_public_ads("BTC", currency, rates=rates_btc)
2022-01-27 12:08:26 +00:00
if public_ad_dict_btc[currency] is False:
return False
2022-01-09 17:52:26 +00:00
except KeyError:
self.log.error("Error getting public ads for currency {currency}", currency=currency)
continue
for asset, ad_id, country, currency in our_ads:
2022-01-09 14:34:37 +00:00
# Get the ads for this country/currency pair
try:
if asset == "XMR":
public_ads = public_ad_dict_xmr[currency.lower()]
elif asset == "BTC":
public_ads = public_ad_dict_btc[currency.lower()]
2022-01-09 14:34:37 +00:00
except KeyError:
continue
2022-01-26 19:46:23 +00:00
if not public_ads:
continue
if xmr:
if asset == "XMR":
new_margin = self.autoprice(public_ads, currency)
new_formula = f"coingeckoxmrusd*usd{currency.lower()}*{new_margin}"
if btc:
if asset == "BTC":
new_margin = self.autoprice(public_ads, currency)
new_formula = f"coingeckobtcusd*usd{currency.lower()}*{new_margin}"
# Get all of our ads
our_ads_list = [ad for ad in public_ads if ad[1] == settings.Agora.Username]
if not len(our_ads_list) == 1:
2022-01-27 12:08:26 +00:00
if not len(set([x[3] for x in our_ads_list])) == 1:
self.log.error("Our ads have different margins: {ads}", ads=our_ads_list)
# Get one from the list, they're probably the same, if not we will deal with the other
# ones in a later pass
# Get one
our_ad = our_ads_list.pop()
# Take the 4th argument, the margin
our_margin = our_ad[3]
# Don't waste API rate limits on setting the same margin as before
if new_margin != our_margin:
# rtrn = self.agora.ad_equation(ad_id, new_formula)
2022-01-27 19:12:15 +00:00
to_update.append([ad_id, new_formula, asset, currency, False])
# if not rtrn["success"]:
# self.log.error("Error updating ad {ad_id}: {response}", ad_id=ad_id, response=rtrn["response"])
self.log.info("Rate for {currency}: {margin}", currency=currency, margin=new_margin)
else:
self.log.info("Not changed rate for {currency}, keeping old margin of {margin}", currency=currency, margin=our_margin)
2022-01-27 19:12:15 +00:00
print("TO UPDATE", to_update)
self.slow_ad_update(to_update)
def slow_ad_update(self, ads):
"""
Slow ad equation update utilising exponential backoff in order to guarantee all ads are updated.
:param ads: our list of ads
"""
self.log.info("Beginning slow ad update for {num} ads", num=len(ads))
iterations = 0
throttled = 0
2022-01-27 16:52:32 +00:00
assets = set()
2022-01-27 19:12:15 +00:00
currencies = set()
while not all([x[4] for x in ads]) or iterations == 1000:
for ad_index in range(len(ads)):
2022-01-27 19:12:15 +00:00
ad_id, new_formula, asset, currency, actioned = ads[ad_index]
print("SLOW ITER", ad_id, new_formula, asset, currency, actioned)
2022-01-27 16:52:32 +00:00
assets.add(asset)
2022-01-27 19:12:15 +00:00
currencies.add(currency)
2022-01-27 16:52:32 +00:00
self.log.error("ASSET {a}", a=asset)
if not actioned:
rtrn = self.agora.ad_equation(ad_id, new_formula)
if rtrn["success"]:
2022-01-27 19:12:15 +00:00
ads[ad_index][4] = True
throttled = 0
self.log.info("Successfully updated ad: {id}", id=ad_id)
continue
else:
if rtrn["response"]["error"]["error_code"] == 429:
throttled += 1
sleep_time = pow(throttled, float(settings.Agora.SleepExponent))
self.log.info(
"Throttled {x} times while updating {id}, sleeping for {sleep} seconds",
x=throttled,
id=ad_id,
sleep=sleep_time,
)
# We're running in a thread, so this is fine
sleep(sleep_time)
self.log.error("Error updating ad {ad_id}: {response}", ad_id=ad_id, response=rtrn["response"])
continue
iterations += 1
2022-01-27 16:52:32 +00:00
if iterations == 0:
self.log.info("Slow ad update finished, no ads to update")
self.irc.sendmsg("Slow ad update finished, no ads to update")
else:
2022-01-27 19:12:15 +00:00
self.log.info(
"Slow ad update completed with {iterations} iterations: [{assets}] | [{currencies}]",
iterations=iterations,
assets=", ".join(assets),
currencies=", ".join(currencies),
)
self.irc.sendmsg(f"Slow ad update completed with {iterations} iterations: [{', '.join(assets)}] | [{', '.join(currencies)}]")
2022-01-09 14:34:37 +00:00
2022-01-09 17:52:26 +00:00
def autoprice(self, ads, currency):
2022-01-09 14:34:37 +00:00
"""
Helper function to automatically adjust the price up/down in certain markets
in order to gain the most profits and sales.
2022-01-12 19:17:53 +00:00
:param ads: list of ads
:type ads: list of lists
:param currency: currency of the ads
:type currency: string
:return: the rate we should use for this currency
:rtype: float
2022-01-09 14:34:37 +00:00
"""
2022-01-12 19:17:53 +00:00
self.log.debug("Autoprice starting for {x}", x=currency)
# Find cheapest ad
# Filter by 3rd index on each ad list to find the cheapest
min_margin_ad = min(ads, key=lambda x: x[3])
self.log.debug("Minimum margin ad: {x}", x=min_margin_ad)
# Find second cheapest that is not us
# Remove results from ads that are us
ads_without_us = [ad for ad in ads if not ad[1] == settings.Agora.Username]
self.log.debug("Ads without us: {x}", x=ads_without_us)
# Find ads above our min that are not us
ads_above_our_min_not_us = [ad for ad in ads_without_us if ad[3] > float(settings.Agora.MinMargin)]
self.log.debug("Ads above our min not us: {x}", x=ads_above_our_min_not_us)
# Check that this list without us is not empty
if ads_without_us:
# Find the cheapest from these
min_margin_ad_not_us = min(ads_without_us, key=lambda x: x[3])
self.log.debug("Min margin ad not us: {x}", x=min_margin_ad_not_us)
# Lowball the lowest ad that is not ours
lowball_lowest_not_ours = min_margin_ad_not_us[3] - 0.001
self.log.debug("Lowball lowest not ours: {x}", x=lowball_lowest_not_ours)
# Check if the username field of the cheapest ad matches ours
if min_margin_ad[1] == settings.Agora.Username:
self.log.debug("We are the cheapest for: {x}", x=currency)
# We are the cheapest!
# Are all of the ads ours?
all_ads_ours = all([ad[1] == settings.Agora.Username for ad in ads])
if all_ads_ours:
self.log.debug("All ads are ours for: {x}", x=currency)
# Now we know it's safe to return the maximum value
return float(settings.Agora.MaxMargin)
2022-01-09 14:34:37 +00:00
else:
2022-01-12 19:17:53 +00:00
self.log.debug("All ads are NOT ours for: {x}", x=currency)
# All the ads are not ours, but we are first...
# Check if the lowballed, lowest (that is not ours) ad's margin
# is less than our minimum
if lowball_lowest_not_ours < float(settings.Agora.MinMargin):
self.log.debug("Lowball lowest not ours less than MinMargin")
return float(settings.Agora.MinMargin)
2022-01-27 16:52:32 +00:00
elif lowball_lowest_not_ours > float(settings.Agora.MaxMargin):
self.log.debug("Lowball lowest not ours more than MaxMargin")
return float(settings.Agora.MaxMargin)
2022-01-12 19:17:53 +00:00
else:
self.log.debug("Returning lowballed figure: {x}", x=lowball_lowest_not_ours)
return lowball_lowest_not_ours
else:
self.log.debug("We are NOT the cheapest for: {x}", x=currency)
# We are not the cheapest :(
# Check if this list is empty
if not ads_above_our_min_not_us:
# Return the maximum margin?
return float(settings.Agora.MaxMargin)
# Find cheapest ad above our min that is not us
cheapest_ad = min(ads_above_our_min_not_us, key=lambda x: x[3])
2022-01-27 16:52:32 +00:00
cheapest_ad_margin = cheapest_ad[3] - 0.001
if cheapest_ad_margin > float(settings.Agora.MaxMargin):
self.log.debug("Cheapest ad not ours more than MaxMargin")
return float(settings.Agora.MaxMargin)
2022-01-12 19:17:53 +00:00
self.log.debug("Cheapest ad above our min that is not us: {x}", x=cheapest_ad)
2022-01-27 16:52:32 +00:00
return cheapest_ad_margin
2022-01-09 14:34:37 +00:00
@handle_exceptions
2021-12-30 14:15:24 +00:00
def nuke_ads(self):
"""
Delete all of our adverts.
:return: True or False
:rtype: bool
"""
2021-12-31 00:39:52 +00:00
ads = self.enum_ad_ids()
return_ids = []
2022-01-27 12:08:26 +00:00
if ads is False:
2021-12-30 14:15:24 +00:00
return False
for ad_id in ads:
2021-12-31 00:39:52 +00:00
rtrn = self.agora.ad_delete(ad_id)
return_ids.append(rtrn["success"])
return all(return_ids)
2021-12-24 02:23:38 +00:00
2021-12-24 17:26:31 +00:00
def get_rates_all(self):
2021-12-27 21:33:49 +00:00
"""
Get all rates that pair with USD.
:return: dictionary of USD/XXX rates
:rtype: dict
"""
2021-12-24 17:26:31 +00:00
rates = self.cr.get_rates("USD")
return rates
2021-12-24 02:23:38 +00:00
2021-12-28 23:56:32 +00:00
def get_acceptable_margins(self, currency, amount):
"""
Get the minimum and maximum amounts we would accept a trade for.
:param currency: currency code
:param amount: amount
:return: (min, max)
:rtype: tuple
"""
rates = self.get_rates_all()
if currency == "USD":
min_amount = amount - float(settings.Agora.AcceptableUSDMargin)
max_amount = amount + float(settings.Agora.AcceptableUSDMargin)
return (min_amount, max_amount)
amount_usd = amount / rates[currency]
min_usd = amount_usd - float(settings.Agora.AcceptableUSDMargin)
max_usd = amount_usd + float(settings.Agora.AcceptableUSDMargin)
min_local = min_usd * rates[currency]
max_local = max_usd * rates[currency]
return (min_local, max_local)
@handle_exceptions
2022-01-21 13:54:47 +00:00
def create_ad(self, asset, countrycode, currency):
2021-12-27 21:33:49 +00:00
"""
2022-01-21 13:54:47 +00:00
Post an ad with the given asset in a country with a given currency.
2021-12-27 21:33:49 +00:00
Convert the min and max amounts from settings to the given currency with CurrencyRates.
2022-01-21 13:54:47 +00:00
:param asset: the crypto asset to list (XMR or BTC)
:type asset: string
2021-12-27 21:33:49 +00:00
:param countrycode: country code
:param currency: currency code
:type countrycode: string
:type currency: string
:return: data about created object or error
:rtype: dict
"""
ad = settings.Agora.Ad
paymentdetails = settings.Agora.PaymentDetails
2022-01-24 19:17:12 +00:00
# Substitute the currency
ad = ad.replace("$CURRENCY$", currency)
if countrycode == "GB" and currency == "GBP":
2022-01-24 19:17:12 +00:00
ad = ad.replace("$PAYMENT$", settings.Agora.GBPDetailsAd)
paymentdetailstext = paymentdetails.replace("$PAYMENT$", settings.Agora.GBPDetailsPayment)
else:
2022-01-24 19:17:12 +00:00
ad = ad.replace("$PAYMENT$", settings.Agora.DefaultDetailsAd)
paymentdetailstext = paymentdetails.replace("$PAYMENT$", settings.Agora.DefaultDetailsPayment)
2022-01-24 19:17:12 +00:00
# Substitute the asset
2022-01-21 13:54:47 +00:00
ad = ad.replace("$ASSET$", asset)
2022-01-24 19:17:12 +00:00
2021-12-24 17:26:31 +00:00
rates = self.get_rates_all()
2022-01-23 17:08:01 +00:00
if asset == "XMR":
min_usd = float(settings.Agora.MinUSDXMR)
max_usd = float(settings.Agora.MaxUSDXMR)
elif asset == "BTC":
min_usd = float(settings.Agora.MinUSDBTC)
max_usd = float(settings.Agora.MaxUSDBTC)
if currency == "USD":
2022-01-23 17:08:01 +00:00
min_amount = min_usd
max_amount = max_usd
else:
2022-01-23 17:08:01 +00:00
min_amount = rates[currency] * min_usd
max_amount = rates[currency] * max_usd
price_formula = f"coingecko{asset.lower()}usd*usd{currency.lower()}*{settings.Agora.Margin}"
# price_formula = f"coingeckoxmrusd*{settings.Agora.Margin}"
2022-01-24 19:17:12 +00:00
# Remove extra tabs
2021-12-27 13:50:32 +00:00
ad = ad.replace("\\t", "\t")
2022-01-23 17:08:01 +00:00
2022-01-25 08:43:53 +00:00
form = {
"country_code": countrycode,
"currency": currency,
"trade_type": "ONLINE_SELL",
"asset": asset,
"price_equation": price_formula,
"track_max_amount": False,
"require_trusted_by_advertiser": False,
"online_provider": "REVOLUT",
"msg": ad,
"min_amount": min_amount,
"max_amount": max_amount,
"payment_method_details": settings.Agora.PaymentMethodDetails,
"account_info": paymentdetailstext,
}
2022-01-23 17:08:01 +00:00
# Dirty hack to test
2022-01-25 08:43:53 +00:00
# if asset == "BTC":
2022-01-23 17:08:01 +00:00
# del form["min_amount"]
ad = self.agora.ad_create(**form)
return ad
2021-12-24 02:23:38 +00:00
def dist_countries(self):
2021-12-27 21:33:49 +00:00
"""
Distribute our advert into all countries listed in the config.
Exits on errors.
:return: False or dict with response
:rtype: bool or dict
2021-12-27 21:33:49 +00:00
"""
2022-01-21 13:54:47 +00:00
for asset in loads(settings.Agora.AssetList):
for currency, countrycode in loads(settings.Agora.DistList):
rtrn = self.create_ad(asset, countrycode, currency)
2022-01-27 12:08:26 +00:00
if rtrn is False:
2022-01-21 13:54:47 +00:00
return False
2021-12-31 00:39:52 +00:00
yield rtrn
2022-01-25 08:43:53 +00:00
# def get_combinations(self):
# """
# Get all combinations of currencies and countries from the configuration.
# :return: list of [country, currency]
# :rtype: list
# """
# currencies = loads(settings.Agora.BruteCurrencies)
# countries = loads(settings.Agora.BruteCountries)
# combinations = [[country, currency] for country in countries for currency in currencies]
# return combinations
# def dist_bruteforce(self):
# """
# Bruteforce all possible ads from the currencies and countries in the config.
# Does not exit on errors.
# :return: False or dict with response
# :rtype: bool or dict
# """
# combinations = self.get_combinations()
# for country, currency in combinations:
# rtrn = self.create_ad(country, currency)
# if not rtrn:
# yield False
# yield rtrn
#
# def bruteforce_fill_blanks(self):
# """
# Get the ads that we want to configure but have not, and fill in the blanks.
# :return: False or dict with response
# :rtype: bool or dict
# """
# existing_ads = self.enum_ads()
# combinations = self.get_combinations()
# for country, currency in combinations:
# if not [country, currency] in existing_ads:
# rtrn = self.create_ad(country, currency)
# if not rtrn:
# yield False
# yield rtrn
2022-01-21 13:54:47 +00:00
@handle_exceptions
2021-12-31 00:39:52 +00:00
def strip_duplicate_ads(self):
"""
Remove duplicate ads.
:return: list of duplicate ads
:rtype: list
"""
existing_ads = self.enum_ads()
_size = len(existing_ads)
repeated = []
for i in range(_size):
k = i + 1
for j in range(k, _size):
if existing_ads[i] == existing_ads[j] and existing_ads[i] not in repeated:
repeated.append(existing_ads[i])
actioned = []
for ad_id, country, currency in repeated:
rtrn = self.agora.ad_delete(ad_id)
actioned.append(rtrn["success"])
return all(actioned)
@handle_exceptions
2021-12-29 19:21:05 +00:00
def release_funds(self, contact_id):
"""
Release funds for a contact_id.
:param contact_id: trade/contact ID
:type contact_id: string
:return: response dict
:rtype: dict
"""
payload = {"tradeId": contact_id, "password": settings.Agora.Pass}
rtrn = self.agora._api_call(api_method=f"contact_release/{contact_id}", http_method="POST", query_values=payload)
2022-01-25 18:39:35 +00:00
# Check if we can withdraw funds
self.withdraw_funds()
2021-12-29 19:21:05 +00:00
return rtrn
2022-01-25 17:54:42 +00:00
@handle_exceptions
2022-01-25 17:54:42 +00:00
def withdraw_funds(self):
"""
Withdraw excess funds to our XMR/BTC wallets.
"""
2022-01-25 18:39:35 +00:00
totals_all = self.tx.get_total()
if totals_all is False:
return False
2022-01-25 18:39:35 +00:00
wallet_xmr, _ = totals_all[2]
# Get the wallet balances in USD
total_usd = totals_all[0][1]
profit_usd = total_usd - float(settings.Money.BaseUSD)
# Get the XMR -> USD exchange rate
xmr_usd = self.cg.get_price(ids="monero", vs_currencies=["USD"])
# Convert the USD total to XMR
profit_usd_in_xmr = float(profit_usd) / xmr_usd["monero"]["usd"]
# Check profit is above zero
if not profit_usd >= 0:
return
if not float(wallet_xmr) > profit_usd_in_xmr:
# Not enough funds to withdraw
self.log.error(
"Not enough funds to withdraw {profit}, as wallet only contains {wallet}", profit=profit_usd_in_xmr, wallet=wallet_xmr
)
self.irc.sendmsg(f"Not enough funds to withdraw {profit_usd_in_xmr}, as wallet only contains {wallet_xmr}")
2022-01-25 18:39:35 +00:00
return
if not profit_usd >= float(settings.Money.WithdrawLimit):
# Not enough profit to withdraw
return
half = profit_usd_in_xmr / 2
half_rounded = round(half, 8)
# Set up the format for calling wallet_send_xmr
send_cast = {
"address": None,
"amount": half_rounded,
"password": settings.Agora.Pass,
}
send_cast["address"] = settings.XMR.Wallet1
rtrn1 = self.agora.wallet_send_xmr(**send_cast)
send_cast["address"] = settings.XMR.Wallet2
rtrn2 = self.agora.wallet_send_xmr(**send_cast)
self.irc.sendmsg(f"Withdrawal: {rtrn1['success']} | {rtrn2['success']}")
2022-01-27 19:12:15 +00:00
self.notify.notify_withdrawal(profit_usd / 2)
def to_usd(self, amount, currency):
if currency == "USD":
return float(amount)
else:
rates = self.get_rates_all()
return float(amount) / rates[currency]