2024-07-01 16:00:17 +00:00
|
|
|
import asyncio
|
2022-01-16 12:31:25 +00:00
|
|
|
from celery import shared_task
|
2023-05-10 19:07:28 +00:00
|
|
|
from celery.exceptions import SoftTimeLimitExceeded
|
2022-01-16 12:31:25 +00:00
|
|
|
|
2022-10-20 09:56:10 +00:00
|
|
|
|
2023-04-08 09:24:47 +00:00
|
|
|
@shared_task(name="users_cleansing", time_limit=600)
|
2022-01-16 12:31:25 +00:00
|
|
|
def users_cleansing():
|
2022-02-17 19:50:10 +00:00
|
|
|
"""
|
2022-01-16 12:31:25 +00:00
|
|
|
Deletes users never used 12 hours after creation
|
2022-02-17 19:50:10 +00:00
|
|
|
"""
|
2022-10-25 18:04:12 +00:00
|
|
|
from datetime import timedelta
|
|
|
|
|
2022-01-17 16:41:55 +00:00
|
|
|
from django.contrib.auth.models import User
|
|
|
|
from django.db.models import Q
|
|
|
|
from django.utils import timezone
|
|
|
|
|
2022-10-25 18:04:12 +00:00
|
|
|
from api.logics import Logics
|
|
|
|
|
2022-01-18 17:52:48 +00:00
|
|
|
# Users who's last login has not been in the last 6 hours
|
|
|
|
active_time_range = (timezone.now() - timedelta(hours=6), timezone.now())
|
2022-01-16 12:31:25 +00:00
|
|
|
queryset = User.objects.filter(~Q(last_login__range=active_time_range))
|
2022-02-17 19:50:10 +00:00
|
|
|
queryset = queryset.filter(is_staff=False) # Do not delete staff users
|
|
|
|
|
2022-03-05 18:43:15 +00:00
|
|
|
# And do not have an active trade, any past contract or any reward.
|
2022-01-16 12:31:25 +00:00
|
|
|
deleted_users = []
|
|
|
|
for user in queryset:
|
2023-05-01 10:30:53 +00:00
|
|
|
# Try an except, due to unknown cause for users lacking robots.
|
2022-03-06 19:12:47 +00:00
|
|
|
try:
|
2022-10-20 09:56:10 +00:00
|
|
|
if (
|
2023-05-05 12:30:25 +00:00
|
|
|
user.robot.earned_rewards > 0
|
2023-05-01 10:30:53 +00:00
|
|
|
or user.robot.claimed_rewards > 0
|
|
|
|
or user.robot.telegram_enabled is True
|
2022-10-20 09:56:10 +00:00
|
|
|
):
|
2022-03-06 19:12:47 +00:00
|
|
|
continue
|
2023-05-01 10:30:53 +00:00
|
|
|
if not user.robot.total_contracts == 0:
|
2022-03-06 19:12:47 +00:00
|
|
|
continue
|
|
|
|
valid, _, _ = Logics.validate_already_maker_or_taker(user)
|
|
|
|
if valid:
|
|
|
|
deleted_users.append(str(user))
|
|
|
|
user.delete()
|
2022-10-20 20:53:51 +00:00
|
|
|
except Exception:
|
2022-03-06 19:12:47 +00:00
|
|
|
pass
|
2022-01-16 12:31:25 +00:00
|
|
|
|
|
|
|
results = {
|
2022-02-17 19:50:10 +00:00
|
|
|
"num_deleted": len(deleted_users),
|
|
|
|
"deleted_users": deleted_users,
|
2022-01-16 12:31:25 +00:00
|
|
|
}
|
2022-01-16 18:32:34 +00:00
|
|
|
return results
|
2022-01-16 12:31:25 +00:00
|
|
|
|
2022-10-20 09:56:10 +00:00
|
|
|
|
2023-05-10 19:07:28 +00:00
|
|
|
@shared_task(name="follow_send_payment", time_limit=180, soft_time_limit=175)
|
2022-04-15 23:26:39 +00:00
|
|
|
def follow_send_payment(hash):
|
2022-02-17 19:50:10 +00:00
|
|
|
"""Sends sats to buyer, continuous update"""
|
2022-01-17 16:41:55 +00:00
|
|
|
|
2023-05-12 10:10:14 +00:00
|
|
|
from datetime import timedelta
|
|
|
|
|
2022-01-17 16:41:55 +00:00
|
|
|
from decouple import config
|
2022-01-24 22:53:55 +00:00
|
|
|
from django.utils import timezone
|
2022-01-17 16:41:55 +00:00
|
|
|
|
2023-04-22 18:54:03 +00:00
|
|
|
from api.lightning.node import LNNode
|
|
|
|
from api.models import LNPayment
|
2022-01-16 21:54:42 +00:00
|
|
|
|
2023-04-25 10:13:28 +00:00
|
|
|
if config("DEBUG_PERMISSIONED_PAYOUTS", cast=bool, default=False):
|
|
|
|
return
|
|
|
|
|
2022-04-15 23:26:39 +00:00
|
|
|
lnpayment = LNPayment.objects.get(payment_hash=hash)
|
2023-03-16 00:53:37 +00:00
|
|
|
lnpayment.last_routing_time = timezone.now()
|
2023-05-08 18:10:37 +00:00
|
|
|
lnpayment.save(update_fields=["last_routing_time"])
|
2023-02-10 13:34:35 +00:00
|
|
|
|
|
|
|
# Default is 0ppm. Set by the user over API. Client's default is 1000 ppm.
|
|
|
|
fee_limit_sat = int(
|
2023-05-01 10:30:53 +00:00
|
|
|
float(lnpayment.num_satoshis) * float(lnpayment.routing_budget_ppm) / 1_000_000
|
2023-02-10 13:34:35 +00:00
|
|
|
)
|
2023-04-22 18:54:03 +00:00
|
|
|
timeout_seconds = config("PAYOUT_TIMEOUT_SECONDS", cast=int, default=90)
|
2023-05-10 19:07:28 +00:00
|
|
|
try:
|
|
|
|
results = LNNode.follow_send_payment(lnpayment, fee_limit_sat, timeout_seconds)
|
|
|
|
|
|
|
|
except SoftTimeLimitExceeded:
|
2023-05-12 10:10:14 +00:00
|
|
|
# If the 175 seconds have been consumed without follow_send_payment()
|
|
|
|
# returning, we set the last routing time as 'in 10 minutes'
|
|
|
|
# so the next check happens in 10 minutes, instead of right now.
|
|
|
|
lnpayment.last_routing_time = timezone.now() + timedelta(minutes=10)
|
2023-05-10 19:07:28 +00:00
|
|
|
lnpayment.save(update_fields=["last_routing_time"])
|
2023-05-12 10:10:14 +00:00
|
|
|
print(
|
|
|
|
f"Order: {lnpayment.order_paid_LN.id} SOFT TIME LIMIT REACHED. Hash: {hash}"
|
|
|
|
)
|
2023-05-10 19:07:28 +00:00
|
|
|
results = {}
|
2022-01-17 16:41:55 +00:00
|
|
|
|
2023-04-22 18:54:03 +00:00
|
|
|
return results
|
2023-03-16 18:11:39 +00:00
|
|
|
|
2022-10-20 09:56:10 +00:00
|
|
|
|
2023-05-16 17:12:15 +00:00
|
|
|
@shared_task(name="send_devfund_donation", time_limit=300, soft_time_limit=295)
|
|
|
|
def send_devfund_donation(order_id, proceeds, reason):
|
|
|
|
"""Sends a fraction of order.proceeds via keysend as
|
|
|
|
donation to the RoboSats Open Source project devfund.
|
|
|
|
"""
|
|
|
|
from decouple import config
|
|
|
|
from django.contrib.auth.models import User
|
|
|
|
|
|
|
|
from api.lightning.node import LNNode
|
|
|
|
from api.models import LNPayment, Order
|
2024-01-08 14:13:37 +00:00
|
|
|
from api.utils import get_devfund_pubkey
|
2023-05-16 17:12:15 +00:00
|
|
|
|
2024-01-08 14:13:37 +00:00
|
|
|
target_pubkey = get_devfund_pubkey(config("NETWORK", cast=str))
|
2023-05-16 17:12:15 +00:00
|
|
|
|
|
|
|
order = Order.objects.get(id=order_id)
|
|
|
|
coordinator_alias = config("COORDINATOR_ALIAS", cast=str, default="NoAlias")
|
2024-01-22 16:28:08 +00:00
|
|
|
donation_fraction = min(1.0, max(0.00, config("DEVFUND", cast=float, default=0.2)))
|
2023-05-16 17:12:15 +00:00
|
|
|
message = f"Devfund donation; {coordinator_alias}; {order}; {donation_fraction}; {reason};"
|
|
|
|
num_satoshis = int(proceeds * donation_fraction)
|
|
|
|
routing_budget_sats = int(max(5, num_satoshis * 0.000_1))
|
|
|
|
timeout = 280
|
|
|
|
sign = False
|
|
|
|
|
|
|
|
valid, keysend_payment = LNNode.send_keysend(
|
|
|
|
target_pubkey, message, num_satoshis, routing_budget_sats, timeout, sign
|
|
|
|
)
|
|
|
|
if not valid:
|
|
|
|
return False
|
|
|
|
|
2023-08-06 17:48:20 +00:00
|
|
|
lnpayment = LNPayment.objects.create(
|
2023-05-16 17:12:15 +00:00
|
|
|
concept=LNPayment.Concepts.DEVDONAT,
|
|
|
|
type=LNPayment.Types.KEYS,
|
|
|
|
sender=User.objects.get(
|
|
|
|
username=config("ESCROW_USERNAME", cast=str, default="admin")
|
|
|
|
),
|
|
|
|
invoice=f"Target pubkey: {target_pubkey}; At: {keysend_payment['created_at']}",
|
|
|
|
routing_budget_sats=routing_budget_sats,
|
|
|
|
description=message,
|
|
|
|
num_satoshis=num_satoshis,
|
|
|
|
order_donated=order,
|
|
|
|
**keysend_payment,
|
|
|
|
)
|
|
|
|
|
2023-08-06 17:48:20 +00:00
|
|
|
order.log(
|
|
|
|
f"Development fund donation LNPayment({lnpayment.payment_hash},{str(lnpayment)}) was made via keysend for {num_satoshis} Sats"
|
|
|
|
)
|
2023-05-16 17:12:15 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2023-04-08 09:24:47 +00:00
|
|
|
@shared_task(name="payments_cleansing", time_limit=600)
|
2022-06-20 17:56:08 +00:00
|
|
|
def payments_cleansing():
|
2022-05-31 10:35:36 +00:00
|
|
|
"""
|
2022-10-20 09:56:10 +00:00
|
|
|
Deletes cancelled payments (hodl invoices never locked) that
|
2022-06-20 17:56:08 +00:00
|
|
|
belong to orders expired more than 3 days ago.
|
2022-10-20 09:56:10 +00:00
|
|
|
Deletes 'cancelled' or 'create' onchain_payments
|
2022-05-31 10:35:36 +00:00
|
|
|
"""
|
|
|
|
|
|
|
|
from datetime import timedelta
|
2022-10-25 18:04:12 +00:00
|
|
|
|
|
|
|
from django.db.models import Q
|
2022-05-31 10:35:36 +00:00
|
|
|
from django.utils import timezone
|
|
|
|
|
2022-10-25 18:04:12 +00:00
|
|
|
from api.models import LNPayment, OnchainPayment
|
|
|
|
|
2022-05-31 10:35:36 +00:00
|
|
|
# Orders that have expired more than -3 days ago
|
|
|
|
# Usually expiry is 1 day for every finished order. So ~4 days until
|
|
|
|
# a never locked hodl invoice is removed.
|
|
|
|
finished_time = timezone.now() - timedelta(days=3)
|
2022-10-20 09:56:10 +00:00
|
|
|
queryset = LNPayment.objects.filter(
|
|
|
|
Q(status=LNPayment.Status.CANCEL),
|
|
|
|
Q(order_made__expires_at__lt=finished_time)
|
|
|
|
| Q(order_taken__expires_at__lt=finished_time),
|
|
|
|
)
|
2022-05-31 10:35:36 +00:00
|
|
|
|
|
|
|
# And do not have an active trade, any past contract or any reward.
|
|
|
|
deleted_lnpayments = []
|
|
|
|
for lnpayment in queryset:
|
2022-06-20 17:56:08 +00:00
|
|
|
# Try and except. In case some payment is already missing.
|
2022-05-31 10:35:36 +00:00
|
|
|
try:
|
|
|
|
name = str(lnpayment)
|
|
|
|
lnpayment.delete()
|
|
|
|
deleted_lnpayments.append(name)
|
2022-10-20 20:53:51 +00:00
|
|
|
except Exception:
|
2022-05-31 10:35:36 +00:00
|
|
|
pass
|
2022-10-20 09:56:10 +00:00
|
|
|
|
2022-06-20 17:56:08 +00:00
|
|
|
# same for onchain payments
|
2022-10-20 09:56:10 +00:00
|
|
|
queryset = OnchainPayment.objects.filter(
|
|
|
|
Q(status__in=[OnchainPayment.Status.CANCE, OnchainPayment.Status.CREAT]),
|
|
|
|
Q(order_paid_TX__expires_at__lt=finished_time) | Q(order_paid_TX__isnull=True),
|
|
|
|
)
|
2022-06-20 17:56:08 +00:00
|
|
|
|
|
|
|
# And do not have an active trade, any past contract or any reward.
|
|
|
|
deleted_onchainpayments = []
|
|
|
|
for onchainpayment in queryset:
|
|
|
|
# Try and except. In case some payment is already missing.
|
|
|
|
try:
|
|
|
|
name = str(onchainpayment)
|
|
|
|
onchainpayment.delete()
|
|
|
|
deleted_onchainpayments.append(name)
|
2022-10-20 20:53:51 +00:00
|
|
|
except Exception:
|
2022-06-20 17:56:08 +00:00
|
|
|
pass
|
2022-05-31 10:35:36 +00:00
|
|
|
|
|
|
|
results = {
|
2022-06-20 17:56:08 +00:00
|
|
|
"num_lnpayments_deleted": len(deleted_lnpayments),
|
2022-05-31 10:35:36 +00:00
|
|
|
"deleted_lnpayments": deleted_lnpayments,
|
2022-06-20 17:56:08 +00:00
|
|
|
"num_onchainpayments_deleted": len(deleted_onchainpayments),
|
|
|
|
"deleted_onchainpayments": deleted_onchainpayments,
|
2022-05-31 10:35:36 +00:00
|
|
|
}
|
|
|
|
return results
|
|
|
|
|
2022-10-20 09:56:10 +00:00
|
|
|
|
2023-05-10 19:07:28 +00:00
|
|
|
@shared_task(
|
|
|
|
name="cache_external_market_prices",
|
|
|
|
ignore_result=True,
|
|
|
|
time_limit=120,
|
|
|
|
soft_time_limit=115,
|
|
|
|
)
|
2022-01-16 12:31:25 +00:00
|
|
|
def cache_market():
|
2023-05-10 19:07:28 +00:00
|
|
|
import math
|
|
|
|
|
2022-10-25 18:04:12 +00:00
|
|
|
from django.utils import timezone
|
|
|
|
|
2022-01-17 16:41:55 +00:00
|
|
|
from .models import Currency
|
|
|
|
from .utils import get_exchange_rates
|
|
|
|
|
2022-02-16 22:02:21 +00:00
|
|
|
currency_codes = list(Currency.currency_dict.values())
|
2023-05-10 19:07:28 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
exchange_rates = get_exchange_rates(currency_codes)
|
2022-02-16 22:02:21 +00:00
|
|
|
|
2023-05-12 10:10:14 +00:00
|
|
|
if not exchange_rates:
|
|
|
|
return
|
2023-05-10 20:57:33 +00:00
|
|
|
|
2023-05-12 10:10:14 +00:00
|
|
|
results = {}
|
|
|
|
for i in range(
|
|
|
|
len(Currency.currency_dict.values())
|
|
|
|
): # currencies are indexed starting at 1 (USD)
|
|
|
|
rate = exchange_rates[i]
|
|
|
|
results[i] = {currency_codes[i], rate}
|
|
|
|
|
|
|
|
# Do not update if no new rate was found
|
|
|
|
if math.isnan(rate):
|
|
|
|
continue
|
|
|
|
|
|
|
|
# Create / Update database cached prices
|
|
|
|
currency_key = list(Currency.currency_dict.keys())[i]
|
|
|
|
Currency.objects.update_or_create(
|
|
|
|
id=int(currency_key),
|
|
|
|
currency=int(currency_key),
|
|
|
|
# if there is a Cached market prices matching that id, it updates it with defaults below
|
|
|
|
defaults={
|
|
|
|
"exchange_rate": float(rate),
|
|
|
|
"timestamp": timezone.now(),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
return results
|
|
|
|
|
|
|
|
except SoftTimeLimitExceeded:
|
|
|
|
print("SOFT LIMIT REACHED. Could not fetch current external market prices.")
|
|
|
|
return
|
2022-02-23 16:15:48 +00:00
|
|
|
|
2022-10-20 09:56:10 +00:00
|
|
|
|
2024-07-01 16:00:17 +00:00
|
|
|
@shared_task(name="", ignore_result=True, time_limit=120)
|
|
|
|
def send_order_nostr_event(order_id=None, message=None):
|
|
|
|
if order_id:
|
|
|
|
from api.models import Order
|
|
|
|
from api.nostr import Nostr
|
|
|
|
|
|
|
|
order = Order.objects.get(id=order_id)
|
|
|
|
|
|
|
|
nostr = Nostr()
|
|
|
|
if message == "new":
|
|
|
|
coroutine = nostr.send_new_order_event(order)
|
|
|
|
|
|
|
|
if coroutine:
|
|
|
|
loop = asyncio.get_event_loop()
|
|
|
|
loop.run_until_complete(coroutine)
|
|
|
|
|
|
|
|
return
|
|
|
|
|
2023-04-08 09:24:47 +00:00
|
|
|
@shared_task(name="send_notification", ignore_result=True, time_limit=120)
|
2023-03-27 09:37:36 +00:00
|
|
|
def send_notification(order_id=None, chat_message_id=None, message=None):
|
|
|
|
if order_id:
|
|
|
|
from api.models import Order
|
|
|
|
|
|
|
|
order = Order.objects.get(id=order_id)
|
|
|
|
elif chat_message_id:
|
|
|
|
from chat.models import Message
|
|
|
|
|
|
|
|
chat_message = Message.objects.get(id=chat_message_id)
|
|
|
|
order = chat_message.order
|
2022-10-20 09:56:10 +00:00
|
|
|
|
2024-06-27 16:47:23 +00:00
|
|
|
from api.notifications import Notifications
|
2022-10-20 09:56:10 +00:00
|
|
|
|
2024-06-27 16:47:23 +00:00
|
|
|
notifications = Notifications()
|
2022-02-23 16:15:48 +00:00
|
|
|
|
2022-10-20 09:56:10 +00:00
|
|
|
if message == "welcome":
|
2024-06-27 16:47:23 +00:00
|
|
|
notifications.welcome(order)
|
2022-10-20 09:56:10 +00:00
|
|
|
|
|
|
|
elif message == "order_expired_untaken":
|
2024-06-27 16:47:23 +00:00
|
|
|
notifications.order_expired_untaken(order)
|
2022-02-23 16:15:48 +00:00
|
|
|
|
2022-10-20 09:56:10 +00:00
|
|
|
elif message == "trade_successful":
|
2024-06-27 16:47:23 +00:00
|
|
|
notifications.trade_successful(order)
|
2022-02-23 16:15:48 +00:00
|
|
|
|
2022-10-20 09:56:10 +00:00
|
|
|
elif message == "public_order_cancelled":
|
2024-06-27 16:47:23 +00:00
|
|
|
notifications.public_order_cancelled(order)
|
2022-02-23 16:15:48 +00:00
|
|
|
|
2022-10-20 09:56:10 +00:00
|
|
|
elif message == "taker_expired_b4bond":
|
2024-06-27 16:47:23 +00:00
|
|
|
notifications.taker_expired_b4bond(order)
|
2022-02-23 16:15:48 +00:00
|
|
|
|
2022-10-20 09:56:10 +00:00
|
|
|
elif message == "order_published":
|
2024-06-27 16:47:23 +00:00
|
|
|
notifications.order_published(order)
|
2022-10-20 09:56:10 +00:00
|
|
|
|
|
|
|
elif message == "order_taken_confirmed":
|
2024-06-27 16:47:23 +00:00
|
|
|
notifications.order_taken_confirmed(order)
|
2022-10-20 09:56:10 +00:00
|
|
|
|
|
|
|
elif message == "fiat_exchange_starts":
|
2024-06-27 16:47:23 +00:00
|
|
|
notifications.fiat_exchange_starts(order)
|
2022-05-01 13:41:55 +00:00
|
|
|
|
2022-10-20 09:56:10 +00:00
|
|
|
elif message == "dispute_opened":
|
2024-06-27 16:47:23 +00:00
|
|
|
notifications.dispute_opened(order)
|
2022-06-02 22:32:01 +00:00
|
|
|
|
2022-10-20 09:56:10 +00:00
|
|
|
elif message == "collaborative_cancelled":
|
2024-06-27 16:47:23 +00:00
|
|
|
notifications.collaborative_cancelled(order)
|
2022-06-02 22:32:01 +00:00
|
|
|
|
2023-03-27 09:37:36 +00:00
|
|
|
elif message == "new_chat_message":
|
2024-06-27 16:47:23 +00:00
|
|
|
notifications.new_chat_message(order, chat_message)
|
2023-03-27 09:37:36 +00:00
|
|
|
|
2024-01-28 13:52:30 +00:00
|
|
|
elif message == "coordinator_cancelled":
|
2024-06-27 16:47:23 +00:00
|
|
|
notifications.coordinator_cancelled(order)
|
2024-01-28 13:52:30 +00:00
|
|
|
|
2024-06-28 13:56:58 +00:00
|
|
|
elif message == "dispute_closed":
|
|
|
|
notifications.dispute_closed(order)
|
|
|
|
|
|
|
|
elif message == "lightning_failed":
|
|
|
|
notifications.lightning_failed(order)
|
|
|
|
|
2022-10-20 09:56:10 +00:00
|
|
|
return
|